-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmiddleware.ts
80 lines (67 loc) · 2.26 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
import type { NextRequest } from "next/server"
import { NextResponse } from "next/server"
/** 参考: https://blog.s2n.tech/articles/nextjs-app-dir-csp */
const middleware = (req: NextRequest): NextResponse | void => {
// Nonceの生成
const nonce = generateNonce()
// CSPヘッダの生成
const csp = generateCspHeader(nonce)
// リクエストヘッダを取得
const headers = new Headers(req.headers)
// コンポーネント側で取得できるようにリクエストヘッダにも設定
headers.set("X-CSP-Nonce", nonce)
// Next.jsが差し込むインラインスクリプトにもNonceが設定されるようにリクエストヘッダにもCSPを設定
headers.set("Content-Security-Policy", csp)
// 改変したリクエストヘッダをNextResponseに渡す
const response = NextResponse.next({
request: {
headers
}
})
// レスポンスヘッダにCSPを設定
response.headers.set("Content-Security-Policy", csp)
return response
}
// Nonceのビット長
// 参考: https://w3c.github.io/webappsec-csp/#security-nonces
const NONCE_BIT_LENGTH = 128
// Nonceの生成
// Node.jsのAPIは利用できないので、Web Crypto APIを使用
const generateNonce = (): string => {
return bufferToHex(
crypto.getRandomValues(new Uint8Array(NONCE_BIT_LENGTH / 8))
)
}
// CSPヘッダの生成
const generateCspHeader = (nonce: string): string => {
const scriptSrc = [
"'self'",
"'unsafe-inline'",
"https:",
"http:",
// 開発環境ではevalを許可
process.env.NODE_ENV === "development" && "'unsafe-eval'",
`'nonce-${nonce}'`,
// Twitterの埋め込みやGoogle Tag Managerを使っている場合は適宜設定
//"https://www.googletagmanager.com",
//"https://platform.twitter.com"
"'strict-dynamic'"
]
.filter(Boolean)
.join(" ")
// CSPの設定
// 自分のサイトの状況に応じて適宜設定
const csp = [
"object-src 'none'",
"base-uri 'none'",
`script-src ${scriptSrc}`
].join("; ")
return csp
}
// ArrayBufferを16進数の文字列に変換する
const bufferToHex = (buffer: ArrayBuffer): string => {
return Array.from(new Uint8Array(buffer))
.map(byte => byte.toString(16).padStart(2, "0"))
.join("")
}
export default middleware