msw 2.5.1 → 2.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (190) hide show
  1. package/lib/browser/index.d.mts +7 -6
  2. package/lib/browser/index.d.ts +7 -6
  3. package/lib/browser/index.js +29 -1
  4. package/lib/browser/index.js.map +1 -1
  5. package/lib/browser/index.mjs +29 -1
  6. package/lib/browser/index.mjs.map +1 -1
  7. package/lib/core/{GraphQLHandler-ClMB0BOy.d.mts → GraphQLHandler-Doool6Q_.d.mts} +1 -1
  8. package/lib/core/{GraphQLHandler-D6mLMXGZ.d.ts → GraphQLHandler-udzgBRPf.d.ts} +1 -1
  9. package/lib/core/{HttpResponse-vn-Pb4Bi.d.mts → HttpResponse-BLGmJolh.d.mts} +1 -1
  10. package/lib/core/{HttpResponse-DaYkf3ml.d.ts → HttpResponse-Cgbkdkje.d.ts} +1 -1
  11. package/lib/core/HttpResponse.d.mts +1 -1
  12. package/lib/core/HttpResponse.d.ts +1 -1
  13. package/lib/core/SetupApi.d.mts +15 -12
  14. package/lib/core/SetupApi.d.ts +15 -12
  15. package/lib/core/SetupApi.js +3 -1
  16. package/lib/core/SetupApi.js.map +1 -1
  17. package/lib/core/SetupApi.mjs +3 -1
  18. package/lib/core/SetupApi.mjs.map +1 -1
  19. package/lib/core/getResponse.d.mts +1 -1
  20. package/lib/core/getResponse.d.ts +1 -1
  21. package/lib/core/graphql.d.mts +2 -2
  22. package/lib/core/graphql.d.ts +2 -2
  23. package/lib/core/handlers/GraphQLHandler.d.mts +2 -2
  24. package/lib/core/handlers/GraphQLHandler.d.ts +2 -2
  25. package/lib/core/handlers/HttpHandler.d.mts +1 -1
  26. package/lib/core/handlers/HttpHandler.d.ts +1 -1
  27. package/lib/core/handlers/RequestHandler.d.mts +1 -1
  28. package/lib/core/handlers/RequestHandler.d.ts +1 -1
  29. package/lib/core/handlers/WebSocketHandler.d.mts +33 -0
  30. package/lib/core/handlers/WebSocketHandler.d.ts +33 -0
  31. package/lib/core/handlers/WebSocketHandler.js +120 -0
  32. package/lib/core/handlers/WebSocketHandler.js.map +1 -0
  33. package/lib/core/handlers/WebSocketHandler.mjs +102 -0
  34. package/lib/core/handlers/WebSocketHandler.mjs.map +1 -0
  35. package/lib/core/http.d.mts +1 -1
  36. package/lib/core/http.d.ts +1 -1
  37. package/lib/core/index.d.mts +5 -2
  38. package/lib/core/index.d.ts +5 -2
  39. package/lib/core/index.js +5 -1
  40. package/lib/core/index.js.map +1 -1
  41. package/lib/core/index.mjs +7 -1
  42. package/lib/core/index.mjs.map +1 -1
  43. package/lib/core/passthrough.d.mts +1 -1
  44. package/lib/core/passthrough.d.ts +1 -1
  45. package/lib/core/utils/HttpResponse/decorators.d.mts +1 -1
  46. package/lib/core/utils/HttpResponse/decorators.d.ts +1 -1
  47. package/lib/core/utils/executeHandlers.d.mts +1 -1
  48. package/lib/core/utils/executeHandlers.d.ts +1 -1
  49. package/lib/core/utils/executeHandlers.js +4 -0
  50. package/lib/core/utils/executeHandlers.js.map +1 -1
  51. package/lib/core/utils/executeHandlers.mjs +6 -0
  52. package/lib/core/utils/executeHandlers.mjs.map +1 -1
  53. package/lib/core/utils/handleRequest.d.mts +2 -2
  54. package/lib/core/utils/handleRequest.d.ts +2 -2
  55. package/lib/core/utils/handleRequest.js.map +1 -1
  56. package/lib/core/utils/handleRequest.mjs.map +1 -1
  57. package/lib/core/utils/internal/parseGraphQLRequest.d.mts +2 -2
  58. package/lib/core/utils/internal/parseGraphQLRequest.d.ts +2 -2
  59. package/lib/core/utils/internal/parseMultipartData.d.mts +1 -1
  60. package/lib/core/utils/internal/parseMultipartData.d.ts +1 -1
  61. package/lib/core/utils/internal/requestHandlerUtils.d.mts +1 -1
  62. package/lib/core/utils/internal/requestHandlerUtils.d.ts +1 -1
  63. package/lib/core/utils/logging/getTimestamp.d.mts +4 -1
  64. package/lib/core/utils/logging/getTimestamp.d.ts +4 -1
  65. package/lib/core/utils/logging/getTimestamp.js +6 -2
  66. package/lib/core/utils/logging/getTimestamp.js.map +1 -1
  67. package/lib/core/utils/logging/getTimestamp.mjs +6 -2
  68. package/lib/core/utils/logging/getTimestamp.mjs.map +1 -1
  69. package/lib/core/utils/matching/matchRequestUrl.d.mts +2 -1
  70. package/lib/core/utils/matching/matchRequestUrl.d.ts +2 -1
  71. package/lib/core/utils/matching/matchRequestUrl.js +4 -0
  72. package/lib/core/utils/matching/matchRequestUrl.js.map +1 -1
  73. package/lib/core/utils/matching/matchRequestUrl.mjs +4 -0
  74. package/lib/core/utils/matching/matchRequestUrl.mjs.map +1 -1
  75. package/lib/core/ws/WebSocketClientManager.d.mts +63 -0
  76. package/lib/core/ws/WebSocketClientManager.d.ts +63 -0
  77. package/lib/core/ws/WebSocketClientManager.js +149 -0
  78. package/lib/core/ws/WebSocketClientManager.js.map +1 -0
  79. package/lib/core/ws/WebSocketClientManager.mjs +129 -0
  80. package/lib/core/ws/WebSocketClientManager.mjs.map +1 -0
  81. package/lib/core/ws/WebSocketClientStore.d.mts +13 -0
  82. package/lib/core/ws/WebSocketClientStore.d.ts +13 -0
  83. package/lib/core/ws/WebSocketClientStore.js +26 -0
  84. package/lib/core/ws/WebSocketClientStore.js.map +1 -0
  85. package/lib/core/ws/WebSocketClientStore.mjs +6 -0
  86. package/lib/core/ws/WebSocketClientStore.mjs.map +1 -0
  87. package/lib/core/ws/WebSocketIndexedDBClientStore.d.mts +15 -0
  88. package/lib/core/ws/WebSocketIndexedDBClientStore.d.ts +15 -0
  89. package/lib/core/ws/WebSocketIndexedDBClientStore.js +130 -0
  90. package/lib/core/ws/WebSocketIndexedDBClientStore.js.map +1 -0
  91. package/lib/core/ws/WebSocketIndexedDBClientStore.mjs +110 -0
  92. package/lib/core/ws/WebSocketIndexedDBClientStore.mjs.map +1 -0
  93. package/lib/core/ws/WebSocketMemoryClientStore.d.mts +13 -0
  94. package/lib/core/ws/WebSocketMemoryClientStore.d.ts +13 -0
  95. package/lib/core/ws/WebSocketMemoryClientStore.js +41 -0
  96. package/lib/core/ws/WebSocketMemoryClientStore.js.map +1 -0
  97. package/lib/core/ws/WebSocketMemoryClientStore.mjs +21 -0
  98. package/lib/core/ws/WebSocketMemoryClientStore.mjs.map +1 -0
  99. package/lib/core/ws/handleWebSocketEvent.d.mts +19 -0
  100. package/lib/core/ws/handleWebSocketEvent.d.ts +19 -0
  101. package/lib/core/ws/handleWebSocketEvent.js +73 -0
  102. package/lib/core/ws/handleWebSocketEvent.js.map +1 -0
  103. package/lib/core/ws/handleWebSocketEvent.mjs +55 -0
  104. package/lib/core/ws/handleWebSocketEvent.mjs.map +1 -0
  105. package/lib/core/ws/utils/attachWebSocketLogger.d.mts +12 -0
  106. package/lib/core/ws/utils/attachWebSocketLogger.d.ts +12 -0
  107. package/lib/core/ws/utils/attachWebSocketLogger.js +198 -0
  108. package/lib/core/ws/utils/attachWebSocketLogger.js.map +1 -0
  109. package/lib/core/ws/utils/attachWebSocketLogger.mjs +178 -0
  110. package/lib/core/ws/utils/attachWebSocketLogger.mjs.map +1 -0
  111. package/lib/core/ws/utils/getMessageLength.d.mts +11 -0
  112. package/lib/core/ws/utils/getMessageLength.d.ts +11 -0
  113. package/lib/core/ws/utils/getMessageLength.js +33 -0
  114. package/lib/core/ws/utils/getMessageLength.js.map +1 -0
  115. package/lib/core/ws/utils/getMessageLength.mjs +13 -0
  116. package/lib/core/ws/utils/getMessageLength.mjs.map +1 -0
  117. package/lib/core/ws/utils/getPublicData.d.mts +5 -0
  118. package/lib/core/ws/utils/getPublicData.d.ts +5 -0
  119. package/lib/core/ws/utils/getPublicData.js +36 -0
  120. package/lib/core/ws/utils/getPublicData.js.map +1 -0
  121. package/lib/core/ws/utils/getPublicData.mjs +16 -0
  122. package/lib/core/ws/utils/getPublicData.mjs.map +1 -0
  123. package/lib/core/ws/utils/truncateMessage.d.mts +3 -0
  124. package/lib/core/ws/utils/truncateMessage.d.ts +3 -0
  125. package/lib/core/ws/utils/truncateMessage.js +31 -0
  126. package/lib/core/ws/utils/truncateMessage.js.map +1 -0
  127. package/lib/core/ws/utils/truncateMessage.mjs +11 -0
  128. package/lib/core/ws/utils/truncateMessage.mjs.map +1 -0
  129. package/lib/core/ws/webSocketInterceptor.d.mts +5 -0
  130. package/lib/core/ws/webSocketInterceptor.d.ts +5 -0
  131. package/lib/core/ws/webSocketInterceptor.js +26 -0
  132. package/lib/core/ws/webSocketInterceptor.js.map +1 -0
  133. package/lib/core/ws/webSocketInterceptor.mjs +6 -0
  134. package/lib/core/ws/webSocketInterceptor.mjs.map +1 -0
  135. package/lib/core/ws.d.mts +75 -0
  136. package/lib/core/ws.d.ts +75 -0
  137. package/lib/core/ws.js +71 -0
  138. package/lib/core/ws.js.map +1 -0
  139. package/lib/core/ws.mjs +54 -0
  140. package/lib/core/ws.mjs.map +1 -0
  141. package/lib/iife/index.js +1413 -85
  142. package/lib/iife/index.js.map +1 -1
  143. package/lib/mockServiceWorker.js +1 -1
  144. package/lib/native/index.d.mts +6 -5
  145. package/lib/native/index.d.ts +6 -5
  146. package/lib/native/index.js +22 -4
  147. package/lib/native/index.js.map +1 -1
  148. package/lib/native/index.mjs +22 -4
  149. package/lib/native/index.mjs.map +1 -1
  150. package/lib/node/index.d.mts +8 -7
  151. package/lib/node/index.d.ts +8 -7
  152. package/lib/node/index.js +22 -4
  153. package/lib/node/index.js.map +1 -1
  154. package/lib/node/index.mjs +22 -4
  155. package/lib/node/index.mjs.map +1 -1
  156. package/package.json +10 -1
  157. package/src/browser/setupWorker/glossary.ts +10 -10
  158. package/src/browser/setupWorker/setupWorker.ts +32 -3
  159. package/src/browser/setupWorker/start/createRequestListener.ts +7 -1
  160. package/src/browser/setupWorker/start/createStartHandler.ts +5 -0
  161. package/src/browser/setupWorker/stop/createStop.ts +6 -0
  162. package/src/core/SetupApi.ts +28 -20
  163. package/src/core/handlers/WebSocketHandler.ts +142 -0
  164. package/src/core/index.ts +11 -1
  165. package/src/core/utils/executeHandlers.ts +6 -2
  166. package/src/core/utils/handleRequest.ts +1 -1
  167. package/src/core/utils/logging/getTimestamp.test.ts +20 -6
  168. package/src/core/utils/logging/getTimestamp.ts +11 -6
  169. package/src/core/utils/matching/matchRequestUrl.test.ts +44 -0
  170. package/src/core/utils/matching/matchRequestUrl.ts +4 -0
  171. package/src/core/ws/WebSocketClientManager.test.ts +164 -0
  172. package/src/core/ws/WebSocketClientManager.ts +211 -0
  173. package/src/core/ws/WebSocketClientStore.ts +14 -0
  174. package/src/core/ws/WebSocketIndexedDBClientStore.ts +145 -0
  175. package/src/core/ws/WebSocketMemoryClientStore.ts +27 -0
  176. package/src/core/ws/handleWebSocketEvent.ts +82 -0
  177. package/src/core/ws/utils/attachWebSocketLogger.ts +259 -0
  178. package/src/core/ws/utils/getMessageLength.test.ts +16 -0
  179. package/src/core/ws/utils/getMessageLength.ts +19 -0
  180. package/src/core/ws/utils/getPublicData.test.ts +38 -0
  181. package/src/core/ws/utils/getPublicData.ts +17 -0
  182. package/src/core/ws/utils/truncateMessage.test.ts +12 -0
  183. package/src/core/ws/utils/truncateMessage.ts +9 -0
  184. package/src/core/ws/webSocketInterceptor.ts +3 -0
  185. package/src/core/ws.test.ts +23 -0
  186. package/src/core/ws.ts +166 -0
  187. package/src/node/SetupServerApi.ts +8 -7
  188. package/src/node/SetupServerCommonApi.ts +29 -5
  189. package/src/node/glossary.ts +5 -7
  190. package/src/node/setupServer.ts +2 -1
