stage 6 attempt 1
This commit is contained in:
parent
93d2853907
commit
f02334b0ad
78
src/main.rs
78
src/main.rs
@ -1,26 +1,37 @@
|
|||||||
// #![feature(if_let_guard)]
|
// #![feature(if_let_guard)]
|
||||||
|
|
||||||
use std::{ collections::HashMap, io::{ BufRead, BufReader, Write }, net::TcpListener };
|
use std::collections::HashMap;
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
|
|
||||||
|
use tokio::{
|
||||||
|
io::{ AsyncBufReadExt, AsyncWriteExt, BufReader, Lines },
|
||||||
|
net::{ TcpListener, TcpStream }
|
||||||
|
};
|
||||||
|
|
||||||
mod utils;
|
mod utils;
|
||||||
use utils::*;
|
use utils::*;
|
||||||
|
|
||||||
fn main() -> Result<()> {
|
#[tokio::main]
|
||||||
let listener = TcpListener::bind("127.0.0.1:4221").unwrap();
|
async fn main() -> Result<()> {
|
||||||
|
let listener = TcpListener::bind("127.0.0.1:4221").await.unwrap();
|
||||||
|
|
||||||
for stream in listener.incoming() {
|
loop {
|
||||||
match stream {
|
let Ok((socket, info)) = listener.accept().await else { continue; } ;
|
||||||
Ok(mut stream) => {
|
println!("Connection from {:?} accepted, processing…", info);
|
||||||
|
let _ = process(socket).await;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
async fn process (mut stream: TcpStream) -> 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();
|
||||||
.map(|result| result.unwrap())
|
|
||||||
.take_while(|line| !line.is_empty());
|
|
||||||
|
|
||||||
let (_method, path, _ver) = {
|
let (_method, path, _ver) = {
|
||||||
let start_line = data.next().ok_or(E::InvalidRequest)?; // should be 500;
|
let start_line = data.next_line().await?.ok_or(E::InvalidRequest)?; // should be 500;
|
||||||
let mut parts = start_line.split_whitespace().map(ToOwned::to_owned);
|
let mut parts = start_line.split_whitespace().map(ToOwned::to_owned);
|
||||||
let method = parts.next().ok_or(E::InvalidRequest)?;
|
let method = parts.next().ok_or(E::InvalidRequest)?;
|
||||||
let path = parts.next().ok_or(E::InvalidRequest)?;
|
let path = parts.next().ok_or(E::InvalidRequest)?;
|
||||||
@ -29,44 +40,43 @@ fn main() -> Result<()> {
|
|||||||
(method, path, ver)
|
(method, path, ver)
|
||||||
};
|
};
|
||||||
|
|
||||||
let headers = Headers::parse(data);
|
|
||||||
|
|
||||||
|
let headers = Headers::parse(data.into_inner()).await;
|
||||||
|
|
||||||
let response = match path.trim_start_matches("/") {
|
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")),
|
||||||
// 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/")),
|
||||||
_ => Response::_404,
|
_ => Response::_404,
|
||||||
};
|
};
|
||||||
|
|
||||||
println!("accepted new connection");
|
println!("{:?}", response);
|
||||||
|
|
||||||
let _ = stream.write(response.build().as_bytes());
|
let _ = stream.write_all(response.build().as_bytes()).await;
|
||||||
let _ = stream.flush();
|
let _ = stream.flush().await;
|
||||||
|
|
||||||
}
|
|
||||||
Err(e) => {
|
|
||||||
println!("error: {}", e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
pub struct Headers (HashMap<String, String>);
|
pub struct Headers (HashMap<String, String>);
|
||||||
|
|
||||||
impl Headers {
|
impl Headers {
|
||||||
pub fn parse (lines: impl Iterator<Item = String>) -> Self {
|
pub async fn parse (mut reader: BufReader<&'_ mut TcpStream>) -> Self {
|
||||||
Self(HashMap::from_iter(lines.filter_map(|line|
|
let mut map = HashMap::new();
|
||||||
line.split_once(":")
|
let mut buf = String::new();
|
||||||
.map(|(a, b)| (
|
while let Ok(_) = reader.read_line(&mut buf).await {
|
||||||
a.trim().to_lowercase(),
|
if let Some((k, v)) = buf.split_once(":") {
|
||||||
b.trim().to_owned()
|
println!("{:?}", (k, v));
|
||||||
))
|
map.insert(k.trim().to_lowercase(), v.trim().to_owned());
|
||||||
)))
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
buf.clear();
|
||||||
|
}
|
||||||
|
Self(map)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get <'a> (&'a self, key: &str) -> &'a str {
|
pub fn get <'a> (&'a self, key: &str) -> &'a str {
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user