From 18824c9c9f9913b8bbecbaa1b022428bae9bf710 Mon Sep 17 00:00:00 2001 From: Yuri Astrakhan Date: Fri, 20 Oct 2023 16:04:52 -0400 Subject: [PATCH] Use SomeEnum instead of Option in cfg Simplify code by adding `None` to the enums we use for configuration --- martin/src/args/pg.rs | 26 +-- martin/src/args/root.rs | 16 +- martin/src/config.rs | 70 +++---- martin/src/file_config.rs | 177 +++++++++--------- martin/src/lib.rs | 2 +- martin/src/pg/config.rs | 48 ++--- martin/src/pg/configurator.rs | 144 +++++++------- martin/src/pg/mod.rs | 2 - martin/src/sprites/mod.rs | 27 ++- .../{one_or_many.rs => cfg_containers.rs} | 56 ++++-- martin/src/utils/mod.rs | 4 +- martin/src/utils/utilities.rs | 10 +- martin/tests/pg_server_test.rs | 4 +- martin/tests/utils/pg_utils.rs | 2 - 14 files changed, 291 insertions(+), 297 deletions(-) rename martin/src/utils/{one_or_many.rs => cfg_containers.rs} (59%) diff --git a/martin/src/args/pg.rs b/martin/src/args/pg.rs index f77c06716..e02a062e7 100644 --- a/martin/src/args/pg.rs +++ b/martin/src/args/pg.rs @@ -4,7 +4,7 @@ use crate::args::connections::Arguments; use crate::args::connections::State::{Ignore, Take}; use crate::args::environment::Env; use crate::pg::{PgConfig, PgSslCerts, POOL_SIZE_DEFAULT}; -use crate::utils::OneOrMany; +use crate::utils::{NoneBoolObj, NoneOneMany}; #[derive(clap::Args, Debug, PartialEq, Default)] #[command(about, version)] @@ -30,7 +30,7 @@ impl PgArgs { self, cli_strings: &mut Arguments, env: &impl Env<'a>, - ) -> Option> { + ) -> NoneOneMany { let connections = Self::extract_conn_strings(cli_strings, env); let default_srid = self.get_default_srid(env); let certs = self.get_certs(env); @@ -48,20 +48,20 @@ impl PgArgs { }, max_feature_count: self.max_feature_count, pool_size: self.pool_size, - auto_publish: None, + auto_publish: NoneBoolObj::None, tables: None, functions: None, }) .collect(); match results.len() { - 0 => None, - 1 => Some(OneOrMany::One(results.into_iter().next().unwrap())), - _ => Some(OneOrMany::Many(results)), + 0 => NoneOneMany::None, + 1 => NoneOneMany::One(results.into_iter().next().unwrap()), + _ => NoneOneMany::Many(results), } } - pub fn override_config<'a>(self, pg_config: &mut OneOrMany, env: &impl Env<'a>) { + pub fn override_config<'a>(self, pg_config: &mut NoneOneMany, env: &impl Env<'a>) { if self.default_srid.is_some() { info!("Overriding configured default SRID to {} on all Postgres connections because of a CLI parameter", self.default_srid.unwrap()); pg_config.iter_mut().for_each(|c| { @@ -224,10 +224,10 @@ mod tests { let config = PgArgs::default().into_config(&mut args, &FauxEnv::default()); assert_eq!( config, - Some(OneOrMany::One(PgConfig { + NoneOneMany::One(PgConfig { connection_string: some("postgres://localhost:5432"), ..Default::default() - })) + }) ); assert!(args.check().is_ok()); } @@ -248,7 +248,7 @@ mod tests { let config = PgArgs::default().into_config(&mut args, &env); assert_eq!( config, - Some(OneOrMany::One(PgConfig { + NoneOneMany::One(PgConfig { connection_string: some("postgres://localhost:5432"), default_srid: Some(10), ssl_certificates: PgSslCerts { @@ -256,7 +256,7 @@ mod tests { ..Default::default() }, ..Default::default() - })) + }) ); assert!(args.check().is_ok()); } @@ -282,7 +282,7 @@ mod tests { let config = pg_args.into_config(&mut args, &env); assert_eq!( config, - Some(OneOrMany::One(PgConfig { + NoneOneMany::One(PgConfig { connection_string: some("postgres://localhost:5432"), default_srid: Some(20), ssl_certificates: PgSslCerts { @@ -291,7 +291,7 @@ mod tests { ssl_root_cert: Some(PathBuf::from("root")), }, ..Default::default() - })) + }) ); assert!(args.check().is_ok()); } diff --git a/martin/src/args/root.rs b/martin/src/args/root.rs index 1201279b9..5b2b11d5d 100644 --- a/martin/src/args/root.rs +++ b/martin/src/args/root.rs @@ -62,11 +62,11 @@ impl Args { let mut cli_strings = Arguments::new(self.meta.connection); let pg_args = self.pg.unwrap_or_default(); - if let Some(pg_config) = &mut config.postgres { - // config was loaded from a file, we can only apply a few CLI overrides to it - pg_args.override_config(pg_config, env); - } else { + if config.postgres.is_none() { config.postgres = pg_args.into_config(&mut cli_strings, env); + } else { + // config was loaded from a file, we can only apply a few CLI overrides to it + pg_args.override_config(&mut config.postgres, env); } if !cli_strings.is_empty() { @@ -85,7 +85,7 @@ impl Args { } } -pub fn parse_file_args(cli_strings: &mut Arguments, extension: &str) -> Option { +pub fn parse_file_args(cli_strings: &mut Arguments, extension: &str) -> FileConfigEnum { let paths = cli_strings.process(|v| match PathBuf::try_from(v) { Ok(v) => { if v.is_dir() { @@ -107,7 +107,7 @@ mod tests { use super::*; use crate::pg::PgConfig; use crate::test_utils::{some, FauxEnv}; - use crate::utils::OneOrMany; + use crate::utils::NoneOneMany; fn parse(args: &[&str]) -> Result<(Config, MetaArgs)> { let args = Args::parse_from(args); @@ -143,10 +143,10 @@ mod tests { let args = parse(&["martin", "postgres://connection"]).unwrap(); let cfg = Config { - postgres: Some(OneOrMany::One(PgConfig { + postgres: NoneOneMany::One(PgConfig { connection_string: some("postgres://connection"), ..Default::default() - })), + }), ..Default::default() }; let meta = MetaArgs { diff --git a/martin/src/config.rs b/martin/src/config.rs index 5856bed23..52db0851e 100644 --- a/martin/src/config.rs +++ b/martin/src/config.rs @@ -17,7 +17,7 @@ use crate::source::{TileInfoSources, TileSources}; use crate::sprites::SpriteSources; use crate::srv::SrvConfig; use crate::Error::{ConfigLoadError, ConfigParseError, NoSources}; -use crate::{IdResolver, OneOrMany, Result}; +use crate::{IdResolver, NoneOneMany, Result}; pub type UnrecognizedValues = HashMap; @@ -31,17 +31,17 @@ pub struct Config { #[serde(flatten)] pub srv: SrvConfig, - #[serde(skip_serializing_if = "Option::is_none")] - pub postgres: Option>, + #[serde(default, skip_serializing_if = "NoneOneMany::is_none")] + pub postgres: NoneOneMany, - #[serde(skip_serializing_if = "Option::is_none")] - pub pmtiles: Option, + #[serde(default, skip_serializing_if = "FileConfigEnum::is_none")] + pub pmtiles: FileConfigEnum, - #[serde(skip_serializing_if = "Option::is_none")] - pub mbtiles: Option, + #[serde(default, skip_serializing_if = "FileConfigEnum::is_none")] + pub mbtiles: FileConfigEnum, - #[serde(skip_serializing_if = "Option::is_none")] - pub sprites: Option, + #[serde(default, skip_serializing_if = "FileConfigEnum::is_none")] + pub sprites: FileConfigEnum, #[serde(flatten)] pub unrecognized: UnrecognizedValues, @@ -53,40 +53,22 @@ impl Config { let mut res = UnrecognizedValues::new(); copy_unrecognized_config(&mut res, "", &self.unrecognized); - let mut any = if let Some(pg) = &mut self.postgres { - for pg in pg.iter_mut() { - res.extend(pg.finalize()?); - } - !pg.is_empty() - } else { - false - }; - - any |= if let Some(cfg) = &mut self.pmtiles { - res.extend(cfg.finalize("pmtiles.")?); - !cfg.is_empty() - } else { - false - }; + for pg in self.postgres.iter_mut() { + res.extend(pg.finalize()?); + } - any |= if let Some(cfg) = &mut self.mbtiles { - res.extend(cfg.finalize("mbtiles.")?); - !cfg.is_empty() - } else { - false - }; + res.extend(self.pmtiles.finalize("pmtiles.")?); + res.extend(self.mbtiles.finalize("mbtiles.")?); + res.extend(self.sprites.finalize("sprites.")?); - any |= if let Some(cfg) = &mut self.sprites { - res.extend(cfg.finalize("sprites.")?); - !cfg.is_empty() + if self.postgres.is_empty() + && self.pmtiles.is_empty() + && self.mbtiles.is_empty() + && self.sprites.is_empty() + { + Err(NoSources) } else { - false - }; - - if any { Ok(res) - } else { - Err(NoSources) } } @@ -102,18 +84,16 @@ impl Config { let create_mbt_src = &mut MbtSource::new_box; let mut sources: Vec>>>> = Vec::new(); - if let Some(v) = self.postgres.as_mut() { - for s in v.iter_mut() { - sources.push(Box::pin(s.resolve(idr.clone()))); - } + for s in self.postgres.iter_mut() { + sources.push(Box::pin(s.resolve(idr.clone()))); } - if self.pmtiles.is_some() { + if !self.pmtiles.is_empty() { let val = resolve_files(&mut self.pmtiles, idr.clone(), "pmtiles", create_pmt_src); sources.push(Box::pin(val)); } - if self.mbtiles.is_some() { + if !self.mbtiles.is_empty() { let val = resolve_files(&mut self.mbtiles, idr.clone(), "mbtiles", create_mbt_src); sources.push(Box::pin(val)); } diff --git a/martin/src/file_config.rs b/martin/src/file_config.rs index 9cb55a359..acfcd813b 100644 --- a/martin/src/file_config.rs +++ b/martin/src/file_config.rs @@ -10,8 +10,8 @@ use serde::{Deserialize, Serialize}; use crate::config::{copy_unrecognized_config, UnrecognizedValues}; use crate::file_config::FileError::{InvalidFilePath, InvalidSourceFilePath, IoError}; use crate::source::{Source, TileInfoSources}; -use crate::utils::{sorted_opt_map, Error, IdResolver, OneOrMany}; -use crate::OneOrMany::{Many, One}; +use crate::utils::{sorted_opt_map, Error, IdResolver, NoneOneMany}; +use crate::NoneOneMany::{Many, One}; #[derive(thiserror::Error, Debug)] pub enum FileError { @@ -31,9 +31,11 @@ pub enum FileError { AquireConnError(String), } -#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)] +#[derive(Clone, Debug, Default, PartialEq, Serialize, Deserialize)] #[serde(untagged)] pub enum FileConfigEnum { + #[default] + None, Path(PathBuf), Paths(Vec), Config(FileConfig), @@ -41,7 +43,7 @@ pub enum FileConfigEnum { impl FileConfigEnum { #[must_use] - pub fn new(paths: Vec) -> Option { + pub fn new(paths: Vec) -> FileConfigEnum { Self::new_extended(paths, HashMap::new(), UnrecognizedValues::new()) } @@ -50,46 +52,70 @@ impl FileConfigEnum { paths: Vec, configs: HashMap, unrecognized: UnrecognizedValues, - ) -> Option { + ) -> FileConfigEnum { if configs.is_empty() && unrecognized.is_empty() { match paths.len() { - 0 => None, - 1 => Some(FileConfigEnum::Path(paths.into_iter().next().unwrap())), - _ => Some(FileConfigEnum::Paths(paths)), + 0 => FileConfigEnum::None, + 1 => FileConfigEnum::Path(paths.into_iter().next().unwrap()), + _ => FileConfigEnum::Paths(paths), } } else { - Some(FileConfigEnum::Config(FileConfig { - paths: OneOrMany::new_opt(paths), + FileConfigEnum::Config(FileConfig { + paths: NoneOneMany::new(paths), sources: if configs.is_empty() { None } else { Some(configs) }, unrecognized, - })) + }) + } + } + + #[must_use] + pub fn is_none(&self) -> bool { + matches!(self, Self::None) + } + + #[must_use] + pub fn is_empty(&self) -> bool { + match self { + Self::None => true, + Self::Path(_) => false, + Self::Paths(v) => v.is_empty(), + Self::Config(c) => c.is_empty(), } } - pub fn extract_file_config(&mut self) -> FileConfig { + pub fn extract_file_config(&mut self) -> Option { match self { - FileConfigEnum::Path(path) => FileConfig { - paths: Some(One(mem::take(path))), + FileConfigEnum::None => None, + FileConfigEnum::Path(path) => Some(FileConfig { + paths: One(mem::take(path)), ..FileConfig::default() - }, - FileConfigEnum::Paths(paths) => FileConfig { - paths: Some(Many(mem::take(paths))), + }), + FileConfigEnum::Paths(paths) => Some(FileConfig { + paths: Many(mem::take(paths)), ..Default::default() - }, - FileConfigEnum::Config(cfg) => mem::take(cfg), + }), + FileConfigEnum::Config(cfg) => Some(mem::take(cfg)), } } + + pub fn finalize(&self, prefix: &str) -> Result { + let mut res = UnrecognizedValues::new(); + if let Self::Config(cfg) = self { + copy_unrecognized_config(&mut res, prefix, &cfg.unrecognized); + } + Ok(res) + } } #[derive(Clone, Debug, Default, PartialEq, Serialize, Deserialize)] pub struct FileConfig { /// A list of file paths - #[serde(skip_serializing_if = "Option::is_none")] - pub paths: Option>, + #[serde(default, skip_serializing_if = "NoneOneMany::is_none")] + pub paths: NoneOneMany, /// A map of source IDs to file paths or config objects #[serde(skip_serializing_if = "Option::is_none")] #[serde(serialize_with = "sorted_opt_map")] @@ -128,27 +154,8 @@ pub struct FileConfigSource { pub path: PathBuf, } -impl FileConfigEnum { - pub fn finalize(&self, prefix: &str) -> Result { - let mut res = UnrecognizedValues::new(); - if let Self::Config(cfg) = self { - copy_unrecognized_config(&mut res, prefix, &cfg.unrecognized); - } - Ok(res) - } - - #[must_use] - pub fn is_empty(&self) -> bool { - match self { - Self::Path(_) => false, - Self::Paths(v) => v.is_empty(), - Self::Config(c) => c.is_empty(), - } - } -} - pub async fn resolve_files( - config: &mut Option, + config: &mut FileConfigEnum, idr: IdResolver, extension: &str, create_source: &mut impl FnMut(String, PathBuf) -> Fut, @@ -162,7 +169,7 @@ where } async fn resolve_int( - config: &mut Option, + config: &mut FileConfigEnum, idr: IdResolver, extension: &str, create_source: &mut impl FnMut(String, PathBuf) -> Fut, @@ -170,10 +177,9 @@ async fn resolve_int( where Fut: Future, FileError>>, { - let Some(cfg) = config else { + let Some(cfg) = config.extract_file_config() else { return Ok(TileInfoSources::default()); }; - let cfg = cfg.extract_file_config(); let mut results = TileInfoSources::default(); let mut configs = HashMap::new(); @@ -202,50 +208,47 @@ where } } - if let Some(paths) = cfg.paths { - for path in paths { - let is_dir = path.is_dir(); - let dir_files = if is_dir { - // directories will be kept in the config just in case there are new files - directories.push(path.clone()); - path.read_dir() - .map_err(|e| IoError(e, path.clone()))? - .filter_map(Result::ok) - .filter(|f| { - f.path().extension().filter(|e| *e == extension).is_some() - && f.path().is_file() - }) - .map(|f| f.path()) - .collect() - } else if path.is_file() { - vec![path] - } else { - return Err(InvalidFilePath(path.canonicalize().unwrap_or(path))); - }; - for path in dir_files { - let can = path.canonicalize().map_err(|e| IoError(e, path.clone()))?; - if files.contains(&can) { - if !is_dir { - warn!("Ignoring duplicate MBTiles path: {}", can.display()); - } - continue; + for path in cfg.paths { + let is_dir = path.is_dir(); + let dir_files = if is_dir { + // directories will be kept in the config just in case there are new files + directories.push(path.clone()); + path.read_dir() + .map_err(|e| IoError(e, path.clone()))? + .filter_map(Result::ok) + .filter(|f| { + f.path().extension().filter(|e| *e == extension).is_some() && f.path().is_file() + }) + .map(|f| f.path()) + .collect() + } else if path.is_file() { + vec![path] + } else { + return Err(InvalidFilePath(path.canonicalize().unwrap_or(path))); + }; + for path in dir_files { + let can = path.canonicalize().map_err(|e| IoError(e, path.clone()))?; + if files.contains(&can) { + if !is_dir { + warn!("Ignoring duplicate MBTiles path: {}", can.display()); } - let id = path.file_stem().map_or_else( - || "_unknown".to_string(), - |s| s.to_string_lossy().to_string(), - ); - let source = FileConfigSrc::Path(path); - let id = idr.resolve(&id, can.to_string_lossy().to_string()); - info!("Configured source {id} from {}", can.display()); - files.insert(can); - configs.insert(id.clone(), source.clone()); - - let path = match source { - FileConfigSrc::Obj(pmt) => pmt.path, - FileConfigSrc::Path(path) => path, - }; - results.push(create_source(id, path).await?); + continue; } + let id = path.file_stem().map_or_else( + || "_unknown".to_string(), + |s| s.to_string_lossy().to_string(), + ); + let source = FileConfigSrc::Path(path); + let id = idr.resolve(&id, can.to_string_lossy().to_string()); + info!("Configured source {id} from {}", can.display()); + files.insert(can); + configs.insert(id.clone(), source.clone()); + + let path = match source { + FileConfigSrc::Obj(pmt) => pmt.path, + FileConfigSrc::Path(path) => path, + }; + results.push(create_source(id, path).await?); } } @@ -280,7 +283,7 @@ mod tests { let FileConfigEnum::Config(cfg) = cfg else { panic!(); }; - let paths = cfg.paths.clone().unwrap().into_iter().collect::>(); + let paths = cfg.paths.clone().into_iter().collect::>(); assert_eq!( paths, vec![ diff --git a/martin/src/lib.rs b/martin/src/lib.rs index 6915d0517..04f6cfe49 100644 --- a/martin/src/lib.rs +++ b/martin/src/lib.rs @@ -31,7 +31,7 @@ pub use crate::args::Env; pub use crate::config::{read_config, Config, ServerState}; pub use crate::source::Source; pub use crate::utils::{ - decode_brotli, decode_gzip, BoolOrObject, Error, IdResolver, OneOrMany, Result, + decode_brotli, decode_gzip, Error, IdResolver, NoneBoolObj, NoneOneMany, Result, }; // Ensure README.md contains valid code diff --git a/martin/src/pg/config.rs b/martin/src/pg/config.rs index 1934756cb..9959e5b78 100644 --- a/martin/src/pg/config.rs +++ b/martin/src/pg/config.rs @@ -11,7 +11,7 @@ use crate::pg::config_table::TableInfoSources; use crate::pg::configurator::PgBuilder; use crate::pg::Result; use crate::source::TileInfoSources; -use crate::utils::{on_slow, sorted_opt_map, BoolOrObject, IdResolver, OneOrMany}; +use crate::utils::{on_slow, sorted_opt_map, IdResolver, NoneBoolObj, NoneOneMany}; pub trait PgInfo { fn format_id(&self) -> String; @@ -47,8 +47,8 @@ pub struct PgConfig { pub max_feature_count: Option, #[serde(skip_serializing_if = "Option::is_none")] pub pool_size: Option, - #[serde(skip_serializing_if = "Option::is_none")] - pub auto_publish: Option>, + #[serde(default, skip_serializing_if = "NoneBoolObj::is_none")] + pub auto_publish: NoneBoolObj, #[serde(skip_serializing_if = "Option::is_none")] #[serde(serialize_with = "sorted_opt_map")] pub tables: Option, @@ -59,29 +59,29 @@ pub struct PgConfig { #[derive(Clone, Debug, Default, PartialEq, Serialize, Deserialize)] pub struct PgCfgPublish { - #[serde(skip_serializing_if = "Option::is_none")] + #[serde(default, skip_serializing_if = "NoneOneMany::is_none")] #[serde(alias = "from_schema")] - pub from_schemas: Option>, - #[serde(skip_serializing_if = "Option::is_none")] - pub tables: Option>, - #[serde(skip_serializing_if = "Option::is_none")] - pub functions: Option>, + pub from_schemas: NoneOneMany, + #[serde(default, skip_serializing_if = "NoneBoolObj::is_none")] + pub tables: NoneBoolObj, + #[serde(default, skip_serializing_if = "NoneBoolObj::is_none")] + pub functions: NoneBoolObj, } #[derive(Clone, Debug, Default, PartialEq, Serialize, Deserialize)] pub struct PgCfgPublishType { - #[serde(skip_serializing_if = "Option::is_none")] + #[serde(default, skip_serializing_if = "NoneOneMany::is_none")] #[serde(alias = "from_schema")] - pub from_schemas: Option>, + pub from_schemas: NoneOneMany, #[serde(skip_serializing_if = "Option::is_none")] #[serde(alias = "id_format")] pub source_id_format: Option, /// A table column to use as the feature ID /// If a table has no column with this name, `id_column` will not be set for that table. /// If a list of strings is given, the first found column will be treated as a feature ID. - #[serde(skip_serializing_if = "Option::is_none")] + #[serde(default, skip_serializing_if = "NoneOneMany::is_none")] #[serde(alias = "id_column")] - pub id_columns: Option>, + pub id_columns: NoneOneMany, #[serde(skip_serializing_if = "Option::is_none")] pub clip_geom: Option, #[serde(skip_serializing_if = "Option::is_none")] @@ -105,7 +105,7 @@ impl PgConfig { } } if self.tables.is_none() && self.functions.is_none() && self.auto_publish.is_none() { - self.auto_publish = Some(BoolOrObject::Bool(true)); + self.auto_publish = NoneBoolObj::Bool(true); } Ok(res) @@ -143,7 +143,7 @@ mod tests { use crate::pg::config_function::FunctionInfo; use crate::pg::config_table::TableInfo; use crate::test_utils::some; - use crate::utils::OneOrMany::{Many, One}; + use crate::utils::NoneOneMany::{Many, One}; #[test] fn parse_pg_one() { @@ -153,11 +153,11 @@ mod tests { connection_string: 'postgresql://postgres@localhost/db' "}, &Config { - postgres: Some(One(PgConfig { + postgres: One(PgConfig { connection_string: some("postgresql://postgres@localhost/db"), - auto_publish: Some(BoolOrObject::Bool(true)), + auto_publish: NoneBoolObj::Bool(true), ..Default::default() - })), + }), ..Default::default() }, ); @@ -172,18 +172,18 @@ mod tests { - connection_string: 'postgresql://postgres@localhost:5433/db' "}, &Config { - postgres: Some(Many(vec![ + postgres: Many(vec![ PgConfig { connection_string: some("postgres://postgres@localhost:5432/db"), - auto_publish: Some(BoolOrObject::Bool(true)), + auto_publish: NoneBoolObj::Bool(true), ..Default::default() }, PgConfig { connection_string: some("postgresql://postgres@localhost:5433/db"), - auto_publish: Some(BoolOrObject::Bool(true)), + auto_publish: NoneBoolObj::Bool(true), ..Default::default() }, - ])), + ]), ..Default::default() }, ); @@ -225,7 +225,7 @@ mod tests { bounds: [-180.0, -90.0, 180.0, 90.0] "}, &Config { - postgres: Some(One(PgConfig { + postgres: One(PgConfig { connection_string: some("postgres://postgres@localhost:5432/db"), default_srid: Some(4326), pool_size: Some(20), @@ -262,7 +262,7 @@ mod tests { ), )])), ..Default::default() - })), + }), ..Default::default() }, ); diff --git a/martin/src/pg/configurator.rs b/martin/src/pg/configurator.rs index 0cd51d615..debaf8c3c 100755 --- a/martin/src/pg/configurator.rs +++ b/martin/src/pg/configurator.rs @@ -18,7 +18,7 @@ use crate::pg::utils::{find_info, find_kv_ignore_case, normalize_key, InfoMap}; use crate::pg::PgError::InvalidTableExtent; use crate::pg::Result; use crate::source::TileInfoSources; -use crate::utils::{BoolOrObject, IdResolver, OneOrMany}; +use crate::utils::{IdResolver, NoneBoolObj, NoneOneMany}; pub type SqlFuncInfoMapMap = InfoMap>; pub type SqlTableInfoMapMapMap = InfoMap>>; @@ -346,70 +346,70 @@ fn new_auto_publish(config: &PgConfig, is_function: bool) -> Option match if is_function { &a.functions } else { &a.tables } { - Some(bo_i) => match bo_i { - BoolOrObject::Object(item) => Some(PgBuilderAuto { - source_id_format: item - .source_id_format - .as_ref() - .cloned() - .unwrap_or_else(|| default_id_fmt(is_function)), - schemas: merge_opt_hs(&a.from_schemas, &item.from_schemas), - id_columns: item.id_columns.as_ref().and_then(|ids| { - if is_function { - error!("Configuration parameter auto_publish.functions.id_columns is not supported"); - None - } else { - Some(ids.iter().cloned().collect()) - } - }), - clip_geom: { - if is_function { - error!("Configuration parameter auto_publish.functions.clip_geom is not supported"); - None - } else { - item.clip_geom - } - }, - buffer: { - if is_function { - error!("Configuration parameter auto_publish.functions.buffer is not supported"); - None - } else { - item.buffer - } - }, - extent: { - if is_function { - error!("Configuration parameter auto_publish.functions.extent is not supported"); - None - } else { - item.extent - } - }, - }), - BoolOrObject::Bool(true) => default(merge_opt_hs(&a.from_schemas, &None)), - BoolOrObject::Bool(false) => None, + match &config.auto_publish { + NoneBoolObj::None => { + if config.tables.is_some() || config.functions.is_some() { + None + } else { + default(None) + } + } + NoneBoolObj::Object(a) => match if is_function { &a.functions } else { &a.tables } { + // If auto_publish.functions is set, and currently asking for .tables which is missing, + // .tables becomes the inverse of functions (i.e. an obj or true in tables means false in functions) + NoneBoolObj::None => match if is_function { &a.tables } else { &a.functions } { + NoneBoolObj::None | NoneBoolObj::Bool(false) => { + default(merge_opt_hs(&a.from_schemas, &NoneOneMany::None)) + } + NoneBoolObj::Object(_) | NoneBoolObj::Bool(true) => None, + }, + NoneBoolObj::Object(item) => Some(PgBuilderAuto { + source_id_format: item + .source_id_format + .as_ref() + .cloned() + .unwrap_or_else(|| default_id_fmt(is_function)), + schemas: merge_opt_hs(&a.from_schemas, &item.from_schemas), + id_columns: { + if item.id_columns.is_none() { + None + } else if is_function { + error!("Configuration parameter auto_publish.functions.id_columns is not supported"); + None + } else { + Some(item.id_columns.iter().cloned().collect()) + } }, - // If auto_publish.functions is set, and currently asking for .tables which is missing, - // .tables becomes the inverse of functions (i.e. an obj or true in tables means false in functions) - None => match if is_function { &a.tables } else { &a.functions } { - Some(bo_i) => match bo_i { - BoolOrObject::Object(_) | BoolOrObject::Bool(true) => None, - BoolOrObject::Bool(false) => default(merge_opt_hs(&a.from_schemas, &None)), - }, - None => default(merge_opt_hs(&a.from_schemas, &None)), + clip_geom: { + if is_function { + error!("Configuration parameter auto_publish.functions.clip_geom is not supported"); + None + } else { + item.clip_geom + } }, - }, - BoolOrObject::Bool(true) => default(None), - BoolOrObject::Bool(false) => None, - } - } else if config.tables.is_some() || config.functions.is_some() { - None - } else { - default(None) + buffer: { + if is_function { + error!("Configuration parameter auto_publish.functions.buffer is not supported"); + None + } else { + item.buffer + } + }, + extent: { + if is_function { + error!("Configuration parameter auto_publish.functions.extent is not supported"); + None + } else { + item.extent + } + }, + }), + NoneBoolObj::Bool(true) => default(merge_opt_hs(&a.from_schemas, &NoneOneMany::None)), + NoneBoolObj::Bool(false) => None, + }, + NoneBoolObj::Bool(true) => default(None), + NoneBoolObj::Bool(false) => None, } } @@ -443,18 +443,16 @@ fn by_key(a: &(String, T), b: &(String, T)) -> Ordering { } /// Merge two optional list of strings into a hashset -fn merge_opt_hs( - a: &Option>, - b: &Option>, -) -> Option> { - if let Some(a) = a { - let mut res: HashSet<_> = a.iter().cloned().collect(); - if let Some(b) = b { +fn merge_opt_hs(a: &NoneOneMany, b: &NoneOneMany) -> Option> { + match (a.is_none(), b.is_none()) { + (true, true) => None, + (true, false) => Some(b.iter().cloned().collect()), + (false, true) => Some(a.iter().cloned().collect()), + (false, false) => { + let mut res: HashSet<_> = a.iter().cloned().collect(); res.extend(b.iter().cloned()); + Some(res) } - Some(res) - } else { - b.as_ref().map(|b| b.iter().cloned().collect()) } } diff --git a/martin/src/pg/mod.rs b/martin/src/pg/mod.rs index 3437322e7..d16468fac 100644 --- a/martin/src/pg/mod.rs +++ b/martin/src/pg/mod.rs @@ -16,5 +16,3 @@ pub use config_table::TableInfo; pub use errors::{PgError, Result}; pub use function_source::query_available_function; pub use pool::{PgPool, POOL_SIZE_DEFAULT}; - -pub use crate::utils::BoolOrObject; diff --git a/martin/src/sprites/mod.rs b/martin/src/sprites/mod.rs index cb120d4b9..ac2b3f4a3 100644 --- a/martin/src/sprites/mod.rs +++ b/martin/src/sprites/mod.rs @@ -55,12 +55,11 @@ pub type SpriteCatalog = BTreeMap; pub struct SpriteSources(HashMap); impl SpriteSources { - pub fn resolve(config: &mut Option) -> Result { - let Some(cfg) = config else { + pub fn resolve(config: &mut FileConfigEnum) -> Result { + let Some(cfg) = config.extract_file_config() else { return Ok(Self::default()); }; - let cfg = cfg.extract_file_config(); let mut results = Self::default(); let mut directories = Vec::new(); let mut configs = HashMap::new(); @@ -72,18 +71,16 @@ impl SpriteSources { } }; - if let Some(paths) = cfg.paths { - for path in paths { - let Some(name) = path.file_name() else { - warn!( - "Ignoring sprite source with no name from {}", - path.display() - ); - continue; - }; - directories.push(path.clone()); - results.add_source(name.to_string_lossy().to_string(), path); - } + for path in cfg.paths { + let Some(name) = path.file_name() else { + warn!( + "Ignoring sprite source with no name from {}", + path.display() + ); + continue; + }; + directories.push(path.clone()); + results.add_source(name.to_string_lossy().to_string(), path); } *config = FileConfigEnum::new_extended(directories, configs, cfg.unrecognized); diff --git a/martin/src/utils/one_or_many.rs b/martin/src/utils/cfg_containers.rs similarity index 59% rename from martin/src/utils/one_or_many.rs rename to martin/src/utils/cfg_containers.rs index c7cdd7808..ba2b0675e 100644 --- a/martin/src/utils/one_or_many.rs +++ b/martin/src/utils/cfg_containers.rs @@ -2,27 +2,47 @@ use std::vec::IntoIter; use serde::{Deserialize, Serialize}; -#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)] +/// A serde helper to store a boolean as an object. +#[derive(Clone, Debug, Default, PartialEq, Serialize, Deserialize)] #[serde(untagged)] -pub enum OneOrMany { +pub enum NoneBoolObj { + #[default] + #[serde(skip)] + None, + Bool(bool), + Object(T), +} + +impl NoneBoolObj { + pub fn is_none(&self) -> bool { + matches!(self, Self::None) + } +} + +#[derive(Debug, Default, Clone, PartialEq, Serialize, Deserialize)] +#[serde(untagged)] +pub enum NoneOneMany { + #[default] + None, One(T), Many(Vec), } -impl IntoIterator for OneOrMany { +impl IntoIterator for NoneOneMany { type Item = T; type IntoIter = IntoIter; fn into_iter(self) -> Self::IntoIter { match self { + Self::None => Vec::new().into_iter(), Self::One(v) => vec![v].into_iter(), Self::Many(v) => v.into_iter(), } } } -impl OneOrMany { - pub fn new_opt>(iter: I) -> Option { +impl NoneOneMany { + pub fn new>(iter: I) -> Self { let mut iter = iter.into_iter(); match (iter.next(), iter.next()) { (Some(first), Some(second)) => { @@ -30,15 +50,20 @@ impl OneOrMany { vec.push(first); vec.push(second); vec.extend(iter); - Some(Self::Many(vec)) + Self::Many(vec) } - (Some(first), None) => Some(Self::One(first)), - (None, _) => None, + (Some(first), None) => Self::One(first), + (None, _) => Self::None, } } + pub fn is_none(&self) -> bool { + matches!(self, Self::None) + } + pub fn is_empty(&self) -> bool { match self { + Self::None => true, Self::One(_) => false, Self::Many(v) => v.is_empty(), } @@ -46,13 +71,15 @@ impl OneOrMany { pub fn iter(&self) -> impl Iterator { match self { - OneOrMany::Many(v) => v.iter(), - OneOrMany::One(v) => std::slice::from_ref(v).iter(), + Self::None => [].iter(), + NoneOneMany::Many(v) => v.iter(), + NoneOneMany::One(v) => std::slice::from_ref(v).iter(), } } pub fn iter_mut(&mut self) -> impl Iterator { match self { + Self::None => [].iter_mut(), Self::Many(v) => v.iter_mut(), Self::One(v) => std::slice::from_mut(v).iter_mut(), } @@ -60,6 +87,7 @@ impl OneOrMany { pub fn as_slice(&self) -> &[T] { match self { + Self::None => &[], Self::One(item) => std::slice::from_ref(item), Self::Many(v) => v.as_slice(), } @@ -69,16 +97,16 @@ impl OneOrMany { #[cfg(test)] mod tests { use super::*; - use crate::OneOrMany::{Many, One}; + use crate::NoneOneMany::{Many, None, One}; #[test] fn test_one_or_many() { let mut one = One(1); let mut many = Many(vec![1, 2, 3]); - assert_eq!(OneOrMany::new_opt(vec![1, 2, 3]), Some(Many(vec![1, 2, 3]))); - assert_eq!(OneOrMany::new_opt(vec![1]), Some(One(1))); - assert_eq!(OneOrMany::new_opt(Vec::::new()), None); + assert_eq!(NoneOneMany::new(vec![1, 2, 3]), Many(vec![1, 2, 3])); + assert_eq!(NoneOneMany::new(vec![1]), One(1)); + assert_eq!(NoneOneMany::new(Vec::::new()), None); assert_eq!(one.iter_mut().collect::>(), vec![&1]); assert_eq!(many.iter_mut().collect::>(), vec![&1, &2, &3]); diff --git a/martin/src/utils/mod.rs b/martin/src/utils/mod.rs index 85534e500..51d7f807f 100644 --- a/martin/src/utils/mod.rs +++ b/martin/src/utils/mod.rs @@ -1,11 +1,11 @@ +mod cfg_containers; mod error; mod id_resolver; -mod one_or_many; mod utilities; mod xyz; +pub use cfg_containers::{NoneBoolObj, NoneOneMany}; pub use error::*; pub use id_resolver::IdResolver; -pub use one_or_many::OneOrMany; pub use utilities::*; pub use xyz::Xyz; diff --git a/martin/src/utils/utilities.rs b/martin/src/utils/utilities.rs index e05e79276..c4893c34c 100644 --- a/martin/src/utils/utilities.rs +++ b/martin/src/utils/utilities.rs @@ -6,17 +6,9 @@ use std::time::Duration; use flate2::read::GzDecoder; use flate2::write::GzEncoder; use futures::pin_mut; -use serde::{Deserialize, Serialize, Serializer}; +use serde::{Serialize, Serializer}; use tokio::time::timeout; -/// A serde helper to store a boolean as an object. -#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)] -#[serde(untagged)] -pub enum BoolOrObject { - Bool(bool), - Object(T), -} - /// Sort an optional hashmap by key, case-insensitive first, then case-sensitive pub fn sorted_opt_map( value: &Option>, diff --git a/martin/tests/pg_server_test.rs b/martin/tests/pg_server_test.rs index c814f554d..74c53206b 100644 --- a/martin/tests/pg_server_test.rs +++ b/martin/tests/pg_server_test.rs @@ -4,7 +4,7 @@ use actix_web::test::{call_and_read_body_json, call_service, read_body, TestRequ use ctor::ctor; use indoc::indoc; use insta::assert_yaml_snapshot; -use martin::OneOrMany; +use martin::NoneOneMany; use tilejson::TileJSON; pub mod utils; @@ -1092,7 +1092,7 @@ tables: ) .await; - let OneOrMany::One(cfg) = cfg.postgres.unwrap() else { + let NoneOneMany::One(cfg) = cfg.postgres else { panic!() }; for (name, _) in cfg.tables.unwrap_or_default() { diff --git a/martin/tests/utils/pg_utils.rs b/martin/tests/utils/pg_utils.rs index da5da7d3d..27b689420 100644 --- a/martin/tests/utils/pg_utils.rs +++ b/martin/tests/utils/pg_utils.rs @@ -34,8 +34,6 @@ pub fn table<'a>(mock: &'a MockSource, name: &str) -> &'a TableInfo { let (_, config) = mock; let vals: Vec<&TableInfo> = config .postgres - .as_ref() - .unwrap() .iter() .flat_map(|v| v.tables.iter().map(|vv| vv.get(name))) .flatten()