@msw/playwright 0.2.0 → 0.3.1

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.d.ts CHANGED
@@ -1,16 +1,18 @@
1
- import { LifeCycleEventsMap, RequestHandler, SetupApi } from "msw";
1
+ import { LifeCycleEventsMap, RequestHandler, SetupApi, WebSocketHandler } from "msw";
2
2
  import { Page, TestFixture } from "@playwright/test";
3
3
 
4
4
  //#region src/index.d.ts
5
5
  interface CreateWorkerFixtureArgs {
6
- initialHandlers: Array<RequestHandler>;
6
+ initialHandlers: Array<RequestHandler | WebSocketHandler>;
7
7
  }
8
- declare function createWorkerFixture(args?: CreateWorkerFixtureArgs): TestFixture<WorkerFixture, any>;
8
+ declare function createWorkerFixture(args?: CreateWorkerFixtureArgs): [TestFixture<WorkerFixture, any>, {
9
+ auto: boolean;
10
+ }];
9
11
  declare class WorkerFixture extends SetupApi<LifeCycleEventsMap> {
10
12
  #private;
11
13
  constructor(args: {
12
14
  page: Page;
13
- initialHandlers: Array<RequestHandler>;
15
+ initialHandlers: Array<RequestHandler | WebSocketHandler>;
14
16
  });
15
17
  start(): Promise<void>;
16
18
  stop(): Promise<void>;
package/build/index.js CHANGED
@@ -1,8 +1,10 @@
1
- import { RequestHandler, SetupApi, getResponse } from "msw";
1
+ import { invariant } from "outvariant";
2
+ import { RequestHandler, SetupApi, WebSocketHandler, getResponse } from "msw";
3
+ import { CancelableCloseEvent, CancelableMessageEvent } from "@mswjs/interceptors/WebSocket";
2
4
 
3
5
  //#region src/index.ts
4
6
  function createWorkerFixture(args) {
5
- return async ({ page }, use) => {
7
+ return [async ({ page }, use) => {
6
8
  const worker = new WorkerFixture({
7
9
  page,
8
10
  initialHandlers: args?.initialHandlers || []
@@ -10,7 +12,7 @@ function createWorkerFixture(args) {
10
12
  await worker.start();
11
13
  await use(worker);
12
14
  await worker.stop();
13
- };
15
+ }, { auto: true }];
14
16
  }
15
17
  var WorkerFixture = class extends SetupApi {
16
18
  #page;
@@ -38,12 +40,167 @@ var WorkerFixture = class extends SetupApi {
38
40
  }
39
41
  route.continue();
40
42
  });
43
+ await this.#page.routeWebSocket(/.+/, async (ws) => {
44
+ const allWebSocketHandlers = this.handlersController.currentHandlers().filter((handler) => {
45
+ return handler instanceof WebSocketHandler;
46
+ });
47
+ if (allWebSocketHandlers.length === 0) {
48
+ ws.connectToServer();
49
+ return;
50
+ }
51
+ const client = new PlaywrightWebSocketClientConnection(ws);
52
+ const server = new PlaywrightWebSocketServerConnection(ws);
53
+ for (const handler of allWebSocketHandlers) await handler.run({
54
+ client,
55
+ server,
56
+ info: { protocols: [] }
57
+ });
58
+ });
41
59
  }
42
60
  async stop() {
43
61
  super.dispose();
44
62
  await this.#page.unroute(/.+/);
45
63
  }
46
64
  };
