use crate::config::Config; use actix_web::HttpServer; use actix_web::http::KeepAlive; use anyhow::Result; use app::make_app; 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::notifier::push_notifier; use rustical_frontend::nextcloud_login::NextcloudFlows; 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; 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?; if config.dav_push.enabled { tokio::spawn(push_notifier( config.dav_push.allowed_push_servers, update_recv, subscription_store.clone(), )); } let nextcloud_flows = Arc::new(NextcloudFlows::default()); HttpServer::new(move || { make_app( addr_store.clone(), cal_store.clone(), subscription_store.clone(), principal_store.clone(), config.frontend.clone(), config.oidc.clone(), config.nextcloud_login.clone(), nextcloud_flows.clone(), ) }) .bind((config.http.host, config.http.port))? // Workaround for a weird bug where // new requests might timeout since they cannot properly reuse the connection // https://github.com/lennart-k/rustical/issues/10 .keep_alive(KeepAlive::Disabled) .run() .await?; } } Ok(()) } #[cfg(test)] mod tests { use crate::{ app::make_app, commands::generate_frontend_secret, config::NextcloudLoginConfig, get_data_stores, }; use actix_web::{http::StatusCode, test::TestRequest}; use rustical_frontend::FrontendConfig; use rustical_frontend::nextcloud_login::NextcloudFlows; use std::sync::Arc; #[tokio::test] async fn test_main() { let (addr_store, cal_store, subscription_store, principal_store, _update_recv) = get_data_stores( true, &crate::config::DataStoreConfig::Sqlite(crate::config::SqliteDataStoreConfig { db_url: "".to_owned(), }), ) .await .unwrap(); let app = make_app( addr_store, cal_store, subscription_store, principal_store, FrontendConfig { enabled: false, secret_key: generate_frontend_secret(), allow_password_login: false, }, None, NextcloudLoginConfig { enabled: false }, Arc::new(NextcloudFlows::default()), ); let app = actix_web::test::init_service(app).await; let req = TestRequest::get().uri("/").to_request(); let resp = actix_web::test::call_service(&app, req).await; assert_eq!(resp.status(), StatusCode::NOT_FOUND); } }