use async_trait::async_trait;
use std::sync::Arc;
use crate::analytics::{AnalyticsProvider, FingerprintProvider};
use crate::auth::UserProvider;
pub trait AppContext: Send + Sync {
fn config(&self) -> Arc<dyn ConfigProvider>;
fn database_handle(&self) -> Arc<dyn DatabaseHandle>;
fn analytics_provider(&self) -> Option<Arc<dyn AnalyticsProvider>>;
fn fingerprint_provider(&self) -> Option<Arc<dyn FingerprintProvider>>;
fn user_provider(&self) -> Option<Arc<dyn UserProvider>>;
}
pub trait InjectContextHeaders {
fn inject_headers(&self, headers: &mut http::HeaderMap);
}
pub trait ContextPropagation {
fn from_headers(headers: &http::HeaderMap) -> anyhow::Result<Self>
where
Self: Sized;
fn to_headers(&self) -> http::HeaderMap;
}
pub trait ConfigProvider: Send + Sync {
fn get(&self, key: &str) -> Option<String>;
fn database_url(&self) -> &str;
fn database_write_url(&self) -> Option<&str> {
None
}
fn system_path(&self) -> &str;
fn api_port(&self) -> u16;
fn as_any(&self) -> &dyn std::any::Any;
}
pub trait ModuleRegistry: Send + Sync {
fn get_module(&self, name: &str) -> Option<Arc<dyn Module>>;
fn list_modules(&self) -> Vec<String>;
}
pub trait DatabaseHandle: Send + Sync {
fn is_connected(&self) -> bool;
fn as_any(&self) -> &dyn std::any::Any;
}
#[async_trait]
pub trait Module: Send + Sync {
fn name(&self) -> &str;
fn version(&self) -> &str;
fn display_name(&self) -> &str;
async fn initialize(&self) -> Result<(), Box<dyn std::error::Error>>;
}
#[cfg(feature = "web")]
#[async_trait]
pub trait ApiModule: Module {
fn router(&self, ctx: Arc<dyn AppContext>) -> axum::Router;
}