Compare commits

..

10 Commits

Author SHA1 Message Date
YK
ac173e1ca4 http compressions stage 3 attempt 2 2024-05-11 09:07:08 +03:00
YK
be3740f6ed http compressions stage 3 2024-05-11 09:01:57 +03:00
YK
16b90fe61c http compressions stage 2 2024-05-11 04:42:02 +03:00
YK
38ded8a8c3 http compressions stage 1 attempt 3 2024-05-11 04:31:48 +03:00
YK
35001306fb http compressions stage 1 attempt 2 2024-05-11 04:29:06 +03:00
YK
ef899b461e http compressions stage 1 attempt 1 2024-05-11 04:23:33 +03:00
YK
d916e58f20 stage 8 attempt 2 2024-05-11 04:07:10 +03:00
YK
edfd90b787 stage 8 2024-05-11 04:05:58 +03:00
YK
d5e6878703 stage 7 refactoring 2024-05-11 01:20:00 +03:00
YK
f4b8eb502d stage 7 attempt 4 (compatibility with previous tests) 2024-05-11 01:14:34 +03:00
4 changed files with 192 additions and 49 deletions

20
Cargo.lock generated
View File

@ -68,6 +68,15 @@ version = "1.0.0"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd"
[[package]]
name = "crc32fast"
version = "1.4.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "b3855a8a784b474f333699ef2bbca9db2c4a1f6d9088a90a2d25b1eb53111eaa"
dependencies = [
"cfg-if",
]
[[package]] [[package]]
name = "diff" name = "diff"
version = "0.1.13" version = "0.1.13"
@ -80,6 +89,16 @@ version = "1.9.0"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a26ae43d7bcc3b814de94796a5e736d4029efb0ee900c12e2d54c993ad1a1e07" checksum = "a26ae43d7bcc3b814de94796a5e736d4029efb0ee900c12e2d54c993ad1a1e07"
[[package]]
name = "flate2"
version = "1.0.30"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "5f54427cfd1c7829e2a139fcefea601bf088ebca651d2bf53ebc600eac295dae"
dependencies = [
"crc32fast",
"miniz_oxide",
]
[[package]] [[package]]
name = "gimli" name = "gimli"
version = "0.27.3" version = "0.27.3"
@ -98,6 +117,7 @@ version = "0.1.0"
dependencies = [ dependencies = [
"anyhow", "anyhow",
"bytes", "bytes",
"flate2",
"itertools", "itertools",
"nom", "nom",
"pretty_assertions", "pretty_assertions",

View File

@ -25,7 +25,7 @@ thiserror = "1.0.38" # error handling
tokio = { version = "1.23.0", features = ["full"] } # async networking tokio = { version = "1.23.0", features = ["full"] } # async networking
nom = "7.1.3" # parser combinators nom = "7.1.3" # parser combinators
itertools = "0.11.0" # General iterator helpers itertools = "0.11.0" # General iterator helpers
flate2 = "1.0.30"
[dev-dependencies] [dev-dependencies]
pretty_assertions = "1.3.0" # nicer looking assertions pretty_assertions = "1.3.0" # nicer looking assertions

View File

@ -1,32 +1,35 @@
// #![feature(if_let_guard)] // #![feature(if_let_guard)]
use std::{ collections::HashMap, path::PathBuf, sync::Arc }; use std::{ collections::HashMap, io::Write, path::PathBuf, sync::Arc };
use anyhow::Result; use anyhow::{bail, Result};
use itertools::Itertools;
use tokio::{ use tokio::{
fs::File, fs::File,
io::{ AsyncBufReadExt, AsyncReadExt, AsyncWriteExt, BufReader }, io::{ AsyncBufReadExt, AsyncReadExt, AsyncWriteExt, BufReader },
net::{ TcpListener, TcpStream }, net::{ TcpListener, TcpStream },
}; };
use flate2::{ write::GzEncoder, Compression };
mod utils; mod utils;
use utils::*; use utils::*;
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
struct Args { struct Args {
pub directory: PathBuf, pub directory: Option<PathBuf>,
} }
type A = Arc<Args>; type A = Arc<Args>;
fn parse_args () -> Args { fn parse_args () -> Args {
let directory = std::env::args().position(|e| e == "--directory").unwrap() + 1; let directory = std::env::args()
let directory = PathBuf::from(std::env::args().nth(directory).unwrap()); .position(|e| e == "--directory")
.and_then(|d| std::env::args().nth(d + 1))
.map(|d| PathBuf::from(d));
Args { Args { directory }
directory
}
} }
#[tokio::main] #[tokio::main]
@ -47,35 +50,55 @@ async fn main() -> Result<()> {
async fn process (mut stream: TcpStream, args: A) -> Result<()> { async fn process (mut stream: TcpStream, args: A) -> Result<()> {
let buf_reader = BufReader::new(&mut stream); let buf_reader = BufReader::new(&mut stream);
let mut data = buf_reader let mut data = buf_reader.lines();
.lines();
let (_method, path, _ver) = { let (method, target, _version) = 'outer : {
let start_line = data.next_line().await?.ok_or(E::InvalidRequest)?; // should be 500; 'inner : {
let mut parts = start_line.split_whitespace().map(ToOwned::to_owned); let Ok(Some(start_line)) = data.next_line().await else { break 'inner };
let method = parts.next().ok_or(E::InvalidRequest)?; let mut parts = start_line.split_ascii_whitespace();
let path = parts.next().ok_or(E::InvalidRequest)?; let Some(Ok(method)) = parts.next().map(Method::try_from) else { break 'inner };
let ver = parts.next().ok_or(E::InvalidRequest)?; 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
(method, path, ver) let _ = stream.write_all(&Response::_400.build()).await;
let _ = stream.flush();
bail!(E::InvalidRequest)
}; };
let headers = Headers::parse(data.into_inner()).await;
let response = match path.as_str() { let mut data = data.into_inner();
"/" => Response::Empty,
"/user-agent" => Response::TextPlain(headers.get("User-Agent").to_owned()), let headers = Headers::parse(&mut data).await;
let encoding = Encoding::parse(headers.get("Accept-Encoding"));
use Method as M;
let response = match (method, target.as_str()) {
(M::GET, "/") => Response::Empty,
(M::GET, "/user-agent") => Response::TextPlain(headers.get("User-Agent").to_owned(), encoding),
// p if let Some(echo) = p.strip_prefix("/echo/") => Response::TextPlain(echo), // a nicer way to do that, not available in stable yet // p if let Some(echo) = p.strip_prefix("/echo/") => Response::TextPlain(echo), // a nicer way to do that, not available in stable yet
p if p.starts_with("/echo/") => Response::TextPlain(p.trim_start_matches("/echo/").to_owned()), (M::GET, r) if r.starts_with("/echo/") => Response::TextPlain(r.trim_start_matches("/echo/").to_owned(), encoding),
p if p.starts_with("/files/") => { (M::GET, r) if r.starts_with("/files/") => 'file : {
let path = args.directory.join(p.trim_start_matches("/files/")); 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; };
let mut buf = vec![]; let mut buf = vec![];
if let Ok(mut f) = File::open(path).await {
let _ = f.read_to_end(&mut buf).await; let _ = f.read_to_end(&mut buf).await;
Response::OctetStream(buf) Response::OctetStream(buf)
} else { },
Response::_404 (M::POST, r) if r.starts_with("/files") => 'file : {
} let length = headers.get("Content-Length").parse().unwrap();
let body = parse_req_body(&mut data, length).await;
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
}, },
_ => Response::_404, _ => Response::_404,
}; };
@ -86,11 +109,17 @@ async fn process (mut stream: TcpStream, args: A) -> Result<()> {
Ok(()) Ok(())
} }
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
}
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
pub struct Headers (HashMap<String, String>); pub struct Headers (HashMap<String, String>);
impl Headers { impl Headers {
pub async fn parse (mut reader: BufReader<&'_ mut TcpStream>) -> Self { pub async fn parse (reader: &mut BufReader<&'_ mut TcpStream>) -> Self {
let mut map = HashMap::new(); let mut map = HashMap::new();
let mut buf = String::new(); let mut buf = String::new();
while let Ok(_) = reader.read_line(&mut buf).await { while let Ok(_) = reader.read_line(&mut buf).await {
@ -110,44 +139,136 @@ impl Headers {
} }
} }
#[derive(Debug, Clone)]
enum Response { #[derive(Debug, Clone, PartialEq, Eq)]
_404, enum Encoding {
Empty, Gzip,
TextPlain (String), Invalid,
OctetStream (Vec<u8>) None,
} }
impl Encoding {
pub fn header (&self) -> &'static str {
match self {
Self::Gzip => "gzip",
_ => d!()
}
}
pub fn parse (s: &str) -> Vec<Self> {
s.split(',').map(str::trim).map(From::from).collect()
}
}
impl From<&str> for Encoding {
fn from (s: &str) -> Self {
match s.to_ascii_lowercase().as_str() {
"" => Self::None,
"gzip" => Self::Gzip,
_ => Self::Invalid
}
}
}
#[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)
})
}
}
#[derive(Debug, Clone)]
enum Response {
_201,
_400,
_404,
_500,
Empty,
TextPlain (String, Vec<Encoding>),
OctetStream (Vec<u8>)
}
#[allow(non_upper_case_globals)] #[allow(non_upper_case_globals)]
impl Response { impl Response {
fn build (self) -> Vec<u8> { fn build (self) -> Vec<u8> {
let headers = self.headers().join("\r\n"); let mut headers = self.headers();
let code = self.code();
let code = match self { let mut v: Vec<u8> = vec![];
Self::_404 => "404 Not Found", let mut write_response_header = |headers: Vec<String>| v.extend_from_slice(format!("HTTP/1.1 {code}\r\n{}\r\n\r\n", headers.join("\r\n")).as_bytes());
_ => "200 OK",
};
let mut v: Vec<u8> = f!("HTTP/1.1 {code}\r\n{headers}\r\n\r\n").into();
match self { match self {
Self::OctetStream(bytes) => { Self::OctetStream(bytes) => {
write_response_header(headers);
v.extend_from_slice(&bytes); v.extend_from_slice(&bytes);
}, },
Self::TextPlain(text) => { Self::TextPlain(text, encodings) => {
if encodings.contains(&Encoding::Gzip) {
let mut enc = GzEncoder::new(vec![], Compression::default());
enc.write_all(text.as_bytes()).unwrap();
let b = enc.finish().unwrap();
headers.push(format!("Content-Length: {}", b.len()));
write_response_header(headers);
v.extend_from_slice(&b);
} else {
write_response_header(headers);
v.extend_from_slice(text.as_bytes()); v.extend_from_slice(text.as_bytes());
}
}, },
_ => () _ => write_response_header(headers)
} }
v v
} }
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",
}
}
fn headers (&self) -> Vec<String> { fn headers (&self) -> Vec<String> {
match self { match self {
Self::TextPlain(text) => vec![f!("Content-Type: text/plain"), format!("Content-Length: {}", text.len())], Self::TextPlain(text, enc) => {
Self::OctetStream(bytes) => vec![f!("Content-Type: application/octet-stream"), format!("Content-Length: {}", bytes.len())], let mut v = vec![
f!("Content-Type: text/plain"),
];
if !enc.contains(&Encoding::Gzip) { v.push(format!("Content-Length: {}", text.len())) }
let enc = enc.into_iter().map(Encoding::header).filter(|e| !e.is_empty()).join(", ");
if !enc.is_empty() { v.push(f!("Content-Encoding: {enc}")) }
v
},
Self::OctetStream(bytes) => vec![
f!("Content-Type: application/octet-stream"),
format!("Content-Length: {}", bytes.len())
],
_ => d!() _ => d!()
} }
} }

View File

@ -9,4 +9,6 @@ macro_rules! f { ($s: expr) => { format!($s) }; }
pub enum E { pub enum E {
#[error("Invalid request data found during parsing")] #[error("Invalid request data found during parsing")]
InvalidRequest, InvalidRequest,
#[error("Cannot parse method")]
UnknownMethod,
} }