use log::{debug, error, info}; use std::env; use http_types::Url; use hyper::client::{HttpConnector,connect::dns::GaiResolver}; use hyper_rustls::HttpsConnector; use warp::{Filter, host::Authority, path::FullPath}; #[tokio::main] async fn main() -> Result<(), kittybox::database::StorageError> { // TODO json logging in the future? let logger_env = env_logger::Env::new().filter_or("RUST_LOG", "info"); env_logger::init_from_env(logger_env); info!("Starting the kittybox server..."); let backend_uri: String; match env::var("BACKEND_URI") { Ok(val) => { debug!("Backend URI: {}", val); backend_uri = val } Err(_) => { error!("BACKEND_URI is not set, cannot find a database"); std::process::exit(1); } }; let token_endpoint: Url; match env::var("TOKEN_ENDPOINT") { Ok(val) => { debug!("Token endpoint: {}", val); match Url::parse(&val) { Ok(val) => token_endpoint = val, _ => { error!("Token endpoint URL cannot be parsed, aborting."); std::process::exit(1) } } } Err(_) => { error!("TOKEN_ENDPOINT is not set, will not be able to authorize users!"); std::process::exit(1) } } let authorization_endpoint: Url; match env::var("AUTHORIZATION_ENDPOINT") { Ok(val) => { debug!("Auth endpoint: {}", val); match Url::parse(&val) { Ok(val) => authorization_endpoint = val, _ => { error!("Authorization endpoint URL cannot be parsed, aborting."); std::process::exit(1) } } } Err(_) => { error!("AUTHORIZATION_ENDPOINT is not set, will not be able to confirm token and ID requests using IndieAuth!"); std::process::exit(1) } } let media_endpoint: Option<String> = env::var("MEDIA_ENDPOINT").ok(); let internal_token: Option<String> = env::var("KITTYBOX_INTERNAL_TOKEN").ok(); let cookie_secret: String = match env::var("COOKIE_SECRET").ok() { Some(value) => value, None => { if let Ok(filename) = env::var("COOKIE_SECRET_FILE") { use tokio::io::AsyncReadExt; let mut file = tokio::fs::File::open(filename).await?; let mut temp_string = String::new(); file.read_to_string(&mut temp_string).await?; temp_string } else { error!("COOKIE_SECRET or COOKIE_SECRET_FILE is not set, will not be able to log in users securely!"); std::process::exit(1); } } }; let host: std::net::SocketAddr = match env::var("SERVE_AT") .ok() .unwrap_or_else(|| "0.0.0.0:8080".to_string()) .parse() { Ok(addr) => addr, Err(e) => { error!("Cannot parse SERVE_AT: {}", e); std::process::exit(1); } }; let http_client: hyper::Client<HttpsConnector<HttpConnector<GaiResolver>>, hyper::Body> = { let builder = hyper::Client::builder(); let https = hyper_rustls::HttpsConnectorBuilder::new() .with_webpki_roots() .https_only() .enable_http1() .enable_http2() .build(); builder.build(https) }; if backend_uri.starts_with("redis") { println!("The Redis backend is deprecated."); std::process::exit(1); } else if backend_uri.starts_with("file") { let database = { let folder = backend_uri.strip_prefix("file://").unwrap(); let path = std::path::PathBuf::from(folder); kittybox::database::FileStorage::new(path).await? }; // TODO interpret HEAD let homepage = kittybox::util::require_host() .and(warp::get()) .and(warp::path::end()) // TODO fetch content from the database // TODO parse content-type and determine appropriate response .map(|host| format!("front page for {}!", host)); let micropub = warp::path("micropub") .and(warp::path::end() .and(kittybox::micropub::micropub( database.clone(), token_endpoint.to_string(), http_client.clone() )) .or(warp::get() .and(warp::path("client")) .and(warp::path::end()) .map(|| kittybox::MICROPUB_CLIENT))); let media = warp::path("media") .and(warp::path::end() .and(kittybox::util::require_host()) .map(|host| "media endpoint?...") .or(kittybox::util::require_host() .and(warp::path::param()) .map(|host: Authority, path: String| format!("media file {}", path)))); // TODO remember how login logic works because I forgor let login = warp::path("login") .and(warp::path("callback") .map(|| "callback!") // TODO form on GET and handler on POST .or(warp::path::end().map(|| "login page!"))); // TODO prettier error response let coffee = warp::path("coffee") .map(|| warp::reply::with_status("I'm a teapot!", warp::http::StatusCode::IM_A_TEAPOT)); // TODO interpret HEAD let static_files = warp::get() .and(warp::path!("static" / String)) .map(|path| path); // TODO interpret HEAD let catchall = warp::get() .and(kittybox::util::require_host()) .and(warp::path::full()) .map(|host: Authority, path: FullPath| host.to_string() + path.as_str() + ".json") // TODO fetch content from the database // TODO parse content-type and determine appropriate response ; let health = warp::path("health").and(warp::path::end()).map(|| "OK"); // TODO instrumentation middleware (see metrics.rs for comments) //let metrics = warp::path("metrics").and(warp::path::end()).map(kittybox::metrics::gather); let app = homepage .or(login) .or(static_files) .or(coffee) .or(health) .or(micropub) .or(media) .or(catchall) ; let server = warp::serve(app); // TODO https://github.com/seanmonstar/warp/issues/854 // TODO move to Hyper to wrap the requests in metrics info!("Listening on {:?}", host); server.bind(host).await; Ok(()) } else { println!("Unknown backend, not starting."); std::process::exit(1); } }