@nocobase/plugin-idp-oauth 2.1.0-alpha.17 → 2.1.0-alpha.19

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (221) hide show
  1. package/build.config.ts +1 -1
  2. package/dist/externalVersion.js +6 -4
  3. package/dist/node_modules/light-my-request/package.json +1 -1
  4. package/dist/node_modules/undici/LICENSE +21 -0
  5. package/dist/node_modules/undici/README.md +741 -0
  6. package/dist/node_modules/undici/docs/docs/api/Agent.md +84 -0
  7. package/dist/node_modules/undici/docs/docs/api/BalancedPool.md +99 -0
  8. package/dist/node_modules/undici/docs/docs/api/CacheStorage.md +30 -0
  9. package/dist/node_modules/undici/docs/docs/api/CacheStore.md +164 -0
  10. package/dist/node_modules/undici/docs/docs/api/Client.md +285 -0
  11. package/dist/node_modules/undici/docs/docs/api/ClientStats.md +27 -0
  12. package/dist/node_modules/undici/docs/docs/api/Connector.md +115 -0
  13. package/dist/node_modules/undici/docs/docs/api/ContentType.md +57 -0
  14. package/dist/node_modules/undici/docs/docs/api/Cookies.md +101 -0
  15. package/dist/node_modules/undici/docs/docs/api/Debug.md +62 -0
  16. package/dist/node_modules/undici/docs/docs/api/DiagnosticsChannel.md +315 -0
  17. package/dist/node_modules/undici/docs/docs/api/Dispatcher.md +1392 -0
  18. package/dist/node_modules/undici/docs/docs/api/EnvHttpProxyAgent.md +159 -0
  19. package/dist/node_modules/undici/docs/docs/api/Errors.md +49 -0
  20. package/dist/node_modules/undici/docs/docs/api/EventSource.md +45 -0
  21. package/dist/node_modules/undici/docs/docs/api/Fetch.md +60 -0
  22. package/dist/node_modules/undici/docs/docs/api/GlobalInstallation.md +139 -0
  23. package/dist/node_modules/undici/docs/docs/api/H2CClient.md +263 -0
  24. package/dist/node_modules/undici/docs/docs/api/MockAgent.md +603 -0
  25. package/dist/node_modules/undici/docs/docs/api/MockCallHistory.md +197 -0
  26. package/dist/node_modules/undici/docs/docs/api/MockCallHistoryLog.md +43 -0
  27. package/dist/node_modules/undici/docs/docs/api/MockClient.md +81 -0
  28. package/dist/node_modules/undici/docs/docs/api/MockErrors.md +12 -0
  29. package/dist/node_modules/undici/docs/docs/api/MockPool.md +555 -0
  30. package/dist/node_modules/undici/docs/docs/api/Pool.md +84 -0
  31. package/dist/node_modules/undici/docs/docs/api/PoolStats.md +35 -0
  32. package/dist/node_modules/undici/docs/docs/api/ProxyAgent.md +229 -0
  33. package/dist/node_modules/undici/docs/docs/api/RedirectHandler.md +93 -0
  34. package/dist/node_modules/undici/docs/docs/api/RetryAgent.md +50 -0
  35. package/dist/node_modules/undici/docs/docs/api/RetryHandler.md +118 -0
  36. package/dist/node_modules/undici/docs/docs/api/RoundRobinPool.md +145 -0
  37. package/dist/node_modules/undici/docs/docs/api/SnapshotAgent.md +616 -0
  38. package/dist/node_modules/undici/docs/docs/api/Socks5ProxyAgent.md +274 -0
  39. package/dist/node_modules/undici/docs/docs/api/Util.md +25 -0
  40. package/dist/node_modules/undici/docs/docs/api/WebSocket.md +141 -0
  41. package/dist/node_modules/undici/docs/docs/api/api-lifecycle.md +91 -0
  42. package/dist/node_modules/undici/docs/docs/best-practices/client-certificate.md +64 -0
  43. package/dist/node_modules/undici/docs/docs/best-practices/crawling.md +58 -0
  44. package/dist/node_modules/undici/docs/docs/best-practices/mocking-request.md +190 -0
  45. package/dist/node_modules/undici/docs/docs/best-practices/proxy.md +127 -0
  46. package/dist/node_modules/undici/docs/docs/best-practices/undici-vs-builtin-fetch.md +224 -0
  47. package/dist/node_modules/undici/docs/docs/best-practices/writing-tests.md +63 -0
  48. package/dist/node_modules/undici/index-fetch.js +65 -0
  49. package/dist/node_modules/undici/index.d.ts +3 -0
  50. package/dist/node_modules/undici/index.js +234 -0
  51. package/dist/node_modules/undici/lib/api/abort-signal.js +59 -0
  52. package/dist/node_modules/undici/lib/api/api-connect.js +110 -0
  53. package/dist/node_modules/undici/lib/api/api-pipeline.js +252 -0
  54. package/dist/node_modules/undici/lib/api/api-request.js +214 -0
  55. package/dist/node_modules/undici/lib/api/api-stream.js +209 -0
  56. package/dist/node_modules/undici/lib/api/api-upgrade.js +111 -0
  57. package/dist/node_modules/undici/lib/api/index.js +7 -0
  58. package/dist/node_modules/undici/lib/api/readable.js +580 -0
  59. package/dist/node_modules/undici/lib/cache/memory-cache-store.js +234 -0
  60. package/dist/node_modules/undici/lib/cache/sqlite-cache-store.js +461 -0
  61. package/dist/node_modules/undici/lib/core/connect.js +137 -0
  62. package/dist/node_modules/undici/lib/core/constants.js +143 -0
  63. package/dist/node_modules/undici/lib/core/diagnostics.js +227 -0
  64. package/dist/node_modules/undici/lib/core/errors.js +477 -0
  65. package/dist/node_modules/undici/lib/core/request.js +438 -0
  66. package/dist/node_modules/undici/lib/core/socks5-client.js +407 -0
  67. package/dist/node_modules/undici/lib/core/socks5-utils.js +203 -0
  68. package/dist/node_modules/undici/lib/core/symbols.js +75 -0
  69. package/dist/node_modules/undici/lib/core/tree.js +160 -0
  70. package/dist/node_modules/undici/lib/core/util.js +992 -0
  71. package/dist/node_modules/undici/lib/dispatcher/agent.js +158 -0
  72. package/dist/node_modules/undici/lib/dispatcher/balanced-pool.js +219 -0
  73. package/dist/node_modules/undici/lib/dispatcher/client-h1.js +1610 -0
  74. package/dist/node_modules/undici/lib/dispatcher/client-h2.js +995 -0
  75. package/dist/node_modules/undici/lib/dispatcher/client.js +659 -0
  76. package/dist/node_modules/undici/lib/dispatcher/dispatcher-base.js +165 -0
  77. package/dist/node_modules/undici/lib/dispatcher/dispatcher.js +48 -0
  78. package/dist/node_modules/undici/lib/dispatcher/env-http-proxy-agent.js +146 -0
  79. package/dist/node_modules/undici/lib/dispatcher/fixed-queue.js +135 -0
  80. package/dist/node_modules/undici/lib/dispatcher/h2c-client.js +51 -0
  81. package/dist/node_modules/undici/lib/dispatcher/pool-base.js +214 -0
  82. package/dist/node_modules/undici/lib/dispatcher/pool.js +118 -0
  83. package/dist/node_modules/undici/lib/dispatcher/proxy-agent.js +318 -0
  84. package/dist/node_modules/undici/lib/dispatcher/retry-agent.js +35 -0
  85. package/dist/node_modules/undici/lib/dispatcher/round-robin-pool.js +137 -0
  86. package/dist/node_modules/undici/lib/dispatcher/socks5-proxy-agent.js +249 -0
  87. package/dist/node_modules/undici/lib/encoding/index.js +33 -0
  88. package/dist/node_modules/undici/lib/global.js +50 -0
  89. package/dist/node_modules/undici/lib/handler/cache-handler.js +578 -0
  90. package/dist/node_modules/undici/lib/handler/cache-revalidation-handler.js +124 -0
  91. package/dist/node_modules/undici/lib/handler/decorator-handler.js +67 -0
  92. package/dist/node_modules/undici/lib/handler/deduplication-handler.js +460 -0
  93. package/dist/node_modules/undici/lib/handler/redirect-handler.js +238 -0
  94. package/dist/node_modules/undici/lib/handler/retry-handler.js +394 -0
  95. package/dist/node_modules/undici/lib/handler/unwrap-handler.js +100 -0
  96. package/dist/node_modules/undici/lib/handler/wrap-handler.js +105 -0
  97. package/dist/node_modules/undici/lib/interceptor/cache.js +495 -0
  98. package/dist/node_modules/undici/lib/interceptor/decompress.js +259 -0
  99. package/dist/node_modules/undici/lib/interceptor/deduplicate.js +117 -0
  100. package/dist/node_modules/undici/lib/interceptor/dns.js +571 -0
  101. package/dist/node_modules/undici/lib/interceptor/dump.js +112 -0
  102. package/dist/node_modules/undici/lib/interceptor/redirect.js +21 -0
  103. package/dist/node_modules/undici/lib/interceptor/response-error.js +95 -0
  104. package/dist/node_modules/undici/lib/interceptor/retry.js +19 -0
  105. package/dist/node_modules/undici/lib/llhttp/.gitkeep +0 -0
  106. package/dist/node_modules/undici/lib/llhttp/constants.d.ts +195 -0
  107. package/dist/node_modules/undici/lib/llhttp/constants.js +531 -0
  108. package/dist/node_modules/undici/lib/llhttp/llhttp-wasm.js +15 -0
  109. package/dist/node_modules/undici/lib/llhttp/llhttp_simd-wasm.js +15 -0
  110. package/dist/node_modules/undici/lib/llhttp/utils.d.ts +2 -0
  111. package/dist/node_modules/undici/lib/llhttp/utils.js +12 -0
  112. package/dist/node_modules/undici/lib/mock/mock-agent.js +232 -0
  113. package/dist/node_modules/undici/lib/mock/mock-call-history.js +248 -0
  114. package/dist/node_modules/undici/lib/mock/mock-client.js +68 -0
  115. package/dist/node_modules/undici/lib/mock/mock-errors.js +29 -0
  116. package/dist/node_modules/undici/lib/mock/mock-interceptor.js +209 -0
  117. package/dist/node_modules/undici/lib/mock/mock-pool.js +68 -0
  118. package/dist/node_modules/undici/lib/mock/mock-symbols.js +32 -0
  119. package/dist/node_modules/undici/lib/mock/mock-utils.js +486 -0
  120. package/dist/node_modules/undici/lib/mock/pending-interceptors-formatter.js +43 -0
  121. package/dist/node_modules/undici/lib/mock/snapshot-agent.js +353 -0
  122. package/dist/node_modules/undici/lib/mock/snapshot-recorder.js +588 -0
  123. package/dist/node_modules/undici/lib/mock/snapshot-utils.js +158 -0
  124. package/dist/node_modules/undici/lib/util/cache.js +407 -0
  125. package/dist/node_modules/undici/lib/util/date.js +653 -0
  126. package/dist/node_modules/undici/lib/util/promise.js +28 -0
  127. package/dist/node_modules/undici/lib/util/runtime-features.js +124 -0
  128. package/dist/node_modules/undici/lib/util/stats.js +32 -0
  129. package/dist/node_modules/undici/lib/util/timers.js +425 -0
  130. package/dist/node_modules/undici/lib/web/cache/cache.js +864 -0
  131. package/dist/node_modules/undici/lib/web/cache/cachestorage.js +152 -0
  132. package/dist/node_modules/undici/lib/web/cache/util.js +45 -0
  133. package/dist/node_modules/undici/lib/web/cookies/constants.js +12 -0
  134. package/dist/node_modules/undici/lib/web/cookies/index.js +199 -0
  135. package/dist/node_modules/undici/lib/web/cookies/parse.js +322 -0
  136. package/dist/node_modules/undici/lib/web/cookies/util.js +282 -0
  137. package/dist/node_modules/undici/lib/web/eventsource/eventsource-stream.js +399 -0
  138. package/dist/node_modules/undici/lib/web/eventsource/eventsource.js +501 -0
  139. package/dist/node_modules/undici/lib/web/eventsource/util.js +29 -0
  140. package/dist/node_modules/undici/lib/web/fetch/LICENSE +21 -0
  141. package/dist/node_modules/undici/lib/web/fetch/body.js +509 -0
  142. package/dist/node_modules/undici/lib/web/fetch/constants.js +131 -0
  143. package/dist/node_modules/undici/lib/web/fetch/data-url.js +596 -0
  144. package/dist/node_modules/undici/lib/web/fetch/formdata-parser.js +575 -0
  145. package/dist/node_modules/undici/lib/web/fetch/formdata.js +259 -0
  146. package/dist/node_modules/undici/lib/web/fetch/global.js +40 -0
  147. package/dist/node_modules/undici/lib/web/fetch/headers.js +719 -0
  148. package/dist/node_modules/undici/lib/web/fetch/index.js +2397 -0
  149. package/dist/node_modules/undici/lib/web/fetch/request.js +1115 -0
  150. package/dist/node_modules/undici/lib/web/fetch/response.js +641 -0
  151. package/dist/node_modules/undici/lib/web/fetch/util.js +1520 -0
  152. package/dist/node_modules/undici/lib/web/infra/index.js +229 -0
  153. package/dist/node_modules/undici/lib/web/subresource-integrity/Readme.md +9 -0
  154. package/dist/node_modules/undici/lib/web/subresource-integrity/subresource-integrity.js +307 -0
  155. package/dist/node_modules/undici/lib/web/webidl/index.js +1006 -0
  156. package/dist/node_modules/undici/lib/web/websocket/connection.js +329 -0
  157. package/dist/node_modules/undici/lib/web/websocket/constants.js +126 -0
  158. package/dist/node_modules/undici/lib/web/websocket/events.js +331 -0
  159. package/dist/node_modules/undici/lib/web/websocket/frame.js +133 -0
  160. package/dist/node_modules/undici/lib/web/websocket/permessage-deflate.js +118 -0
  161. package/dist/node_modules/undici/lib/web/websocket/receiver.js +450 -0
  162. package/dist/node_modules/undici/lib/web/websocket/sender.js +109 -0
  163. package/dist/node_modules/undici/lib/web/websocket/stream/websocketerror.js +104 -0
  164. package/dist/node_modules/undici/lib/web/websocket/stream/websocketstream.js +497 -0
  165. package/dist/node_modules/undici/lib/web/websocket/util.js +347 -0
  166. package/dist/node_modules/undici/lib/web/websocket/websocket.js +751 -0
  167. package/dist/node_modules/undici/package.json +152 -0
  168. package/dist/node_modules/undici/scripts/strip-comments.js +10 -0
  169. package/dist/node_modules/undici/types/README.md +6 -0
  170. package/dist/node_modules/undici/types/agent.d.ts +32 -0
  171. package/dist/node_modules/undici/types/api.d.ts +43 -0
  172. package/dist/node_modules/undici/types/balanced-pool.d.ts +30 -0
  173. package/dist/node_modules/undici/types/cache-interceptor.d.ts +179 -0
  174. package/dist/node_modules/undici/types/cache.d.ts +36 -0
  175. package/dist/node_modules/undici/types/client-stats.d.ts +15 -0
  176. package/dist/node_modules/undici/types/client.d.ts +123 -0
  177. package/dist/node_modules/undici/types/connector.d.ts +36 -0
  178. package/dist/node_modules/undici/types/content-type.d.ts +21 -0
  179. package/dist/node_modules/undici/types/cookies.d.ts +30 -0
  180. package/dist/node_modules/undici/types/diagnostics-channel.d.ts +74 -0
  181. package/dist/node_modules/undici/types/dispatcher.d.ts +273 -0
  182. package/dist/node_modules/undici/types/env-http-proxy-agent.d.ts +22 -0
  183. package/dist/node_modules/undici/types/errors.d.ts +177 -0
  184. package/dist/node_modules/undici/types/eventsource.d.ts +66 -0
  185. package/dist/node_modules/undici/types/fetch.d.ts +231 -0
  186. package/dist/node_modules/undici/types/formdata.d.ts +114 -0
  187. package/dist/node_modules/undici/types/global-dispatcher.d.ts +9 -0
  188. package/dist/node_modules/undici/types/global-origin.d.ts +7 -0
  189. package/dist/node_modules/undici/types/h2c-client.d.ts +73 -0
  190. package/dist/node_modules/undici/types/handlers.d.ts +14 -0
  191. package/dist/node_modules/undici/types/header.d.ts +160 -0
  192. package/dist/node_modules/undici/types/index.d.ts +91 -0
  193. package/dist/node_modules/undici/types/interceptors.d.ts +80 -0
  194. package/dist/node_modules/undici/types/mock-agent.d.ts +68 -0
  195. package/dist/node_modules/undici/types/mock-call-history.d.ts +111 -0
  196. package/dist/node_modules/undici/types/mock-client.d.ts +27 -0
  197. package/dist/node_modules/undici/types/mock-errors.d.ts +12 -0
  198. package/dist/node_modules/undici/types/mock-interceptor.d.ts +94 -0
  199. package/dist/node_modules/undici/types/mock-pool.d.ts +27 -0
  200. package/dist/node_modules/undici/types/patch.d.ts +29 -0
  201. package/dist/node_modules/undici/types/pool-stats.d.ts +19 -0
  202. package/dist/node_modules/undici/types/pool.d.ts +41 -0
  203. package/dist/node_modules/undici/types/proxy-agent.d.ts +29 -0
  204. package/dist/node_modules/undici/types/readable.d.ts +68 -0
  205. package/dist/node_modules/undici/types/retry-agent.d.ts +8 -0
  206. package/dist/node_modules/undici/types/retry-handler.d.ts +125 -0
  207. package/dist/node_modules/undici/types/round-robin-pool.d.ts +41 -0
  208. package/dist/node_modules/undici/types/snapshot-agent.d.ts +109 -0
  209. package/dist/node_modules/undici/types/socks5-proxy-agent.d.ts +25 -0
  210. package/dist/node_modules/undici/types/util.d.ts +18 -0
  211. package/dist/node_modules/undici/types/utility.d.ts +7 -0
  212. package/dist/node_modules/undici/types/webidl.d.ts +347 -0
  213. package/dist/node_modules/undici/types/websocket.d.ts +188 -0
  214. package/dist/server/collections/oidcStates.d.ts +10 -0
  215. package/dist/server/collections/oidcStates.js +96 -0
  216. package/dist/server/db-adapter.d.ts +25 -0
  217. package/dist/server/db-adapter.js +156 -0
  218. package/dist/server/service.js +11 -10
  219. package/package.json +2 -2
  220. package/dist/server/cache-adapter.d.ts +0 -33
  221. package/dist/server/cache-adapter.js +0 -159
