-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathmiddleware.ts
76 lines (64 loc) · 2.12 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
import { NextResponse, type NextRequest } from "next/server";
export function generateCsp() {
// generate random nonce converted to base64. Must be different on every HTTP page load
// const nonce = crypto.randomBytes(16).toString('base64')
const nonce = crypto.randomUUID();
const csp = [
{ name: "default-src", values: ["'self'"] },
{
name: "script-src",
values: [
"'report-sample'",
"'self'",
`'nonce-${nonce}'`,
"'strict-dynamic'",
],
},
{
name: "style-src",
values: ["'report-sample'", "'self'", `'nonce-${nonce}'`],
},
{
name: "connect-src",
values: ["'self'", "*.vercel-insights.com", "plausible.io"],
},
{ name: "font-src", values: ["'self'", "data:"] },
{ name: "img-src", values: ["'self'", "data:"] },
{ name: "worker-src", values: ["'self'", "blob:"] },
{ name: "frame-ancestors", values: ["'none'"] },
{ name: "form-action", values: ["'self'"] },
];
const cspString = csp
.map((directive) => {
return `${directive.name} ${directive.values.join(" ")}`;
})
.join("; ");
return { csp: cspString, nonce };
}
export async function middleware(request: NextRequest) {
// generate CSP and nonce
const { csp, nonce } = generateCsp();
// Clone the request headers
const requestHeaders = new Headers(request.headers);
// set nonce request header to read in pages if needed
requestHeaders.set("x-nonce", nonce);
// set CSP header
// switching for report-only or regular for repro on
// https://github.com/vercel/next.js/issues/48966
const headerKey =
request.nextUrl.pathname === "/csp-report-only"
? "content-security-policy-report-only"
: "content-security-policy";
// Set the CSP header so that `app-render` can read it and generate tags with the nonce
requestHeaders.set(headerKey, csp);
// create new response
const response = NextResponse.next({
request: {
// New request headers
headers: requestHeaders,
},
});
// Also set the CSP so that it is outputted to the browser
response.headers.set(headerKey, csp);
return response;
}