@depup/mswjs__interceptors 0.41.3-depup.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (248) hide show
  1. package/ClientRequest/package.json +11 -0
  2. package/LICENSE.md +9 -0
  3. package/README.md +31 -0
  4. package/RemoteHttpInterceptor/package.json +11 -0
  5. package/WebSocket/package.json +12 -0
  6. package/XMLHttpRequest/package.json +12 -0
  7. package/changes.json +10 -0
  8. package/fetch/package.json +12 -0
  9. package/lib/browser/Interceptor-Deczogc8.d.cts +65 -0
  10. package/lib/browser/Interceptor-gqKgs-aF.d.mts +65 -0
  11. package/lib/browser/XMLHttpRequest-BACqefB-.cjs +761 -0
  12. package/lib/browser/XMLHttpRequest-BACqefB-.cjs.map +1 -0
  13. package/lib/browser/XMLHttpRequest-BvxZV0WU.mjs +756 -0
  14. package/lib/browser/XMLHttpRequest-BvxZV0WU.mjs.map +1 -0
  15. package/lib/browser/bufferUtils-BiiO6HZv.mjs +20 -0
  16. package/lib/browser/bufferUtils-BiiO6HZv.mjs.map +1 -0
  17. package/lib/browser/bufferUtils-Uc0eRItL.cjs +38 -0
  18. package/lib/browser/bufferUtils-Uc0eRItL.cjs.map +1 -0
  19. package/lib/browser/createRequestId-Cs4oXfa1.cjs +205 -0
  20. package/lib/browser/createRequestId-Cs4oXfa1.cjs.map +1 -0
  21. package/lib/browser/createRequestId-DQcIlohW.mjs +170 -0
  22. package/lib/browser/createRequestId-DQcIlohW.mjs.map +1 -0
  23. package/lib/browser/fetch-DdKEdDOR.mjs +248 -0
  24. package/lib/browser/fetch-DdKEdDOR.mjs.map +1 -0
  25. package/lib/browser/fetch-U3v3Y4ap.cjs +253 -0
  26. package/lib/browser/fetch-U3v3Y4ap.cjs.map +1 -0
  27. package/lib/browser/getRawRequest-BTaNLFr0.mjs +218 -0
  28. package/lib/browser/getRawRequest-BTaNLFr0.mjs.map +1 -0
  29. package/lib/browser/getRawRequest-zx8rUJL2.cjs +259 -0
  30. package/lib/browser/getRawRequest-zx8rUJL2.cjs.map +1 -0
  31. package/lib/browser/glossary-BdLS4k1H.d.cts +70 -0
  32. package/lib/browser/glossary-DYwOrogs.d.mts +70 -0
  33. package/lib/browser/handleRequest-CvX2G-Lz.cjs +189 -0
  34. package/lib/browser/handleRequest-CvX2G-Lz.cjs.map +1 -0
  35. package/lib/browser/handleRequest-D7kpTI5U.mjs +178 -0
  36. package/lib/browser/handleRequest-D7kpTI5U.mjs.map +1 -0
  37. package/lib/browser/hasConfigurableGlobal-BvCTG97d.cjs +45 -0
  38. package/lib/browser/hasConfigurableGlobal-BvCTG97d.cjs.map +1 -0
  39. package/lib/browser/hasConfigurableGlobal-npXitu1-.mjs +33 -0
  40. package/lib/browser/hasConfigurableGlobal-npXitu1-.mjs.map +1 -0
  41. package/lib/browser/index.cjs +70 -0
  42. package/lib/browser/index.cjs.map +1 -0
  43. package/lib/browser/index.d.cts +96 -0
  44. package/lib/browser/index.d.mts +96 -0
  45. package/lib/browser/index.mjs +56 -0
  46. package/lib/browser/index.mjs.map +1 -0
  47. package/lib/browser/interceptors/WebSocket/index.cjs +622 -0
  48. package/lib/browser/interceptors/WebSocket/index.cjs.map +1 -0
  49. package/lib/browser/interceptors/WebSocket/index.d.cts +277 -0
  50. package/lib/browser/interceptors/WebSocket/index.d.mts +277 -0
  51. package/lib/browser/interceptors/WebSocket/index.mjs +615 -0
  52. package/lib/browser/interceptors/WebSocket/index.mjs.map +1 -0
  53. package/lib/browser/interceptors/XMLHttpRequest/index.cjs +7 -0
  54. package/lib/browser/interceptors/XMLHttpRequest/index.d.cts +15 -0
  55. package/lib/browser/interceptors/XMLHttpRequest/index.d.mts +15 -0
  56. package/lib/browser/interceptors/XMLHttpRequest/index.mjs +7 -0
  57. package/lib/browser/interceptors/fetch/index.cjs +6 -0
  58. package/lib/browser/interceptors/fetch/index.d.cts +13 -0
  59. package/lib/browser/interceptors/fetch/index.d.mts +13 -0
  60. package/lib/browser/interceptors/fetch/index.mjs +6 -0
  61. package/lib/browser/presets/browser.cjs +17 -0
  62. package/lib/browser/presets/browser.cjs.map +1 -0
  63. package/lib/browser/presets/browser.d.cts +12 -0
  64. package/lib/browser/presets/browser.d.mts +14 -0
  65. package/lib/browser/presets/browser.mjs +17 -0
  66. package/lib/browser/presets/browser.mjs.map +1 -0
  67. package/lib/browser/resolveWebSocketUrl-6K6EgqsA.cjs +31 -0
  68. package/lib/browser/resolveWebSocketUrl-6K6EgqsA.cjs.map +1 -0
  69. package/lib/browser/resolveWebSocketUrl-C83-x9iE.mjs +25 -0
  70. package/lib/browser/resolveWebSocketUrl-C83-x9iE.mjs.map +1 -0
  71. package/lib/node/BatchInterceptor-3LnAnLTx.cjs +49 -0
  72. package/lib/node/BatchInterceptor-3LnAnLTx.cjs.map +1 -0
  73. package/lib/node/BatchInterceptor-D7mXzHcQ.d.mts +26 -0
  74. package/lib/node/BatchInterceptor-DFaBPilf.mjs +44 -0
  75. package/lib/node/BatchInterceptor-DFaBPilf.mjs.map +1 -0
  76. package/lib/node/BatchInterceptor-D_YqR8qU.d.cts +26 -0
  77. package/lib/node/ClientRequest-2rDe54Ui.cjs +1043 -0
  78. package/lib/node/ClientRequest-2rDe54Ui.cjs.map +1 -0
  79. package/lib/node/ClientRequest-Ca8Qykuv.mjs +1034 -0
  80. package/lib/node/ClientRequest-Ca8Qykuv.mjs.map +1 -0
  81. package/lib/node/Interceptor-DEazpLJd.d.mts +133 -0
  82. package/lib/node/Interceptor-DJ2akVWI.d.cts +133 -0
  83. package/lib/node/RemoteHttpInterceptor.cjs +154 -0
  84. package/lib/node/RemoteHttpInterceptor.cjs.map +1 -0
  85. package/lib/node/RemoteHttpInterceptor.d.cts +39 -0
  86. package/lib/node/RemoteHttpInterceptor.d.mts +39 -0
  87. package/lib/node/RemoteHttpInterceptor.mjs +152 -0
  88. package/lib/node/RemoteHttpInterceptor.mjs.map +1 -0
  89. package/lib/node/XMLHttpRequest-B7kJdYYI.cjs +763 -0
  90. package/lib/node/XMLHttpRequest-B7kJdYYI.cjs.map +1 -0
  91. package/lib/node/XMLHttpRequest-C8dIZpds.mjs +757 -0
  92. package/lib/node/XMLHttpRequest-C8dIZpds.mjs.map +1 -0
  93. package/lib/node/bufferUtils-DiCTqG-7.cjs +38 -0
  94. package/lib/node/bufferUtils-DiCTqG-7.cjs.map +1 -0
  95. package/lib/node/bufferUtils-_8XfKIfX.mjs +20 -0
  96. package/lib/node/bufferUtils-_8XfKIfX.mjs.map +1 -0
  97. package/lib/node/chunk-CbDLau6x.cjs +34 -0
  98. package/lib/node/fetch-BmXpK10r.cjs +272 -0
  99. package/lib/node/fetch-BmXpK10r.cjs.map +1 -0
  100. package/lib/node/fetch-G1DVwDKG.mjs +265 -0
  101. package/lib/node/fetch-G1DVwDKG.mjs.map +1 -0
  102. package/lib/node/fetchUtils-BaY5iWXw.cjs +419 -0
  103. package/lib/node/fetchUtils-BaY5iWXw.cjs.map +1 -0
  104. package/lib/node/fetchUtils-CoU35g3M.mjs +359 -0
  105. package/lib/node/fetchUtils-CoU35g3M.mjs.map +1 -0
  106. package/lib/node/getRawRequest-BavnMWh_.cjs +36 -0
  107. package/lib/node/getRawRequest-BavnMWh_.cjs.map +1 -0
  108. package/lib/node/getRawRequest-DnwmXyOW.mjs +24 -0
  109. package/lib/node/getRawRequest-DnwmXyOW.mjs.map +1 -0
  110. package/lib/node/glossary-BLKRyLBd.cjs +12 -0
  111. package/lib/node/glossary-BLKRyLBd.cjs.map +1 -0
  112. package/lib/node/glossary-glQBRnVD.mjs +6 -0
  113. package/lib/node/glossary-glQBRnVD.mjs.map +1 -0
  114. package/lib/node/handleRequest-Bb7Y-XLw.cjs +220 -0
  115. package/lib/node/handleRequest-Bb7Y-XLw.cjs.map +1 -0
  116. package/lib/node/handleRequest-Y97UwBbF.mjs +190 -0
  117. package/lib/node/handleRequest-Y97UwBbF.mjs.map +1 -0
  118. package/lib/node/hasConfigurableGlobal-C97fWuaA.cjs +26 -0
  119. package/lib/node/hasConfigurableGlobal-C97fWuaA.cjs.map +1 -0
  120. package/lib/node/hasConfigurableGlobal-DBJA0vjm.mjs +20 -0
  121. package/lib/node/hasConfigurableGlobal-DBJA0vjm.mjs.map +1 -0
  122. package/lib/node/index-BMbJ8FXL.d.cts +113 -0
  123. package/lib/node/index-C0YAQ36w.d.mts +113 -0
  124. package/lib/node/index.cjs +54 -0
  125. package/lib/node/index.cjs.map +1 -0
  126. package/lib/node/index.d.cts +75 -0
  127. package/lib/node/index.d.mts +75 -0
  128. package/lib/node/index.mjs +40 -0
  129. package/lib/node/index.mjs.map +1 -0
  130. package/lib/node/interceptors/ClientRequest/index.cjs +6 -0
  131. package/lib/node/interceptors/ClientRequest/index.d.cts +2 -0
  132. package/lib/node/interceptors/ClientRequest/index.d.mts +3 -0
  133. package/lib/node/interceptors/ClientRequest/index.mjs +6 -0
  134. package/lib/node/interceptors/XMLHttpRequest/index.cjs +6 -0
  135. package/lib/node/interceptors/XMLHttpRequest/index.d.cts +14 -0
  136. package/lib/node/interceptors/XMLHttpRequest/index.d.mts +14 -0
  137. package/lib/node/interceptors/XMLHttpRequest/index.mjs +6 -0
  138. package/lib/node/interceptors/fetch/index.cjs +5 -0
  139. package/lib/node/interceptors/fetch/index.d.cts +12 -0
  140. package/lib/node/interceptors/fetch/index.d.mts +12 -0
  141. package/lib/node/interceptors/fetch/index.mjs +5 -0
  142. package/lib/node/node-DwCc6iuP.mjs +27 -0
  143. package/lib/node/node-DwCc6iuP.mjs.map +1 -0
  144. package/lib/node/node-dKdAf3tC.cjs +39 -0
  145. package/lib/node/node-dKdAf3tC.cjs.map +1 -0
  146. package/lib/node/presets/node.cjs +22 -0
  147. package/lib/node/presets/node.cjs.map +1 -0
  148. package/lib/node/presets/node.d.cts +13 -0
  149. package/lib/node/presets/node.d.mts +15 -0
  150. package/lib/node/presets/node.mjs +22 -0
  151. package/lib/node/presets/node.mjs.map +1 -0
  152. package/lib/node/utils/node/index.cjs +4 -0
  153. package/lib/node/utils/node/index.d.cts +16 -0
  154. package/lib/node/utils/node/index.d.mts +16 -0
  155. package/lib/node/utils/node/index.mjs +3 -0
  156. package/package.json +204 -0
  157. package/presets/browser/package.json +5 -0
  158. package/presets/node/package.json +11 -0
  159. package/src/BatchInterceptor.test.ts +255 -0
  160. package/src/BatchInterceptor.ts +95 -0
  161. package/src/Interceptor.test.ts +205 -0
  162. package/src/Interceptor.ts +249 -0
  163. package/src/InterceptorError.ts +7 -0
  164. package/src/RemoteHttpInterceptor.ts +251 -0
  165. package/src/RequestController.test.ts +104 -0
  166. package/src/RequestController.ts +109 -0
  167. package/src/createRequestId.test.ts +7 -0
  168. package/src/createRequestId.ts +9 -0
  169. package/src/getRawRequest.ts +21 -0
  170. package/src/glossary.ts +37 -0
  171. package/src/index.ts +15 -0
  172. package/src/interceptors/ClientRequest/MockHttpSocket.ts +724 -0
  173. package/src/interceptors/ClientRequest/agents.ts +110 -0
  174. package/src/interceptors/ClientRequest/index.test.ts +75 -0
  175. package/src/interceptors/ClientRequest/index.ts +193 -0
  176. package/src/interceptors/ClientRequest/utils/getIncomingMessageBody.test.ts +54 -0
  177. package/src/interceptors/ClientRequest/utils/getIncomingMessageBody.ts +45 -0
  178. package/src/interceptors/ClientRequest/utils/normalizeClientRequestArgs.test.ts +427 -0
  179. package/src/interceptors/ClientRequest/utils/normalizeClientRequestArgs.ts +268 -0
  180. package/src/interceptors/ClientRequest/utils/parserUtils.ts +48 -0
  181. package/src/interceptors/ClientRequest/utils/recordRawHeaders.test.ts +258 -0
  182. package/src/interceptors/ClientRequest/utils/recordRawHeaders.ts +262 -0
  183. package/src/interceptors/Socket/MockSocket.test.ts +264 -0
  184. package/src/interceptors/Socket/MockSocket.ts +58 -0
  185. package/src/interceptors/Socket/utils/baseUrlFromConnectionOptions.ts +26 -0
  186. package/src/interceptors/Socket/utils/normalizeSocketWriteArgs.test.ts +52 -0
  187. package/src/interceptors/Socket/utils/normalizeSocketWriteArgs.ts +33 -0
  188. package/src/interceptors/WebSocket/WebSocketClassTransport.ts +116 -0
  189. package/src/interceptors/WebSocket/WebSocketClientConnection.ts +152 -0
  190. package/src/interceptors/WebSocket/WebSocketOverride.ts +252 -0
  191. package/src/interceptors/WebSocket/WebSocketServerConnection.ts +420 -0
  192. package/src/interceptors/WebSocket/WebSocketTransport.ts +39 -0
  193. package/src/interceptors/WebSocket/index.ts +191 -0
  194. package/src/interceptors/WebSocket/utils/bindEvent.test.ts +27 -0
  195. package/src/interceptors/WebSocket/utils/bindEvent.ts +21 -0
  196. package/src/interceptors/WebSocket/utils/events.test.ts +101 -0
  197. package/src/interceptors/WebSocket/utils/events.ts +94 -0
  198. package/src/interceptors/XMLHttpRequest/XMLHttpRequestController.ts +746 -0
  199. package/src/interceptors/XMLHttpRequest/XMLHttpRequestProxy.ts +121 -0
  200. package/src/interceptors/XMLHttpRequest/index.ts +61 -0
  201. package/src/interceptors/XMLHttpRequest/polyfills/EventPolyfill.ts +51 -0
  202. package/src/interceptors/XMLHttpRequest/polyfills/ProgressEventPolyfill.ts +17 -0
  203. package/src/interceptors/XMLHttpRequest/utils/concatArrayBuffer.ts +12 -0
  204. package/src/interceptors/XMLHttpRequest/utils/concateArrayBuffer.test.ts +12 -0
  205. package/src/interceptors/XMLHttpRequest/utils/createEvent.test.ts +26 -0
  206. package/src/interceptors/XMLHttpRequest/utils/createEvent.ts +41 -0
  207. package/src/interceptors/XMLHttpRequest/utils/createResponse.ts +49 -0
  208. package/src/interceptors/XMLHttpRequest/utils/getBodyByteLength.test.ts +164 -0
  209. package/src/interceptors/XMLHttpRequest/utils/getBodyByteLength.ts +16 -0
  210. package/src/interceptors/XMLHttpRequest/utils/isDomParserSupportedType.ts +14 -0
  211. package/src/interceptors/fetch/index.ts +214 -0
  212. package/src/interceptors/fetch/utils/brotli-decompress.browser.ts +14 -0
  213. package/src/interceptors/fetch/utils/brotli-decompress.ts +31 -0
  214. package/src/interceptors/fetch/utils/createNetworkError.ts +5 -0
  215. package/src/interceptors/fetch/utils/decompression.ts +85 -0
  216. package/src/interceptors/fetch/utils/followRedirect.ts +114 -0
  217. package/src/presets/browser.ts +11 -0
  218. package/src/presets/node.ts +13 -0
  219. package/src/utils/bufferUtils.test.ts +21 -0
  220. package/src/utils/bufferUtils.ts +22 -0
  221. package/src/utils/canParseUrl.ts +13 -0
  222. package/src/utils/cloneObject.test.ts +94 -0
  223. package/src/utils/cloneObject.ts +36 -0
  224. package/src/utils/createProxy.test.ts +164 -0
  225. package/src/utils/createProxy.ts +104 -0
  226. package/src/utils/emitAsync.ts +25 -0
  227. package/src/utils/fetchUtils.ts +119 -0
  228. package/src/utils/findPropertySource.test.ts +27 -0
  229. package/src/utils/findPropertySource.ts +20 -0
  230. package/src/utils/getCleanUrl.test.ts +32 -0
  231. package/src/utils/getCleanUrl.ts +6 -0
  232. package/src/utils/getUrlByRequestOptions.test.ts +163 -0
  233. package/src/utils/getUrlByRequestOptions.ts +152 -0
  234. package/src/utils/getValueBySymbol.test.ts +14 -0
  235. package/src/utils/getValueBySymbol.ts +19 -0
  236. package/src/utils/handleRequest.ts +205 -0
  237. package/src/utils/hasConfigurableGlobal.test.ts +83 -0
  238. package/src/utils/hasConfigurableGlobal.ts +34 -0
  239. package/src/utils/isNodeLikeError.ts +13 -0
  240. package/src/utils/isObject.test.ts +21 -0
  241. package/src/utils/isObject.ts +8 -0
  242. package/src/utils/isPropertyAccessible.ts +19 -0
  243. package/src/utils/nextTick.ts +11 -0
  244. package/src/utils/node/index.ts +39 -0
  245. package/src/utils/parseJson.test.ts +10 -0
  246. package/src/utils/parseJson.ts +12 -0
  247. package/src/utils/resolveWebSocketUrl.ts +45 -0
  248. package/src/utils/responseUtils.ts +59 -0