@@ -0,0 +1,259 @@
1
+ import type {
2
+ WebSocketClientConnection,
3
+ WebSocketConnectionData,
4
+ WebSocketData,
5
+ } from '@mswjs/interceptors/WebSocket'
6
+ import { devUtils } from '../../utils/internal/devUtils'
7
+ import { getTimestamp } from '../../utils/logging/getTimestamp'
8
+ import { toPublicUrl } from '../../utils/request/toPublicUrl'
9
+ import { getMessageLength } from './getMessageLength'
10
+ import { getPublicData } from './getPublicData'
11
+
12
+ const colors = {
13
+ system: '#3b82f6',
14
+ outgoing: '#22c55e',
15
+ incoming: '#ef4444',
16
+ mocked: '#ff6a33',
17
+ }
18
+
19
+ export function attachWebSocketLogger(
20
+ connection: WebSocketConnectionData,
21
+ ): void {
22
+ const { client, server } = connection
23
+
24
+ logConnectionOpen(client)
25
+
26
+ // Log the events sent from the WebSocket client.
27
+ // WebSocket client connection object is written from the
28
+ // server's perspective so these message events are outgoing.
29
+ /**
30
+ * @todo Provide the reference to the exact event handler
31
+ * that called this `client.send()`.
32
+ */
33
+ client.addEventListener('message', (event) => {
34
+ logOutgoingClientMessage(event)
35
+ })
36
+
37
+ client.addEventListener('close', (event) => {
38
+ logConnectionClose(event)
39
+ })
40
+
41
+ // Log client errors (connection closures due to errors).
42
+ client.socket.addEventListener('error', (event) => {
43
+ logClientError(event)
44
+ })
45
+
46
+ client.send = new Proxy(client.send, {
47
+ apply(target, thisArg, args) {
48
+ const [data] = args
49
+ const messageEvent = new MessageEvent('message', { data })
50
+ Object.defineProperties(messageEvent, {
51
+ currentTarget: {
52
+ enumerable: true,
53
+ writable: false,
54
+ value: client.socket,
55
+ },
56
+ target: {
57
+ enumerable: true,
58
+ writable: false,
59
+ value: client.socket,
60
+ },
61
+ })
62
+
63
+ queueMicrotask(() => {
64
+ logIncomingMockedClientMessage(messageEvent)
65
+ })
66
+
67
+ return Reflect.apply(target, thisArg, args)
68
+ },
69
+ })
70
+
71
+ server.addEventListener(
72
+ 'open',
73
+ () => {
74
+ server.addEventListener('message', (event) => {
75
+ logIncomingServerMessage(event)
76
+ })
77
+ },
78
+ { once: true },
79
+ )
80
+
81
+ // Log outgoing client events initiated by the event handler.
82
+ // The actual client never sent these but the handler did.
83
+ server.send = new Proxy(server.send, {
84
+ apply(target, thisArg, args) {
85
+ const [data] = args
86
+ const messageEvent = new MessageEvent('message', { data })
87
+ Object.defineProperties(messageEvent, {
88
+ currentTarget: {
89
+ enumerable: true,
90
+ writable: false,
91
+ value: server.socket,
92
+ },
93
+ target: {
94
+ enumerable: true,
95
+ writable: false,
96
+ value: server.socket,
97
+ },
98
+ })
99
+
100
+ logOutgoingMockedClientMessage(messageEvent)
101
+
102
+ return Reflect.apply(target, thisArg, args)
103
+ },
104
+ })
105
+ }
106
+
107
+ /**
108
+ * Prints the WebSocket connection.
109
+ * This is meant to be logged by every WebSocket handler
110
+ * that intercepted this connection. This helps you see
111
+ * what handlers observe this connection.
112
+ */
113
+ export function logConnectionOpen(client: WebSocketClientConnection) {
114
+ const publicUrl = toPublicUrl(client.url)
115
+
116
+ // eslint-disable-next-line no-console
117
+ console.groupCollapsed(
118
+ devUtils.formatMessage(`${getTimestamp()} %c▶%c ${publicUrl}`),
119
+ `color:${colors.system}`,
120
+ 'color:inherit',
121
+ )
122
+ // eslint-disable-next-line no-console
123
+ console.log('Client:', client.socket)
124
+ // eslint-disable-next-line no-console
125
+ console.groupEnd()
126
+ }
127
+
128
+ function logConnectionClose(event: CloseEvent) {
129
+ const target = event.target as WebSocket
130
+ const publicUrl = toPublicUrl(target.url)
131
+
132
+ // eslint-disable-next-line no-console
133
+ console.groupCollapsed(
134
+ devUtils.formatMessage(
135
+ `${getTimestamp({ milliseconds: true })} %c■%c ${publicUrl}`,
136
+ ),
137
+ `color:${colors.system}`,
138
+ 'color:inherit',
139
+ )
140
+ // eslint-disable-next-line no-console
141
+ console.log(event)
142
+ // eslint-disable-next-line no-console
143
+ console.groupEnd()
144
+ }
145
+
146
+ function logClientError(event: Event) {
147
+ const socket = event.target as WebSocket
148
+ const publicUrl = toPublicUrl(socket.url)
149
+
150
+ // eslint-disable-next-line no-console
151
+ console.groupCollapsed(
152
+ devUtils.formatMessage(
153
+ `${getTimestamp({ milliseconds: true })} %c\u00D7%c ${publicUrl}`,
154
+ ),
155
+ `color:${colors.system}`,
156
+ 'color:inherit',
157
+ )
158
+ // eslint-disable-next-line no-console
159
+ console.log(event)
160
+ // eslint-disable-next-line no-console
161
+ console.groupEnd()
162
+ }
163
+
164
+ /**
165
+ * Prints the outgoing client message.
166
+ */
167
+ async function logOutgoingClientMessage(event: MessageEvent<WebSocketData>) {
168
+ const byteLength = getMessageLength(event.data)
169
+ const publicData = await getPublicData(event.data)
170
+ const arrow = event.defaultPrevented ? '⇡' : '⬆'
171
+
172
+ // eslint-disable-next-line no-console
173
+ console.groupCollapsed(
174
+ devUtils.formatMessage(
175
+ `${getTimestamp({ milliseconds: true })} %c${arrow}%c ${publicData} %c${byteLength}%c`,
176
+ ),
177
+ `color:${colors.outgoing}`,
178
+ 'color:inherit',
179
+ 'color:gray;font-weight:normal',
180
+ 'color:inherit;font-weight:inherit',
181
+ )
182
+ // eslint-disable-next-line no-console
183
+ console.log(event)
184
+ // eslint-disable-next-line no-console
185
+ console.groupEnd()
186
+ }
187
+
188
+ /**
189
+ * Prints the outgoing client message initiated
190
+ * by `server.send()` in the event handler.
191
+ */
192
+ async function logOutgoingMockedClientMessage(
193
+ event: MessageEvent<WebSocketData>,
194
+ ) {
195
+ const byteLength = getMessageLength(event.data)
196
+ const publicData = await getPublicData(event.data)
197
+
198
+ // eslint-disable-next-line no-console
199
+ console.groupCollapsed(
200
+ devUtils.formatMessage(
201
+ `${getTimestamp({ milliseconds: true })} %c⬆%c ${publicData} %c${byteLength}%c`,
202
+ ),
203
+ `color:${colors.mocked}`,
204
+ 'color:inherit',
205
+ 'color:gray;font-weight:normal',
206
+ 'color:inherit;font-weight:inherit',
207
+ )
208
+ // eslint-disable-next-line no-console
209
+ console.log(event)
210
+ // eslint-disable-next-line no-console
211
+ console.groupEnd()
212
+ }
213
+
214
+ /**
215
+ * Prints the outgoing client message initiated
216
+ * by `client.send()` in the event handler.
217
+ */
218
+ async function logIncomingMockedClientMessage(
219
+ event: MessageEvent<WebSocketData>,
220
+ ) {
221
+ const byteLength = getMessageLength(event.data)
222
+ const publicData = await getPublicData(event.data)
223
+
224
+ // eslint-disable-next-line no-console
225
+ console.groupCollapsed(
226
+ devUtils.formatMessage(
227
+ `${getTimestamp({ milliseconds: true })} %c⬇%c ${publicData} %c${byteLength}%c`,
228
+ ),
229
+ `color:${colors.mocked}`,
230
+ 'color:inherit',
231
+ 'color:gray;font-weight:normal',
232
+ 'color:inherit;font-weight:inherit',
233
+ )
234
+ // eslint-disable-next-line no-console
235
+ console.log(event)
236
+ // eslint-disable-next-line no-console
237
+ console.groupEnd()
238
+ }
239
+
240
+ async function logIncomingServerMessage(event: MessageEvent<WebSocketData>) {
241
+ const byteLength = getMessageLength(event.data)
242
+ const publicData = await getPublicData(event.data)
243
+ const arrow = event.defaultPrevented ? '⇣' : '⬇'
244
+
245
+ // eslint-disable-next-line no-console
246
+ console.groupCollapsed(
247
+ devUtils.formatMessage(
248
+ `${getTimestamp({ milliseconds: true })} %c${arrow}%c ${publicData} %c${byteLength}%c`,
249
+ ),
250
+ `color:${colors.incoming}`,
251
+ 'color:inherit',
252
+ 'color:gray;font-weight:normal',
253
+ 'color:inherit;font-weight:inherit',
254
+ )
255
+ // eslint-disable-next-line no-console
256
+ console.log(event)
257
+ // eslint-disable-next-line no-console
258
+ console.groupEnd()
259
+ }
@@ -0,0 +1,16 @@
1
+ import { getMessageLength } from './getMessageLength'
2
+
3
+ it('returns the length of the string', () => {
4
+ expect(getMessageLength('')).toBe(0)
5
+ expect(getMessageLength('hello')).toBe(5)
6
+ })
7
+
8
+ it('returns the size of the Blob', () => {
9
+ expect(getMessageLength(new Blob())).toBe(0)
10
+ expect(getMessageLength(new Blob(['hello']))).toBe(5)
11
+ })
12
+
13
+ it('returns the byte length of ArrayBuffer', () => {
14
+ expect(getMessageLength(new ArrayBuffer(0))).toBe(0)
15
+ expect(getMessageLength(new ArrayBuffer(5))).toBe(5)
16
+ })
@@ -0,0 +1,19 @@
1
+ import type { WebSocketData } from '@mswjs/interceptors/lib/browser/interceptors/WebSocket'
2
+
3
+ /**
4
+ * Returns the byte length of the given WebSocket message.
5
+ * @example
6
+ * getMessageLength('hello') // 5
7
+ * getMessageLength(new Blob(['hello'])) // 5
8
+ */
9
+ export function getMessageLength(data: WebSocketData): number {
10
+ if (data instanceof Blob) {
11
+ return data.size
12
+ }
13
+
14
+ if (data instanceof ArrayBuffer) {
15
+ return data.byteLength
16
+ }
17
+
18
+ return new Blob([data]).size
19
+ }
@@ -0,0 +1,38 @@
1
+ import { getPublicData } from './getPublicData'
2
+
3
+ it('returns a short string as-is', async () => {
4
+ expect(await getPublicData('')).toBe('')
5
+ expect(await getPublicData('hello')).toBe('hello')
6
+ })
7
+
8
+ it('returns a truncated long string', async () => {
9
+ expect(await getPublicData('this is a very long string')).toBe(
10
+ 'this is a very long stri…',
11
+ )
12
+ })
13
+
14
+ it('returns a short Blob text as-is', async () => {
15
+ expect(await getPublicData(new Blob(['']))).toBe('Blob()')
16
+ expect(await getPublicData(new Blob(['hello']))).toBe('Blob(hello)')
17
+ })
18
+
19
+ it('returns a truncated long Blob text', async () => {
20
+ expect(await getPublicData(new Blob(['this is a very long string']))).toBe(
21
+ 'Blob(this is a very long stri…)',
22
+ )
23
+ })
24
+
25
+ it('returns a short ArrayBuffer text as-is', async () => {
26
+ expect(await getPublicData(new TextEncoder().encode(''))).toBe(
27
+ 'ArrayBuffer()',
28
+ )
29
+ expect(await getPublicData(new TextEncoder().encode('hello'))).toBe(
30
+ 'ArrayBuffer(hello)',
31
+ )
32
+ })
33
+
34
+ it('returns a truncated ArrayBuffer text', async () => {
35
+ expect(
36
+ await getPublicData(new TextEncoder().encode('this is a very long string')),
37
+ ).toBe('ArrayBuffer(this is a very long stri…)')
38
+ })
@@ -0,0 +1,17 @@
1
+ import { WebSocketData } from '@mswjs/interceptors/WebSocket'
2
+ import { truncateMessage } from './truncateMessage'
3
+
4
+ export async function getPublicData(data: WebSocketData): Promise<string> {
5
+ if (data instanceof Blob) {
6
+ const text = await data.text()
7
+ return `Blob(${truncateMessage(text)})`
8
+ }
9
+
10
+ // Handle all ArrayBuffer-like objects.
11
+ if (typeof data === 'object' && 'byteLength' in data) {
12
+ const text = new TextDecoder().decode(data)
13
+ return `ArrayBuffer(${truncateMessage(text)})`
14
+ }
15
+
16
+ return truncateMessage(data)
17
+ }
@@ -0,0 +1,12 @@
1
+ import { truncateMessage } from './truncateMessage'
2
+
3
+ it('returns a short string as-is', () => {
4
+ expect(truncateMessage('')).toBe('')
5
+ expect(truncateMessage('hello')).toBe('hello')
6
+ })
7
+
8
+ it('truncates a long string', () => {
9
+ expect(truncateMessage('this is a very long string')).toBe(
10
+ 'this is a very long stri…',
11
+ )
12
+ })
@@ -0,0 +1,9 @@
1
+ const MAX_LENGTH = 24
2
+
3
+ export function truncateMessage(message: string): string {
4
+ if (message.length <= MAX_LENGTH) {
5
+ return message
6
+ }
7
+
8
+ return `${message.slice(0, MAX_LENGTH)}…`
9
+ }
@@ -0,0 +1,3 @@
1
+ import { WebSocketInterceptor } from '@mswjs/interceptors/WebSocket'
2
+
3
+ export const webSocketInterceptor = new WebSocketInterceptor()
@@ -0,0 +1,23 @@
1
+ /**
2
+ * @vitest-environment node-websocket
3
+ */
4
+ import { ws } from './ws'
5
+
6
+ it('exports the "link()" method', () => {
7
+ expect(ws).toHaveProperty('link')
8
+ expect(ws.link).toBeInstanceOf(Function)
9
+ })
10
+
11
+ it('throws an error when calling "ws.link()" without a URL argument', () => {
12
+ expect(() =>
13
+ // @ts-expect-error Intentionally invalid call.
14
+ ws.link(),
15
+ ).toThrow('Expected a WebSocket server URL but got undefined')
16
+ })
17
+
18
+ it('throws an error when given a non-path argument to "ws.link()"', () => {
19
+ expect(() =>
20
+ // @ts-expect-error Intentionally invalid argument.
21
+ ws.link(2),
22
+ ).toThrow('Expected a WebSocket server URL to be a valid path but got number')
23
+ })
package/src/core/ws.ts ADDED
@@ -0,0 +1,166 @@
1
+ import { invariant } from 'outvariant'
2
+ import type {
3
+ WebSocketData,
4
+ WebSocketClientConnectionProtocol,
5
+ } from '@mswjs/interceptors/WebSocket'
6
+ import {
7
+ WebSocketHandler,
8
+ kEmitter,
9
+ type WebSocketHandlerEventMap,
10
+ } from './handlers/WebSocketHandler'
11
+ import { Path, isPath } from './utils/matching/matchRequestUrl'
12
+ import { WebSocketClientManager } from './ws/WebSocketClientManager'
13
+
14
+ function isBroadcastChannelWithUnref(
15
+ channel: BroadcastChannel,
16
+ ): channel is BroadcastChannel & NodeJS.RefCounted {
17
+ return typeof Reflect.get(channel, 'unref') !== 'undefined'
18
+ }
19
+
20
+ const webSocketChannel = new BroadcastChannel('msw:websocket-client-manager')
21
+
22
+ if (isBroadcastChannelWithUnref(webSocketChannel)) {
23
+ // Allows the Node.js thread to exit if it is the only active handle in the event system.
24
+ // https://nodejs.org/api/worker_threads.html#broadcastchannelunref
25
+ webSocketChannel.unref()
26
+ }
27
+
28
+ export type WebSocketEventListener<
29
+ EventType extends keyof WebSocketHandlerEventMap,
30
+ > = (...args: WebSocketHandlerEventMap[EventType]) => void
31
+
32
+ export type WebSocketLink = {
33
+ /**
34
+ * A set of all WebSocket clients connected
35
+ * to this link.
36
+ *
37
+ * @see {@link https://mswjs.io/docs/api/ws#clients `clients` API reference}
38
+ */
39
+ clients: Set<WebSocketClientConnectionProtocol>
40
+
41
+ /**
42
+ * Adds an event listener to this WebSocket link.
43
+ *
44
+ * @example
45
+ * const chat = ws.link('wss://chat.example.com')
46
+ * chat.addEventListener('connection', listener)
47
+ *
48
+ * @see {@link https://mswjs.io/docs/api/ws#onevent-listener `on()` API reference}
49
+ */
50
+ addEventListener<EventType extends keyof WebSocketHandlerEventMap>(
51
+ event: EventType,
52
+ listener: WebSocketEventListener<EventType>,
53
+ ): WebSocketHandler
54
+
55
+ /**
56
+ * Broadcasts the given data to all WebSocket clients.
57
+ *
58
+ * @example
59
+ * const service = ws.link('wss://example.com')
60
+ * service.addEventListener('connection', () => {
61
+ * service.broadcast('hello, everyone!')
62
+ * })
63
+ *
64
+ * @see {@link https://mswjs.io/docs/api/ws#broadcastdata `broadcast()` API reference}
65
+ */
66
+ broadcast(data: WebSocketData): void
67
+
68
+ /**
69
+ * Broadcasts the given data to all WebSocket clients
70
+ * except the ones provided in the `clients` argument.
71
+ *
72
+ * @example
73
+ * const service = ws.link('wss://example.com')
74
+ * service.addEventListener('connection', ({ client }) => {
75
+ * service.broadcastExcept(client, 'hi, the rest of you!')
76
+ * })
77
+ *
78
+ * @see {@link https://mswjs.io/docs/api/ws#broadcastexceptclients-data `broadcast()` API reference}
79
+ */
80
+ broadcastExcept(
81
+ clients:
82
+ | WebSocketClientConnectionProtocol
83
+ | Array<WebSocketClientConnectionProtocol>,
84
+ data: WebSocketData,
85
+ ): void
86
+ }
87
+
88
+ /**
89
+ * Intercepts outgoing WebSocket connections to the given URL.
90
+ *
91
+ * @example
92
+ * const chat = ws.link('wss://chat.example.com')
93
+ * chat.addEventListener('connection', ({ client }) => {
94
+ * client.send('hello from server!')
95
+ * })
96
+ */
97
+ function createWebSocketLinkHandler(url: Path): WebSocketLink {
98
+ invariant(url, 'Expected a WebSocket server URL but got undefined')
99
+
100
+ invariant(
101
+ isPath(url),
102
+ 'Expected a WebSocket server URL to be a valid path but got %s',
103
+ typeof url,
104
+ )
105
+
106
+ const clientManager = new WebSocketClientManager(webSocketChannel)
107
+
108
+ return {
109
+ get clients() {
110
+ return clientManager.clients
111
+ },
112
+ addEventListener(event, listener) {
113
+ const handler = new WebSocketHandler(url)
114
+
115
+ // Add the connection event listener for when the
116
+ // handler matches and emits a connection event.
117
+ // When that happens, store that connection in the
118
+ // set of all connections for reference.
119
+ handler[kEmitter].on('connection', async ({ client }) => {
120
+ await clientManager.addConnection(client)
121
+ })
122
+
123
+ // The "handleWebSocketEvent" function will invoke
124
+ // the "run()" method on the WebSocketHandler.
125
+ // If the handler matches, it will emit the "connection"
126
+ // event. Attach the user-defined listener to that event.
127
+ handler[kEmitter].on(event, listener)
128
+
129
+ return handler
130
+ },
131
+
132
+ broadcast(data) {
133
+ // This will invoke "send()" on the immediate clients
134
+ // in this runtime and post a message to the broadcast channel
135
+ // to trigger send for the clients in other runtimes.
136
+ this.broadcastExcept([], data)
137
+ },
138
+
139
+ broadcastExcept(clients, data) {
140
+ const ignoreClients = Array.prototype
141
+ .concat(clients)
142
+ .map((client) => client.id)
143
+
144
+ clientManager.clients.forEach((otherClient) => {
145
+ if (!ignoreClients.includes(otherClient.id)) {
146
+ otherClient.send(data)
147
+ }
148
+ })
149
+ },
150
+ }
151
+ }
152
+
153
+ /**
154
+ * A namespace to intercept and mock WebSocket connections.
155
+ *
156
+ * @example
157
+ * const chat = ws.link('wss://chat.example.com')
158
+ *
159
+ * @see {@link https://mswjs.io/docs/api/ws `ws` API reference}
160
+ * @see {@link https://mswjs.io/docs/basics/handling-websocket-events Handling WebSocket events}
161
+ */
162
+ export const ws = {
163
+ link: createWebSocketLinkHandler,
164
+ }
165
+
166
+ export { WebSocketData }
@@ -4,14 +4,15 @@ import { XMLHttpRequestInterceptor } from '@mswjs/interceptors/XMLHttpRequest'
4
4
  import { FetchInterceptor } from '@mswjs/interceptors/fetch'
