forked from erik/strava-heatmap-proxy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrouter.js
121 lines (100 loc) · 2.66 KB
/
router.js
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
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
/**
* Helper functions that when passed a request will return a
* boolean indicating if the request uses that HTTP method,
* header, host or referrer.
*/
const Method = (method) => (req) =>
req.method.toLowerCase() === method.toLowerCase();
const Connect = Method("connect");
const Delete = Method("delete");
const Get = Method("get");
const Head = Method("head");
const Options = Method("options");
const Patch = Method("patch");
const Post = Method("post");
const Put = Method("put");
const Trace = Method("trace");
const Header = (header, val) => (req) => req.headers.get(header) === val;
const Host = (host) => Header("host", host.toLowerCase());
const Referrer = (host) => Header("referrer", host.toLowerCase());
const Path = (regExp) => (req) => {
const url = new URL(req.url);
const path = url.pathname;
const match = path.match(regExp) || [];
return match[0] === path;
};
/**
* The Router handles determines which handler is matched given the
* conditions present for each request.
*/
class Router {
constructor() {
this.routes = [];
}
handle(conditions, handler) {
this.routes.push({
conditions,
handler,
});
return this;
}
connect(url, handler) {
return this.handle([Connect, Path(url)], handler);
}
delete(url, handler) {
return this.handle([Delete, Path(url)], handler);
}
get(url, handler) {
return this.handle([Get, Path(url)], handler);
}
head(url, handler) {
return this.handle([Head, Path(url)], handler);
}
options(url, handler) {
return this.handle([Options, Path(url)], handler);
}
patch(url, handler) {
return this.handle([Patch, Path(url)], handler);
}
post(url, handler) {
return this.handle([Post, Path(url)], handler);
}
put(url, handler) {
return this.handle([Put, Path(url)], handler);
}
trace(url, handler) {
return this.handle([Trace, Path(url)], handler);
}
all(handler) {
return this.handle([], handler);
}
route(req) {
const route = this.resolve(req);
if (route) {
return route.handler(req);
}
return new Response("resource not found", {
status: 404,
statusText: "not found",
headers: {
"content-type": "text/plain",
},
});
}
/**
* resolve returns the matching route for a request that returns
* true for all conditions (if any).
*/
resolve(req) {
return this.routes.find((r) => {
if (!r.conditions || (Array.isArray(r) && !r.conditions.length)) {
return true;
}
if (typeof r.conditions === "function") {
return r.conditions(req);
}
return r.conditions.every((c) => c(req));
});
}
}
module.exports = Router;