2022-03-10 01:33:29 -08:00
|
|
|
/// Parse the configuration file.
|
2022-06-24 14:52:38 -07:00
|
|
|
use arc_swap::ArcSwap;
|
2022-02-20 22:47:08 -08:00
|
|
|
use log::{error, info};
|
2022-02-19 13:57:35 -08:00
|
|
|
use once_cell::sync::Lazy;
|
2022-02-08 09:25:59 -08:00
|
|
|
use serde_derive::Deserialize;
|
2022-03-10 01:33:29 -08:00
|
|
|
use std::collections::{HashMap, HashSet};
|
|
|
|
|
use std::sync::Arc;
|
2022-02-08 09:25:59 -08:00
|
|
|
use tokio::fs::File;
|
|
|
|
|
use tokio::io::AsyncReadExt;
|
|
|
|
|
use toml;
|
|
|
|
|
|
|
|
|
|
use crate::errors::Error;
|
2022-06-24 14:52:38 -07:00
|
|
|
use crate::{ClientServerMap, ConnectionPool};
|
2022-02-08 09:25:59 -08:00
|
|
|
|
2022-03-10 01:33:29 -08:00
|
|
|
/// Globally available configuration.
|
2022-02-19 13:57:35 -08:00
|
|
|
static CONFIG: Lazy<ArcSwap<Config>> = Lazy::new(|| ArcSwap::from_pointee(Config::default()));
|
|
|
|
|
|
2022-03-10 01:33:29 -08:00
|
|
|
/// Server role: primary or replica.
|
2022-02-09 20:02:20 -08:00
|
|
|
#[derive(Clone, PartialEq, Deserialize, Hash, std::cmp::Eq, Debug, Copy)]
|
|
|
|
|
pub enum Role {
|
|
|
|
|
Primary,
|
|
|
|
|
Replica,
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-28 17:22:28 -08:00
|
|
|
impl ToString for Role {
|
|
|
|
|
fn to_string(&self) -> String {
|
|
|
|
|
match *self {
|
|
|
|
|
Role::Primary => "primary".to_string(),
|
|
|
|
|
Role::Replica => "replica".to_string(),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-18 07:10:18 -08:00
|
|
|
impl PartialEq<Option<Role>> for Role {
|
|
|
|
|
fn eq(&self, other: &Option<Role>) -> bool {
|
|
|
|
|
match other {
|
|
|
|
|
None => true,
|
|
|
|
|
Some(role) => *self == *role,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl PartialEq<Role> for Option<Role> {
|
|
|
|
|
fn eq(&self, other: &Role) -> bool {
|
|
|
|
|
match *self {
|
|
|
|
|
None => true,
|
|
|
|
|
Some(role) => role == *other,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-10 01:33:29 -08:00
|
|
|
/// Address identifying a PostgreSQL server uniquely.
|
2022-02-05 13:15:53 -08:00
|
|
|
#[derive(Clone, PartialEq, Hash, std::cmp::Eq, Debug)]
|
2022-02-05 10:02:13 -08:00
|
|
|
pub struct Address {
|
2022-03-04 17:04:27 -08:00
|
|
|
pub id: usize,
|
2022-02-05 10:02:13 -08:00
|
|
|
pub host: String,
|
|
|
|
|
pub port: String,
|
2022-02-15 08:18:01 -08:00
|
|
|
pub shard: usize,
|
2022-02-09 20:02:20 -08:00
|
|
|
pub role: Role,
|
2022-03-01 22:49:43 -08:00
|
|
|
pub replica_number: usize,
|
2022-02-05 10:02:13 -08:00
|
|
|
}
|
|
|
|
|
|
2022-02-19 13:57:35 -08:00
|
|
|
impl Default for Address {
|
|
|
|
|
fn default() -> Address {
|
|
|
|
|
Address {
|
2022-03-04 17:04:27 -08:00
|
|
|
id: 0,
|
2022-02-19 13:57:35 -08:00
|
|
|
host: String::from("127.0.0.1"),
|
|
|
|
|
port: String::from("5432"),
|
|
|
|
|
shard: 0,
|
2022-03-01 22:49:43 -08:00
|
|
|
replica_number: 0,
|
2022-02-19 13:57:35 -08:00
|
|
|
role: Role::Replica,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-01 22:49:43 -08:00
|
|
|
impl Address {
|
2022-03-10 01:33:29 -08:00
|
|
|
/// Address name (aka database) used in `SHOW STATS`, `SHOW DATABASES`, and `SHOW POOLS`.
|
2022-03-01 22:49:43 -08:00
|
|
|
pub fn name(&self) -> String {
|
|
|
|
|
match self.role {
|
|
|
|
|
Role::Primary => format!("shard_{}_primary", self.shard),
|
|
|
|
|
|
|
|
|
|
Role::Replica => format!("shard_{}_replica_{}", self.shard, self.replica_number),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-10 01:33:29 -08:00
|
|
|
/// PostgreSQL user.
|
2022-02-08 09:25:59 -08:00
|
|
|
#[derive(Clone, PartialEq, Hash, std::cmp::Eq, Deserialize, Debug)]
|
2022-02-05 10:02:13 -08:00
|
|
|
pub struct User {
|
|
|
|
|
pub name: String,
|
|
|
|
|
pub password: String,
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-19 13:57:35 -08:00
|
|
|
impl Default for User {
|
|
|
|
|
fn default() -> User {
|
|
|
|
|
User {
|
|
|
|
|
name: String::from("postgres"),
|
|
|
|
|
password: String::new(),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-10 01:33:29 -08:00
|
|
|
/// General configuration.
|
2022-06-25 11:46:20 -07:00
|
|
|
#[derive(Deserialize, Debug, Clone, PartialEq)]
|
2022-02-08 09:25:59 -08:00
|
|
|
pub struct General {
|
|
|
|
|
pub host: String,
|
|
|
|
|
pub port: i16,
|
|
|
|
|
pub pool_size: u32,
|
|
|
|
|
pub pool_mode: String,
|
|
|
|
|
pub connect_timeout: u64,
|
|
|
|
|
pub healthcheck_timeout: u64,
|
|
|
|
|
pub ban_time: i64,
|
2022-06-25 11:46:20 -07:00
|
|
|
pub autoreload: bool,
|
2022-02-19 13:57:35 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl Default for General {
|
|
|
|
|
fn default() -> General {
|
|
|
|
|
General {
|
|
|
|
|
host: String::from("localhost"),
|
|
|
|
|
port: 5432,
|
|
|
|
|
pool_size: 15,
|
|
|
|
|
pool_mode: String::from("transaction"),
|
|
|
|
|
connect_timeout: 5000,
|
|
|
|
|
healthcheck_timeout: 1000,
|
|
|
|
|
ban_time: 60,
|
2022-06-25 11:46:20 -07:00
|
|
|
autoreload: false,
|
2022-02-19 13:57:35 -08:00
|
|
|
}
|
|
|
|
|
}
|
2022-02-08 09:25:59 -08:00
|
|
|
}
|
|
|
|
|
|
2022-03-10 01:33:29 -08:00
|
|
|
/// Shard configuration.
|
2022-06-24 14:52:38 -07:00
|
|
|
#[derive(Deserialize, Debug, Clone, PartialEq)]
|
2022-02-08 09:25:59 -08:00
|
|
|
pub struct Shard {
|
2022-02-09 20:02:20 -08:00
|
|
|
pub servers: Vec<(String, u16, String)>,
|
2022-02-08 09:25:59 -08:00
|
|
|
pub database: String,
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-19 13:57:35 -08:00
|
|
|
impl Default for Shard {
|
|
|
|
|
fn default() -> Shard {
|
|
|
|
|
Shard {
|
|
|
|
|
servers: vec![(String::from("localhost"), 5432, String::from("primary"))],
|
|
|
|
|
database: String::from("postgres"),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-10 01:33:29 -08:00
|
|
|
/// Query Router configuration.
|
2022-06-25 11:46:20 -07:00
|
|
|
#[derive(Deserialize, Debug, Clone, PartialEq)]
|
2022-02-11 11:19:40 -08:00
|
|
|
pub struct QueryRouter {
|
|
|
|
|
pub default_role: String,
|
2022-02-18 07:10:18 -08:00
|
|
|
pub query_parser_enabled: bool,
|
|
|
|
|
pub primary_reads_enabled: bool,
|
2022-02-23 11:47:24 -08:00
|
|
|
pub sharding_function: String,
|
2022-02-11 11:19:40 -08:00
|
|
|
}
|
|
|
|
|
|
2022-02-19 13:57:35 -08:00
|
|
|
impl Default for QueryRouter {
|
|
|
|
|
fn default() -> QueryRouter {
|
|
|
|
|
QueryRouter {
|
|
|
|
|
default_role: String::from("any"),
|
|
|
|
|
query_parser_enabled: false,
|
|
|
|
|
primary_reads_enabled: true,
|
2022-02-23 11:47:24 -08:00
|
|
|
sharding_function: "pg_bigint_hash".to_string(),
|
2022-02-19 13:57:35 -08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-24 14:52:38 -07:00
|
|
|
fn default_path() -> String {
|
|
|
|
|
String::from("pgcat.toml")
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-10 01:33:29 -08:00
|
|
|
/// Configuration wrapper.
|
2022-06-25 11:46:20 -07:00
|
|
|
#[derive(Deserialize, Debug, Clone, PartialEq)]
|
2022-02-08 09:25:59 -08:00
|
|
|
pub struct Config {
|
2022-06-24 14:52:38 -07:00
|
|
|
#[serde(default = "default_path")]
|
|
|
|
|
pub path: String,
|
|
|
|
|
|
2022-02-08 09:25:59 -08:00
|
|
|
pub general: General,
|
|
|
|
|
pub user: User,
|
|
|
|
|
pub shards: HashMap<String, Shard>,
|
2022-02-11 11:19:40 -08:00
|
|
|
pub query_router: QueryRouter,
|
2022-02-08 09:25:59 -08:00
|
|
|
}
|
|
|
|
|
|
2022-02-19 13:57:35 -08:00
|
|
|
impl Default for Config {
|
|
|
|
|
fn default() -> Config {
|
|
|
|
|
Config {
|
2022-06-24 14:52:38 -07:00
|
|
|
path: String::from("pgcat.toml"),
|
2022-02-19 13:57:35 -08:00
|
|
|
general: General::default(),
|
|
|
|
|
user: User::default(),
|
|
|
|
|
shards: HashMap::from([(String::from("1"), Shard::default())]),
|
|
|
|
|
query_router: QueryRouter::default(),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-28 08:14:39 -08:00
|
|
|
impl From<&Config> for std::collections::HashMap<String, String> {
|
|
|
|
|
fn from(config: &Config) -> HashMap<String, String> {
|
|
|
|
|
HashMap::from([
|
|
|
|
|
("host".to_string(), config.general.host.to_string()),
|
|
|
|
|
("port".to_string(), config.general.port.to_string()),
|
|
|
|
|
(
|
|
|
|
|
"pool_size".to_string(),
|
|
|
|
|
config.general.pool_size.to_string(),
|
|
|
|
|
),
|
|
|
|
|
(
|
|
|
|
|
"pool_mode".to_string(),
|
|
|
|
|
config.general.pool_mode.to_string(),
|
|
|
|
|
),
|
|
|
|
|
(
|
|
|
|
|
"connect_timeout".to_string(),
|
|
|
|
|
config.general.connect_timeout.to_string(),
|
|
|
|
|
),
|
|
|
|
|
(
|
|
|
|
|
"healthcheck_timeout".to_string(),
|
|
|
|
|
config.general.healthcheck_timeout.to_string(),
|
|
|
|
|
),
|
|
|
|
|
("ban_time".to_string(), config.general.ban_time.to_string()),
|
|
|
|
|
(
|
|
|
|
|
"default_role".to_string(),
|
|
|
|
|
config.query_router.default_role.to_string(),
|
|
|
|
|
),
|
|
|
|
|
(
|
|
|
|
|
"query_parser_enabled".to_string(),
|
|
|
|
|
config.query_router.query_parser_enabled.to_string(),
|
|
|
|
|
),
|
|
|
|
|
(
|
|
|
|
|
"primary_reads_enabled".to_string(),
|
|
|
|
|
config.query_router.primary_reads_enabled.to_string(),
|
|
|
|
|
),
|
|
|
|
|
(
|
|
|
|
|
"sharding_function".to_string(),
|
|
|
|
|
config.query_router.sharding_function.to_string(),
|
|
|
|
|
),
|
|
|
|
|
])
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-19 13:57:35 -08:00
|
|
|
impl Config {
|
2022-03-10 01:33:29 -08:00
|
|
|
/// Print current configuration.
|
2022-02-19 13:57:35 -08:00
|
|
|
pub fn show(&self) {
|
2022-02-20 22:47:08 -08:00
|
|
|
info!("Pool size: {}", self.general.pool_size);
|
|
|
|
|
info!("Pool mode: {}", self.general.pool_mode);
|
|
|
|
|
info!("Ban time: {}s", self.general.ban_time);
|
|
|
|
|
info!(
|
|
|
|
|
"Healthcheck timeout: {}ms",
|
2022-02-19 13:57:35 -08:00
|
|
|
self.general.healthcheck_timeout
|
|
|
|
|
);
|
2022-02-20 22:47:08 -08:00
|
|
|
info!("Connection timeout: {}ms", self.general.connect_timeout);
|
2022-02-23 11:47:24 -08:00
|
|
|
info!("Sharding function: {}", self.query_router.sharding_function);
|
2022-06-24 14:52:38 -07:00
|
|
|
info!("Primary reads: {}", self.query_router.primary_reads_enabled);
|
|
|
|
|
info!("Query router: {}", self.query_router.query_parser_enabled);
|
2022-02-23 11:47:24 -08:00
|
|
|
info!("Number of shards: {}", self.shards.len());
|
2022-02-19 13:57:35 -08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-10 01:33:29 -08:00
|
|
|
/// Get a read-only instance of the configuration
|
|
|
|
|
/// from anywhere in the app.
|
|
|
|
|
/// ArcSwap makes this cheap and quick.
|
2022-06-24 14:52:38 -07:00
|
|
|
pub fn get_config() -> Config {
|
|
|
|
|
(*(*CONFIG.load())).clone()
|
2022-02-19 13:57:35 -08:00
|
|
|
}
|
|
|
|
|
|
2022-03-10 01:33:29 -08:00
|
|
|
/// Parse the configuration file located at the path.
|
2022-02-19 13:57:35 -08:00
|
|
|
pub async fn parse(path: &str) -> Result<(), Error> {
|
2022-02-08 09:25:59 -08:00
|
|
|
let mut contents = String::new();
|
|
|
|
|
let mut file = match File::open(path).await {
|
|
|
|
|
Ok(file) => file,
|
|
|
|
|
Err(err) => {
|
2022-02-21 20:41:32 -08:00
|
|
|
error!("Could not open '{}': {}", path, err.to_string());
|
2022-02-08 09:25:59 -08:00
|
|
|
return Err(Error::BadConfig);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
match file.read_to_string(&mut contents).await {
|
|
|
|
|
Ok(_) => (),
|
|
|
|
|
Err(err) => {
|
2022-02-21 20:41:32 -08:00
|
|
|
error!("Could not read config file: {}", err.to_string());
|
2022-02-08 09:25:59 -08:00
|
|
|
return Err(Error::BadConfig);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2022-02-27 10:21:24 -08:00
|
|
|
let mut config: Config = match toml::from_str(&contents) {
|
2022-02-08 09:25:59 -08:00
|
|
|
Ok(config) => config,
|
|
|
|
|
Err(err) => {
|
2022-02-21 20:41:32 -08:00
|
|
|
error!("Could not parse config file: {}", err.to_string());
|
2022-02-08 09:25:59 -08:00
|
|
|
return Err(Error::BadConfig);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2022-02-23 11:47:24 -08:00
|
|
|
match config.query_router.sharding_function.as_ref() {
|
|
|
|
|
"pg_bigint_hash" => (),
|
|
|
|
|
"sha1" => (),
|
|
|
|
|
_ => {
|
|
|
|
|
error!(
|
|
|
|
|
"Supported sharding functions are: 'pg_bigint_hash', 'sha1', got: '{}'",
|
|
|
|
|
config.query_router.sharding_function
|
|
|
|
|
);
|
|
|
|
|
return Err(Error::BadConfig);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2022-02-10 09:07:10 -08:00
|
|
|
// Quick config sanity check.
|
2022-02-09 21:19:14 -08:00
|
|
|
for shard in &config.shards {
|
2022-02-10 09:07:10 -08:00
|
|
|
// We use addresses as unique identifiers,
|
|
|
|
|
// let's make sure they are unique in the config as well.
|
2022-02-09 21:19:14 -08:00
|
|
|
let mut dup_check = HashSet::new();
|
2022-02-10 09:07:10 -08:00
|
|
|
let mut primary_count = 0;
|
2022-02-09 21:19:14 -08:00
|
|
|
|
2022-02-21 20:41:32 -08:00
|
|
|
match shard.0.parse::<usize>() {
|
|
|
|
|
Ok(_) => (),
|
|
|
|
|
Err(_) => {
|
|
|
|
|
error!(
|
|
|
|
|
"Shard '{}' is not a valid number, shards must be numbered starting at 0",
|
|
|
|
|
shard.0
|
|
|
|
|
);
|
|
|
|
|
return Err(Error::BadConfig);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2022-02-11 22:19:49 -08:00
|
|
|
if shard.1.servers.len() == 0 {
|
2022-02-20 22:47:08 -08:00
|
|
|
error!("Shard {} has no servers configured", shard.0);
|
2022-02-11 22:19:49 -08:00
|
|
|
return Err(Error::BadConfig);
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-09 21:19:14 -08:00
|
|
|
for server in &shard.1.servers {
|
|
|
|
|
dup_check.insert(server);
|
2022-02-10 09:07:10 -08:00
|
|
|
|
|
|
|
|
// Check that we define only zero or one primary.
|
|
|
|
|
match server.2.as_ref() {
|
|
|
|
|
"primary" => primary_count += 1,
|
|
|
|
|
_ => (),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Check role spelling.
|
|
|
|
|
match server.2.as_ref() {
|
|
|
|
|
"primary" => (),
|
|
|
|
|
"replica" => (),
|
|
|
|
|
_ => {
|
2022-02-20 22:47:08 -08:00
|
|
|
error!(
|
|
|
|
|
"Shard {} server role must be either 'primary' or 'replica', got: '{}'",
|
2022-02-10 09:07:10 -08:00
|
|
|
shard.0, server.2
|
|
|
|
|
);
|
|
|
|
|
return Err(Error::BadConfig);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if primary_count > 1 {
|
2022-02-20 22:47:08 -08:00
|
|
|
error!("Shard {} has more than on primary configured", &shard.0);
|
2022-02-10 09:07:10 -08:00
|
|
|
return Err(Error::BadConfig);
|
2022-02-09 21:19:14 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if dup_check.len() != shard.1.servers.len() {
|
2022-02-20 22:47:08 -08:00
|
|
|
error!("Shard {} contains duplicate server configs", &shard.0);
|
2022-02-09 21:19:14 -08:00
|
|
|
return Err(Error::BadConfig);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-11 11:19:40 -08:00
|
|
|
match config.query_router.default_role.as_ref() {
|
|
|
|
|
"any" => (),
|
|
|
|
|
"primary" => (),
|
|
|
|
|
"replica" => (),
|
|
|
|
|
other => {
|
2022-02-20 22:47:08 -08:00
|
|
|
error!(
|
|
|
|
|
"Query router default_role must be 'primary', 'replica', or 'any', got: '{}'",
|
2022-02-11 11:19:40 -08:00
|
|
|
other
|
|
|
|
|
);
|
|
|
|
|
return Err(Error::BadConfig);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2022-06-24 14:52:38 -07:00
|
|
|
config.path = path.to_string();
|
2022-02-27 10:21:24 -08:00
|
|
|
|
2022-03-10 01:33:29 -08:00
|
|
|
// Update the configuration globally.
|
2022-02-19 13:57:35 -08:00
|
|
|
CONFIG.store(Arc::new(config.clone()));
|
|
|
|
|
|
|
|
|
|
Ok(())
|
2022-02-08 09:25:59 -08:00
|
|
|
}
|
2022-02-08 17:08:17 -08:00
|
|
|
|
2022-06-25 11:46:20 -07:00
|
|
|
pub async fn reload_config(client_server_map: ClientServerMap) -> Result<bool, Error> {
|
2022-06-24 14:52:38 -07:00
|
|
|
let old_config = get_config();
|
|
|
|
|
|
|
|
|
|
match parse(&old_config.path).await {
|
|
|
|
|
Ok(()) => (),
|
|
|
|
|
Err(err) => {
|
|
|
|
|
error!("Config reload error: {:?}", err);
|
|
|
|
|
return Err(Error::BadConfig);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let new_config = get_config();
|
|
|
|
|
|
2022-06-25 11:46:20 -07:00
|
|
|
if old_config.shards != new_config.shards || old_config.user != new_config.user {
|
2022-06-24 14:52:38 -07:00
|
|
|
info!("Sharding configuration changed, re-creating server pools");
|
2022-06-25 11:46:20 -07:00
|
|
|
ConnectionPool::from_config(client_server_map).await?;
|
|
|
|
|
Ok(true)
|
|
|
|
|
} else if old_config != new_config {
|
|
|
|
|
Ok(true)
|
2022-06-24 14:52:38 -07:00
|
|
|
} else {
|
2022-06-25 11:46:20 -07:00
|
|
|
Ok(false)
|
2022-06-24 14:52:38 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-08 17:08:17 -08:00
|
|
|
#[cfg(test)]
|
|
|
|
|
mod test {
|
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
|
|
#[tokio::test]
|
|
|
|
|
async fn test_config() {
|
2022-02-19 13:57:35 -08:00
|
|
|
parse("pgcat.toml").await.unwrap();
|
|
|
|
|
assert_eq!(get_config().general.pool_size, 15);
|
|
|
|
|
assert_eq!(get_config().shards.len(), 3);
|
|
|
|
|
assert_eq!(get_config().shards["1"].servers[0].0, "127.0.0.1");
|
|
|
|
|
assert_eq!(get_config().shards["0"].servers[0].2, "primary");
|
|
|
|
|
assert_eq!(get_config().query_router.default_role, "any");
|
2022-06-24 14:52:38 -07:00
|
|
|
assert_eq!(get_config().path, "pgcat.toml".to_string());
|
2022-02-08 17:08:17 -08:00
|
|
|
}
|
|
|
|
|
}
|