use std::{
path::PathBuf,
time::Duration,
};
use clap::ValueEnum;
use fuel_core_poa::signer::SignMode;
use strum_macros::{
Display,
EnumString,
EnumVariantNames,
};
use fuel_core_chain_config::SnapshotReader;
#[cfg(feature = "test-helpers")]
use fuel_core_chain_config::{
ChainConfig,
StateConfig,
};
pub use fuel_core_consensus_module::RelayerConsensusConfig;
pub use fuel_core_importer;
#[cfg(feature = "p2p")]
use fuel_core_p2p::config::{
Config as P2PConfig,
NotInitialized,
};
pub use fuel_core_poa::Trigger;
#[cfg(feature = "relayer")]
use fuel_core_relayer::Config as RelayerConfig;
use fuel_core_txpool::config::Config as TxPoolConfig;
use fuel_core_types::blockchain::header::StateTransitionBytecodeVersion;
use crate::{
combined_database::CombinedDatabaseConfig,
graphql_api::{
worker_service::DaCompressionConfig,
ServiceConfig as GraphQLConfig,
},
};
#[derive(Clone, Debug)]
pub struct Config {
pub graphql_config: GraphQLConfig,
pub combined_db_config: CombinedDatabaseConfig,
pub snapshot_reader: SnapshotReader,
pub continue_on_error: bool,
pub debug: bool,
pub utxo_validation: bool,
pub native_executor_version: Option<StateTransitionBytecodeVersion>,
pub block_production: Trigger,
pub predefined_blocks_path: Option<PathBuf>,
pub vm: VMConfig,
pub txpool: TxPoolConfig,
pub block_producer: fuel_core_producer::Config,
pub starting_gas_price: u64,
pub gas_price_change_percent: u64,
pub min_gas_price: u64,
pub gas_price_threshold_percent: u64,
pub da_compression: DaCompressionConfig,
pub block_importer: fuel_core_importer::Config,
#[cfg(feature = "relayer")]
pub relayer: Option<RelayerConfig>,
#[cfg(feature = "p2p")]
pub p2p: Option<P2PConfig<NotInitialized>>,
#[cfg(feature = "p2p")]
pub sync: fuel_core_sync::Config,
pub consensus_signer: SignMode,
pub name: String,
pub relayer_consensus_config: fuel_core_consensus_module::RelayerConsensusConfig,
pub min_connected_reserved_peers: usize,
pub time_until_synced: Duration,
pub memory_pool_size: usize,
}
impl Config {
#[cfg(feature = "test-helpers")]
pub fn local_node() -> Self {
Self::local_node_with_state_config(StateConfig::local_testnet())
}
#[cfg(feature = "test-helpers")]
pub fn local_node_with_state_config(state_config: StateConfig) -> Self {
Self::local_node_with_configs(ChainConfig::local_testnet(), state_config)
}
#[cfg(feature = "test-helpers")]
pub fn local_node_with_configs(
chain_config: ChainConfig,
state_config: StateConfig,
) -> Self {
Self::local_node_with_reader(SnapshotReader::new_in_memory(
chain_config,
state_config,
))
}
#[cfg(feature = "test-helpers")]
pub fn local_node_with_reader(snapshot_reader: SnapshotReader) -> Self {
let block_importer = fuel_core_importer::Config::new(false);
let latest_block = snapshot_reader.last_block_config();
let native_executor_version = latest_block
.map(|last_block| last_block.state_transition_version.saturating_add(1))
.unwrap_or(
fuel_core_types::blockchain::header::LATEST_STATE_TRANSITION_VERSION,
);
let utxo_validation = false;
let combined_db_config = CombinedDatabaseConfig {
max_database_cache_size: 10 * 1024 * 1024,
database_path: Default::default(),
#[cfg(feature = "rocksdb")]
database_type: DbType::RocksDb,
#[cfg(not(feature = "rocksdb"))]
database_type: DbType::InMemory,
#[cfg(feature = "rocksdb")]
state_rewind_policy:
crate::state::historical_rocksdb::StateRewindPolicy::RewindFullRange,
};
let starting_gas_price = 0;
let gas_price_change_percent = 0;
let min_gas_price = 0;
let gas_price_threshold_percent = 50;
Self {
graphql_config: GraphQLConfig {
addr: std::net::SocketAddr::new(
std::net::Ipv4Addr::new(127, 0, 0, 1).into(),
0,
),
number_of_threads: 0,
database_batch_size: 100,
max_queries_depth: 16,
max_queries_complexity: 80000,
max_queries_recursive_depth: 16,
max_queries_resolver_recursive_depth: 1,
max_queries_directives: 10,
max_concurrent_queries: 1024,
request_body_bytes_limit: 16 * 1024 * 1024,
query_log_threshold_time: Duration::from_secs(2),
api_request_timeout: Duration::from_secs(60),
costs: Default::default(),
},
combined_db_config,
continue_on_error: false,
debug: true,
utxo_validation,
native_executor_version: Some(native_executor_version),
snapshot_reader,
block_production: Trigger::Instant,
predefined_blocks_path: None,
vm: Default::default(),
txpool: TxPoolConfig {
utxo_validation,
max_txs_ttl: Duration::from_secs(60 * 100000000),
..Default::default()
},
block_producer: fuel_core_producer::Config {
..Default::default()
},
da_compression: DaCompressionConfig::Disabled,
starting_gas_price,
gas_price_change_percent,
min_gas_price,
gas_price_threshold_percent,
block_importer,
#[cfg(feature = "relayer")]
relayer: None,
#[cfg(feature = "p2p")]
p2p: Some(P2PConfig::<NotInitialized>::default("test_network")),
#[cfg(feature = "p2p")]
sync: fuel_core_sync::Config::default(),
consensus_signer: SignMode::Key(fuel_core_types::secrecy::Secret::new(
fuel_core_chain_config::default_consensus_dev_key().into(),
)),
name: String::default(),
relayer_consensus_config: Default::default(),
min_connected_reserved_peers: 0,
time_until_synced: Duration::ZERO,
memory_pool_size: 4,
}
}
pub fn make_config_consistent(mut self) -> Config {
if !self.debug && !self.utxo_validation {
tracing::warn!(
"The `utxo_validation` should be `true` with disabled `debug`"
);
self.utxo_validation = true;
}
if self.txpool.utxo_validation != self.utxo_validation {
tracing::warn!("The `utxo_validation` of `TxPool` was inconsistent");
self.txpool.utxo_validation = self.utxo_validation;
}
self
}
}
impl From<&Config> for fuel_core_poa::Config {
fn from(config: &Config) -> Self {
fuel_core_poa::Config {
trigger: config.block_production,
signer: config.consensus_signer.clone(),
metrics: false,
min_connected_reserved_peers: config.min_connected_reserved_peers,
time_until_synced: config.time_until_synced,
chain_id: config
.snapshot_reader
.chain_config()
.consensus_parameters
.chain_id(),
}
}
}
#[derive(Clone, Debug, Default)]
pub struct VMConfig {
pub backtrace: bool,
}
#[derive(
Clone, Debug, Display, Eq, PartialEq, EnumString, EnumVariantNames, ValueEnum,
)]
#[strum(serialize_all = "kebab_case")]
pub enum DbType {
InMemory,
RocksDb,
}