pub mod api; pub mod auth; pub mod db; pub mod env; pub mod executor; pub mod rpc; #[cfg(test)] mod tests; use axum::{http::StatusCode, response::IntoResponse}; use db::Database; use executor::Executor; use serde::Deserialize; use std::{path::PathBuf, sync::Arc}; pub type Result = std::result::Result; pub enum Error { Http(StatusCode, String), Database(sea_orm::error::DbErr), Internal(anyhow::Error), } impl From for Error { fn from(error: anyhow::Error) -> Self { Self::Internal(error) } } impl From for Error { fn from(error: sea_orm::error::DbErr) -> Self { Self::Database(error) } } impl From for Error { fn from(error: axum::Error) -> Self { Self::Internal(error.into()) } } impl From for Error { fn from(error: hyper::Error) -> Self { Self::Internal(error.into()) } } impl From for Error { fn from(error: serde_json::Error) -> Self { Self::Internal(error.into()) } } impl IntoResponse for Error { fn into_response(self) -> axum::response::Response { match self { Error::Http(code, message) => (code, message).into_response(), Error::Database(error) => { (StatusCode::INTERNAL_SERVER_ERROR, format!("{}", &error)).into_response() } Error::Internal(error) => { (StatusCode::INTERNAL_SERVER_ERROR, format!("{}", &error)).into_response() } } } } impl std::fmt::Debug for Error { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { match self { Error::Http(code, message) => (code, message).fmt(f), Error::Database(error) => error.fmt(f), Error::Internal(error) => error.fmt(f), } } } impl std::fmt::Display for Error { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { match self { Error::Http(code, message) => write!(f, "{code}: {message}"), Error::Database(error) => error.fmt(f), Error::Internal(error) => error.fmt(f), } } } impl std::error::Error for Error {} #[derive(Default, Deserialize)] pub struct Config { pub http_port: u16, pub database_url: String, pub database_max_connections: u32, pub api_token: String, pub invite_link_prefix: String, pub live_kit_server: Option, pub live_kit_key: Option, pub live_kit_secret: Option, pub rust_log: Option, pub log_json: Option, pub zed_environment: String, } #[derive(Default, Deserialize)] pub struct MigrateConfig { pub database_url: String, pub migrations_path: Option, } pub struct AppState { pub db: Arc, pub live_kit_client: Option>, pub config: Config, } impl AppState { pub async fn new(config: Config) -> Result> { let mut db_options = db::ConnectOptions::new(config.database_url.clone()); db_options.max_connections(config.database_max_connections); let mut db = Database::new(db_options, Executor::Production).await?; db.initialize_notification_kinds().await?; let live_kit_client = if let Some(((server, key), secret)) = config .live_kit_server .as_ref() .zip(config.live_kit_key.as_ref()) .zip(config.live_kit_secret.as_ref()) { Some(Arc::new(live_kit_server::api::LiveKitClient::new( server.clone(), key.clone(), secret.clone(), )) as Arc) } else { None }; let this = Self { db: Arc::new(db), live_kit_client, config, }; Ok(Arc::new(this)) } }