@@ -0,0 +1,205 @@
1
+ import { describe, vi, it, expect, afterEach } from 'vitest'
2
+ import {
3
+ Interceptor,
4
+ getGlobalSymbol,
5
+ deleteGlobalSymbol,
6
+ InterceptorReadyState,
7
+ } from './Interceptor'
8
+ import { nextTickAsync } from './utils/nextTick'
9
+
10
+ const symbol = Symbol('test')
11
+
12
+ afterEach(() => {
13
+ deleteGlobalSymbol(symbol)
14
+ })
15
+
16
+ it('does not set a maximum listeners limit', () => {
17
+ const interceptor = new Interceptor(symbol)
18
+ expect(interceptor['emitter'].getMaxListeners()).toBe(0)
19
+ })
20
+
21
+ describe('on()', () => {
22
+ it('adds a new listener using "on()"', () => {
23
+ const interceptor = new Interceptor(symbol)
24
+ expect(interceptor['emitter'].listenerCount('event')).toBe(0)
25
+
26
+ const listener = vi.fn()
27
+ interceptor.on('event', listener)
28
+ expect(interceptor['emitter'].listenerCount('event')).toBe(1)
29
+ })
30
+ })
31
+
32
+ describe('once()', () => {
33
+ it('calls the listener only once', () => {
34
+ const interceptor = new Interceptor(symbol)
35
+ const listener = vi.fn()
36
+
37
+ interceptor.once('foo', listener)
38
+ expect(listener).not.toHaveBeenCalled()
39
+
40
+ interceptor['emitter'].emit('foo', 'bar')
41
+
42
+ expect(listener).toHaveBeenCalledTimes(1)
43
+ expect(listener).toHaveBeenCalledWith('bar')
44
+
45
+ listener.mockReset()
46
+
47
+ interceptor['emitter'].emit('foo', 'baz')
48
+ interceptor['emitter'].emit('foo', 'xyz')
49
+ expect(listener).toHaveBeenCalledTimes(0)
50
+ })
51
+ })
52
+
53
+ describe('off()', () => {
54
+ it('removes a listener using "off()"', () => {
55
+ const interceptor = new Interceptor(symbol)
56
+ expect(interceptor['emitter'].listenerCount('event')).toBe(0)
57
+
58
+ const listener = vi.fn()
59
+ interceptor.on('event', listener)
60
+ expect(interceptor['emitter'].listenerCount('event')).toBe(1)
61
+
62
+ interceptor.off('event', listener)
63
+ expect(interceptor['emitter'].listenerCount('event')).toBe(0)
64
+ })
65
+ })
66
+
67
+ describe('persistence', () => {
68
+ it('stores global reference to the applied interceptor', () => {
69
+ const interceptor = new Interceptor(symbol)
70
+ interceptor.apply()
71
+
72
+ expect(getGlobalSymbol(symbol)).toEqual(interceptor)
73
+ })
74
+
75
+ it('deletes global reference when the interceptor is disposed', () => {
76
+ const interceptor = new Interceptor(symbol)
77
+
78
+ interceptor.apply()
79
+ interceptor.dispose()
80
+
81
+ expect(getGlobalSymbol(symbol)).toBeUndefined()
82
+ })
83
+ })
84
+
85
+ describe('readyState', () => {
86
+ it('sets the state to "INACTIVE" when the interceptor is created', () => {
87
+ const interceptor = new Interceptor(symbol)
88
+ expect(interceptor.readyState).toBe(InterceptorReadyState.INACTIVE)
89
+ })
90
+
91
+ it('leaves state as "INACTIVE" if the interceptor failed the environment check', async () => {
92
+ class MyInterceptor extends Interceptor<any> {
93
+ protected checkEnvironment(): boolean {
94
+ return false
95
+ }
96
+ }
97
+ const interceptor = new MyInterceptor(symbol)
98
+ interceptor.apply()
99
+
100
+ expect(interceptor.readyState).toBe(InterceptorReadyState.INACTIVE)
101
+ })
102
+
103
+ it('performs state transition when the interceptor is applying', async () => {
104
+ const interceptor = new Interceptor(symbol)
105
+ interceptor.apply()
106
+
107
+ // The interceptor's state transitions to APPLIED immediately.
108
+ // The only exception is if something throws during the setup.
109
+ expect(interceptor.readyState).toBe(InterceptorReadyState.APPLIED)
110
+ })
111
+
112
+ it('performs state transition when disposing of the interceptor', async () => {
113
+ const interceptor = new Interceptor(symbol)
114
+ interceptor.apply()
115
+ interceptor.dispose()
116
+
117
+ // The interceptor's state transitions to DISPOSED immediately.
118
+ // The only exception is if something throws during the teardown.
119
+ expect(interceptor.readyState).toBe(InterceptorReadyState.DISPOSED)
120
+ })
121
+ })
122
+
123
+ describe('apply', () => {
124
+ it('does not apply the same interceptor multiple times', () => {
125
+ const interceptor = new Interceptor(symbol)
126
+ const setupSpy = vi.spyOn(
127
+ interceptor,
128
+ // @ts-expect-error Protected property spy.
129
+ 'setup'
130
+ )
131
+
132
+ // Intentionally apply the same interceptor multiple times.
133
+ interceptor.apply()
134
+ interceptor.apply()
135
+ interceptor.apply()
136
+
137
+ // The "setup" must not be called repeatedly.
138
+ expect(setupSpy).toHaveBeenCalledTimes(1)
139
+
140
+ expect(getGlobalSymbol(symbol)).toEqual(interceptor)
141
+ })
142
+
143
+ it('does not call "apply" if the interceptor fails environment check', () => {
144
+ class MyInterceptor extends Interceptor<{}> {
145
+ checkEnvironment() {
146
+ return false
147
+ }
148
+ }
149
+
150
+ const interceptor = new MyInterceptor(Symbol('test'))
151
+ const setupSpy = vi.spyOn(
152
+ interceptor,
153
+ // @ts-expect-error Protected property spy.
154
+ 'setup'
155
+ )
156
+ interceptor.apply()
157
+
158
+ expect(setupSpy).not.toHaveBeenCalled()
159
+ })
160
+
161
+ it('proxies listeners from new interceptor to already running interceptor', () => {
162
+ const firstInterceptor = new Interceptor(symbol)
163
+ const secondInterceptor = new Interceptor(symbol)
164
+
165
+ firstInterceptor.apply()
166
+ const firstListener = vi.fn()
167
+ firstInterceptor.on('test', firstListener)
168
+
169
+ secondInterceptor.apply()
170
+ const secondListener = vi.fn()
171
+ secondInterceptor.on('test', secondListener)
172
+
173
+ // Emitting event in the first interceptor will bubble to the second one.
174
+ firstInterceptor['emitter'].emit('test', 'hello world')
175
+
176
+ expect(firstListener).toHaveBeenCalledTimes(1)
177
+ expect(firstListener).toHaveBeenCalledWith('hello world')
178
+
179
+ expect(secondListener).toHaveBeenCalledTimes(1)
180
+ expect(secondListener).toHaveBeenCalledWith('hello world')
181
+
182
+ expect(secondInterceptor['emitter'].listenerCount('test')).toBe(0)
183
+ })
184
+ })
185
+
186
+ describe('dispose', () => {
187
+ it('removes all listeners when the interceptor is disposed', async () => {
188
+ const interceptor = new Interceptor(symbol)
189
+
190
+ interceptor.apply()
191
+ const listener = vi.fn()
192
+ interceptor.on('test', listener)
193
+ interceptor.dispose()
194
+
195
+ // Even after emitting an event, the listener must not get called.
196
+ interceptor['emitter'].emit('test')
197
+ expect(listener).not.toHaveBeenCalled()
198
+
199
+ // The listener must not be called on the next tick either.
200
+ await nextTickAsync(() => {
201
+ interceptor['emitter'].emit('test')
202
+ expect(listener).not.toHaveBeenCalled()
203
+ })
204
+ })
205
+ })
@@ -0,0 +1,249 @@
1
+ import { Logger } from '@open-draft/logger'
2
+ import { Emitter, Listener } from 'strict-event-emitter'
3
+
4
+ export type InterceptorEventMap = Record<string, any>
5
+ export type InterceptorSubscription = () => void
6
+
7
+ /**
8
+ * Request header name to detect when a single request
9
+ * is being handled by nested interceptors (XHR -> ClientRequest).
10
+ * Obscure by design to prevent collisions with user-defined headers.
11
+ * Ideally, come up with the Interceptor-level mechanism for this.
12
+ * @see https://github.com/mswjs/interceptors/issues/378
13
+ */
14
+ export const INTERNAL_REQUEST_ID_HEADER_NAME =
15
+ 'x-interceptors-internal-request-id'
16
+
17
+ export function getGlobalSymbol<V>(symbol: Symbol): V | undefined {
18
+ return (
19
+ // @ts-ignore https://github.com/Microsoft/TypeScript/issues/24587
20
+ globalThis[symbol] || undefined
21
+ )
22
+ }
23
+
24
+ function setGlobalSymbol(symbol: Symbol, value: any): void {
25
+ // @ts-ignore
26
+ globalThis[symbol] = value
27
+ }
28
+
29
+ export function deleteGlobalSymbol(symbol: Symbol): void {
30
+ // @ts-ignore
31
+ delete globalThis[symbol]
32
+ }
33
+
34
+ export enum InterceptorReadyState {
35
+ INACTIVE = 'INACTIVE',
36
+ APPLYING = 'APPLYING',
37
+ APPLIED = 'APPLIED',
38
+ DISPOSING = 'DISPOSING',
39
+ DISPOSED = 'DISPOSED',
40
+ }
41
+
42
+ export type ExtractEventNames<Events extends Record<string, any>> =
43
+ Events extends Record<infer EventName, any> ? EventName : never
44
+
45
+ export class Interceptor<Events extends InterceptorEventMap> {
46
+ protected emitter: Emitter<Events>
47
+ protected subscriptions: Array<InterceptorSubscription>
48
+ protected logger: Logger
49
+
50
+ public readyState: InterceptorReadyState
51
+
52
+ constructor(private readonly symbol: symbol) {
53
+ this.readyState = InterceptorReadyState.INACTIVE
54
+
55
+ this.emitter = new Emitter()
56
+ this.subscriptions = []
57
+ this.logger = new Logger(symbol.description!)
58
+
59
+ // Do not limit the maximum number of listeners
60
+ // so not to limit the maximum amount of parallel events emitted.
61
+ this.emitter.setMaxListeners(0)
62
+
63
+ this.logger.info('constructing the interceptor...')
64
+ }
65
+
66
+ /**
67
+ * Determine if this interceptor can be applied
68
+ * in the current environment.
69
+ */
70
+ protected checkEnvironment(): boolean {
71
+ return true
72
+ }
73
+
74
+ /**
75
+ * Apply this interceptor to the current process.
76
+ * Returns an already running interceptor instance if it's present.
77
+ */
78
+ public apply(): void {
79
+ const logger = this.logger.extend('apply')
80
+ logger.info('applying the interceptor...')
81
+
82
+ if (this.readyState === InterceptorReadyState.APPLIED) {
83
+ logger.info('intercepted already applied!')
84
+ return
85
+ }
86
+
87
+ const shouldApply = this.checkEnvironment()
88
+
89
+ if (!shouldApply) {
90
+ logger.info('the interceptor cannot be applied in this environment!')
91
+ return
92
+ }
93
+
94
+ this.readyState = InterceptorReadyState.APPLYING
95
+
96
+ // Whenever applying a new interceptor, check if it hasn't been applied already.
97
+ // This enables to apply the same interceptor multiple times, for example from a different
98
+ // interceptor, only proxying events but keeping the stubs in a single place.
99
+ const runningInstance = this.getInstance()
100
+
101
+ if (runningInstance) {
102
+ logger.info('found a running instance, reusing...')
103
+
104
+ // Proxy any listeners you set on this instance to the running instance.
105
+ this.on = (event, listener) => {
106
+ logger.info('proxying the "%s" listener', event)
107
+
108
+ // Add listeners to the running instance so they appear
109
+ // at the top of the event listeners list and are executed first.
110
+ runningInstance.emitter.addListener(event, listener)
111
+
112
+ // Ensure that once this interceptor instance is disposed,
113
+ // it removes all listeners it has appended to the running interceptor instance.
114
+ this.subscriptions.push(() => {
115
+ runningInstance.emitter.removeListener(event, listener)
116
+ logger.info('removed proxied "%s" listener!', event)
117
+ })
118
+
119
+ return this
120
+ }
121
+
122
+ this.readyState = InterceptorReadyState.APPLIED
123
+
124
+ return
125
+ }
126
+
127
+ logger.info('no running instance found, setting up a new instance...')
128
+
129
+ // Setup the interceptor.
130
+ this.setup()
131
+
132
+ // Store the newly applied interceptor instance globally.
133
+ this.setInstance()
134
+
135
+ this.readyState = InterceptorReadyState.APPLIED
136
+ }
137
+
138
+ /**
139
+ * Setup the module augments and stubs necessary for this interceptor.
140
+ * This method is not run if there's a running interceptor instance
141
+ * to prevent instantiating an interceptor multiple times.
142
+ */
143
+ protected setup(): void {}
144
+
145
+ /**
146
+ * Listen to the interceptor's public events.
147
+ */
148
+ public on<EventName extends ExtractEventNames<Events>>(
149
+ event: EventName,
150
+ listener: Listener<Events[EventName]>
151
+ ): this {
152
+ const logger = this.logger.extend('on')
153
+
154
+ if (
155
+ this.readyState === InterceptorReadyState.DISPOSING ||
156
+ this.readyState === InterceptorReadyState.DISPOSED
157
+ ) {
158
+ logger.info('cannot listen to events, already disposed!')
159
+ return this
160
+ }
161
+
162
+ logger.info('adding "%s" event listener:', event, listener)
163
+
164
+ this.emitter.on(event, listener)
165
+ return this
166
+ }
167
+
168
+ public once<EventName extends ExtractEventNames<Events>>(
169
+ event: EventName,
170
+ listener: Listener<Events[EventName]>
171
+ ): this {
172
+ this.emitter.once(event, listener)
173
+ return this
174
+ }
175
+
176
+ public off<EventName extends ExtractEventNames<Events>>(
177
+ event: EventName,
178
+ listener: Listener<Events[EventName]>
179
+ ): this {
180
+ this.emitter.off(event, listener)
181
+ return this
182
+ }
183
+
184
+ public removeAllListeners<EventName extends ExtractEventNames<Events>>(
185
+ event?: EventName
186
+ ): this {
187
+ this.emitter.removeAllListeners(event)
188
+ return this
189
+ }
190
+
191
+ /**
192
+ * Disposes of any side-effects this interceptor has introduced.
193
+ */
194
+ public dispose(): void {
195
+ const logger = this.logger.extend('dispose')
196
+
197
+ if (this.readyState === InterceptorReadyState.DISPOSED) {
198
+ logger.info('cannot dispose, already disposed!')
199
+ return
200
+ }
201
+
202
+ logger.info('disposing the interceptor...')
203
+ this.readyState = InterceptorReadyState.DISPOSING
204
+
205
+ if (!this.getInstance()) {
206
+ logger.info('no interceptors running, skipping dispose...')
207
+ return
208
+ }
209
+
210
+ // Delete the global symbol as soon as possible,
211
+ // indicating that the interceptor is no longer running.
212
+ this.clearInstance()
213
+
214
+ logger.info('global symbol deleted:', getGlobalSymbol(this.symbol))
215
+
216
+ if (this.subscriptions.length > 0) {
217
+ logger.info('disposing of %d subscriptions...', this.subscriptions.length)
218
+
219
+ for (const dispose of this.subscriptions) {
220
+ dispose()
221
+ }
222
+
223
+ this.subscriptions = []
224
+
225
+ logger.info('disposed of all subscriptions!', this.subscriptions.length)
226
+ }
227
+
228
+ this.emitter.removeAllListeners()
229
+ logger.info('destroyed the listener!')
230
+
231
+ this.readyState = InterceptorReadyState.DISPOSED
232
+ }
233
+
234
+ private getInstance(): this | undefined {
235
+ const instance = getGlobalSymbol<this>(this.symbol)
236
+ this.logger.info('retrieved global instance:', instance?.constructor?.name)
237
+ return instance
238
+ }
239
+
240
+ private setInstance(): void {
241
+ setGlobalSymbol(this.symbol, this)
242
+ this.logger.info('set global instance!', this.symbol.description)
243
+ }
244
+
245
+ private clearInstance(): void {
246
+ deleteGlobalSymbol(this.symbol)
247
+ this.logger.info('cleared global instance!', this.symbol.description)
248
+ }
249
+ }
@@ -0,0 +1,7 @@
1
+ export class InterceptorError extends Error {
2
+ constructor(message?: string) {
3
+ super(message)
4
+ this.name = 'InterceptorError'
5
+ Object.setPrototypeOf(this, InterceptorError.prototype)
6
+ }
7
+ }
@@ -0,0 +1,251 @@
1
+ import { ChildProcess } from 'child_process'
2
+ import { HttpRequestEventMap } from './glossary'
3
+ import { Interceptor } from './Interceptor'
4
+ import { BatchInterceptor } from './BatchInterceptor'
5
+ import { ClientRequestInterceptor } from './interceptors/ClientRequest'
6
+ import { XMLHttpRequestInterceptor } from './interceptors/XMLHttpRequest'
7
+ import { FetchInterceptor } from './interceptors/fetch'
8
+ import { handleRequest } from './utils/handleRequest'
9
+ import { RequestController } from './RequestController'
10
+ import { FetchResponse } from './utils/fetchUtils'
11
+ import { isResponseError } from './utils/responseUtils'
12
+
13
+ export interface SerializedRequest {
14
+ id: string
15
+ url: string
16
+ method: string
17
+ headers: Array<[string, string]>
18
+ credentials: RequestCredentials
19
+ body: string
20
+ }
21
+
22
+ interface RevivedRequest extends Omit<SerializedRequest, 'url' | 'headers'> {
23
+ url: URL
24
+ headers: Headers
25
+ }
26
+
27
+ export interface SerializedResponse {
28
+ status: number
29
+ statusText: string
30
+ headers: Array<[string, string]>
31
+ body: string
32
+ }
33
+
34
+ export class RemoteHttpInterceptor extends BatchInterceptor<
35
+ [ClientRequestInterceptor, XMLHttpRequestInterceptor, FetchInterceptor]
36
+ > {
37
+ constructor() {
38
+ super({
39
+ name: 'remote-interceptor',
40
+ interceptors: [
41
+ new ClientRequestInterceptor(),
42
+ new XMLHttpRequestInterceptor(),
43
+ new FetchInterceptor(),
44
+ ],
45
+ })
46
+ }
47
+
48
+ protected setup() {
49
+ super.setup()
50
+
51
+ let handleParentMessage: NodeJS.MessageListener
52
+
53
+ this.on('request', async ({ request, requestId, controller }) => {
54
+ // Send the stringified intercepted request to
55
+ // the parent process where the remote resolver is established.
56
+ const serializedRequest = JSON.stringify({
57
+ id: requestId,
58
+ method: request.method,
59
+ url: request.url,
60
+ headers: Array.from(request.headers.entries()),
61
+ credentials: request.credentials,
62
+ body: ['GET', 'HEAD'].includes(request.method)
63
+ ? null
64
+ : await request.text(),
65
+ } as SerializedRequest)
66
+
67
+ this.logger.info(
68
+ 'sent serialized request to the child:',
69
+ serializedRequest
70
+ )
71
+
72
+ process.send?.(`request:${serializedRequest}`)
73
+
74
+ const responsePromise = new Promise<void>((resolve) => {
75
+ handleParentMessage = (message) => {
76
+ if (typeof message !== 'string') {
77
+ return resolve()
78
+ }
79
+
80
+ if (message.startsWith(`response:${requestId}`)) {
81
+ const [, serializedResponse] =
82
+ message.match(/^response:.+?:(.+)$/) || []
83
+
84
+ if (!serializedResponse) {
85
+ return resolve()
86
+ }
87
+
88
+ const responseInit = JSON.parse(
89
+ serializedResponse
90
+ ) as SerializedResponse
91
+
92
+ const mockedResponse = new FetchResponse(responseInit.body, {
93
+ url: request.url,
94
+ status: responseInit.status,
95
+ statusText: responseInit.statusText,
96
+ headers: responseInit.headers,
97
+ })
98
+
99
+ /**
100
+ * @todo Support "errorWith" as well.
101
+ * This response handling from the child is incomplete.
102
+ */
103
+
104
+ controller.respondWith(mockedResponse)
105
+ return resolve()
106
+ }
107
+ }
108
+ })
109
+
110
+ // Listen for the mocked response message from the parent.
111
+ this.logger.info(
112
+ 'add "message" listener to the parent process',
113
+ handleParentMessage
114
+ )
115
+ process.addListener('message', handleParentMessage)
116
+
117
+ return responsePromise
118
+ })
119
+
120
+ this.subscriptions.push(() => {
121
+ process.removeListener('message', handleParentMessage)
122
+ })
123
+ }
124
+ }
125
+
126
+ export function requestReviver(key: string, value: any) {
127
+ switch (key) {
128
+ case 'url':
129
+ return new URL(value)
130
+
131
+ case 'headers':
132
+ return new Headers(value)
133
+
134
+ default:
135
+ return value
136
+ }
137
+ }
138
+
139
+ export interface RemoveResolverOptions {
140
+ process: ChildProcess
141
+ }
142
+
143
+ export class RemoteHttpResolver extends Interceptor<HttpRequestEventMap> {
144
+ static symbol = Symbol('remote-resolver')
145
+ private process: ChildProcess
146
+
147
+ constructor(options: RemoveResolverOptions) {
148
+ super(RemoteHttpResolver.symbol)
149
+ this.process = options.process
150
+ }
151
+
152
+ protected setup() {
153
+ const logger = this.logger.extend('setup')
154
+
155
+ const handleChildMessage: NodeJS.MessageListener = async (message) => {
156
+ logger.info('received message from child!', message)
157
+
158
+ if (typeof message !== 'string' || !message.startsWith('request:')) {
159
+ logger.info('unknown message, ignoring...')
160
+ return
161
+ }
162
+
163
+ const [, serializedRequest] = message.match(/^request:(.+)$/) || []
164
+ if (!serializedRequest) {
165
+ return
166
+ }
167
+
168
+ const requestJson = JSON.parse(
169
+ serializedRequest,
170
+ requestReviver
171
+ ) as RevivedRequest
172
+
173
+ logger.info('parsed intercepted request', requestJson)
174
+
175
+ const request = new Request(requestJson.url, {
176
+ method: requestJson.method,
177
+ headers: new Headers(requestJson.headers),
178
+ credentials: requestJson.credentials,
179
+ body: requestJson.body,
180
+ })
181
+
182
+ const controller = new RequestController(request, {
183
+ passthrough: () => {},
184
+ respondWith: async (response) => {
185
+ if (isResponseError(response)) {
186
+ this.logger.info('received a network error!', { response })
187
+ throw new Error('Not implemented')
188
+ }
189
+
190
+ this.logger.info('received mocked response!', { response })
191
+
192
+ const responseClone = response.clone()
193
+ const responseText = await responseClone.text()
194
+
195
+ // // Send the mocked response to the child process.
196
+ const serializedResponse = JSON.stringify({
197
+ status: response.status,
198
+ statusText: response.statusText,
199
+ headers: Array.from(response.headers.entries()),
200
+ body: responseText,
201
+ } as SerializedResponse)
202
+
203
+ this.process.send(
204
+ `response:${requestJson.id}:${serializedResponse}`,
205
+ (error) => {
206
+ if (error) {
207
+ return
208
+ }
209
+
210
+ // Emit an optimistic "response" event at this point,
211
+ // not to rely on the back-and-forth signaling for the sake of the event.
212
+ this.emitter.emit('response', {
213
+ request,
214
+ requestId: requestJson.id,
215
+ response: responseClone,
216
+ isMockedResponse: true,
217
+ })
218
+ }
219
+ )
220
+
221
+ logger.info(
222
+ 'sent serialized mocked response to the parent:',
223
+ serializedResponse
224
+ )
225
+ },
226
+ errorWith: (reason) => {
227
+ this.logger.info('request has errored!', { error: reason })
228
+ throw new Error('Not implemented')
229
+ },
230
+ })
231
+
232
+ await handleRequest({
233
+ request,
234
+ requestId: requestJson.id,
235
+ controller,
236
+ emitter: this.emitter,
237
+ })
238
+ }
239
+
240
+ this.subscriptions.push(() => {
241
+ this.process.removeListener('message', handleChildMessage)
242
+ logger.info('removed the "message" listener from the child process!')
243
+ })
244
+
245
+ logger.info('adding a "message" listener to the child process')
246
+ this.process.addListener('message', handleChildMessage)
247
+
248
+ this.process.once('error', () => this.dispose())
249
+ this.process.once('exit', () => this.dispose())
250
+ }
251
+ }