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-11 01:05:58 +00:00
|
|
|
use anyhow::{bail, Result};
|
2024-05-10 17:06:42 +00:00
|
|
|
|
2024-05-11 01:42:02 +00:00
|
|
|
use itertools::Itertools;
|
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 {
|
2024-05-10 22:14:34 +00:00
|
|
|
pub directory: Option<PathBuf>,
|
2024-05-10 22:04:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type A = Arc<Args>;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
fn parse_args () -> Args {
|
2024-05-10 22:14:34 +00:00
|
|
|
let directory = std::env::args()
|
|
|
|
|
.position(|e| e == "--directory")
|
2024-05-11 01:05:58 +00:00
|
|
|
.and_then(|d| std::env::args().nth(d + 1))
|
2024-05-10 22:14:34 +00:00
|
|
|
.map(|d| PathBuf::from(d));
|
2024-05-10 22:04:51 +00:00
|
|
|
|
2024-05-11 01:05:58 +00:00
|
|
|
Args { directory }
|
2024-05-10 22:04:51 +00:00
|
|
|
}
|
|
|
|
|
|
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-11 01:05:58 +00:00
|
|
|
let mut data = buf_reader.lines();
|
2024-05-10 19:29:51 +00:00
|
|
|
|
2024-05-11 01:05:58 +00:00
|
|
|
let (method, target, _version) = 'outer : {
|
|
|
|
|
'inner : {
|
|
|
|
|
let Ok(Some(start_line)) = data.next_line().await else { break 'inner };
|
|
|
|
|
let mut parts = start_line.split_ascii_whitespace();
|
|
|
|
|
let Some(Ok(method)) = parts.next().map(Method::try_from) else { break 'inner };
|
|
|
|
|
let Some(path) = parts.next() else { break 'inner };
|
|
|
|
|
let Some(version) = parts.next() else { break 'inner };
|
|
|
|
|
break 'outer (method, path.to_owned(), version.to_owned());
|
|
|
|
|
}
|
|
|
|
|
// this is either the best or the worst piece of code i've written
|
2024-05-10 19:29:51 +00:00
|
|
|
|
2024-05-11 01:05:58 +00:00
|
|
|
let _ = stream.write_all(&Response::_400.build()).await;
|
|
|
|
|
let _ = stream.flush();
|
|
|
|
|
bail!(E::InvalidRequest)
|
2024-05-10 20:27:11 +00:00
|
|
|
};
|
2024-05-10 17:50:36 +00:00
|
|
|
|
|
|
|
|
|
2024-05-11 01:05:58 +00:00
|
|
|
let mut data = data.into_inner();
|
|
|
|
|
|
|
|
|
|
let headers = Headers::parse(&mut data).await;
|
2024-05-11 01:42:02 +00:00
|
|
|
let encoding = Encoding::parse(headers.get("Accept-Encoding"));
|
2024-05-11 01:23:33 +00:00
|
|
|
|
2024-05-11 01:29:06 +00:00
|
|
|
println!("{:?}", headers);
|
|
|
|
|
println!("{:?}", encoding);
|
2024-05-11 01:23:33 +00:00
|
|
|
|
2024-05-11 01:05:58 +00:00
|
|
|
use Method as M;
|
|
|
|
|
let response = match (method, target.as_str()) {
|
|
|
|
|
(M::GET, "/") => Response::Empty,
|
2024-05-11 01:23:33 +00:00
|
|
|
(M::GET, "/user-agent") => Response::TextPlain(headers.get("User-Agent").to_owned(), encoding),
|
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-11 01:23:33 +00:00
|
|
|
(M::GET, r) if r.starts_with("/echo/") => Response::TextPlain(r.trim_start_matches("/echo/").to_owned(), encoding),
|
2024-05-11 01:05:58 +00:00
|
|
|
(M::GET, r) if r.starts_with("/files/") => 'file : {
|
|
|
|
|
let Some(path) = &args.directory else { break 'file Response::_500; };
|
|
|
|
|
let path = path.join(r.trim_start_matches("/files/"));
|
|
|
|
|
let Ok(mut f) = File::open(path).await else { break 'file Response::_404; };
|
2024-05-10 22:20:00 +00:00
|
|
|
|
|
|
|
|
let mut buf = vec![];
|
|
|
|
|
let _ = f.read_to_end(&mut buf).await;
|
|
|
|
|
Response::OctetStream(buf)
|
2024-05-10 22:04:51 +00:00
|
|
|
},
|
2024-05-11 01:05:58 +00:00
|
|
|
(M::POST, r) if r.starts_with("/files") => 'file : {
|
2024-05-11 01:07:10 +00:00
|
|
|
let length = headers.get("Content-Length").parse().unwrap();
|
|
|
|
|
let body = parse_req_body(&mut data, length).await;
|
|
|
|
|
|
2024-05-11 01:05:58 +00:00
|
|
|
let Some(path) = &args.directory else { break 'file Response::_500; };
|
|
|
|
|
let path = path.join(r.trim_start_matches("/files/"));
|
|
|
|
|
let Ok(mut f) = File::create(path).await else { break 'file Response::_500; };
|
|
|
|
|
let Ok(_) = f.write_all(&body).await else { break 'file Response::_500 };
|
|
|
|
|
|
|
|
|
|
Response::_201
|
|
|
|
|
},
|
2024-05-10 20:27:11 +00:00
|
|
|
_ => Response::_404,
|
|
|
|
|
};
|
|
|
|
|
|
2024-05-11 01:29:06 +00:00
|
|
|
println!("{:?}", String::from_utf8_lossy(&response.clone().build()));
|
|
|
|
|
|
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-11 01:05:58 +00:00
|
|
|
pub async fn parse_req_body (reader: &mut BufReader<&mut TcpStream>, length: usize) -> Vec<u8> {
|
|
|
|
|
let mut v = vec![0; length];
|
|
|
|
|
let _ = reader.read_exact(&mut v).await;
|
|
|
|
|
v
|
|
|
|
|
}
|
|
|
|
|
|
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-11 01:05:58 +00:00
|
|
|
pub async fn parse (reader: &mut BufReader<&'_ mut TcpStream>) -> Self {
|
2024-05-10 20:27:11 +00:00
|
|
|
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-11 01:23:33 +00:00
|
|
|
|
2024-05-10 18:33:15 +00:00
|
|
|
#[derive(Debug, Clone)]
|
2024-05-11 01:23:33 +00:00
|
|
|
enum Encoding {
|
|
|
|
|
Gzip,
|
|
|
|
|
Invalid,
|
|
|
|
|
None,
|
2024-05-10 18:33:15 +00:00
|
|
|
}
|
|
|
|
|
|
2024-05-11 01:23:33 +00:00
|
|
|
|
|
|
|
|
impl Encoding {
|
|
|
|
|
pub fn header (&self) -> &'static str {
|
|
|
|
|
match self {
|
2024-05-11 01:42:02 +00:00
|
|
|
Self::Gzip => "gzip",
|
2024-05-11 01:23:33 +00:00
|
|
|
_ => d!()
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-05-11 01:42:02 +00:00
|
|
|
|
|
|
|
|
pub fn parse (s: &str) -> Vec<Self> {
|
|
|
|
|
s.split(',').map(str::trim).map(From::from).collect()
|
|
|
|
|
}
|
2024-05-11 01:23:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl From<&str> for Encoding {
|
|
|
|
|
fn from (s: &str) -> Self {
|
|
|
|
|
match s.to_ascii_lowercase().as_str() {
|
|
|
|
|
"" => Self::None,
|
|
|
|
|
"gzip" => Self::Gzip,
|
|
|
|
|
_ => Self::Invalid
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2024-05-11 01:05:58 +00:00
|
|
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
|
|
|
|
enum Method {
|
|
|
|
|
GET,
|
|
|
|
|
POST,
|
|
|
|
|
PUT,
|
|
|
|
|
DELETE,
|
|
|
|
|
UPDATE,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl TryFrom<&str> for Method {
|
|
|
|
|
type Error = anyhow::Error;
|
|
|
|
|
fn try_from (s: &str) -> Result<Self> {
|
|
|
|
|
let mut s = s.to_string();
|
|
|
|
|
s.make_ascii_lowercase();
|
|
|
|
|
Ok(match s.as_str() {
|
|
|
|
|
"get" => Self::GET,
|
|
|
|
|
"post" => Self::POST,
|
|
|
|
|
"put" => Self::PUT,
|
|
|
|
|
"delete" => Self::DELETE,
|
|
|
|
|
"update" => Self::UPDATE,
|
|
|
|
|
_ => bail!(E::UnknownMethod)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-11 01:23:33 +00:00
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
|
enum Response {
|
|
|
|
|
_201,
|
|
|
|
|
_400,
|
|
|
|
|
_404,
|
|
|
|
|
_500,
|
|
|
|
|
Empty,
|
2024-05-11 01:42:02 +00:00
|
|
|
TextPlain (String, Vec<Encoding>),
|
2024-05-11 01:23:33 +00:00
|
|
|
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");
|
2024-05-11 01:05:58 +00:00
|
|
|
let code = self.code();
|
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();
|
2024-05-11 01:05:58 +00:00
|
|
|
|
2024-05-10 22:04:51 +00:00
|
|
|
match self {
|
|
|
|
|
Self::OctetStream(bytes) => {
|
|
|
|
|
v.extend_from_slice(&bytes);
|
|
|
|
|
},
|
2024-05-11 01:23:33 +00:00
|
|
|
Self::TextPlain(text, _) => {
|
2024-05-10 22:04:51 +00:00
|
|
|
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
|
|
|
}
|
|
|
|
|
|
2024-05-11 01:05:58 +00:00
|
|
|
fn code (&self) -> &'static str {
|
|
|
|
|
match self {
|
|
|
|
|
Self::_201 => "201 Created",
|
|
|
|
|
Self::_400 => "400 Bad Request",
|
|
|
|
|
Self::_404 => "404 Not Found",
|
|
|
|
|
Self::_500 => "500 Internal Server Error",
|
|
|
|
|
_ => "200 OK",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-10 18:33:15 +00:00
|
|
|
fn headers (&self) -> Vec<String> {
|
|
|
|
|
match self {
|
2024-05-11 01:31:48 +00:00
|
|
|
Self::TextPlain(text, enc) => {
|
|
|
|
|
let mut v = vec![
|
|
|
|
|
f!("Content-Type: text/plain"),
|
|
|
|
|
format!("Content-Length: {}", text.len()),
|
|
|
|
|
];
|
2024-05-11 01:42:02 +00:00
|
|
|
let enc = enc.into_iter().map(Encoding::header).filter(|e| !e.is_empty()).join(", ");
|
|
|
|
|
if !enc.is_empty() { v.push(f!("Content-Encoding: {enc}")) }
|
2024-05-11 01:31:48 +00:00
|
|
|
v
|
|
|
|
|
},
|
2024-05-11 01:05:58 +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
|
|
|
}
|