@rozek/nanoclaw 0.0.5 → 0.0.6

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 (132) hide show
  1. package/container/agent-runner/package-lock.json +1524 -0
  2. package/dist/cli.js +39 -4
  3. package/dist/cli.js.map +1 -1
  4. package/dist/index.d.ts.map +1 -1
  5. package/dist/index.js +34 -0
  6. package/dist/index.js.map +1 -1
  7. package/package.json +7 -1
  8. package/.claude/settings.json +0 -1
  9. package/.claude/skills/add-compact/SKILL.md +0 -135
  10. package/.claude/skills/add-discord/SKILL.md +0 -203
  11. package/.claude/skills/add-gmail/SKILL.md +0 -220
  12. package/.claude/skills/add-image-vision/SKILL.md +0 -94
  13. package/.claude/skills/add-ollama-tool/SKILL.md +0 -153
  14. package/.claude/skills/add-parallel/SKILL.md +0 -290
  15. package/.claude/skills/add-pdf-reader/SKILL.md +0 -104
  16. package/.claude/skills/add-reactions/SKILL.md +0 -117
  17. package/.claude/skills/add-slack/SKILL.md +0 -207
  18. package/.claude/skills/add-telegram/SKILL.md +0 -222
  19. package/.claude/skills/add-telegram-swarm/SKILL.md +0 -384
  20. package/.claude/skills/add-voice-transcription/SKILL.md +0 -148
  21. package/.claude/skills/add-whatsapp/SKILL.md +0 -372
  22. package/.claude/skills/convert-to-apple-container/SKILL.md +0 -175
  23. package/.claude/skills/customize/SKILL.md +0 -110
  24. package/.claude/skills/debug/SKILL.md +0 -349
  25. package/.claude/skills/get-qodo-rules/SKILL.md +0 -122
  26. package/.claude/skills/get-qodo-rules/references/output-format.md +0 -41
  27. package/.claude/skills/get-qodo-rules/references/pagination.md +0 -33
  28. package/.claude/skills/get-qodo-rules/references/repository-scope.md +0 -26
  29. package/.claude/skills/qodo-pr-resolver/SKILL.md +0 -326
  30. package/.claude/skills/qodo-pr-resolver/resources/providers.md +0 -329
  31. package/.claude/skills/setup/SKILL.md +0 -218
  32. package/.claude/skills/update-nanoclaw/SKILL.md +0 -235
  33. package/.claude/skills/update-skills/SKILL.md +0 -130
  34. package/.claude/skills/use-local-whisper/SKILL.md +0 -152
  35. package/.claude/skills/x-integration/SKILL.md +0 -417
  36. package/.claude/skills/x-integration/agent.ts +0 -243
  37. package/.claude/skills/x-integration/host.ts +0 -159
  38. package/.claude/skills/x-integration/lib/browser.ts +0 -148
  39. package/.claude/skills/x-integration/lib/config.ts +0 -62
  40. package/.claude/skills/x-integration/scripts/like.ts +0 -56
  41. package/.claude/skills/x-integration/scripts/post.ts +0 -66
  42. package/.claude/skills/x-integration/scripts/quote.ts +0 -80
  43. package/.claude/skills/x-integration/scripts/reply.ts +0 -74
  44. package/.claude/skills/x-integration/scripts/retweet.ts +0 -62
  45. package/.claude/skills/x-integration/scripts/setup.ts +0 -87
  46. package/.env.example +0 -1
  47. package/.github/CODEOWNERS +0 -10
  48. package/.github/PULL_REQUEST_TEMPLATE.md +0 -14
  49. package/.github/workflows/bump-version.yml +0 -32
  50. package/.github/workflows/ci.yml +0 -25
  51. package/.github/workflows/merge-forward-skills.yml +0 -160
  52. package/.github/workflows/update-tokens.yml +0 -42
  53. package/.husky/pre-commit +0 -1
  54. package/.mcp.json +0 -3
  55. package/.nvmrc +0 -1
  56. package/.prettierrc +0 -3
  57. package/CHANGELOG.md +0 -8
  58. package/CONTRIBUTING.md +0 -23
  59. package/CONTRIBUTORS.md +0 -15
  60. package/NanoClaw_with_Web-Support.md +0 -326
  61. package/README_zh.md +0 -200
  62. package/assets/nanoclaw-favicon.png +0 -0
  63. package/assets/nanoclaw-icon.png +0 -0
  64. package/assets/nanoclaw-logo-dark.png +0 -0
  65. package/assets/nanoclaw-logo.png +0 -0
  66. package/assets/nanoclaw-profile.jpeg +0 -0
  67. package/assets/nanoclaw-sales.png +0 -0
  68. package/assets/social-preview.jpg +0 -0
  69. package/config-examples/mount-allowlist.json +0 -25
  70. package/docs/APPLE-CONTAINER-NETWORKING.md +0 -90
  71. package/docs/DEBUG_CHECKLIST.md +0 -143
  72. package/docs/REQUIREMENTS.md +0 -196
  73. package/docs/SDK_DEEP_DIVE.md +0 -643
  74. package/docs/SECURITY.md +0 -122
  75. package/docs/SPEC.md +0 -785
  76. package/docs/docker-sandboxes.md +0 -359
  77. package/docs/nanoclaw-architecture-final.md +0 -1063
  78. package/docs/nanorepo-architecture.md +0 -168
  79. package/docs/skills-as-branches.md +0 -662
  80. package/groups/global/CLAUDE.md +0 -58
  81. package/groups/main/CLAUDE.md +0 -246
  82. package/launchd/com.nanoclaw.plist +0 -32
  83. package/repo-tokens/README.md +0 -113
  84. package/repo-tokens/action.yml +0 -186
  85. package/repo-tokens/badge.svg +0 -23
  86. package/repo-tokens/examples/green.svg +0 -14
  87. package/repo-tokens/examples/red.svg +0 -14
  88. package/repo-tokens/examples/yellow-green.svg +0 -14
  89. package/repo-tokens/examples/yellow.svg +0 -14
  90. package/scripts/run-migrations.ts +0 -105
  91. package/setup.sh +0 -161
  92. package/src/channels/index.ts +0 -15
  93. package/src/channels/registry.test.ts +0 -42
  94. package/src/channels/registry.ts +0 -32
  95. package/src/channels/web.ts +0 -1931
  96. package/src/cli.ts +0 -254
  97. package/src/config.ts +0 -73
  98. package/src/container-runner.test.ts +0 -210
  99. package/src/container-runner.ts +0 -768
  100. package/src/container-runtime.test.ts +0 -149
  101. package/src/container-runtime.ts +0 -127
  102. package/src/credential-proxy.test.ts +0 -192
  103. package/src/credential-proxy.ts +0 -125
  104. package/src/db.test.ts +0 -484
  105. package/src/db.ts +0 -803
  106. package/src/env.ts +0 -42
  107. package/src/formatting.test.ts +0 -256
  108. package/src/group-folder.test.ts +0 -43
  109. package/src/group-folder.ts +0 -44
  110. package/src/group-queue.test.ts +0 -484
  111. package/src/group-queue.ts +0 -379
  112. package/src/index.ts +0 -854
  113. package/src/ipc-auth.test.ts +0 -679
  114. package/src/ipc.ts +0 -461
  115. package/src/logger.ts +0 -16
  116. package/src/mount-security.ts +0 -419
  117. package/src/remote-control.test.ts +0 -397
  118. package/src/remote-control.ts +0 -224
  119. package/src/router.ts +0 -52
  120. package/src/routing.test.ts +0 -170
  121. package/src/sender-allowlist.test.ts +0 -216
  122. package/src/sender-allowlist.ts +0 -128
  123. package/src/session-commands.test.ts +0 -247
  124. package/src/session-commands.ts +0 -163
  125. package/src/task-scheduler.test.ts +0 -129
  126. package/src/task-scheduler.ts +0 -328
  127. package/src/timezone.test.ts +0 -29
  128. package/src/timezone.ts +0 -16
  129. package/src/types.ts +0 -109
  130. package/tsconfig.json +0 -20
  131. package/vitest.config.ts +0 -7
  132. package/vitest.skills.config.ts +0 -7