5
5
  import { HandlersController } from '~/core/SetupApi'
6
6
  import type { RequestHandler } from '~/core/handlers/RequestHandler'
7
+ import type { WebSocketHandler } from '~/core/handlers/WebSocketHandler'
7
8
  import type { SetupServer } from './glossary'
8
9
  import { SetupServerCommonApi } from './SetupServerCommonApi'
9
10
 
10
11
  const store = new AsyncLocalStorage<RequestHandlersContext>()
11
12
 
12
13
  type RequestHandlersContext = {
13
- initialHandlers: Array<RequestHandler>
14
- handlers: Array<RequestHandler>
14
+ initialHandlers: Array<RequestHandler | WebSocketHandler>
15
+ handlers: Array<RequestHandler | WebSocketHandler>
15
16
  }
16
17
 
17
18
  /**
@@ -22,7 +23,7 @@ type RequestHandlersContext = {
22
23
  class AsyncHandlersController implements HandlersController {
23
24
  private rootContext: RequestHandlersContext
24
25
 
25
- constructor(initialHandlers: Array<RequestHandler>) {
26
+ constructor(initialHandlers: Array<RequestHandler | WebSocketHandler>) {
26
27
  this.rootContext = { initialHandlers, handlers: [] }
27
28
  }
28
29
 
@@ -30,18 +31,18 @@ class AsyncHandlersController implements HandlersController {
30
31
  return store.getStore() || this.rootContext
31
32
  }
32
33
 
33
- public prepend(runtimeHandlers: Array<RequestHandler>) {
34
+ public prepend(runtimeHandlers: Array<RequestHandler | WebSocketHandler>) {
34
35
  this.context.handlers.unshift(...runtimeHandlers)
35
36
  }
36
37
 
37
- public reset(nextHandlers: Array<RequestHandler>) {
38
+ public reset(nextHandlers: Array<RequestHandler | WebSocketHandler>) {
38
39
  const context = this.context
39
40
  context.handlers = []
40
41
  context.initialHandlers =
41
42
  nextHandlers.length > 0 ? nextHandlers : context.initialHandlers
42
43
  }
43
44
 
44
- public currentHandlers(): Array<RequestHandler> {
45
+ public currentHandlers(): Array<RequestHandler | WebSocketHandler> {
45
46
  const { initialHandlers, handlers } = this.context
46
47
  return handlers.concat(initialHandlers)
47
48
  }
@@ -51,7 +52,7 @@ export class SetupServerApi
51
52
  extends SetupServerCommonApi
52
53
  implements SetupServer
53
54
  {
54
- constructor(handlers: Array<RequestHandler>) {
55
+ constructor(handlers: Array<RequestHandler | WebSocketHandler>) {
55
56
  super(
56
57
  [ClientRequestInterceptor, XMLHttpRequestInterceptor, FetchInterceptor],
57
58
  handlers,