Compare commits
No commits in common. "ac173e1ca4537ab4b85baa396d8f147eb8aad00c" and "4a298c8b1ece254e88320965ba42cc55f73a3544" have entirely different histories.
ac173e1ca4
...
4a298c8b1e
20
Cargo.lock
generated
20
Cargo.lock
generated
@ -68,15 +68,6 @@ 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"
|
||||||
@ -89,16 +80,6 @@ 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"
|
||||||
@ -117,7 +98,6 @@ version = "0.1.0"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"bytes",
|
"bytes",
|
||||||
"flate2",
|
|
||||||
"itertools",
|
"itertools",
|
||||||
"nom",
|
"nom",
|
||||||
"pretty_assertions",
|
"pretty_assertions",
|
||||||
|
|||||||
@ -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
|
||||||
|
|
||||||
|
|||||||
209
src/main.rs
209
src/main.rs
@ -1,35 +1,32 @@
|
|||||||
// #![feature(if_let_guard)]
|
// #![feature(if_let_guard)]
|
||||||
|
|
||||||
use std::{ collections::HashMap, io::Write, path::PathBuf, sync::Arc };
|
use std::{ collections::HashMap, path::PathBuf, sync::Arc };
|
||||||
use anyhow::{bail, Result};
|
use anyhow::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: Option<PathBuf>,
|
pub directory: PathBuf,
|
||||||
}
|
}
|
||||||
|
|
||||||
type A = Arc<Args>;
|
type A = Arc<Args>;
|
||||||
|
|
||||||
|
|
||||||
fn parse_args () -> Args {
|
fn parse_args () -> Args {
|
||||||
let directory = std::env::args()
|
let directory = std::env::args().position(|e| e == "--directory").unwrap() + 1;
|
||||||
.position(|e| e == "--directory")
|
let directory = PathBuf::from(std::env::args().nth(directory).unwrap());
|
||||||
.and_then(|d| std::env::args().nth(d + 1))
|
|
||||||
.map(|d| PathBuf::from(d));
|
|
||||||
|
|
||||||
Args { directory }
|
Args {
|
||||||
|
directory
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
@ -50,55 +47,35 @@ 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.lines();
|
let mut data = buf_reader
|
||||||
|
.lines();
|
||||||
|
|
||||||
let (method, target, _version) = 'outer : {
|
let (_method, path, _ver) = {
|
||||||
'inner : {
|
let start_line = data.next_line().await?.ok_or(E::InvalidRequest)?; // should be 500;
|
||||||
let Ok(Some(start_line)) = data.next_line().await else { break 'inner };
|
let mut parts = start_line.split_whitespace().map(ToOwned::to_owned);
|
||||||
let mut parts = start_line.split_ascii_whitespace();
|
let method = parts.next().ok_or(E::InvalidRequest)?;
|
||||||
let Some(Ok(method)) = parts.next().map(Method::try_from) else { break 'inner };
|
let path = parts.next().ok_or(E::InvalidRequest)?;
|
||||||
let Some(path) = parts.next() else { break 'inner };
|
let ver = parts.next().ok_or(E::InvalidRequest)?;
|
||||||
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
|
|
||||||
|
|
||||||
let _ = stream.write_all(&Response::_400.build()).await;
|
(method, path, ver)
|
||||||
let _ = stream.flush();
|
|
||||||
bail!(E::InvalidRequest)
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
let headers = Headers::parse(data.into_inner()).await;
|
||||||
|
|
||||||
let mut data = data.into_inner();
|
let response = match path.as_str() {
|
||||||
|
"/" => Response::Empty,
|
||||||
let headers = Headers::parse(&mut data).await;
|
"/user-agent" => Response::TextPlain(headers.get("User-Agent").to_owned()),
|
||||||
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
|
||||||
(M::GET, r) if r.starts_with("/echo/") => Response::TextPlain(r.trim_start_matches("/echo/").to_owned(), encoding),
|
p if p.starts_with("/echo/") => Response::TextPlain(p.trim_start_matches("/echo/").to_owned()),
|
||||||
(M::GET, r) if r.starts_with("/files/") => 'file : {
|
p if p.starts_with("/files/") => {
|
||||||
let Some(path) = &args.directory else { break 'file Response::_500; };
|
let path = args.directory.join(p.trim_start_matches("/files/"));
|
||||||
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![];
|
||||||
let _ = f.read_to_end(&mut buf).await;
|
if let Ok(mut f) = File::open(path).await {
|
||||||
Response::OctetStream(buf)
|
let _ = f.read_to_end(&mut buf).await;
|
||||||
},
|
Response::OctetStream(buf)
|
||||||
(M::POST, r) if r.starts_with("/files") => 'file : {
|
} else {
|
||||||
let length = headers.get("Content-Length").parse().unwrap();
|
Response::_404
|
||||||
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,
|
||||||
};
|
};
|
||||||
@ -109,17 +86,11 @@ 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 (reader: &mut BufReader<&'_ mut TcpStream>) -> Self {
|
pub async fn parse (mut reader: 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 {
|
||||||
@ -139,136 +110,44 @@ impl Headers {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq, Eq)]
|
|
||||||
enum Encoding {
|
|
||||||
Gzip,
|
|
||||||
Invalid,
|
|
||||||
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)]
|
#[derive(Debug, Clone)]
|
||||||
enum Response {
|
enum Response {
|
||||||
_201,
|
|
||||||
_400,
|
|
||||||
_404,
|
_404,
|
||||||
_500,
|
|
||||||
Empty,
|
Empty,
|
||||||
TextPlain (String, Vec<Encoding>),
|
TextPlain (String),
|
||||||
OctetStream (Vec<u8>)
|
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 mut headers = self.headers();
|
let headers = self.headers().join("\r\n");
|
||||||
let code = self.code();
|
|
||||||
|
|
||||||
let mut v: Vec<u8> = vec![];
|
let code = match self {
|
||||||
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());
|
Self::_404 => "404 Not Found",
|
||||||
|
_ => "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, encodings) => {
|
Self::TextPlain(text) => {
|
||||||
if encodings.contains(&Encoding::Gzip) {
|
v.extend_from_slice(text.as_bytes());
|
||||||
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());
|
|
||||||
}
|
|
||||||
|
|
||||||
},
|
},
|
||||||
_ => 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, enc) => {
|
Self::TextPlain(text) => vec![f!("Content-Type: text/plain"), format!("Content-Length: {}", text.len())],
|
||||||
let mut v = vec![
|
Self::OctetStream(bytes) => vec![f!("Content-Type: application/octet-stream"), format!("Content-Length: {}", bytes.len())],
|
||||||
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!()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -9,6 +9,4 @@ 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,
|
|
||||||
}
|
}
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user