@depup/mswjs__interceptors 0.41.3-depup.0

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 (248) hide show
  1. package/ClientRequest/package.json +11 -0
  2. package/LICENSE.md +9 -0
  3. package/README.md +31 -0
  4. package/RemoteHttpInterceptor/package.json +11 -0
  5. package/WebSocket/package.json +12 -0
  6. package/XMLHttpRequest/package.json +12 -0
  7. package/changes.json +10 -0
  8. package/fetch/package.json +12 -0
  9. package/lib/browser/Interceptor-Deczogc8.d.cts +65 -0
  10. package/lib/browser/Interceptor-gqKgs-aF.d.mts +65 -0
  11. package/lib/browser/XMLHttpRequest-BACqefB-.cjs +761 -0
  12. package/lib/browser/XMLHttpRequest-BACqefB-.cjs.map +1 -0
  13. package/lib/browser/XMLHttpRequest-BvxZV0WU.mjs +756 -0
  14. package/lib/browser/XMLHttpRequest-BvxZV0WU.mjs.map +1 -0
  15. package/lib/browser/bufferUtils-BiiO6HZv.mjs +20 -0
  16. package/lib/browser/bufferUtils-BiiO6HZv.mjs.map +1 -0
  17. package/lib/browser/bufferUtils-Uc0eRItL.cjs +38 -0
  18. package/lib/browser/bufferUtils-Uc0eRItL.cjs.map +1 -0
  19. package/lib/browser/createRequestId-Cs4oXfa1.cjs +205 -0
  20. package/lib/browser/createRequestId-Cs4oXfa1.cjs.map +1 -0
  21. package/lib/browser/createRequestId-DQcIlohW.mjs +170 -0
  22. package/lib/browser/createRequestId-DQcIlohW.mjs.map +1 -0
  23. package/lib/browser/fetch-DdKEdDOR.mjs +248 -0
  24. package/lib/browser/fetch-DdKEdDOR.mjs.map +1 -0
  25. package/lib/browser/fetch-U3v3Y4ap.cjs +253 -0
  26. package/lib/browser/fetch-U3v3Y4ap.cjs.map +1 -0
  27. package/lib/browser/getRawRequest-BTaNLFr0.mjs +218 -0
  28. package/lib/browser/getRawRequest-BTaNLFr0.mjs.map +1 -0
  29. package/lib/browser/getRawRequest-zx8rUJL2.cjs +259 -0
  30. package/lib/browser/getRawRequest-zx8rUJL2.cjs.map +1 -0
  31. package/lib/browser/glossary-BdLS4k1H.d.cts +70 -0
  32. package/lib/browser/glossary-DYwOrogs.d.mts +70 -0
  33. package/lib/browser/handleRequest-CvX2G-Lz.cjs +189 -0
  34. package/lib/browser/handleRequest-CvX2G-Lz.cjs.map +1 -0
  35. package/lib/browser/handleRequest-D7kpTI5U.mjs +178 -0
  36. package/lib/browser/handleRequest-D7kpTI5U.mjs.map +1 -0
  37. package/lib/browser/hasConfigurableGlobal-BvCTG97d.cjs +45 -0
  38. package/lib/browser/hasConfigurableGlobal-BvCTG97d.cjs.map +1 -0
  39. package/lib/browser/hasConfigurableGlobal-npXitu1-.mjs +33 -0
  40. package/lib/browser/hasConfigurableGlobal-npXitu1-.mjs.map +1 -0
  41. package/lib/browser/index.cjs +70 -0
  42. package/lib/browser/index.cjs.map +1 -0
  43. package/lib/browser/index.d.cts +96 -0
  44. package/lib/browser/index.d.mts +96 -0
  45. package/lib/browser/index.mjs +56 -0
  46. package/lib/browser/index.mjs.map +1 -0
  47. package/lib/browser/interceptors/WebSocket/index.cjs +622 -0
  48. package/lib/browser/interceptors/WebSocket/index.cjs.map +1 -0
  49. package/lib/browser/interceptors/WebSocket/index.d.cts +277 -0
  50. package/lib/browser/interceptors/WebSocket/index.d.mts +277 -0
  51. package/lib/browser/interceptors/WebSocket/index.mjs +615 -0
  52. package/lib/browser/interceptors/WebSocket/index.mjs.map +1 -0
  53. package/lib/browser/interceptors/XMLHttpRequest/index.cjs +7 -0
  54. package/lib/browser/interceptors/XMLHttpRequest/index.d.cts +15 -0
  55. package/lib/browser/interceptors/XMLHttpRequest/index.d.mts +15 -0
  56. package/lib/browser/interceptors/XMLHttpRequest/index.mjs +7 -0
  57. package/lib/browser/interceptors/fetch/index.cjs +6 -0
  58. package/lib/browser/interceptors/fetch/index.d.cts +13 -0
  59. package/lib/browser/interceptors/fetch/index.d.mts +13 -0
  60. package/lib/browser/interceptors/fetch/index.mjs +6 -0
  61. package/lib/browser/presets/browser.cjs +17 -0
  62. package/lib/browser/presets/browser.cjs.map +1 -0
  63. package/lib/browser/presets/browser.d.cts +12 -0
  64. package/lib/browser/presets/browser.d.mts +14 -0
  65. package/lib/browser/presets/browser.mjs +17 -0
  66. package/lib/browser/presets/browser.mjs.map +1 -0
  67. package/lib/browser/resolveWebSocketUrl-6K6EgqsA.cjs +31 -0
  68. package/lib/browser/resolveWebSocketUrl-6K6EgqsA.cjs.map +1 -0
  69. package/lib/browser/resolveWebSocketUrl-C83-x9iE.mjs +25 -0
  70. package/lib/browser/resolveWebSocketUrl-C83-x9iE.mjs.map +1 -0
  71. package/lib/node/BatchInterceptor-3LnAnLTx.cjs +49 -0
  72. package/lib/node/BatchInterceptor-3LnAnLTx.cjs.map +1 -0
  73. package/lib/node/BatchInterceptor-D7mXzHcQ.d.mts +26 -0
  74. package/lib/node/BatchInterceptor-DFaBPilf.mjs +44 -0
  75. package/lib/node/BatchInterceptor-DFaBPilf.mjs.map +1 -0
  76. package/lib/node/BatchInterceptor-D_YqR8qU.d.cts +26 -0
  77. package/lib/node/ClientRequest-2rDe54Ui.cjs +1043 -0
  78. package/lib/node/ClientRequest-2rDe54Ui.cjs.map +1 -0
  79. package/lib/node/ClientRequest-Ca8Qykuv.mjs +1034 -0
  80. package/lib/node/ClientRequest-Ca8Qykuv.mjs.map +1 -0
  81. package/lib/node/Interceptor-DEazpLJd.d.mts +133 -0
  82. package/lib/node/Interceptor-DJ2akVWI.d.cts +133 -0
  83. package/lib/node/RemoteHttpInterceptor.cjs +154 -0
  84. package/lib/node/RemoteHttpInterceptor.cjs.map +1 -0
  85. package/lib/node/RemoteHttpInterceptor.d.cts +39 -0
  86. package/lib/node/RemoteHttpInterceptor.d.mts +39 -0
  87. package/lib/node/RemoteHttpInterceptor.mjs +152 -0
  88. package/lib/node/RemoteHttpInterceptor.mjs.map +1 -0
  89. package/lib/node/XMLHttpRequest-B7kJdYYI.cjs +763 -0
  90. package/lib/node/XMLHttpRequest-B7kJdYYI.cjs.map +1 -0
  91. package/lib/node/XMLHttpRequest-C8dIZpds.mjs +757 -0
  92. package/lib/node/XMLHttpRequest-C8dIZpds.mjs.map +1 -0
  93. package/lib/node/bufferUtils-DiCTqG-7.cjs +38 -0
  94. package/lib/node/bufferUtils-DiCTqG-7.cjs.map +1 -0
  95. package/lib/node/bufferUtils-_8XfKIfX.mjs +20 -0
  96. package/lib/node/bufferUtils-_8XfKIfX.mjs.map +1 -0
  97. package/lib/node/chunk-CbDLau6x.cjs +34 -0
  98. package/lib/node/fetch-BmXpK10r.cjs +272 -0
  99. package/lib/node/fetch-BmXpK10r.cjs.map +1 -0
  100. package/lib/node/fetch-G1DVwDKG.mjs +265 -0
  101. package/lib/node/fetch-G1DVwDKG.mjs.map +1 -0
  102. package/lib/node/fetchUtils-BaY5iWXw.cjs +419 -0
  103. package/lib/node/fetchUtils-BaY5iWXw.cjs.map +1 -0
  104. package/lib/node/fetchUtils-CoU35g3M.mjs +359 -0
  105. package/lib/node/fetchUtils-CoU35g3M.mjs.map +1 -0
  106. package/lib/node/getRawRequest-BavnMWh_.cjs +36 -0
  107. package/lib/node/getRawRequest-BavnMWh_.cjs.map +1 -0
  108. package/lib/node/getRawRequest-DnwmXyOW.mjs +24 -0
  109. package/lib/node/getRawRequest-DnwmXyOW.mjs.map +1 -0
  110. package/lib/node/glossary-BLKRyLBd.cjs +12 -0
  111. package/lib/node/glossary-BLKRyLBd.cjs.map +1 -0
  112. package/lib/node/glossary-glQBRnVD.mjs +6 -0
  113. package/lib/node/glossary-glQBRnVD.mjs.map +1 -0
  114. package/lib/node/handleRequest-Bb7Y-XLw.cjs +220 -0
  115. package/lib/node/handleRequest-Bb7Y-XLw.cjs.map +1 -0
  116. package/lib/node/handleRequest-Y97UwBbF.mjs +190 -0
  117. package/lib/node/handleRequest-Y97UwBbF.mjs.map +1 -0
  118. package/lib/node/hasConfigurableGlobal-C97fWuaA.cjs +26 -0
  119. package/lib/node/hasConfigurableGlobal-C97fWuaA.cjs.map +1 -0
  120. package/lib/node/hasConfigurableGlobal-DBJA0vjm.mjs +20 -0
  121. package/lib/node/hasConfigurableGlobal-DBJA0vjm.mjs.map +1 -0
  122. package/lib/node/index-BMbJ8FXL.d.cts +113 -0
  123. package/lib/node/index-C0YAQ36w.d.mts +113 -0
  124. package/lib/node/index.cjs +54 -0
  125. package/lib/node/index.cjs.map +1 -0
  126. package/lib/node/index.d.cts +75 -0
  127. package/lib/node/index.d.mts +75 -0
  128. package/lib/node/index.mjs +40 -0
  129. package/lib/node/index.mjs.map +1 -0
  130. package/lib/node/interceptors/ClientRequest/index.cjs +6 -0
  131. package/lib/node/interceptors/ClientRequest/index.d.cts +2 -0
  132. package/lib/node/interceptors/ClientRequest/index.d.mts +3 -0
  133. package/lib/node/interceptors/ClientRequest/index.mjs +6 -0
  134. package/lib/node/interceptors/XMLHttpRequest/index.cjs +6 -0
  135. package/lib/node/interceptors/XMLHttpRequest/index.d.cts +14 -0
  136. package/lib/node/interceptors/XMLHttpRequest/index.d.mts +14 -0
  137. package/lib/node/interceptors/XMLHttpRequest/index.mjs +6 -0
  138. package/lib/node/interceptors/fetch/index.cjs +5 -0
  139. package/lib/node/interceptors/fetch/index.d.cts +12 -0
  140. package/lib/node/interceptors/fetch/index.d.mts +12 -0
  141. package/lib/node/interceptors/fetch/index.mjs +5 -0
  142. package/lib/node/node-DwCc6iuP.mjs +27 -0
  143. package/lib/node/node-DwCc6iuP.mjs.map +1 -0
  144. package/lib/node/node-dKdAf3tC.cjs +39 -0
  145. package/lib/node/node-dKdAf3tC.cjs.map +1 -0
  146. package/lib/node/presets/node.cjs +22 -0
  147. package/lib/node/presets/node.cjs.map +1 -0
  148. package/lib/node/presets/node.d.cts +13 -0
  149. package/lib/node/presets/node.d.mts +15 -0
  150. package/lib/node/presets/node.mjs +22 -0
  151. package/lib/node/presets/node.mjs.map +1 -0
  152. package/lib/node/utils/node/index.cjs +4 -0
  153. package/lib/node/utils/node/index.d.cts +16 -0
  154. package/lib/node/utils/node/index.d.mts +16 -0
  155. package/lib/node/utils/node/index.mjs +3 -0
  156. package/package.json +204 -0
  157. package/presets/browser/package.json +5 -0
  158. package/presets/node/package.json +11 -0
  159. package/src/BatchInterceptor.test.ts +255 -0
  160. package/src/BatchInterceptor.ts +95 -0
  161. package/src/Interceptor.test.ts +205 -0
  162. package/src/Interceptor.ts +249 -0
  163. package/src/InterceptorError.ts +7 -0
  164. package/src/RemoteHttpInterceptor.ts +251 -0
  165. package/src/RequestController.test.ts +104 -0
  166. package/src/RequestController.ts +109 -0
  167. package/src/createRequestId.test.ts +7 -0
  168. package/src/createRequestId.ts +9 -0
  169. package/src/getRawRequest.ts +21 -0
  170. package/src/glossary.ts +37 -0
  171. package/src/index.ts +15 -0
  172. package/src/interceptors/ClientRequest/MockHttpSocket.ts +724 -0
  173. package/src/interceptors/ClientRequest/agents.ts +110 -0
  174. package/src/interceptors/ClientRequest/index.test.ts +75 -0
  175. package/src/interceptors/ClientRequest/index.ts +193 -0
  176. package/src/interceptors/ClientRequest/utils/getIncomingMessageBody.test.ts +54 -0
  177. package/src/interceptors/ClientRequest/utils/getIncomingMessageBody.ts +45 -0
  178. package/src/interceptors/ClientRequest/utils/normalizeClientRequestArgs.test.ts +427 -0
  179. package/src/interceptors/ClientRequest/utils/normalizeClientRequestArgs.ts +268 -0
  180. package/src/interceptors/ClientRequest/utils/parserUtils.ts +48 -0
  181. package/src/interceptors/ClientRequest/utils/recordRawHeaders.test.ts +258 -0
  182. package/src/interceptors/ClientRequest/utils/recordRawHeaders.ts +262 -0
  183. package/src/interceptors/Socket/MockSocket.test.ts +264 -0
  184. package/src/interceptors/Socket/MockSocket.ts +58 -0
  185. package/src/interceptors/Socket/utils/baseUrlFromConnectionOptions.ts +26 -0
  186. package/src/interceptors/Socket/utils/normalizeSocketWriteArgs.test.ts +52 -0
  187. package/src/interceptors/Socket/utils/normalizeSocketWriteArgs.ts +33 -0
  188. package/src/interceptors/WebSocket/WebSocketClassTransport.ts +116 -0
  189. package/src/interceptors/WebSocket/WebSocketClientConnection.ts +152 -0
  190. package/src/interceptors/WebSocket/WebSocketOverride.ts +252 -0
  191. package/src/interceptors/WebSocket/WebSocketServerConnection.ts +420 -0
  192. package/src/interceptors/WebSocket/WebSocketTransport.ts +39 -0
  193. package/src/interceptors/WebSocket/index.ts +191 -0
  194. package/src/interceptors/WebSocket/utils/bindEvent.test.ts +27 -0
  195. package/src/interceptors/WebSocket/utils/bindEvent.ts +21 -0
  196. package/src/interceptors/WebSocket/utils/events.test.ts +101 -0
  197. package/src/interceptors/WebSocket/utils/events.ts +94 -0
  198. package/src/interceptors/XMLHttpRequest/XMLHttpRequestController.ts +746 -0
  199. package/src/interceptors/XMLHttpRequest/XMLHttpRequestProxy.ts +121 -0
  200. package/src/interceptors/XMLHttpRequest/index.ts +61 -0
  201. package/src/interceptors/XMLHttpRequest/polyfills/EventPolyfill.ts +51 -0
  202. package/src/interceptors/XMLHttpRequest/polyfills/ProgressEventPolyfill.ts +17 -0
  203. package/src/interceptors/XMLHttpRequest/utils/concatArrayBuffer.ts +12 -0
  204. package/src/interceptors/XMLHttpRequest/utils/concateArrayBuffer.test.ts +12 -0
  205. package/src/interceptors/XMLHttpRequest/utils/createEvent.test.ts +26 -0
  206. package/src/interceptors/XMLHttpRequest/utils/createEvent.ts +41 -0
  207. package/src/interceptors/XMLHttpRequest/utils/createResponse.ts +49 -0
  208. package/src/interceptors/XMLHttpRequest/utils/getBodyByteLength.test.ts +164 -0
  209. package/src/interceptors/XMLHttpRequest/utils/getBodyByteLength.ts +16 -0
  210. package/src/interceptors/XMLHttpRequest/utils/isDomParserSupportedType.ts +14 -0
  211. package/src/interceptors/fetch/index.ts +214 -0
  212. package/src/interceptors/fetch/utils/brotli-decompress.browser.ts +14 -0
  213. package/src/interceptors/fetch/utils/brotli-decompress.ts +31 -0
  214. package/src/interceptors/fetch/utils/createNetworkError.ts +5 -0
  215. package/src/interceptors/fetch/utils/decompression.ts +85 -0
  216. package/src/interceptors/fetch/utils/followRedirect.ts +114 -0
  217. package/src/presets/browser.ts +11 -0
  218. package/src/presets/node.ts +13 -0
  219. package/src/utils/bufferUtils.test.ts +21 -0
  220. package/src/utils/bufferUtils.ts +22 -0
  221. package/src/utils/canParseUrl.ts +13 -0
  222. package/src/utils/cloneObject.test.ts +94 -0
  223. package/src/utils/cloneObject.ts +36 -0
  224. package/src/utils/createProxy.test.ts +164 -0
  225. package/src/utils/createProxy.ts +104 -0
  226. package/src/utils/emitAsync.ts +25 -0
  227. package/src/utils/fetchUtils.ts +119 -0
  228. package/src/utils/findPropertySource.test.ts +27 -0
  229. package/src/utils/findPropertySource.ts +20 -0
  230. package/src/utils/getCleanUrl.test.ts +32 -0
  231. package/src/utils/getCleanUrl.ts +6 -0
  232. package/src/utils/getUrlByRequestOptions.test.ts +163 -0
  233. package/src/utils/getUrlByRequestOptions.ts +152 -0
  234. package/src/utils/getValueBySymbol.test.ts +14 -0
  235. package/src/utils/getValueBySymbol.ts +19 -0
  236. package/src/utils/handleRequest.ts +205 -0
  237. package/src/utils/hasConfigurableGlobal.test.ts +83 -0
  238. package/src/utils/hasConfigurableGlobal.ts +34 -0
  239. package/src/utils/isNodeLikeError.ts +13 -0
  240. package/src/utils/isObject.test.ts +21 -0
  241. package/src/utils/isObject.ts +8 -0
  242. package/src/utils/isPropertyAccessible.ts +19 -0
  243. package/src/utils/nextTick.ts +11 -0
  244. package/src/utils/node/index.ts +39 -0
  245. package/src/utils/parseJson.test.ts +10 -0
  246. package/src/utils/parseJson.ts +12 -0
  247. package/src/utils/resolveWebSocketUrl.ts +45 -0
  248. package/src/utils/responseUtils.ts +59 -0
