-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmiddleware.ts
108 lines (92 loc) · 3.98 KB
/
middleware.ts
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
import CONFIG from '@/config';
import { error } from '@/lib/utils';
import { NextRequest, NextResponse } from 'next/server';
import { injectSpeedInsights } from '@vercel/speed-insights';
import { inject as injectAnalytics } from "@vercel/analytics"
// API config validation
const API: { [base: string]: string[] } = {};
Object.entries(CONFIG.API).forEach(([base, endpoints]) => {
if (!/^(?:\/[a-z_\d\-]+)+$/i.test(base)) {
console.error(`Invalid API base: ${base}`);
return;
}
if (!Array.isArray(endpoints)) {
console.error(`Invalid API endpoints: ${endpoints}`);
return;
}
if (!API[base]) API[base] = [];
endpoints.forEach(endpoint => {
if (!/^(?:\/|(?:\/(?:[a-z_\d\-]+|:[a-z_][a-z_\d]*[^\/]*))+)$/i.test(endpoint)) {
console.error(`Invalid API endpoint: ${endpoint}`);
return;
}
API[base].push(endpoint);
});
});
export const config = { matcher: '/(.*)' };
export function middleware(request: NextRequest) {
// Inject speed insights and analytics
injectSpeedInsights();
injectAnalytics();
const path = request.nextUrl.pathname;
if (path.startsWith('/docs')) NextResponse.redirect(new URL(path,
CONFIG.DOCS_URL || process.env.DOCS_URL || 'https://docs.hfut.info'
));
// Find the matched base
for (const [base, endpoints] of Object.entries(API)) {
// Not starts with the base
if (!path.startsWith(base)) continue;
// Base matched, find the matched endpoint
FindNext: for (let endpoint of endpoints) {
// Create copies of the path
let pathcopy = path.slice(base.length);
if (!pathcopy.endsWith('/'))
pathcopy += '/'; // compate with `/` endpoint
let realpath = '';
// Check each part of the endpoint
let part: string | undefined;
let params: [string, string][] = [];
while (part = /^\/(?:[a-z_\d\-]+|:[a-z_][a-z_\d]*[^\/]*)/i.exec(endpoint)?.[0]) {
// Check if the part is a parameter
let param = /^\/:([a-z_][a-z_\d]*)([^\/]*)/i.exec(part);
// Is a `/:key{pattern}` like part
if (param) {
const key = param[1];
const pattern = param[2] !== '' ? param[2] : '[a-z_\d\-]+';
const vmatch = new RegExp(`^/(${pattern})(?=/|$)`, 'i').exec(pathcopy);
if (!vmatch) continue FindNext;
params.push([key, vmatch[1]]);
pathcopy = pathcopy.slice(vmatch[0].length);
endpoint = endpoint.slice(part.length);
realpath += '/$';
continue;
}
// Is a `/path` like part and not starts with it
else if (!pathcopy.startsWith(part)) continue FindNext;
// Is a `/path` like part and starts with it
pathcopy = pathcopy.slice(part.length);
endpoint = endpoint.slice(part.length);
realpath += part;
}
// Return the matched endpoint
const url = new URL(`${base}${realpath}`, request.url);
request.nextUrl.searchParams.forEach((value, key) =>
request.headers.append(`${
CONFIG.HEADER_PARAM_PREFIX ||
process.env.HEADER_PARAM_PREFIX ||
'X-HFUTINFO-'
}${key}`, value));
params.forEach(([key, value]) =>
request.headers.append(`${
CONFIG.HEADER_PARAM_PREFIX ||
process.env.HEADER_PARAM_PREFIX ||
'X-HFUTINFO-'
}${key}`, value));
return NextResponse.rewrite(url, { request });
}
// No endpoint matched
return error(new Error(`Resource ${path} not found`), 404);
}
// No base matched
return error(new Error(`Resource ${path} not found`), 404);
}