@indigoai-us/hq-cloud 6.3.0 → 6.3.2

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 (36) hide show
  1. package/dist/bin/sync-runner.d.ts +22 -2
  2. package/dist/bin/sync-runner.d.ts.map +1 -1
  3. package/dist/bin/sync-runner.js +85 -2
  4. package/dist/bin/sync-runner.js.map +1 -1
  5. package/dist/bin/sync-runner.test.js +201 -0
  6. package/dist/bin/sync-runner.test.js.map +1 -1
  7. package/dist/cli/rescue-core.js +14 -2
  8. package/dist/cli/rescue-core.js.map +1 -1
  9. package/dist/cli/rescue-hq-root-guard.test.d.ts +2 -0
  10. package/dist/cli/rescue-hq-root-guard.test.d.ts.map +1 -0
  11. package/dist/cli/rescue-hq-root-guard.test.js +176 -0
  12. package/dist/cli/rescue-hq-root-guard.test.js.map +1 -0
  13. package/dist/skill-telemetry.d.ts +42 -6
  14. package/dist/skill-telemetry.d.ts.map +1 -1
  15. package/dist/skill-telemetry.js +253 -10
  16. package/dist/skill-telemetry.js.map +1 -1
  17. package/dist/skill-telemetry.test.js +287 -1
  18. package/dist/skill-telemetry.test.js.map +1 -1
  19. package/dist/sync/event-sync.d.ts +181 -0
  20. package/dist/sync/event-sync.d.ts.map +1 -0
  21. package/dist/sync/event-sync.js +316 -0
  22. package/dist/sync/event-sync.js.map +1 -0
  23. package/dist/sync/event-sync.test.d.ts +14 -0
  24. package/dist/sync/event-sync.test.d.ts.map +1 -0
  25. package/dist/sync/event-sync.test.js +440 -0
  26. package/dist/sync/event-sync.test.js.map +1 -0
  27. package/package.json +1 -1
  28. package/src/bin/sync-runner.test.ts +246 -0
  29. package/src/bin/sync-runner.ts +117 -3
  30. package/src/cli/rescue-core.ts +15 -2
  31. package/src/cli/rescue-hq-root-guard.test.ts +193 -0
  32. package/src/skill-telemetry.test.ts +433 -0
  33. package/src/skill-telemetry.ts +260 -10
  34. package/src/sync/event-sync.test.ts +533 -0
  35. package/src/sync/event-sync.ts +481 -0
  36. package/test/e2e/sync/cross-tenant-isolation.test.ts +126 -0
