Skip to main content

server/
config.rs

1use crate::db::get_connection;
2use derive_more::From;
3use rust_i18n::t;
4use sqlx::Error::RowNotFound;
5use std::env::var;
6use std::fmt;
7use std::ops::Deref;
8use std::sync::LazyLock;
9use thiserror::Error;
10
11/// Represents an error that can occur while accessing the configuration.
12///
13/// # Variants
14///
15/// - `NoConfig(String)`: Returned when a specific configuration field is not found.
16/// - `DB`: Indicates a database access error.
17/// - `Sqlx`: An error propagated from the `SQLx` crate.
18///
19/// # Example
20///
21/// ```rust
22/// use thiserror::Error;
23///
24/// #[derive(Debug, Error)]
25/// pub enum ConfigError {
26///     #[error("No such config field: {0}")]
27///     NoConfig(String),
28///     #[error("Can't access db")]
29///     DB,
30///     #[error("Sqlx")]
31///     Sqlx(#[from] sqlx::Error),
32/// }
33/// ```
34#[derive(Debug, Error)]
35pub enum ConfigError {
36    #[error("No such config field: {0}")]
37    NoConfig(String),
38    #[error("Can't access db")]
39    DB,
40    #[error("Sqlx")]
41    Sqlx(#[from] sqlx::Error),
42}
43
44/// Represents a configuration value stored in the system.
45///
46/// # Variants
47///
48/// - `String`: Stores the configuration as a string.
49/// - `Blob`: Stores the configuration as a binary blob.
50///
51/// # Example
52///
53/// ```rust
54/// use server::config::ConfigOption;
55/// let config_str = ConfigOption::String("example".to_string());
56/// let config_blob = ConfigOption::Blob(vec![1, 2, 3, 4]);
57/// ```
58///
59/// You can also convert `String` and `Vec<u8>` directly into `ConfigOption`
60/// using the `From` trait:
61///
62/// ```rust
63/// use server::config::ConfigOption;
64/// let config: ConfigOption = "example".to_string().into();
65/// let blob: ConfigOption = vec![1, 2, 3, 4].into();
66/// ```
67#[derive(Debug, From, PartialEq, Clone)]
68pub enum ConfigOption {
69    String(#[from] String),
70    Blob(#[from] Vec<u8>),
71}
72
73impl fmt::Display for ConfigOption {
74    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
75        match self {
76            ConfigOption::String(s) => write!(f, "{s}"),
77            ConfigOption::Blob(b) => write!(f, "ConfigOption<Blob>: \"{}\" bytes", b.len()),
78        }
79    }
80}
81
82/// Convert `ConfigOption` into a `&str` reference when the variant is a `String`.
83///
84/// # Panics
85///
86/// This will panic if the `ConfigOption` is of type `Blob`, as it cannot be
87/// converted to a `&str`.
88///
89/// # Example
90///
91/// ```rust,should_panic
92/// use server::config::ConfigOption;
93/// let config = ConfigOption::Blob(vec![1, 2, 3]);
94/// let str_ref: &str = config.as_ref(); // Will panic
95/// ```
96impl AsRef<str> for ConfigOption {
97    fn as_ref(&self) -> &str {
98        match self {
99            ConfigOption::String(s) => s.as_str(),
100            ConfigOption::Blob(_) => panic!("ConfigOption::Blob cannot be converted to &str"),
101        }
102    }
103}
104
105/// Dereference `ConfigOption::String` to `&str`.
106///
107/// This allows you to use `ConfigOption` directly in contexts where a `&str`
108/// is expected, without having to manually call `.as_ref()`.
109///
110/// # Panics
111///
112/// This will panic if the `ConfigOption` is of type `Blob`.
113///
114/// # Example
115///
116/// ```rust,should_panic
117/// use server::config::ConfigOption;
118/// let config = ConfigOption::Blob(vec![1, 2, 3]);
119/// let str_ref: &str = &*config; // Will panic
120/// ```
121impl Deref for ConfigOption {
122    type Target = str;
123
124    fn deref(&self) -> &Self::Target {
125        match self {
126            ConfigOption::String(s) => s.as_str(),
127            ConfigOption::Blob(_) => panic!("ConfigOption::Blob cannot be dereferenced as &str"),
128        }
129    }
130}
131
132/// Allows for the conversion from a `&str` to `ConfigOption`.
133///
134/// This is useful for passing string literals or string slices directly as `ConfigOption`.
135/// The conversion will wrap the `&str` in the `ConfigOption::String` variant.
136///
137/// # Example
138///
139/// ```rust
140/// use server::config::ConfigOption;
141/// let config_option: ConfigOption = "example text".into();
142/// assert_eq!(config_option, ConfigOption::String("example text".to_string()));
143/// ```
144impl From<&str> for ConfigOption {
145    fn from(value: &str) -> Self {
146        ConfigOption::String(value.to_string())
147    }
148}
149
150/// Convert `ConfigOption` into a `&[u8]` reference when the variant is a `Blob`.
151///
152/// # Example
153///
154/// ```rust
155/// use server::config::ConfigOption;
156/// let config = ConfigOption::Blob(vec![1, 2, 3]);
157/// let byte_ref: &[u8] = config.as_ref(); // Works
158/// ```
159impl AsRef<[u8]> for ConfigOption {
160    fn as_ref(&self) -> &[u8] {
161        match self {
162            ConfigOption::String(s) => s.as_bytes(),
163            ConfigOption::Blob(b) => b.as_slice(),
164        }
165    }
166}
167
168/// Fetches the `contents` of a field from the configdata.
169///
170/// This function attempts to retrieve the configuration stored in the
171/// `configdata` table.  If the field is not found, it returns a
172/// `ConfigError::NoConfig`.
173///
174/// # Arguments
175///
176/// - `field`: The name of the field to retrieve.
177///
178/// # Returns
179///
180/// - `Ok(Some(ConfigOption))` if the field exists.
181/// - `Ok(None)` if the field doesn't exist.
182/// - `Err(ConfigError)` if there is an error during retrieval.
183///
184/// # Example
185///
186/// ```rust,ignore
187/// let config = system_configdata("example_field").await?;
188/// if let Some(option) = config {
189///     println!("Config value: {}", option);
190/// }
191/// ```
192///
193/// # Errors
194///
195/// Returns a `ConfigError::DB` if there is an issue with the database connection.
196// The config read/write logic is identical for the system (admin DB) and
197// per-user surfaces — only the connection differs. The `*_on` helpers take an
198// explicit connection so both surfaces share one implementation; `User::config`
199// / `User::set_config` (see `user.rs`) pass a per-user connection, while the
200// `system_*` wrappers below pass the global admin connection.
201
202async fn configdata_on(
203    conn: &mut sqlx::PgConnection,
204    field: &str,
205) -> Result<Option<ConfigOption>, ConfigError> {
206    let value = sqlx::query_file_scalar!("../server/sql/select/config/data.sql", field)
207        .fetch_one(&mut *conn)
208        .await
209        .map_err(|err| {
210            if let RowNotFound = err {
211                ConfigError::NoConfig(String::from(field))
212            } else {
213                log::error!("{}", t!("Database error: %{err}", err = err : {:?}));
214                ConfigError::DB
215            }
216        })?;
217
218    Ok(Some(ConfigOption::Blob(value)))
219}
220
221/// Reads a config field on the given connection: the `config` (string) table
222/// first, falling back to `configdata` (blob).
223///
224/// # Errors
225/// `ConfigError::DB` on a database error.
226pub async fn config_on(
227    conn: &mut sqlx::PgConnection,
228    field: &str,
229) -> Result<Option<ConfigOption>, ConfigError> {
230    let value = sqlx::query_file_scalar!("../server/sql/select/config/string.sql", field)
231        .fetch_one(&mut *conn)
232        .await;
233    match value {
234        Ok(l) => Ok(Some(ConfigOption::String(l))),
235        Err(RowNotFound) => configdata_on(conn, field).await,
236        Err(err) => {
237            log::error!("{}", t!("Database error: %{err}", err = err : {:?}));
238            Err(ConfigError::DB)
239        }
240    }
241}
242
243/// Upserts a config field on the given connection (string → `config`, blob →
244/// `configdata`). The upsert keys on `lower(field)` (see the SQL), so it is
245/// idempotent and case-insensitive.
246///
247/// # Errors
248/// `ConfigError::DB` on a database error, `ConfigError::NoConfig` if the row is
249/// missing for an update.
250pub async fn set_config_on(
251    conn: &mut sqlx::PgConnection,
252    field: &str,
253    contents: ConfigOption,
254) -> Result<(), ConfigError> {
255    let id = uuid::Uuid::new_v4();
256    match contents {
257        ConfigOption::String(s) => {
258            sqlx::query_file!("../server/sql/set/config/string.sql", &id, field, s)
259        }
260        ConfigOption::Blob(b) => {
261            sqlx::query_file!("../server/sql/set/config/blob.sql", &id, field, b)
262        }
263    }
264    .execute(&mut *conn)
265    .await
266    .map_err(|err| {
267        if let RowNotFound = err {
268            ConfigError::NoConfig(String::from(field))
269        } else {
270            log::error!("{}", t!("Database error: %{err}", err = err : {:?}));
271            ConfigError::DB
272        }
273    })?;
274
275    Ok(())
276}
277
278/// Reads a SYSTEM (server-wide) config field from the global admin database.
279/// For server-wide keys (infra URLs, locale) and the bootstrap reads that run
280/// before any user is resolved. Per-user config goes through `User::config`.
281///
282/// # Errors
283/// `ConfigError::DB` on a database error.
284pub async fn system_config(field: &str) -> Result<Option<ConfigOption>, ConfigError> {
285    let mut conn = get_connection().await.map_err(|err| {
286        log::error!("{}", t!("Database error: %{err}", err = err : {:?}));
287        ConfigError::DB
288    })?;
289    config_on(&mut conn, field).await
290}
291
292/// Reads a SYSTEM config blob field from the global admin database.
293///
294/// # Errors
295/// `ConfigError::DB` on a database error.
296pub async fn system_configdata(field: &str) -> Result<Option<ConfigOption>, ConfigError> {
297    let mut conn = get_connection().await.map_err(|err| {
298        log::error!("{}", t!("Database error: %{err}", err = err : {:?}));
299        ConfigError::DB
300    })?;
301    configdata_on(&mut conn, field).await
302}
303
304/// Writes a SYSTEM config field to the global admin database.
305///
306/// # Errors
307/// `ConfigError::DB` on a database error.
308pub async fn set_system_config(field: &str, contents: ConfigOption) -> Result<(), ConfigError> {
309    let mut conn = get_connection().await.map_err(|err| {
310        log::error!("{}", t!("Database error: %{err}", err = err : {:?}));
311        ConfigError::DB
312    })?;
313    set_config_on(&mut conn, field, contents).await
314}
315
316static LOCALE_NAME: LazyLock<String> = LazyLock::new(|| var("LANG").unwrap_or(String::from("en")));
317
318pub async fn load_config() -> Result<(), ConfigError> {
319    if let Ok(Some(locale)) = system_config("locale").await {
320        rust_i18n::set_locale(&locale);
321        log::info!(
322            "{}",
323            t!(
324                "Loaded the %{loc} locale from config, setting as main",
325                loc = &locale
326            )
327        );
328    } else {
329        rust_i18n::set_locale(&LOCALE_NAME);
330    }
331    Ok(())
332}
333
334#[cfg(test)]
335mod config_tests {
336    use super::*;
337    use crate::db::DB_POOL;
338    use sqlx::PgPool;
339    use supp_macro::local_db_sqlx_test;
340    use tokio::sync::OnceCell;
341
342    /// Context for keeping environment intact
343    static CONTEXT: OnceCell<()> = OnceCell::const_new();
344
345    async fn setup() {
346        CONTEXT
347            .get_or_init(|| async {
348                #[cfg(feature = "testlog")]
349                let _ = env_logger::builder()
350                    .is_test(true)
351                    .filter_level(log::LevelFilter::Trace)
352                    .try_init();
353            })
354            .await;
355    }
356
357    #[local_db_sqlx_test]
358    async fn test_config_string(pool: PgPool) {
359        let opt = ConfigOption::String("testval".to_string());
360        set_system_config("testfield", opt.clone()).await.unwrap();
361        let val = system_config("testfield").await.unwrap().unwrap();
362        assert_eq!(val, opt);
363    }
364
365    #[local_db_sqlx_test]
366    async fn test_config_configdata(pool: PgPool) {
367        let opt = ConfigOption::Blob(vec![0, 1, 2, 3]);
368        set_system_config("testblob", opt.clone()).await.unwrap();
369        let val = system_config("testblob").await.unwrap().unwrap();
370        assert_eq!(val, opt);
371    }
372}