use std::collections::HashMap;
use std::path::PathBuf;
use std::sync::{Arc, RwLock};
use sonda_core::ScenarioHandle;
#[derive(Clone)]
pub struct AppState {
pub scenarios: Arc<RwLock<HashMap<String, ScenarioHandle>>>,
pub api_key: Option<Arc<String>>,
pub catalog_dir: Option<Arc<PathBuf>>,
}
impl AppState {
pub fn new() -> Self {
Self {
scenarios: Arc::new(RwLock::new(HashMap::new())),
api_key: None,
catalog_dir: None,
}
}
pub fn with_api_key(api_key: Option<String>) -> Self {
Self {
scenarios: Arc::new(RwLock::new(HashMap::new())),
api_key: api_key.map(Arc::new),
catalog_dir: None,
}
}
}
impl Default for AppState {
fn default() -> Self {
Self::new()
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn new_state_has_empty_scenarios() {
let state = AppState::new();
let scenarios = state.scenarios.read().expect("RwLock must not be poisoned");
assert!(
scenarios.is_empty(),
"new AppState must have an empty scenarios map"
);
}
#[test]
fn new_state_has_no_api_key() {
let state = AppState::new();
assert!(
state.api_key.is_none(),
"new AppState must have api_key = None"
);
}
#[test]
fn default_produces_empty_state() {
let state = AppState::default();
let scenarios = state.scenarios.read().expect("RwLock must not be poisoned");
assert!(
scenarios.is_empty(),
"default AppState must have an empty scenarios map"
);
assert!(
state.api_key.is_none(),
"default AppState must have api_key = None"
);
}
#[test]
fn with_api_key_some_stores_key() {
let state = AppState::with_api_key(Some("secret".to_string()));
let key = state.api_key.expect("api_key must be Some");
assert_eq!(*key, "secret", "api_key must contain the provided value");
}
#[test]
fn with_api_key_none_disables_auth() {
let state = AppState::with_api_key(None);
assert!(
state.api_key.is_none(),
"with_api_key(None) must produce api_key = None"
);
}
#[test]
fn with_api_key_has_empty_scenarios() {
let state = AppState::with_api_key(Some("key".to_string()));
let scenarios = state.scenarios.read().expect("RwLock must not be poisoned");
assert!(
scenarios.is_empty(),
"with_api_key must produce an empty scenarios map"
);
}
#[test]
fn clone_shares_same_arc() {
let state1 = AppState::new();
let state2 = state1.clone();
assert!(
Arc::ptr_eq(&state1.scenarios, &state2.scenarios),
"cloned AppState must share the same Arc<RwLock<...>>"
);
}
#[test]
fn clone_shares_api_key_arc() {
let state1 = AppState::with_api_key(Some("secret".to_string()));
let state2 = state1.clone();
assert!(
Arc::ptr_eq(
state1.api_key.as_ref().unwrap(),
state2.api_key.as_ref().unwrap()
),
"cloned AppState must share the same api_key Arc"
);
}
#[test]
fn constructors_default_catalog_dir_to_none() {
assert!(AppState::new().catalog_dir.is_none());
assert!(AppState::with_api_key(None).catalog_dir.is_none());
assert!(AppState::with_api_key(Some("k".to_string()))
.catalog_dir
.is_none());
}
#[test]
fn catalog_dir_is_carried_and_shared_on_clone() {
let mut state = AppState::new();
state.catalog_dir = Some(Arc::new(PathBuf::from("/scenarios")));
let clone = state.clone();
assert!(Arc::ptr_eq(
state.catalog_dir.as_ref().unwrap(),
clone.catalog_dir.as_ref().unwrap()
));
assert_eq!(
clone.catalog_dir.as_ref().unwrap().as_path(),
std::path::Path::new("/scenarios")
);
}
#[test]
fn app_state_is_send_and_sync() {
fn assert_send_sync<T: Send + Sync>() {}
assert_send_sync::<AppState>();
}
#[test]
fn app_state_is_clone() {
fn assert_clone<T: Clone>() {}
assert_clone::<AppState>();
}
}