aboutsummaryrefslogtreecommitdiff
path: root/src/router.rs
blob: d0b4c9d1579947fc0090f13d549d1e81224bf35f (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
use crate::{
    extractor::build_regex_from_path,
    http::{get, post, put, Endpoint, Method, StatusCode},
    request::Request,
    response::Response,
};
use regex::Regex;
use std::collections::HashMap;

type Handler = fn(&Request, Option<&HashMap<String, String>>) -> Response;
type Routes = HashMap<Endpoint, Handler>;

pub struct Router {
    routes: Routes,
}

impl Router {
    // Create a new Router
    pub fn new() -> Self {
        Router {
            routes: HashMap::new(),
        }
    }

    pub fn routes(&self) -> &Routes {
        &self.routes
    }

    // Add a route to the router
    pub fn route(&mut self, endpoint: Endpoint, handler: Handler) -> &mut Self {
        use Method as M;
        match (endpoint.method, endpoint.route) {
            (M::GET, route) => {
                let re = build_regex_from_path(&route);
                let epoint = get(re.as_str());
                self.routes.insert(epoint, handler);
            }
            (M::POST, route) => {
                let re = build_regex_from_path(&route);
                let epoint = post(re.as_str());
                self.routes.insert(epoint, handler);
            }
            (M::PUT, route) => {
                let re = build_regex_from_path(&route);
                let meth = put(re.as_str());
                self.routes.insert(meth, handler);
            }
        }
        self
    }

    // Handle incoming requests
    pub fn handle(&self, request: &Request, ctx: Option<&HashMap<String, String>>) -> Response {
        use Method as M;
        for (endpoint, handler) in self.routes() {
            let repoint = &request.endpoint;
            match (
                &repoint.method,
                &repoint.route,
                &endpoint.method,
                &endpoint.route,
            ) {
                (M::GET, request_route, M::GET, endpoint_regex) => {
                    let re = Regex::new(&endpoint_regex).unwrap();
                    // dbg!(&re, request_method);
                    if re.is_match(&request_route) {
                        return handler(request, ctx);
                    }
                }
                (M::POST, request_route, M::POST, endpoint_regex) => {
                    let re = Regex::new(&endpoint_regex).unwrap();
                    // dbg!(&re, request_method);
                    if re.is_match(&request_route) {
                        return handler(request, ctx);
                    }
                }
                (M::PUT, request_route, M::PUT, endpoint_regex) => {
                    let re = Regex::new(&endpoint_regex).unwrap();
                    // dbg!(&re, request_method);
                    if re.is_match(&request_route) {
                        return handler(request, ctx);
                    }
                }
                _ => {}
            }
        }
        Response::new("1.1".to_string(), StatusCode::NotFound, None, None)
    }
}