@mswjs/interceptors 0.25.15 → 0.26.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/README.md +142 -0
- package/lib/browser/{Interceptor-8d5fd4c6.d.ts → Interceptor-b7c08a9f.d.ts} +1 -32
- package/lib/browser/{chunk-DZVB7JEV.mjs → chunk-7FP6PMUG.mjs} +5 -3
- package/lib/browser/{chunk-DZVB7JEV.mjs.map → chunk-7FP6PMUG.mjs.map} +1 -1
- package/lib/browser/{chunk-44V5AUD6.mjs → chunk-E5YM6K3N.mjs} +16 -13
- package/lib/browser/chunk-E5YM6K3N.mjs.map +1 -0
- package/lib/browser/{chunk-3O7223NM.js → chunk-EM6NYHQV.js} +3 -7
- package/lib/browser/chunk-EM6NYHQV.js.map +1 -0
- package/lib/browser/chunk-FZJKKO5H.js +7 -0
- package/lib/browser/chunk-FZJKKO5H.js.map +1 -0
- package/lib/browser/chunk-HAGW22AN.mjs +7 -0
- package/lib/browser/chunk-HAGW22AN.mjs.map +1 -0
- package/lib/browser/{chunk-EILJI62P.js → chunk-KWMV32LI.js} +8 -6
- package/lib/browser/chunk-KWMV32LI.js.map +1 -0
- package/lib/browser/{chunk-D3X43XAK.js → chunk-NNNHH4BN.js} +19 -16
- package/lib/browser/chunk-NNNHH4BN.js.map +1 -0
- package/lib/browser/{chunk-WZQN3FMY.mjs → chunk-SGO3INLV.mjs} +1 -5
- package/lib/browser/{chunk-3O7223NM.js.map → chunk-SGO3INLV.mjs.map} +1 -1
- package/lib/browser/glossary-640c9679.d.ts +33 -0
- package/lib/browser/index.d.ts +3 -2
- package/lib/browser/index.js +5 -3
- package/lib/browser/index.js.map +1 -1
- package/lib/browser/index.mjs +4 -2
- package/lib/browser/index.mjs.map +1 -1
- package/lib/browser/interceptors/WebSocket/index.d.ts +207 -0
- package/lib/browser/interceptors/WebSocket/index.js +433 -0
- package/lib/browser/interceptors/WebSocket/index.js.map +1 -0
- package/lib/browser/interceptors/WebSocket/index.mjs +433 -0
- package/lib/browser/interceptors/WebSocket/index.mjs.map +1 -0
- package/lib/browser/interceptors/XMLHttpRequest/index.d.ts +2 -1
- package/lib/browser/interceptors/XMLHttpRequest/index.js +4 -3
- package/lib/browser/interceptors/XMLHttpRequest/index.mjs +3 -2
- package/lib/browser/interceptors/fetch/index.d.ts +2 -1
- package/lib/browser/interceptors/fetch/index.js +4 -3
- package/lib/browser/interceptors/fetch/index.mjs +3 -2
- package/lib/browser/presets/browser.d.ts +2 -1
- package/lib/browser/presets/browser.js +6 -5
- package/lib/browser/presets/browser.js.map +1 -1
- package/lib/browser/presets/browser.mjs +4 -3
- package/lib/browser/presets/browser.mjs.map +1 -1
- package/lib/node/interceptors/fetch/index.js +11 -10
- package/lib/node/interceptors/fetch/index.js.map +1 -1
- package/lib/node/interceptors/fetch/index.mjs +11 -10
- package/lib/node/interceptors/fetch/index.mjs.map +1 -1
- package/package.json +15 -2
- package/src/interceptors/WebSocket/WebSocketClassTransport.ts +55 -0
- package/src/interceptors/WebSocket/WebSocketClientConnection.ts +88 -0
- package/src/interceptors/WebSocket/WebSocketOverride.ts +229 -0
- package/src/interceptors/WebSocket/WebSocketServerConnection.ts +186 -0
- package/src/interceptors/WebSocket/WebSocketTransport.ts +35 -0
- package/src/interceptors/WebSocket/index.ts +84 -0
- package/src/interceptors/WebSocket/utils/bindEvent.test.ts +27 -0
- package/src/interceptors/WebSocket/utils/bindEvent.ts +13 -0
- package/src/interceptors/WebSocket/utils/events.ts +18 -0
- package/src/interceptors/fetch/index.ts +15 -11
- package/lib/browser/chunk-44V5AUD6.mjs.map +0 -1
- package/lib/browser/chunk-D3X43XAK.js.map +0 -1
- package/lib/browser/chunk-EILJI62P.js.map +0 -1
- package/lib/browser/chunk-WZQN3FMY.mjs.map +0 -1
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../../../src/interceptors/WebSocket/utils/bindEvent.ts","../../../../src/interceptors/WebSocket/utils/events.ts","../../../../src/interceptors/WebSocket/WebSocketClientConnection.ts","../../../../src/interceptors/WebSocket/WebSocketServerConnection.ts","../../../../src/interceptors/WebSocket/WebSocketTransport.ts","../../../../src/interceptors/WebSocket/WebSocketOverride.ts","../../../../src/interceptors/WebSocket/WebSocketClassTransport.ts","../../../../src/interceptors/WebSocket/index.ts"],"names":["kEmitter","invariant"],"mappings":"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","sourcesContent":["type EventWithTarget<E extends Event, T> = E & { target: T }\n\nexport function bindEvent<E extends Event, T>(\n target: T,\n event: E\n): EventWithTarget<E, T> {\n Object.defineProperty(event, 'target', {\n enumerable: true,\n writable: true,\n value: target,\n })\n return event as EventWithTarget<E, T>\n}\n","interface CloseEventInit extends EventInit {\n code?: number\n reason?: string\n wasClean?: boolean\n}\n\nexport class CloseEvent extends Event {\n public code: number\n public reason: string\n public wasClean: boolean\n\n constructor(type: string, init: CloseEventInit = {}) {\n super(type, init)\n this.code = init.code === undefined ? 0 : init.code\n this.reason = init.reason === undefined ? '' : init.reason\n this.wasClean = init.wasClean === undefined ? false : init.wasClean\n }\n}\n","/**\n * WebSocket client class.\n * This represents an incoming WebSocket client connection.\n * @note Keep this class implementation-agnostic because it's\n * meant to be used over any WebSocket implementation\n * (not all of them follow the one from WHATWG).\n */\nimport type { WebSocketRawData, WebSocketTransport } from './WebSocketTransport'\nimport { WebSocketMessageListener } from './WebSocketOverride'\nimport { bindEvent } from './utils/bindEvent'\nimport { CloseEvent } from './utils/events'\n\nconst kEmitter = Symbol('kEmitter')\n\n/**\n * The WebSocket client instance represents an incoming\n * client connection. The user can control the connection,\n * send and receive events.\n */\nexport class WebSocketClientConnection {\n public readonly url: URL\n\n protected [kEmitter]: EventTarget\n\n constructor(\n protected readonly ws: WebSocket,\n protected readonly transport: WebSocketTransport\n ) {\n this.url = new URL(ws.url)\n this[kEmitter] = new EventTarget()\n\n // Emit outgoing client data (\"ws.send()\") as \"message\"\n // events on the client connection.\n this.transport.onOutgoing = (data) => {\n this[kEmitter].dispatchEvent(\n bindEvent(this.ws, new MessageEvent('message', { data }))\n )\n }\n\n this.transport.onClose = (event) => {\n this[kEmitter].dispatchEvent(\n bindEvent(this.ws, new CloseEvent('close', event))\n )\n }\n }\n\n /**\n * Listen for the outgoing events from the connected WebSocket client.\n */\n public addEventListener(\n event: string,\n listener: WebSocketMessageListener,\n options?: AddEventListenerOptions | boolean\n ): void {\n this[kEmitter].addEventListener(event, listener as EventListener, options)\n }\n\n /**\n * Removes the listener for the given event.\n */\n public removeEventListener(\n event: string,\n listener: WebSocketMessageListener,\n options?: EventListenerOptions | boolean\n ): void {\n this[kEmitter].removeEventListener(\n event,\n listener as EventListener,\n options\n )\n }\n\n /**\n * Send data to the connected client.\n */\n public send(data: WebSocketRawData): void {\n this.transport.send(data)\n }\n\n /**\n * Close the WebSocket connection.\n * @param {number} code A status code (see https://www.rfc-editor.org/rfc/rfc6455#section-7.4.1).\n * @param {string} reason A custom connection close reason.\n */\n public close(code?: number, reason?: string): void {\n this.transport.close(code, reason)\n }\n}\n","import { invariant } from 'outvariant'\nimport type { WebSocketOverride } from './WebSocketOverride'\nimport type { WebSocketRawData } from './WebSocketTransport'\nimport type { WebSocketClassTransport } from './WebSocketClassTransport'\nimport { bindEvent } from './utils/bindEvent'\nimport { CloseEvent } from './utils/events'\n\nconst kEmitter = Symbol('kEmitter')\n\n/**\n * The WebSocket server instance represents the actual production\n * WebSocket server connection. It's idle by default but you can\n * establish it by calling `server.connect()`.\n */\nexport class WebSocketServerConnection {\n /**\n * A WebSocket instance connected to the original server.\n */\n private realWebSocket?: WebSocket\n private [kEmitter]: EventTarget\n\n constructor(\n private readonly mockWebSocket: WebSocketOverride,\n private readonly createConnection: () => WebSocket,\n private readonly transport: WebSocketClassTransport\n ) {\n this[kEmitter] = new EventTarget()\n\n // Handle incoming events from the actual server.\n // The (mock) WebSocket instance will call this\n // whenever a \"message\" event from the actual server\n // is dispatched on it (the dispatch will be skipped).\n this.transport.onIncoming = (event) => {\n /**\n * @note Emit \"message\" event on the WebSocketClassServer\n * instance to let the interceptor know about these\n * incoming events from the original server. In that listener,\n * the interceptor can modify or skip the event forwarding\n * to the mock WebSocket instance.\n */\n this[kEmitter].dispatchEvent(event)\n }\n }\n\n /**\n * Server ready state.\n * Proxies the ready state of the original WebSocket instance,\n * if set. If the original connection hasn't been established,\n * defaults to `-1`.\n */\n public get readyState(): number {\n if (this.realWebSocket) {\n return this.realWebSocket.readyState\n }\n\n return -1\n }\n\n /**\n * Open connection to the original WebSocket server.\n */\n public connect(): void {\n invariant(\n this.readyState === -1,\n 'Failed to call \"connect()\" on the original WebSocket instance: the connection already open'\n )\n\n const ws = this.createConnection()\n\n // Close the original connection when the (mock)\n // client closes, regardless of the reason.\n this.mockWebSocket.addEventListener(\n 'close',\n (event) => {\n ws.close(event.code, event.reason)\n },\n { once: true }\n )\n\n // Once the connection is open, forward any incoming\n // events directly to the (override) WebSocket instance.\n ws.addEventListener('message', (event) => {\n // Clone the event to dispatch it on this class\n // once again and prevent the \"already being dispatched\"\n // exception. Clone it here so we can observe this event\n // being prevented in the \"server.on()\" listeners.\n const messageEvent = bindEvent(\n ws,\n new MessageEvent('message', {\n data: event.data,\n origin: event.origin,\n cancelable: true,\n })\n )\n\n this.transport.onIncoming(messageEvent)\n\n // Unless the default is prevented, forward the\n // messages from the original server to the mock client.\n // This is the only way the user can receive them.\n if (!messageEvent.defaultPrevented) {\n this.mockWebSocket.dispatchEvent(\n bindEvent(\n /**\n * @note Bind the forwarded original server events\n * to the mock WebSocket instance so it would\n * dispatch them straight away.\n */\n this.mockWebSocket,\n // Clone the message event again to prevent\n // the \"already being dispatched\" exception.\n new MessageEvent('message', {\n data: event.data,\n origin: event.origin,\n })\n )\n )\n }\n })\n\n this.realWebSocket = ws\n }\n\n /**\n * Listen for the incoming events from the original WebSocket server.\n */\n public addEventListener<K extends keyof WebSocketEventMap>(\n event: K,\n listener: (this: WebSocket, event: WebSocketEventMap[K]) => void,\n options?: AddEventListenerOptions | boolean\n ): void {\n this[kEmitter].addEventListener(\n event,\n listener.bind(this.realWebSocket!) as EventListener,\n options\n )\n }\n\n /**\n * Removes the listener for the given event.\n */\n public removeEventListener<K extends keyof WebSocketEventMap>(\n event: K,\n listener: (this: WebSocket, event: WebSocketEventMap[K]) => void,\n options?: EventListenerOptions | boolean\n ): void {\n this[kEmitter].removeEventListener(\n event,\n listener as EventListener,\n options\n )\n }\n\n /**\n * Send data to the original WebSocket server.\n * @example\n * server.send('hello')\n * server.send(new Blob(['hello']))\n * server.send(new TextEncoder().encode('hello'))\n */\n public send(data: WebSocketRawData): void {\n const { realWebSocket } = this\n invariant(\n realWebSocket,\n 'Failed to call \"server.send()\" for \"%s\": the connection is not open. Did you forget to call \"await server.connect()\"?',\n this.mockWebSocket.url\n )\n\n // Delegate the send to when the original connection is open.\n // Unlike the mock, connecting to the original server may take time\n // so we cannot call this on the next tick.\n if (realWebSocket.readyState === realWebSocket.CONNECTING) {\n realWebSocket.addEventListener(\n 'open',\n () => {\n realWebSocket.send(data)\n },\n { once: true }\n )\n return\n }\n\n // Send the data to the original WebSocket server.\n realWebSocket.send(data)\n }\n}\n","export type WebSocketRawData = string | ArrayBufferLike | Blob | ArrayBufferView\n\nexport type WebSocketTransportOnIncomingCallback = (\n event: MessageEvent<WebSocketRawData>\n) => void\n\nexport type WebSocketTransportOnOutgoingCallback = (\n data: WebSocketRawData\n) => void\n\nexport type WebSocketTransportOnCloseCallback = (event: CloseEvent) => void\n\nexport abstract class WebSocketTransport {\n /**\n * Listener for the incoming server events.\n * This is called when the client receives the\n * event from the original server connection.\n *\n * This way, we can trigger the \"message\" event\n * on the mocked connection to let the user know.\n */\n abstract onIncoming: WebSocketTransportOnIncomingCallback\n abstract onOutgoing: WebSocketTransportOnOutgoingCallback\n abstract onClose: WebSocketTransportOnCloseCallback\n\n /**\n * Send the data from the server to this client.\n */\n abstract send(data: WebSocketRawData): void\n\n /**\n * Close the client connection.\n */\n abstract close(code?: number, reason?: string): void\n}\n","import { invariant } from 'outvariant'\nimport type { WebSocketRawData } from './WebSocketTransport'\nimport { bindEvent } from './utils/bindEvent'\nimport { CloseEvent } from './utils/events'\n\ntype WebSocketEventListener = (this: WebSocket, event: Event) => void\n\nexport type WebSocketMessageListener = (\n this: WebSocket,\n event: MessageEvent\n) => void\n\ntype WebSocketCloseListener = (this: WebSocket, event: CloseEvent) => void\n\nconst WEBSOCKET_CLOSE_CODE_RANGE_ERROR =\n 'InvalidAccessError: close code out of user configurable range'\n\nexport const kOnSend = Symbol('kOnSend')\n\nexport class WebSocketOverride extends EventTarget implements WebSocket {\n static readonly CONNECTING = WebSocket.CONNECTING\n static readonly OPEN = WebSocket.OPEN\n static readonly CLOSING = WebSocket.CLOSING\n static readonly CLOSED = WebSocket.CLOSED\n readonly CONNECTING = WebSocket.CONNECTING\n readonly OPEN = WebSocket.OPEN\n readonly CLOSING = WebSocket.CLOSING\n readonly CLOSED = WebSocket.CLOSED\n\n public url: string\n public protocol: string\n public extensions: string\n public binaryType: BinaryType\n public readyState: number\n public bufferedAmount: number\n\n private _onopen: WebSocketEventListener | null = null\n private _onmessage: WebSocketMessageListener | null = null\n private _onerror: WebSocketEventListener | null = null\n private _onclose: WebSocketCloseListener | null = null\n\n private [kOnSend]?: (data: WebSocketRawData) => void\n\n constructor(url: string | URL, protocols?: string | Array<string>) {\n super()\n this.url = url.toString()\n this.protocol = ''\n this.extensions = ''\n this.binaryType = 'blob'\n this.readyState = this.CONNECTING\n this.bufferedAmount = 0\n\n Reflect.set(this, 'readyState', this.CONNECTING)\n queueMicrotask(() => {\n Reflect.set(this, 'readyState', this.OPEN)\n this.protocol = protocols ? protocols[0] : 'ws'\n\n this.dispatchEvent(bindEvent(this, new Event('open')))\n })\n }\n\n set onopen(listener: WebSocketEventListener | null) {\n this.removeEventListener('open', this._onopen)\n this._onopen = listener\n if (listener !== null) {\n this.addEventListener('open', listener)\n }\n }\n get onopen(): WebSocketEventListener | null {\n return this._onopen\n }\n\n set onmessage(listener: WebSocketMessageListener | null) {\n this.removeEventListener(\n 'message',\n this._onmessage as WebSocketEventListener\n )\n this._onmessage = listener\n if (listener !== null) {\n this.addEventListener('message', listener)\n }\n }\n get onmessage(): WebSocketMessageListener | null {\n return this._onmessage\n }\n\n set onerror(listener: WebSocketEventListener | null) {\n this.removeEventListener('error', this._onerror)\n this._onerror = listener\n if (listener !== null) {\n this.addEventListener('error', listener)\n }\n }\n get onerror(): WebSocketEventListener | null {\n return this._onerror\n }\n\n set onclose(listener: WebSocketCloseListener | null) {\n this.removeEventListener('close', this._onclose as WebSocketEventListener)\n this._onclose = listener\n if (listener !== null) {\n this.addEventListener('close', listener)\n }\n }\n get onclose(): WebSocketCloseListener | null {\n return this._onclose\n }\n\n /**\n * @see https://websockets.spec.whatwg.org/#ref-for-dom-websocket-send%E2%91%A0\n */\n public send(data: WebSocketRawData): void {\n if (this.readyState === this.CONNECTING) {\n this.close()\n throw new DOMException('InvalidStateError')\n }\n\n // Sending when the socket is about to close\n // discards the sent data.\n if (this.readyState === this.CLOSING || this.readyState === this.CLOSED) {\n return\n }\n\n // Buffer the data to send in this even loop\n // but send it in the next.\n this.bufferedAmount += getDataSize(data)\n\n queueMicrotask(() => {\n // This is a bit optimistic but since no actual data transfer\n // is involved, all the data will be \"sent\" on the next tick.\n this.bufferedAmount = 0\n\n /**\n * @note Notify the parent about outgoing data.\n * This notifies the transport and the connection\n * listens to the outgoing data to emit the \"message\" event.\n */\n this[kOnSend]?.(data)\n })\n }\n\n public close(code: number = 1000, reason?: string): void {\n invariant(code, WEBSOCKET_CLOSE_CODE_RANGE_ERROR)\n invariant(\n code === 1000 || (code >= 3000 && code <= 4999),\n WEBSOCKET_CLOSE_CODE_RANGE_ERROR\n )\n\n if (this.readyState === this.CLOSING || this.readyState === this.CLOSED) {\n return\n }\n\n this._close(code, reason)\n }\n\n private _close(code: number = 1000, reason?: string): void {\n this.readyState = this.CLOSING\n\n queueMicrotask(() => {\n this.readyState = this.CLOSED\n\n // Non-user-configurable close status codes\n // represent connection termination and must\n // emit the \"error\" event before closing.\n if (code > 1000 && code <= 1015) {\n this.dispatchEvent(bindEvent(this, new Event('error')))\n }\n\n this.dispatchEvent(\n bindEvent(\n this,\n new CloseEvent('close', {\n code,\n reason,\n wasClean: code === 1000,\n })\n )\n )\n\n // Remove all event listeners once the socket is closed.\n this._onopen = null\n this._onmessage = null\n this._onerror = null\n this._onclose = null\n })\n }\n\n public addEventListener<K extends keyof WebSocketEventMap>(\n type: K,\n listener: (this: WebSocket, event: WebSocketEventMap[K]) => void,\n options?: boolean | AddEventListenerOptions\n ): void\n public addEventListener(\n type: string,\n listener: EventListenerOrEventListenerObject,\n options?: boolean | AddEventListenerOptions\n ): void\n public addEventListener(\n type: unknown,\n listener: unknown,\n options?: unknown\n ): void {\n return super.addEventListener(\n type as string,\n listener as EventListener,\n options as AddEventListenerOptions\n )\n }\n\n removeEventListener<K extends keyof WebSocketEventMap>(\n type: K,\n callback: EventListenerOrEventListenerObject | null,\n options?: boolean | EventListenerOptions\n ): void {\n return super.removeEventListener(type, callback, options)\n }\n}\n\nfunction getDataSize(data: WebSocketRawData): number {\n if (typeof data === 'string') {\n return data.length\n }\n\n if (data instanceof Blob) {\n return data.size\n }\n\n return data.byteLength\n}\n","import { bindEvent } from './utils/bindEvent'\nimport {\n WebSocketRawData,\n WebSocketTransport,\n WebSocketTransportOnCloseCallback,\n WebSocketTransportOnIncomingCallback,\n WebSocketTransportOnOutgoingCallback,\n} from './WebSocketTransport'\nimport { kOnSend, WebSocketOverride } from './WebSocketOverride'\n\nexport class WebSocketClassTransport extends WebSocketTransport {\n public onOutgoing: WebSocketTransportOnOutgoingCallback = () => {}\n public onIncoming: WebSocketTransportOnIncomingCallback = () => {}\n public onClose: WebSocketTransportOnCloseCallback = () => {}\n\n constructor(protected readonly ws: WebSocketOverride) {\n super()\n\n this.ws.addEventListener('close', (event) => this.onClose(event), {\n once: true,\n })\n this.ws[kOnSend] = (...args) => this.onOutgoing(...args)\n }\n\n public send(data: WebSocketRawData): void {\n queueMicrotask(() => {\n const message = bindEvent(\n /**\n * @note Setting this event's \"target\" to the\n * WebSocket override instance is important.\n * This way it can tell apart original incoming events\n * (must be forwarded to the transport) from the\n * mocked message events like the one below\n * (must be dispatched on the client instance).\n */\n this.ws,\n new MessageEvent('message', {\n data,\n origin: this.ws.url,\n })\n )\n\n this.ws.dispatchEvent(message)\n })\n }\n\n public close(code: number, reason?: string): void {\n /**\n * @note Call the internal close method directly\n * to allow closing the connection with the status codes\n * that are non-configurable by the user (> 1000 <= 1015).\n */\n this.ws['_close'](code, reason)\n }\n}\n","import { Interceptor } from '../../Interceptor'\nimport { WebSocketClientConnection } from './WebSocketClientConnection'\nimport { WebSocketServerConnection } from './WebSocketServerConnection'\nimport { WebSocketClassTransport } from './WebSocketClassTransport'\nimport { WebSocketOverride } from './WebSocketOverride'\n\nexport type { WebSocketRawData } from './WebSocketTransport'\nexport { WebSocketClientConnection, WebSocketServerConnection }\n\nexport type WebSocketEventMap = {\n connection: [\n args: {\n /**\n * The incoming WebSocket client connection.\n */\n client: WebSocketClientConnection\n\n /**\n * The original WebSocket server connection.\n */\n server: WebSocketServerConnection\n }\n ]\n}\n\n/**\n * Intercept the outgoing WebSocket connections created using\n * the global `WebSocket` class.\n */\nexport class WebSocketInterceptor extends Interceptor<WebSocketEventMap> {\n static symbol = Symbol('websocket')\n\n constructor() {\n super(WebSocketInterceptor.symbol)\n }\n\n protected checkEnvironment(): boolean {\n // Enable this interceptor in any environment\n // that has a global WebSocket API.\n return typeof globalThis.WebSocket !== 'undefined'\n }\n\n protected setup(): void {\n const webSocketProxy = Proxy.revocable(globalThis.WebSocket, {\n construct: (\n target,\n args: ConstructorParameters<typeof globalThis.WebSocket>,\n newTarget\n ) => {\n const [url, protocols] = args\n\n const createConnection = (): WebSocket => {\n return Reflect.construct(target, args, newTarget)\n }\n\n // All WebSocket instances are mocked and don't forward\n // any events to the original server (no connection established).\n // To forward the events, the user must use the \"server.send()\" API.\n const mockWs = new WebSocketOverride(url, protocols)\n const transport = new WebSocketClassTransport(mockWs)\n\n // The \"globalThis.WebSocket\" class stands for\n // the client-side connection. Assume it's established\n // as soon as the WebSocket instance is constructed.\n this.emitter.emit('connection', {\n client: new WebSocketClientConnection(mockWs, transport),\n server: new WebSocketServerConnection(\n mockWs,\n createConnection,\n transport\n ),\n })\n\n return mockWs\n },\n })\n\n globalThis.WebSocket = webSocketProxy.proxy\n\n this.subscriptions.push(() => {\n webSocketProxy.revoke()\n })\n }\n}\n"]}
|
|
@@ -0,0 +1,433 @@
|
|
|
1
|
+
import {
|
|
2
|
+
Interceptor
|
|
3
|
+
} from "../../chunk-SGO3INLV.mjs";
|
|
4
|
+
|
|
5
|
+
// src/interceptors/WebSocket/utils/bindEvent.ts
|
|
6
|
+
function bindEvent(target, event) {
|
|
7
|
+
Object.defineProperty(event, "target", {
|
|
8
|
+
enumerable: true,
|
|
9
|
+
writable: true,
|
|
10
|
+
value: target
|
|
11
|
+
});
|
|
12
|
+
return event;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
// src/interceptors/WebSocket/utils/events.ts
|
|
16
|
+
var CloseEvent = class extends Event {
|
|
17
|
+
constructor(type, init = {}) {
|
|
18
|
+
super(type, init);
|
|
19
|
+
this.code = init.code === void 0 ? 0 : init.code;
|
|
20
|
+
this.reason = init.reason === void 0 ? "" : init.reason;
|
|
21
|
+
this.wasClean = init.wasClean === void 0 ? false : init.wasClean;
|
|
22
|
+
}
|
|
23
|
+
};
|
|
24
|
+
|
|
25
|
+
// src/interceptors/WebSocket/WebSocketClientConnection.ts
|
|
26
|
+
var kEmitter = Symbol("kEmitter");
|
|
27
|
+
var WebSocketClientConnection = class {
|
|
28
|
+
constructor(ws, transport) {
|
|
29
|
+
this.ws = ws;
|
|
30
|
+
this.transport = transport;
|
|
31
|
+
this.url = new URL(ws.url);
|
|
32
|
+
this[kEmitter] = new EventTarget();
|
|
33
|
+
this.transport.onOutgoing = (data) => {
|
|
34
|
+
this[kEmitter].dispatchEvent(
|
|
35
|
+
bindEvent(this.ws, new MessageEvent("message", { data }))
|
|
36
|
+
);
|
|
37
|
+
};
|
|
38
|
+
this.transport.onClose = (event) => {
|
|
39
|
+
this[kEmitter].dispatchEvent(
|
|
40
|
+
bindEvent(this.ws, new CloseEvent("close", event))
|
|
41
|
+
);
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* Listen for the outgoing events from the connected WebSocket client.
|
|
46
|
+
*/
|
|
47
|
+
addEventListener(event, listener, options) {
|
|
48
|
+
this[kEmitter].addEventListener(event, listener, options);
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* Removes the listener for the given event.
|
|
52
|
+
*/
|
|
53
|
+
removeEventListener(event, listener, options) {
|
|
54
|
+
this[kEmitter].removeEventListener(
|
|
55
|
+
event,
|
|
56
|
+
listener,
|
|
57
|
+
options
|
|
58
|
+
);
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* Send data to the connected client.
|
|
62
|
+
*/
|
|
63
|
+
send(data) {
|
|
64
|
+
this.transport.send(data);
|
|
65
|
+
}
|
|
66
|
+
/**
|
|
67
|
+
* Close the WebSocket connection.
|
|
68
|
+
* @param {number} code A status code (see https://www.rfc-editor.org/rfc/rfc6455#section-7.4.1).
|
|
69
|
+
* @param {string} reason A custom connection close reason.
|
|
70
|
+
*/
|
|
71
|
+
close(code, reason) {
|
|
72
|
+
this.transport.close(code, reason);
|
|
73
|
+
}
|
|
74
|
+
};
|
|
75
|
+
kEmitter;
|
|
76
|
+
|
|
77
|
+
// src/interceptors/WebSocket/WebSocketServerConnection.ts
|
|
78
|
+
import { invariant } from "outvariant";
|
|
79
|
+
var kEmitter2 = Symbol("kEmitter");
|
|
80
|
+
var WebSocketServerConnection = class {
|
|
81
|
+
constructor(mockWebSocket, createConnection, transport) {
|
|
82
|
+
this.mockWebSocket = mockWebSocket;
|
|
83
|
+
this.createConnection = createConnection;
|
|
84
|
+
this.transport = transport;
|
|
85
|
+
this[kEmitter2] = new EventTarget();
|
|
86
|
+
this.transport.onIncoming = (event) => {
|
|
87
|
+
this[kEmitter2].dispatchEvent(event);
|
|
88
|
+
};
|
|
89
|
+
}
|
|
90
|
+
/**
|
|
91
|
+
* Server ready state.
|
|
92
|
+
* Proxies the ready state of the original WebSocket instance,
|
|
93
|
+
* if set. If the original connection hasn't been established,
|
|
94
|
+
* defaults to `-1`.
|
|
95
|
+
*/
|
|
96
|
+
get readyState() {
|
|
97
|
+
if (this.realWebSocket) {
|
|
98
|
+
return this.realWebSocket.readyState;
|
|
99
|
+
}
|
|
100
|
+
return -1;
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* Open connection to the original WebSocket server.
|
|
104
|
+
*/
|
|
105
|
+
connect() {
|
|
106
|
+
invariant(
|
|
107
|
+
this.readyState === -1,
|
|
108
|
+
'Failed to call "connect()" on the original WebSocket instance: the connection already open'
|
|
109
|
+
);
|
|
110
|
+
const ws = this.createConnection();
|
|
111
|
+
this.mockWebSocket.addEventListener(
|
|
112
|
+
"close",
|
|
113
|
+
(event) => {
|
|
114
|
+
ws.close(event.code, event.reason);
|
|
115
|
+
},
|
|
116
|
+
{ once: true }
|
|
117
|
+
);
|
|
118
|
+
ws.addEventListener("message", (event) => {
|
|
119
|
+
const messageEvent = bindEvent(
|
|
120
|
+
ws,
|
|
121
|
+
new MessageEvent("message", {
|
|
122
|
+
data: event.data,
|
|
123
|
+
origin: event.origin,
|
|
124
|
+
cancelable: true
|
|
125
|
+
})
|
|
126
|
+
);
|
|
127
|
+
this.transport.onIncoming(messageEvent);
|
|
128
|
+
if (!messageEvent.defaultPrevented) {
|
|
129
|
+
this.mockWebSocket.dispatchEvent(
|
|
130
|
+
bindEvent(
|
|
131
|
+
/**
|
|
132
|
+
* @note Bind the forwarded original server events
|
|
133
|
+
* to the mock WebSocket instance so it would
|
|
134
|
+
* dispatch them straight away.
|
|
135
|
+
*/
|
|
136
|
+
this.mockWebSocket,
|
|
137
|
+
// Clone the message event again to prevent
|
|
138
|
+
// the "already being dispatched" exception.
|
|
139
|
+
new MessageEvent("message", {
|
|
140
|
+
data: event.data,
|
|
141
|
+
origin: event.origin
|
|
142
|
+
})
|
|
143
|
+
)
|
|
144
|
+
);
|
|
145
|
+
}
|
|
146
|
+
});
|
|
147
|
+
this.realWebSocket = ws;
|
|
148
|
+
}
|
|
149
|
+
/**
|
|
150
|
+
* Listen for the incoming events from the original WebSocket server.
|
|
151
|
+
*/
|
|
152
|
+
addEventListener(event, listener, options) {
|
|
153
|
+
this[kEmitter2].addEventListener(
|
|
154
|
+
event,
|
|
155
|
+
listener.bind(this.realWebSocket),
|
|
156
|
+
options
|
|
157
|
+
);
|
|
158
|
+
}
|
|
159
|
+
/**
|
|
160
|
+
* Removes the listener for the given event.
|
|
161
|
+
*/
|
|
162
|
+
removeEventListener(event, listener, options) {
|
|
163
|
+
this[kEmitter2].removeEventListener(
|
|
164
|
+
event,
|
|
165
|
+
listener,
|
|
166
|
+
options
|
|
167
|
+
);
|
|
168
|
+
}
|
|
169
|
+
/**
|
|
170
|
+
* Send data to the original WebSocket server.
|
|
171
|
+
* @example
|
|
172
|
+
* server.send('hello')
|
|
173
|
+
* server.send(new Blob(['hello']))
|
|
174
|
+
* server.send(new TextEncoder().encode('hello'))
|
|
175
|
+
*/
|
|
176
|
+
send(data) {
|
|
177
|
+
const { realWebSocket } = this;
|
|
178
|
+
invariant(
|
|
179
|
+
realWebSocket,
|
|
180
|
+
'Failed to call "server.send()" for "%s": the connection is not open. Did you forget to call "await server.connect()"?',
|
|
181
|
+
this.mockWebSocket.url
|
|
182
|
+
);
|
|
183
|
+
if (realWebSocket.readyState === realWebSocket.CONNECTING) {
|
|
184
|
+
realWebSocket.addEventListener(
|
|
185
|
+
"open",
|
|
186
|
+
() => {
|
|
187
|
+
realWebSocket.send(data);
|
|
188
|
+
},
|
|
189
|
+
{ once: true }
|
|
190
|
+
);
|
|
191
|
+
return;
|
|
192
|
+
}
|
|
193
|
+
realWebSocket.send(data);
|
|
194
|
+
}
|
|
195
|
+
};
|
|
196
|
+
kEmitter2;
|
|
197
|
+
|
|
198
|
+
// src/interceptors/WebSocket/WebSocketTransport.ts
|
|
199
|
+
var WebSocketTransport = class {
|
|
200
|
+
};
|
|
201
|
+
|
|
202
|
+
// src/interceptors/WebSocket/WebSocketOverride.ts
|
|
203
|
+
import { invariant as invariant2 } from "outvariant";
|
|
204
|
+
var WEBSOCKET_CLOSE_CODE_RANGE_ERROR = "InvalidAccessError: close code out of user configurable range";
|
|
205
|
+
var kOnSend = Symbol("kOnSend");
|
|
206
|
+
var WebSocketOverride = class extends EventTarget {
|
|
207
|
+
constructor(url, protocols) {
|
|
208
|
+
super();
|
|
209
|
+
this.CONNECTING = WebSocket.CONNECTING;
|
|
210
|
+
this.OPEN = WebSocket.OPEN;
|
|
211
|
+
this.CLOSING = WebSocket.CLOSING;
|
|
212
|
+
this.CLOSED = WebSocket.CLOSED;
|
|
213
|
+
this._onopen = null;
|
|
214
|
+
this._onmessage = null;
|
|
215
|
+
this._onerror = null;
|
|
216
|
+
this._onclose = null;
|
|
217
|
+
this.url = url.toString();
|
|
218
|
+
this.protocol = "";
|
|
219
|
+
this.extensions = "";
|
|
220
|
+
this.binaryType = "blob";
|
|
221
|
+
this.readyState = this.CONNECTING;
|
|
222
|
+
this.bufferedAmount = 0;
|
|
223
|
+
Reflect.set(this, "readyState", this.CONNECTING);
|
|
224
|
+
queueMicrotask(() => {
|
|
225
|
+
Reflect.set(this, "readyState", this.OPEN);
|
|
226
|
+
this.protocol = protocols ? protocols[0] : "ws";
|
|
227
|
+
this.dispatchEvent(bindEvent(this, new Event("open")));
|
|
228
|
+
});
|
|
229
|
+
}
|
|
230
|
+
set onopen(listener) {
|
|
231
|
+
this.removeEventListener("open", this._onopen);
|
|
232
|
+
this._onopen = listener;
|
|
233
|
+
if (listener !== null) {
|
|
234
|
+
this.addEventListener("open", listener);
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
get onopen() {
|
|
238
|
+
return this._onopen;
|
|
239
|
+
}
|
|
240
|
+
set onmessage(listener) {
|
|
241
|
+
this.removeEventListener(
|
|
242
|
+
"message",
|
|
243
|
+
this._onmessage
|
|
244
|
+
);
|
|
245
|
+
this._onmessage = listener;
|
|
246
|
+
if (listener !== null) {
|
|
247
|
+
this.addEventListener("message", listener);
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
get onmessage() {
|
|
251
|
+
return this._onmessage;
|
|
252
|
+
}
|
|
253
|
+
set onerror(listener) {
|
|
254
|
+
this.removeEventListener("error", this._onerror);
|
|
255
|
+
this._onerror = listener;
|
|
256
|
+
if (listener !== null) {
|
|
257
|
+
this.addEventListener("error", listener);
|
|
258
|
+
}
|
|
259
|
+
}
|
|
260
|
+
get onerror() {
|
|
261
|
+
return this._onerror;
|
|
262
|
+
}
|
|
263
|
+
set onclose(listener) {
|
|
264
|
+
this.removeEventListener("close", this._onclose);
|
|
265
|
+
this._onclose = listener;
|
|
266
|
+
if (listener !== null) {
|
|
267
|
+
this.addEventListener("close", listener);
|
|
268
|
+
}
|
|
269
|
+
}
|
|
270
|
+
get onclose() {
|
|
271
|
+
return this._onclose;
|
|
272
|
+
}
|
|
273
|
+
/**
|
|
274
|
+
* @see https://websockets.spec.whatwg.org/#ref-for-dom-websocket-send%E2%91%A0
|
|
275
|
+
*/
|
|
276
|
+
send(data) {
|
|
277
|
+
if (this.readyState === this.CONNECTING) {
|
|
278
|
+
this.close();
|
|
279
|
+
throw new DOMException("InvalidStateError");
|
|
280
|
+
}
|
|
281
|
+
if (this.readyState === this.CLOSING || this.readyState === this.CLOSED) {
|
|
282
|
+
return;
|
|
283
|
+
}
|
|
284
|
+
this.bufferedAmount += getDataSize(data);
|
|
285
|
+
queueMicrotask(() => {
|
|
286
|
+
var _a;
|
|
287
|
+
this.bufferedAmount = 0;
|
|
288
|
+
(_a = this[kOnSend]) == null ? void 0 : _a.call(this, data);
|
|
289
|
+
});
|
|
290
|
+
}
|
|
291
|
+
close(code = 1e3, reason) {
|
|
292
|
+
invariant2(code, WEBSOCKET_CLOSE_CODE_RANGE_ERROR);
|
|
293
|
+
invariant2(
|
|
294
|
+
code === 1e3 || code >= 3e3 && code <= 4999,
|
|
295
|
+
WEBSOCKET_CLOSE_CODE_RANGE_ERROR
|
|
296
|
+
);
|
|
297
|
+
if (this.readyState === this.CLOSING || this.readyState === this.CLOSED) {
|
|
298
|
+
return;
|
|
299
|
+
}
|
|
300
|
+
this._close(code, reason);
|
|
301
|
+
}
|
|
302
|
+
_close(code = 1e3, reason) {
|
|
303
|
+
this.readyState = this.CLOSING;
|
|
304
|
+
queueMicrotask(() => {
|
|
305
|
+
this.readyState = this.CLOSED;
|
|
306
|
+
if (code > 1e3 && code <= 1015) {
|
|
307
|
+
this.dispatchEvent(bindEvent(this, new Event("error")));
|
|
308
|
+
}
|
|
309
|
+
this.dispatchEvent(
|
|
310
|
+
bindEvent(
|
|
311
|
+
this,
|
|
312
|
+
new CloseEvent("close", {
|
|
313
|
+
code,
|
|
314
|
+
reason,
|
|
315
|
+
wasClean: code === 1e3
|
|
316
|
+
})
|
|
317
|
+
)
|
|
318
|
+
);
|
|
319
|
+
this._onopen = null;
|
|
320
|
+
this._onmessage = null;
|
|
321
|
+
this._onerror = null;
|
|
322
|
+
this._onclose = null;
|
|
323
|
+
});
|
|
324
|
+
}
|
|
325
|
+
addEventListener(type, listener, options) {
|
|
326
|
+
return super.addEventListener(
|
|
327
|
+
type,
|
|
328
|
+
listener,
|
|
329
|
+
options
|
|
330
|
+
);
|
|
331
|
+
}
|
|
332
|
+
removeEventListener(type, callback, options) {
|
|
333
|
+
return super.removeEventListener(type, callback, options);
|
|
334
|
+
}
|
|
335
|
+
};
|
|
336
|
+
kOnSend;
|
|
337
|
+
WebSocketOverride.CONNECTING = WebSocket.CONNECTING;
|
|
338
|
+
WebSocketOverride.OPEN = WebSocket.OPEN;
|
|
339
|
+
WebSocketOverride.CLOSING = WebSocket.CLOSING;
|
|
340
|
+
WebSocketOverride.CLOSED = WebSocket.CLOSED;
|
|
341
|
+
function getDataSize(data) {
|
|
342
|
+
if (typeof data === "string") {
|
|
343
|
+
return data.length;
|
|
344
|
+
}
|
|
345
|
+
if (data instanceof Blob) {
|
|
346
|
+
return data.size;
|
|
347
|
+
}
|
|
348
|
+
return data.byteLength;
|
|
349
|
+
}
|
|
350
|
+
|
|
351
|
+
// src/interceptors/WebSocket/WebSocketClassTransport.ts
|
|
352
|
+
var WebSocketClassTransport = class extends WebSocketTransport {
|
|
353
|
+
constructor(ws) {
|
|
354
|
+
super();
|
|
355
|
+
this.ws = ws;
|
|
356
|
+
this.onOutgoing = () => {
|
|
357
|
+
};
|
|
358
|
+
this.onIncoming = () => {
|
|
359
|
+
};
|
|
360
|
+
this.onClose = () => {
|
|
361
|
+
};
|
|
362
|
+
this.ws.addEventListener("close", (event) => this.onClose(event), {
|
|
363
|
+
once: true
|
|
364
|
+
});
|
|
365
|
+
this.ws[kOnSend] = (...args) => this.onOutgoing(...args);
|
|
366
|
+
}
|
|
367
|
+
send(data) {
|
|
368
|
+
queueMicrotask(() => {
|
|
369
|
+
const message = bindEvent(
|
|
370
|
+
/**
|
|
371
|
+
* @note Setting this event's "target" to the
|
|
372
|
+
* WebSocket override instance is important.
|
|
373
|
+
* This way it can tell apart original incoming events
|
|
374
|
+
* (must be forwarded to the transport) from the
|
|
375
|
+
* mocked message events like the one below
|
|
376
|
+
* (must be dispatched on the client instance).
|
|
377
|
+
*/
|
|
378
|
+
this.ws,
|
|
379
|
+
new MessageEvent("message", {
|
|
380
|
+
data,
|
|
381
|
+
origin: this.ws.url
|
|
382
|
+
})
|
|
383
|
+
);
|
|
384
|
+
this.ws.dispatchEvent(message);
|
|
385
|
+
});
|
|
386
|
+
}
|
|
387
|
+
close(code, reason) {
|
|
388
|
+
this.ws["_close"](code, reason);
|
|
389
|
+
}
|
|
390
|
+
};
|
|
391
|
+
|
|
392
|
+
// src/interceptors/WebSocket/index.ts
|
|
393
|
+
var _WebSocketInterceptor = class extends Interceptor {
|
|
394
|
+
constructor() {
|
|
395
|
+
super(_WebSocketInterceptor.symbol);
|
|
396
|
+
}
|
|
397
|
+
checkEnvironment() {
|
|
398
|
+
return typeof globalThis.WebSocket !== "undefined";
|
|
399
|
+
}
|
|
400
|
+
setup() {
|
|
401
|
+
const webSocketProxy = Proxy.revocable(globalThis.WebSocket, {
|
|
402
|
+
construct: (target, args, newTarget) => {
|
|
403
|
+
const [url, protocols] = args;
|
|
404
|
+
const createConnection = () => {
|
|
405
|
+
return Reflect.construct(target, args, newTarget);
|
|
406
|
+
};
|
|
407
|
+
const mockWs = new WebSocketOverride(url, protocols);
|
|
408
|
+
const transport = new WebSocketClassTransport(mockWs);
|
|
409
|
+
this.emitter.emit("connection", {
|
|
410
|
+
client: new WebSocketClientConnection(mockWs, transport),
|
|
411
|
+
server: new WebSocketServerConnection(
|
|
412
|
+
mockWs,
|
|
413
|
+
createConnection,
|
|
414
|
+
transport
|
|
415
|
+
)
|
|
416
|
+
});
|
|
417
|
+
return mockWs;
|
|
418
|
+
}
|
|
419
|
+
});
|
|
420
|
+
globalThis.WebSocket = webSocketProxy.proxy;
|
|
421
|
+
this.subscriptions.push(() => {
|
|
422
|
+
webSocketProxy.revoke();
|
|
423
|
+
});
|
|
424
|
+
}
|
|
425
|
+
};
|
|
426
|
+
var WebSocketInterceptor = _WebSocketInterceptor;
|
|
427
|
+
WebSocketInterceptor.symbol = Symbol("websocket");
|
|
428
|
+
export {
|
|
429
|
+
WebSocketClientConnection,
|
|
430
|
+
WebSocketInterceptor,
|
|
431
|
+
WebSocketServerConnection
|
|
432
|
+
};
|
|
433
|
+
//# sourceMappingURL=index.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../../../src/interceptors/WebSocket/utils/bindEvent.ts","../../../../src/interceptors/WebSocket/utils/events.ts","../../../../src/interceptors/WebSocket/WebSocketClientConnection.ts","../../../../src/interceptors/WebSocket/WebSocketServerConnection.ts","../../../../src/interceptors/WebSocket/WebSocketTransport.ts","../../../../src/interceptors/WebSocket/WebSocketOverride.ts","../../../../src/interceptors/WebSocket/WebSocketClassTransport.ts","../../../../src/interceptors/WebSocket/index.ts"],"sourcesContent":["type EventWithTarget<E extends Event, T> = E & { target: T }\n\nexport function bindEvent<E extends Event, T>(\n target: T,\n event: E\n): EventWithTarget<E, T> {\n Object.defineProperty(event, 'target', {\n enumerable: true,\n writable: true,\n value: target,\n })\n return event as EventWithTarget<E, T>\n}\n","interface CloseEventInit extends EventInit {\n code?: number\n reason?: string\n wasClean?: boolean\n}\n\nexport class CloseEvent extends Event {\n public code: number\n public reason: string\n public wasClean: boolean\n\n constructor(type: string, init: CloseEventInit = {}) {\n super(type, init)\n this.code = init.code === undefined ? 0 : init.code\n this.reason = init.reason === undefined ? '' : init.reason\n this.wasClean = init.wasClean === undefined ? false : init.wasClean\n }\n}\n","/**\n * WebSocket client class.\n * This represents an incoming WebSocket client connection.\n * @note Keep this class implementation-agnostic because it's\n * meant to be used over any WebSocket implementation\n * (not all of them follow the one from WHATWG).\n */\nimport type { WebSocketRawData, WebSocketTransport } from './WebSocketTransport'\nimport { WebSocketMessageListener } from './WebSocketOverride'\nimport { bindEvent } from './utils/bindEvent'\nimport { CloseEvent } from './utils/events'\n\nconst kEmitter = Symbol('kEmitter')\n\n/**\n * The WebSocket client instance represents an incoming\n * client connection. The user can control the connection,\n * send and receive events.\n */\nexport class WebSocketClientConnection {\n public readonly url: URL\n\n protected [kEmitter]: EventTarget\n\n constructor(\n protected readonly ws: WebSocket,\n protected readonly transport: WebSocketTransport\n ) {\n this.url = new URL(ws.url)\n this[kEmitter] = new EventTarget()\n\n // Emit outgoing client data (\"ws.send()\") as \"message\"\n // events on the client connection.\n this.transport.onOutgoing = (data) => {\n this[kEmitter].dispatchEvent(\n bindEvent(this.ws, new MessageEvent('message', { data }))\n )\n }\n\n this.transport.onClose = (event) => {\n this[kEmitter].dispatchEvent(\n bindEvent(this.ws, new CloseEvent('close', event))\n )\n }\n }\n\n /**\n * Listen for the outgoing events from the connected WebSocket client.\n */\n public addEventListener(\n event: string,\n listener: WebSocketMessageListener,\n options?: AddEventListenerOptions | boolean\n ): void {\n this[kEmitter].addEventListener(event, listener as EventListener, options)\n }\n\n /**\n * Removes the listener for the given event.\n */\n public removeEventListener(\n event: string,\n listener: WebSocketMessageListener,\n options?: EventListenerOptions | boolean\n ): void {\n this[kEmitter].removeEventListener(\n event,\n listener as EventListener,\n options\n )\n }\n\n /**\n * Send data to the connected client.\n */\n public send(data: WebSocketRawData): void {\n this.transport.send(data)\n }\n\n /**\n * Close the WebSocket connection.\n * @param {number} code A status code (see https://www.rfc-editor.org/rfc/rfc6455#section-7.4.1).\n * @param {string} reason A custom connection close reason.\n */\n public close(code?: number, reason?: string): void {\n this.transport.close(code, reason)\n }\n}\n","import { invariant } from 'outvariant'\nimport type { WebSocketOverride } from './WebSocketOverride'\nimport type { WebSocketRawData } from './WebSocketTransport'\nimport type { WebSocketClassTransport } from './WebSocketClassTransport'\nimport { bindEvent } from './utils/bindEvent'\nimport { CloseEvent } from './utils/events'\n\nconst kEmitter = Symbol('kEmitter')\n\n/**\n * The WebSocket server instance represents the actual production\n * WebSocket server connection. It's idle by default but you can\n * establish it by calling `server.connect()`.\n */\nexport class WebSocketServerConnection {\n /**\n * A WebSocket instance connected to the original server.\n */\n private realWebSocket?: WebSocket\n private [kEmitter]: EventTarget\n\n constructor(\n private readonly mockWebSocket: WebSocketOverride,\n private readonly createConnection: () => WebSocket,\n private readonly transport: WebSocketClassTransport\n ) {\n this[kEmitter] = new EventTarget()\n\n // Handle incoming events from the actual server.\n // The (mock) WebSocket instance will call this\n // whenever a \"message\" event from the actual server\n // is dispatched on it (the dispatch will be skipped).\n this.transport.onIncoming = (event) => {\n /**\n * @note Emit \"message\" event on the WebSocketClassServer\n * instance to let the interceptor know about these\n * incoming events from the original server. In that listener,\n * the interceptor can modify or skip the event forwarding\n * to the mock WebSocket instance.\n */\n this[kEmitter].dispatchEvent(event)\n }\n }\n\n /**\n * Server ready state.\n * Proxies the ready state of the original WebSocket instance,\n * if set. If the original connection hasn't been established,\n * defaults to `-1`.\n */\n public get readyState(): number {\n if (this.realWebSocket) {\n return this.realWebSocket.readyState\n }\n\n return -1\n }\n\n /**\n * Open connection to the original WebSocket server.\n */\n public connect(): void {\n invariant(\n this.readyState === -1,\n 'Failed to call \"connect()\" on the original WebSocket instance: the connection already open'\n )\n\n const ws = this.createConnection()\n\n // Close the original connection when the (mock)\n // client closes, regardless of the reason.\n this.mockWebSocket.addEventListener(\n 'close',\n (event) => {\n ws.close(event.code, event.reason)\n },\n { once: true }\n )\n\n // Once the connection is open, forward any incoming\n // events directly to the (override) WebSocket instance.\n ws.addEventListener('message', (event) => {\n // Clone the event to dispatch it on this class\n // once again and prevent the \"already being dispatched\"\n // exception. Clone it here so we can observe this event\n // being prevented in the \"server.on()\" listeners.\n const messageEvent = bindEvent(\n ws,\n new MessageEvent('message', {\n data: event.data,\n origin: event.origin,\n cancelable: true,\n })\n )\n\n this.transport.onIncoming(messageEvent)\n\n // Unless the default is prevented, forward the\n // messages from the original server to the mock client.\n // This is the only way the user can receive them.\n if (!messageEvent.defaultPrevented) {\n this.mockWebSocket.dispatchEvent(\n bindEvent(\n /**\n * @note Bind the forwarded original server events\n * to the mock WebSocket instance so it would\n * dispatch them straight away.\n */\n this.mockWebSocket,\n // Clone the message event again to prevent\n // the \"already being dispatched\" exception.\n new MessageEvent('message', {\n data: event.data,\n origin: event.origin,\n })\n )\n )\n }\n })\n\n this.realWebSocket = ws\n }\n\n /**\n * Listen for the incoming events from the original WebSocket server.\n */\n public addEventListener<K extends keyof WebSocketEventMap>(\n event: K,\n listener: (this: WebSocket, event: WebSocketEventMap[K]) => void,\n options?: AddEventListenerOptions | boolean\n ): void {\n this[kEmitter].addEventListener(\n event,\n listener.bind(this.realWebSocket!) as EventListener,\n options\n )\n }\n\n /**\n * Removes the listener for the given event.\n */\n public removeEventListener<K extends keyof WebSocketEventMap>(\n event: K,\n listener: (this: WebSocket, event: WebSocketEventMap[K]) => void,\n options?: EventListenerOptions | boolean\n ): void {\n this[kEmitter].removeEventListener(\n event,\n listener as EventListener,\n options\n )\n }\n\n /**\n * Send data to the original WebSocket server.\n * @example\n * server.send('hello')\n * server.send(new Blob(['hello']))\n * server.send(new TextEncoder().encode('hello'))\n */\n public send(data: WebSocketRawData): void {\n const { realWebSocket } = this\n invariant(\n realWebSocket,\n 'Failed to call \"server.send()\" for \"%s\": the connection is not open. Did you forget to call \"await server.connect()\"?',\n this.mockWebSocket.url\n )\n\n // Delegate the send to when the original connection is open.\n // Unlike the mock, connecting to the original server may take time\n // so we cannot call this on the next tick.\n if (realWebSocket.readyState === realWebSocket.CONNECTING) {\n realWebSocket.addEventListener(\n 'open',\n () => {\n realWebSocket.send(data)\n },\n { once: true }\n )\n return\n }\n\n // Send the data to the original WebSocket server.\n realWebSocket.send(data)\n }\n}\n","export type WebSocketRawData = string | ArrayBufferLike | Blob | ArrayBufferView\n\nexport type WebSocketTransportOnIncomingCallback = (\n event: MessageEvent<WebSocketRawData>\n) => void\n\nexport type WebSocketTransportOnOutgoingCallback = (\n data: WebSocketRawData\n) => void\n\nexport type WebSocketTransportOnCloseCallback = (event: CloseEvent) => void\n\nexport abstract class WebSocketTransport {\n /**\n * Listener for the incoming server events.\n * This is called when the client receives the\n * event from the original server connection.\n *\n * This way, we can trigger the \"message\" event\n * on the mocked connection to let the user know.\n */\n abstract onIncoming: WebSocketTransportOnIncomingCallback\n abstract onOutgoing: WebSocketTransportOnOutgoingCallback\n abstract onClose: WebSocketTransportOnCloseCallback\n\n /**\n * Send the data from the server to this client.\n */\n abstract send(data: WebSocketRawData): void\n\n /**\n * Close the client connection.\n */\n abstract close(code?: number, reason?: string): void\n}\n","import { invariant } from 'outvariant'\nimport type { WebSocketRawData } from './WebSocketTransport'\nimport { bindEvent } from './utils/bindEvent'\nimport { CloseEvent } from './utils/events'\n\ntype WebSocketEventListener = (this: WebSocket, event: Event) => void\n\nexport type WebSocketMessageListener = (\n this: WebSocket,\n event: MessageEvent\n) => void\n\ntype WebSocketCloseListener = (this: WebSocket, event: CloseEvent) => void\n\nconst WEBSOCKET_CLOSE_CODE_RANGE_ERROR =\n 'InvalidAccessError: close code out of user configurable range'\n\nexport const kOnSend = Symbol('kOnSend')\n\nexport class WebSocketOverride extends EventTarget implements WebSocket {\n static readonly CONNECTING = WebSocket.CONNECTING\n static readonly OPEN = WebSocket.OPEN\n static readonly CLOSING = WebSocket.CLOSING\n static readonly CLOSED = WebSocket.CLOSED\n readonly CONNECTING = WebSocket.CONNECTING\n readonly OPEN = WebSocket.OPEN\n readonly CLOSING = WebSocket.CLOSING\n readonly CLOSED = WebSocket.CLOSED\n\n public url: string\n public protocol: string\n public extensions: string\n public binaryType: BinaryType\n public readyState: number\n public bufferedAmount: number\n\n private _onopen: WebSocketEventListener | null = null\n private _onmessage: WebSocketMessageListener | null = null\n private _onerror: WebSocketEventListener | null = null\n private _onclose: WebSocketCloseListener | null = null\n\n private [kOnSend]?: (data: WebSocketRawData) => void\n\n constructor(url: string | URL, protocols?: string | Array<string>) {\n super()\n this.url = url.toString()\n this.protocol = ''\n this.extensions = ''\n this.binaryType = 'blob'\n this.readyState = this.CONNECTING\n this.bufferedAmount = 0\n\n Reflect.set(this, 'readyState', this.CONNECTING)\n queueMicrotask(() => {\n Reflect.set(this, 'readyState', this.OPEN)\n this.protocol = protocols ? protocols[0] : 'ws'\n\n this.dispatchEvent(bindEvent(this, new Event('open')))\n })\n }\n\n set onopen(listener: WebSocketEventListener | null) {\n this.removeEventListener('open', this._onopen)\n this._onopen = listener\n if (listener !== null) {\n this.addEventListener('open', listener)\n }\n }\n get onopen(): WebSocketEventListener | null {\n return this._onopen\n }\n\n set onmessage(listener: WebSocketMessageListener | null) {\n this.removeEventListener(\n 'message',\n this._onmessage as WebSocketEventListener\n )\n this._onmessage = listener\n if (listener !== null) {\n this.addEventListener('message', listener)\n }\n }\n get onmessage(): WebSocketMessageListener | null {\n return this._onmessage\n }\n\n set onerror(listener: WebSocketEventListener | null) {\n this.removeEventListener('error', this._onerror)\n this._onerror = listener\n if (listener !== null) {\n this.addEventListener('error', listener)\n }\n }\n get onerror(): WebSocketEventListener | null {\n return this._onerror\n }\n\n set onclose(listener: WebSocketCloseListener | null) {\n this.removeEventListener('close', this._onclose as WebSocketEventListener)\n this._onclose = listener\n if (listener !== null) {\n this.addEventListener('close', listener)\n }\n }\n get onclose(): WebSocketCloseListener | null {\n return this._onclose\n }\n\n /**\n * @see https://websockets.spec.whatwg.org/#ref-for-dom-websocket-send%E2%91%A0\n */\n public send(data: WebSocketRawData): void {\n if (this.readyState === this.CONNECTING) {\n this.close()\n throw new DOMException('InvalidStateError')\n }\n\n // Sending when the socket is about to close\n // discards the sent data.\n if (this.readyState === this.CLOSING || this.readyState === this.CLOSED) {\n return\n }\n\n // Buffer the data to send in this even loop\n // but send it in the next.\n this.bufferedAmount += getDataSize(data)\n\n queueMicrotask(() => {\n // This is a bit optimistic but since no actual data transfer\n // is involved, all the data will be \"sent\" on the next tick.\n this.bufferedAmount = 0\n\n /**\n * @note Notify the parent about outgoing data.\n * This notifies the transport and the connection\n * listens to the outgoing data to emit the \"message\" event.\n */\n this[kOnSend]?.(data)\n })\n }\n\n public close(code: number = 1000, reason?: string): void {\n invariant(code, WEBSOCKET_CLOSE_CODE_RANGE_ERROR)\n invariant(\n code === 1000 || (code >= 3000 && code <= 4999),\n WEBSOCKET_CLOSE_CODE_RANGE_ERROR\n )\n\n if (this.readyState === this.CLOSING || this.readyState === this.CLOSED) {\n return\n }\n\n this._close(code, reason)\n }\n\n private _close(code: number = 1000, reason?: string): void {\n this.readyState = this.CLOSING\n\n queueMicrotask(() => {\n this.readyState = this.CLOSED\n\n // Non-user-configurable close status codes\n // represent connection termination and must\n // emit the \"error\" event before closing.\n if (code > 1000 && code <= 1015) {\n this.dispatchEvent(bindEvent(this, new Event('error')))\n }\n\n this.dispatchEvent(\n bindEvent(\n this,\n new CloseEvent('close', {\n code,\n reason,\n wasClean: code === 1000,\n })\n )\n )\n\n // Remove all event listeners once the socket is closed.\n this._onopen = null\n this._onmessage = null\n this._onerror = null\n this._onclose = null\n })\n }\n\n public addEventListener<K extends keyof WebSocketEventMap>(\n type: K,\n listener: (this: WebSocket, event: WebSocketEventMap[K]) => void,\n options?: boolean | AddEventListenerOptions\n ): void\n public addEventListener(\n type: string,\n listener: EventListenerOrEventListenerObject,\n options?: boolean | AddEventListenerOptions\n ): void\n public addEventListener(\n type: unknown,\n listener: unknown,\n options?: unknown\n ): void {\n return super.addEventListener(\n type as string,\n listener as EventListener,\n options as AddEventListenerOptions\n )\n }\n\n removeEventListener<K extends keyof WebSocketEventMap>(\n type: K,\n callback: EventListenerOrEventListenerObject | null,\n options?: boolean | EventListenerOptions\n ): void {\n return super.removeEventListener(type, callback, options)\n }\n}\n\nfunction getDataSize(data: WebSocketRawData): number {\n if (typeof data === 'string') {\n return data.length\n }\n\n if (data instanceof Blob) {\n return data.size\n }\n\n return data.byteLength\n}\n","import { bindEvent } from './utils/bindEvent'\nimport {\n WebSocketRawData,\n WebSocketTransport,\n WebSocketTransportOnCloseCallback,\n WebSocketTransportOnIncomingCallback,\n WebSocketTransportOnOutgoingCallback,\n} from './WebSocketTransport'\nimport { kOnSend, WebSocketOverride } from './WebSocketOverride'\n\nexport class WebSocketClassTransport extends WebSocketTransport {\n public onOutgoing: WebSocketTransportOnOutgoingCallback = () => {}\n public onIncoming: WebSocketTransportOnIncomingCallback = () => {}\n public onClose: WebSocketTransportOnCloseCallback = () => {}\n\n constructor(protected readonly ws: WebSocketOverride) {\n super()\n\n this.ws.addEventListener('close', (event) => this.onClose(event), {\n once: true,\n })\n this.ws[kOnSend] = (...args) => this.onOutgoing(...args)\n }\n\n public send(data: WebSocketRawData): void {\n queueMicrotask(() => {\n const message = bindEvent(\n /**\n * @note Setting this event's \"target\" to the\n * WebSocket override instance is important.\n * This way it can tell apart original incoming events\n * (must be forwarded to the transport) from the\n * mocked message events like the one below\n * (must be dispatched on the client instance).\n */\n this.ws,\n new MessageEvent('message', {\n data,\n origin: this.ws.url,\n })\n )\n\n this.ws.dispatchEvent(message)\n })\n }\n\n public close(code: number, reason?: string): void {\n /**\n * @note Call the internal close method directly\n * to allow closing the connection with the status codes\n * that are non-configurable by the user (> 1000 <= 1015).\n */\n this.ws['_close'](code, reason)\n }\n}\n","import { Interceptor } from '../../Interceptor'\nimport { WebSocketClientConnection } from './WebSocketClientConnection'\nimport { WebSocketServerConnection } from './WebSocketServerConnection'\nimport { WebSocketClassTransport } from './WebSocketClassTransport'\nimport { WebSocketOverride } from './WebSocketOverride'\n\nexport type { WebSocketRawData } from './WebSocketTransport'\nexport { WebSocketClientConnection, WebSocketServerConnection }\n\nexport type WebSocketEventMap = {\n connection: [\n args: {\n /**\n * The incoming WebSocket client connection.\n */\n client: WebSocketClientConnection\n\n /**\n * The original WebSocket server connection.\n */\n server: WebSocketServerConnection\n }\n ]\n}\n\n/**\n * Intercept the outgoing WebSocket connections created using\n * the global `WebSocket` class.\n */\nexport class WebSocketInterceptor extends Interceptor<WebSocketEventMap> {\n static symbol = Symbol('websocket')\n\n constructor() {\n super(WebSocketInterceptor.symbol)\n }\n\n protected checkEnvironment(): boolean {\n // Enable this interceptor in any environment\n // that has a global WebSocket API.\n return typeof globalThis.WebSocket !== 'undefined'\n }\n\n protected setup(): void {\n const webSocketProxy = Proxy.revocable(globalThis.WebSocket, {\n construct: (\n target,\n args: ConstructorParameters<typeof globalThis.WebSocket>,\n newTarget\n ) => {\n const [url, protocols] = args\n\n const createConnection = (): WebSocket => {\n return Reflect.construct(target, args, newTarget)\n }\n\n // All WebSocket instances are mocked and don't forward\n // any events to the original server (no connection established).\n // To forward the events, the user must use the \"server.send()\" API.\n const mockWs = new WebSocketOverride(url, protocols)\n const transport = new WebSocketClassTransport(mockWs)\n\n // The \"globalThis.WebSocket\" class stands for\n // the client-side connection. Assume it's established\n // as soon as the WebSocket instance is constructed.\n this.emitter.emit('connection', {\n client: new WebSocketClientConnection(mockWs, transport),\n server: new WebSocketServerConnection(\n mockWs,\n createConnection,\n transport\n ),\n })\n\n return mockWs\n },\n })\n\n globalThis.WebSocket = webSocketProxy.proxy\n\n this.subscriptions.push(() => {\n webSocketProxy.revoke()\n })\n }\n}\n"],"mappings":"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","names":["kEmitter","invariant","invariant"]}
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { Emitter } from 'strict-event-emitter';
|
|
2
|
-
import {
|
|
2
|
+
import { a as InteractiveRequest, H as HttpRequestEventMap } from '../../glossary-640c9679.js';
|
|
3
|
+
import { I as Interceptor } from '../../Interceptor-b7c08a9f.js';
|
|
3
4
|
import '@open-draft/deferred-promise';
|
|
4
5
|
import '@open-draft/logger';
|
|
5
6
|
|
|
@@ -1,10 +1,11 @@
|
|
|
1
1
|
"use strict";Object.defineProperty(exports, "__esModule", {value: true});
|
|
2
2
|
|
|
3
|
-
var
|
|
3
|
+
var _chunkKWMV32LIjs = require('../../chunk-KWMV32LI.js');
|
|
4
4
|
require('../../chunk-OJ2CN4LS.js');
|
|
5
5
|
require('../../chunk-QMV2V6GV.js');
|
|
6
|
-
require('../../chunk-
|
|
6
|
+
require('../../chunk-FZJKKO5H.js');
|
|
7
|
+
require('../../chunk-EM6NYHQV.js');
|
|
7
8
|
|
|
8
9
|
|
|
9
|
-
exports.XMLHttpRequestInterceptor =
|
|
10
|
+
exports.XMLHttpRequestInterceptor = _chunkKWMV32LIjs.XMLHttpRequestInterceptor;
|
|
10
11
|
//# sourceMappingURL=index.js.map
|
|
@@ -1,9 +1,10 @@
|
|
|
1
1
|
import {
|
|
2
2
|
XMLHttpRequestInterceptor
|
|
3
|
-
} from "../../chunk-
|
|
3
|
+
} from "../../chunk-7FP6PMUG.mjs";
|
|
4
4
|
import "../../chunk-UJZOJSMP.mjs";
|
|
5
5
|
import "../../chunk-72HT65NX.mjs";
|
|
6
|
-
import "../../chunk-
|
|
6
|
+
import "../../chunk-HAGW22AN.mjs";
|
|
7
|
+
import "../../chunk-SGO3INLV.mjs";
|
|
7
8
|
export {
|
|
8
9
|
XMLHttpRequestInterceptor
|
|
9
10
|
};
|
|
@@ -1,4 +1,5 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { H as HttpRequestEventMap } from '../../glossary-640c9679.js';
|
|
2
|
+
import { I as Interceptor } from '../../Interceptor-b7c08a9f.js';
|
|
2
3
|
import '@open-draft/deferred-promise';
|
|
3
4
|
import '@open-draft/logger';
|
|
4
5
|
import 'strict-event-emitter';
|
|
@@ -1,9 +1,10 @@
|
|
|
1
1
|
"use strict";Object.defineProperty(exports, "__esModule", {value: true});
|
|
2
2
|
|
|
3
|
-
var
|
|
3
|
+
var _chunkNNNHH4BNjs = require('../../chunk-NNNHH4BN.js');
|
|
4
4
|
require('../../chunk-QMV2V6GV.js');
|
|
5
|
-
require('../../chunk-
|
|
5
|
+
require('../../chunk-FZJKKO5H.js');
|
|
6
|
+
require('../../chunk-EM6NYHQV.js');
|
|
6
7
|
|
|
7
8
|
|
|
8
|
-
exports.FetchInterceptor =
|
|
9
|
+
exports.FetchInterceptor = _chunkNNNHH4BNjs.FetchInterceptor;
|
|
9
10
|
//# sourceMappingURL=index.js.map
|
|
@@ -1,8 +1,9 @@
|
|
|
1
1
|
import {
|
|
2
2
|
FetchInterceptor
|
|
3
|
-
} from "../../chunk-
|
|
3
|
+
} from "../../chunk-E5YM6K3N.mjs";
|
|
4
4
|
import "../../chunk-72HT65NX.mjs";
|
|
5
|
-
import "../../chunk-
|
|
5
|
+
import "../../chunk-HAGW22AN.mjs";
|
|
6
|
+
import "../../chunk-SGO3INLV.mjs";
|
|
6
7
|
export {
|
|
7
8
|
FetchInterceptor
|
|
8
9
|
};
|
|
@@ -1,7 +1,8 @@
|
|
|
1
1
|
import { FetchInterceptor } from '../interceptors/fetch/index.js';
|
|
2
2
|
import { XMLHttpRequestInterceptor } from '../interceptors/XMLHttpRequest/index.js';
|
|
3
|
-
import '../
|
|
3
|
+
import '../glossary-640c9679.js';
|
|
4
4
|
import '@open-draft/deferred-promise';
|
|
5
|
+
import '../Interceptor-b7c08a9f.js';
|
|
5
6
|
import '@open-draft/logger';
|
|
6
7
|
import 'strict-event-emitter';
|
|
7
8
|
|