use crate::config::Config; use anyhow::Result; use app::make_app; use axum::ServiceExt; use axum::extract::Request; use clap::{Parser, Subcommand}; use commands::principals::{PrincipalsArgs, cmd_principals}; use commands::{cmd_gen_config, cmd_pwhash}; use config::{DataStoreConfig, SqliteDataStoreConfig}; use figment::Figment; use figment::providers::{Env, Format, Toml}; use rustical_dav_push::DavPushController; use rustical_store::auth::AuthenticationProvider; use rustical_store::{AddressbookStore, CalendarStore, CollectionOperation, SubscriptionStore}; use rustical_store_sqlite::addressbook_store::SqliteAddressbookStore; use rustical_store_sqlite::calendar_store::SqliteCalendarStore; use rustical_store_sqlite::principal_store::SqlitePrincipalStore; use rustical_store_sqlite::{SqliteStore, create_db_pool}; use setup_tracing::setup_tracing; use std::sync::Arc; use tokio::sync::mpsc::Receiver; use tower::Layer; use tower_http::normalize_path::NormalizePathLayer; use tracing::info; mod app; mod commands; mod config; mod setup_tracing; #[derive(Parser, Debug)] #[command(author, version, about, long_about = None)] struct Args { #[arg(short, long, env, default_value = "/etc/rustical/config.toml")] config_file: String, #[arg(long, env, help = "Do no run database migrations (only for sql store)")] no_migrations: bool, #[command(subcommand)] command: Option, } #[derive(Debug, Subcommand)] enum Command { GenConfig(commands::GenConfigArgs), Pwhash(commands::PwhashArgs), Principals(PrincipalsArgs), } async fn get_data_stores( migrate: bool, config: &DataStoreConfig, ) -> Result<( Arc, Arc, Arc, Arc, Receiver, )> { Ok(match &config { DataStoreConfig::Sqlite(SqliteDataStoreConfig { db_url }) => { let db = create_db_pool(db_url, migrate).await?; // Channel to watch for changes (for DAV Push) let (send, recv) = tokio::sync::mpsc::channel(1000); let addressbook_store = Arc::new(SqliteAddressbookStore::new(db.clone(), send.clone())); let cal_store = Arc::new(SqliteCalendarStore::new(db.clone(), send)); let subscription_store = Arc::new(SqliteStore::new(db.clone())); let principal_store = Arc::new(SqlitePrincipalStore::new(db.clone())); ( addressbook_store, cal_store, subscription_store, principal_store, recv, ) } }) } #[tokio::main] async fn main() -> Result<()> { let args = Args::parse(); match args.command { Some(Command::GenConfig(gen_config_args)) => cmd_gen_config(gen_config_args)?, Some(Command::Pwhash(pwhash_args)) => cmd_pwhash(pwhash_args)?, Some(Command::Principals(principals_args)) => cmd_principals(principals_args).await?, None => { let config: Config = Figment::new() .merge(Toml::file(&args.config_file)) .merge(Env::prefixed("RUSTICAL_").split("__")) .extract()?; setup_tracing(&config.tracing); let (addr_store, cal_store, subscription_store, principal_store, update_recv) = get_data_stores(!args.no_migrations, &config.data_store).await?; let mut tasks = vec![]; if config.dav_push.enabled { let dav_push_controller = DavPushController::new( config.dav_push.allowed_push_servers, subscription_store.clone(), ); tasks.push(tokio::spawn(async move { dav_push_controller.notifier(update_recv).await; })); } let app = make_app( addr_store.clone(), cal_store.clone(), subscription_store.clone(), principal_store.clone(), config.frontend.clone(), config.oidc.clone(), config.nextcloud_login.clone(), config.dav_push.enabled, ); let app = ServiceExt::::into_make_service( NormalizePathLayer::trim_trailing_slash().layer(app), ); let address = format!("{}:{}", config.http.host, config.http.port); let listener = tokio::net::TcpListener::bind(&address).await?; tasks.push(tokio::spawn(async move { info!("RustiCal serving on http://{address}"); axum::serve(listener, app).await.unwrap() })); for task in tasks { task.await?; } } } Ok(()) }