@ricsam/isolate-test-utils 0.1.9 → 0.1.11

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 (46) hide show
  1. package/README.md +2 -2
  2. package/dist/cjs/fetch-context.cjs +66 -0
  3. package/dist/cjs/fetch-context.cjs.map +10 -0
  4. package/dist/cjs/fs-context.cjs +73 -0
  5. package/dist/cjs/fs-context.cjs.map +10 -0
  6. package/dist/cjs/index.cjs +121 -0
  7. package/dist/cjs/index.cjs.map +10 -0
  8. package/{src/mock-fs.ts → dist/cjs/mock-fs.cjs} +63 -97
  9. package/dist/cjs/mock-fs.cjs.map +10 -0
  10. package/dist/cjs/native-input-test.cjs +347 -0
  11. package/dist/cjs/native-input-test.cjs.map +10 -0
  12. package/dist/cjs/package.json +5 -0
  13. package/dist/cjs/runtime-context.cjs +97 -0
  14. package/dist/cjs/runtime-context.cjs.map +10 -0
  15. package/dist/cjs/server.cjs +109 -0
  16. package/dist/cjs/server.cjs.map +10 -0
  17. package/dist/mjs/fetch-context.mjs +23 -0
  18. package/dist/mjs/fetch-context.mjs.map +10 -0
  19. package/dist/mjs/fs-context.mjs +30 -0
  20. package/dist/mjs/fs-context.mjs.map +10 -0
  21. package/dist/mjs/index.mjs +78 -0
  22. package/dist/mjs/index.mjs.map +10 -0
  23. package/dist/mjs/mock-fs.mjs +181 -0
  24. package/dist/mjs/mock-fs.mjs.map +10 -0
  25. package/{src/native-input-test.ts → dist/mjs/native-input-test.mjs} +60 -169
  26. package/dist/mjs/native-input-test.mjs.map +10 -0
  27. package/dist/mjs/package.json +5 -0
  28. package/dist/mjs/runtime-context.mjs +67 -0
  29. package/dist/mjs/runtime-context.mjs.map +10 -0
  30. package/dist/mjs/server.mjs +79 -0
  31. package/dist/mjs/server.mjs.map +10 -0
  32. package/dist/types/fetch-context.d.ts +7 -0
  33. package/dist/types/fs-context.d.ts +30 -0
  34. package/dist/types/index.d.ts +88 -0
  35. package/dist/types/mock-fs.d.ts +59 -0
  36. package/dist/types/native-input-test.d.ts +29 -0
  37. package/dist/types/runtime-context.d.ts +73 -0
  38. package/dist/types/server.d.ts +53 -0
  39. package/package.json +45 -18
  40. package/CHANGELOG.md +0 -109
  41. package/src/fetch-context.ts +0 -33
  42. package/src/fs-context.ts +0 -65
  43. package/src/index.test.ts +0 -472
  44. package/src/index.ts +0 -177
  45. package/src/runtime-context.ts +0 -148
  46. package/src/server.ts +0 -150