@@ -0,0 +1,252 @@
1
+ import { invariant } from 'outvariant'
2
+ import { DeferredPromise } from '@open-draft/deferred-promise'
3
+ import type { WebSocketData } from './WebSocketTransport'
4
+ import { bindEvent } from './utils/bindEvent'
5
+ import { CloseEvent } from './utils/events'
6
+ import { resolveWebSocketUrl } from '../../utils/resolveWebSocketUrl'
7
+
8
+ export type WebSocketEventListener<
9
+ EventType extends WebSocketEventMap[keyof WebSocketEventMap] = Event,
10
+ > = (this: WebSocket, event: EventType) => void
11
+
12
+ const WEBSOCKET_CLOSE_CODE_RANGE_ERROR =
13
+ 'InvalidAccessError: close code out of user configurable range'
14
+
15
+ export const kPassthroughPromise = Symbol('kPassthroughPromise')
16
+ export const kOnSend = Symbol('kOnSend')
17
+ export const kClose = Symbol('kClose')
18
+
19
+ export class WebSocketOverride extends EventTarget implements WebSocket {
20
+ static readonly CONNECTING = 0
21
+ static readonly OPEN = 1
22
+ static readonly CLOSING = 2
23
+ static readonly CLOSED = 3
24
+ readonly CONNECTING = 0
25
+ readonly OPEN = 1
26
+ readonly CLOSING = 2
27
+ readonly CLOSED = 3
28
+
29
+ public url: string
30
+ public protocol: string
31
+ public extensions: string
32
+ public binaryType: BinaryType
33
+ public readyState: WebSocket['readyState']
34
+ public bufferedAmount: number
35
+
36
+ private _onopen: WebSocketEventListener | null = null
37
+ private _onmessage: WebSocketEventListener<
38
+ MessageEvent<WebSocketData>
39
+ > | null = null
40
+ private _onerror: WebSocketEventListener | null = null
41
+ private _onclose: WebSocketEventListener<CloseEvent> | null = null
42
+
43
+ private [kPassthroughPromise]: DeferredPromise<boolean>
44
+ private [kOnSend]?: (data: WebSocketData) => void
45
+
46
+ constructor(url: string | URL, protocols?: string | Array<string>) {
47
+ super()
48
+ this.url = resolveWebSocketUrl(url)
49
+ this.protocol = ''
50
+ this.extensions = ''
51
+ this.binaryType = 'blob'
52
+ this.readyState = this.CONNECTING
53
+ this.bufferedAmount = 0
54
+
55
+ this[kPassthroughPromise] = new DeferredPromise<boolean>()
56
+
57
+ queueMicrotask(async () => {
58
+ if (await this[kPassthroughPromise]) {
59
+ return
60
+ }
61
+
62
+ this.protocol =
63
+ typeof protocols === 'string'
64
+ ? protocols
65
+ : Array.isArray(protocols) && protocols.length > 0
66
+ ? protocols[0]
67
+ : ''
68
+
69
+ /**
70
+ * @note Check that nothing has prevented this connection
71
+ * (e.g. called `client.close()` in the connection listener).
72
+ * If the connection has been prevented, never dispatch the open event,.
73
+ */
74
+ if (this.readyState === this.CONNECTING) {
75
+ this.readyState = this.OPEN
76
+ this.dispatchEvent(bindEvent(this, new Event('open')))
77
+ }
78
+ })
79
+ }
80
+
81
+ set onopen(listener: WebSocketEventListener | null) {
82
+ this.removeEventListener('open', this._onopen)
83
+ this._onopen = listener
84
+ if (listener !== null) {
85
+ this.addEventListener('open', listener)
86
+ }
87
+ }
88
+ get onopen(): WebSocketEventListener | null {
89
+ return this._onopen
90
+ }
91
+
92
+ set onmessage(
93
+ listener: WebSocketEventListener<MessageEvent<WebSocketData>> | null
94
+ ) {
95
+ this.removeEventListener(
96
+ 'message',
97
+ this._onmessage as WebSocketEventListener
98
+ )
99
+ this._onmessage = listener
100
+ if (listener !== null) {
101
+ this.addEventListener('message', listener)
102
+ }
103
+ }
104
+ get onmessage(): WebSocketEventListener<MessageEvent<WebSocketData>> | null {
105
+ return this._onmessage
106
+ }
107
+
108
+ set onerror(listener: WebSocketEventListener | null) {
109
+ this.removeEventListener('error', this._onerror)
110
+ this._onerror = listener
111
+ if (listener !== null) {
112
+ this.addEventListener('error', listener)
113
+ }
114
+ }
115
+ get onerror(): WebSocketEventListener | null {
116
+ return this._onerror
117
+ }
118
+
119
+ set onclose(listener: WebSocketEventListener<CloseEvent> | null) {
120
+ this.removeEventListener('close', this._onclose as WebSocketEventListener)
121
+ this._onclose = listener
122
+ if (listener !== null) {
123
+ this.addEventListener('close', listener)
124
+ }
125
+ }
126
+ get onclose(): WebSocketEventListener<CloseEvent> | null {
127
+ return this._onclose
128
+ }
129
+
130
+ /**
131
+ * @see https://websockets.spec.whatwg.org/#ref-for-dom-websocket-send%E2%91%A0
132
+ */
133
+ public send(data: WebSocketData): void {
134
+ if (this.readyState === this.CONNECTING) {
135
+ this.close()
136
+ throw new DOMException('InvalidStateError')
137
+ }
138
+
139
+ // Sending when the socket is about to close
140
+ // discards the sent data.
141
+ if (this.readyState === this.CLOSING || this.readyState === this.CLOSED) {
142
+ return
143
+ }
144
+
145
+ // Buffer the data to send in this even loop
146
+ // but send it in the next.
147
+ this.bufferedAmount += getDataSize(data)
148
+
149
+ queueMicrotask(() => {
150
+ // This is a bit optimistic but since no actual data transfer
151
+ // is involved, all the data will be "sent" on the next tick.
152
+ this.bufferedAmount = 0
153
+
154
+ /**
155
+ * @note Notify the parent about outgoing data.
156
+ * This notifies the transport and the connection
157
+ * listens to the outgoing data to emit the "message" event.
158
+ */
159
+ this[kOnSend]?.(data)
160
+ })
161
+ }
162
+
163
+ public close(code: number = 1000, reason?: string): void {
164
+ invariant(code, WEBSOCKET_CLOSE_CODE_RANGE_ERROR)
165
+ invariant(
166
+ code === 1000 || (code >= 3000 && code <= 4999),
167
+ WEBSOCKET_CLOSE_CODE_RANGE_ERROR
168
+ )
169
+
170
+ this[kClose](code, reason)
171
+ }
172
+
173
+ private [kClose](
174
+ code: number = 1000,
175
+ reason?: string,
176
+ wasClean = true
177
+ ): void {
178
+ /**
179
+ * @note Move this check here so that even internal closures,
180
+ * like those triggered by the `server` connection, are not
181
+ * performed twice.
182
+ */
183
+ if (this.readyState === this.CLOSING || this.readyState === this.CLOSED) {
184
+ return
185
+ }
186
+
187
+ this.readyState = this.CLOSING
188
+
189
+ queueMicrotask(() => {
190
+ this.readyState = this.CLOSED
191
+
192
+ this.dispatchEvent(
193
+ bindEvent(
194
+ this,
195
+ new CloseEvent('close', {
196
+ code,
197
+ reason,
198
+ wasClean,
199
+ })
200
+ )
201
+ )
202
+
203
+ // Remove all event listeners once the socket is closed.
204
+ this._onopen = null
205
+ this._onmessage = null
206
+ this._onerror = null
207
+ this._onclose = null
208
+ })
209
+ }
210
+
211
+ public addEventListener<K extends keyof WebSocketEventMap>(
212
+ type: K,
213
+ listener: (this: WebSocket, event: WebSocketEventMap[K]) => void,
214
+ options?: boolean | AddEventListenerOptions
215
+ ): void
216
+ public addEventListener(
217
+ type: string,
218
+ listener: EventListenerOrEventListenerObject,
219
+ options?: boolean | AddEventListenerOptions
220
+ ): void
221
+ public addEventListener(
222
+ type: unknown,
223
+ listener: unknown,
224
+ options?: unknown
225
+ ): void {
226
+ return super.addEventListener(
227
+ type as string,
228
+ listener as EventListener,
229
+ options as AddEventListenerOptions
230
+ )
231
+ }
232
+
233
+ removeEventListener<K extends keyof WebSocketEventMap>(
234
+ type: K,
235
+ callback: EventListenerOrEventListenerObject | null,
236
+ options?: boolean | EventListenerOptions
237
+ ): void {
238
+ return super.removeEventListener(type, callback, options)
239
+ }
240
+ }
241
+
242
+ function getDataSize(data: WebSocketData): number {
243
+ if (typeof data === 'string') {
244
+ return data.length
245
+ }
246
+
247
+ if (data instanceof Blob) {
248
+ return data.size
249
+ }
250
+
251
+ return data.byteLength
252
+ }
@@ -0,0 +1,420 @@
1
+ import { invariant } from 'outvariant'
2
+ import {
3
+ kClose,
4
+ WebSocketEventListener,
5
+ WebSocketOverride,
6
+ } from './WebSocketOverride'
7
+ import type { WebSocketData } from './WebSocketTransport'
8
+ import type { WebSocketClassTransport } from './WebSocketClassTransport'
9
+ import { bindEvent } from './utils/bindEvent'
10
+ import {
11
+ CancelableMessageEvent,
12
+ CancelableCloseEvent,
13
+ CloseEvent,
14
+ } from './utils/events'
15
+
16
+ const kEmitter = Symbol('kEmitter')
17
+ const kBoundListener = Symbol('kBoundListener')
18
+ const kSend = Symbol('kSend')
19
+
20
+ export interface WebSocketServerEventMap {
21
+ open: Event
22
+ message: MessageEvent<WebSocketData>
23
+ error: Event
24
+ close: CloseEvent
25
+ }
26
+
27
+ export abstract class WebSocketServerConnectionProtocol {
28
+ public abstract connect(): void
29
+ public abstract send(data: WebSocketData): void
30
+ public abstract close(): void
31
+
32
+ public abstract addEventListener<
33
+ EventType extends keyof WebSocketServerEventMap,
34
+ >(
35
+ event: EventType,
36
+ listener: WebSocketEventListener<WebSocketServerEventMap[EventType]>,
37
+ options?: AddEventListenerOptions | boolean
38
+ ): void
39
+
40
+ public abstract removeEventListener<
41
+ EventType extends keyof WebSocketServerEventMap,
42
+ >(
43
+ event: EventType,
44
+ listener: WebSocketEventListener<WebSocketServerEventMap[EventType]>,
45
+ options?: EventListenerOptions | boolean
46
+ ): void
47
+ }
48
+
49
+ /**
50
+ * The WebSocket server instance represents the actual production
51
+ * WebSocket server connection. It's idle by default but you can
52
+ * establish it by calling `server.connect()`.
53
+ */
54
+ export class WebSocketServerConnection implements WebSocketServerConnectionProtocol {
55
+ /**
56
+ * A WebSocket instance connected to the original server.
57
+ */
58
+ private realWebSocket?: WebSocket
59
+ private mockCloseController: AbortController
60
+ private realCloseController: AbortController
61
+ private [kEmitter]: EventTarget
62
+
63
+ constructor(
64
+ private readonly client: WebSocketOverride,
65
+ private readonly transport: WebSocketClassTransport,
66
+ private readonly createConnection: () => WebSocket
67
+ ) {
68
+ this[kEmitter] = new EventTarget()
69
+ this.mockCloseController = new AbortController()
70
+ this.realCloseController = new AbortController()
71
+
72
+ // Automatically forward outgoing client events
73
+ // to the actual server unless the outgoing message event
74
+ // has been prevented. The "outgoing" transport event it
75
+ // dispatched by the "client" connection.
76
+ this.transport.addEventListener('outgoing', (event) => {
77
+ // Ignore client messages if the server connection
78
+ // hasn't been established yet. Nowhere to forward.
79
+ if (typeof this.realWebSocket === 'undefined') {
80
+ return
81
+ }
82
+
83
+ // Every outgoing client message can prevent this forwarding
84
+ // by preventing the default of the outgoing message event.
85
+ // This listener will be added before user-defined listeners,
86
+ // so execute the logic on the next tick.
87
+ queueMicrotask(() => {
88
+ if (!event.defaultPrevented) {
89
+ /**
90
+ * @note Use the internal send mechanism so consumers can tell
91
+ * apart direct user calls to `server.send()` and internal calls.
92
+ * E.g. MSW has to ignore this internal call to log out messages correctly.
93
+ */
94
+ this[kSend](event.data)
95
+ }
96
+ })
97
+ })
98
+
99
+ this.transport.addEventListener(
100
+ 'incoming',
101
+ this.handleIncomingMessage.bind(this)
102
+ )
103
+ }
104
+
105
+ /**
106
+ * The `WebSocket` instance connected to the original server.
107
+ * Accessing this before calling `server.connect()` will throw.
108
+ */
109
+ public get socket(): WebSocket {
110
+ invariant(
111
+ this.realWebSocket,
112
+ 'Cannot access "socket" on the original WebSocket server object: the connection is not open. Did you forget to call `server.connect()`?'
113
+ )
114
+
115
+ return this.realWebSocket
116
+ }
117
+
118
+ /**
119
+ * Open connection to the original WebSocket server.
120
+ */
121
+ public connect(): void {
122
+ invariant(
123
+ !this.realWebSocket || this.realWebSocket.readyState !== WebSocket.OPEN,
124
+ 'Failed to call "connect()" on the original WebSocket instance: the connection already open'
125
+ )
126
+
127
+ const realWebSocket = this.createConnection()
128
+
129
+ // Inherit the binary type from the mock WebSocket client.
130
+ realWebSocket.binaryType = this.client.binaryType
131
+
132
+ // Allow the interceptor to listen to when the server connection
133
+ // has been established. This isn't necessary to operate with the connection
134
+ // but may be beneficial in some cases (like conditionally adding logging).
135
+ realWebSocket.addEventListener(
136
+ 'open',
137
+ (event) => {
138
+ this[kEmitter].dispatchEvent(
139
+ bindEvent(this.realWebSocket!, new Event('open', event))
140
+ )
141
+ },
142
+ { once: true }
143
+ )
144
+
145
+ realWebSocket.addEventListener('message', (event) => {
146
+ // Dispatch the "incoming" transport event instead of
147
+ // invoking the internal handler directly. This way,
148
+ // anyone can listen to the "incoming" event but this
149
+ // class is the one resulting in it.
150
+ this.transport.dispatchEvent(
151
+ bindEvent(
152
+ this.realWebSocket!,
153
+ new MessageEvent('incoming', {
154
+ data: event.data,
155
+ origin: event.origin,
156
+ })
157
+ )
158
+ )
159
+ })
160
+
161
+ // Close the original connection when the mock client closes.
162
+ // E.g. "client.close()" was called. This is never forwarded anywhere.
163
+ this.client.addEventListener(
164
+ 'close',
165
+ (event) => {
166
+ this.handleMockClose(event)
167
+ },
168
+ {
169
+ signal: this.mockCloseController.signal,
170
+ }
171
+ )
172
+
173
+ // Forward the "close" event to let the interceptor handle
174
+ // closures initiated by the original server.
175
+ realWebSocket.addEventListener(
176
+ 'close',
177
+ (event) => {
178
+ this.handleRealClose(event)
179
+ },
180
+ {
181
+ signal: this.realCloseController.signal,
182
+ }
183
+ )
184
+
185
+ realWebSocket.addEventListener('error', () => {
186
+ const errorEvent = bindEvent(
187
+ realWebSocket,
188
+ new Event('error', { cancelable: true })
189
+ )
190
+
191
+ // Emit the "error" event on the `server` connection
192
+ // to let the interceptor react to original server errors.
193
+ this[kEmitter].dispatchEvent(errorEvent)
194
+
195
+ // If the error event from the original server hasn't been prevented,
196
+ // forward it to the underlying client.
197
+ if (!errorEvent.defaultPrevented) {
198
+ this.client.dispatchEvent(bindEvent(this.client, new Event('error')))
199
+ }
200
+ })
201
+
202
+ this.realWebSocket = realWebSocket
203
+ }
204
+
205
+ /**
206
+ * Listen for the incoming events from the original WebSocket server.
207
+ */
208
+ public addEventListener<EventType extends keyof WebSocketServerEventMap>(
209
+ event: EventType,
210
+ listener: WebSocketEventListener<WebSocketServerEventMap[EventType]>,
211
+ options?: AddEventListenerOptions | boolean
212
+ ): void {
213
+ if (!Reflect.has(listener, kBoundListener)) {
214
+ const boundListener = listener.bind(this.client)
215
+
216
+ // Store the bound listener on the original listener
217
+ // so the exact bound function can be accessed in "removeEventListener()".
218
+ Object.defineProperty(listener, kBoundListener, {
219
+ value: boundListener,
220
+ enumerable: false,
221
+ })
222
+ }
223
+
224
+ this[kEmitter].addEventListener(
225
+ event,
226
+ Reflect.get(listener, kBoundListener) as EventListener,
227
+ options
228
+ )
229
+ }
230
+
231
+ /**
232
+ * Remove the listener for the given event.
233
+ */
234
+ public removeEventListener<EventType extends keyof WebSocketServerEventMap>(
235
+ event: EventType,
236
+ listener: WebSocketEventListener<WebSocketServerEventMap[EventType]>,
237
+ options?: EventListenerOptions | boolean
238
+ ): void {
239
+ this[kEmitter].removeEventListener(
240
+ event,
241
+ Reflect.get(listener, kBoundListener) as EventListener,
242
+ options
243
+ )
244
+ }
245
+
246
+ /**
247
+ * Send data to the original WebSocket server.
248
+ * @example
249
+ * server.send('hello')
250
+ * server.send(new Blob(['hello']))
251
+ * server.send(new TextEncoder().encode('hello'))
252
+ */
253
+ public send(data: WebSocketData): void {
254
+ this[kSend](data)
255
+ }
256
+
257
+ private [kSend](data: WebSocketData): void {
258
+ const { realWebSocket } = this
259
+
260
+ invariant(
261
+ realWebSocket,
262
+ 'Failed to call "server.send()" for "%s": the connection is not open. Did you forget to call "server.connect()"?',
263
+ this.client.url
264
+ )
265
+
266
+ // Silently ignore writes on the closed original WebSocket.
267
+ if (
268
+ realWebSocket.readyState === WebSocket.CLOSING ||
269
+ realWebSocket.readyState === WebSocket.CLOSED
270
+ ) {
271
+ return
272
+ }
273
+
274
+ // Delegate the send to when the original connection is open.
275
+ // Unlike the mock, connecting to the original server may take time
276
+ // so we cannot call this on the next tick.
277
+ if (realWebSocket.readyState === WebSocket.CONNECTING) {
278
+ realWebSocket.addEventListener(
279
+ 'open',
280
+ () => {
281
+ realWebSocket.send(data)
282
+ },
283
+ { once: true }
284
+ )
285
+ return
286
+ }
287
+
288
+ // Send the data to the original WebSocket server.
289
+ realWebSocket.send(data)
290
+ }
291
+
292
+ /**
293
+ * Close the actual server connection.
294
+ */
295
+ public close(): void {
296
+ const { realWebSocket } = this
297
+
298
+ invariant(
299
+ realWebSocket,
300
+ 'Failed to close server connection for "%s": the connection is not open. Did you forget to call "server.connect()"?',
301
+ this.client.url
302
+ )
303
+
304
+ // Remove the "close" event listener from the server
305
+ // so it doesn't close the underlying WebSocket client
306
+ // when you call "server.close()". This also prevents the
307
+ // `close` event on the `server` connection from being dispatched twice.
308
+ this.realCloseController.abort()
309
+
310
+ if (
311
+ realWebSocket.readyState === WebSocket.CLOSING ||
312
+ realWebSocket.readyState === WebSocket.CLOSED
313
+ ) {
314
+ return
315
+ }
316
+
317
+ // Close the actual client connection.
318
+ realWebSocket.close()
319
+
320
+ // Dispatch the "close" event on the `server` connection.
321
+ queueMicrotask(() => {
322
+ this[kEmitter].dispatchEvent(
323
+ bindEvent(
324
+ this.realWebSocket,
325
+ new CancelableCloseEvent('close', {
326
+ /**
327
+ * @note `server.close()` in the interceptor
328
+ * always results in clean closures.
329
+ */
330
+ code: 1000,
331
+ cancelable: true,
332
+ })
333
+ )
334
+ )
335
+ })
336
+ }
337
+
338
+ private handleIncomingMessage(event: MessageEvent<WebSocketData>): void {
339
+ // Clone the event to dispatch it on this class
340
+ // once again and prevent the "already being dispatched"
341
+ // exception. Clone it here so we can observe this event
342
+ // being prevented in the "server.on()" listeners.
343
+ const messageEvent = bindEvent(
344
+ event.target,
345
+ new CancelableMessageEvent('message', {
346
+ data: event.data,
347
+ origin: event.origin,
348
+ cancelable: true,
349
+ })
350
+ )
351
+
352
+ /**
353
+ * @note Emit "message" event on the server connection
354
+ * instance to let the interceptor know about these
355
+ * incoming events from the original server. In that listener,
356
+ * the interceptor can modify or skip the event forwarding
357
+ * to the mock WebSocket instance.
358
+ */
359
+ this[kEmitter].dispatchEvent(messageEvent)
360
+
361
+ /**
362
+ * @note Forward the incoming server events to the client.
363
+ * Preventing the default on the message event stops this.
364
+ */
365
+ if (!messageEvent.defaultPrevented) {
366
+ this.client.dispatchEvent(
367
+ bindEvent(
368
+ /**
369
+ * @note Bind the forwarded original server events
370
+ * to the mock WebSocket instance so it would
371
+ * dispatch them straight away.
372
+ */
373
+ this.client,
374
+ // Clone the message event again to prevent
375
+ // the "already being dispatched" exception.
376
+ new MessageEvent('message', {
377
+ data: event.data,
378
+ origin: event.origin,
379
+ })
380
+ )
381
+ )
382
+ }
383
+ }
384
+
385
+ private handleMockClose(_event: Event): void {
386
+ // Close the original connection if the mock client closes.
387
+ if (this.realWebSocket) {
388
+ this.realWebSocket.close()
389
+ }
390
+ }
391
+
392
+ private handleRealClose(event: CloseEvent): void {
393
+ // For closures originating from the original server,
394
+ // remove the "close" listener from the mock client.
395
+ // original close -> (?) client[kClose]() --X--> "close" (again).
396
+ this.mockCloseController.abort()
397
+
398
+ const closeEvent = bindEvent(
399
+ this.realWebSocket,
400
+ new CancelableCloseEvent('close', {
401
+ code: event.code,
402
+ reason: event.reason,
403
+ wasClean: event.wasClean,
404
+ cancelable: true,
405
+ })
406
+ )
407
+
408
+ this[kEmitter].dispatchEvent(closeEvent)
409
+
410
+ // If the close event from the server hasn't been prevented,
411
+ // forward the closure to the mock client.
412
+ if (!closeEvent.defaultPrevented) {
413
+ // Close the intercepted client forcefully to
414
+ // allow non-configurable status codes from the server.
415
+ // If the socket has been closed by now, no harm calling
416
+ // this again—it will have no effect.
417
+ this.client[kClose](event.code, event.reason)
418
+ }
419
+ }
420
+ }
@@ -0,0 +1,39 @@
1
+ import { CloseEvent } from './utils/events'
2
+
3
+ export type WebSocketData = string | ArrayBufferLike | Blob | ArrayBufferView
4
+
5
+ export type WebSocketTransportEventMap = {
6
+ incoming: MessageEvent<WebSocketData>
7
+ outgoing: MessageEvent<WebSocketData>
8
+ close: CloseEvent
9
+ }
10
+
11
+ export type StrictEventListenerOrEventListenerObject<EventType extends Event> =
12
+ | ((this: WebSocket, event: EventType) => void)
13
+ | {
14
+ handleEvent(this: WebSocket, event: EventType): void
15
+ }
16
+
17
+ export interface WebSocketTransport {
18
+ addEventListener<EventType extends keyof WebSocketTransportEventMap>(
19
+ event: EventType,
20
+ listener: StrictEventListenerOrEventListenerObject<
21
+ WebSocketTransportEventMap[EventType]
22
+ > | null,
23
+ options?: boolean | AddEventListenerOptions
24
+ ): void
25
+
26
+ dispatchEvent<EventType extends keyof WebSocketTransportEventMap>(
27
+ event: WebSocketTransportEventMap[EventType]
28
+ ): boolean
29
+
30
+ /**
31
+ * Send the data from the server to this client.
32
+ */
33
+ send(data: WebSocketData): void
34
+
35
+ /**
36
+ * Close the client connection.
37
+ */
38
+ close(code?: number, reason?: string): void
39
+ }