first commit
This commit is contained in:
1
.gitignore
vendored
Normal file
1
.gitignore
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
/target
|
||||||
7
Cargo.lock
generated
Normal file
7
Cargo.lock
generated
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
# This file is automatically @generated by Cargo.
|
||||||
|
# It is not intended for manual editing.
|
||||||
|
version = 4
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "mini_web_server"
|
||||||
|
version = "0.1.0"
|
||||||
6
Cargo.toml
Normal file
6
Cargo.toml
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
[package]
|
||||||
|
name = "mini_web_server"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2024"
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
97
src/lib.rs
Normal file
97
src/lib.rs
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
use std::sync::mpsc;
|
||||||
|
use std::sync::Arc;
|
||||||
|
use std::sync::Mutex;
|
||||||
|
use std::thread;
|
||||||
|
|
||||||
|
|
||||||
|
enum Message{
|
||||||
|
NewJob(Job),
|
||||||
|
Terminate,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct ThreadPool{
|
||||||
|
workers: Vec<Worker>,
|
||||||
|
sender: mpsc::Sender<Message>,
|
||||||
|
}
|
||||||
|
|
||||||
|
type Job = Box<dyn FnOnce() + Send + 'static>;
|
||||||
|
|
||||||
|
impl ThreadPool {
|
||||||
|
/// Create a new ThreadPool.
|
||||||
|
///
|
||||||
|
/// this size is the number of threads in the pool.
|
||||||
|
///
|
||||||
|
/// # Panics
|
||||||
|
///
|
||||||
|
/// the `new` function will panic if the size is not greater than 0.
|
||||||
|
pub fn new(size: usize) -> ThreadPool {
|
||||||
|
assert!(size > 0);
|
||||||
|
|
||||||
|
let (sender, receiver) = mpsc::channel();
|
||||||
|
let receiver = Arc::new(Mutex::new(receiver));
|
||||||
|
|
||||||
|
let mut workers = Vec::with_capacity(size);
|
||||||
|
|
||||||
|
for id in 0..size {
|
||||||
|
workers.push(Worker::new(id, Arc::clone(&receiver)));
|
||||||
|
}
|
||||||
|
|
||||||
|
ThreadPool { workers , sender}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn execute<F>(&self, f: F)
|
||||||
|
where F: FnOnce() + Send + 'static,
|
||||||
|
{
|
||||||
|
let job = Box::new(f);
|
||||||
|
self.sender.send(Message::NewJob(job)).unwrap();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Drop for ThreadPool {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
|
||||||
|
for _ in &mut self.workers {
|
||||||
|
self.sender.send(Message::Terminate).unwrap();
|
||||||
|
}
|
||||||
|
println!("shutting down all workers");
|
||||||
|
|
||||||
|
for workder in &mut self.workers {
|
||||||
|
println!("shutting down worker {}", workder.id);
|
||||||
|
|
||||||
|
if let Some(thread) = workder.thread.take(){
|
||||||
|
thread.join().unwrap();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
struct Worker{
|
||||||
|
id: usize,
|
||||||
|
thread: Option<thread::JoinHandle<()>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Worker {
|
||||||
|
fn new(id: usize, receiver: Arc<Mutex<mpsc::Receiver<Message>>>) -> Worker {
|
||||||
|
let thread = thread::spawn(move || loop {
|
||||||
|
let message = receiver.lock().unwrap().recv().unwrap();
|
||||||
|
match message {
|
||||||
|
Message::NewJob(job) => {
|
||||||
|
println!("Worker {} got a job. executing.", id);
|
||||||
|
job();
|
||||||
|
}
|
||||||
|
Message::Terminate => {
|
||||||
|
println!("Worker {} was told to terminate.", id);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
Worker {
|
||||||
|
id,
|
||||||
|
thread: Some(thread)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
152
src/main.rs
Normal file
152
src/main.rs
Normal file
@ -0,0 +1,152 @@
|
|||||||
|
use std::fs;
|
||||||
|
use std::path::Path;
|
||||||
|
use std::io::{BufRead, BufReader, Write};
|
||||||
|
use std::net::{TcpListener, TcpStream};
|
||||||
|
use mini_web_server::ThreadPool;
|
||||||
|
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
let listener = TcpListener::bind("127.0.0.1:7878").unwrap();
|
||||||
|
|
||||||
|
let pool = ThreadPool::new(10);
|
||||||
|
|
||||||
|
for stream in listener.incoming() {
|
||||||
|
let stream = stream.unwrap();
|
||||||
|
pool.execute(||{
|
||||||
|
handle_connection(stream);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
println!("shutdown");
|
||||||
|
}
|
||||||
|
|
||||||
|
fn handle_connection(mut stream: TcpStream) {
|
||||||
|
// 1. 用 BufReader 包装 stream,提供按行读取的能力
|
||||||
|
let mut reader = BufReader::new(&mut stream);
|
||||||
|
|
||||||
|
// 2. 读取请求行(第一行)
|
||||||
|
let mut request_line = String::new();
|
||||||
|
|
||||||
|
if reader.read_line(&mut request_line).is_err() {
|
||||||
|
// 读取出错,返回 400
|
||||||
|
stream.write(b"HTTP/1.1 400 Bad Request\r\n\r\n").unwrap();
|
||||||
|
stream.flush().unwrap();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let request_line = request_line.trim(); // 去掉末尾的 \r\n
|
||||||
|
|
||||||
|
// 3. 解析请求行:方法 路径 版本
|
||||||
|
let parts: Vec<&str> = request_line.split_whitespace().collect();
|
||||||
|
if parts.len() < 3 {
|
||||||
|
stream.write(b"HTTP/1.1 400 Bad Request\r\n\r\n").unwrap();
|
||||||
|
stream.flush().unwrap();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let method = parts[0];
|
||||||
|
let path = parts[1];
|
||||||
|
|
||||||
|
// 4. 只允许 GET
|
||||||
|
if method != "GET" {
|
||||||
|
stream.write(b"HTTP/1.1 405 Method Not Allowed\r\n\r\n").unwrap();
|
||||||
|
stream.flush().unwrap();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 5. 继续读取请求头,直到空行
|
||||||
|
loop {
|
||||||
|
let mut header_line = String::new();
|
||||||
|
if reader.read_line(&mut header_line).is_err() {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
// 空行表示头部结束
|
||||||
|
if header_line == "\r\n" || header_line == "\n" {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
// 这里可以解析头部(目前我们不需要)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 6. 安全地构建文件路径
|
||||||
|
let base_dir = "./public";
|
||||||
|
let requested_path = path.trim_start_matches('/');
|
||||||
|
if requested_path.contains("..") {
|
||||||
|
stream.write(b"HTTP/1.1 403 Forbidden\r\n\r\n").unwrap();
|
||||||
|
stream.flush().unwrap();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// v1
|
||||||
|
// let file_path = if requested_path.is_empty() {
|
||||||
|
// format!("{}/index.html", base_dir)
|
||||||
|
// } else {
|
||||||
|
// format!("{}/{}", base_dir, requested_path)
|
||||||
|
// };
|
||||||
|
|
||||||
|
// v2
|
||||||
|
// let file_path = format!("{}/{}", base_dir, requested_path);
|
||||||
|
|
||||||
|
// v3
|
||||||
|
// let file_path = if requested_path.is_empty() || path.ends_with('/') {
|
||||||
|
// format!("{}/index.html", requested_path.trim_end_matches('/'))
|
||||||
|
// } else {
|
||||||
|
// requested_path
|
||||||
|
// };
|
||||||
|
|
||||||
|
// v4
|
||||||
|
let base_path = Path::new(base_dir);
|
||||||
|
let requested_path = path.trim_start_matches('/'); // 去掉开头的 '/'
|
||||||
|
|
||||||
|
// 禁止路径遍历
|
||||||
|
if requested_path.contains("..") {
|
||||||
|
stream.write(b"HTTP/1.1 403 Forbidden\r\n\r\n").unwrap();
|
||||||
|
stream.flush().unwrap();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut full_path = base_path.join(requested_path);
|
||||||
|
|
||||||
|
if full_path.is_dir() {
|
||||||
|
full_path = full_path.join("index.html");
|
||||||
|
}
|
||||||
|
|
||||||
|
// temp debug print
|
||||||
|
println!("file: {}", full_path.display());
|
||||||
|
|
||||||
|
let path = Path::new(&full_path);
|
||||||
|
|
||||||
|
// 7. 尝试读取文件并响应
|
||||||
|
if path.exists() && path.is_file() {
|
||||||
|
match fs::read(path) {
|
||||||
|
Ok(contents) => {
|
||||||
|
let content_type = match path.extension().and_then(|ext| ext.to_str()) {
|
||||||
|
Some("html") => "text/html; charset=utf-8",
|
||||||
|
Some("css") => "text/css; charset=utf-8",
|
||||||
|
Some("js") => "application/javascript",
|
||||||
|
Some("png") => "image/png",
|
||||||
|
Some("jpg" | "jpeg") => "image/jpeg",
|
||||||
|
Some("gif") => "image/gif",
|
||||||
|
Some("svg") => "image/svg+xml",
|
||||||
|
Some("ico") => "image/x-icon",
|
||||||
|
Some("json") => "application/json",
|
||||||
|
Some("txt") => "text/plain; charset=utf-8",
|
||||||
|
_ => "application/octet-stream",
|
||||||
|
};
|
||||||
|
|
||||||
|
let status_line = "HTTP/1.1 200 OK\r\n";
|
||||||
|
let length_header = format!("Content-Length: {}\r\n", contents.len());
|
||||||
|
let type_header = format!("Content-Type: {}\r\n", content_type);
|
||||||
|
let response_head = format!("{}{}{}\r\n", status_line, type_header, length_header);
|
||||||
|
|
||||||
|
stream.write(response_head.as_bytes()).unwrap();
|
||||||
|
stream.write(&contents).unwrap();
|
||||||
|
stream.flush().unwrap();
|
||||||
|
}
|
||||||
|
Err(_) => {
|
||||||
|
stream.write(b"http/1.1 500 internal server error\r\n\r\n").unwrap();
|
||||||
|
stream.flush().unwrap();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
stream.write(b"HTTP/1.1 404 Not Found\r\n\r\n404 Not Found").unwrap();
|
||||||
|
stream.flush().unwrap();
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user