1use std::{
6 net::{IpAddr, Ipv4Addr, SocketAddr},
7 num::NonZeroUsize,
8 path::{Path, PathBuf},
9 sync::Arc,
10 time::Duration,
11};
12
13use anyhow::Result;
14use iota_keys::keypair_file::{read_authority_keypair_from_file, read_keypair_from_file};
15use iota_names::config::IotaNamesConfig;
16use iota_sdk_types::Address;
17use iota_types::{
18 committee::EpochId,
19 crypto::{
20 AccountKeyPair, AuthorityKeyPair, AuthorityPublicKeyBytes, IotaKeyPair, KeypairTraits,
21 NetworkKeyPair, get_key_pair_from_rng,
22 },
23 messages_checkpoint::CheckpointSequenceNumber,
24 multiaddr::Multiaddr,
25 supported_protocol_versions::{Chain, SupportedProtocolVersions},
26 traffic_control::{PolicyConfig, RemoteFirewallConfig},
27};
28use once_cell::sync::OnceCell;
29use rand::rngs::OsRng;
30use serde::{Deserialize, Serialize};
31use starfish_config::Parameters as StarfishParameters;
32use tracing::info;
33
34use crate::{
35 Config, certificate_deny_config::CertificateDenyConfig, genesis,
36 migration_tx_data::MigrationTxData, object_storage_config::ObjectStoreConfig, p2p::P2pConfig,
37 transaction_deny_config::TransactionDenyConfig, verifier_signing_config::VerifierSigningConfig,
38};
39
40pub const DEFAULT_GRPC_CONCURRENCY_LIMIT: usize = 20000000000;
42
43pub const DEFAULT_VALIDATOR_GAS_PRICE: u64 = iota_types::transaction::DEFAULT_VALIDATOR_GAS_PRICE;
45
46pub const DEFAULT_COMMISSION_RATE: u64 = 200;
48
49#[derive(Clone, Debug, Deserialize, Serialize)]
50#[serde(rename_all = "kebab-case")]
51pub struct NodeConfig {
52 #[serde(default = "default_authority_key_pair")]
55 pub authority_key_pair: AuthorityKeyPairWithPath,
56 #[serde(default = "default_key_pair")]
59 pub protocol_key_pair: KeyPairWithPath,
60 #[serde(default = "default_key_pair")]
61 pub account_key_pair: KeyPairWithPath,
62 #[serde(default = "default_key_pair")]
65 pub network_key_pair: KeyPairWithPath,
66 pub db_path: PathBuf,
67
68 #[serde(default = "default_grpc_address")]
72 pub network_address: Multiaddr,
73 #[serde(default = "default_json_rpc_address")]
74 pub json_rpc_address: SocketAddr,
75
76 #[serde(default = "default_metrics_address")]
78 pub metrics_address: SocketAddr,
79
80 #[serde(default = "default_admin_interface_address")]
84 pub admin_interface_address: SocketAddr,
85
86 #[serde(skip_serializing_if = "Option::is_none")]
88 pub consensus_config: Option<ConsensusConfig>,
89
90 #[serde(default = "default_enable_index_processing")]
95 pub enable_index_processing: bool,
96
97 #[serde(default)]
99 pub jsonrpc_server_type: Option<ServerType>,
104
105 #[serde(default)]
109 pub grpc_load_shed: Option<bool>,
110
111 #[serde(default = "default_concurrency_limit")]
112 pub grpc_concurrency_limit: Option<usize>,
113
114 #[serde(default)]
116 pub p2p_config: P2pConfig,
117
118 pub genesis: Genesis,
122
123 pub migration_tx_data_path: Option<PathBuf>,
125
126 #[serde(default = "default_authority_store_pruning_config")]
129 pub authority_store_pruning_config: AuthorityStorePruningConfig,
130
131 #[serde(default = "default_end_of_epoch_broadcast_channel_capacity")]
136 pub end_of_epoch_broadcast_channel_capacity: usize,
137
138 #[serde(default)]
142 pub checkpoint_executor_config: CheckpointExecutorConfig,
143
144 #[serde(skip_serializing_if = "Option::is_none")]
145 pub metrics: Option<MetricsConfig>,
146
147 #[serde(skip)]
152 pub supported_protocol_versions: Option<SupportedProtocolVersions>,
153
154 #[serde(default)]
158 pub db_checkpoint_config: DBCheckpointConfig,
159
160 #[serde(default)]
162 pub expensive_safety_check_config: ExpensiveSafetyCheckConfig,
163
164 #[serde(default)]
168 pub transaction_deny_config: TransactionDenyConfig,
169
170 #[serde(default)]
176 pub certificate_deny_config: CertificateDenyConfig,
177
178 #[serde(default)]
181 pub state_debug_dump_config: StateDebugDumpConfig,
182
183 #[serde(default)]
187 pub state_archive_write_config: StateArchiveConfig,
188
189 #[serde(default)]
190 pub state_archive_read_config: Vec<StateArchiveConfig>,
191
192 #[serde(default)]
194 pub state_snapshot_write_config: StateSnapshotConfig,
195
196 #[serde(default)]
197 pub indexer_max_subscriptions: Option<usize>,
198
199 #[serde(default = "default_transaction_kv_store_config")]
200 pub transaction_kv_store_read_config: TransactionKeyValueStoreReadConfig,
201
202 #[serde(skip_serializing_if = "Option::is_none")]
204 pub transaction_kv_store_write_config: Option<TransactionKeyValueStoreWriteConfig>,
205
206 #[serde(default = "default_authority_overload_config")]
209 pub authority_overload_config: AuthorityOverloadConfig,
210
211 #[serde(skip_serializing_if = "Option::is_none")]
215 pub run_with_range: Option<RunWithRange>,
216
217 #[serde(
219 skip_serializing_if = "Option::is_none",
220 default = "default_traffic_controller_policy_config"
221 )]
222 pub policy_config: Option<PolicyConfig>,
223
224 #[serde(skip_serializing_if = "Option::is_none")]
225 pub firewall_config: Option<RemoteFirewallConfig>,
226
227 #[serde(default)]
228 pub execution_cache_config: ExecutionCacheConfig,
229
230 #[serde(default = "bool_true")]
231 pub enable_validator_tx_finalizer: bool,
232
233 #[serde(default = "bool_true")]
239 pub enable_soft_locking: bool,
240
241 #[serde(default)]
242 pub verifier_signing_config: VerifierSigningConfig,
243
244 #[serde(skip_serializing_if = "Option::is_none")]
248 pub enable_db_write_stall: Option<bool>,
249
250 #[serde(default, skip_serializing_if = "Option::is_none")]
251 pub iota_names_config: Option<IotaNamesConfig>,
252
253 #[serde(default)]
255 pub enable_grpc_api: bool,
256 #[serde(
257 default = "default_grpc_api_config",
258 skip_serializing_if = "Option::is_none"
259 )]
260 pub grpc_api_config: Option<GrpcApiConfig>,
261
262 #[serde(skip_serializing_if = "Option::is_none")]
267 pub chain_override_for_testing: Option<Chain>,
268
269 #[serde(default, skip_serializing_if = "Option::is_none")]
272 pub validator_client_monitor_config:
273 Option<crate::validator_client_monitor_config::ValidatorClientMonitorConfig>,
274}
275
276#[derive(Clone, Debug, Default, serde::Deserialize, serde::Serialize)]
277#[serde(rename_all = "kebab-case")]
278pub struct TlsConfig {
279 cert: String,
281 key: String,
283}
284
285impl TlsConfig {
286 pub fn cert(&self) -> &str {
287 &self.cert
288 }
289
290 pub fn key(&self) -> &str {
291 &self.key
292 }
293}
294
295#[derive(Clone, Debug, serde::Deserialize, serde::Serialize)]
297#[serde(rename_all = "kebab-case")]
298pub struct GrpcApiConfig {
299 #[serde(default = "default_grpc_api_address")]
301 pub address: SocketAddr,
302
303 #[serde(skip_serializing_if = "Option::is_none")]
307 pub tls: Option<TlsConfig>,
308
309 #[serde(default = "default_grpc_api_max_message_size_bytes")]
311 pub max_message_size_bytes: u32,
312
313 #[serde(default = "default_grpc_api_broadcast_buffer_size")]
315 pub broadcast_buffer_size: u32,
316
317 #[serde(default = "default_grpc_api_max_concurrent_stream_subscribers")]
323 pub max_concurrent_stream_subscribers: u32,
324
325 #[serde(default = "default_grpc_api_max_json_move_value_size")]
328 pub max_json_move_value_size: usize,
329
330 #[serde(default = "default_grpc_api_max_execute_transaction_batch_size")]
333 pub max_execute_transaction_batch_size: u32,
334
335 #[serde(default = "default_grpc_api_max_simulate_transaction_batch_size")]
338 pub max_simulate_transaction_batch_size: u32,
339
340 #[serde(default = "default_grpc_api_max_checkpoint_inclusion_timeout_ms")]
344 pub max_checkpoint_inclusion_timeout_ms: u64,
345}
346
347fn default_grpc_api_address() -> SocketAddr {
348 SocketAddr::new(IpAddr::V4(Ipv4Addr::new(0, 0, 0, 0)), 50051)
349}
350
351fn default_grpc_api_broadcast_buffer_size() -> u32 {
352 100
353}
354
355fn default_grpc_api_max_concurrent_stream_subscribers() -> u32 {
356 1024
357}
358
359fn default_grpc_api_max_message_size_bytes() -> u32 {
360 128 * 1024 * 1024 }
362
363fn default_grpc_api_max_json_move_value_size() -> usize {
364 1024 * 1024 }
366
367fn default_grpc_api_max_execute_transaction_batch_size() -> u32 {
368 20
369}
370
371fn default_grpc_api_max_simulate_transaction_batch_size() -> u32 {
372 20
373}
374
375fn default_grpc_api_max_checkpoint_inclusion_timeout_ms() -> u64 {
376 60_000 }
378
379impl Default for GrpcApiConfig {
380 fn default() -> Self {
381 Self {
382 address: default_grpc_api_address(),
383 tls: None,
384 max_message_size_bytes: default_grpc_api_max_message_size_bytes(),
385 broadcast_buffer_size: default_grpc_api_broadcast_buffer_size(),
386 max_concurrent_stream_subscribers: default_grpc_api_max_concurrent_stream_subscribers(),
387 max_json_move_value_size: default_grpc_api_max_json_move_value_size(),
388 max_execute_transaction_batch_size: default_grpc_api_max_execute_transaction_batch_size(
389 ),
390 max_simulate_transaction_batch_size:
391 default_grpc_api_max_simulate_transaction_batch_size(),
392 max_checkpoint_inclusion_timeout_ms:
393 default_grpc_api_max_checkpoint_inclusion_timeout_ms(),
394 }
395 }
396}
397
398impl GrpcApiConfig {
399 const GRPC_TONIC_DEFAULT_MAX_RECV_MESSAGE_SIZE: u32 = 4 * 1024 * 1024; const GRPC_MIN_CLIENT_MAX_MESSAGE_SIZE_BYTES: u32 =
403 Self::GRPC_TONIC_DEFAULT_MAX_RECV_MESSAGE_SIZE;
404
405 pub fn tls_config(&self) -> Option<&TlsConfig> {
406 self.tls.as_ref()
407 }
408
409 pub fn max_message_size_bytes(&self) -> u32 {
410 self.max_message_size_bytes
412 .max(Self::GRPC_MIN_CLIENT_MAX_MESSAGE_SIZE_BYTES)
413 }
414
415 pub fn max_message_size_client_bytes(&self, client_max_message_size_bytes: Option<u32>) -> u32 {
419 client_max_message_size_bytes
420 .unwrap_or(Self::GRPC_TONIC_DEFAULT_MAX_RECV_MESSAGE_SIZE)
423 .clamp(
425 Self::GRPC_MIN_CLIENT_MAX_MESSAGE_SIZE_BYTES,
426 self.max_message_size_bytes(),
427 )
428 }
429}
430
431#[derive(Clone, Debug, Default, Deserialize, Serialize)]
432#[serde(rename_all = "kebab-case")]
433pub struct ExecutionCacheConfig {
434 #[serde(default)]
435 pub writeback_cache: WritebackCacheConfig,
436}
437
438#[derive(Clone, Debug, Default, Deserialize, Serialize)]
439#[serde(rename_all = "kebab-case")]
440pub struct WritebackCacheConfig {
441 #[serde(default, skip_serializing_if = "Option::is_none")]
444 pub max_cache_size: Option<u64>, #[serde(default, skip_serializing_if = "Option::is_none")]
447 pub package_cache_size: Option<u64>, #[serde(default, skip_serializing_if = "Option::is_none")]
450 pub object_cache_size: Option<u64>, #[serde(default, skip_serializing_if = "Option::is_none")]
452 pub marker_cache_size: Option<u64>, #[serde(default, skip_serializing_if = "Option::is_none")]
454 pub object_by_id_cache_size: Option<u64>, #[serde(default, skip_serializing_if = "Option::is_none")]
457 pub transaction_cache_size: Option<u64>, #[serde(default, skip_serializing_if = "Option::is_none")]
459 pub executed_effect_cache_size: Option<u64>, #[serde(default, skip_serializing_if = "Option::is_none")]
461 pub effect_cache_size: Option<u64>, #[serde(default, skip_serializing_if = "Option::is_none")]
464 pub events_cache_size: Option<u64>, #[serde(default, skip_serializing_if = "Option::is_none")]
467 pub transaction_objects_cache_size: Option<u64>, #[serde(default, skip_serializing_if = "Option::is_none")]
472 pub backpressure_threshold: Option<u64>, #[serde(default, skip_serializing_if = "Option::is_none")]
478 pub backpressure_threshold_for_rpc: Option<u64>, #[serde(default, skip_serializing_if = "Option::is_none")]
488 pub backpressure_soft_limit_pct: Option<u32>,
489}
490
491impl WritebackCacheConfig {
492 pub fn max_cache_size(&self) -> u64 {
493 std::env::var("IOTA_CACHE_WRITEBACK_SIZE_MAX")
494 .ok()
495 .and_then(|s| s.parse().ok())
496 .or(self.max_cache_size)
497 .unwrap_or(100000)
498 }
499
500 pub fn package_cache_size(&self) -> u64 {
501 std::env::var("IOTA_CACHE_WRITEBACK_SIZE_PACKAGE")
502 .ok()
503 .and_then(|s| s.parse().ok())
504 .or(self.package_cache_size)
505 .unwrap_or(1000)
506 }
507
508 pub fn object_cache_size(&self) -> u64 {
509 std::env::var("IOTA_CACHE_WRITEBACK_SIZE_OBJECT")
510 .ok()
511 .and_then(|s| s.parse().ok())
512 .or(self.object_cache_size)
513 .unwrap_or_else(|| self.max_cache_size())
514 }
515
516 pub fn marker_cache_size(&self) -> u64 {
517 std::env::var("IOTA_CACHE_WRITEBACK_SIZE_MARKER")
518 .ok()
519 .and_then(|s| s.parse().ok())
520 .or(self.marker_cache_size)
521 .unwrap_or_else(|| self.object_cache_size())
522 }
523
524 pub fn object_by_id_cache_size(&self) -> u64 {
525 std::env::var("IOTA_CACHE_WRITEBACK_SIZE_OBJECT_BY_ID")
526 .ok()
527 .and_then(|s| s.parse().ok())
528 .or(self.object_by_id_cache_size)
529 .unwrap_or_else(|| self.object_cache_size())
530 }
531
532 pub fn transaction_cache_size(&self) -> u64 {
533 std::env::var("IOTA_CACHE_WRITEBACK_SIZE_TRANSACTION")
534 .ok()
535 .and_then(|s| s.parse().ok())
536 .or(self.transaction_cache_size)
537 .unwrap_or_else(|| self.max_cache_size())
538 }
539
540 pub fn executed_effect_cache_size(&self) -> u64 {
541 std::env::var("IOTA_CACHE_WRITEBACK_SIZE_EXECUTED_EFFECT")
542 .ok()
543 .and_then(|s| s.parse().ok())
544 .or(self.executed_effect_cache_size)
545 .unwrap_or_else(|| self.transaction_cache_size())
546 }
547
548 pub fn effect_cache_size(&self) -> u64 {
549 std::env::var("IOTA_CACHE_WRITEBACK_SIZE_EFFECT")
550 .ok()
551 .and_then(|s| s.parse().ok())
552 .or(self.effect_cache_size)
553 .unwrap_or_else(|| self.executed_effect_cache_size())
554 }
555
556 pub fn events_cache_size(&self) -> u64 {
557 std::env::var("IOTA_CACHE_WRITEBACK_SIZE_EVENTS")
558 .ok()
559 .and_then(|s| s.parse().ok())
560 .or(self.events_cache_size)
561 .unwrap_or_else(|| self.transaction_cache_size())
562 }
563
564 pub fn transaction_objects_cache_size(&self) -> u64 {
565 std::env::var("IOTA_CACHE_WRITEBACK_SIZE_TRANSACTION_OBJECTS")
566 .ok()
567 .and_then(|s| s.parse().ok())
568 .or(self.transaction_objects_cache_size)
569 .unwrap_or(1000)
570 }
571
572 pub fn backpressure_threshold(&self) -> u64 {
573 std::env::var("IOTA_CACHE_WRITEBACK_BACKPRESSURE_THRESHOLD")
574 .ok()
575 .and_then(|s| s.parse().ok())
576 .or(self.backpressure_threshold)
577 .unwrap_or(100_000)
578 }
579
580 pub fn backpressure_threshold_for_rpc(&self) -> u64 {
581 std::env::var("IOTA_CACHE_WRITEBACK_BACKPRESSURE_THRESHOLD_FOR_RPC")
582 .ok()
583 .and_then(|s| s.parse().ok())
584 .or(self.backpressure_threshold_for_rpc)
585 .unwrap_or(self.backpressure_threshold())
586 }
587
588 pub fn backpressure_soft_limit_pct(&self) -> u32 {
589 std::env::var("IOTA_CACHE_WRITEBACK_BACKPRESSURE_SOFT_LIMIT_PCT")
590 .ok()
591 .and_then(|s| s.parse().ok())
592 .or(self.backpressure_soft_limit_pct)
593 .unwrap_or(50)
594 .min(100)
595 }
596}
597
598#[derive(Clone, Copy, Debug, Deserialize, Serialize)]
599#[serde(rename_all = "lowercase")]
600pub enum ServerType {
601 WebSocket,
602 Http,
603 Both,
604}
605
606#[derive(Clone, Debug, Deserialize, Serialize)]
607#[serde(rename_all = "kebab-case")]
608pub struct TransactionKeyValueStoreReadConfig {
609 #[serde(default = "default_base_url")]
610 pub base_url: String,
611
612 #[serde(default = "default_cache_size")]
613 pub cache_size: u64,
614}
615
616impl Default for TransactionKeyValueStoreReadConfig {
617 fn default() -> Self {
618 Self {
619 base_url: default_base_url(),
620 cache_size: default_cache_size(),
621 }
622 }
623}
624
625fn default_base_url() -> String {
626 "".to_string()
627}
628
629fn default_cache_size() -> u64 {
630 100_000
631}
632
633fn default_transaction_kv_store_config() -> TransactionKeyValueStoreReadConfig {
634 TransactionKeyValueStoreReadConfig::default()
635}
636
637fn default_authority_store_pruning_config() -> AuthorityStorePruningConfig {
638 AuthorityStorePruningConfig::default()
639}
640
641pub fn default_enable_index_processing() -> bool {
642 true
643}
644
645fn default_grpc_address() -> Multiaddr {
646 "/ip4/0.0.0.0/tcp/8080".parse().unwrap()
647}
648fn default_authority_key_pair() -> AuthorityKeyPairWithPath {
649 AuthorityKeyPairWithPath::new(get_key_pair_from_rng::<AuthorityKeyPair, _>(&mut OsRng).1)
650}
651
652fn default_key_pair() -> KeyPairWithPath {
653 KeyPairWithPath::new(
654 get_key_pair_from_rng::<AccountKeyPair, _>(&mut OsRng)
655 .1
656 .into(),
657 )
658}
659
660fn default_metrics_address() -> SocketAddr {
661 SocketAddr::new(IpAddr::V4(Ipv4Addr::new(0, 0, 0, 0)), 9184)
662}
663
664pub fn default_admin_interface_address() -> SocketAddr {
665 SocketAddr::new(IpAddr::V4(Ipv4Addr::LOCALHOST), 1337)
666}
667
668pub fn default_json_rpc_address() -> SocketAddr {
669 SocketAddr::new(IpAddr::V4(Ipv4Addr::new(0, 0, 0, 0)), 9000)
670}
671
672pub fn default_grpc_api_config() -> Option<GrpcApiConfig> {
673 Some(GrpcApiConfig::default())
674}
675
676pub fn default_concurrency_limit() -> Option<usize> {
677 Some(DEFAULT_GRPC_CONCURRENCY_LIMIT)
678}
679
680pub fn default_end_of_epoch_broadcast_channel_capacity() -> usize {
681 128
682}
683
684pub fn bool_true() -> bool {
685 true
686}
687
688fn is_true(value: &bool) -> bool {
689 *value
690}
691
692impl Config for NodeConfig {}
693
694impl NodeConfig {
695 pub fn authority_key_pair(&self) -> &AuthorityKeyPair {
696 self.authority_key_pair.authority_keypair()
697 }
698
699 pub fn protocol_key_pair(&self) -> &NetworkKeyPair {
700 match self.protocol_key_pair.keypair() {
701 IotaKeyPair::Ed25519(kp) => kp,
702 other => {
703 panic!("invalid keypair type: {other:?}, only Ed25519 is allowed for protocol key")
704 }
705 }
706 }
707
708 pub fn network_key_pair(&self) -> &NetworkKeyPair {
709 match self.network_key_pair.keypair() {
710 IotaKeyPair::Ed25519(kp) => kp,
711 other => {
712 panic!("invalid keypair type: {other:?}, only Ed25519 is allowed for network key")
713 }
714 }
715 }
716
717 pub fn authority_public_key(&self) -> AuthorityPublicKeyBytes {
718 self.authority_key_pair().public().into()
719 }
720
721 pub fn db_path(&self) -> PathBuf {
722 self.db_path.join("live")
723 }
724
725 pub fn db_checkpoint_path(&self) -> PathBuf {
726 self.db_path.join("db_checkpoints")
727 }
728
729 pub fn archive_path(&self) -> PathBuf {
730 self.db_path.join("archive")
731 }
732
733 pub fn snapshot_path(&self) -> PathBuf {
734 self.db_path.join("snapshot")
735 }
736
737 pub fn network_address(&self) -> &Multiaddr {
738 &self.network_address
739 }
740
741 pub fn consensus_config(&self) -> Option<&ConsensusConfig> {
742 self.consensus_config.as_ref()
743 }
744
745 pub fn genesis(&self) -> Result<&genesis::Genesis> {
746 self.genesis.genesis()
747 }
748
749 pub fn load_migration_tx_data(&self) -> Result<MigrationTxData> {
750 let Some(location) = &self.migration_tx_data_path else {
751 anyhow::bail!("no file location set");
752 };
753
754 let migration_tx_data = MigrationTxData::load(location)?;
756
757 migration_tx_data.validate_from_genesis(self.genesis.genesis()?)?;
759 Ok(migration_tx_data)
760 }
761
762 pub fn iota_address(&self) -> Address {
763 (&self.account_key_pair.keypair().public()).into()
764 }
765
766 pub fn archive_reader_config(&self) -> Vec<ArchiveReaderConfig> {
767 self.state_archive_read_config
768 .iter()
769 .flat_map(|config| {
770 config
771 .object_store_config
772 .as_ref()
773 .map(|remote_store_config| ArchiveReaderConfig {
774 remote_store_config: remote_store_config.clone(),
775 download_concurrency: NonZeroUsize::new(config.concurrency)
776 .unwrap_or(NonZeroUsize::new(5).unwrap()),
777 use_for_pruning_watermark: config.use_for_pruning_watermark,
778 })
779 })
780 .collect()
781 }
782
783 pub fn jsonrpc_server_type(&self) -> ServerType {
784 self.jsonrpc_server_type.unwrap_or(ServerType::Http)
785 }
786}
787
788#[derive(Debug, Clone, Deserialize, Serialize)]
789#[serde(rename_all = "kebab-case")]
790pub struct ConsensusConfig {
791 pub db_path: PathBuf,
793
794 pub db_retention_epochs: Option<u64>,
798
799 pub db_pruner_period_secs: Option<u64>,
803
804 pub max_pending_transactions: Option<usize>,
815
816 pub max_submit_position: Option<usize>,
822
823 pub submit_delay_step_override_millis: Option<u64>,
829
830 #[serde(skip_serializing_if = "Option::is_none", alias = "starfish_parameters")]
832 pub parameters: Option<StarfishParameters>,
833
834 #[serde(skip_serializing_if = "Option::is_none")]
840 pub graduated_load_shedding_soft_limit_pct: Option<u32>,
841}
842
843impl ConsensusConfig {
844 pub fn db_path(&self) -> &Path {
845 &self.db_path
846 }
847
848 pub fn max_pending_transactions(&self) -> usize {
852 self.max_pending_transactions.unwrap_or(20_000)
853 }
854
855 pub fn graduated_load_shedding_soft_limit_pct(&self) -> u32 {
860 self.graduated_load_shedding_soft_limit_pct
861 .unwrap_or(50)
862 .min(100)
863 }
864
865 pub fn submit_delay_step_override(&self) -> Option<Duration> {
866 self.submit_delay_step_override_millis
867 .map(Duration::from_millis)
868 }
869
870 pub fn db_retention_epochs(&self) -> u64 {
871 self.db_retention_epochs.unwrap_or(0)
872 }
873
874 pub fn db_pruner_period(&self) -> Duration {
875 self.db_pruner_period_secs
877 .map(Duration::from_secs)
878 .unwrap_or(Duration::from_secs(3_600))
879 }
880}
881
882#[derive(Clone, Debug, Deserialize, Serialize)]
883#[serde(rename_all = "kebab-case")]
884pub struct CheckpointExecutorConfig {
885 #[serde(default = "default_checkpoint_execution_max_concurrency")]
890 pub checkpoint_execution_max_concurrency: usize,
891
892 #[serde(default = "default_local_execution_timeout_sec")]
898 pub local_execution_timeout_sec: u64,
899
900 #[serde(default, skip_serializing_if = "Option::is_none")]
905 pub data_ingestion_dir: Option<PathBuf>,
906}
907
908#[derive(Clone, Debug, Default, Deserialize, Serialize)]
909#[serde(rename_all = "kebab-case")]
910pub struct ExpensiveSafetyCheckConfig {
911 #[serde(default)]
916 enable_epoch_iota_conservation_check: bool,
917
918 #[serde(default)]
922 enable_deep_per_tx_iota_conservation_check: bool,
923
924 #[serde(default)]
927 force_disable_epoch_iota_conservation_check: bool,
928
929 #[serde(default)]
932 enable_state_consistency_check: bool,
933
934 #[serde(default)]
936 force_disable_state_consistency_check: bool,
937
938 #[serde(default)]
939 enable_secondary_index_checks: bool,
940 }
942
943impl ExpensiveSafetyCheckConfig {
944 pub fn new_enable_all() -> Self {
945 Self {
946 enable_epoch_iota_conservation_check: true,
947 enable_deep_per_tx_iota_conservation_check: true,
948 force_disable_epoch_iota_conservation_check: false,
949 enable_state_consistency_check: true,
950 force_disable_state_consistency_check: false,
951 enable_secondary_index_checks: false, }
953 }
954
955 pub fn new_disable_all() -> Self {
956 Self {
957 enable_epoch_iota_conservation_check: false,
958 enable_deep_per_tx_iota_conservation_check: false,
959 force_disable_epoch_iota_conservation_check: true,
960 enable_state_consistency_check: false,
961 force_disable_state_consistency_check: true,
962 enable_secondary_index_checks: false,
963 }
964 }
965
966 pub fn force_disable_epoch_iota_conservation_check(&mut self) {
967 self.force_disable_epoch_iota_conservation_check = true;
968 }
969
970 pub fn enable_epoch_iota_conservation_check(&self) -> bool {
971 (self.enable_epoch_iota_conservation_check || cfg!(debug_assertions))
972 && !self.force_disable_epoch_iota_conservation_check
973 }
974
975 pub fn force_disable_state_consistency_check(&mut self) {
976 self.force_disable_state_consistency_check = true;
977 }
978
979 pub fn enable_state_consistency_check(&self) -> bool {
980 (self.enable_state_consistency_check || cfg!(debug_assertions))
981 && !self.force_disable_state_consistency_check
982 }
983
984 pub fn enable_deep_per_tx_iota_conservation_check(&self) -> bool {
985 self.enable_deep_per_tx_iota_conservation_check || cfg!(debug_assertions)
986 }
987
988 pub fn enable_secondary_index_checks(&self) -> bool {
989 self.enable_secondary_index_checks
990 }
991}
992
993fn default_checkpoint_execution_max_concurrency() -> usize {
994 4
995}
996
997fn default_local_execution_timeout_sec() -> u64 {
998 30
999}
1000
1001impl Default for CheckpointExecutorConfig {
1002 fn default() -> Self {
1003 Self {
1004 checkpoint_execution_max_concurrency: default_checkpoint_execution_max_concurrency(),
1005 local_execution_timeout_sec: default_local_execution_timeout_sec(),
1006 data_ingestion_dir: None,
1007 }
1008 }
1009}
1010
1011#[derive(Debug, Clone, Deserialize, Serialize)]
1012#[serde(rename_all = "kebab-case")]
1013pub struct AuthorityStorePruningConfig {
1014 #[serde(default = "default_num_latest_epoch_dbs_to_retain")]
1016 pub num_latest_epoch_dbs_to_retain: usize,
1017 #[serde(default)]
1022 pub num_epochs_to_retain: u64,
1023 #[serde(skip_serializing_if = "Option::is_none")]
1025 pub pruning_run_delay_seconds: Option<u64>,
1026 #[serde(default = "default_max_checkpoints_in_batch")]
1029 pub max_checkpoints_in_batch: usize,
1030 #[serde(default = "default_max_transactions_in_batch")]
1032 pub max_transactions_in_batch: usize,
1033 #[serde(
1038 default = "default_periodic_compaction_threshold_days",
1039 skip_serializing_if = "Option::is_none"
1040 )]
1041 pub periodic_compaction_threshold_days: Option<usize>,
1042 #[serde(skip_serializing_if = "Option::is_none")]
1045 pub num_epochs_to_retain_for_checkpoints: Option<u64>,
1046 #[serde(default = "default_smoothing", skip_serializing_if = "is_true")]
1047 pub smooth: bool,
1048 #[serde(default, skip_serializing_if = "std::ops::Not::not")]
1054 pub enable_compaction_filter: bool,
1055 #[serde(skip_serializing_if = "Option::is_none")]
1056 pub num_epochs_to_retain_for_indexes: Option<u64>,
1057}
1058
1059fn default_num_latest_epoch_dbs_to_retain() -> usize {
1060 3
1061}
1062
1063fn default_max_transactions_in_batch() -> usize {
1064 1000
1065}
1066
1067fn default_max_checkpoints_in_batch() -> usize {
1068 10
1069}
1070
1071fn default_smoothing() -> bool {
1072 cfg!(not(test))
1073}
1074
1075fn default_periodic_compaction_threshold_days() -> Option<usize> {
1076 Some(1)
1077}
1078
1079impl Default for AuthorityStorePruningConfig {
1080 fn default() -> Self {
1081 Self {
1082 num_latest_epoch_dbs_to_retain: default_num_latest_epoch_dbs_to_retain(),
1083 num_epochs_to_retain: 0,
1084 pruning_run_delay_seconds: if cfg!(msim) { Some(2) } else { None },
1085 max_checkpoints_in_batch: default_max_checkpoints_in_batch(),
1086 max_transactions_in_batch: default_max_transactions_in_batch(),
1087 periodic_compaction_threshold_days: None,
1088 num_epochs_to_retain_for_checkpoints: if cfg!(msim) { Some(2) } else { None },
1089 smooth: true,
1090 enable_compaction_filter: cfg!(test) || cfg!(msim),
1091 num_epochs_to_retain_for_indexes: None,
1092 }
1093 }
1094}
1095
1096impl AuthorityStorePruningConfig {
1097 pub fn set_num_epochs_to_retain(&mut self, num_epochs_to_retain: u64) {
1098 self.num_epochs_to_retain = num_epochs_to_retain;
1099 }
1100
1101 pub fn set_num_epochs_to_retain_for_checkpoints(&mut self, num_epochs_to_retain: Option<u64>) {
1102 self.num_epochs_to_retain_for_checkpoints = num_epochs_to_retain;
1103 }
1104
1105 pub fn num_epochs_to_retain_for_checkpoints(&self) -> Option<u64> {
1106 self.num_epochs_to_retain_for_checkpoints
1107 .map(|n| {
1109 if n < 2 {
1110 info!("num_epochs_to_retain_for_checkpoints must be at least 2, rounding up from {}", n);
1111 2
1112 } else {
1113 n
1114 }
1115 })
1116 }
1117}
1118
1119#[derive(Debug, Clone, Deserialize, Serialize)]
1120#[serde(rename_all = "kebab-case")]
1121pub struct MetricsConfig {
1122 #[serde(skip_serializing_if = "Option::is_none")]
1123 pub push_interval_seconds: Option<u64>,
1124 #[serde(skip_serializing_if = "Option::is_none")]
1125 pub push_url: Option<String>,
1126}
1127
1128#[derive(Default, Debug, Clone, Deserialize, Serialize)]
1129#[serde(rename_all = "kebab-case")]
1130pub struct DBCheckpointConfig {
1131 #[serde(default)]
1132 pub perform_db_checkpoints_at_epoch_end: bool,
1133 #[serde(skip_serializing_if = "Option::is_none")]
1134 pub checkpoint_path: Option<PathBuf>,
1135 #[serde(skip_serializing_if = "Option::is_none")]
1136 pub object_store_config: Option<ObjectStoreConfig>,
1137 #[serde(skip_serializing_if = "Option::is_none")]
1138 pub perform_index_db_checkpoints_at_epoch_end: Option<bool>,
1139 #[serde(skip_serializing_if = "Option::is_none")]
1140 pub prune_and_compact_before_upload: Option<bool>,
1141}
1142
1143#[derive(Debug, Clone)]
1144pub struct ArchiveReaderConfig {
1145 pub remote_store_config: ObjectStoreConfig,
1146 pub download_concurrency: NonZeroUsize,
1147 pub use_for_pruning_watermark: bool,
1148}
1149
1150#[derive(Default, Debug, Clone, Deserialize, Serialize)]
1151#[serde(rename_all = "kebab-case")]
1152pub struct StateArchiveConfig {
1153 #[serde(skip_serializing_if = "Option::is_none")]
1154 pub object_store_config: Option<ObjectStoreConfig>,
1155 pub concurrency: usize,
1156 pub use_for_pruning_watermark: bool,
1157}
1158
1159#[derive(Default, Debug, Clone, Deserialize, Serialize)]
1160#[serde(rename_all = "kebab-case")]
1161pub struct StateSnapshotConfig {
1162 #[serde(skip_serializing_if = "Option::is_none")]
1163 pub object_store_config: Option<ObjectStoreConfig>,
1164 pub concurrency: usize,
1165}
1166
1167#[derive(Default, Debug, Clone, Deserialize, Serialize)]
1168#[serde(rename_all = "kebab-case")]
1169pub struct TransactionKeyValueStoreWriteConfig {
1170 pub aws_access_key_id: String,
1171 pub aws_secret_access_key: String,
1172 pub aws_region: String,
1173 pub table_name: String,
1174 pub bucket_name: String,
1175 pub concurrency: usize,
1176}
1177
1178#[derive(Clone, Debug, Deserialize, Serialize)]
1183#[serde(rename_all = "kebab-case")]
1184pub struct AuthorityOverloadConfig {
1185 #[serde(default = "default_max_txn_age_in_queue")]
1189 pub max_txn_age_in_queue: Duration,
1190
1191 #[serde(default = "default_overload_monitor_interval")]
1193 pub overload_monitor_interval: Duration,
1194
1195 #[serde(default = "default_execution_queue_latency_soft_limit")]
1197 pub execution_queue_latency_soft_limit: Duration,
1198
1199 #[serde(default = "default_execution_queue_latency_hard_limit")]
1202 pub execution_queue_latency_hard_limit: Duration,
1203
1204 #[serde(default = "default_max_load_shedding_percentage")]
1206 pub max_load_shedding_percentage: u32,
1207
1208 #[serde(default = "default_min_load_shedding_percentage_above_hard_limit")]
1211 pub min_load_shedding_percentage_above_hard_limit: u32,
1212
1213 #[serde(default = "default_safe_transaction_ready_rate")]
1216 pub safe_transaction_ready_rate: u32,
1217
1218 #[serde(default = "default_check_system_overload_at_signing")]
1221 pub check_system_overload_at_signing: bool,
1222
1223 #[serde(default, skip_serializing_if = "std::ops::Not::not")]
1226 pub check_system_overload_at_execution: bool,
1227
1228 #[serde(default = "default_max_transaction_manager_queue_length")]
1232 pub max_transaction_manager_queue_length: usize,
1233
1234 #[serde(default = "default_max_transaction_manager_per_object_queue_length")]
1237 pub max_transaction_manager_per_object_queue_length: usize,
1238
1239 #[serde(default = "default_max_transaction_manager_queue_length_soft_limit_pct")]
1243 pub max_transaction_manager_queue_length_soft_limit_pct: u32,
1244}
1245
1246impl AuthorityOverloadConfig {
1247 pub fn max_transaction_manager_queue_length_soft_limit_pct(&self) -> u32 {
1250 self.max_transaction_manager_queue_length_soft_limit_pct
1251 .min(100)
1252 }
1253}
1254
1255fn default_max_txn_age_in_queue() -> Duration {
1256 Duration::from_millis(500)
1257}
1258
1259fn default_overload_monitor_interval() -> Duration {
1260 Duration::from_secs(10)
1261}
1262
1263fn default_execution_queue_latency_soft_limit() -> Duration {
1264 Duration::from_secs(1)
1265}
1266
1267fn default_execution_queue_latency_hard_limit() -> Duration {
1268 Duration::from_secs(10)
1269}
1270
1271fn default_max_load_shedding_percentage() -> u32 {
1272 95
1273}
1274
1275fn default_min_load_shedding_percentage_above_hard_limit() -> u32 {
1276 50
1277}
1278
1279fn default_safe_transaction_ready_rate() -> u32 {
1280 100
1281}
1282
1283fn default_check_system_overload_at_signing() -> bool {
1284 true
1285}
1286
1287fn default_max_transaction_manager_queue_length() -> usize {
1288 100_000
1289}
1290
1291fn default_max_transaction_manager_queue_length_soft_limit_pct() -> u32 {
1292 50
1293}
1294
1295fn default_max_transaction_manager_per_object_queue_length() -> usize {
1296 20
1297}
1298
1299impl Default for AuthorityOverloadConfig {
1300 fn default() -> Self {
1301 Self {
1302 max_txn_age_in_queue: default_max_txn_age_in_queue(),
1303 overload_monitor_interval: default_overload_monitor_interval(),
1304 execution_queue_latency_soft_limit: default_execution_queue_latency_soft_limit(),
1305 execution_queue_latency_hard_limit: default_execution_queue_latency_hard_limit(),
1306 max_load_shedding_percentage: default_max_load_shedding_percentage(),
1307 min_load_shedding_percentage_above_hard_limit:
1308 default_min_load_shedding_percentage_above_hard_limit(),
1309 safe_transaction_ready_rate: default_safe_transaction_ready_rate(),
1310 check_system_overload_at_signing: true,
1311 check_system_overload_at_execution: false,
1312 max_transaction_manager_queue_length: default_max_transaction_manager_queue_length(),
1313 max_transaction_manager_queue_length_soft_limit_pct:
1314 default_max_transaction_manager_queue_length_soft_limit_pct(),
1315 max_transaction_manager_per_object_queue_length:
1316 default_max_transaction_manager_per_object_queue_length(),
1317 }
1318 }
1319}
1320
1321fn default_authority_overload_config() -> AuthorityOverloadConfig {
1322 AuthorityOverloadConfig::default()
1323}
1324
1325fn default_traffic_controller_policy_config() -> Option<PolicyConfig> {
1326 Some(PolicyConfig::default_dos_protection_policy())
1327}
1328
1329#[derive(Debug, Clone, PartialEq, Deserialize, Serialize, Eq)]
1330pub struct Genesis {
1331 #[serde(flatten)]
1332 location: Option<GenesisLocation>,
1333
1334 #[serde(skip)]
1335 genesis: once_cell::sync::OnceCell<genesis::Genesis>,
1336}
1337
1338impl Genesis {
1339 pub fn new(genesis: genesis::Genesis) -> Self {
1340 Self {
1341 location: Some(GenesisLocation::InPlace {
1342 genesis: Box::new(genesis),
1343 }),
1344 genesis: Default::default(),
1345 }
1346 }
1347
1348 pub fn new_from_file<P: Into<PathBuf>>(path: P) -> Self {
1349 Self {
1350 location: Some(GenesisLocation::File {
1351 genesis_file_location: path.into(),
1352 }),
1353 genesis: Default::default(),
1354 }
1355 }
1356
1357 pub fn new_empty() -> Self {
1358 Self {
1359 location: None,
1360 genesis: Default::default(),
1361 }
1362 }
1363
1364 pub fn genesis(&self) -> Result<&genesis::Genesis> {
1365 match &self.location {
1366 Some(GenesisLocation::InPlace { genesis }) => Ok(genesis),
1367 Some(GenesisLocation::File {
1368 genesis_file_location,
1369 }) => self
1370 .genesis
1371 .get_or_try_init(|| genesis::Genesis::load(genesis_file_location)),
1372 None => anyhow::bail!("no genesis location set"),
1373 }
1374 }
1375}
1376
1377#[derive(Debug, Clone, PartialEq, Deserialize, Serialize, Eq)]
1378#[serde(untagged)]
1379enum GenesisLocation {
1380 InPlace {
1381 genesis: Box<genesis::Genesis>,
1382 },
1383 File {
1384 #[serde(rename = "genesis-file-location")]
1385 genesis_file_location: PathBuf,
1386 },
1387}
1388
1389#[derive(Clone, Debug, PartialEq, Eq, Deserialize, Serialize)]
1392pub struct KeyPairWithPath {
1393 #[serde(flatten)]
1394 location: KeyPairLocation,
1395
1396 #[serde(skip)]
1397 keypair: OnceCell<Arc<IotaKeyPair>>,
1398}
1399
1400#[derive(Debug, Clone, PartialEq, Deserialize, Serialize, Eq)]
1401#[serde(untagged)]
1402enum KeyPairLocation {
1403 InPlace {
1404 #[serde(with = "bech32_formatted_keypair")]
1405 value: Arc<IotaKeyPair>,
1406 },
1407 File {
1408 path: PathBuf,
1409 },
1410}
1411
1412impl KeyPairWithPath {
1413 pub fn new(kp: IotaKeyPair) -> Self {
1414 let cell: OnceCell<Arc<IotaKeyPair>> = OnceCell::new();
1415 let arc_kp = Arc::new(kp);
1416 cell.set(arc_kp.clone()).expect("failed to set keypair");
1419 Self {
1420 location: KeyPairLocation::InPlace { value: arc_kp },
1421 keypair: cell,
1422 }
1423 }
1424
1425 pub fn new_from_path(path: PathBuf) -> Self {
1426 let cell: OnceCell<Arc<IotaKeyPair>> = OnceCell::new();
1427 cell.set(Arc::new(read_keypair_from_file(&path).unwrap_or_else(
1430 |e| panic!("invalid keypair file at path {:?}: {e}", &path),
1431 )))
1432 .expect("failed to set keypair");
1433 Self {
1434 location: KeyPairLocation::File { path },
1435 keypair: cell,
1436 }
1437 }
1438
1439 pub fn keypair(&self) -> &IotaKeyPair {
1440 self.keypair
1441 .get_or_init(|| match &self.location {
1442 KeyPairLocation::InPlace { value } => value.clone(),
1443 KeyPairLocation::File { path } => {
1444 Arc::new(
1447 read_keypair_from_file(path).unwrap_or_else(|e| {
1448 panic!("invalid keypair file at path {path:?}: {e}")
1449 }),
1450 )
1451 }
1452 })
1453 .as_ref()
1454 }
1455}
1456
1457#[derive(Clone, Debug, PartialEq, Eq, Deserialize, Serialize)]
1460pub struct AuthorityKeyPairWithPath {
1461 #[serde(flatten)]
1462 location: AuthorityKeyPairLocation,
1463
1464 #[serde(skip)]
1465 keypair: OnceCell<Arc<AuthorityKeyPair>>,
1466}
1467
1468#[derive(Debug, Clone, PartialEq, Deserialize, Serialize, Eq)]
1469#[serde(untagged)]
1470enum AuthorityKeyPairLocation {
1471 InPlace { value: Arc<AuthorityKeyPair> },
1472 File { path: PathBuf },
1473}
1474
1475impl AuthorityKeyPairWithPath {
1476 pub fn new(kp: AuthorityKeyPair) -> Self {
1477 let cell: OnceCell<Arc<AuthorityKeyPair>> = OnceCell::new();
1478 let arc_kp = Arc::new(kp);
1479 cell.set(arc_kp.clone())
1482 .expect("failed to set authority keypair");
1483 Self {
1484 location: AuthorityKeyPairLocation::InPlace { value: arc_kp },
1485 keypair: cell,
1486 }
1487 }
1488
1489 pub fn new_from_path(path: PathBuf) -> Self {
1490 let cell: OnceCell<Arc<AuthorityKeyPair>> = OnceCell::new();
1491 cell.set(Arc::new(
1494 read_authority_keypair_from_file(&path)
1495 .unwrap_or_else(|_| panic!("invalid authority keypair file at path {:?}", &path)),
1496 ))
1497 .expect("failed to set authority keypair");
1498 Self {
1499 location: AuthorityKeyPairLocation::File { path },
1500 keypair: cell,
1501 }
1502 }
1503
1504 pub fn authority_keypair(&self) -> &AuthorityKeyPair {
1505 self.keypair
1506 .get_or_init(|| match &self.location {
1507 AuthorityKeyPairLocation::InPlace { value } => value.clone(),
1508 AuthorityKeyPairLocation::File { path } => {
1509 Arc::new(
1512 read_authority_keypair_from_file(path).unwrap_or_else(|_| {
1513 panic!("invalid authority keypair file {:?}", &path)
1514 }),
1515 )
1516 }
1517 })
1518 .as_ref()
1519 }
1520}
1521
1522#[derive(Clone, Debug, Deserialize, Serialize, Default)]
1525#[serde(rename_all = "kebab-case")]
1526pub struct StateDebugDumpConfig {
1527 #[serde(skip_serializing_if = "Option::is_none")]
1528 pub dump_file_directory: Option<PathBuf>,
1529}
1530
1531#[cfg(test)]
1532mod tests {
1533 use std::path::PathBuf;
1534
1535 use fastcrypto::traits::KeyPair;
1536 use iota_keys::keypair_file::{write_authority_keypair_to_file, write_keypair_to_file};
1537 use iota_types::crypto::{
1538 AuthorityKeyPair, IotaKeyPair, NetworkKeyPair, get_key_pair_from_rng,
1539 };
1540 use rand::{SeedableRng, rngs::StdRng};
1541
1542 use super::Genesis;
1543 use crate::NodeConfig;
1544
1545 #[test]
1546 fn serialize_genesis_from_file() {
1547 let g = Genesis::new_from_file("path/to/file");
1548
1549 let s = serde_yaml::to_string(&g).unwrap();
1550 assert_eq!("---\ngenesis-file-location: path/to/file\n", s);
1551 let loaded_genesis: Genesis = serde_yaml::from_str(&s).unwrap();
1552 assert_eq!(g, loaded_genesis);
1553 }
1554
1555 #[test]
1556 fn fullnode_template() {
1557 const TEMPLATE: &str = include_str!("../data/fullnode-template.yaml");
1558
1559 let _template: NodeConfig = serde_yaml::from_str(TEMPLATE).unwrap();
1560 }
1561
1562 #[test]
1563 fn enable_soft_locking_defaults_to_enabled() {
1564 const TEMPLATE: &str = include_str!("../data/fullnode-template.yaml");
1567
1568 let config: NodeConfig = serde_yaml::from_str(TEMPLATE).unwrap();
1569 assert!(config.enable_soft_locking);
1570 }
1571
1572 #[test]
1573 fn load_key_pairs_to_node_config() {
1574 let authority_key_pair: AuthorityKeyPair =
1575 get_key_pair_from_rng(&mut StdRng::from_seed([0; 32])).1;
1576 let protocol_key_pair: NetworkKeyPair =
1577 get_key_pair_from_rng(&mut StdRng::from_seed([0; 32])).1;
1578 let network_key_pair: NetworkKeyPair =
1579 get_key_pair_from_rng(&mut StdRng::from_seed([0; 32])).1;
1580
1581 write_authority_keypair_to_file(&authority_key_pair, PathBuf::from("authority.key"))
1582 .unwrap();
1583 write_keypair_to_file(
1584 &IotaKeyPair::Ed25519(protocol_key_pair.copy()),
1585 PathBuf::from("protocol.key"),
1586 )
1587 .unwrap();
1588 write_keypair_to_file(
1589 &IotaKeyPair::Ed25519(network_key_pair.copy()),
1590 PathBuf::from("network.key"),
1591 )
1592 .unwrap();
1593
1594 const TEMPLATE: &str = include_str!("../data/fullnode-template-with-path.yaml");
1595 let template: NodeConfig = serde_yaml::from_str(TEMPLATE).unwrap();
1596 assert_eq!(
1597 template.authority_key_pair().public(),
1598 authority_key_pair.public()
1599 );
1600 assert_eq!(
1601 template.network_key_pair().public(),
1602 network_key_pair.public()
1603 );
1604 assert_eq!(
1605 template.protocol_key_pair().public(),
1606 protocol_key_pair.public()
1607 );
1608 }
1609}
1610
1611#[derive(Clone, Copy, PartialEq, Debug, Serialize, Deserialize)]
1615pub enum RunWithRange {
1616 Epoch(EpochId),
1617 Checkpoint(CheckpointSequenceNumber),
1618}
1619
1620impl RunWithRange {
1621 pub fn is_epoch_gt(&self, epoch_id: EpochId) -> bool {
1623 matches!(self, RunWithRange::Epoch(e) if epoch_id > *e)
1624 }
1625
1626 pub fn matches_checkpoint(&self, seq_num: CheckpointSequenceNumber) -> bool {
1627 matches!(self, RunWithRange::Checkpoint(seq) if *seq == seq_num)
1628 }
1629
1630 pub fn into_checkpoint_bound(self) -> Option<CheckpointSequenceNumber> {
1631 match self {
1632 RunWithRange::Epoch(_) => None,
1633 RunWithRange::Checkpoint(seq) => Some(seq),
1634 }
1635 }
1636}
1637
1638mod bech32_formatted_keypair {
1642 use std::ops::Deref;
1643
1644 use iota_types::crypto::{EncodeDecodeBase64, IotaKeyPair};
1645 use serde::{Deserialize, Deserializer, Serializer};
1646
1647 pub fn serialize<S, T>(kp: &T, serializer: S) -> Result<S::Ok, S::Error>
1648 where
1649 S: Serializer,
1650 T: Deref<Target = IotaKeyPair>,
1651 {
1652 use serde::ser::Error;
1653
1654 let s = kp.encode().map_err(Error::custom)?;
1656
1657 serializer.serialize_str(&s)
1658 }
1659
1660 pub fn deserialize<'de, D, T>(deserializer: D) -> Result<T, D::Error>
1661 where
1662 D: Deserializer<'de>,
1663 T: From<IotaKeyPair>,
1664 {
1665 use serde::de::Error;
1666
1667 let s = String::deserialize(deserializer)?;
1668
1669 IotaKeyPair::decode(&s)
1671 .or_else(|_| {
1672 IotaKeyPair::decode_base64(&s)
1674 })
1675 .map(Into::into)
1676 .map_err(Error::custom)
1677 }
1678}