@@ -1,484 +0,0 @@
1
- import { describe, it, expect, beforeEach, vi, afterEach } from 'vitest';
2
-
3
- import { GroupQueue } from './group-queue.js';
4
-
5
- // Mock config to control concurrency limit
6
- vi.mock('./config.js', () => ({
7
- DATA_DIR: '/tmp/nanoclaw-test-data',
8
- MAX_CONCURRENT_CONTAINERS: 2,
9
- }));
10
-
11
- // Mock fs operations used by sendMessage/closeStdin
12
- vi.mock('fs', async () => {
13
- const actual = await vi.importActual<typeof import('fs')>('fs');
14
- return {
15
- ...actual,
16
- default: {
17
- ...actual,
18
- mkdirSync: vi.fn(),
19
- writeFileSync: vi.fn(),
20
- renameSync: vi.fn(),
21
- },
22
- };
23
- });
24
-
25
- describe('GroupQueue', () => {
26
- let queue: GroupQueue;
27
-
28
- beforeEach(() => {
29
- vi.useFakeTimers();
30
- queue = new GroupQueue();
31
- });
32
-
33
- afterEach(() => {
34
- vi.useRealTimers();
35
- });
36
-
37
- // --- Single group at a time ---
38
-
39
- it('only runs one container per group at a time', async () => {
40
- let concurrentCount = 0;
41
- let maxConcurrent = 0;
42
-
43
- const processMessages = vi.fn(async (groupJid: string) => {
44
- concurrentCount++;
45
- maxConcurrent = Math.max(maxConcurrent, concurrentCount);
46
- // Simulate async work
47
- await new Promise((resolve) => setTimeout(resolve, 100));
48
- concurrentCount--;
49
- return true;
50
- });
51
-
52
- queue.setProcessMessagesFn(processMessages);
53
-
54
- // Enqueue two messages for the same group
55
- queue.enqueueMessageCheck('group1@g.us');
56
- queue.enqueueMessageCheck('group1@g.us');
57
-
58
- // Advance timers to let the first process complete
59
- await vi.advanceTimersByTimeAsync(200);
60
-
61
- // Second enqueue should have been queued, not concurrent
62
- expect(maxConcurrent).toBe(1);
63
- });
64
-
65
- // --- Global concurrency limit ---
66
-
67
- it('respects global concurrency limit', async () => {
68
- let activeCount = 0;
69
- let maxActive = 0;
70
- const completionCallbacks: Array<() => void> = [];
71
-
72
- const processMessages = vi.fn(async (groupJid: string) => {
73
- activeCount++;
74
- maxActive = Math.max(maxActive, activeCount);
75
- await new Promise<void>((resolve) => completionCallbacks.push(resolve));
76
- activeCount--;
77
- return true;
78
- });
79
-
80
- queue.setProcessMessagesFn(processMessages);
81
-
82
- // Enqueue 3 groups (limit is 2)
83
- queue.enqueueMessageCheck('group1@g.us');
84
- queue.enqueueMessageCheck('group2@g.us');
85
- queue.enqueueMessageCheck('group3@g.us');
86
-
87
- // Let promises settle
88
- await vi.advanceTimersByTimeAsync(10);
89
-
90
- // Only 2 should be active (MAX_CONCURRENT_CONTAINERS = 2)
91
- expect(maxActive).toBe(2);
92
- expect(activeCount).toBe(2);
93
-
94
- // Complete one — third should start
95
- completionCallbacks[0]();
96
- await vi.advanceTimersByTimeAsync(10);
97
-
98
- expect(processMessages).toHaveBeenCalledTimes(3);
99
- });
100
-
101
- // --- Tasks prioritized over messages ---
102
-
103
- it('drains tasks before messages for same group', async () => {
104
- const executionOrder: string[] = [];
105
- let resolveFirst: () => void;
106
-
107
- const processMessages = vi.fn(async (groupJid: string) => {
108
- if (executionOrder.length === 0) {
109
- // First call: block until we release it
110
- await new Promise<void>((resolve) => {
111
- resolveFirst = resolve;
112
- });
113
- }
114
- executionOrder.push('messages');
115
- return true;
116
- });
117
-
118
- queue.setProcessMessagesFn(processMessages);
119
-
120
- // Start processing messages (takes the active slot)
121
- queue.enqueueMessageCheck('group1@g.us');
122
- await vi.advanceTimersByTimeAsync(10);
123
-
124
- // While active, enqueue both a task and pending messages
125
- const taskFn = vi.fn(async () => {
126
- executionOrder.push('task');
127
- });
128
- queue.enqueueTask('group1@g.us', 'task-1', taskFn);
129
- queue.enqueueMessageCheck('group1@g.us');
130
-
131
- // Release the first processing
132
- resolveFirst!();
133
- await vi.advanceTimersByTimeAsync(10);
134
-
135
- // Task should have run before the second message check
136
- expect(executionOrder[0]).toBe('messages'); // first call
137
- expect(executionOrder[1]).toBe('task'); // task runs first in drain
138
- // Messages would run after task completes
139
- });
140
-
141
- // --- Retry with backoff on failure ---
142
-
143
- it('retries with exponential backoff on failure', async () => {
144
- let callCount = 0;
145
-
146
- const processMessages = vi.fn(async () => {
147
- callCount++;
148
- return false; // failure
149
- });
150
-
151
- queue.setProcessMessagesFn(processMessages);
152
- queue.enqueueMessageCheck('group1@g.us');
153
-
154
- // First call happens immediately
155
- await vi.advanceTimersByTimeAsync(10);
156
- expect(callCount).toBe(1);
157
-
158
- // First retry after 5000ms (BASE_RETRY_MS * 2^0)
159
- await vi.advanceTimersByTimeAsync(5000);
160
- await vi.advanceTimersByTimeAsync(10);
161
- expect(callCount).toBe(2);
162
-
163
- // Second retry after 10000ms (BASE_RETRY_MS * 2^1)
164
- await vi.advanceTimersByTimeAsync(10000);
165
- await vi.advanceTimersByTimeAsync(10);
166
- expect(callCount).toBe(3);
167
- });
168
-
169
- // --- Shutdown prevents new enqueues ---
170
-
171
- it('prevents new enqueues after shutdown', async () => {
172
- const processMessages = vi.fn(async () => true);
173
- queue.setProcessMessagesFn(processMessages);
174
-
175
- await queue.shutdown(1000);
176
-
177
- queue.enqueueMessageCheck('group1@g.us');
178
- await vi.advanceTimersByTimeAsync(100);
179
-
180
- expect(processMessages).not.toHaveBeenCalled();
181
- });
182
-
183
- // --- Max retries exceeded ---
184
-
185
- it('stops retrying after MAX_RETRIES and resets', async () => {
186
- let callCount = 0;
187
-
188
- const processMessages = vi.fn(async () => {
189
- callCount++;
190
- return false; // always fail
191
- });
192
-
193
- queue.setProcessMessagesFn(processMessages);
194
- queue.enqueueMessageCheck('group1@g.us');
195
-
196
- // Run through all 5 retries (MAX_RETRIES = 5)
197
- // Initial call
198
- await vi.advanceTimersByTimeAsync(10);
199
- expect(callCount).toBe(1);
200
-
201
- // Retry 1: 5000ms, Retry 2: 10000ms, Retry 3: 20000ms, Retry 4: 40000ms, Retry 5: 80000ms
202
- const retryDelays = [5000, 10000, 20000, 40000, 80000];
203
- for (let i = 0; i < retryDelays.length; i++) {
204
- await vi.advanceTimersByTimeAsync(retryDelays[i] + 10);
205
- expect(callCount).toBe(i + 2);
206
- }
207
-
208
- // After 5 retries (6 total calls), should stop — no more retries
209
- const countAfterMaxRetries = callCount;
210
- await vi.advanceTimersByTimeAsync(200000); // Wait a long time
211
- expect(callCount).toBe(countAfterMaxRetries);
212
- });
213
-
214
- // --- Waiting groups get drained when slots free up ---
215
-
216
- it('drains waiting groups when active slots free up', async () => {
217
- const processed: string[] = [];
218
- const completionCallbacks: Array<() => void> = [];
219
-
220
- const processMessages = vi.fn(async (groupJid: string) => {
221
- processed.push(groupJid);
222
- await new Promise<void>((resolve) => completionCallbacks.push(resolve));
223
- return true;
224
- });
225
-
226
- queue.setProcessMessagesFn(processMessages);
227
-
228
- // Fill both slots
229
- queue.enqueueMessageCheck('group1@g.us');
230
- queue.enqueueMessageCheck('group2@g.us');
231
- await vi.advanceTimersByTimeAsync(10);
232
-
233
- // Queue a third
234
- queue.enqueueMessageCheck('group3@g.us');
235
- await vi.advanceTimersByTimeAsync(10);
236
-
237
- expect(processed).toEqual(['group1@g.us', 'group2@g.us']);
238
-
239
- // Free up a slot
240
- completionCallbacks[0]();
241
- await vi.advanceTimersByTimeAsync(10);
242
-
243
- expect(processed).toContain('group3@g.us');
244
- });
245
-
246
- // --- Running task dedup (Issue #138) ---
247
-
248
- it('rejects duplicate enqueue of a currently-running task', async () => {
249
- let resolveTask: () => void;
250
- let taskCallCount = 0;
251
-
252
- const taskFn = vi.fn(async () => {
253
- taskCallCount++;
254
- await new Promise<void>((resolve) => {
255
- resolveTask = resolve;
256
- });
257
- });
258
-
259
- // Start the task (runs immediately — slot available)
260
- queue.enqueueTask('group1@g.us', 'task-1', taskFn);
261
- await vi.advanceTimersByTimeAsync(10);
262
- expect(taskCallCount).toBe(1);
263
-
264
- // Scheduler poll re-discovers the same task while it's running —
265
- // this must be silently dropped
266
- const dupFn = vi.fn(async () => {});
267
- queue.enqueueTask('group1@g.us', 'task-1', dupFn);
268
- await vi.advanceTimersByTimeAsync(10);
269
-
270
- // Duplicate was NOT queued
271
- expect(dupFn).not.toHaveBeenCalled();
272
-
273
- // Complete the original task
274
- resolveTask!();
275
- await vi.advanceTimersByTimeAsync(10);
276
-
277
- // Only one execution total
278
- expect(taskCallCount).toBe(1);
279
- });
280
-
281
- // --- Idle preemption ---
282
-
283
- it('does NOT preempt active container when not idle', async () => {
284
- const fs = await import('fs');
285
- let resolveProcess: () => void;
286
-
287
- const processMessages = vi.fn(async () => {
288
- await new Promise<void>((resolve) => {
289
- resolveProcess = resolve;
290
- });
291
- return true;
292
- });
293
-
294
- queue.setProcessMessagesFn(processMessages);
295
-
296
- // Start processing (takes the active slot)
297
- queue.enqueueMessageCheck('group1@g.us');
298
- await vi.advanceTimersByTimeAsync(10);
299
-
300
- // Register a process so closeStdin has a groupFolder
301
- queue.registerProcess(
302
- 'group1@g.us',
303
- {} as any,
304
- 'container-1',
305
- 'test-group',
306
- );
307
-
308
- // Enqueue a task while container is active but NOT idle
309
- const taskFn = vi.fn(async () => {});
310
- queue.enqueueTask('group1@g.us', 'task-1', taskFn);
311
-
312
- // _close should NOT have been written (container is working, not idle)
313
- const writeFileSync = vi.mocked(fs.default.writeFileSync);
314
- const closeWrites = writeFileSync.mock.calls.filter(
315
- (call) => typeof call[0] === 'string' && call[0].endsWith('_close'),
316
- );
317
- expect(closeWrites).toHaveLength(0);
318
-
319
- resolveProcess!();
320
- await vi.advanceTimersByTimeAsync(10);
321
- });
322
-
323
- it('preempts idle container when task is enqueued', async () => {
324
- const fs = await import('fs');
325
- let resolveProcess: () => void;
326
-
327
- const processMessages = vi.fn(async () => {
328
- await new Promise<void>((resolve) => {
329
- resolveProcess = resolve;
330
- });
331
- return true;
332
- });
333
-
334
- queue.setProcessMessagesFn(processMessages);
335
-
336
- // Start processing
337
- queue.enqueueMessageCheck('group1@g.us');
338
- await vi.advanceTimersByTimeAsync(10);
339
-
340
- // Register process and mark idle
341
- queue.registerProcess(
342
- 'group1@g.us',
343
- {} as any,
344
- 'container-1',
345
- 'test-group',
346
- );
347
- queue.notifyIdle('group1@g.us');
348
-
349
- // Clear previous writes, then enqueue a task
350
- const writeFileSync = vi.mocked(fs.default.writeFileSync);
351
- writeFileSync.mockClear();
352
-
353
- const taskFn = vi.fn(async () => {});
354
- queue.enqueueTask('group1@g.us', 'task-1', taskFn);
355
-
356
- // _close SHOULD have been written (container is idle)
357
- const closeWrites = writeFileSync.mock.calls.filter(
358
- (call) => typeof call[0] === 'string' && call[0].endsWith('_close'),
359
- );
360
- expect(closeWrites).toHaveLength(1);
361
-
362
- resolveProcess!();
363
- await vi.advanceTimersByTimeAsync(10);
364
- });
365
-
366
- it('sendMessage resets idleWaiting so a subsequent task enqueue does not preempt', async () => {
367
- const fs = await import('fs');
368
- let resolveProcess: () => void;
369
-
370
- const processMessages = vi.fn(async () => {
371
- await new Promise<void>((resolve) => {
372
- resolveProcess = resolve;
373
- });
374
- return true;
375
- });
376
-
377
- queue.setProcessMessagesFn(processMessages);
378
- queue.enqueueMessageCheck('group1@g.us');
379
- await vi.advanceTimersByTimeAsync(10);
380
- queue.registerProcess(
381
- 'group1@g.us',
382
- {} as any,
383
- 'container-1',
384
- 'test-group',
385
- );
386
-
387
- // Container becomes idle
388
- queue.notifyIdle('group1@g.us');
389
-
390
- // A new user message arrives — resets idleWaiting
391
- queue.sendMessage('group1@g.us', 'hello');
392
-
393
- // Task enqueued after message reset — should NOT preempt (agent is working)
394
- const writeFileSync = vi.mocked(fs.default.writeFileSync);
395
- writeFileSync.mockClear();
396
-
397
- const taskFn = vi.fn(async () => {});
398
- queue.enqueueTask('group1@g.us', 'task-1', taskFn);
399
-
400
- const closeWrites = writeFileSync.mock.calls.filter(
401
- (call) => typeof call[0] === 'string' && call[0].endsWith('_close'),
402
- );
403
- expect(closeWrites).toHaveLength(0);
404
-
405
- resolveProcess!();
406
- await vi.advanceTimersByTimeAsync(10);
407
- });
408
-
409
- it('sendMessage returns false for task containers so user messages queue up', async () => {
410
- let resolveTask: () => void;
411
-
412
- const taskFn = vi.fn(async () => {
413
- await new Promise<void>((resolve) => {
414
- resolveTask = resolve;
415
- });
416
- });
417
-
418
- // Start a task (sets isTaskContainer = true)
419
- queue.enqueueTask('group1@g.us', 'task-1', taskFn);
420
- await vi.advanceTimersByTimeAsync(10);
421
- queue.registerProcess(
422
- 'group1@g.us',
423
- {} as any,
424
- 'container-1',
425
- 'test-group',
426
- );
427
-
428
- // sendMessage should return false — user messages must not go to task containers
429
- const result = queue.sendMessage('group1@g.us', 'hello');
430
- expect(result).toBe(false);
431
-
432
- resolveTask!();
433
- await vi.advanceTimersByTimeAsync(10);
434
- });
435
-
436
- it('preempts when idle arrives with pending tasks', async () => {
437
- const fs = await import('fs');
438
- let resolveProcess: () => void;
439
-
440
- const processMessages = vi.fn(async () => {
441
- await new Promise<void>((resolve) => {
442
- resolveProcess = resolve;
443
- });
444
- return true;
445
- });
446
-
447
- queue.setProcessMessagesFn(processMessages);
448
-
449
- // Start processing
450
- queue.enqueueMessageCheck('group1@g.us');
451
- await vi.advanceTimersByTimeAsync(10);
452
-
453
- // Register process and enqueue a task (no idle yet — no preemption)
454
- queue.registerProcess(
455
- 'group1@g.us',
456
- {} as any,
457
- 'container-1',
458
- 'test-group',
459
- );
460
-
461
- const writeFileSync = vi.mocked(fs.default.writeFileSync);
462
- writeFileSync.mockClear();
463
-
464
- const taskFn = vi.fn(async () => {});
465
- queue.enqueueTask('group1@g.us', 'task-1', taskFn);
466
-
467
- let closeWrites = writeFileSync.mock.calls.filter(
468
- (call) => typeof call[0] === 'string' && call[0].endsWith('_close'),
469
- );
470
- expect(closeWrites).toHaveLength(0);
471
-
472
- // Now container becomes idle — should preempt because task is pending
473
- writeFileSync.mockClear();
474
- queue.notifyIdle('group1@g.us');
475
-
476
- closeWrites = writeFileSync.mock.calls.filter(
477
- (call) => typeof call[0] === 'string' && call[0].endsWith('_close'),
478
- );
479
- expect(closeWrites).toHaveLength(1);
480
-
481
- resolveProcess!();
482
- await vi.advanceTimersByTimeAsync(10);
483
- });
484
- });