stage 7 attempt 1
This commit is contained in:
parent
f02334b0ad
commit
629d57de50
84
src/main.rs
84
src/main.rs
@ -1,30 +1,50 @@
|
|||||||
// #![feature(if_let_guard)]
|
// #![feature(if_let_guard)]
|
||||||
|
|
||||||
use std::collections::HashMap;
|
use std::{ collections::HashMap, path::PathBuf, sync::Arc };
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
|
|
||||||
use tokio::{
|
use tokio::{
|
||||||
io::{ AsyncBufReadExt, AsyncWriteExt, BufReader, Lines },
|
fs::File,
|
||||||
net::{ TcpListener, TcpStream }
|
io::{ AsyncBufReadExt, AsyncReadExt, AsyncWriteExt, BufReader },
|
||||||
|
net::{ TcpListener, TcpStream },
|
||||||
};
|
};
|
||||||
|
|
||||||
mod utils;
|
mod utils;
|
||||||
use utils::*;
|
use utils::*;
|
||||||
|
|
||||||
|
#[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
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
async fn main() -> Result<()> {
|
async fn main() -> Result<()> {
|
||||||
let listener = TcpListener::bind("127.0.0.1:4221").await.unwrap();
|
let listener = TcpListener::bind("127.0.0.1:4221").await.unwrap();
|
||||||
|
let args = Arc::new(parse_args());
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
|
let args = args.clone();
|
||||||
let Ok((socket, info)) = listener.accept().await else { continue; } ;
|
let Ok((socket, info)) = listener.accept().await else { continue; } ;
|
||||||
println!("Connection from {:?} accepted, processing…", info);
|
println!("Connection from {:?} accepted, processing…", info);
|
||||||
let _ = process(socket).await;
|
let _ = process(socket, args).await;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
async fn process (mut stream: TcpStream) -> 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
|
||||||
@ -40,20 +60,27 @@ async fn process (mut stream: TcpStream) -> Result<()> {
|
|||||||
(method, path, ver)
|
(method, path, ver)
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
let headers = Headers::parse(data.into_inner()).await;
|
let headers = Headers::parse(data.into_inner()).await;
|
||||||
|
|
||||||
let response = match path.as_str() {
|
let response = match path.as_str() {
|
||||||
"/" => Response::Empty,
|
"/" => Response::Empty,
|
||||||
"/user-agent" => Response::TextPlain(headers.get("User-Agent")),
|
"/user-agent" => Response::TextPlain(headers.get("User-Agent").to_owned()),
|
||||||
// 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/")),
|
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![];
|
||||||
|
let mut f = File::open(path).await?;
|
||||||
|
|
||||||
|
|
||||||
|
let _ = f.read_to_end(&mut buf).await;
|
||||||
|
|
||||||
|
Response::OctetStream(buf)
|
||||||
|
},
|
||||||
_ => Response::_404,
|
_ => Response::_404,
|
||||||
};
|
};
|
||||||
|
|
||||||
println!("{:?}", response);
|
let _ = stream.write_all(&response.build()).await;
|
||||||
|
|
||||||
let _ = stream.write_all(response.build().as_bytes()).await;
|
|
||||||
let _ = stream.flush().await;
|
let _ = stream.flush().await;
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
@ -68,7 +95,6 @@ impl Headers {
|
|||||||
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 {
|
||||||
if let Some((k, v)) = buf.split_once(":") {
|
if let Some((k, v)) = buf.split_once(":") {
|
||||||
println!("{:?}", (k, v));
|
|
||||||
map.insert(k.trim().to_lowercase(), v.trim().to_owned());
|
map.insert(k.trim().to_lowercase(), v.trim().to_owned());
|
||||||
} else {
|
} else {
|
||||||
break;
|
break;
|
||||||
@ -85,31 +111,43 @@ impl Headers {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
enum Response <'a> {
|
enum Response {
|
||||||
_404,
|
_404,
|
||||||
Empty,
|
Empty,
|
||||||
TextPlain (&'a str),
|
TextPlain (String),
|
||||||
|
OctetStream (Vec<u8>)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[allow(non_upper_case_globals)]
|
#[allow(non_upper_case_globals)]
|
||||||
impl Response <'_> {
|
impl Response {
|
||||||
fn build (self) -> String {
|
fn build (self) -> Vec<u8> {
|
||||||
|
|
||||||
let (code, body) = match self {
|
|
||||||
Self::_404 => ("404 Not Found", d!()),
|
|
||||||
Self::Empty => ("200 OK", d!()),
|
|
||||||
Self::TextPlain(text) => ("200 OK", text)
|
|
||||||
};
|
|
||||||
|
|
||||||
let headers = self.headers().join("\r\n");
|
let headers = self.headers().join("\r\n");
|
||||||
|
|
||||||
f!("HTTP/1.1 {code}\r\n{headers}\r\n\r\n{body}").into()
|
let code = match self {
|
||||||
|
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 {
|
||||||
|
Self::OctetStream(bytes) => {
|
||||||
|
v.extend_from_slice(&bytes);
|
||||||
|
},
|
||||||
|
Self::TextPlain(text) => {
|
||||||
|
v.extend_from_slice(text.as_bytes());
|
||||||
|
},
|
||||||
|
_ => ()
|
||||||
|
}
|
||||||
|
|
||||||
|
v
|
||||||
}
|
}
|
||||||
|
|
||||||
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) => vec![f!("Content-Type: text/plain"), format!("Content-Length: {}", text.len())],
|
||||||
|
Self::OctetStream(bytes) => vec![f!("Content-Type: application/octet-stream"), format!("Content-Length: {}", bytes.len())],
|
||||||
_ => d!()
|
_ => d!()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user