@@ -0,0 +1,533 @@
1
+ /**
2
+ * Phase 3 event-sync tests (US-017/US-018/US-019).
3
+ *
4
+ * - resolveEventSync: exact-email gate semantics + env-override precedence.
5
+ * - subscribeSyncReceive: request shape, response validation, failure surface.
6
+ * - sqsClientFromAwsSdk: command mapping + abort-signal forwarding.
7
+ * - createRefreshingSqsClient: proactive skew refresh, reactive retry-once on
8
+ * expiry-class errors, non-expiry errors propagate, concurrent refresh
9
+ * collapse.
10
+ * - startEventSync: happy-path handle wiring (self-echo filter, publish leg),
11
+ * poll-only degradation on subscribe/tenant failure.
12
+ */
13
+
14
+ import { describe, expect, it, vi } from "vitest";
15
+
16
+ import {
17
+ EVENT_SYNC_ROLLOUT_EMAILS,
18
+ resolveEventSync,
19
+ subscribeSyncReceive,
20
+ sqsClientFromAwsSdk,
21
+ createRefreshingSqsClient,
22
+ startEventSync,
23
+ type SubscribeSyncResponse,
24
+ } from "./event-sync.js";
25
+ import type { SqsClientLike, PushReceiverContext } from "./push-receiver.js";
26
+ import type { PushEvent } from "./push-event.js";
27
+
28
+ const ENROLLED = "hassaan@getindigo.ai";
29
+
30
+ function subscribeResponse(
31
+ overrides: Partial<SubscribeSyncResponse["credentials"]> = {},
32
+ queueUrl = "https://sqs.us-east-1.amazonaws.com/000000000000/sync-push-prs_t-dev1",
33
+ ): SubscribeSyncResponse {
34
+ return {
35
+ queueUrl,
36
+ region: "us-east-1",
37
+ credentials: {
38
+ accessKeyId: "AKIA-1",
39
+ secretAccessKey: "sec-1",
40
+ sessionToken: "tok-1",
41
+ expiration: "2026-06-10T13:00:00.000Z",
42
+ ...overrides,
43
+ },
44
+ };
45
+ }
46
+
47
+ // ─── resolveEventSync (US-019) ──────────────────────────────────────────────
48
+
49
+ describe("resolveEventSync — exact-email rollout gate", () => {
50
+ it("enrolls exactly the allowlisted address (case-insensitive, trimmed)", () => {
51
+ expect(EVENT_SYNC_ROLLOUT_EMAILS.has(ENROLLED)).toBe(true);
52
+ expect(resolveEventSync(ENROLLED, undefined)).toBe(true);
53
+ expect(resolveEventSync("HASSAAN@GetIndigo.AI", undefined)).toBe(true);
54
+ expect(resolveEventSync(` ${ENROLLED} `, undefined)).toBe(true);
55
+ });
56
+
57
+ it("never matches look-alikes — exact full address, not suffix/superset", () => {
58
+ expect(resolveEventSync("xhassaan@getindigo.ai", undefined)).toBe(false);
59
+ expect(resolveEventSync("hassaan@getindigo.ai.evil.com", undefined)).toBe(
60
+ false,
61
+ );
62
+ expect(resolveEventSync("hassaan@evil-getindigo.ai", undefined)).toBe(
63
+ false,
64
+ );
65
+ expect(resolveEventSync("corey@getindigo.ai", undefined)).toBe(false);
66
+ expect(resolveEventSync(undefined, undefined)).toBe(false);
67
+ expect(resolveEventSync("", undefined)).toBe(false);
68
+ });
69
+
70
+ it.each(["1", "true", "yes", "on", " ON "])(
71
+ "override %j forces ON for unenrolled accounts",
72
+ (o) => {
73
+ expect(resolveEventSync("nobody@example.com", o)).toBe(true);
74
+ expect(resolveEventSync(undefined, o)).toBe(true);
75
+ },
76
+ );
77
+
78
+ it.each(["0", "false", "no", "off"])(
79
+ "override %j forces OFF even for the enrolled account",
80
+ (o) => {
81
+ expect(resolveEventSync(ENROLLED, o)).toBe(false);
82
+ },
83
+ );
84
+
85
+ it("unset/blank/garbage override falls through to the email check", () => {
86
+ expect(resolveEventSync(ENROLLED, undefined)).toBe(true);
87
+ expect(resolveEventSync(ENROLLED, "")).toBe(true);
88
+ expect(resolveEventSync(ENROLLED, "maybe")).toBe(true);
89
+ expect(resolveEventSync("nobody@example.com", "maybe")).toBe(false);
90
+ });
91
+ });
92
+
93
+ // ─── subscribeSyncReceive ───────────────────────────────────────────────────
94
+
95
+ describe("subscribeSyncReceive", () => {
96
+ it("POSTs deviceId with a Bearer token and parses the response", async () => {
97
+ const calls: Array<{ url: string; init: Record<string, unknown> }> = [];
98
+ const resp = subscribeResponse();
99
+ const result = await subscribeSyncReceive({
100
+ apiUrl: "https://api.example.com/",
101
+ authToken: async () => "jwt-123",
102
+ deviceId: "dev1",
103
+ fetchImpl: async (url, init) => {
104
+ calls.push({ url, init: init as unknown as Record<string, unknown> });
105
+ return { ok: true, status: 200, text: async () => JSON.stringify(resp) };
106
+ },
107
+ });
108
+
109
+ expect(result).toEqual(resp);
110
+ expect(calls).toHaveLength(1);
111
+ expect(calls[0].url).toBe("https://api.example.com/v1/sync/subscribe");
112
+ const init = calls[0].init as {
113
+ headers: Record<string, string>;
114
+ body: string;
115
+ };
116
+ expect(init.headers.Authorization).toBe("Bearer jwt-123");
117
+ expect(JSON.parse(init.body)).toEqual({ deviceId: "dev1" });
118
+ });
119
+
120
+ it("throws with status + body excerpt on non-2xx", async () => {
121
+ await expect(
122
+ subscribeSyncReceive({
123
+ apiUrl: "https://api.example.com",
124
+ authToken: "jwt",
125
+ deviceId: "dev1",
126
+ fetchImpl: async () => ({
127
+ ok: false,
128
+ status: 503,
129
+ text: async () => "subscribe exploded",
130
+ }),
131
+ }),
132
+ ).rejects.toThrow(/503.*subscribe exploded/s);
133
+ });
134
+
135
+ it("rejects a response missing credentials (old server) loudly", async () => {
136
+ await expect(
137
+ subscribeSyncReceive({
138
+ apiUrl: "https://api.example.com",
139
+ authToken: "jwt",
140
+ deviceId: "dev1",
141
+ fetchImpl: async () => ({
142
+ ok: true,
143
+ status: 200,
144
+ text: async () =>
145
+ JSON.stringify({ queueUrl: "https://q", region: "us-east-1" }),
146
+ }),
147
+ }),
148
+ ).rejects.toThrow(/credentials\.accessKeyId/);
149
+ });
150
+ });
151
+
152
+ // ─── sqsClientFromAwsSdk ────────────────────────────────────────────────────
153
+
154
+ describe("sqsClientFromAwsSdk", () => {
155
+ it("maps receive/delete onto SDK commands and forwards the abort signal", async () => {
156
+ const sent: Array<{ input: Record<string, unknown>; opts: unknown }> = [];
157
+ const fakeSdk = {
158
+ send: async (cmd: { input: Record<string, unknown> }, opts?: unknown) => {
159
+ sent.push({ input: cmd.input, opts });
160
+ return {
161
+ Messages: [{ Body: "b", ReceiptHandle: "rh", MessageId: "m1" }],
162
+ };
163
+ },
164
+ };
165
+ const adapted = sqsClientFromAwsSdk(
166
+ fakeSdk as unknown as Parameters<typeof sqsClientFromAwsSdk>[0],
167
+ );
168
+
169
+ const ac = new AbortController();
170
+ const out = await adapted.receiveMessage({
171
+ queueUrl: "https://q",
172
+ maxMessages: 5,
173
+ waitTimeSeconds: 20,
174
+ signal: ac.signal,
175
+ });
176
+ expect(out.messages).toEqual([
177
+ { Body: "b", ReceiptHandle: "rh", MessageId: "m1" },
178
+ ]);
179
+ expect(sent[0].input).toMatchObject({
180
+ QueueUrl: "https://q",
181
+ MaxNumberOfMessages: 5,
182
+ WaitTimeSeconds: 20,
183
+ });
184
+ expect(sent[0].opts).toEqual({ abortSignal: ac.signal });
185
+
186
+ await adapted.deleteMessage({ queueUrl: "https://q", receiptHandle: "rh" });
187
+ expect(sent[1].input).toMatchObject({
188
+ QueueUrl: "https://q",
189
+ ReceiptHandle: "rh",
190
+ });
191
+ });
192
+
193
+ it("returns an empty list when the SDK omits Messages", async () => {
194
+ const adapted = sqsClientFromAwsSdk({
195
+ send: async () => ({}),
196
+ } as unknown as Parameters<typeof sqsClientFromAwsSdk>[0]);
197
+ const out = await adapted.receiveMessage({
198
+ queueUrl: "https://q",
199
+ maxMessages: 1,
200
+ waitTimeSeconds: 1,
201
+ signal: new AbortController().signal,
202
+ });
203
+ expect(out.messages).toEqual([]);
204
+ });
205
+ });
206
+
207
+ // ─── createRefreshingSqsClient ──────────────────────────────────────────────
208
+
209
+ function expiringError(name: string): Error {
210
+ const e = new Error(name);
211
+ e.name = name;
212
+ return e;
213
+ }
214
+
215
+ function recordingSqs(): SqsClientLike & { receives: number } {
216
+ const client = {
217
+ receives: 0,
218
+ async receiveMessage() {
219
+ client.receives += 1;
220
+ return { messages: [] };
221
+ },
222
+ async deleteMessage() {},
223
+ };
224
+ return client;
225
+ }
226
+
227
+ describe("createRefreshingSqsClient", () => {
228
+ const T0 = Date.parse("2026-06-10T12:00:00.000Z");
229
+
230
+ it("uses the initial client while creds are fresh (no refresh)", async () => {
231
+ const subscribe = vi.fn();
232
+ const built: SqsClientLike[] = [];
233
+ const client = createRefreshingSqsClient({
234
+ initial: subscribeResponse(), // expires 13:00
235
+ subscribe,
236
+ buildSqs: () => {
237
+ const c = recordingSqs();
238
+ built.push(c);
239
+ return c;
240
+ },
241
+ now: () => T0, // 60 min before expiry — outside the 2-min skew window
242
+ });
243
+ await client.receiveMessage({
244
+ queueUrl: "https://q",
245
+ maxMessages: 1,
246
+ waitTimeSeconds: 1,
247
+ signal: new AbortController().signal,
248
+ });
249
+ expect(subscribe).not.toHaveBeenCalled();
250
+ expect(built).toHaveLength(1);
251
+ });
252
+
253
+ it("proactively re-vends inside the expiry skew window", async () => {
254
+ const subscribe = vi
255
+ .fn()
256
+ .mockResolvedValue(
257
+ subscribeResponse({ expiration: "2026-06-10T15:00:00.000Z" }),
258
+ );
259
+ const built: SqsClientLike[] = [];
260
+ const client = createRefreshingSqsClient({
261
+ initial: subscribeResponse(), // expires 13:00
262
+ subscribe,
263
+ buildSqs: () => {
264
+ const c = recordingSqs();
265
+ built.push(c);
266
+ return c;
267
+ },
268
+ now: () => Date.parse("2026-06-10T12:59:30.000Z"), // inside 2-min skew
269
+ });
270
+ await client.receiveMessage({
271
+ queueUrl: "https://q",
272
+ maxMessages: 1,
273
+ waitTimeSeconds: 1,
274
+ signal: new AbortController().signal,
275
+ });
276
+ expect(subscribe).toHaveBeenCalledTimes(1);
277
+ expect(built).toHaveLength(2); // initial + rebuilt
278
+ });
279
+
280
+ it("reactively re-vends ONCE on an expiry-class error, then retries", async () => {
281
+ const subscribe = vi
282
+ .fn()
283
+ .mockResolvedValue(
284
+ subscribeResponse({ expiration: "2026-06-10T15:00:00.000Z" }),
285
+ );
286
+ const failing: SqsClientLike = {
287
+ async receiveMessage() {
288
+ throw expiringError("ExpiredTokenException");
289
+ },
290
+ async deleteMessage() {},
291
+ };
292
+ const fresh = recordingSqs();
293
+ let builds = 0;
294
+ const client = createRefreshingSqsClient({
295
+ initial: subscribeResponse(),
296
+ subscribe,
297
+ // First build (construction) returns the always-failing client; the
298
+ // refresh-triggered rebuild returns the fresh one.
299
+ buildSqs: () => (builds++ === 0 ? failing : fresh),
300
+ now: () => T0,
301
+ });
302
+ const out = await client.receiveMessage({
303
+ queueUrl: "https://q",
304
+ maxMessages: 1,
305
+ waitTimeSeconds: 1,
306
+ signal: new AbortController().signal,
307
+ });
308
+ expect(out.messages).toEqual([]);
309
+ expect(subscribe).toHaveBeenCalledTimes(1);
310
+ expect(fresh.receives).toBe(1);
311
+ });
312
+
313
+ it("non-expiry errors propagate untouched (receiver backoff owns them)", async () => {
314
+ const subscribe = vi.fn();
315
+ const client = createRefreshingSqsClient({
316
+ initial: subscribeResponse(),
317
+ subscribe,
318
+ buildSqs: () => ({
319
+ async receiveMessage() {
320
+ throw expiringError("AccessDeniedException");
321
+ },
322
+ async deleteMessage() {},
323
+ }),
324
+ now: () => T0,
325
+ });
326
+ await expect(
327
+ client.receiveMessage({
328
+ queueUrl: "https://q",
329
+ maxMessages: 1,
330
+ waitTimeSeconds: 1,
331
+ signal: new AbortController().signal,
332
+ }),
333
+ ).rejects.toThrow("AccessDeniedException");
334
+ expect(subscribe).not.toHaveBeenCalled();
335
+ });
336
+
337
+ it("collapses concurrent refreshes onto one subscribe call", async () => {
338
+ // Indirection object (not a nullable local) so TS doesn't narrow the
339
+ // callback assignment from inside the promise executor to `never`.
340
+ const sub: { resolve?: (r: SubscribeSyncResponse) => void } = {};
341
+ const subscribe = vi.fn(
342
+ () =>
343
+ new Promise<SubscribeSyncResponse>((res) => {
344
+ sub.resolve = res;
345
+ }),
346
+ );
347
+ const client = createRefreshingSqsClient({
348
+ initial: subscribeResponse(),
349
+ subscribe,
350
+ buildSqs: () => recordingSqs(),
351
+ now: () => Date.parse("2026-06-10T13:30:00.000Z"), // past expiry
352
+ });
353
+ const sig = new AbortController().signal;
354
+ const p1 = client.receiveMessage({
355
+ queueUrl: "https://q",
356
+ maxMessages: 1,
357
+ waitTimeSeconds: 1,
358
+ signal: sig,
359
+ });
360
+ const p2 = client.receiveMessage({
361
+ queueUrl: "https://q",
362
+ maxMessages: 1,
363
+ waitTimeSeconds: 1,
364
+ signal: sig,
365
+ });
366
+ // Let both calls hit ensureFresh before resolving the vend.
367
+ await new Promise((r) => setTimeout(r, 0));
368
+ sub.resolve?.(subscribeResponse({ expiration: "2026-06-10T15:00:00.000Z" }));
369
+ await Promise.all([p1, p2]);
370
+ expect(subscribe).toHaveBeenCalledTimes(1);
371
+ });
372
+ });
373
+
374
+ // ─── startEventSync ─────────────────────────────────────────────────────────
375
+
376
+ function pushEvent(overrides: Partial<PushEvent> = {}): PushEvent {
377
+ return {
378
+ relativePath: "companies/indigo/docs/a.md",
379
+ contentHash: `sha256:${"a".repeat(64)}`,
380
+ mtime: "2026-06-10T12:00:00.000Z",
381
+ originDeviceId: "peer-device",
382
+ originTenantId: "prs_tenant",
383
+ sequenceNumber: 1,
384
+ eventTimestamp: "2026-06-10T12:00:00.000Z",
385
+ ...overrides,
386
+ };
387
+ }
388
+
389
+ describe("startEventSync", () => {
390
+ function happyOpts(syncCalls: PushEvent[]) {
391
+ return {
392
+ hqRoot: "/tmp/hq",
393
+ apiUrl: "https://api.example.com",
394
+ authToken: async () => "jwt",
395
+ deviceId: "this-device",
396
+ resolveTenantId: async () => "prs_tenant",
397
+ syncFn: async (ctx: PushReceiverContext) => {
398
+ syncCalls.push(ctx.event);
399
+ },
400
+ subscribe: async () => subscribeResponse(),
401
+ buildSqs: () =>
402
+ ({
403
+ // Parks the receiver's poll loop, but honors the abort signal (the
404
+ // SqsClientLike contract) so dispose() doesn't block on the drain
405
+ // deadline.
406
+ receiveMessage: ({ signal }: { signal: AbortSignal }) =>
407
+ new Promise<{ messages: [] }>((resolve) => {
408
+ signal.addEventListener("abort", () => resolve({ messages: [] }), {
409
+ once: true,
410
+ });
411
+ }),
412
+ deleteMessage: async () => {},
413
+ }) as SqsClientLike,
414
+ log: () => {},
415
+ };
416
+ }
417
+
418
+ it("brings up publish + receive and self-echo-filters the receiver", async () => {
419
+ const syncCalls: PushEvent[] = [];
420
+ const opts = happyOpts(syncCalls);
421
+
422
+ // The wrapped syncFn is private to the receiver — exercise the filter
423
+ // through a receiver-visible seam instead: deliver one own-device and one
424
+ // peer message through the (fake) queue.
425
+ const delivered: string[] = [];
426
+ let deliverNext: ((body: string) => void) | null = null;
427
+ const queueSqs: SqsClientLike = {
428
+ receiveMessage: ({ signal }) =>
429
+ new Promise((resolve) => {
430
+ signal.addEventListener("abort", () => resolve({ messages: [] }), {
431
+ once: true,
432
+ });
433
+ deliverNext = (body: string) => {
434
+ deliverNext = null; // consumed — re-armed on the next poll
435
+ delivered.push(body);
436
+ resolve({ messages: [{ Body: body, ReceiptHandle: "rh" }] });
437
+ };
438
+ }),
439
+ deleteMessage: async () => {},
440
+ };
441
+ const handles = await startEventSync({
442
+ ...opts,
443
+ buildSqs: () => queueSqs,
444
+ });
445
+ expect(handles).not.toBeNull();
446
+ expect(handles!.ownDeviceId).toBe("this-device");
447
+
448
+ const waitForPoll = () =>
449
+ vi.waitFor(() => {
450
+ if (!deliverNext) throw new Error("receiver not polling yet");
451
+ });
452
+
453
+ // Own-device event → filtered out (no syncFn call).
454
+ await waitForPoll();
455
+ deliverNext!(JSON.stringify(pushEvent({ originDeviceId: "this-device" })));
456
+ await vi.waitFor(() => expect(delivered).toHaveLength(1));
457
+
458
+ // Peer event → bridged into syncFn.
459
+ await waitForPoll();
460
+ deliverNext!(
461
+ JSON.stringify(
462
+ pushEvent({ originDeviceId: "peer-device", sequenceNumber: 2 }),
463
+ ),
464
+ );
465
+ await vi.waitFor(() => expect(syncCalls).toHaveLength(1));
466
+ expect(syncCalls[0].originDeviceId).toBe("peer-device");
467
+ // The own-device event was delivered but never bridged.
468
+ expect(syncCalls.every((e) => e.originDeviceId === "peer-device")).toBe(
469
+ true,
470
+ );
471
+
472
+ await handles!.dispose();
473
+ });
474
+
475
+ it("returns null (poll-only) when subscribe fails — never throws", async () => {
476
+ const logged: string[] = [];
477
+ const handles = await startEventSync({
478
+ ...happyOpts([]),
479
+ subscribe: async () => {
480
+ throw new Error("subscribe 503");
481
+ },
482
+ log: (m) => logged.push(m),
483
+ });
484
+ expect(handles).toBeNull();
485
+ expect(logged.join("\n")).toContain("poll-only");
486
+ expect(logged.join("\n")).toContain("subscribe 503");
487
+ });
488
+
489
+ it("returns null (poll-only) when tenant resolution fails", async () => {
490
+ const logged: string[] = [];
491
+ const handles = await startEventSync({
492
+ ...happyOpts([]),
493
+ resolveTenantId: async () => {
494
+ throw new Error("no canonical person entity for this account");
495
+ },
496
+ log: (m) => logged.push(m),
497
+ });
498
+ expect(handles).toBeNull();
499
+ expect(logged.join("\n")).toContain("no canonical person entity");
500
+ });
501
+
502
+ it("publishBatch ships one event per changed path AFTER being invoked (publish leg)", async () => {
503
+ const published: PushEvent[] = [];
504
+ const fakeTransport = {
505
+ connected: true,
506
+ start: async () => {},
507
+ dispose: async () => {},
508
+ publish: async (e: PushEvent) => {
509
+ published.push(e);
510
+ },
511
+ };
512
+ const handles = await startEventSync({
513
+ ...happyOpts([]),
514
+ transport: fakeTransport as never,
515
+ now: () => 1765372800000,
516
+ });
517
+ expect(handles).not.toBeNull();
518
+
519
+ // NOTE: paths must exist for content hashing — use this test file itself.
520
+ const abs = new URL(import.meta.url).pathname;
521
+ handles!.publishBatch({ paths: new Map([[abs, "docs/self.ts"]]) });
522
+ await vi.waitFor(() => expect(published).toHaveLength(1));
523
+ const ev = published[0];
524
+ expect(ev.relativePath).toBe("docs/self.ts");
525
+ expect(ev.originDeviceId).toBe("this-device");
526
+ expect(ev.originTenantId).toBe("prs_tenant");
527
+ // Wall-clock sequence numbers (restart-safe monotonicity).
528
+ expect(ev.sequenceNumber).toBe(1765372800000);
529
+ expect(ev.contentHash).toMatch(/^sha256:[0-9a-f]{64}$/);
530
+
531
+ await handles!.dispose();
532
+ });
533
+ });