65
+ var PlaywrightWebSocketClientConnection = class {
66
+ id;
67
+ url;
68
+ constructor(ws) {
69
+ this.ws = ws;
70
+ this.id = crypto.randomUUID();
71
+ this.url = new URL(ws.url());
72
+ }
73
+ send(data) {
74
+ if (data instanceof Blob) {
75
+ /**
76
+ * @note Playwright does not support sending Blob data.
77
+ * Read the blob as buffer, then send the buffer instead.
78
+ */
79
+ data.bytes().then((bytes) => {
80
+ this.ws.send(Buffer.from(bytes));
81
+ });
82
+ return;
83
+ }
84
+ if (typeof data === "string") {
85
+ this.ws.send(data);
86
+ return;
87
+ }
88
+ this.ws.send(
89
+ /**
90
+ * @note Forcefully cast all data to Buffer because Playwright
91
+ * has trouble digesting ArrayBuffer and Blob directly.
92
+ */
93
+ Buffer.from(
94
+ /**
95
+ * @note Playwright type definitions are tailored to Node.js
96
+ * while MSW describes all data types that can be sent over
97
+ * the WebSocket protocol, like ArrayBuffer and Blob.
98
+ */
99
+ data
100
+ )
101
+ );
102
+ }
103
+ close(code, reason) {
104
+ const resolvedCode = code ?? 1e3;
105
+ this.ws.close({
106
+ code: resolvedCode,
107
+ reason
108
+ });
109
+ }
110
+ addEventListener(type, listener, options) {
111
+ /**
112
+ * @note Playwright does not expose the actual WebSocket reference.
113
+ */
114
+ const target = {};
115
+ switch (type) {
116
+ case "message": {
117
+ this.ws.onMessage((data) => {
118
+ listener.call(target, new CancelableMessageEvent("message", { data }));
119
+ });
120
+ break;
121
+ }
122
+ case "close": {
123
+ this.ws.onClose((code, reason) => {
124
+ listener.call(target, new CancelableCloseEvent("close", {
125
+ code,
126
+ reason
127
+ }));
128
+ });
129
+ break;
130
+ }
131
+ }
132
+ }
133
+ removeEventListener(event, listener, options) {
134
+ console.warn("@msw/playwright: WebSocketRoute does not support removing event listeners");
135
+ }
136
+ };
137
+ var PlaywrightWebSocketServerConnection = class {
138
+ #server;
139
+ #bufferedEvents;
140
+ #bufferedData;
141
+ constructor(ws) {
142
+ this.ws = ws;
143
+ this.#bufferedEvents = [];
144
+ this.#bufferedData = [];
145
+ }
146
+ connect() {
147
+ this.#server = this.ws.connectToServer();
148
+ /**
149
+ * @note Playwright does not support event buffering.
150
+ * Manually add event listeners that might have been registered
151
+ * before `connect()` was called.
152
+ */
153
+ for (const [type, listener, options] of this.#bufferedEvents) this.addEventListener(type, listener, options);
154
+ this.#bufferedEvents.length = 0;
155
+ for (const data of this.#bufferedData) this.send(data);
156
+ this.#bufferedData.length = 0;
157
+ }
158
+ send(data) {
159
+ if (this.#server == null) {
160
+ this.#bufferedData.push(data);
161
+ return;
162
+ }
163
+ this.#server.send(data);
164
+ }
165
+ close(code, reason) {
166
+ invariant(this.#server, "Failed to close connection to the actual WebSocket server: connection not established. Did you forget to call `connect()`?");
167
+ this.#server.close({
168
+ code,
169
+ reason
170
+ });
171
+ }
172
+ addEventListener(type, listener, options) {
173
+ if (this.#server == null) {
174
+ this.#bufferedEvents.push([
175
+ type,
176
+ listener,
177
+ options
178
+ ]);
179
+ return;
180
+ }
181
+ const target = {};
182
+ switch (type) {
183
+ case "message": {
184
+ this.#server.onMessage((data) => {
185
+ listener.call(target, new CancelableMessageEvent("message", { data }));
186
+ });
187
+ break;
188
+ }
189
+ case "close": {
190
+ this.#server.onClose((code, reason) => {
191
+ listener.call(target, new CancelableCloseEvent("close", {
192
+ code,
193
+ reason
194
+ }));
195
+ });
196
+ break;
197
+ }
198
+ }
199
+ }
200
+ removeEventListener(type, listener, options) {
201
+ console.warn("@msw/playwright: WebSocketRoute does not support removing event listeners");
202
+ }
203
+ };
47
204
 
