@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.
- package/ClientRequest/package.json +11 -0
- package/LICENSE.md +9 -0
- package/README.md +31 -0
- package/RemoteHttpInterceptor/package.json +11 -0
- package/WebSocket/package.json +12 -0
- package/XMLHttpRequest/package.json +12 -0
- package/changes.json +10 -0
- package/fetch/package.json +12 -0
- package/lib/browser/Interceptor-Deczogc8.d.cts +65 -0
- package/lib/browser/Interceptor-gqKgs-aF.d.mts +65 -0
- package/lib/browser/XMLHttpRequest-BACqefB-.cjs +761 -0
- package/lib/browser/XMLHttpRequest-BACqefB-.cjs.map +1 -0
- package/lib/browser/XMLHttpRequest-BvxZV0WU.mjs +756 -0
- package/lib/browser/XMLHttpRequest-BvxZV0WU.mjs.map +1 -0
- package/lib/browser/bufferUtils-BiiO6HZv.mjs +20 -0
- package/lib/browser/bufferUtils-BiiO6HZv.mjs.map +1 -0
- package/lib/browser/bufferUtils-Uc0eRItL.cjs +38 -0
- package/lib/browser/bufferUtils-Uc0eRItL.cjs.map +1 -0
- package/lib/browser/createRequestId-Cs4oXfa1.cjs +205 -0
- package/lib/browser/createRequestId-Cs4oXfa1.cjs.map +1 -0
- package/lib/browser/createRequestId-DQcIlohW.mjs +170 -0
- package/lib/browser/createRequestId-DQcIlohW.mjs.map +1 -0
- package/lib/browser/fetch-DdKEdDOR.mjs +248 -0
- package/lib/browser/fetch-DdKEdDOR.mjs.map +1 -0
- package/lib/browser/fetch-U3v3Y4ap.cjs +253 -0
- package/lib/browser/fetch-U3v3Y4ap.cjs.map +1 -0
- package/lib/browser/getRawRequest-BTaNLFr0.mjs +218 -0
- package/lib/browser/getRawRequest-BTaNLFr0.mjs.map +1 -0
- package/lib/browser/getRawRequest-zx8rUJL2.cjs +259 -0
- package/lib/browser/getRawRequest-zx8rUJL2.cjs.map +1 -0
- package/lib/browser/glossary-BdLS4k1H.d.cts +70 -0
- package/lib/browser/glossary-DYwOrogs.d.mts +70 -0
- package/lib/browser/handleRequest-CvX2G-Lz.cjs +189 -0
- package/lib/browser/handleRequest-CvX2G-Lz.cjs.map +1 -0
- package/lib/browser/handleRequest-D7kpTI5U.mjs +178 -0
- package/lib/browser/handleRequest-D7kpTI5U.mjs.map +1 -0
- package/lib/browser/hasConfigurableGlobal-BvCTG97d.cjs +45 -0
- package/lib/browser/hasConfigurableGlobal-BvCTG97d.cjs.map +1 -0
- package/lib/browser/hasConfigurableGlobal-npXitu1-.mjs +33 -0
- package/lib/browser/hasConfigurableGlobal-npXitu1-.mjs.map +1 -0
- package/lib/browser/index.cjs +70 -0
- package/lib/browser/index.cjs.map +1 -0
- package/lib/browser/index.d.cts +96 -0
- package/lib/browser/index.d.mts +96 -0
- package/lib/browser/index.mjs +56 -0
- package/lib/browser/index.mjs.map +1 -0
- package/lib/browser/interceptors/WebSocket/index.cjs +622 -0
- package/lib/browser/interceptors/WebSocket/index.cjs.map +1 -0
- package/lib/browser/interceptors/WebSocket/index.d.cts +277 -0
- package/lib/browser/interceptors/WebSocket/index.d.mts +277 -0
- package/lib/browser/interceptors/WebSocket/index.mjs +615 -0
- package/lib/browser/interceptors/WebSocket/index.mjs.map +1 -0
- package/lib/browser/interceptors/XMLHttpRequest/index.cjs +7 -0
- package/lib/browser/interceptors/XMLHttpRequest/index.d.cts +15 -0
- package/lib/browser/interceptors/XMLHttpRequest/index.d.mts +15 -0
- package/lib/browser/interceptors/XMLHttpRequest/index.mjs +7 -0
- package/lib/browser/interceptors/fetch/index.cjs +6 -0
- package/lib/browser/interceptors/fetch/index.d.cts +13 -0
- package/lib/browser/interceptors/fetch/index.d.mts +13 -0
- package/lib/browser/interceptors/fetch/index.mjs +6 -0
- package/lib/browser/presets/browser.cjs +17 -0
- package/lib/browser/presets/browser.cjs.map +1 -0
- package/lib/browser/presets/browser.d.cts +12 -0
- package/lib/browser/presets/browser.d.mts +14 -0
- package/lib/browser/presets/browser.mjs +17 -0
- package/lib/browser/presets/browser.mjs.map +1 -0
- package/lib/browser/resolveWebSocketUrl-6K6EgqsA.cjs +31 -0
- package/lib/browser/resolveWebSocketUrl-6K6EgqsA.cjs.map +1 -0
- package/lib/browser/resolveWebSocketUrl-C83-x9iE.mjs +25 -0
- package/lib/browser/resolveWebSocketUrl-C83-x9iE.mjs.map +1 -0
- package/lib/node/BatchInterceptor-3LnAnLTx.cjs +49 -0
- package/lib/node/BatchInterceptor-3LnAnLTx.cjs.map +1 -0
- package/lib/node/BatchInterceptor-D7mXzHcQ.d.mts +26 -0
- package/lib/node/BatchInterceptor-DFaBPilf.mjs +44 -0
- package/lib/node/BatchInterceptor-DFaBPilf.mjs.map +1 -0
- package/lib/node/BatchInterceptor-D_YqR8qU.d.cts +26 -0
- package/lib/node/ClientRequest-2rDe54Ui.cjs +1043 -0
- package/lib/node/ClientRequest-2rDe54Ui.cjs.map +1 -0
- package/lib/node/ClientRequest-Ca8Qykuv.mjs +1034 -0
- package/lib/node/ClientRequest-Ca8Qykuv.mjs.map +1 -0
- package/lib/node/Interceptor-DEazpLJd.d.mts +133 -0
- package/lib/node/Interceptor-DJ2akVWI.d.cts +133 -0
- package/lib/node/RemoteHttpInterceptor.cjs +154 -0
- package/lib/node/RemoteHttpInterceptor.cjs.map +1 -0
- package/lib/node/RemoteHttpInterceptor.d.cts +39 -0
- package/lib/node/RemoteHttpInterceptor.d.mts +39 -0
- package/lib/node/RemoteHttpInterceptor.mjs +152 -0
- package/lib/node/RemoteHttpInterceptor.mjs.map +1 -0
- package/lib/node/XMLHttpRequest-B7kJdYYI.cjs +763 -0
- package/lib/node/XMLHttpRequest-B7kJdYYI.cjs.map +1 -0
- package/lib/node/XMLHttpRequest-C8dIZpds.mjs +757 -0
- package/lib/node/XMLHttpRequest-C8dIZpds.mjs.map +1 -0
- package/lib/node/bufferUtils-DiCTqG-7.cjs +38 -0
- package/lib/node/bufferUtils-DiCTqG-7.cjs.map +1 -0
- package/lib/node/bufferUtils-_8XfKIfX.mjs +20 -0
- package/lib/node/bufferUtils-_8XfKIfX.mjs.map +1 -0
- package/lib/node/chunk-CbDLau6x.cjs +34 -0
- package/lib/node/fetch-BmXpK10r.cjs +272 -0
- package/lib/node/fetch-BmXpK10r.cjs.map +1 -0
- package/lib/node/fetch-G1DVwDKG.mjs +265 -0
- package/lib/node/fetch-G1DVwDKG.mjs.map +1 -0
- package/lib/node/fetchUtils-BaY5iWXw.cjs +419 -0
- package/lib/node/fetchUtils-BaY5iWXw.cjs.map +1 -0
- package/lib/node/fetchUtils-CoU35g3M.mjs +359 -0
- package/lib/node/fetchUtils-CoU35g3M.mjs.map +1 -0
- package/lib/node/getRawRequest-BavnMWh_.cjs +36 -0
- package/lib/node/getRawRequest-BavnMWh_.cjs.map +1 -0
- package/lib/node/getRawRequest-DnwmXyOW.mjs +24 -0
- package/lib/node/getRawRequest-DnwmXyOW.mjs.map +1 -0
- package/lib/node/glossary-BLKRyLBd.cjs +12 -0
- package/lib/node/glossary-BLKRyLBd.cjs.map +1 -0
- package/lib/node/glossary-glQBRnVD.mjs +6 -0
- package/lib/node/glossary-glQBRnVD.mjs.map +1 -0
- package/lib/node/handleRequest-Bb7Y-XLw.cjs +220 -0
- package/lib/node/handleRequest-Bb7Y-XLw.cjs.map +1 -0
- package/lib/node/handleRequest-Y97UwBbF.mjs +190 -0
- package/lib/node/handleRequest-Y97UwBbF.mjs.map +1 -0
- package/lib/node/hasConfigurableGlobal-C97fWuaA.cjs +26 -0
- package/lib/node/hasConfigurableGlobal-C97fWuaA.cjs.map +1 -0
- package/lib/node/hasConfigurableGlobal-DBJA0vjm.mjs +20 -0
- package/lib/node/hasConfigurableGlobal-DBJA0vjm.mjs.map +1 -0
- package/lib/node/index-BMbJ8FXL.d.cts +113 -0
- package/lib/node/index-C0YAQ36w.d.mts +113 -0
- package/lib/node/index.cjs +54 -0
- package/lib/node/index.cjs.map +1 -0
- package/lib/node/index.d.cts +75 -0
- package/lib/node/index.d.mts +75 -0
- package/lib/node/index.mjs +40 -0
- package/lib/node/index.mjs.map +1 -0
- package/lib/node/interceptors/ClientRequest/index.cjs +6 -0
- package/lib/node/interceptors/ClientRequest/index.d.cts +2 -0
- package/lib/node/interceptors/ClientRequest/index.d.mts +3 -0
- package/lib/node/interceptors/ClientRequest/index.mjs +6 -0
- package/lib/node/interceptors/XMLHttpRequest/index.cjs +6 -0
- package/lib/node/interceptors/XMLHttpRequest/index.d.cts +14 -0
- package/lib/node/interceptors/XMLHttpRequest/index.d.mts +14 -0
- package/lib/node/interceptors/XMLHttpRequest/index.mjs +6 -0
- package/lib/node/interceptors/fetch/index.cjs +5 -0
- package/lib/node/interceptors/fetch/index.d.cts +12 -0
- package/lib/node/interceptors/fetch/index.d.mts +12 -0
- package/lib/node/interceptors/fetch/index.mjs +5 -0
- package/lib/node/node-DwCc6iuP.mjs +27 -0
- package/lib/node/node-DwCc6iuP.mjs.map +1 -0
- package/lib/node/node-dKdAf3tC.cjs +39 -0
- package/lib/node/node-dKdAf3tC.cjs.map +1 -0
- package/lib/node/presets/node.cjs +22 -0
- package/lib/node/presets/node.cjs.map +1 -0
- package/lib/node/presets/node.d.cts +13 -0
- package/lib/node/presets/node.d.mts +15 -0
- package/lib/node/presets/node.mjs +22 -0
- package/lib/node/presets/node.mjs.map +1 -0
- package/lib/node/utils/node/index.cjs +4 -0
- package/lib/node/utils/node/index.d.cts +16 -0
- package/lib/node/utils/node/index.d.mts +16 -0
- package/lib/node/utils/node/index.mjs +3 -0
- package/package.json +204 -0
- package/presets/browser/package.json +5 -0
- package/presets/node/package.json +11 -0
- package/src/BatchInterceptor.test.ts +255 -0
- package/src/BatchInterceptor.ts +95 -0
- package/src/Interceptor.test.ts +205 -0
- package/src/Interceptor.ts +249 -0
- package/src/InterceptorError.ts +7 -0
- package/src/RemoteHttpInterceptor.ts +251 -0
- package/src/RequestController.test.ts +104 -0
- package/src/RequestController.ts +109 -0
- package/src/createRequestId.test.ts +7 -0
- package/src/createRequestId.ts +9 -0
- package/src/getRawRequest.ts +21 -0
- package/src/glossary.ts +37 -0
- package/src/index.ts +15 -0
- package/src/interceptors/ClientRequest/MockHttpSocket.ts +724 -0
- package/src/interceptors/ClientRequest/agents.ts +110 -0
- package/src/interceptors/ClientRequest/index.test.ts +75 -0
- package/src/interceptors/ClientRequest/index.ts +193 -0
- package/src/interceptors/ClientRequest/utils/getIncomingMessageBody.test.ts +54 -0
- package/src/interceptors/ClientRequest/utils/getIncomingMessageBody.ts +45 -0
- package/src/interceptors/ClientRequest/utils/normalizeClientRequestArgs.test.ts +427 -0
- package/src/interceptors/ClientRequest/utils/normalizeClientRequestArgs.ts +268 -0
- package/src/interceptors/ClientRequest/utils/parserUtils.ts +48 -0
- package/src/interceptors/ClientRequest/utils/recordRawHeaders.test.ts +258 -0
- package/src/interceptors/ClientRequest/utils/recordRawHeaders.ts +262 -0
- package/src/interceptors/Socket/MockSocket.test.ts +264 -0
- package/src/interceptors/Socket/MockSocket.ts +58 -0
- package/src/interceptors/Socket/utils/baseUrlFromConnectionOptions.ts +26 -0
- package/src/interceptors/Socket/utils/normalizeSocketWriteArgs.test.ts +52 -0
- package/src/interceptors/Socket/utils/normalizeSocketWriteArgs.ts +33 -0
- package/src/interceptors/WebSocket/WebSocketClassTransport.ts +116 -0
- package/src/interceptors/WebSocket/WebSocketClientConnection.ts +152 -0
- package/src/interceptors/WebSocket/WebSocketOverride.ts +252 -0
- package/src/interceptors/WebSocket/WebSocketServerConnection.ts +420 -0
- package/src/interceptors/WebSocket/WebSocketTransport.ts +39 -0
- package/src/interceptors/WebSocket/index.ts +191 -0
- package/src/interceptors/WebSocket/utils/bindEvent.test.ts +27 -0
- package/src/interceptors/WebSocket/utils/bindEvent.ts +21 -0
- package/src/interceptors/WebSocket/utils/events.test.ts +101 -0
- package/src/interceptors/WebSocket/utils/events.ts +94 -0
- package/src/interceptors/XMLHttpRequest/XMLHttpRequestController.ts +746 -0
- package/src/interceptors/XMLHttpRequest/XMLHttpRequestProxy.ts +121 -0
- package/src/interceptors/XMLHttpRequest/index.ts +61 -0
- package/src/interceptors/XMLHttpRequest/polyfills/EventPolyfill.ts +51 -0
- package/src/interceptors/XMLHttpRequest/polyfills/ProgressEventPolyfill.ts +17 -0
- package/src/interceptors/XMLHttpRequest/utils/concatArrayBuffer.ts +12 -0
- package/src/interceptors/XMLHttpRequest/utils/concateArrayBuffer.test.ts +12 -0
- package/src/interceptors/XMLHttpRequest/utils/createEvent.test.ts +26 -0
- package/src/interceptors/XMLHttpRequest/utils/createEvent.ts +41 -0
- package/src/interceptors/XMLHttpRequest/utils/createResponse.ts +49 -0
- package/src/interceptors/XMLHttpRequest/utils/getBodyByteLength.test.ts +164 -0
- package/src/interceptors/XMLHttpRequest/utils/getBodyByteLength.ts +16 -0
- package/src/interceptors/XMLHttpRequest/utils/isDomParserSupportedType.ts +14 -0
- package/src/interceptors/fetch/index.ts +214 -0
- package/src/interceptors/fetch/utils/brotli-decompress.browser.ts +14 -0
- package/src/interceptors/fetch/utils/brotli-decompress.ts +31 -0
- package/src/interceptors/fetch/utils/createNetworkError.ts +5 -0
- package/src/interceptors/fetch/utils/decompression.ts +85 -0
- package/src/interceptors/fetch/utils/followRedirect.ts +114 -0
- package/src/presets/browser.ts +11 -0
- package/src/presets/node.ts +13 -0
- package/src/utils/bufferUtils.test.ts +21 -0
- package/src/utils/bufferUtils.ts +22 -0
- package/src/utils/canParseUrl.ts +13 -0
- package/src/utils/cloneObject.test.ts +94 -0
- package/src/utils/cloneObject.ts +36 -0
- package/src/utils/createProxy.test.ts +164 -0
- package/src/utils/createProxy.ts +104 -0
- package/src/utils/emitAsync.ts +25 -0
- package/src/utils/fetchUtils.ts +119 -0
- package/src/utils/findPropertySource.test.ts +27 -0
- package/src/utils/findPropertySource.ts +20 -0
- package/src/utils/getCleanUrl.test.ts +32 -0
- package/src/utils/getCleanUrl.ts +6 -0
- package/src/utils/getUrlByRequestOptions.test.ts +163 -0
- package/src/utils/getUrlByRequestOptions.ts +152 -0
- package/src/utils/getValueBySymbol.test.ts +14 -0
- package/src/utils/getValueBySymbol.ts +19 -0
- package/src/utils/handleRequest.ts +205 -0
- package/src/utils/hasConfigurableGlobal.test.ts +83 -0
- package/src/utils/hasConfigurableGlobal.ts +34 -0
- package/src/utils/isNodeLikeError.ts +13 -0
- package/src/utils/isObject.test.ts +21 -0
- package/src/utils/isObject.ts +8 -0
- package/src/utils/isPropertyAccessible.ts +19 -0
- package/src/utils/nextTick.ts +11 -0
- package/src/utils/node/index.ts +39 -0
- package/src/utils/parseJson.test.ts +10 -0
- package/src/utils/parseJson.ts +12 -0
- package/src/utils/resolveWebSocketUrl.ts +45 -0
- 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
|
+
}
|