http-server-cc/src/main.rs

155 lines
4.3 KiB
Rust
Raw Normal View History

2024-05-10 18:37:00 +00:00
// #![feature(if_let_guard)]
2024-05-10 18:33:15 +00:00
2024-05-10 22:04:51 +00:00
use std::{ collections::HashMap, path::PathBuf, sync::Arc };
2024-05-10 17:50:36 +00:00
use anyhow::Result;
2024-05-10 17:06:42 +00:00
2024-05-10 20:27:11 +00:00
use tokio::{
2024-05-10 22:04:51 +00:00
fs::File,
io::{ AsyncBufReadExt, AsyncReadExt, AsyncWriteExt, BufReader },
net::{ TcpListener, TcpStream },
2024-05-10 20:27:11 +00:00
};
2024-05-10 19:29:51 +00:00
mod utils;
use utils::*;
2024-05-10 17:50:36 +00:00
2024-05-10 22:04:51 +00:00
#[derive(Debug, Clone)]
struct Args {
pub directory: PathBuf,
}
type A = Arc<Args>;
fn parse_args () -> Args {
let directory = std::env::args().position(|e| e == "--directory").unwrap() + 1;
let directory = PathBuf::from(std::env::args().nth(directory).unwrap());
Args {
directory
}
}
2024-05-10 20:27:11 +00:00
#[tokio::main]
async fn main() -> Result<()> {
let listener = TcpListener::bind("127.0.0.1:4221").await.unwrap();
2024-05-10 22:04:51 +00:00
let args = Arc::new(parse_args());
2024-05-10 17:23:14 +00:00
2024-05-10 20:27:11 +00:00
loop {
2024-05-10 22:04:51 +00:00
let args = args.clone();
2024-05-10 20:27:11 +00:00
let Ok((socket, info)) = listener.accept().await else { continue; } ;
println!("Connection from {:?} accepted, processing…", info);
2024-05-10 22:04:51 +00:00
let _ = process(socket, args).await;
2024-05-10 20:27:11 +00:00
}
2024-05-10 17:50:36 +00:00
2024-05-10 20:27:11 +00:00
}
2024-05-10 17:50:36 +00:00
2024-05-10 22:04:51 +00:00
async fn process (mut stream: TcpStream, args: A) -> Result<()> {
2024-05-10 20:27:11 +00:00
let buf_reader = BufReader::new(&mut stream);
2024-05-10 17:50:36 +00:00
2024-05-10 20:27:11 +00:00
let mut data = buf_reader
.lines();
2024-05-10 19:29:51 +00:00
2024-05-10 20:27:11 +00:00
let (_method, path, _ver) = {
let start_line = data.next_line().await?.ok_or(E::InvalidRequest)?; // should be 500;
let mut parts = start_line.split_whitespace().map(ToOwned::to_owned);
let method = parts.next().ok_or(E::InvalidRequest)?;
let path = parts.next().ok_or(E::InvalidRequest)?;
let ver = parts.next().ok_or(E::InvalidRequest)?;
2024-05-10 19:29:51 +00:00
2024-05-10 20:27:11 +00:00
(method, path, ver)
};
2024-05-10 17:50:36 +00:00
2024-05-10 20:27:11 +00:00
let headers = Headers::parse(data.into_inner()).await;
2024-05-10 17:50:36 +00:00
2024-05-10 20:27:11 +00:00
let response = match path.as_str() {
"/" => Response::Empty,
2024-05-10 22:04:51 +00:00
"/user-agent" => Response::TextPlain(headers.get("User-Agent").to_owned()),
2024-05-10 20:27:11 +00:00
// p if let Some(echo) = p.strip_prefix("/echo/") => Response::TextPlain(echo), // a nicer way to do that, not available in stable yet
2024-05-10 22:04:51 +00:00
p if p.starts_with("/echo/") => Response::TextPlain(p.trim_start_matches("/echo/").to_owned()),
p if p.starts_with("/files/") => {
let path = args.directory.join(p.trim_start_matches("/files/"));
let mut buf = vec![];
2024-05-10 22:09:54 +00:00
if let Ok(mut f) = File::open(path).await {
let _ = f.read_to_end(&mut buf).await;
Response::OctetStream(buf)
} else {
Response::_404
}
2024-05-10 22:04:51 +00:00
},
2024-05-10 20:27:11 +00:00
_ => Response::_404,
};
2024-05-10 22:04:51 +00:00
let _ = stream.write_all(&response.build()).await;
2024-05-10 20:27:11 +00:00
let _ = stream.flush().await;
2024-05-10 17:50:36 +00:00
Ok(())
}
2024-05-10 20:27:11 +00:00
#[derive(Debug, Clone)]
2024-05-10 19:29:51 +00:00
pub struct Headers (HashMap<String, String>);
impl Headers {
2024-05-10 20:27:11 +00:00
pub async fn parse (mut reader: BufReader<&'_ mut TcpStream>) -> Self {
let mut map = HashMap::new();
let mut buf = String::new();
while let Ok(_) = reader.read_line(&mut buf).await {
if let Some((k, v)) = buf.split_once(":") {
map.insert(k.trim().to_lowercase(), v.trim().to_owned());
} else {
break;
}
buf.clear();
}
Self(map)
2024-05-10 19:29:51 +00:00
}
pub fn get <'a> (&'a self, key: &str) -> &'a str {
self.0.get(&key.to_lowercase()).map(|e| e.as_str()).unwrap_or_default()
}
}
2024-05-10 18:33:15 +00:00
#[derive(Debug, Clone)]
2024-05-10 22:04:51 +00:00
enum Response {
2024-05-10 18:33:15 +00:00
_404,
Empty,
2024-05-10 22:04:51 +00:00
TextPlain (String),
OctetStream (Vec<u8>)
2024-05-10 18:33:15 +00:00
}
#[allow(non_upper_case_globals)]
2024-05-10 22:04:51 +00:00
impl Response {
fn build (self) -> Vec<u8> {
2024-05-10 18:33:15 +00:00
2024-05-10 22:04:51 +00:00
let headers = self.headers().join("\r\n");
let code = match self {
Self::_404 => "404 Not Found",
_ => "200 OK",
2024-05-10 18:33:15 +00:00
};
2024-05-10 17:50:36 +00:00
2024-05-10 22:04:51 +00:00
let mut v: Vec<u8> = f!("HTTP/1.1 {code}\r\n{headers}\r\n\r\n").into();
match self {
Self::OctetStream(bytes) => {
v.extend_from_slice(&bytes);
},
Self::TextPlain(text) => {
v.extend_from_slice(text.as_bytes());
},
_ => ()
}
2024-05-10 18:33:15 +00:00
2024-05-10 22:04:51 +00:00
v
2024-05-10 18:33:15 +00:00
}
fn headers (&self) -> Vec<String> {
match self {
Self::TextPlain(text) => vec![f!("Content-Type: text/plain"), format!("Content-Length: {}", text.len())],
2024-05-10 22:04:51 +00:00
Self::OctetStream(bytes) => vec![f!("Content-Type: application/octet-stream"), format!("Content-Length: {}", bytes.len())],
2024-05-10 18:33:15 +00:00
_ => d!()
}
}
2024-05-10 17:06:42 +00:00
}