http-server-cc/src/main.rs

107 lines
3.1 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 19:29:51 +00:00
use std::{ collections::HashMap, io::{ BufRead, BufReader, Write }, net::TcpListener };
2024-05-10 17:50:36 +00:00
use anyhow::Result;
2024-05-10 17:06:42 +00:00
2024-05-10 19:29:51 +00:00
mod utils;
use utils::*;
2024-05-10 17:50:36 +00:00
fn main() -> Result<()> {
2024-05-10 17:23:14 +00:00
let listener = TcpListener::bind("127.0.0.1:4221").unwrap();
for stream in listener.incoming() {
match stream {
2024-05-10 17:26:28 +00:00
Ok(mut stream) => {
2024-05-10 18:07:14 +00:00
let buf_reader = BufReader::new(&mut stream);
2024-05-10 17:50:36 +00:00
2024-05-10 18:07:14 +00:00
let mut data = buf_reader
.lines()
.map(|result| result.unwrap())
.take_while(|line| !line.is_empty());
2024-05-10 17:50:36 +00:00
2024-05-10 18:07:14 +00:00
let (_method, path, _ver) = {
2024-05-10 17:50:36 +00:00
let start_line = data.next().ok_or(E::InvalidRequest)?; // should be 500;
2024-05-10 18:07:14 +00:00
let mut parts = start_line.split_whitespace().map(ToOwned::to_owned);
2024-05-10 17:50:36 +00:00
let method = parts.next().ok_or(E::InvalidRequest)?;
2024-05-10 18:37:00 +00:00
let path = parts.next().ok_or(E::InvalidRequest)?;
let ver = parts.next().ok_or(E::InvalidRequest)?;
2024-05-10 17:50:36 +00:00
(method, path, ver)
};
2024-05-10 19:29:51 +00:00
let headers = Headers::parse(data);
let response = match path.trim_start_matches("/") {
"" => Response::Empty,
"user-agent" => Response::TextPlain(headers.get("User-Agent")),
2024-05-10 18:37:00 +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 19:29:51 +00:00
p if p.starts_with("echo/") => Response::TextPlain(p.trim_start_matches("echo/")),
2024-05-10 18:33:15 +00:00
_ => Response::_404,
2024-05-10 17:50:36 +00:00
};
2024-05-10 17:23:14 +00:00
println!("accepted new connection");
2024-05-10 17:50:36 +00:00
2024-05-10 18:33:15 +00:00
let _ = stream.write(response.build().as_bytes());
2024-05-10 17:50:36 +00:00
let _ = stream.flush();
2024-05-10 17:23:14 +00:00
}
Err(e) => {
println!("error: {}", e);
}
}
}
2024-05-10 17:50:36 +00:00
Ok(())
}
2024-05-10 19:29:51 +00:00
pub struct Headers (HashMap<String, String>);
impl Headers {
pub fn parse (lines: impl Iterator<Item = String>) -> Self {
Self(HashMap::from_iter(lines.filter_map(|line|
line.split_once(":")
.map(|(a, b)| (
a.trim().to_lowercase(),
b.trim().to_owned()
))
)))
}
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)]
enum Response <'a> {
_404,
Empty,
TextPlain (&'a str),
}
#[allow(non_upper_case_globals)]
impl Response <'_> {
fn build (self) -> String {
let (code, body) = match self {
Self::_404 => ("404 Not Found", d!()),
Self::Empty => ("200 OK", d!()),
Self::TextPlain(text) => ("200 OK", text)
};
2024-05-10 17:50:36 +00:00
2024-05-10 18:33:15 +00:00
let headers = self.headers().join("\r\n");
f!("HTTP/1.1 {code}\r\n{headers}\r\n\r\n{body}").into()
}
fn headers (&self) -> Vec<String> {
match self {
Self::TextPlain(text) => vec![f!("Content-Type: text/plain"), format!("Content-Length: {}", text.len())],
_ => d!()
}
}
2024-05-10 17:06:42 +00:00
}