use { async_std::{ net::{TcpListener, TcpStream}, stream::StreamExt, task, }, async_tls::{TlsAcceptor, server::TlsStream}, lazy_static::lazy_static, rustls::internal::pemfile::{certs, pkcs8_private_keys}, std::{ error::Error, fs::{File, read}, io::BufReader, path::PathBuf, sync::Arc, }, url::Url, }; pub type Result = std::result::Result>; lazy_static! { static ref ARGS: Args = args().expect("usage: agate "); static ref ACCEPTOR: TlsAcceptor = acceptor().unwrap(); } struct Args { sock_addr: String, content_dir: String, cert_file: String, key_file: String, } fn main() -> Result { task::block_on(async { let listener = TcpListener::bind(&ARGS.sock_addr).await?; let mut incoming = listener.incoming(); while let Some(Ok(stream)) = incoming.next().await { task::spawn(async { if let Err(e) = connection(stream).await { eprintln!("Error: {:?}", e); } }); } Ok(()) }) } fn args() -> Option { let mut args = std::env::args().skip(1); Some(Args { sock_addr: args.next()?, content_dir: args.next()?, cert_file: args.next()?, key_file: args.next()?, }) } fn acceptor() -> Result { let cert_file = File::open(&ARGS.cert_file)?; let key_file = File::open(&ARGS.key_file)?; let certs = certs(&mut BufReader::new(cert_file)).or(Err("bad cert"))?; let mut keys = pkcs8_private_keys(&mut BufReader::new(key_file)).or(Err("bad key"))?; let mut config = rustls::ServerConfig::new(rustls::NoClientAuth::new()); config.set_single_cert(certs, keys.remove(0))?; Ok(TlsAcceptor::from(Arc::new(config))) } async fn connection(stream: TcpStream) -> Result { use async_std::io::prelude::*; let mut stream = ACCEPTOR.accept(stream).await?; let url = match parse_request(&mut stream).await { Ok(url) => url, Err(e) => { stream.write_all(b"50 Invalid request.\r\n").await?; return Err(e) } }; match get(&url) { Ok(response) => { stream.write_all(b"20 text/gemini\r\n").await?; stream.write_all(&response).await?; } Err(e) => { stream.write_all(b"40 Not found, sorry.\r\n").await?; return Err(e) } } Ok(()) } async fn parse_request(stream: &mut TlsStream) -> Result { use async_std::io::prelude::*; let mut stream = async_std::io::BufReader::new(stream); let mut request = String::new(); stream.read_line(&mut request).await?; let url = Url::parse(request.trim())?; Ok(url) } fn get(url: &Url) -> Result> { let mut path = PathBuf::from(&ARGS.content_dir); path.extend(url.path_segments().unwrap()); if !path.starts_with(&ARGS.content_dir) { Err("invalid path")? } eprintln!("Got request for {:?}", path); if path.is_dir() { path.push("index.gemini"); } let response = read(&path)?; Ok(response) }