@msw/playwright 0.4.2 → 0.4.3

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/build/index.js CHANGED
@@ -40,7 +40,7 @@ var NetworkFixture = class extends SetupApi {
40
40
  this.#page = args.page;
41
41
  }
42
42
  async start() {
43
- await this.#page.route(/.+/, async (route, request) => {
43
+ const httpRequestListener = async (route, request) => {
44
44
  const fetchRequest = new Request(request.url(), {
45
45
  method: request.method(),
46
46
  headers: new Headers(await request.allHeaders()),
@@ -62,7 +62,8 @@ var NetworkFixture = class extends SetupApi {
62
62
  return;
63
63
  }
64
64
  route.continue();
65
- });
65
+ };
66
+ await this.#page.route(/.+/, httpRequestListener);
66
67
  await this.#page.routeWebSocket(/.+/, async (ws) => {
67
68
  const allWebSocketHandlers = this.handlersController.currentHandlers().filter((handler) => {
68
69
  return handler instanceof WebSocketHandler;
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "type": "module",
3
3
  "name": "@msw/playwright",
4
- "version": "0.4.2",
4
+ "version": "0.4.3",
5
5
  "description": "Mock Service Worker binding for Playwright",
6
6
  "main": "./build/index.js",
7
7
  "types": "./build/index.d.ts",
@@ -20,8 +20,7 @@
20
20
  "funding": "https://github.com/sponsors/mswjs",
21
21
  "homepage": "https://mswjs.io",
22
22
  "repository": {
23
- "type": "git",
24
- "url": "git+https://github.com/mswjs/playwright.git"
23
+ "url": "https://github.com/mswjs/playwright"
25
24
  },
26
25
  "author": {
27
26
  "name": "Artem Zakharchenko",
@@ -38,13 +37,13 @@
38
37
  },
39
38
  "devDependencies": {
40
39
  "@epic-web/test-server": "^0.1.6",
41
- "@ossjs/release": "^0.8.1",
42
- "@playwright/test": "^1.52.0",
40
+ "@ossjs/release": "^0.10.1",
41
+ "@playwright/test": "^1.57.0",
43
42
  "@types/node": "^22.15.29",
44
43
  "msw": "^2.10.3",
45
44
  "tsdown": "^0.12.7",
46
45
  "typescript": "^5.8.3",
47
- "vite": "^7.0.2"
46
+ "vite": "^7.3.1"
48
47
  },
49
48
  "dependencies": {
50
49
  "@mswjs/interceptors": "^0.39.2",
package/src/fixture.ts ADDED
@@ -0,0 +1,411 @@
1
+ import { invariant } from 'outvariant'
2
+ import type {
3
+ Page,
4
+ PlaywrightTestArgs,
5
+ PlaywrightWorkerArgs,
6
+ Request,
7
+ Route,
8
+ TestFixture,
9
+ WebSocketRoute,
10
+ } from '@playwright/test'
11
+ import {
12
+ type LifeCycleEventsMap,
13
+ SetupApi,
14
+ RequestHandler,
15
+ WebSocketHandler,
16
+ getResponse,
17
+ } from 'msw'
18
+ import {
19
+ type WebSocketClientEventMap,
20
+ type WebSocketData,
21
+ type WebSocketServerEventMap,
22
+ CancelableMessageEvent,
23
+ CancelableCloseEvent,
24
+ WebSocketClientConnectionProtocol,
25
+ WebSocketServerConnectionProtocol,
26
+ } from '@mswjs/interceptors/WebSocket'
27
+
28
+ export interface CreateNetworkFixtureArgs {
29
+ initialHandlers: Array<RequestHandler | WebSocketHandler>
30
+ }
31
+
32
+ /**
33
+ * Creates a fixture that controls the network in your tests.
34
+ *
35
+ * @note The returned fixture already has the `auto` option set to `true`.
36
+ *
37
+ * **Usage**
38
+ * ```ts
39
+ * import { test as testBase } from '@playwright/test'
40
+ * import { createNetworkFixture, type WorkerFixture } from '@msw/playwright'
41
+ *
42
+ * interface Fixtures {
43
+ * network: WorkerFixture
44
+ * }
45
+ *
46
+ * export const test = testBase.extend<Fixtures>({
47
+ * network: createNetworkFixture()
48
+ * })
49
+ * ```
50
+ */
51
+ export function createNetworkFixture(
52
+ args?: CreateNetworkFixtureArgs,
53
+ /** @todo `onUnhandledRequest`? */
54
+ ): [
55
+ TestFixture<NetworkFixture, PlaywrightTestArgs & PlaywrightWorkerArgs>,
56
+ { auto: boolean },
57
+ ] {
58
+ return [
59
+ async ({ page }, use) => {
60
+ const worker = new NetworkFixture({
61
+ page,
62
+ initialHandlers: args?.initialHandlers || [],
63
+ })
64
+
65
+ await worker.start()
66
+ await use(worker)
67
+ await worker.stop()
68
+ },
69
+ { auto: true },
70
+ ]
71
+ }
72
+
73
+ /**
74
+ * @note Use a match-all RegExp with an optional group as the predicate
75
+ * for the `page.route()`/`page.unroute()` calls. Playwright treats given RegExp
76
+ * as the handler ID, which allows us to remove only those handlers introduces by us
77
+ * without carrying the reference to the handler function around.
78
+ */
79
+ export const INTERNAL_MATCH_ALL_REG_EXP = /.+(__MSW_PLAYWRIGHT_PREDICATE__)?/
80
+
81
+ export class NetworkFixture extends SetupApi<LifeCycleEventsMap> {
82
+ #page: Page
83
+
84
+ constructor(args: {
85
+ page: Page
86
+ initialHandlers: Array<RequestHandler | WebSocketHandler>
87
+ }) {
88
+ super(...args.initialHandlers)
89
+ this.#page = args.page
90
+ }
91
+
92
+ public async start(): Promise<void> {
93
+ // Handle HTTP requests.
94
+ await this.#page.route(
95
+ INTERNAL_MATCH_ALL_REG_EXP,
96
+ async (route: Route, request: Request) => {
97
+ const fetchRequest = new Request(request.url(), {
98
+ method: request.method(),
99
+ headers: new Headers(await request.allHeaders()),
100
+ body: request.postDataBuffer(),
101
+ })
102
+
103
+ const response = await getResponse(
104
+ this.handlersController.currentHandlers().filter((handler) => {
105
+ return handler instanceof RequestHandler
106
+ }),
107
+ fetchRequest,
108
+ {
109
+ baseUrl: this.getPageUrl(),
110
+ },
111
+ )
112
+
113
+ if (response) {
114
+ if (response.status === 0) {
115
+ route.abort()
116
+ return
117
+ }
118
+
119
+ route.fulfill({
120
+ status: response.status,
121
+ headers: Object.fromEntries(response.headers),
122
+ body: response.body
123
+ ? Buffer.from(await response.arrayBuffer())
124
+ : undefined,
125
+ })
126
+ return
127
+ }
128
+
129
+ route.continue()
130
+ },
131
+ )
132
+
133
+ // Handle WebSocket connections.
134
+ await this.#page.routeWebSocket(INTERNAL_MATCH_ALL_REG_EXP, async (ws) => {
135
+ const allWebSocketHandlers = this.handlersController
136
+ .currentHandlers()
137
+ .filter((handler) => {
138
+ return handler instanceof WebSocketHandler
139
+ })
140
+
141
+ if (allWebSocketHandlers.length === 0) {
142
+ ws.connectToServer()
143
+ return
144
+ }
145
+
146
+ const client = new PlaywrightWebSocketClientConnection(ws)
147
+ const server = new PlaywrightWebSocketServerConnection(ws)
148
+
149
+ for (const handler of allWebSocketHandlers) {
150
+ await handler.run(
151
+ {
152
+ client,
153
+ server,
154
+ info: { protocols: [] },
155
+ },
156
+ {
157
+ baseUrl: this.getPageUrl(),
158
+ },
159
+ )
160
+ }
161
+ })
162
+ }
163
+
164
+ public async stop(): Promise<void> {
165
+ super.dispose()
166
+ await this.#page.unroute(INTERNAL_MATCH_ALL_REG_EXP)
167
+ await unrouteWebSocket(this.#page, INTERNAL_MATCH_ALL_REG_EXP)
168
+ }
169
+
170
+ private getPageUrl(): string | undefined {
171
+ const url = this.#page.url()
172
+ return url !== 'about:blank' ? url : undefined
173
+ }
174
+ }
175
+
176
+ class PlaywrightWebSocketClientConnection
177
+ implements WebSocketClientConnectionProtocol
178
+ {
179
+ public id: string
180
+ public url: URL
181
+
182
+ constructor(protected readonly ws: WebSocketRoute) {
183
+ this.id = crypto.randomUUID()
184
+ this.url = new URL(ws.url())
185
+ }
186
+
187
+ public send(data: WebSocketData): void {
188
+ if (data instanceof Blob) {
189
+ /**
190
+ * @note Playwright does not support sending Blob data.
191
+ * Read the blob as buffer, then send the buffer instead.
192
+ */
193
+ data.bytes().then((bytes) => {
194
+ this.ws.send(Buffer.from(bytes))
195
+ })
196
+ return
197
+ }
198
+
199
+ if (typeof data === 'string') {
200
+ this.ws.send(data)
201
+ return
202
+ }
203
+
204
+ this.ws.send(
205
+ /**
206
+ * @note Forcefully cast all data to Buffer because Playwright
207
+ * has trouble digesting ArrayBuffer and Blob directly.
208
+ */
209
+ Buffer.from(
210
+ /**
211
+ * @note Playwright type definitions are tailored to Node.js
212
+ * while MSW describes all data types that can be sent over
213
+ * the WebSocket protocol, like ArrayBuffer and Blob.
214
+ */
215
+ data as any,
216
+ ),
217
+ )
218
+ }
219
+
220
+ public close(code?: number, reason?: string): void {
221
+ const resolvedCode = code ?? 1000
222
+ this.ws.close({ code: resolvedCode, reason })
223
+ }
224
+
225
+ public addEventListener<EventType extends keyof WebSocketClientEventMap>(
226
+ type: EventType,
227
+ listener: (
228
+ this: WebSocket,
229
+ event: WebSocketClientEventMap[EventType],
230
+ ) => void,
231
+ options?: AddEventListenerOptions | boolean,
232
+ ): void {
233
+ /**
234
+ * @note Playwright does not expose the actual WebSocket reference.
235
+ */
236
+ const target = {} as WebSocket
237
+
238
+ switch (type) {
239
+ case 'message': {
240
+ this.ws.onMessage((data) => {
241
+ listener.call(
242
+ target,
243
+ new CancelableMessageEvent('message', {
244
+ data,
245
+ }) as any,
246
+ )
247
+ })
248
+ break
249
+ }
250
+
251
+ case 'close': {
252
+ this.ws.onClose((code, reason) => {
253
+ listener.call(
254
+ target,
255
+ new CancelableCloseEvent('close', {
256
+ code,
257
+ reason,
258
+ }) as any,
259
+ )
260
+ })
261
+ break
262
+ }
263
+ }
264
+ }
265
+
266
+ public removeEventListener<EventType extends keyof WebSocketClientEventMap>(
267
+ event: EventType,
268
+ listener: (
269
+ this: WebSocket,
270
+ event: WebSocketClientEventMap[EventType],
271
+ ) => void,
272
+ options?: EventListenerOptions | boolean,
273
+ ): void {
274
+ console.warn(
275
+ '@msw/playwright: WebSocketRoute does not support removing event listeners',
276
+ )
277
+ }
278
+ }
279
+
280
+ class PlaywrightWebSocketServerConnection
281
+ implements WebSocketServerConnectionProtocol
282
+ {
283
+ #server?: WebSocketRoute
284
+ #bufferedEvents: Array<
285
+ Parameters<WebSocketServerConnectionProtocol['addEventListener']>
286
+ >
287
+ #bufferedData: Array<WebSocketData>
288
+
289
+ constructor(protected readonly ws: WebSocketRoute) {
290
+ this.#bufferedEvents = []
291
+ this.#bufferedData = []
292
+ }
293
+
294
+ public connect(): void {
295
+ this.#server = this.ws.connectToServer()
296
+
297
+ /**
298
+ * @note Playwright does not support event buffering.
299
+ * Manually add event listeners that might have been registered
300
+ * before `connect()` was called.
301
+ */
302
+ for (const [type, listener, options] of this.#bufferedEvents) {
303
+ this.addEventListener(type, listener, options)
304
+ }
305
+ this.#bufferedEvents.length = 0
306
+
307
+ // Same for the buffered data.
308
+ for (const data of this.#bufferedData) {
309
+ this.send(data)
310
+ }
311
+ this.#bufferedData.length = 0
312
+ }
313
+
314
+ public send(data: WebSocketData): void {
315
+ if (this.#server == null) {
316
+ this.#bufferedData.push(data)
317
+ return
318
+ }
319
+
320
+ this.#server.send(data as any)
321
+ }
322
+
323
+ public close(code?: number, reason?: string): void {
324
+ invariant(
325
+ this.#server,
326
+ 'Failed to close connection to the actual WebSocket server: connection not established. Did you forget to call `connect()`?',
327
+ )
328
+
329
+ this.#server.close({ code, reason })
330
+ }
331
+
332
+ public addEventListener<EventType extends keyof WebSocketServerEventMap>(
333
+ type: EventType,
334
+ listener: (
335
+ this: WebSocket,
336
+ event: WebSocketServerEventMap[EventType],
337
+ ) => void,
338
+ options?: AddEventListenerOptions | boolean,
339
+ ): void {
340
+ if (this.#server == null) {
341
+ this.#bufferedEvents.push([type, listener as any, options])
342
+ return
343
+ }
344
+
345
+ const target = {} as WebSocket
346
+ switch (type) {
347
+ case 'message': {
348
+ this.#server.onMessage((data) => {
349
+ listener.call(
350
+ target,
351
+ new CancelableMessageEvent('message', { data }) as any,
352
+ )
353
+ })
354
+ break
355
+ }
356
+
357
+ case 'close': {
358
+ this.#server.onClose((code, reason) => {
359
+ listener.call(
360
+ target,
361
+ new CancelableCloseEvent('close', { code, reason }) as any,
362
+ )
363
+ })
364
+ break
365
+ }
366
+ }
367
+ }
368
+
369
+ public removeEventListener<EventType extends keyof WebSocketServerEventMap>(
370
+ type: EventType,
371
+ listener: (
372
+ this: WebSocket,
373
+ event: WebSocketServerEventMap[EventType],
374
+ ) => void,
375
+ options?: EventListenerOptions | boolean,
376
+ ): void {
377
+ console.warn(
378
+ '@msw/playwright: WebSocketRoute does not support removing event listeners',
379
+ )
380
+ }
381
+ }
382
+
383
+ interface InternalWebSocketRoute {
384
+ url: Parameters<Page['routeWebSocket']>[0]
385
+ handler: Parameters<Page['routeWebSocket']>[1]
386
+ }
387
+
388
+ /**
389
+ * Custom implementation of the missing `page.unrouteWebSocket()` to remove
390
+ * WebSocket route handlers from the page. Loosely inspired by `page.unroute()`.
391
+ */
392
+ async function unrouteWebSocket(
393
+ page: Page,
394
+ url: InternalWebSocketRoute['url'],
395
+ handler?: InternalWebSocketRoute['handler'],
396
+ ): Promise<void> {
397
+ if (!('_webSocketRoutes' in page && Array.isArray(page._webSocketRoutes))) {
398
+ return
399
+ }
400
+
401
+ for (let i = page._webSocketRoutes.length - 1; i >= 0; i--) {
402
+ const route = page._webSocketRoutes[i] as InternalWebSocketRoute
403
+
404
+ if (
405
+ route.url === url &&
406
+ (handler != null ? route.handler === handler : true)
407
+ ) {
408
+ page._webSocketRoutes.splice(i, 1)
409
+ }
410
+ }
411
+ }
package/src/index.ts CHANGED
@@ -1,367 +1,5 @@
1
- import { invariant } from 'outvariant'
2
- import type {
3
- Page,
4
- PlaywrightTestArgs,
5
- PlaywrightWorkerArgs,
6
- TestFixture,
7
- WebSocketRoute,
8
- } from '@playwright/test'
9
- import {
10
- type LifeCycleEventsMap,
11
- SetupApi,
12
- RequestHandler,
13
- WebSocketHandler,
14
- getResponse,
15
- } from 'msw'
16
- import {
17
- type WebSocketClientEventMap,
18
- type WebSocketData,
19
- type WebSocketServerEventMap,
20
- CancelableMessageEvent,
21
- CancelableCloseEvent,
22
- WebSocketClientConnectionProtocol,
23
- WebSocketServerConnectionProtocol,
24
- } from '@mswjs/interceptors/WebSocket'
25
-
26
- export interface CreateNetworkFixtureArgs {
27
- initialHandlers: Array<RequestHandler | WebSocketHandler>
28
- }
29
-
30
- /**
31
- * Creates a fixture that controls the network in your tests.
32
- *
33
- * @note The returned fixture already has the `auto` option set to `true`.
34
- *
35
- * **Usage**
36
- * ```ts
37
- * import { test as testBase } from '@playwright/test'
38
- * import { createNetworkFixture, type WorkerFixture } from '@msw/playwright'
39
- *
40
- * interface Fixtures {
41
- * network: WorkerFixture
42
- * }
43
- *
44
- * export const test = testBase.extend<Fixtures>({
45
- * network: createNetworkFixture()
46
- * })
47
- * ```
48
- */
49
- export function createNetworkFixture(
50
- args?: CreateNetworkFixtureArgs,
51
- /** @todo `onUnhandledRequest`? */
52
- ): [
53
- TestFixture<NetworkFixture, PlaywrightTestArgs & PlaywrightWorkerArgs>,
54
- { auto: boolean },
55
- ] {
56
- return [
57
- async ({ page }, use) => {
58
- const worker = new NetworkFixture({
59
- page,
60
- initialHandlers: args?.initialHandlers || [],
61
- })
62
-
63
- await worker.start()
64
- await use(worker)
65
- await worker.stop()
66
- },
67
- { auto: true },
68
- ]
69
- }
70
-
71
- export class NetworkFixture extends SetupApi<LifeCycleEventsMap> {
72
- #page: Page
73
-
74
- constructor(args: {
75
- page: Page
76
- initialHandlers: Array<RequestHandler | WebSocketHandler>
77
- }) {
78
- super(...args.initialHandlers)
79
- this.#page = args.page
80
- }
81
-
82
- public async start() {
83
- // Handle HTTP requests.
84
- await this.#page.route(/.+/, async (route, request) => {
85
- const fetchRequest = new Request(request.url(), {
86
- method: request.method(),
87
- headers: new Headers(await request.allHeaders()),
88
- body: request.postDataBuffer(),
89
- })
90
-
91
- const response = await getResponse(
92
- this.handlersController.currentHandlers().filter((handler) => {
93
- return handler instanceof RequestHandler
94
- }),
95
- fetchRequest,
96
- {
97
- baseUrl: this.getPageUrl(),
98
- },
99
- )
100
-
101
- if (response) {
102
- if (response.status === 0) {
103
- route.abort()
104
- return
105
- }
106
-
107
- route.fulfill({
108
- status: response.status,
109
- headers: Object.fromEntries(response.headers),
110
- body: response.body
111
- ? Buffer.from(await response.arrayBuffer())
112
- : undefined,
113
- })
114
- return
115
- }
116
-
117
- route.continue()
118
- })
119
-
120
- // Handle WebSocket connections.
121
- await this.#page.routeWebSocket(/.+/, async (ws) => {
122
- const allWebSocketHandlers = this.handlersController
123
- .currentHandlers()
124
- .filter((handler) => {
125
- return handler instanceof WebSocketHandler
126
- })
127
-
128
- if (allWebSocketHandlers.length === 0) {
129
- ws.connectToServer()
130
- return
131
- }
132
-
133
- const client = new PlaywrightWebSocketClientConnection(ws)
134
- const server = new PlaywrightWebSocketServerConnection(ws)
135
-
136
- for (const handler of allWebSocketHandlers) {
137
- await handler.run(
138
- {
139
- client,
140
- server,
141
- info: { protocols: [] },
142
- },
143
- {
144
- baseUrl: this.getPageUrl(),
145
- },
146
- )
147
- }
148
- })
149
- }
150
-
151
- public async stop() {
152
- super.dispose()
153
- await this.#page.unroute(/.+/)
154
- }
155
-
156
- private getPageUrl(): string | undefined {
157
- const url = this.#page.url()
158
- return url !== 'about:blank' ? url : undefined
159
- }
160
- }
161
-
162
- class PlaywrightWebSocketClientConnection
163
- implements WebSocketClientConnectionProtocol
164
- {
165
- public id: string
166
- public url: URL
167
-
168
- constructor(protected readonly ws: WebSocketRoute) {
169
- this.id = crypto.randomUUID()
170
- this.url = new URL(ws.url())
171
- }
172
-
173
- public send(data: WebSocketData): void {
174
- if (data instanceof Blob) {
175
- /**
176
- * @note Playwright does not support sending Blob data.
177
- * Read the blob as buffer, then send the buffer instead.
178
- */
179
- data.bytes().then((bytes) => {
180
- this.ws.send(Buffer.from(bytes))
181
- })
182
- return
183
- }
184
-
185
- if (typeof data === 'string') {
186
- this.ws.send(data)
187
- return
188
- }
189
-
190
- this.ws.send(
191
- /**
192
- * @note Forcefully cast all data to Buffer because Playwright
193
- * has trouble digesting ArrayBuffer and Blob directly.
194
- */
195
- Buffer.from(
196
- /**
197
- * @note Playwright type definitions are tailored to Node.js
198
- * while MSW describes all data types that can be sent over
199
- * the WebSocket protocol, like ArrayBuffer and Blob.
200
- */
201
- data as any,
202
- ),
203
- )
204
- }
205
-
206
- public close(code?: number, reason?: string): void {
207
- const resolvedCode = code ?? 1000
208
- this.ws.close({ code: resolvedCode, reason })
209
- }
210
-
211
- public addEventListener<EventType extends keyof WebSocketClientEventMap>(
212
- type: EventType,
213
- listener: (
214
- this: WebSocket,
215
- event: WebSocketClientEventMap[EventType],
216
- ) => void,
217
- options?: AddEventListenerOptions | boolean,
218
- ): void {
219
- /**
220
- * @note Playwright does not expose the actual WebSocket reference.
221
- */
222
- const target = {} as WebSocket
223
-
224
- switch (type) {
225
- case 'message': {
226
- this.ws.onMessage((data) => {
227
- listener.call(
228
- target,
229
- new CancelableMessageEvent('message', {
230
- data,
231
- }) as any,
232
- )
233
- })
234
- break
235
- }
236
-
237
- case 'close': {
238
- this.ws.onClose((code, reason) => {
239
- listener.call(
240
- target,
241
- new CancelableCloseEvent('close', {
242
- code,
243
- reason,
244
- }) as any,
245
- )
246
- })
247
- break
248
- }
249
- }
250
- }
251
-
252
- public removeEventListener<EventType extends keyof WebSocketClientEventMap>(
253
- event: EventType,
254
- listener: (
255
- this: WebSocket,
256
- event: WebSocketClientEventMap[EventType],
257
- ) => void,
258
- options?: EventListenerOptions | boolean,
259
- ): void {
260
- console.warn(
261
- '@msw/playwright: WebSocketRoute does not support removing event listeners',
262
- )
263
- }
264
- }
265
-
266
- class PlaywrightWebSocketServerConnection
267
- implements WebSocketServerConnectionProtocol
268
- {
269
- #server?: WebSocketRoute
270
- #bufferedEvents: Array<
271
- Parameters<WebSocketServerConnectionProtocol['addEventListener']>
272
- >
273
- #bufferedData: Array<WebSocketData>
274
-
275
- constructor(protected readonly ws: WebSocketRoute) {
276
- this.#bufferedEvents = []
277
- this.#bufferedData = []
278
- }
279
-
280
- public connect(): void {
281
- this.#server = this.ws.connectToServer()
282
-
283
- /**
284
- * @note Playwright does not support event buffering.
285
- * Manually add event listeners that might have been registered
286
- * before `connect()` was called.
287
- */
288
- for (const [type, listener, options] of this.#bufferedEvents) {
289
- this.addEventListener(type, listener, options)
290
- }
291
- this.#bufferedEvents.length = 0
292
-
293
- // Same for the buffered data.
294
- for (const data of this.#bufferedData) {
295
- this.send(data)
296
- }
297
- this.#bufferedData.length = 0
298
- }
299
-
300
- public send(data: WebSocketData): void {
301
- if (this.#server == null) {
302
- this.#bufferedData.push(data)
303
- return
304
- }
305
-
306
- this.#server.send(data as any)
307
- }
308
-
309
- public close(code?: number, reason?: string): void {
310
- invariant(
311
- this.#server,
312
- 'Failed to close connection to the actual WebSocket server: connection not established. Did you forget to call `connect()`?',
313
- )
314
-
315
- this.#server.close({ code, reason })
316
- }
317
-
318
- public addEventListener<EventType extends keyof WebSocketServerEventMap>(
319
- type: EventType,
320
- listener: (
321
- this: WebSocket,
322
- event: WebSocketServerEventMap[EventType],
323
- ) => void,
324
- options?: AddEventListenerOptions | boolean,
325
- ): void {
326
- if (this.#server == null) {
327
- this.#bufferedEvents.push([type, listener as any, options])
328
- return
329
- }
330
-
331
- const target = {} as WebSocket
332
- switch (type) {
333
- case 'message': {
334
- this.#server.onMessage((data) => {
335
- listener.call(
336
- target,
337
- new CancelableMessageEvent('message', { data }) as any,
338
- )
339
- })
340
- break
341
- }
342
-
343
- case 'close': {
344
- this.#server.onClose((code, reason) => {
345
- listener.call(
346
- target,
347
- new CancelableCloseEvent('close', { code, reason }) as any,
348
- )
349
- })
350
- break
351
- }
352
- }
353
- }
354
-
355
- public removeEventListener<EventType extends keyof WebSocketServerEventMap>(
356
- type: EventType,
357
- listener: (
358
- this: WebSocket,
359
- event: WebSocketServerEventMap[EventType],
360
- ) => void,
361
- options?: EventListenerOptions | boolean,
362
- ): void {
363
- console.warn(
364
- '@msw/playwright: WebSocketRoute does not support removing event listeners',
365
- )
366
- }
367
- }
1
+ export {
2
+ type CreateNetworkFixtureArgs,
3
+ type NetworkFixture,
4
+ createNetworkFixture,
5
+ } from './fixture.js'