48
205
  //#endregion
49
206
  export { WorkerFixture, createWorkerFixture };
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "type": "module",
3
3
  "name": "@msw/playwright",
4
- "version": "0.2.0",
4
+ "version": "0.3.1",
5
5
  "description": "Mock Service Worker binding for Playwright",
6
6
  "main": "./build/index.js",
7
7
  "types": "./build/index.d.ts",
@@ -34,16 +34,21 @@
34
34
  "node": ">=20.0.0"
35
35
  },
36
36
  "peerDependencies": {
37
- "msw": "^2.9.0"
37
+ "msw": "^2.10.1"
38
38
  },
39
39
  "devDependencies": {
40
+ "@epic-web/test-server": "^0.1.6",
40
41
  "@ossjs/release": "^0.8.1",
41
42
  "@playwright/test": "^1.52.0",
42
43
  "@types/node": "^22.15.29",
43
- "msw": "^2.9.0",
44
+ "msw": "^2.10.1",
44
45
  "tsdown": "^0.12.7",
45
46
  "typescript": "^5.8.3"
46
47
  },
48
+ "dependencies": {
49
+ "@mswjs/interceptors": "^0.39.2",
50
+ "outvariant": "^1.4.3"
51
+ },
47
52
  "scripts": {
48
53
  "dev": "tsdown --watch",
49
54
  "test": "playwright test",
package/src/index.ts CHANGED
@@ -1,40 +1,58 @@
1
- import type { Page, TestFixture } from '@playwright/test'
1
+ import { invariant } from 'outvariant'
2
+ import type { Page, TestFixture, WebSocketRoute } from '@playwright/test'
2
3
  import {
3
4
  type LifeCycleEventsMap,
4
5
  SetupApi,
5
6
  RequestHandler,
7
+ WebSocketHandler,
6
8
  getResponse,
7
9
  } from 'msw'
10
+ import {
11
+ type WebSocketClientEventMap,
12
+ type WebSocketData,
13
+ type WebSocketServerEventMap,
14
+ CancelableMessageEvent,
15
+ CancelableCloseEvent,
16
+ WebSocketClientConnectionProtocol,
17
+ WebSocketServerConnectionProtocol,
18
+ } from '@mswjs/interceptors/WebSocket'
8
19
 
9
20
  export interface CreateWorkerFixtureArgs {
10
- initialHandlers: Array<RequestHandler>
21
+ initialHandlers: Array<RequestHandler | WebSocketHandler>
11
22
  }
12
23
 
13
24
  export function createWorkerFixture(
14
25
  args?: CreateWorkerFixtureArgs,
15
26
  /** @todo `onUnhandledRequest`? */
16
- ): TestFixture<WorkerFixture, any> {
17
- return async ({ page }, use) => {
18
- const worker = new WorkerFixture({
19
- page,
20
- initialHandlers: args?.initialHandlers || [],
21
- })
27
+ ): [TestFixture<WorkerFixture, any>, { auto: boolean }] {
28
+ return [
29
+ async ({ page }, use) => {
30
+ const worker = new WorkerFixture({
31
+ page,
32
+ initialHandlers: args?.initialHandlers || [],
33
+ })
22
34
 
23
- await worker.start()
24
- await use(worker)
25
- await worker.stop()
26
- }
35
+ await worker.start()
36
+ await use(worker)
37
+ await worker.stop()
38
+ },
39
+ { auto: true },
40
+ ]
27
41
  }
28
42
 
29
43
  export class WorkerFixture extends SetupApi<LifeCycleEventsMap> {
30
44
  #page: Page
31
45
 
32
- constructor(args: { page: Page; initialHandlers: Array<RequestHandler> }) {
46
+ constructor(args: {
47
+ page: Page
48
+ initialHandlers: Array<RequestHandler | WebSocketHandler>
49
+ }) {
33
50
  super(...args.initialHandlers)
34
51
  this.#page = args.page
35
52
  }
36
53
 
37
54
  public async start() {
55
+ // Handle HTTP requests.
38
56
  await this.#page.route(/.+/, async (route, request) => {
39
57
  const fetchRequest = new Request(request.url(), {
40
58
  method: request.method(),
@@ -62,6 +80,31 @@ export class WorkerFixture extends SetupApi<LifeCycleEventsMap> {
62
80
 
63
81
  route.continue()
64
82
  })
83
+
84
+ // Handle WebSocket connections.
85
+ await this.#page.routeWebSocket(/.+/, async (ws) => {
86
+ const allWebSocketHandlers = this.handlersController
87
+ .currentHandlers()
88
+ .filter((handler) => {
89
+ return handler instanceof WebSocketHandler
90
+ })
91
+
92
+ if (allWebSocketHandlers.length === 0) {
93
+ ws.connectToServer()
94
+ return
95
+ }
96
+
97
+ const client = new PlaywrightWebSocketClientConnection(ws)
98
+ const server = new PlaywrightWebSocketServerConnection(ws)
99
+
100
+ for (const handler of allWebSocketHandlers) {
101
+ await handler.run({
102
+ client,
103
+ server,
104
+ info: { protocols: [] },
105
+ })
106
+ }
107
+ })
65
108
  }
66
109
 
67
110
  public async stop() {
@@ -69,3 +112,210 @@ export class WorkerFixture extends SetupApi<LifeCycleEventsMap> {
69
112
  await this.#page.unroute(/.+/)
70
113
  }
71
114
  }
115
+
116
+ class PlaywrightWebSocketClientConnection
117
+ implements WebSocketClientConnectionProtocol
118
+ {
119
+ public id: string
120
+ public url: URL
121
+
122
+ constructor(protected readonly ws: WebSocketRoute) {
123
+ this.id = crypto.randomUUID()
124
+ this.url = new URL(ws.url())
125
+ }
126
+
127
+ public send(data: WebSocketData): void {
128
+ if (data instanceof Blob) {
129
+ /**
130
+ * @note Playwright does not support sending Blob data.
131
+ * Read the blob as buffer, then send the buffer instead.
132
+ */
133
+ data.bytes().then((bytes) => {
134
+ this.ws.send(Buffer.from(bytes))
135
+ })
136
+ return
137
+ }
138
+
139
+ if (typeof data === 'string') {
140
+ this.ws.send(data)
141
+ return
142
+ }
143
+
144
+ this.ws.send(
145
+ /**
146
+ * @note Forcefully cast all data to Buffer because Playwright
147
+ * has trouble digesting ArrayBuffer and Blob directly.
148
+ */
149
+ Buffer.from(
150
+ /**
151
+ * @note Playwright type definitions are tailored to Node.js
152
+ * while MSW describes all data types that can be sent over
153
+ * the WebSocket protocol, like ArrayBuffer and Blob.
154
+ */
155
+ data as any,
156
+ ),
157
+ )
158
+ }
159
+
160
+ public close(code?: number, reason?: string): void {
161
+ const resolvedCode = code ?? 1000
162
+ this.ws.close({ code: resolvedCode, reason })
163
+ }
164
+
165
+ public addEventListener<EventType extends keyof WebSocketClientEventMap>(
166
+ type: EventType,
167
+ listener: (
168
+ this: WebSocket,
169
+ event: WebSocketClientEventMap[EventType],
170
+ ) => void,
171
+ options?: AddEventListenerOptions | boolean,
172
+ ): void {
173
+ /**
174
+ * @note Playwright does not expose the actual WebSocket reference.
175
+ */
176
+ const target = {} as WebSocket
177
+
178
+ switch (type) {
179
+ case 'message': {
180
+ this.ws.onMessage((data) => {
181
+ listener.call(
182
+ target,
183
+ new CancelableMessageEvent('message', {
184
+ data,
185
+ }) as any,
186
+ )
187
+ })
188
+ break
189
+ }
190
+
191
+ case 'close': {
192
+ this.ws.onClose((code, reason) => {
193
+ listener.call(
194
+ target,
195
+ new CancelableCloseEvent('close', {
196
+ code,
197
+ reason,
198
+ }) as any,
199
+ )
200
+ })
201
+ break
202
+ }
203
+ }
204
+ }
205
+
206
+ public removeEventListener<EventType extends keyof WebSocketClientEventMap>(
207
+ event: EventType,
208
+ listener: (
209
+ this: WebSocket,
210
+ event: WebSocketClientEventMap[EventType],
211
+ ) => void,
212
+ options?: EventListenerOptions | boolean,
213
+ ): void {
214
+ console.warn(
215
+ '@msw/playwright: WebSocketRoute does not support removing event listeners',
216
+ )
217
+ }
218
+ }
219
+
220
+ class PlaywrightWebSocketServerConnection
221
+ implements WebSocketServerConnectionProtocol
222
+ {
223
+ #server?: WebSocketRoute
224
+ #bufferedEvents: Array<
225
+ Parameters<WebSocketServerConnectionProtocol['addEventListener']>
226
+ >
227
+ #bufferedData: Array<WebSocketData>
228
+
229
+ constructor(protected readonly ws: WebSocketRoute) {
230
+ this.#bufferedEvents = []
231
+ this.#bufferedData = []
232
+ }
233
+
234
+ public connect(): void {
235
+ this.#server = this.ws.connectToServer()
236
+
237
+ /**
238
+ * @note Playwright does not support event buffering.
239
+ * Manually add event listeners that might have been registered
240
+ * before `connect()` was called.
241
+ */
242
+ for (const [type, listener, options] of this.#bufferedEvents) {
243
+ this.addEventListener(type, listener, options)
244
+ }
245
+ this.#bufferedEvents.length = 0
246
+
247
+ // Same for the buffered data.
248
+ for (const data of this.#bufferedData) {
249
+ this.send(data)
250
+ }
251
+ this.#bufferedData.length = 0
252
+ }
253
+
254
+ public send(data: WebSocketData): void {
255
+ if (this.#server == null) {
256
+ this.#bufferedData.push(data)
257
+ return
258
+ }
259
+
260
+ this.#server.send(data as any)
261
+ }
262
+
263
+ public close(code?: number, reason?: string): void {
264
+ invariant(
265
+ this.#server,
266
+ 'Failed to close connection to the actual WebSocket server: connection not established. Did you forget to call `connect()`?',
267
+ )
268
+
269
+ this.#server.close({ code, reason })
270
+ }
271
+
272
+ public addEventListener<EventType extends keyof WebSocketServerEventMap>(
273
+ type: EventType,
274
+ listener: (
275
+ this: WebSocket,
276
+ event: WebSocketServerEventMap[EventType],
277
+ ) => void,
278
+ options?: AddEventListenerOptions | boolean,
279
+ ): void {
280
+ if (this.#server == null) {
281
+ this.#bufferedEvents.push([type, listener as any, options])
282
+ return
283
+ }
284
+
285
+ const target = {} as WebSocket
286
+ switch (type) {
287
+ case 'message': {
288
+ this.#server.onMessage((data) => {
289
+ listener.call(
290
+ target,
291
+ new CancelableMessageEvent('message', { data }) as any,
292
+ )
293
+ })
294
+ break
295
+ }
296
+
297
+ case 'close': {
298
+ this.#server.onClose((code, reason) => {
299
+ listener.call(
300
+ target,
301
+ new CancelableCloseEvent('close', { code, reason }) as any,
302
+ )
303
+ })
304
+ break
305
+ }
306
+ }
307
+ }
308
+
309
+ public removeEventListener<EventType extends keyof WebSocketServerEventMap>(
310
+ type: EventType,
311
+ listener: (
312
+ this: WebSocket,
313
+ event: WebSocketServerEventMap[EventType],
314
+ ) => void,
315
+ options?: EventListenerOptions | boolean,
316
+ ): void {
317
+ console.warn(
318
+ '@msw/playwright: WebSocketRoute does not support removing event listeners',
319
+ )
320
+ }
321
+ }