@@ -0,0 +1,229 @@
1
+ 'use strict'
2
+
3
+ const assert = require('node:assert')
4
+ const { utf8DecodeBytes } = require('../../encoding')
5
+
6
+ /**
7
+ * @param {(char: string) => boolean} condition
8
+ * @param {string} input
9
+ * @param {{ position: number }} position
10
+ * @returns {string}
11
+ *
12
+ * @see https://infra.spec.whatwg.org/#collect-a-sequence-of-code-points
13
+ */
14
+ function collectASequenceOfCodePoints (condition, input, position) {
15
+ // 1. Let result be the empty string.
16
+ let result = ''
17
+
18
+ // 2. While position doesn’t point past the end of input and the
19
+ // code point at position within input meets the condition condition:
20
+ while (position.position < input.length && condition(input[position.position])) {
21
+ // 1. Append that code point to the end of result.
22
+ result += input[position.position]
23
+
24
+ // 2. Advance position by 1.
25
+ position.position++
26
+ }
27
+
28
+ // 3. Return result.
29
+ return result
30
+ }
31
+
32
+ /**
33
+ * A faster collectASequenceOfCodePoints that only works when comparing a single character.
34
+ * @param {string} char
35
+ * @param {string} input
36
+ * @param {{ position: number }} position
37
+ * @returns {string}
38
+ *
39
+ * @see https://infra.spec.whatwg.org/#collect-a-sequence-of-code-points
40
+ */
41
+ function collectASequenceOfCodePointsFast (char, input, position) {
42
+ const idx = input.indexOf(char, position.position)
43
+ const start = position.position
44
+
45
+ if (idx === -1) {
46
+ position.position = input.length
47
+ return input.slice(start)
48
+ }
49
+
50
+ position.position = idx
51
+ return input.slice(start, position.position)
52
+ }
53
+
54
+ const ASCII_WHITESPACE_REPLACE_REGEX = /[\u0009\u000A\u000C\u000D\u0020]/g // eslint-disable-line no-control-regex
55
+
56
+ /**
57
+ * @param {string} data
58
+ * @returns {Uint8Array | 'failure'}
59
+ *
60
+ * @see https://infra.spec.whatwg.org/#forgiving-base64-decode
61
+ */
62
+ function forgivingBase64 (data) {
63
+ // 1. Remove all ASCII whitespace from data.
64
+ data = data.replace(ASCII_WHITESPACE_REPLACE_REGEX, '')
65
+
66
+ let dataLength = data.length
67
+ // 2. If data’s code point length divides by 4 leaving
68
+ // no remainder, then:
69
+ if (dataLength % 4 === 0) {
70
+ // 1. If data ends with one or two U+003D (=) code points,
71
+ // then remove them from data.
72
+ if (data.charCodeAt(dataLength - 1) === 0x003D) {
73
+ --dataLength
74
+ if (data.charCodeAt(dataLength - 1) === 0x003D) {
75
+ --dataLength
76
+ }
77
+ }
78
+ }
79
+
80
+ // 3. If data’s code point length divides by 4 leaving
81
+ // a remainder of 1, then return failure.
82
+ if (dataLength % 4 === 1) {
83
+ return 'failure'
84
+ }
85
+
86
+ // 4. If data contains a code point that is not one of
87
+ // U+002B (+)
88
+ // U+002F (/)
89
+ // ASCII alphanumeric
90
+ // then return failure.
91
+ if (/[^+/0-9A-Za-z]/.test(data.length === dataLength ? data : data.substring(0, dataLength))) {
92
+ return 'failure'
93
+ }
94
+
95
+ const buffer = Buffer.from(data, 'base64')
96
+ return new Uint8Array(buffer.buffer, buffer.byteOffset, buffer.byteLength)
97
+ }
98
+
99
+ /**
100
+ * @param {number} char
101
+ * @returns {boolean}
102
+ *
103
+ * @see https://infra.spec.whatwg.org/#ascii-whitespace
104
+ */
105
+ function isASCIIWhitespace (char) {
106
+ return (
107
+ char === 0x09 || // \t
108
+ char === 0x0a || // \n
109
+ char === 0x0c || // \f
110
+ char === 0x0d || // \r
111
+ char === 0x20 // space
112
+ )
113
+ }
114
+
115
+ /**
116
+ * @param {Uint8Array} input
117
+ * @returns {string}
118
+ *
119
+ * @see https://infra.spec.whatwg.org/#isomorphic-decode
120
+ */
121
+ function isomorphicDecode (input) {
122
+ // 1. To isomorphic decode a byte sequence input, return a string whose code point
123
+ // length is equal to input’s length and whose code points have the same values
124
+ // as the values of input’s bytes, in the same order.
125
+ const length = input.length
126
+ if ((2 << 15) - 1 > length) {
127
+ return String.fromCharCode.apply(null, input)
128
+ }
129
+ let result = ''
130
+ let i = 0
131
+ let addition = (2 << 15) - 1
132
+ while (i < length) {
133
+ if (i + addition > length) {
134
+ addition = length - i
135
+ }
136
+ result += String.fromCharCode.apply(null, input.subarray(i, i += addition))
137
+ }
138
+ return result
139
+ }
140
+
141
+ const invalidIsomorphicEncodeValueRegex = /[^\x00-\xFF]/ // eslint-disable-line no-control-regex
142
+
143
+ /**
144
+ * @param {string} input
145
+ * @returns {string}
146
+ *
147
+ * @see https://infra.spec.whatwg.org/#isomorphic-encode
148
+ */
149
+ function isomorphicEncode (input) {
150
+ // 1. Assert: input contains no code points greater than U+00FF.
151
+ assert(!invalidIsomorphicEncodeValueRegex.test(input))
152
+
153
+ // 2. Return a byte sequence whose length is equal to input’s code
154
+ // point length and whose bytes have the same values as the
155
+ // values of input’s code points, in the same order
156
+ return input
157
+ }
158
+
159
+ /**
160
+ * @see https://infra.spec.whatwg.org/#parse-json-bytes-to-a-javascript-value
161
+ * @param {Uint8Array} bytes
162
+ */
163
+ function parseJSONFromBytes (bytes) {
164
+ return JSON.parse(utf8DecodeBytes(bytes))
165
+ }
166
+
167
+ /**
168
+ * @param {string} str
169
+ * @param {boolean} [leading=true]
170
+ * @param {boolean} [trailing=true]
171
+ * @returns {string}
172
+ *
173
+ * @see https://infra.spec.whatwg.org/#strip-leading-and-trailing-ascii-whitespace
174
+ */
175
+ function removeASCIIWhitespace (str, leading = true, trailing = true) {
176
+ return removeChars(str, leading, trailing, isASCIIWhitespace)
177
+ }
178
+
179
+ /**
180
+ * @param {string} str
181
+ * @param {boolean} leading
182
+ * @param {boolean} trailing
183
+ * @param {(charCode: number) => boolean} predicate
184
+ * @returns {string}
185
+ */
186
+ function removeChars (str, leading, trailing, predicate) {
187
+ let lead = 0
188
+ let trail = str.length - 1
189
+
190
+ if (leading) {
191
+ while (lead < str.length && predicate(str.charCodeAt(lead))) lead++
192
+ }
193
+
194
+ if (trailing) {
195
+ while (trail > 0 && predicate(str.charCodeAt(trail))) trail--
196
+ }
197
+
198
+ return lead === 0 && trail === str.length - 1 ? str : str.slice(lead, trail + 1)
199
+ }
200
+
201
+ // https://infra.spec.whatwg.org/#serialize-a-javascript-value-to-a-json-string
202
+ function serializeJavascriptValueToJSONString (value) {
203
+ // 1. Let result be ? Call(%JSON.stringify%, undefined, « value »).
204
+ const result = JSON.stringify(value)
205
+
206
+ // 2. If result is undefined, then throw a TypeError.
207
+ if (result === undefined) {
208
+ throw new TypeError('Value is not JSON serializable')
209
+ }
210
+
211
+ // 3. Assert: result is a string.
212
+ assert(typeof result === 'string')
213
+
214
+ // 4. Return result.
215
+ return result
216
+ }
217
+
218
+ module.exports = {
219
+ collectASequenceOfCodePoints,
220
+ collectASequenceOfCodePointsFast,
221
+ forgivingBase64,
222
+ isASCIIWhitespace,
223
+ isomorphicDecode,
224
+ isomorphicEncode,
225
+ parseJSONFromBytes,
226
+ removeASCIIWhitespace,
227
+ removeChars,
228
+ serializeJavascriptValueToJSONString
229
+ }
@@ -0,0 +1,9 @@
1
+ # Subresource Integrity
2
+
3
+ based on Editor’s Draft, 12 June 2025
4
+
5
+ This module provides support for Subresource Integrity (SRI) in the context of web fetch operations. SRI is a security feature that allows clients to verify that fetched resources are delivered without unexpected manipulation.
6
+
7
+ ## Links
8
+
9
+ - [Subresource Integrity](https://w3c.github.io/webappsec-subresource-integrity/)
@@ -0,0 +1,307 @@
1
+ 'use strict'
2
+
3
+ const assert = require('node:assert')
4
+ const { runtimeFeatures } = require('../../util/runtime-features.js')
5
+
6
+ /**
7
+ * @typedef {object} Metadata
8
+ * @property {SRIHashAlgorithm} alg - The algorithm used for the hash.
9
+ * @property {string} val - The base64-encoded hash value.
10
+ */
11
+
12
+ /**
13
+ * @typedef {Metadata[]} MetadataList
14
+ */
15
+
16
+ /**
17
+ * @typedef {('sha256' | 'sha384' | 'sha512')} SRIHashAlgorithm
18
+ */
19
+
20
+ /**
21
+ * @type {Map<SRIHashAlgorithm, number>}
22
+ *
23
+ * The valid SRI hash algorithm token set is the ordered set « "sha256",
24
+ * "sha384", "sha512" » (corresponding to SHA-256, SHA-384, and SHA-512
25
+ * respectively). The ordering of this set is meaningful, with stronger
26
+ * algorithms appearing later in the set.
27
+ *
28
+ * @see https://w3c.github.io/webappsec-subresource-integrity/#valid-sri-hash-algorithm-token-set
29
+ */
30
+ const validSRIHashAlgorithmTokenSet = new Map([['sha256', 0], ['sha384', 1], ['sha512', 2]])
31
+
32
+ // https://nodejs.org/api/crypto.html#determining-if-crypto-support-is-unavailable
33
+ /** @type {import('node:crypto')} */
34
+ let crypto
35
+
36
+ if (runtimeFeatures.has('crypto')) {
37
+ crypto = require('node:crypto')
38
+ const cryptoHashes = crypto.getHashes()
39
+
40
+ // If no hashes are available, we cannot support SRI.
41
+ if (cryptoHashes.length === 0) {
42
+ validSRIHashAlgorithmTokenSet.clear()
43
+ }
44
+
45
+ for (const algorithm of validSRIHashAlgorithmTokenSet.keys()) {
46
+ // If the algorithm is not supported, remove it from the list.
47
+ if (cryptoHashes.includes(algorithm) === false) {
48
+ validSRIHashAlgorithmTokenSet.delete(algorithm)
49
+ }
50
+ }
51
+ } else {
52
+ // If crypto is not available, we cannot support SRI.
53
+ validSRIHashAlgorithmTokenSet.clear()
54
+ }
55
+
56
+ /**
57
+ * @typedef GetSRIHashAlgorithmIndex
58
+ * @type {(algorithm: SRIHashAlgorithm) => number}
59
+ * @param {SRIHashAlgorithm} algorithm
60
+ * @returns {number} The index of the algorithm in the valid SRI hash algorithm
61
+ * token set.
62
+ */
63
+
64
+ const getSRIHashAlgorithmIndex = /** @type {GetSRIHashAlgorithmIndex} */ (Map.prototype.get.bind(
65
+ validSRIHashAlgorithmTokenSet))
66
+
67
+ /**
68
+ * @typedef IsValidSRIHashAlgorithm
69
+ * @type {(algorithm: string) => algorithm is SRIHashAlgorithm}
70
+ * @param {*} algorithm
71
+ * @returns {algorithm is SRIHashAlgorithm}
72
+ */
73
+
74
+ const isValidSRIHashAlgorithm = /** @type {IsValidSRIHashAlgorithm} */ (
75
+ Map.prototype.has.bind(validSRIHashAlgorithmTokenSet)
76
+ )
77
+
78
+ /**
79
+ * @param {Uint8Array} bytes
80
+ * @param {string} metadataList
81
+ * @returns {boolean}
82
+ *
83
+ * @see https://w3c.github.io/webappsec-subresource-integrity/#does-response-match-metadatalist
84
+ */
85
+ const bytesMatch = runtimeFeatures.has('crypto') === false || validSRIHashAlgorithmTokenSet.size === 0
86
+ // If node is not built with OpenSSL support, we cannot check
87
+ // a request's integrity, so allow it by default (the spec will
88
+ // allow requests if an invalid hash is given, as precedence).
89
+ ? () => true
90
+ : (bytes, metadataList) => {
91
+ // 1. Let parsedMetadata be the result of parsing metadataList.
92
+ const parsedMetadata = parseMetadata(metadataList)
93
+
94
+ // 2. If parsedMetadata is empty set, return true.
95
+ if (parsedMetadata.length === 0) {
96
+ return true
97
+ }
98
+
99
+ // 3. Let metadata be the result of getting the strongest
100
+ // metadata from parsedMetadata.
101
+ const metadata = getStrongestMetadata(parsedMetadata)
102
+
103
+ // 4. For each item in metadata:
104
+ for (const item of metadata) {
105
+ // 1. Let algorithm be the item["alg"].
106
+ const algorithm = item.alg
107
+
108
+ // 2. Let expectedValue be the item["val"].
109
+ const expectedValue = item.val
110
+
111
+ // See https://github.com/web-platform-tests/wpt/commit/e4c5cc7a5e48093220528dfdd1c4012dc3837a0e
112
+ // "be liberal with padding". This is annoying, and it's not even in the spec.
113
+
114
+ // 3. Let actualValue be the result of applying algorithm to bytes .
115
+ const actualValue = applyAlgorithmToBytes(algorithm, bytes)
116
+
117
+ // 4. If actualValue is a case-sensitive match for expectedValue,
118
+ // return true.
119
+ if (caseSensitiveMatch(actualValue, expectedValue)) {
120
+ return true
121
+ }
122
+ }
123
+
124
+ // 5. Return false.
125
+ return false
126
+ }
127
+
128
+ /**
129
+ * @param {MetadataList} metadataList
130
+ * @returns {MetadataList} The strongest hash algorithm from the metadata list.
131
+ */
132
+ function getStrongestMetadata (metadataList) {
133
+ // 1. Let result be the empty set and strongest be the empty string.
134
+ const result = []
135
+ /** @type {Metadata|null} */
136
+ let strongest = null
137
+
138
+ // 2. For each item in set:
139
+ for (const item of metadataList) {
140
+ // 1. Assert: item["alg"] is a valid SRI hash algorithm token.
141
+ assert(isValidSRIHashAlgorithm(item.alg), 'Invalid SRI hash algorithm token')
142
+
143
+ // 2. If result is the empty set, then:
144
+ if (result.length === 0) {
145
+ // 1. Append item to result.
146
+ result.push(item)
147
+
148
+ // 2. Set strongest to item.
149
+ strongest = item
150
+
151
+ // 3. Continue.
152
+ continue
153
+ }
154
+
155
+ // 3. Let currentAlgorithm be strongest["alg"], and currentAlgorithmIndex be
156
+ // the index of currentAlgorithm in the valid SRI hash algorithm token set.
157
+ const currentAlgorithm = /** @type {Metadata} */ (strongest).alg
158
+ const currentAlgorithmIndex = getSRIHashAlgorithmIndex(currentAlgorithm)
159
+
160
+ // 4. Let newAlgorithm be the item["alg"], and newAlgorithmIndex be the
161
+ // index of newAlgorithm in the valid SRI hash algorithm token set.
162
+ const newAlgorithm = item.alg
163
+ const newAlgorithmIndex = getSRIHashAlgorithmIndex(newAlgorithm)
164
+
165
+ // 5. If newAlgorithmIndex is less than currentAlgorithmIndex, then continue.
166
+ if (newAlgorithmIndex < currentAlgorithmIndex) {
167
+ continue
168
+
169
+ // 6. Otherwise, if newAlgorithmIndex is greater than
170
+ // currentAlgorithmIndex:
171
+ } else if (newAlgorithmIndex > currentAlgorithmIndex) {
172
+ // 1. Set strongest to item.
173
+ strongest = item
174
+
175
+ // 2. Set result to « item ».
176
+ result[0] = item
177
+ result.length = 1
178
+
179
+ // 7. Otherwise, newAlgorithmIndex and currentAlgorithmIndex are the same
180
+ // value. Append item to result.
181
+ } else {
182
+ result.push(item)
183
+ }
184
+ }
185
+
186
+ // 3. Return result.
187
+ return result
188
+ }
189
+
190
+ /**
191
+ * @param {string} metadata
192
+ * @returns {MetadataList}
193
+ *
194
+ * @see https://w3c.github.io/webappsec-subresource-integrity/#parse-metadata
195
+ */
196
+ function parseMetadata (metadata) {
197
+ // 1. Let result be the empty set.
198
+ /** @type {MetadataList} */
199
+ const result = []
200
+
201
+ // 2. For each item returned by splitting metadata on spaces:
202
+ for (const item of metadata.split(' ')) {
203
+ // 1. Let expression-and-options be the result of splitting item on U+003F (?).
204
+ const expressionAndOptions = item.split('?', 1)
205
+
206
+ // 2. Let algorithm-expression be expression-and-options[0].
207
+ const algorithmExpression = expressionAndOptions[0]
208
+
209
+ // 3. Let base64-value be the empty string.
210
+ let base64Value = ''
211
+
212
+ // 4. Let algorithm-and-value be the result of splitting algorithm-expression on U+002D (-).
213
+ const algorithmAndValue = [algorithmExpression.slice(0, 6), algorithmExpression.slice(7)]
214
+
215
+ // 5. Let algorithm be algorithm-and-value[0].
216
+ const algorithm = algorithmAndValue[0]
217
+
218
+ // 6. If algorithm is not a valid SRI hash algorithm token, then continue.
219
+ if (!isValidSRIHashAlgorithm(algorithm)) {
220
+ continue
221
+ }
222
+
223
+ // 7. If algorithm-and-value[1] exists, set base64-value to
224
+ // algorithm-and-value[1].
225
+ if (algorithmAndValue[1]) {
226
+ base64Value = algorithmAndValue[1]
227
+ }
228
+
229
+ // 8. Let metadata be the ordered map
230
+ // «["alg" → algorithm, "val" → base64-value]».
231
+ const metadata = {
232
+ alg: algorithm,
233
+ val: base64Value
234
+ }
235
+
236
+ // 9. Append metadata to result.
237
+ result.push(metadata)
238
+ }
239
+
240
+ // 3. Return result.
241
+ return result
242
+ }
243
+
244
+ /**
245
+ * Applies the specified hash algorithm to the given bytes
246
+ *
247
+ * @typedef {(algorithm: SRIHashAlgorithm, bytes: Uint8Array) => string} ApplyAlgorithmToBytes
248
+ * @param {SRIHashAlgorithm} algorithm
249
+ * @param {Uint8Array} bytes
250
+ * @returns {string}
251
+ */
252
+ const applyAlgorithmToBytes = (algorithm, bytes) => {
253
+ return crypto.hash(algorithm, bytes, 'base64')
254
+ }
255
+
256
+ /**
257
+ * Compares two base64 strings, allowing for base64url
258
+ * in the second string.
259
+ *
260
+ * @param {string} actualValue base64 encoded string
261
+ * @param {string} expectedValue base64 or base64url encoded string
262
+ * @returns {boolean}
263
+ */
264
+ function caseSensitiveMatch (actualValue, expectedValue) {
265
+ // Ignore padding characters from the end of the strings by
266
+ // decreasing the length by 1 or 2 if the last characters are `=`.
267
+ let actualValueLength = actualValue.length
268
+ if (actualValueLength !== 0 && actualValue[actualValueLength - 1] === '=') {
269
+ actualValueLength -= 1
270
+ }
271
+ if (actualValueLength !== 0 && actualValue[actualValueLength - 1] === '=') {
272
+ actualValueLength -= 1
273
+ }
274
+ let expectedValueLength = expectedValue.length
275
+ if (expectedValueLength !== 0 && expectedValue[expectedValueLength - 1] === '=') {
276
+ expectedValueLength -= 1
277
+ }
278
+ if (expectedValueLength !== 0 && expectedValue[expectedValueLength - 1] === '=') {
279
+ expectedValueLength -= 1
280
+ }
281
+
282
+ if (actualValueLength !== expectedValueLength) {
283
+ return false
284
+ }
285
+
286
+ for (let i = 0; i < actualValueLength; ++i) {
287
+ if (
288
+ actualValue[i] === expectedValue[i] ||
289
+ (actualValue[i] === '+' && expectedValue[i] === '-') ||
290
+ (actualValue[i] === '/' && expectedValue[i] === '_')
291
+ ) {
292
+ continue
293
+ }
294
+ return false
295
+ }
296
+
297
+ return true
298
+ }
299
+
300
+ module.exports = {
301
+ applyAlgorithmToBytes,
302
+ bytesMatch,
303
+ caseSensitiveMatch,
304
+ isValidSRIHashAlgorithm,
305
+ getStrongestMetadata,
306
+ parseMetadata
307
+ }