@@ -0,0 +1,347 @@
1
+ var __create = Object.create;
2
+ var __getProtoOf = Object.getPrototypeOf;
3
+ var __defProp = Object.defineProperty;
4
+ var __getOwnPropNames = Object.getOwnPropertyNames;
5
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
6
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
7
+ var __toESM = (mod, isNodeMode, target) => {
8
+ target = mod != null ? __create(__getProtoOf(mod)) : {};
9
+ const to = isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target;
10
+ for (let key of __getOwnPropNames(mod))
11
+ if (!__hasOwnProp.call(to, key))
12
+ __defProp(to, key, {
13
+ get: () => mod[key],
14
+ enumerable: true
15
+ });
16
+ return to;
17
+ };
18
+ var __moduleCache = /* @__PURE__ */ new WeakMap;
19
+ var __toCommonJS = (from) => {
20
+ var entry = __moduleCache.get(from), desc;
21
+ if (entry)
22
+ return entry;
23
+ entry = __defProp({}, "__esModule", { value: true });
24
+ if (from && typeof from === "object" || typeof from === "function")
25
+ __getOwnPropNames(from).map((key) => !__hasOwnProp.call(entry, key) && __defProp(entry, key, {
26
+ get: () => from[key],
27
+ enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable
28
+ }));
29
+ __moduleCache.set(from, entry);
30
+ return entry;
31
+ };
32
+ var __export = (target, all) => {
33
+ for (var name in all)
34
+ __defProp(target, name, {
35
+ get: all[name],
36
+ enumerable: true,
37
+ configurable: true,
38
+ set: (newValue) => all[name] = () => newValue
39
+ });
40
+ };
41
+
42
+ // packages/test-utils/src/native-input-test.ts
43
+ var exports_native_input_test = {};
44
+ __export(exports_native_input_test, {
45
+ runTestCode: () => runTestCode
46
+ });
47
+ module.exports = __toCommonJS(exports_native_input_test);
48
+ var import_isolated_vm = __toESM(require("isolated-vm"));
49
+ function runTestCode(context, code) {
50
+ return {
51
+ input(inputs) {
52
+ const logs = {};
53
+ const logCallback = new import_isolated_vm.default.Callback((tag, valueJson) => {
54
+ const value = JSON.parse(valueJson);
55
+ logs[tag] = unmarshalFromJson(value);
56
+ });
57
+ context.global.setSync("__log_callback__", logCallback);
58
+ context.evalSync(`
59
+ globalThis.log = function(tag, value) {
60
+ __log_callback__(tag, JSON.stringify(__serializeForLog__(value)));
61
+ };
62
+
63
+ globalThis.__serializeForLog__ = function(value) {
64
+ if (typeof Headers !== 'undefined' && value instanceof Headers) {
65
+ const pairs = [];
66
+ for (const [k, v] of value) pairs.push([k, v]);
67
+ return { __type__: 'Headers', pairs };
68
+ }
69
+ if (typeof Request !== 'undefined' && value instanceof Request) {
70
+ const headers = [];
71
+ for (const [k, v] of value.headers) headers.push([k, v]);
72
+ return {
73
+ __type__: 'Request',
74
+ url: value.url,
75
+ method: value.method,
76
+ headers,
77
+ mode: value.mode,
78
+ credentials: value.credentials,
79
+ cache: value.cache,
80
+ redirect: value.redirect,
81
+ referrer: value.referrer,
82
+ referrerPolicy: value.referrerPolicy,
83
+ integrity: value.integrity,
84
+ };
85
+ }
86
+ if (typeof Response !== 'undefined' && value instanceof Response) {
87
+ const headers = [];
88
+ for (const [k, v] of value.headers) headers.push([k, v]);
89
+ return {
90
+ __type__: 'Response',
91
+ status: value.status,
92
+ statusText: value.statusText,
93
+ ok: value.ok,
94
+ headers,
95
+ type: value.type,
96
+ redirected: value.redirected,
97
+ url: value.url,
98
+ };
99
+ }
100
+ if (typeof FormData !== 'undefined' && value instanceof FormData) {
101
+ const entries = [];
102
+ for (const [k, v] of value) {
103
+ if (typeof File !== 'undefined' && v instanceof File) {
104
+ entries.push([k, { __type__: 'File', name: v.name, type: v.type, lastModified: v.lastModified }]);
105
+ } else {
106
+ entries.push([k, v]);
107
+ }
108
+ }
109
+ return { __type__: 'FormData', entries };
110
+ }
111
+ if (value instanceof URL) {
112
+ return { __type__: 'URL', href: value.href };
113
+ }
114
+ if (value instanceof File) {
115
+ return { __type__: 'File', name: value.name, type: value.type, lastModified: value.lastModified };
116
+ }
117
+ if (value instanceof Blob) {
118
+ return { __type__: 'Blob', type: value.type, size: value.size };
119
+ }
120
+ if (Array.isArray(value)) {
121
+ return value.map(v => __serializeForLog__(v));
122
+ }
123
+ if (value && typeof value === 'object' && value.constructor === Object) {
124
+ const result = {};
125
+ for (const [k, v] of Object.entries(value)) {
126
+ result[k] = __serializeForLog__(v);
127
+ }
128
+ return result;
129
+ }
130
+ return value;
131
+ };
132
+ `);
133
+ marshalInputs(context, inputs);
134
+ let returnValue = undefined;
135
+ try {
136
+ returnValue = context.evalSync(code);
137
+ } catch (error) {
138
+ context.evalSync(`
139
+ delete globalThis.testingInput;
140
+ delete globalThis.log;
141
+ delete globalThis.__log_callback__;
142
+ delete globalThis.__serializeForLog__;
143
+ `);
144
+ throw error;
145
+ }
146
+ context.evalSync(`
147
+ delete globalThis.testingInput;
148
+ delete globalThis.log;
149
+ delete globalThis.__log_callback__;
150
+ delete globalThis.__serializeForLog__;
151
+ `);
152
+ return { logs, result: returnValue };
153
+ }
154
+ };
155
+ }
156
+ function marshalInputs(context, inputs) {
157
+ context.evalSync(`globalThis.testingInput = {};`);
158
+ for (const [key, value] of Object.entries(inputs)) {
159
+ marshalValue(context, `testingInput.${key}`, value);
160
+ }
161
+ }
162
+ function marshalValue(context, path, value) {
163
+ if (value instanceof Headers) {
164
+ const pairs = [];
165
+ value.forEach((v, k) => pairs.push([k, v]));
166
+ const pairsJson = JSON.stringify(pairs);
167
+ context.evalSync(`${path} = new Headers(${pairsJson});`);
168
+ return;
169
+ }
170
+ if (value instanceof Request) {
171
+ const headerPairs = [];
172
+ value.headers.forEach((v, k) => headerPairs.push([k, v]));
173
+ const headersJson = JSON.stringify(headerPairs);
174
+ const urlJson = JSON.stringify(value.url);
175
+ const methodJson = JSON.stringify(value.method);
176
+ const modeJson = JSON.stringify(value.mode);
177
+ const credentialsJson = JSON.stringify(value.credentials);
178
+ const cacheJson = JSON.stringify(value.cache);
179
+ const redirectJson = JSON.stringify(value.redirect);
180
+ const referrerJson = JSON.stringify(value.referrer);
181
+ const referrerPolicyJson = JSON.stringify(value.referrerPolicy);
182
+ const integrityJson = JSON.stringify(value.integrity);
183
+ context.evalSync(`
184
+ ${path} = new Request(${urlJson}, {
185
+ method: ${methodJson},
186
+ headers: new Headers(${headersJson}),
187
+ mode: ${modeJson},
188
+ credentials: ${credentialsJson},
189
+ cache: ${cacheJson},
190
+ redirect: ${redirectJson},
191
+ referrer: ${referrerJson},
192
+ referrerPolicy: ${referrerPolicyJson},
193
+ integrity: ${integrityJson},
194
+ });
195
+ `);
196
+ return;
197
+ }
198
+ if (value instanceof Response) {
199
+ const headerPairs = [];
200
+ value.headers.forEach((v, k) => headerPairs.push([k, v]));
201
+ const headersJson = JSON.stringify(headerPairs);
202
+ const statusJson = JSON.stringify(value.status);
203
+ const statusTextJson = JSON.stringify(value.statusText);
204
+ context.evalSync(`
205
+ ${path} = new Response(null, {
206
+ status: ${statusJson},
207
+ statusText: ${statusTextJson},
208
+ headers: new Headers(${headersJson}),
209
+ });
210
+ `);
211
+ return;
212
+ }
213
+ if (value instanceof FormData) {
214
+ context.evalSync(`${path} = new FormData();`);
215
+ for (const [key, entryValue] of value.entries()) {
216
+ const keyJson = JSON.stringify(key);
217
+ if (typeof entryValue !== "string") {
218
+ const file = entryValue;
219
+ const nameJson = JSON.stringify(file.name);
220
+ const typeJson = JSON.stringify(file.type);
221
+ const lastModifiedJson = JSON.stringify(file.lastModified);
222
+ context.evalSync(`
223
+ ${path}.append(${keyJson}, new File([], ${nameJson}, { type: ${typeJson}, lastModified: ${lastModifiedJson} }));
224
+ `);
225
+ } else {
226
+ const valueJson2 = JSON.stringify(entryValue);
227
+ context.evalSync(`${path}.append(${keyJson}, ${valueJson2});`);
228
+ }
229
+ }
230
+ return;
231
+ }
232
+ if (value instanceof URL) {
233
+ const hrefJson = JSON.stringify(value.href);
234
+ context.evalSync(`${path} = new URL(${hrefJson});`);
235
+ return;
236
+ }
237
+ if (value instanceof File) {
238
+ const nameJson = JSON.stringify(value.name);
239
+ const typeJson = JSON.stringify(value.type);
240
+ const lastModifiedJson = JSON.stringify(value.lastModified);
241
+ context.evalSync(`${path} = new File([], ${nameJson}, { type: ${typeJson}, lastModified: ${lastModifiedJson} });`);
242
+ return;
243
+ }
244
+ if (value instanceof Blob) {
245
+ const typeJson = JSON.stringify(value.type);
246
+ context.evalSync(`${path} = new Blob([], { type: ${typeJson} });`);
247
+ return;
248
+ }
249
+ if (Array.isArray(value)) {
250
+ context.evalSync(`${path} = [];`);
251
+ for (let i = 0;i < value.length; i++) {
252
+ marshalValue(context, `${path}[${i}]`, value[i]);
253
+ }
254
+ return;
255
+ }
256
+ if (value && typeof value === "object" && value.constructor === Object) {
257
+ context.evalSync(`${path} = {};`);
258
+ for (const [key, val] of Object.entries(value)) {
259
+ marshalValue(context, `${path}[${JSON.stringify(key)}]`, val);
260
+ }
261
+ return;
262
+ }
263
+ const valueJson = JSON.stringify(value);
264
+ context.evalSync(`${path} = ${valueJson};`);
265
+ }
266
+ function unmarshalFromJson(value) {
267
+ if (value === null || value === undefined) {
268
+ return value;
269
+ }
270
+ if (Array.isArray(value)) {
271
+ return value.map((v) => unmarshalFromJson(v));
272
+ }
273
+ if (typeof value === "object") {
274
+ const obj = value;
275
+ if (obj.__type__ === "Headers") {
276
+ const pairs = obj.pairs;
277
+ const headers = new Headers;
278
+ for (const [k, v] of pairs) {
279
+ headers.append(k, v);
280
+ }
281
+ return headers;
282
+ }
283
+ if (obj.__type__ === "Request") {
284
+ const headers = new Headers;
285
+ for (const [k, v] of obj.headers) {
286
+ headers.append(k, v);
287
+ }
288
+ return new Request(obj.url, {
289
+ method: obj.method,
290
+ headers,
291
+ mode: obj.mode,
292
+ credentials: obj.credentials,
293
+ cache: obj.cache,
294
+ redirect: obj.redirect,
295
+ referrer: obj.referrer,
296
+ referrerPolicy: obj.referrerPolicy,
297
+ integrity: obj.integrity
298
+ });
299
+ }
300
+ if (obj.__type__ === "Response") {
301
+ const headers = new Headers;
302
+ for (const [k, v] of obj.headers) {
303
+ headers.append(k, v);
304
+ }
305
+ return new Response(null, {
306
+ status: obj.status,
307
+ statusText: obj.statusText,
308
+ headers
309
+ });
310
+ }
311
+ if (obj.__type__ === "FormData") {
312
+ const formData = new FormData;
313
+ for (const [k, v] of obj.entries) {
314
+ if (typeof v === "object" && v !== null && v.__type__ === "File") {
315
+ const fileObj = v;
316
+ formData.append(k, new File([], fileObj.name, {
317
+ type: fileObj.type,
318
+ lastModified: fileObj.lastModified
319
+ }));
320
+ } else {
321
+ formData.append(k, v);
322
+ }
323
+ }
324
+ return formData;
325
+ }
326
+ if (obj.__type__ === "URL") {
327
+ return new URL(obj.href);
328
+ }
329
+ if (obj.__type__ === "File") {
330
+ return new File([], obj.name, {
331
+ type: obj.type,
332
+ lastModified: obj.lastModified
333
+ });
334
+ }
335
+ if (obj.__type__ === "Blob") {
336
+ return new Blob([], { type: obj.type });
337
+ }
338
+ const result = {};
339
+ for (const [k, v] of Object.entries(obj)) {
340
+ result[k] = unmarshalFromJson(v);
341
+ }
342
+ return result;
343
+ }
344
+ return value;
345
+ }
346
+
347
+ //# debugId=7D8A3E21B94E7B5B64756E2164756E21
@@ -0,0 +1,10 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../src/native-input-test.ts"],
4
+ "sourcesContent": [
5
+ "import ivm from \"isolated-vm\";\n\nexport interface TestRuntime {\n logs: Record<string, unknown>;\n result: unknown;\n}\n\nexport interface TestRunner {\n input(inputs: Record<string, unknown>): TestRuntime;\n}\n\n/**\n * Run isolate code with native objects as input and capture logs\n *\n * This utility allows testing whether native objects passed INTO the isolate\n * behave like isolate instances. It converts native web API classes (Headers,\n * Request, Response, URL, Blob, File, FormData) to their isolate equivalents\n * before executing the test code.\n *\n * @example\n * const runtime = runTestCode(ctx.context, `\n * const headers = testingInput.headers;\n * log(\"instanceof\", headers instanceof Headers);\n * log(\"contentType\", headers.get(\"content-type\"));\n * `).input({\n * headers: new Headers({ \"content-type\": \"application/json\" })\n * });\n *\n * expect(runtime.logs.instanceof).toBe(true);\n * expect(runtime.logs.contentType).toBe(\"application/json\");\n */\nexport function runTestCode(context: ivm.Context, code: string): TestRunner {\n return {\n input(inputs: Record<string, unknown>): TestRuntime {\n const logs: Record<string, unknown> = {};\n\n // Setup log capture - log(tag, value) stores as logs[tag] = value\n // Values are unmarshalled back to native types for bidirectional testing\n const logCallback = new ivm.Callback(\n (tag: string, valueJson: string) => {\n const value = JSON.parse(valueJson);\n logs[tag] = unmarshalFromJson(value);\n }\n );\n context.global.setSync(\"__log_callback__\", logCallback);\n\n // Create a wrapper log function that serializes values\n context.evalSync(`\n globalThis.log = function(tag, value) {\n __log_callback__(tag, JSON.stringify(__serializeForLog__(value)));\n };\n\n globalThis.__serializeForLog__ = function(value) {\n if (typeof Headers !== 'undefined' && value instanceof Headers) {\n const pairs = [];\n for (const [k, v] of value) pairs.push([k, v]);\n return { __type__: 'Headers', pairs };\n }\n if (typeof Request !== 'undefined' && value instanceof Request) {\n const headers = [];\n for (const [k, v] of value.headers) headers.push([k, v]);\n return {\n __type__: 'Request',\n url: value.url,\n method: value.method,\n headers,\n mode: value.mode,\n credentials: value.credentials,\n cache: value.cache,\n redirect: value.redirect,\n referrer: value.referrer,\n referrerPolicy: value.referrerPolicy,\n integrity: value.integrity,\n };\n }\n if (typeof Response !== 'undefined' && value instanceof Response) {\n const headers = [];\n for (const [k, v] of value.headers) headers.push([k, v]);\n return {\n __type__: 'Response',\n status: value.status,\n statusText: value.statusText,\n ok: value.ok,\n headers,\n type: value.type,\n redirected: value.redirected,\n url: value.url,\n };\n }\n if (typeof FormData !== 'undefined' && value instanceof FormData) {\n const entries = [];\n for (const [k, v] of value) {\n if (typeof File !== 'undefined' && v instanceof File) {\n entries.push([k, { __type__: 'File', name: v.name, type: v.type, lastModified: v.lastModified }]);\n } else {\n entries.push([k, v]);\n }\n }\n return { __type__: 'FormData', entries };\n }\n if (value instanceof URL) {\n return { __type__: 'URL', href: value.href };\n }\n if (value instanceof File) {\n return { __type__: 'File', name: value.name, type: value.type, lastModified: value.lastModified };\n }\n if (value instanceof Blob) {\n return { __type__: 'Blob', type: value.type, size: value.size };\n }\n if (Array.isArray(value)) {\n return value.map(v => __serializeForLog__(v));\n }\n if (value && typeof value === 'object' && value.constructor === Object) {\n const result = {};\n for (const [k, v] of Object.entries(value)) {\n result[k] = __serializeForLog__(v);\n }\n return result;\n }\n return value;\n };\n `);\n\n // Marshal inputs with special handling for native web API classes\n marshalInputs(context, inputs);\n\n // Run the code\n let returnValue: unknown = undefined;\n try {\n returnValue = context.evalSync(code);\n } catch (error) {\n // Clean up before re-throwing\n context.evalSync(`\n delete globalThis.testingInput;\n delete globalThis.log;\n delete globalThis.__log_callback__;\n delete globalThis.__serializeForLog__;\n `);\n throw error;\n }\n\n // Cleanup\n context.evalSync(`\n delete globalThis.testingInput;\n delete globalThis.log;\n delete globalThis.__log_callback__;\n delete globalThis.__serializeForLog__;\n `);\n\n return { logs, result: returnValue };\n },\n };\n}\n\n/**\n * Marshal inputs into the isolate, converting native web API classes\n */\nfunction marshalInputs(\n context: ivm.Context,\n inputs: Record<string, unknown>\n): void {\n // Create the testingInput object in the isolate\n context.evalSync(`globalThis.testingInput = {};`);\n\n for (const [key, value] of Object.entries(inputs)) {\n marshalValue(context, `testingInput.${key}`, value);\n }\n}\n\n/**\n * Marshal a single value into the isolate at the given path\n */\nfunction marshalValue(\n context: ivm.Context,\n path: string,\n value: unknown\n): void {\n // Check for native Headers\n if (value instanceof Headers) {\n const pairs: [string, string][] = [];\n value.forEach((v, k) => pairs.push([k, v]));\n const pairsJson = JSON.stringify(pairs);\n context.evalSync(`${path} = new Headers(${pairsJson});`);\n return;\n }\n\n // Check for native Request\n if (value instanceof Request) {\n // First marshal the headers\n const headerPairs: [string, string][] = [];\n value.headers.forEach((v, k) => headerPairs.push([k, v]));\n const headersJson = JSON.stringify(headerPairs);\n\n const urlJson = JSON.stringify(value.url);\n const methodJson = JSON.stringify(value.method);\n const modeJson = JSON.stringify(value.mode);\n const credentialsJson = JSON.stringify(value.credentials);\n const cacheJson = JSON.stringify(value.cache);\n const redirectJson = JSON.stringify(value.redirect);\n const referrerJson = JSON.stringify(value.referrer);\n const referrerPolicyJson = JSON.stringify(value.referrerPolicy);\n const integrityJson = JSON.stringify(value.integrity);\n\n context.evalSync(`\n ${path} = new Request(${urlJson}, {\n method: ${methodJson},\n headers: new Headers(${headersJson}),\n mode: ${modeJson},\n credentials: ${credentialsJson},\n cache: ${cacheJson},\n redirect: ${redirectJson},\n referrer: ${referrerJson},\n referrerPolicy: ${referrerPolicyJson},\n integrity: ${integrityJson},\n });\n `);\n return;\n }\n\n // Check for native Response\n if (value instanceof Response) {\n const headerPairs: [string, string][] = [];\n value.headers.forEach((v, k) => headerPairs.push([k, v]));\n const headersJson = JSON.stringify(headerPairs);\n\n const statusJson = JSON.stringify(value.status);\n const statusTextJson = JSON.stringify(value.statusText);\n\n context.evalSync(`\n ${path} = new Response(null, {\n status: ${statusJson},\n statusText: ${statusTextJson},\n headers: new Headers(${headersJson}),\n });\n `);\n return;\n }\n\n // Check for native FormData\n if (value instanceof FormData) {\n context.evalSync(`${path} = new FormData();`);\n\n for (const [key, entryValue] of value.entries()) {\n const keyJson = JSON.stringify(key);\n\n if (typeof entryValue !== \"string\") {\n const file = entryValue as File;\n const nameJson = JSON.stringify(file.name);\n const typeJson = JSON.stringify(file.type);\n const lastModifiedJson = JSON.stringify(file.lastModified);\n context.evalSync(`\n ${path}.append(${keyJson}, new File([], ${nameJson}, { type: ${typeJson}, lastModified: ${lastModifiedJson} }));\n `);\n } else {\n const valueJson = JSON.stringify(entryValue);\n context.evalSync(`${path}.append(${keyJson}, ${valueJson});`);\n }\n }\n return;\n }\n\n // Check for native URL\n if (value instanceof URL) {\n const hrefJson = JSON.stringify(value.href);\n context.evalSync(`${path} = new URL(${hrefJson});`);\n return;\n }\n\n // Check for native File (before Blob, since File extends Blob)\n if (value instanceof File) {\n const nameJson = JSON.stringify(value.name);\n const typeJson = JSON.stringify(value.type);\n const lastModifiedJson = JSON.stringify(value.lastModified);\n context.evalSync(\n `${path} = new File([], ${nameJson}, { type: ${typeJson}, lastModified: ${lastModifiedJson} });`\n );\n return;\n }\n\n // Check for native Blob\n if (value instanceof Blob) {\n const typeJson = JSON.stringify(value.type);\n context.evalSync(`${path} = new Blob([], { type: ${typeJson} });`);\n return;\n }\n\n // Handle arrays recursively\n if (Array.isArray(value)) {\n context.evalSync(`${path} = [];`);\n for (let i = 0; i < value.length; i++) {\n marshalValue(context, `${path}[${i}]`, value[i]);\n }\n return;\n }\n\n // Handle plain objects recursively\n if (value && typeof value === \"object\" && value.constructor === Object) {\n context.evalSync(`${path} = {};`);\n for (const [key, val] of Object.entries(value)) {\n // Use bracket notation for safe property access\n marshalValue(context, `${path}[${JSON.stringify(key)}]`, val);\n }\n return;\n }\n\n // For primitives, set directly via JSON\n const valueJson = JSON.stringify(value);\n context.evalSync(`${path} = ${valueJson};`);\n}\n\n/**\n * Unmarshal a value from JSON, converting special __type__ markers back to native instances\n */\nfunction unmarshalFromJson(value: unknown): unknown {\n if (value === null || value === undefined) {\n return value;\n }\n\n if (Array.isArray(value)) {\n return value.map((v) => unmarshalFromJson(v));\n }\n\n if (typeof value === \"object\") {\n const obj = value as Record<string, unknown>;\n\n // Check for special type markers\n if (obj.__type__ === \"Headers\") {\n const pairs = obj.pairs as [string, string][];\n const headers = new Headers();\n for (const [k, v] of pairs) {\n headers.append(k, v);\n }\n return headers;\n }\n\n if (obj.__type__ === \"Request\") {\n const headers = new Headers();\n for (const [k, v] of obj.headers as [string, string][]) {\n headers.append(k, v);\n }\n return new Request(obj.url as string, {\n method: obj.method as string,\n headers,\n mode: obj.mode as Request[\"mode\"],\n credentials: obj.credentials as Request[\"credentials\"],\n cache: obj.cache as Request[\"cache\"],\n redirect: obj.redirect as Request[\"redirect\"],\n referrer: obj.referrer as string,\n referrerPolicy: obj.referrerPolicy as Request[\"referrerPolicy\"],\n integrity: obj.integrity as string,\n });\n }\n\n if (obj.__type__ === \"Response\") {\n const headers = new Headers();\n for (const [k, v] of obj.headers as [string, string][]) {\n headers.append(k, v);\n }\n return new Response(null, {\n status: obj.status as number,\n statusText: obj.statusText as string,\n headers,\n });\n }\n\n if (obj.__type__ === \"FormData\") {\n const formData = new FormData();\n for (const [k, v] of obj.entries as [string, unknown][]) {\n if (\n typeof v === \"object\" &&\n v !== null &&\n (v as Record<string, unknown>).__type__ === \"File\"\n ) {\n const fileObj = v as Record<string, unknown>;\n formData.append(\n k,\n new File([], fileObj.name as string, {\n type: fileObj.type as string,\n lastModified: fileObj.lastModified as number,\n })\n );\n } else {\n formData.append(k, v as string);\n }\n }\n return formData;\n }\n\n if (obj.__type__ === \"URL\") {\n return new URL(obj.href as string);\n }\n\n if (obj.__type__ === \"File\") {\n return new File([], obj.name as string, {\n type: obj.type as string,\n lastModified: obj.lastModified as number,\n });\n }\n\n if (obj.__type__ === \"Blob\") {\n return new Blob([], { type: obj.type as string });\n }\n\n // Plain object - recursively unmarshal properties\n const result: Record<string, unknown> = {};\n for (const [k, v] of Object.entries(obj)) {\n result[k] = unmarshalFromJson(v);\n }\n return result;\n }\n\n return value;\n}\n"
6
+ ],
7
+ "mappings": "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",
8
+ "debugId": "7D8A3E21B94E7B5B64756E2164756E21",
9
+ "names": []
10
+ }
@@ -0,0 +1,5 @@
1
+ {
2
+ "name": "@ricsam/isolate-test-utils",
3
+ "version": "0.1.11",
4
+ "type": "commonjs"
5
+ }
@@ -0,0 +1,97 @@
1
+ var __defProp = Object.defineProperty;
2
+ var __getOwnPropNames = Object.getOwnPropertyNames;
3
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
5
+ var __moduleCache = /* @__PURE__ */ new WeakMap;
6
+ var __toCommonJS = (from) => {
7
+ var entry = __moduleCache.get(from), desc;
8
+ if (entry)
9
+ return entry;
10
+ entry = __defProp({}, "__esModule", { value: true });
11
+ if (from && typeof from === "object" || typeof from === "function")
12
+ __getOwnPropNames(from).map((key) => !__hasOwnProp.call(entry, key) && __defProp(entry, key, {
13
+ get: () => from[key],
14
+ enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable
15
+ }));
16
+ __moduleCache.set(from, entry);
17
+ return entry;
18
+ };
19
+ var __export = (target, all) => {
20
+ for (var name in all)
21
+ __defProp(target, name, {
22
+ get: all[name],
23
+ enumerable: true,
24
+ configurable: true,
25
+ set: (newValue) => all[name] = () => newValue
26
+ });
27
+ };
28
+
29
+ // packages/test-utils/src/runtime-context.ts
30
+ var exports_runtime_context = {};
31
+ __export(exports_runtime_context, {
32
+ createRuntimeTestContext: () => createRuntimeTestContext
33
+ });
34
+ module.exports = __toCommonJS(exports_runtime_context);
35
+ var import_mock_fs = require("./mock-fs.cjs");
36
+ var import_isolate_runtime = require("@ricsam/isolate-runtime");
37
+ var import_isolate_core = require("@ricsam/isolate-core");
38
+ async function createRuntimeTestContext(options) {
39
+ const opts = options ?? {};
40
+ import_isolate_core.clearAllInstanceState();
41
+ const logs = [];
42
+ const fetchCalls = [];
43
+ let mockResponse = { status: 200, body: "" };
44
+ let storedResult = undefined;
45
+ const mockFs = new import_mock_fs.MockFileSystem;
46
+ const runtime = await import_isolate_runtime.createRuntime({
47
+ console: {
48
+ onEntry: (entry) => {
49
+ if (entry.type === "output") {
50
+ logs.push({ level: entry.level, stdout: entry.stdout });
51
+ } else if (entry.type === "assert") {
52
+ logs.push({ level: "error", stdout: entry.stdout });
53
+ }
54
+ }
55
+ },
56
+ fetch: async (request) => {
57
+ fetchCalls.push({
58
+ url: request.url,
59
+ method: request.method,
60
+ headers: [...request.headers.entries()]
61
+ });
62
+ return new Response(mockResponse.body ?? "", {
63
+ status: mockResponse.status ?? 200,
64
+ headers: mockResponse.headers
65
+ });
66
+ },
67
+ fs: opts.fs ? { getDirectory: async () => mockFs } : undefined,
68
+ customFunctions: {
69
+ setResult: {
70
+ fn: (value) => {
71
+ storedResult = value;
72
+ },
73
+ type: "sync"
74
+ }
75
+ }
76
+ });
77
+ return {
78
+ eval: runtime.eval.bind(runtime),
79
+ clearTimers: runtime.timers.clearAll.bind(runtime.timers),
80
+ dispatchRequest: runtime.fetch.dispatchRequest.bind(runtime.fetch),
81
+ dispose: runtime.dispose.bind(runtime),
82
+ logs,
83
+ fetchCalls,
84
+ setMockResponse(response) {
85
+ mockResponse = response;
86
+ },
87
+ mockFs,
88
+ getResult() {
89
+ return storedResult;
90
+ },
91
+ clearResult() {
92
+ storedResult = undefined;
93
+ }
94
+ };
95
+ }
96
+
97
+ //# debugId=C62EDF3B37F359C764756E2164756E21
@@ -0,0 +1,10 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../src/runtime-context.ts"],
4
+ "sourcesContent": [
5
+ "import { MockFileSystem } from \"./mock-fs.cjs\";\nimport { createRuntime } from \"@ricsam/isolate-runtime\";\nimport { clearAllInstanceState } from \"@ricsam/isolate-core\";\n\nexport interface MockResponse {\n status?: number;\n body?: string;\n headers?: Record<string, string>;\n}\n\nexport interface RuntimeTestContextOptions {\n /** Enable file system APIs with mock file system */\n fs?: boolean;\n}\n\nexport interface RuntimeTestContext {\n /** Execute code in the runtime (ES module mode, supports top-level await) */\n eval(code: string): Promise<void>;\n /** Clear all pending timers */\n clearTimers(): void;\n /** Dispatch an HTTP request to the serve() handler */\n dispatchRequest(request: Request): Promise<Response>;\n /** Dispose all resources */\n dispose(): Promise<void>;\n /** Captured console.log calls */\n logs: Array<{ level: string; stdout: string }>;\n /** Captured fetch calls */\n fetchCalls: Array<{ url: string; method: string; headers: [string, string][] }>;\n /** Set the mock response for the next fetch call */\n setMockResponse(response: MockResponse): void;\n /** Mock file system (only available if fs option is true) */\n mockFs: MockFileSystem;\n /**\n * Get a result from the isolate. Call `await setResult(value)` in your eval code\n * to pass a value back to the host.\n */\n getResult<T = unknown>(): T | undefined;\n /** Clear the stored result */\n clearResult(): void;\n}\n\n/**\n * Create a full runtime test context with all APIs set up.\n * Includes console logging capture, fetch mocking, and optionally file system.\n *\n * @example\n * const ctx = await createRuntimeTestContext({ fs: true });\n *\n * // Set up mock response for fetch\n * ctx.setMockResponse({ status: 200, body: '{\"data\": \"test\"}' });\n *\n * // Run code and pass result back via setResult\n * await ctx.eval(`\n * console.log(\"Starting fetch...\");\n * const response = await fetch(\"https://api.example.com/data\");\n * const data = await response.json();\n * console.log(\"Got data:\", data);\n * setResult(data);\n * `);\n *\n * // Get the result\n * console.log(ctx.getResult()); // { data: \"test\" }\n *\n * // Check logs\n * console.log(ctx.logs); // [{ level: \"log\", stdout: \"Starting fetch...\" }, ...]\n *\n * // Check fetch calls\n * console.log(ctx.fetchCalls); // [{ url: \"https://api.example.com/data\", method: \"GET\", ... }]\n *\n * await ctx.dispose();\n */\nexport async function createRuntimeTestContext(\n options?: RuntimeTestContextOptions\n): Promise<RuntimeTestContext> {\n const opts = options ?? {};\n\n // Clear any previous instance state\n clearAllInstanceState();\n\n // State for capturing logs and fetch calls\n const logs: Array<{ level: string; stdout: string }> = [];\n const fetchCalls: Array<{\n url: string;\n method: string;\n headers: [string, string][];\n }> = [];\n\n let mockResponse: MockResponse = { status: 200, body: \"\" };\n let storedResult: unknown = undefined;\n\n // Create mock file system\n const mockFs = new MockFileSystem();\n\n // Create runtime with configured handlers\n const runtime = await createRuntime({\n console: {\n onEntry: (entry) => {\n if (entry.type === \"output\") {\n logs.push({ level: entry.level, stdout: entry.stdout });\n } else if (entry.type === \"assert\") {\n logs.push({ level: \"error\", stdout: entry.stdout });\n }\n },\n },\n fetch: async (request: Request) => {\n // Capture fetch call\n fetchCalls.push({\n url: request.url,\n method: request.method,\n headers: [...request.headers.entries()],\n });\n\n // Return mock response\n return new Response(mockResponse.body ?? \"\", {\n status: mockResponse.status ?? 200,\n headers: mockResponse.headers,\n });\n },\n fs: opts.fs ? { getDirectory: async () => mockFs } : undefined,\n customFunctions: {\n setResult: {\n fn: (value: unknown) => {\n storedResult = value;\n },\n type: 'sync',\n },\n },\n });\n\n return {\n eval: runtime.eval.bind(runtime),\n clearTimers: runtime.timers.clearAll.bind(runtime.timers),\n dispatchRequest: runtime.fetch.dispatchRequest.bind(runtime.fetch),\n dispose: runtime.dispose.bind(runtime),\n logs,\n fetchCalls,\n setMockResponse(response: MockResponse) {\n mockResponse = response;\n },\n mockFs,\n getResult<T = unknown>(): T | undefined {\n return storedResult as T | undefined;\n },\n clearResult() {\n storedResult = undefined;\n },\n };\n}\n"
6
+ ],
7
+ "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAA+B,IAA/B;AAC8B,IAA9B;AACsC,IAAtC;AAqEA,eAAsB,wBAAwB,CAC5C,SAC6B;AAAA,EAC7B,MAAM,OAAO,WAAW,CAAC;AAAA,EAGzB,0CAAsB;AAAA,EAGtB,MAAM,OAAiD,CAAC;AAAA,EACxD,MAAM,aAID,CAAC;AAAA,EAEN,IAAI,eAA6B,EAAE,QAAQ,KAAK,MAAM,GAAG;AAAA,EACzD,IAAI,eAAwB;AAAA,EAG5B,MAAM,SAAS,IAAI;AAAA,EAGnB,MAAM,UAAU,MAAM,qCAAc;AAAA,IAClC,SAAS;AAAA,MACP,SAAS,CAAC,UAAU;AAAA,QAClB,IAAI,MAAM,SAAS,UAAU;AAAA,UAC3B,KAAK,KAAK,EAAE,OAAO,MAAM,OAAO,QAAQ,MAAM,OAAO,CAAC;AAAA,QACxD,EAAO,SAAI,MAAM,SAAS,UAAU;AAAA,UAClC,KAAK,KAAK,EAAE,OAAO,SAAS,QAAQ,MAAM,OAAO,CAAC;AAAA,QACpD;AAAA;AAAA,IAEJ;AAAA,IACA,OAAO,OAAO,YAAqB;AAAA,MAEjC,WAAW,KAAK;AAAA,QACd,KAAK,QAAQ;AAAA,QACb,QAAQ,QAAQ;AAAA,QAChB,SAAS,CAAC,GAAG,QAAQ,QAAQ,QAAQ,CAAC;AAAA,MACxC,CAAC;AAAA,MAGD,OAAO,IAAI,SAAS,aAAa,QAAQ,IAAI;AAAA,QAC3C,QAAQ,aAAa,UAAU;AAAA,QAC/B,SAAS,aAAa;AAAA,MACxB,CAAC;AAAA;AAAA,IAEH,IAAI,KAAK,KAAK,EAAE,cAAc,YAAY,OAAO,IAAI;AAAA,IACrD,iBAAiB;AAAA,MACf,WAAW;AAAA,QACT,IAAI,CAAC,UAAmB;AAAA,UACtB,eAAe;AAAA;AAAA,QAEjB,MAAM;AAAA,MACR;AAAA,IACF;AAAA,EACF,CAAC;AAAA,EAED,OAAO;AAAA,IACL,MAAM,QAAQ,KAAK,KAAK,OAAO;AAAA,IAC/B,aAAa,QAAQ,OAAO,SAAS,KAAK,QAAQ,MAAM;AAAA,IACxD,iBAAiB,QAAQ,MAAM,gBAAgB,KAAK,QAAQ,KAAK;AAAA,IACjE,SAAS,QAAQ,QAAQ,KAAK,OAAO;AAAA,IACrC;AAAA,IACA;AAAA,IACA,eAAe,CAAC,UAAwB;AAAA,MACtC,eAAe;AAAA;AAAA,IAEjB;AAAA,IACA,SAAsB,GAAkB;AAAA,MACtC,OAAO;AAAA;AAAA,IAET,WAAW,GAAG;AAAA,MACZ,eAAe;AAAA;AAAA,EAEnB;AAAA;",
8
+ "debugId": "C62EDF3B37F359C764756E2164756E21",
9
+ "names": []
10
+ }
@@ -0,0 +1,109 @@
1
+ var __defProp = Object.defineProperty;
2
+ var __getOwnPropNames = Object.getOwnPropertyNames;
3
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
5
+ var __moduleCache = /* @__PURE__ */ new WeakMap;
6
+ var __toCommonJS = (from) => {
7
+ var entry = __moduleCache.get(from), desc;
8
+ if (entry)
9
+ return entry;
10
+ entry = __defProp({}, "__esModule", { value: true });
11
+ if (from && typeof from === "object" || typeof from === "function")
12
+ __getOwnPropNames(from).map((key) => !__hasOwnProp.call(entry, key) && __defProp(entry, key, {
13
+ get: () => from[key],
14
+ enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable
15
+ }));
16
+ __moduleCache.set(from, entry);
17
+ return entry;
18
+ };
19
+ var __export = (target, all) => {
20
+ for (var name in all)
21
+ __defProp(target, name, {
22
+ get: all[name],
23
+ enumerable: true,
24
+ configurable: true,
25
+ set: (newValue) => all[name] = () => newValue
26
+ });
27
+ };
28
+
29
+ // packages/test-utils/src/server.ts
30
+ var exports_server = {};
31
+ __export(exports_server, {
32
+ startIntegrationServer: () => startIntegrationServer
33
+ });
34
+ module.exports = __toCommonJS(exports_server);
35
+ var import_node_http = require("node:http");
36
+ async function startIntegrationServer(port) {
37
+ const responses = new Map;
38
+ const requests = [];
39
+ let defaultResponse = { status: 404, body: "Not Found" };
40
+ const server = import_node_http.createServer(async (req, res) => {
41
+ const path = req.url ?? "/";
42
+ const method = req.method ?? "GET";
43
+ const chunks = [];
44
+ for await (const chunk of req) {
45
+ chunks.push(chunk);
46
+ }
47
+ const body = chunks.length > 0 ? Buffer.concat(chunks).toString() : undefined;
48
+ const headers = {};
49
+ for (const [key, value] of Object.entries(req.headers)) {
50
+ if (typeof value === "string") {
51
+ headers[key] = value;
52
+ } else if (Array.isArray(value)) {
53
+ headers[key] = value.join(", ");
54
+ }
55
+ }
56
+ requests.push({ method, path, headers, body });
57
+ const mockResponse = responses.get(path) ?? defaultResponse;
58
+ res.statusCode = mockResponse.status ?? 200;
59
+ if (mockResponse.headers) {
60
+ for (const [key, value] of Object.entries(mockResponse.headers)) {
61
+ res.setHeader(key, value);
62
+ }
63
+ }
64
+ res.end(mockResponse.body ?? "");
65
+ });
66
+ const actualPort = await new Promise((resolve, reject) => {
67
+ server.listen(port ?? 0, () => {
68
+ const address = server.address();
69
+ if (address && typeof address === "object") {
70
+ resolve(address.port);
71
+ } else {
72
+ reject(new Error("Failed to get server address"));
73
+ }
74
+ });
75
+ server.on("error", reject);
76
+ });
77
+ return {
78
+ url: `http://localhost:${actualPort}`,
79
+ port: actualPort,
80
+ async close() {
81
+ return new Promise((resolve, reject) => {
82
+ server.close((err) => {
83
+ if (err)
84
+ reject(err);
85
+ else
86
+ resolve();
87
+ });
88
+ });
89
+ },
90
+ setResponse(path, response) {
91
+ responses.set(path, response);
92
+ },
93
+ setDefaultResponse(response) {
94
+ defaultResponse = response;
95
+ },
96
+ getRequests() {
97
+ return [...requests];
98
+ },
99
+ clearRequests() {
100
+ requests.length = 0;
101
+ },
102
+ clearResponses() {
103
+ responses.clear();
104
+ defaultResponse = { status: 404, body: "Not Found" };
105
+ }
106
+ };
107
+ }
108
+
109
+ //# debugId=A81C39DF693539C664756E2164756E21
@@ -0,0 +1,10 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../src/server.ts"],
4
+ "sourcesContent": [
5
+ "import { createServer, type Server, type IncomingMessage, type ServerResponse } from \"node:http\";\n\nexport interface MockServerResponse {\n status?: number;\n body?: string;\n headers?: Record<string, string>;\n}\n\nexport interface RecordedRequest {\n method: string;\n path: string;\n headers: Record<string, string>;\n body?: string;\n}\n\nexport interface IntegrationServer {\n /** The base URL of the server (e.g., \"http://localhost:3000\") */\n url: string;\n /** The port the server is listening on */\n port: number;\n /** Close the server */\n close(): Promise<void>;\n /** Set the response for a specific path */\n setResponse(path: string, response: MockServerResponse): void;\n /** Set a default response for any unmatched path */\n setDefaultResponse(response: MockServerResponse): void;\n /** Get all recorded requests */\n getRequests(): RecordedRequest[];\n /** Clear all recorded requests */\n clearRequests(): void;\n /** Clear all configured responses */\n clearResponses(): void;\n}\n\n/**\n * Start an HTTP server for integration tests.\n * Useful for testing fetch operations against a real server.\n *\n * @example\n * const server = await startIntegrationServer();\n *\n * server.setResponse(\"/api/data\", {\n * status: 200,\n * body: JSON.stringify({ message: \"Hello\" }),\n * headers: { \"Content-Type\": \"application/json\" }\n * });\n *\n * // In your test\n * const response = await fetch(`${server.url}/api/data`);\n * const data = await response.json();\n *\n * // Check what requests were made\n * const requests = server.getRequests();\n * console.log(requests[0].path); // \"/api/data\"\n *\n * await server.close();\n */\nexport async function startIntegrationServer(\n port?: number\n): Promise<IntegrationServer> {\n const responses = new Map<string, MockServerResponse>();\n const requests: RecordedRequest[] = [];\n let defaultResponse: MockServerResponse = { status: 404, body: \"Not Found\" };\n\n const server: Server = createServer(\n async (req: IncomingMessage, res: ServerResponse) => {\n const path = req.url ?? \"/\";\n const method = req.method ?? \"GET\";\n\n // Read request body\n const chunks: Buffer[] = [];\n for await (const chunk of req) {\n chunks.push(chunk as Buffer);\n }\n const body = chunks.length > 0 ? Buffer.concat(chunks).toString() : undefined;\n\n // Record the request\n const headers: Record<string, string> = {};\n for (const [key, value] of Object.entries(req.headers)) {\n if (typeof value === \"string\") {\n headers[key] = value;\n } else if (Array.isArray(value)) {\n headers[key] = value.join(\", \");\n }\n }\n requests.push({ method, path, headers, body });\n\n // Find and send response\n const mockResponse = responses.get(path) ?? defaultResponse;\n\n res.statusCode = mockResponse.status ?? 200;\n\n if (mockResponse.headers) {\n for (const [key, value] of Object.entries(mockResponse.headers)) {\n res.setHeader(key, value);\n }\n }\n\n res.end(mockResponse.body ?? \"\");\n }\n );\n\n // Find an available port\n const actualPort = await new Promise<number>((resolve, reject) => {\n server.listen(port ?? 0, () => {\n const address = server.address();\n if (address && typeof address === \"object\") {\n resolve(address.port);\n } else {\n reject(new Error(\"Failed to get server address\"));\n }\n });\n server.on(\"error\", reject);\n });\n\n return {\n url: `http://localhost:${actualPort}`,\n port: actualPort,\n\n async close() {\n return new Promise((resolve, reject) => {\n server.close((err) => {\n if (err) reject(err);\n else resolve();\n });\n });\n },\n\n setResponse(path: string, response: MockServerResponse) {\n responses.set(path, response);\n },\n\n setDefaultResponse(response: MockServerResponse) {\n defaultResponse = response;\n },\n\n getRequests() {\n return [...requests];\n },\n\n clearRequests() {\n requests.length = 0;\n },\n\n clearResponses() {\n responses.clear();\n defaultResponse = { status: 404, body: \"Not Found\" };\n },\n };\n}\n"
6
+ ],
7
+ "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAqF,IAArF;AAyDA,eAAsB,sBAAsB,CAC1C,MAC4B;AAAA,EAC5B,MAAM,YAAY,IAAI;AAAA,EACtB,MAAM,WAA8B,CAAC;AAAA,EACrC,IAAI,kBAAsC,EAAE,QAAQ,KAAK,MAAM,YAAY;AAAA,EAE3E,MAAM,SAAiB,8BACrB,OAAO,KAAsB,QAAwB;AAAA,IACnD,MAAM,OAAO,IAAI,OAAO;AAAA,IACxB,MAAM,SAAS,IAAI,UAAU;AAAA,IAG7B,MAAM,SAAmB,CAAC;AAAA,IAC1B,iBAAiB,SAAS,KAAK;AAAA,MAC7B,OAAO,KAAK,KAAe;AAAA,IAC7B;AAAA,IACA,MAAM,OAAO,OAAO,SAAS,IAAI,OAAO,OAAO,MAAM,EAAE,SAAS,IAAI;AAAA,IAGpE,MAAM,UAAkC,CAAC;AAAA,IACzC,YAAY,KAAK,UAAU,OAAO,QAAQ,IAAI,OAAO,GAAG;AAAA,MACtD,IAAI,OAAO,UAAU,UAAU;AAAA,QAC7B,QAAQ,OAAO;AAAA,MACjB,EAAO,SAAI,MAAM,QAAQ,KAAK,GAAG;AAAA,QAC/B,QAAQ,OAAO,MAAM,KAAK,IAAI;AAAA,MAChC;AAAA,IACF;AAAA,IACA,SAAS,KAAK,EAAE,QAAQ,MAAM,SAAS,KAAK,CAAC;AAAA,IAG7C,MAAM,eAAe,UAAU,IAAI,IAAI,KAAK;AAAA,IAE5C,IAAI,aAAa,aAAa,UAAU;AAAA,IAExC,IAAI,aAAa,SAAS;AAAA,MACxB,YAAY,KAAK,UAAU,OAAO,QAAQ,aAAa,OAAO,GAAG;AAAA,QAC/D,IAAI,UAAU,KAAK,KAAK;AAAA,MAC1B;AAAA,IACF;AAAA,IAEA,IAAI,IAAI,aAAa,QAAQ,EAAE;AAAA,GAEnC;AAAA,EAGA,MAAM,aAAa,MAAM,IAAI,QAAgB,CAAC,SAAS,WAAW;AAAA,IAChE,OAAO,OAAO,QAAQ,GAAG,MAAM;AAAA,MAC7B,MAAM,UAAU,OAAO,QAAQ;AAAA,MAC/B,IAAI,WAAW,OAAO,YAAY,UAAU;AAAA,QAC1C,QAAQ,QAAQ,IAAI;AAAA,MACtB,EAAO;AAAA,QACL,OAAO,IAAI,MAAM,8BAA8B,CAAC;AAAA;AAAA,KAEnD;AAAA,IACD,OAAO,GAAG,SAAS,MAAM;AAAA,GAC1B;AAAA,EAED,OAAO;AAAA,IACL,KAAK,oBAAoB;AAAA,IACzB,MAAM;AAAA,SAEA,MAAK,GAAG;AAAA,MACZ,OAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AAAA,QACtC,OAAO,MAAM,CAAC,QAAQ;AAAA,UACpB,IAAI;AAAA,YAAK,OAAO,GAAG;AAAA,UACd;AAAA,oBAAQ;AAAA,SACd;AAAA,OACF;AAAA;AAAA,IAGH,WAAW,CAAC,MAAc,UAA8B;AAAA,MACtD,UAAU,IAAI,MAAM,QAAQ;AAAA;AAAA,IAG9B,kBAAkB,CAAC,UAA8B;AAAA,MAC/C,kBAAkB;AAAA;AAAA,IAGpB,WAAW,GAAG;AAAA,MACZ,OAAO,CAAC,GAAG,QAAQ;AAAA;AAAA,IAGrB,aAAa,GAAG;AAAA,MACd,SAAS,SAAS;AAAA;AAAA,IAGpB,cAAc,GAAG;AAAA,MACf,UAAU,MAAM;AAAA,MAChB,kBAAkB,EAAE,QAAQ,KAAK,MAAM,YAAY;AAAA;AAAA,EAEvD;AAAA;",
8
+ "debugId": "A81C39DF693539C664756E2164756E21",
9
+ "names": []
10
+ }