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