use anyhow::Result; use axum::routing::{delete, get, patch, post}; use axum::Router; use tracing_subscriber::{layer::SubscriberExt, util::SubscriberInitExt}; mod auth; mod config; mod db; mod error; mod handlers; mod models; mod services; mod state; use config::AppConfig; use state::AppState; #[tokio::main] async fn main() -> Result<()> { dotenvy::dotenv().ok(); tracing_subscriber::registry() .with(tracing_subscriber::EnvFilter::try_from_default_env().unwrap_or_else(|_| { "eventsnap_backend=debug,tower_http=debug".into() })) .with(tracing_subscriber::fmt::layer()) .init(); let config = AppConfig::from_env()?; let pool = db::create_pool(&config.database_url).await?; let state = AppState::new(pool, config.clone()); // Ensure media directories exist tokio::fs::create_dir_all(&config.media_path).await.ok(); let api = Router::new() // Auth .route("/api/v1/join", post(auth::handlers::join)) .route("/api/v1/recover", post(auth::handlers::recover)) .route("/api/v1/admin/login", post(auth::handlers::admin_login)) .route("/api/v1/session", delete(auth::handlers::logout)) // Upload .route("/api/v1/upload", post(handlers::upload::upload)) .route( "/api/v1/upload/{id}", patch(handlers::upload::edit_upload).delete(handlers::upload::delete_upload), ) // SSE .route("/api/v1/stream", get(handlers::sse::stream)); let router = Router::new() .route("/health", get(|| async { "ok" })) .merge(api) .with_state(state); let listener = tokio::net::TcpListener::bind(("0.0.0.0", config.app_port)).await?; tracing::info!("listening on {}", listener.local_addr()?); axum::serve(listener, router).await?; Ok(()) }