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

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 (214) hide show
  1. package/build.config.ts +1 -1
  2. package/dist/externalVersion.js +4 -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/package.json +2 -2
@@ -0,0 +1,329 @@
1
+ 'use strict'
2
+
3
+ const { uid, states, sentCloseFrameState, emptyBuffer, opcodes } = require('./constants')
4
+ const { parseExtensions, isClosed, isClosing, isEstablished, isConnecting, validateCloseCodeAndReason } = require('./util')
5
+ const { makeRequest } = require('../fetch/request')
6
+ const { fetching } = require('../fetch/index')
7
+ const { Headers, getHeadersList } = require('../fetch/headers')
8
+ const { getDecodeSplit } = require('../fetch/util')
9
+ const { WebsocketFrameSend } = require('./frame')
10
+ const assert = require('node:assert')
11
+ const { runtimeFeatures } = require('../../util/runtime-features')
12
+
13
+ const crypto = runtimeFeatures.has('crypto')
14
+ ? require('node:crypto')
15
+ : null
16
+
17
+ let warningEmitted = false
18
+
19
+ /**
20
+ * @see https://websockets.spec.whatwg.org/#concept-websocket-establish
21
+ * @param {URL} url
22
+ * @param {string|string[]} protocols
23
+ * @param {import('./websocket').Handler} handler
24
+ * @param {Partial<import('../../../types/websocket').WebSocketInit>} options
25
+ */
26
+ function establishWebSocketConnection (url, protocols, client, handler, options) {
27
+ // 1. Let requestURL be a copy of url, with its scheme set to "http", if url’s
28
+ // scheme is "ws", and to "https" otherwise.
29
+ const requestURL = url
30
+
31
+ requestURL.protocol = url.protocol === 'ws:' ? 'http:' : 'https:'
32
+
33
+ // 2. Let request be a new request, whose URL is requestURL, client is client,
34
+ // service-workers mode is "none", referrer is "no-referrer", mode is
35
+ // "websocket", credentials mode is "include", cache mode is "no-store" ,
36
+ // redirect mode is "error", and use-URL-credentials flag is set.
37
+ const request = makeRequest({
38
+ urlList: [requestURL],
39
+ client,
40
+ serviceWorkers: 'none',
41
+ referrer: 'no-referrer',
42
+ mode: 'websocket',
43
+ credentials: 'include',
44
+ cache: 'no-store',
45
+ redirect: 'error',
46
+ useURLCredentials: true
47
+ })
48
+
49
+ // Note: undici extension, allow setting custom headers.
50
+ if (options.headers) {
51
+ const headersList = getHeadersList(new Headers(options.headers))
52
+
53
+ request.headersList = headersList
54
+ }
55
+
56
+ // 3. Append (`Upgrade`, `websocket`) to request’s header list.
57
+ // 4. Append (`Connection`, `Upgrade`) to request’s header list.
58
+ // Note: both of these are handled by undici currently.
59
+ // https://github.com/nodejs/undici/blob/68c269c4144c446f3f1220951338daef4a6b5ec4/lib/client.js#L1397
60
+
61
+ // 5. Let keyValue be a nonce consisting of a randomly selected
62
+ // 16-byte value that has been forgiving-base64-encoded and
63
+ // isomorphic encoded.
64
+ const keyValue = crypto.randomBytes(16).toString('base64')
65
+
66
+ // 6. Append (`Sec-WebSocket-Key`, keyValue) to request’s
67
+ // header list.
68
+ request.headersList.append('sec-websocket-key', keyValue, true)
69
+
70
+ // 7. Append (`Sec-WebSocket-Version`, `13`) to request’s
71
+ // header list.
72
+ request.headersList.append('sec-websocket-version', '13', true)
73
+
74
+ // 8. For each protocol in protocols, combine
75
+ // (`Sec-WebSocket-Protocol`, protocol) in request’s header
76
+ // list.
77
+ for (const protocol of protocols) {
78
+ request.headersList.append('sec-websocket-protocol', protocol, true)
79
+ }
80
+
81
+ // 9. Let permessageDeflate be a user-agent defined
82
+ // "permessage-deflate" extension header value.
83
+ // https://github.com/mozilla/gecko-dev/blob/ce78234f5e653a5d3916813ff990f053510227bc/netwerk/protocol/websocket/WebSocketChannel.cpp#L2673
84
+ const permessageDeflate = 'permessage-deflate; client_max_window_bits'
85
+
86
+ // 10. Append (`Sec-WebSocket-Extensions`, permessageDeflate) to
87
+ // request’s header list.
88
+ request.headersList.append('sec-websocket-extensions', permessageDeflate, true)
89
+
90
+ // 11. Fetch request with useParallelQueue set to true, and
91
+ // processResponse given response being these steps:
92
+ const controller = fetching({
93
+ request,
94
+ useParallelQueue: true,
95
+ dispatcher: options.dispatcher,
96
+ processResponse (response) {
97
+ // 1. If response is a network error or its status is not 101,
98
+ // fail the WebSocket connection.
99
+ // if (response.type === 'error' || ((response.socket?.session != null && response.status !== 200) && response.status !== 101)) {
100
+ if (response.type === 'error' || response.status !== 101) {
101
+ // The presence of a session property on the socket indicates HTTP2
102
+ // HTTP1
103
+ if (response.socket?.session == null) {
104
+ failWebsocketConnection(handler, 1002, 'Received network error or non-101 status code.', response.error)
105
+ return
106
+ }
107
+
108
+ // HTTP2
109
+ if (response.status !== 200) {
110
+ failWebsocketConnection(handler, 1002, 'Received network error or non-200 status code.', response.error)
111
+ return
112
+ }
113
+ }
114
+
115
+ if (warningEmitted === false && response.socket?.session != null) {
116
+ process.emitWarning('WebSocket over HTTP2 is experimental, and subject to change.', 'ExperimentalWarning')
117
+ warningEmitted = true
118
+ }
119
+
120
+ // 2. If protocols is not the empty list and extracting header
121
+ // list values given `Sec-WebSocket-Protocol` and response’s
122
+ // header list results in null, failure, or the empty byte
123
+ // sequence, then fail the WebSocket connection.
124
+ if (protocols.length !== 0 && !response.headersList.get('Sec-WebSocket-Protocol')) {
125
+ failWebsocketConnection(handler, 1002, 'Server did not respond with sent protocols.')
126
+ return
127
+ }
128
+
129
+ // 3. Follow the requirements stated step 2 to step 6, inclusive,
130
+ // of the last set of steps in section 4.1 of The WebSocket
131
+ // Protocol to validate response. This either results in fail
132
+ // the WebSocket connection or the WebSocket connection is
133
+ // established.
134
+
135
+ // 2. If the response lacks an |Upgrade| header field or the |Upgrade|
136
+ // header field contains a value that is not an ASCII case-
137
+ // insensitive match for the value "websocket", the client MUST
138
+ // _Fail the WebSocket Connection_.
139
+ // For H2, no upgrade header is expected.
140
+ if (response.socket.session == null && response.headersList.get('Upgrade')?.toLowerCase() !== 'websocket') {
141
+ failWebsocketConnection(handler, 1002, 'Server did not set Upgrade header to "websocket".')
142
+ return
143
+ }
144
+
145
+ // 3. If the response lacks a |Connection| header field or the
146
+ // |Connection| header field doesn't contain a token that is an
147
+ // ASCII case-insensitive match for the value "Upgrade", the client
148
+ // MUST _Fail the WebSocket Connection_.
149
+ // For H2, no connection header is expected.
150
+ if (response.socket.session == null && response.headersList.get('Connection')?.toLowerCase() !== 'upgrade') {
151
+ failWebsocketConnection(handler, 1002, 'Server did not set Connection header to "upgrade".')
152
+ return
153
+ }
154
+
155
+ // 4. If the response lacks a |Sec-WebSocket-Accept| header field or
156
+ // the |Sec-WebSocket-Accept| contains a value other than the
157
+ // base64-encoded SHA-1 of the concatenation of the |Sec-WebSocket-
158
+ // Key| (as a string, not base64-decoded) with the string "258EAFA5-
159
+ // E914-47DA-95CA-C5AB0DC85B11" but ignoring any leading and
160
+ // trailing whitespace, the client MUST _Fail the WebSocket
161
+ // Connection_.
162
+ const secWSAccept = response.headersList.get('Sec-WebSocket-Accept')
163
+ const digest = crypto.hash('sha1', keyValue + uid, 'base64')
164
+ if (secWSAccept !== digest) {
165
+ failWebsocketConnection(handler, 1002, 'Incorrect hash received in Sec-WebSocket-Accept header.')
166
+ return
167
+ }
168
+
169
+ // 5. If the response includes a |Sec-WebSocket-Extensions| header
170
+ // field and this header field indicates the use of an extension
171
+ // that was not present in the client's handshake (the server has
172
+ // indicated an extension not requested by the client), the client
173
+ // MUST _Fail the WebSocket Connection_. (The parsing of this
174
+ // header field to determine which extensions are requested is
175
+ // discussed in Section 9.1.)
176
+ const secExtension = response.headersList.get('Sec-WebSocket-Extensions')
177
+ let extensions
178
+
179
+ if (secExtension !== null) {
180
+ extensions = parseExtensions(secExtension)
181
+
182
+ if (!extensions.has('permessage-deflate')) {
183
+ failWebsocketConnection(handler, 1002, 'Sec-WebSocket-Extensions header does not match.')
184
+ return
185
+ }
186
+ }
187
+
188
+ // 6. If the response includes a |Sec-WebSocket-Protocol| header field
189
+ // and this header field indicates the use of a subprotocol that was
190
+ // not present in the client's handshake (the server has indicated a
191
+ // subprotocol not requested by the client), the client MUST _Fail
192
+ // the WebSocket Connection_.
193
+ const secProtocol = response.headersList.get('Sec-WebSocket-Protocol')
194
+
195
+ if (secProtocol !== null) {
196
+ const requestProtocols = getDecodeSplit('sec-websocket-protocol', request.headersList)
197
+
198
+ // The client can request that the server use a specific subprotocol by
199
+ // including the |Sec-WebSocket-Protocol| field in its handshake. If it
200
+ // is specified, the server needs to include the same field and one of
201
+ // the selected subprotocol values in its response for the connection to
202
+ // be established.
203
+ if (!requestProtocols.includes(secProtocol)) {
204
+ failWebsocketConnection(handler, 1002, 'Protocol was not set in the opening handshake.')
205
+ return
206
+ }
207
+ }
208
+
209
+ response.socket.on('data', handler.onSocketData)
210
+ response.socket.on('close', handler.onSocketClose)
211
+ response.socket.on('error', handler.onSocketError)
212
+
213
+ handler.wasEverConnected = true
214
+ handler.onConnectionEstablished(response, extensions)
215
+ }
216
+ })
217
+
218
+ return controller
219
+ }
220
+
221
+ /**
222
+ * @see https://whatpr.org/websockets/48.html#close-the-websocket
223
+ * @param {import('./websocket').Handler} object
224
+ * @param {number} [code=null]
225
+ * @param {string} [reason='']
226
+ */
227
+ function closeWebSocketConnection (object, code, reason, validate = false) {
228
+ // 1. If code was not supplied, let code be null.
229
+ code ??= null
230
+
231
+ // 2. If reason was not supplied, let reason be the empty string.
232
+ reason ??= ''
233
+
234
+ // 3. Validate close code and reason with code and reason.
235
+ if (validate) validateCloseCodeAndReason(code, reason)
236
+
237
+ // 4. Run the first matching steps from the following list:
238
+ // - If object’s ready state is CLOSING (2) or CLOSED (3)
239
+ // - If the WebSocket connection is not yet established [WSP]
240
+ // - If the WebSocket closing handshake has not yet been started [WSP]
241
+ // - Otherwise
242
+ if (isClosed(object.readyState) || isClosing(object.readyState)) {
243
+ // Do nothing.
244
+ } else if (!isEstablished(object.readyState)) {
245
+ // Fail the WebSocket connection and set object’s ready state to CLOSING (2). [WSP]
246
+ failWebsocketConnection(object)
247
+ object.readyState = states.CLOSING
248
+ } else if (!object.closeState.has(sentCloseFrameState.SENT) && !object.closeState.has(sentCloseFrameState.RECEIVED)) {
249
+ // Upon either sending or receiving a Close control frame, it is said
250
+ // that _The WebSocket Closing Handshake is Started_ and that the
251
+ // WebSocket connection is in the CLOSING state.
252
+
253
+ const frame = new WebsocketFrameSend()
254
+
255
+ // If neither code nor reason is present, the WebSocket Close
256
+ // message must not have a body.
257
+
258
+ // If code is present, then the status code to use in the
259
+ // WebSocket Close message must be the integer given by code.
260
+ // If code is null and reason is the empty string, the WebSocket Close frame must not have a body.
261
+ // If reason is non-empty but code is null, then set code to 1000 ("Normal Closure").
262
+ if (reason.length !== 0 && code === null) {
263
+ code = 1000
264
+ }
265
+
266
+ // If code is set, then the status code to use in the WebSocket Close frame must be the integer given by code.
267
+ assert(code === null || Number.isInteger(code))
268
+
269
+ if (code === null && reason.length === 0) {
270
+ frame.frameData = emptyBuffer
271
+ } else if (code !== null && reason === null) {
272
+ frame.frameData = Buffer.allocUnsafe(2)
273
+ frame.frameData.writeUInt16BE(code, 0)
274
+ } else if (code !== null && reason !== null) {
275
+ // If reason is also present, then reasonBytes must be
276
+ // provided in the Close message after the status code.
277
+ frame.frameData = Buffer.allocUnsafe(2 + Buffer.byteLength(reason))
278
+ frame.frameData.writeUInt16BE(code, 0)
279
+ // the body MAY contain UTF-8-encoded data with value /reason/
280
+ frame.frameData.write(reason, 2, 'utf-8')
281
+ } else {
282
+ frame.frameData = emptyBuffer
283
+ }
284
+
285
+ object.socket.write(frame.createFrame(opcodes.CLOSE))
286
+
287
+ object.closeState.add(sentCloseFrameState.SENT)
288
+
289
+ // Upon either sending or receiving a Close control frame, it is said
290
+ // that _The WebSocket Closing Handshake is Started_ and that the
291
+ // WebSocket connection is in the CLOSING state.
292
+ object.readyState = states.CLOSING
293
+ } else {
294
+ // Set object’s ready state to CLOSING (2).
295
+ object.readyState = states.CLOSING
296
+ }
297
+ }
298
+
299
+ /**
300
+ * @param {import('./websocket').Handler} handler
301
+ * @param {number} code
302
+ * @param {string|undefined} reason
303
+ * @param {unknown} cause
304
+ * @returns {void}
305
+ */
306
+ function failWebsocketConnection (handler, code, reason, cause) {
307
+ // If _The WebSocket Connection is Established_ prior to the point where
308
+ // the endpoint is required to _Fail the WebSocket Connection_, the
309
+ // endpoint SHOULD send a Close frame with an appropriate status code
310
+ // (Section 7.4) before proceeding to _Close the WebSocket Connection_.
311
+ if (isEstablished(handler.readyState)) {
312
+ closeWebSocketConnection(handler, code, reason, false)
313
+ }
314
+
315
+ handler.controller.abort()
316
+
317
+ if (isConnecting(handler.readyState)) {
318
+ // If the connection was not established, we must still emit an 'error' and 'close' events
319
+ handler.onSocketClose()
320
+ } else if (handler.socket?.destroyed === false) {
321
+ handler.socket.destroy()
322
+ }
323
+ }
324
+
325
+ module.exports = {
326
+ establishWebSocketConnection,
327
+ failWebsocketConnection,
328
+ closeWebSocketConnection
329
+ }
@@ -0,0 +1,126 @@
1
+ 'use strict'
2
+
3
+ /**
4
+ * This is a Globally Unique Identifier unique used to validate that the
5
+ * endpoint accepts websocket connections.
6
+ * @see https://www.rfc-editor.org/rfc/rfc6455.html#section-1.3
7
+ * @type {'258EAFA5-E914-47DA-95CA-C5AB0DC85B11'}
8
+ */
9
+ const uid = '258EAFA5-E914-47DA-95CA-C5AB0DC85B11'
10
+
11
+ /**
12
+ * @type {PropertyDescriptor}
13
+ */
14
+ const staticPropertyDescriptors = {
15
+ enumerable: true,
16
+ writable: false,
17
+ configurable: false
18
+ }
19
+
20
+ /**
21
+ * The states of the WebSocket connection.
22
+ *
23
+ * @readonly
24
+ * @enum
25
+ * @property {0} CONNECTING
26
+ * @property {1} OPEN
27
+ * @property {2} CLOSING
28
+ * @property {3} CLOSED
29
+ */
30
+ const states = {
31
+ CONNECTING: 0,
32
+ OPEN: 1,
33
+ CLOSING: 2,
34
+ CLOSED: 3
35
+ }
36
+
37
+ /**
38
+ * @readonly
39
+ * @enum
40
+ * @property {0} NOT_SENT
41
+ * @property {1} PROCESSING
42
+ * @property {2} SENT
43
+ */
44
+ const sentCloseFrameState = {
45
+ SENT: 1,
46
+ RECEIVED: 2
47
+ }
48
+
49
+ /**
50
+ * The WebSocket opcodes.
51
+ *
52
+ * @readonly
53
+ * @enum
54
+ * @property {0x0} CONTINUATION
55
+ * @property {0x1} TEXT
56
+ * @property {0x2} BINARY
57
+ * @property {0x8} CLOSE
58
+ * @property {0x9} PING
59
+ * @property {0xA} PONG
60
+ * @see https://datatracker.ietf.org/doc/html/rfc6455#section-5.2
61
+ */
62
+ const opcodes = {
63
+ CONTINUATION: 0x0,
64
+ TEXT: 0x1,
65
+ BINARY: 0x2,
66
+ CLOSE: 0x8,
67
+ PING: 0x9,
68
+ PONG: 0xA
69
+ }
70
+
71
+ /**
72
+ * The maximum value for an unsigned 16-bit integer.
73
+ *
74
+ * @type {65535} 2 ** 16 - 1
75
+ */
76
+ const maxUnsigned16Bit = 65535
77
+
78
+ /**
79
+ * The states of the parser.
80
+ *
81
+ * @readonly
82
+ * @enum
83
+ * @property {0} INFO
84
+ * @property {2} PAYLOADLENGTH_16
85
+ * @property {3} PAYLOADLENGTH_64
86
+ * @property {4} READ_DATA
87
+ */
88
+ const parserStates = {
89
+ INFO: 0,
90
+ PAYLOADLENGTH_16: 2,
91
+ PAYLOADLENGTH_64: 3,
92
+ READ_DATA: 4
93
+ }
94
+
95
+ /**
96
+ * An empty buffer.
97
+ *
98
+ * @type {Buffer}
99
+ */
100
+ const emptyBuffer = Buffer.allocUnsafe(0)
101
+
102
+ /**
103
+ * @readonly
104
+ * @property {1} text
105
+ * @property {2} typedArray
106
+ * @property {3} arrayBuffer
107
+ * @property {4} blob
108
+ */
109
+ const sendHints = {
110
+ text: 1,
111
+ typedArray: 2,
112
+ arrayBuffer: 3,
113
+ blob: 4
114
+ }
115
+
116
+ module.exports = {
117
+ uid,
118
+ sentCloseFrameState,
119
+ staticPropertyDescriptors,
120
+ states,
121
+ opcodes,
122
+ maxUnsigned16Bit,
123
+ parserStates,
124
+ emptyBuffer,
125
+ sendHints
126
+ }