From 3cc0346140745f5514baee739da17af3e834b492 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Maciej=20Wo=C5=9B?= Date: Thu, 7 Dec 2023 16:01:34 +0000 Subject: [PATCH] set max_connections sqlx parameter --- Cargo.lock | 33 ++++++------ .../src/commands/run.rs | 6 ++- packages/fuel-indexer-database/Cargo.toml | 1 + packages/fuel-indexer-database/src/lib.rs | 25 +++++++++- packages/fuel-indexer-lib/src/config/cli.rs | 8 +++ packages/fuel-indexer-lib/src/config/mod.rs | 10 ++++ packages/fuel-indexer-lib/src/defaults.rs | 3 ++ packages/fuel-indexer-tests/src/fixtures.rs | 50 +++++++++++-------- packages/fuel-indexer/src/commands/run.rs | 6 ++- .../forc-index/src/ops/forc_index_start.rs | 3 ++ 10 files changed, 104 insertions(+), 41 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index bf17da6ef..38d155742 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -2699,7 +2699,7 @@ checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1" [[package]] name = "forc-index" -version = "0.24.1" +version = "0.24.2" dependencies = [ "actix-web", "anyhow", @@ -2733,7 +2733,7 @@ dependencies = [ [[package]] name = "forc-postgres" -version = "0.24.1" +version = "0.24.2" dependencies = [ "anyhow", "clap 3.2.25", @@ -3232,7 +3232,7 @@ dependencies = [ [[package]] name = "fuel-indexer" -version = "0.24.1" +version = "0.24.2" dependencies = [ "anyhow", "async-std", @@ -3265,7 +3265,7 @@ dependencies = [ [[package]] name = "fuel-indexer-api-server" -version = "0.24.1" +version = "0.24.2" dependencies = [ "anyhow", "async-graphql 5.0.10", @@ -3324,19 +3324,20 @@ dependencies = [ [[package]] name = "fuel-indexer-database" -version = "0.24.1" +version = "0.24.2" dependencies = [ "fuel-indexer-database-types", "fuel-indexer-lib", "fuel-indexer-postgres", "sqlx", "thiserror", + "tracing", "url", ] [[package]] name = "fuel-indexer-database-types" -version = "0.24.1" +version = "0.24.2" dependencies = [ "async-graphql-parser 5.0.10", "async-graphql-value 5.0.10", @@ -3349,7 +3350,7 @@ dependencies = [ [[package]] name = "fuel-indexer-graphql" -version = "0.24.1" +version = "0.24.2" dependencies = [ "async-graphql 5.0.10", "async-graphql-parser 5.0.10", @@ -3367,7 +3368,7 @@ dependencies = [ [[package]] name = "fuel-indexer-lib" -version = "0.24.1" +version = "0.24.2" dependencies = [ "anyhow", "async-graphql-parser 5.0.10", @@ -3391,7 +3392,7 @@ dependencies = [ [[package]] name = "fuel-indexer-macro-utils" -version = "0.24.1" +version = "0.24.2" dependencies = [ "proc-macro-error", "proc-macro2", @@ -3401,7 +3402,7 @@ dependencies = [ [[package]] name = "fuel-indexer-macros" -version = "0.24.1" +version = "0.24.2" dependencies = [ "async-graphql-parser 5.0.10", "async-graphql-value 5.0.10", @@ -3424,7 +3425,7 @@ dependencies = [ [[package]] name = "fuel-indexer-metrics" -version = "0.24.1" +version = "0.24.2" dependencies = [ "axum 0.6.20", "lazy_static", @@ -3434,7 +3435,7 @@ dependencies = [ [[package]] name = "fuel-indexer-plugin" -version = "0.24.1" +version = "0.24.2" dependencies = [ "anyhow", "bincode", @@ -3451,7 +3452,7 @@ dependencies = [ [[package]] name = "fuel-indexer-postgres" -version = "0.24.1" +version = "0.24.2" dependencies = [ "bigdecimal", "chrono", @@ -3466,7 +3467,7 @@ dependencies = [ [[package]] name = "fuel-indexer-schema" -version = "0.24.1" +version = "0.24.2" dependencies = [ "async-graphql-parser 5.0.10", "fuel-indexer-database", @@ -3544,7 +3545,7 @@ dependencies = [ [[package]] name = "fuel-indexer-types" -version = "0.24.1" +version = "0.24.2" dependencies = [ "bytes", "fuel-tx 0.35.4", @@ -3557,7 +3558,7 @@ dependencies = [ [[package]] name = "fuel-indexer-utils" -version = "0.24.1" +version = "0.24.2" dependencies = [ "fuel-indexer-macros", "fuel-indexer-plugin", diff --git a/packages/fuel-indexer-api-server/src/commands/run.rs b/packages/fuel-indexer-api-server/src/commands/run.rs index 836001ef5..27810fc45 100644 --- a/packages/fuel-indexer-api-server/src/commands/run.rs +++ b/packages/fuel-indexer-api-server/src/commands/run.rs @@ -18,7 +18,11 @@ pub async fn exec(args: ApiServerArgs) -> anyhow::Result<()> { let (tx, _) = channel::(SERVICE_REQUEST_CHANNEL_SIZE); - let pool = IndexerConnectionPool::connect(&config.database.to_string()).await?; + let pool = IndexerConnectionPool::connect( + &config.database.to_string(), + config.max_db_connections, + ) + .await?; if config.run_migrations { let mut c = pool.acquire().await?; diff --git a/packages/fuel-indexer-database/Cargo.toml b/packages/fuel-indexer-database/Cargo.toml index 73cc34ca8..a20c03b6d 100644 --- a/packages/fuel-indexer-database/Cargo.toml +++ b/packages/fuel-indexer-database/Cargo.toml @@ -15,4 +15,5 @@ fuel-indexer-lib = { workspace = true } fuel-indexer-postgres = { workspace = true } sqlx = { version = "0.6" } thiserror = { workspace = true } +tracing = { workspace = true } url = "2.2" diff --git a/packages/fuel-indexer-database/src/lib.rs b/packages/fuel-indexer-database/src/lib.rs index 2a351b2e4..62a842450 100644 --- a/packages/fuel-indexer-database/src/lib.rs +++ b/packages/fuel-indexer-database/src/lib.rs @@ -59,6 +59,7 @@ impl IndexerConnectionPool { pub async fn connect( database_url: &str, + max_db_connections: u32, ) -> Result { let url = url::Url::parse(database_url); if url.is_err() { @@ -83,11 +84,18 @@ impl IndexerConnectionPool { opts.disable_statement_logging(); let pool = attempt_database_connection(|| { - sqlx::postgres::PgPoolOptions::new().connect_with(opts.clone()) + sqlx::postgres::PgPoolOptions::new() + .max_connections(max_db_connections) + .connect_with(opts.clone()) }) .await; - Ok(IndexerConnectionPool::Postgres(pool)) + let result = IndexerConnectionPool::Postgres(pool); + let backend_max_connections = result.max_connections().await?; + if backend_max_connections < max_db_connections { + tracing::warn!("Indexer --max-db-connections `{max_db_connections}` exceeds `{backend_max_connections}` value set by db backend") + }; + Ok(result) } err => Err(IndexerDatabaseError::BackendNotSupported(err.into())), } @@ -116,4 +124,17 @@ impl IndexerConnectionPool { } } } + + pub async fn max_connections(&self) -> sqlx::Result { + match self { + IndexerConnectionPool::Postgres(pool) => { + let max_connections: i32 = sqlx::query_scalar( + "SELECT setting::int FROM pg_settings WHERE name = 'max_connections'", + ) + .fetch_one(pool) + .await?; + Ok(max_connections as u32) + } + } + } } diff --git a/packages/fuel-indexer-lib/src/config/cli.rs b/packages/fuel-indexer-lib/src/config/cli.rs index e08611c5f..2f832ee46 100644 --- a/packages/fuel-indexer-lib/src/config/cli.rs +++ b/packages/fuel-indexer-lib/src/config/cli.rs @@ -64,6 +64,10 @@ pub struct IndexerArgs { #[clap(long, help = "Database type.", default_value = defaults::DATABASE, value_parser(["postgres"]))] pub database: String, + /// The maximum number of database connections. + #[clap(long, help = "The maximum number of database connections.", default_value_t = defaults::MAX_DB_CONNECTIONS)] + pub max_db_connections: u32, + /// Max body size for web server requests. #[clap(long, help = "Max body size for web server requests.", default_value_t = defaults::MAX_BODY_SIZE )] pub max_body_size: usize, @@ -262,6 +266,10 @@ pub struct ApiServerArgs { #[clap(long, help = "Database type.", default_value = defaults::DATABASE, value_parser(["postgres"]))] pub database: String, + /// The maximum number of database connections. + #[clap(long, help = "The maximum number of database connections.", default_value_t = defaults::MAX_DB_CONNECTIONS)] + pub max_db_connections: u32, + /// Max body size for web server requests. #[clap(long, help = "Max body size for web requests.", default_value_t = defaults::MAX_BODY_SIZE )] pub max_body_size: usize, diff --git a/packages/fuel-indexer-lib/src/config/mod.rs b/packages/fuel-indexer-lib/src/config/mod.rs index 6b76a678a..815cdd3f8 100644 --- a/packages/fuel-indexer-lib/src/config/mod.rs +++ b/packages/fuel-indexer-lib/src/config/mod.rs @@ -82,6 +82,7 @@ impl Default for IndexerArgs { web_api_host: defaults::WEB_API_HOST.to_string(), web_api_port: defaults::WEB_API_PORT.to_string(), database: defaults::DATABASE.to_string(), + max_db_connections: defaults::MAX_DB_CONNECTIONS, max_body_size: defaults::MAX_BODY_SIZE, postgres_user: Some(defaults::POSTGRES_USER.to_string()), postgres_database: Some(defaults::POSTGRES_DATABASE.to_string()), @@ -132,6 +133,7 @@ pub struct IndexerConfig { pub web_api: WebApiConfig, #[serde(default)] pub database: DatabaseConfig, + pub max_db_connections: u32, pub metrics: bool, pub stop_idle_indexers: bool, pub run_migrations: bool, @@ -157,6 +159,7 @@ impl Default for IndexerConfig { fuel_node: FuelClientConfig::default(), web_api: WebApiConfig::default(), database: DatabaseConfig::default(), + max_db_connections: defaults::MAX_DB_CONNECTIONS, metrics: defaults::USE_METRICS, stop_idle_indexers: defaults::STOP_IDLE_INDEXERS, run_migrations: defaults::RUN_MIGRATIONS, @@ -221,6 +224,7 @@ impl From for IndexerConfig { local_fuel_node: args.local_fuel_node, indexer_net_config: args.indexer_net_config, database, + max_db_connections: args.max_db_connections, fuel_node: FuelClientConfig { host: args.fuel_node_host, port: args.fuel_node_port, @@ -320,6 +324,7 @@ impl From for IndexerConfig { local_fuel_node: defaults::LOCAL_FUEL_NODE, indexer_net_config: defaults::INDEXER_NET_CONFIG, database, + max_db_connections: args.max_db_connections, fuel_node: FuelClientConfig { host: args.fuel_node_host, port: args.fuel_node_port, @@ -452,6 +457,7 @@ impl IndexerConfig { let fuel_config_key = serde_yaml::Value::String("fuel_node".into()); let web_config_key = serde_yaml::Value::String("web_api".into()); let database_config_key = serde_yaml::Value::String("database".into()); + let max_db_connections = serde_yaml::Value::String("max_db_connections".into()); let auth_config_key = serde_yaml::Value::String("authentication".into()); let rate_limit_config_key = serde_yaml::Value::String("rate_limit".into()); @@ -549,6 +555,10 @@ impl IndexerConfig { } } + if let Some(max_db_connections) = content.get(max_db_connections) { + config.max_db_connections = max_db_connections.as_u64().unwrap() as u32; + } + if let Some(section) = content.get(auth_config_key) { let auth_enabled = section.get(&serde_yaml::Value::String("enabled".into())); if let Some(auth_enabled) = auth_enabled { diff --git a/packages/fuel-indexer-lib/src/defaults.rs b/packages/fuel-indexer-lib/src/defaults.rs index 599fc3e6a..d3c19b8bc 100644 --- a/packages/fuel-indexer-lib/src/defaults.rs +++ b/packages/fuel-indexer-lib/src/defaults.rs @@ -143,3 +143,6 @@ pub const DISABLE_TOOLCHAIN_VERSION_CHECK: bool = false; /// Default Fuel network to use. pub const NETWORK: &str = "beta-4"; + +/// Maximum number of database connections. It the number exceeds the database backend setting, a warning will be issued. +pub const MAX_DB_CONNECTIONS: u32 = 100; diff --git a/packages/fuel-indexer-tests/src/fixtures.rs b/packages/fuel-indexer-tests/src/fixtures.rs index c7925edb4..686d74861 100644 --- a/packages/fuel-indexer-tests/src/fixtures.rs +++ b/packages/fuel-indexer-tests/src/fixtures.rs @@ -9,7 +9,7 @@ use fuel_indexer_api_server::api::WebApi; use fuel_indexer_database::IndexerConnectionPool; use fuel_indexer_lib::{ config::{DatabaseConfig, IndexerConfig, WebApiConfig}, - defaults::SERVICE_REQUEST_CHANNEL_SIZE, + defaults::{MAX_DB_CONNECTIONS, SERVICE_REQUEST_CHANNEL_SIZE}, manifest::Manifest, utils::{derive_socket_addr, ServiceRequest}, }; @@ -193,20 +193,22 @@ impl TestPostgresDb { .await?; // Instantiate a pool so that it can be stored in the struct for use in the tests - let pool = - match IndexerConnectionPool::connect(&test_db_config.clone().to_string()) - .await - { - Ok(pool) => match pool { - IndexerConnectionPool::Postgres(p) => { - let mut conn = p.acquire().await?; - - fuel_indexer_postgres::run_migration(&mut conn).await?; - p - } - }, - Err(e) => return Err(TestError::PoolCreationError(e)), - }; + let pool = match IndexerConnectionPool::connect( + &test_db_config.clone().to_string(), + MAX_DB_CONNECTIONS, + ) + .await + { + Ok(pool) => match pool { + IndexerConnectionPool::Postgres(p) => { + let mut conn = p.acquire().await?; + + fuel_indexer_postgres::run_migration(&mut conn).await?; + p + } + }, + Err(e) => return Err(TestError::PoolCreationError(e)), + }; Ok(Self { db_name, @@ -382,9 +384,12 @@ pub async fn api_server_app_postgres( config.database = DatabaseConfig::from_str(url).unwrap(); } - let pool = IndexerConnectionPool::connect(&config.database.to_string()) - .await - .unwrap(); + let pool = IndexerConnectionPool::connect( + &config.database.to_string(), + config.max_db_connections, + ) + .await + .unwrap(); let (tx, rx) = channel::(SERVICE_REQUEST_CHANNEL_SIZE); @@ -405,9 +410,12 @@ pub async fn indexer_service_postgres( let (_tx, rx) = channel::(SERVICE_REQUEST_CHANNEL_SIZE); - let pool = IndexerConnectionPool::connect(&config.database.to_string()) - .await - .unwrap(); + let pool = IndexerConnectionPool::connect( + &config.database.to_string(), + config.max_db_connections, + ) + .await + .unwrap(); IndexerService::new(config, pool, rx).await.unwrap() } diff --git a/packages/fuel-indexer/src/commands/run.rs b/packages/fuel-indexer/src/commands/run.rs index 79288caee..bb8648a1a 100644 --- a/packages/fuel-indexer/src/commands/run.rs +++ b/packages/fuel-indexer/src/commands/run.rs @@ -117,7 +117,11 @@ If the --embedded-database flag demonstrates flaky behavior on your machine, or #[allow(unused)] let (tx, rx) = channel::(defaults::SERVICE_REQUEST_CHANNEL_SIZE); - let pool = IndexerConnectionPool::connect(&config.database.to_string()).await?; + let pool = IndexerConnectionPool::connect( + &config.database.to_string(), + config.max_db_connections, + ) + .await?; if config.run_migrations { let mut c = pool.acquire().await?; diff --git a/plugins/forc-index/src/ops/forc_index_start.rs b/plugins/forc-index/src/ops/forc_index_start.rs index f216c09f8..5dc346b1f 100644 --- a/plugins/forc-index/src/ops/forc_index_start.rs +++ b/plugins/forc-index/src/ops/forc_index_start.rs @@ -32,6 +32,7 @@ pub async fn init(command: StartCommand) -> anyhow::Result<()> { verbose, local_fuel_node, max_body_size, + max_db_connections, stop_idle_indexers, indexer_net_config, rate_limit, @@ -83,6 +84,8 @@ pub async fn init(command: StartCommand) -> anyhow::Result<()> { .arg(OsStr::new(&metering_points.to_string())); cmd.arg("--block-page-size") .arg(OsStr::new(&block_page_size.to_string())); + cmd.arg("--max-db-connections") + .arg(OsStr::new(&max_db_connections.to_string())); // Bool options let options = [