@vellumai/assistant 0.5.5 → 0.5.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 (102) hide show
  1. package/Dockerfile +3 -4
  2. package/package.json +1 -1
  3. package/src/__tests__/actor-token-service.test.ts +113 -0
  4. package/src/__tests__/config-schema.test.ts +2 -2
  5. package/src/__tests__/context-window-manager.test.ts +78 -0
  6. package/src/__tests__/conversation-title-service.test.ts +30 -1
  7. package/src/__tests__/docker-signing-key-bootstrap.test.ts +207 -0
  8. package/src/__tests__/memory-regressions.test.ts +8 -30
  9. package/src/__tests__/require-fresh-approval.test.ts +4 -0
  10. package/src/__tests__/tool-executor-lifecycle-events.test.ts +4 -0
  11. package/src/__tests__/tool-executor.test.ts +4 -0
  12. package/src/cli/commands/conversations.ts +0 -18
  13. package/src/config/env.ts +8 -2
  14. package/src/config/feature-flag-registry.json +0 -8
  15. package/src/config/schema.ts +0 -12
  16. package/src/config/schemas/memory.ts +0 -4
  17. package/src/config/schemas/platform.ts +1 -1
  18. package/src/config/schemas/security.ts +4 -0
  19. package/src/context/window-manager.ts +53 -2
  20. package/src/daemon/config-watcher.ts +1 -4
  21. package/src/daemon/conversation-agent-loop.ts +0 -60
  22. package/src/daemon/conversation-memory.ts +0 -117
  23. package/src/daemon/conversation-runtime-assembly.ts +0 -2
  24. package/src/daemon/handlers/conversations.ts +0 -11
  25. package/src/daemon/lifecycle.ts +3 -46
  26. package/src/followups/followup-store.ts +5 -2
  27. package/src/memory/conversation-crud.ts +0 -236
  28. package/src/memory/conversation-title-service.ts +26 -10
  29. package/src/memory/db-init.ts +5 -13
  30. package/src/memory/indexer.ts +15 -106
  31. package/src/memory/job-handlers/embedding.ts +0 -79
  32. package/src/memory/job-utils.ts +1 -1
  33. package/src/memory/jobs-store.ts +0 -8
  34. package/src/memory/jobs-worker.ts +0 -20
  35. package/src/memory/migrations/189-drop-simplified-memory.ts +42 -0
  36. package/src/memory/migrations/index.ts +1 -3
  37. package/src/memory/qdrant-client.ts +4 -6
  38. package/src/memory/schema/conversations.ts +0 -3
  39. package/src/memory/schema/index.ts +0 -2
  40. package/src/messaging/draft-store.ts +2 -2
  41. package/src/permissions/defaults.ts +3 -3
  42. package/src/permissions/trust-client.ts +2 -13
  43. package/src/permissions/trust-store.ts +8 -3
  44. package/src/runtime/auth/route-policy.ts +14 -0
  45. package/src/runtime/auth/token-service.ts +133 -0
  46. package/src/runtime/http-server.ts +2 -0
  47. package/src/runtime/routes/conversation-management-routes.ts +0 -36
  48. package/src/runtime/routes/conversation-query-routes.ts +44 -2
  49. package/src/runtime/routes/conversation-routes.ts +2 -1
  50. package/src/runtime/routes/memory-item-routes.test.ts +221 -3
  51. package/src/runtime/routes/memory-item-routes.ts +124 -2
  52. package/src/runtime/routes/upgrade-broadcast-routes.ts +151 -0
  53. package/src/schedule/schedule-store.ts +0 -21
  54. package/src/skills/inline-command-render.ts +5 -1
  55. package/src/skills/inline-command-runner.ts +30 -2
  56. package/src/tools/memory/handlers.ts +1 -129
  57. package/src/tools/permission-checker.ts +18 -0
  58. package/src/tools/skills/load.ts +9 -2
  59. package/src/util/platform.ts +5 -5
  60. package/src/util/xml.ts +8 -0
  61. package/src/workspace/heartbeat-service.ts +5 -24
  62. package/src/__tests__/archive-recall.test.ts +0 -560
  63. package/src/__tests__/conversation-memory-dirty-tail.test.ts +0 -150
  64. package/src/__tests__/conversation-switch-memory-reduction.test.ts +0 -474
  65. package/src/__tests__/db-memory-archive-migration.test.ts +0 -372
  66. package/src/__tests__/db-memory-brief-state-migration.test.ts +0 -213
  67. package/src/__tests__/db-memory-reducer-checkpoints.test.ts +0 -273
  68. package/src/__tests__/memory-brief-open-loops.test.ts +0 -530
  69. package/src/__tests__/memory-brief-time.test.ts +0 -285
  70. package/src/__tests__/memory-brief-wrapper.test.ts +0 -311
  71. package/src/__tests__/memory-chunk-archive.test.ts +0 -400
  72. package/src/__tests__/memory-chunk-dual-write.test.ts +0 -453
  73. package/src/__tests__/memory-episode-archive.test.ts +0 -370
  74. package/src/__tests__/memory-episode-dual-write.test.ts +0 -626
  75. package/src/__tests__/memory-observation-archive.test.ts +0 -375
  76. package/src/__tests__/memory-observation-dual-write.test.ts +0 -318
  77. package/src/__tests__/memory-reducer-job.test.ts +0 -538
  78. package/src/__tests__/memory-reducer-scheduling.test.ts +0 -473
  79. package/src/__tests__/memory-reducer-store.test.ts +0 -728
  80. package/src/__tests__/memory-reducer-types.test.ts +0 -707
  81. package/src/__tests__/memory-reducer.test.ts +0 -704
  82. package/src/__tests__/memory-simplified-config.test.ts +0 -281
  83. package/src/__tests__/simplified-memory-e2e.test.ts +0 -666
  84. package/src/__tests__/simplified-memory-runtime.test.ts +0 -616
  85. package/src/config/schemas/memory-simplified.ts +0 -101
  86. package/src/memory/archive-recall.ts +0 -516
  87. package/src/memory/archive-store.ts +0 -400
  88. package/src/memory/brief-formatting.ts +0 -33
  89. package/src/memory/brief-open-loops.ts +0 -266
  90. package/src/memory/brief-time.ts +0 -162
  91. package/src/memory/brief.ts +0 -75
  92. package/src/memory/job-handlers/backfill-simplified-memory.ts +0 -462
  93. package/src/memory/job-handlers/reduce-conversation-memory.ts +0 -229
  94. package/src/memory/migrations/185-memory-brief-state.ts +0 -52
  95. package/src/memory/migrations/186-memory-archive.ts +0 -109
  96. package/src/memory/migrations/187-memory-reducer-checkpoints.ts +0 -19
  97. package/src/memory/reducer-scheduler.ts +0 -242
  98. package/src/memory/reducer-store.ts +0 -271
  99. package/src/memory/reducer-types.ts +0 -106
  100. package/src/memory/reducer.ts +0 -467
  101. package/src/memory/schema/memory-archive.ts +0 -121
  102. package/src/memory/schema/memory-brief.ts +0 -55
@@ -1,704 +0,0 @@
1
- import { beforeEach, describe, expect, mock, test } from "bun:test";
2
-
3
- import type { Provider, ProviderResponse } from "../providers/types.js";
4
-
5
- // ---------------------------------------------------------------------------
6
- // Mocks — declared before imports that depend on them
7
- // ---------------------------------------------------------------------------
8
-
9
- function makeLoggerStub(): Record<string, unknown> {
10
- const stub: Record<string, unknown> = {};
11
- for (const m of [
12
- "info",
13
- "warn",
14
- "error",
15
- "debug",
16
- "trace",
17
- "fatal",
18
- "silent",
19
- "child",
20
- ]) {
21
- stub[m] = m === "child" ? () => makeLoggerStub() : () => {};
22
- }
23
- return stub;
24
- }
25
-
26
- mock.module("../util/logger.js", () => ({
27
- getLogger: () => makeLoggerStub(),
28
- }));
29
-
30
- // ---------------------------------------------------------------------------
31
- // Mock provider
32
- // ---------------------------------------------------------------------------
33
-
34
- const mockSendMessage = mock<Provider["sendMessage"]>();
35
- const mockProvider: Provider = {
36
- name: "mock-reducer-provider",
37
- sendMessage: mockSendMessage,
38
- };
39
-
40
- let providerAvailable = true;
41
-
42
- mock.module("../providers/provider-send-message.js", () => ({
43
- getConfiguredProvider: async () => (providerAvailable ? mockProvider : null),
44
- createTimeout: (ms: number) => {
45
- const controller = new AbortController();
46
- const timer = setTimeout(() => controller.abort(), ms);
47
- return {
48
- signal: controller.signal,
49
- cleanup: () => clearTimeout(timer),
50
- };
51
- },
52
- extractText: (response: ProviderResponse) => {
53
- const block = response.content.find(
54
- (b): b is Extract<(typeof response.content)[number], { type: "text" }> =>
55
- b.type === "text",
56
- );
57
- return block?.text?.trim() ?? "";
58
- },
59
- }));
60
-
61
- // ---------------------------------------------------------------------------
62
- // Import module under test AFTER mocks
63
- // ---------------------------------------------------------------------------
64
-
65
- import {
66
- buildReducerSystemPrompt,
67
- buildReducerUserMessage,
68
- type ReducerPromptInput,
69
- runReducer,
70
- } from "../memory/reducer.js";
71
- import { EMPTY_REDUCER_RESULT } from "../memory/reducer-types.js";
72
-
73
- // ---------------------------------------------------------------------------
74
- // Helpers
75
- // ---------------------------------------------------------------------------
76
-
77
- function makeInput(
78
- overrides: Partial<ReducerPromptInput> = {},
79
- ): ReducerPromptInput {
80
- return {
81
- conversationId: "conv-test-1",
82
- newMessages: [
83
- { role: "user", content: "I'm traveling to Paris next week." },
84
- {
85
- role: "assistant",
86
- content: "That sounds exciting! Do you need help planning?",
87
- },
88
- ],
89
- existingTimeContexts: [],
90
- existingOpenLoops: [],
91
- nowMs: 1700000000000,
92
- scopeId: "scope-default",
93
- ...overrides,
94
- };
95
- }
96
-
97
- function makeProviderResponse(jsonOutput: unknown): ProviderResponse {
98
- return {
99
- content: [{ type: "text", text: JSON.stringify(jsonOutput) }],
100
- model: "mock-model",
101
- usage: { inputTokens: 100, outputTokens: 50 },
102
- stopReason: "end_turn",
103
- };
104
- }
105
-
106
- // ---------------------------------------------------------------------------
107
- // Tests: buildReducerSystemPrompt
108
- // ---------------------------------------------------------------------------
109
-
110
- describe("buildReducerSystemPrompt", () => {
111
- test("contains key structural instructions", () => {
112
- const prompt = buildReducerSystemPrompt();
113
- expect(prompt).toContain("timeContexts");
114
- expect(prompt).toContain("openLoops");
115
- expect(prompt).toContain("archiveObservations");
116
- expect(prompt).toContain("archiveEpisodes");
117
- expect(prompt).toContain("JSON");
118
- });
119
- });
120
-
121
- // ---------------------------------------------------------------------------
122
- // Tests: buildReducerUserMessage
123
- // ---------------------------------------------------------------------------
124
-
125
- describe("buildReducerUserMessage", () => {
126
- test("includes current time, conversation ID, and scope", () => {
127
- const input = makeInput();
128
- const msg = buildReducerUserMessage(input);
129
- expect(msg).toContain("conv-test-1");
130
- expect(msg).toContain("scope-default");
131
- expect(msg).toContain("1700000000000");
132
- });
133
-
134
- test("includes new messages", () => {
135
- const input = makeInput();
136
- const msg = buildReducerUserMessage(input);
137
- expect(msg).toContain("[user]: I'm traveling to Paris next week.");
138
- expect(msg).toContain("[assistant]: That sounds exciting!");
139
- });
140
-
141
- test("includes existing time contexts when provided", () => {
142
- const input = makeInput({
143
- existingTimeContexts: [
144
- { id: "tc-1", summary: "User on vacation until Friday" },
145
- ],
146
- });
147
- const msg = buildReducerUserMessage(input);
148
- expect(msg).toContain("Active time contexts");
149
- expect(msg).toContain("[tc-1] User on vacation until Friday");
150
- });
151
-
152
- test("includes existing open loops when provided", () => {
153
- const input = makeInput({
154
- existingOpenLoops: [
155
- { id: "ol-1", summary: "Follow up with Bob", status: "open" },
156
- ],
157
- });
158
- const msg = buildReducerUserMessage(input);
159
- expect(msg).toContain("Active open loops");
160
- expect(msg).toContain("[ol-1] (open) Follow up with Bob");
161
- });
162
-
163
- test("omits time context section when none exist", () => {
164
- const input = makeInput({ existingTimeContexts: [] });
165
- const msg = buildReducerUserMessage(input);
166
- expect(msg).not.toContain("Active time contexts");
167
- });
168
-
169
- test("omits open loop section when none exist", () => {
170
- const input = makeInput({ existingOpenLoops: [] });
171
- const msg = buildReducerUserMessage(input);
172
- expect(msg).not.toContain("Active open loops");
173
- });
174
- });
175
-
176
- // ---------------------------------------------------------------------------
177
- // Tests: runReducer — event extraction
178
- // ---------------------------------------------------------------------------
179
-
180
- describe("runReducer — event extraction", () => {
181
- beforeEach(() => {
182
- mockSendMessage.mockClear();
183
- providerAvailable = true;
184
- });
185
-
186
- test("extracts archive observations from provider response", async () => {
187
- mockSendMessage.mockResolvedValueOnce(
188
- makeProviderResponse({
189
- archiveObservations: [
190
- {
191
- content: "User is planning a trip to Paris",
192
- role: "user",
193
- modality: "text",
194
- source: "vellum",
195
- },
196
- {
197
- content: "User prefers window seats on flights",
198
- role: "user",
199
- },
200
- ],
201
- }),
202
- );
203
-
204
- const result = await runReducer(makeInput());
205
-
206
- expect(result.archiveObservations).toHaveLength(2);
207
- expect(result.archiveObservations[0]).toEqual({
208
- content: "User is planning a trip to Paris",
209
- role: "user",
210
- modality: "text",
211
- source: "vellum",
212
- });
213
- expect(result.archiveObservations[1]).toEqual({
214
- content: "User prefers window seats on flights",
215
- role: "user",
216
- });
217
- });
218
-
219
- test("extracts archive episodes from provider response", async () => {
220
- mockSendMessage.mockResolvedValueOnce(
221
- makeProviderResponse({
222
- archiveEpisodes: [
223
- {
224
- title: "Paris trip planning",
225
- summary:
226
- "User discussed upcoming trip to Paris, mentioned interest in museums.",
227
- source: "vellum",
228
- },
229
- ],
230
- }),
231
- );
232
-
233
- const result = await runReducer(makeInput());
234
-
235
- expect(result.archiveEpisodes).toHaveLength(1);
236
- expect(result.archiveEpisodes[0]).toEqual({
237
- title: "Paris trip planning",
238
- summary:
239
- "User discussed upcoming trip to Paris, mentioned interest in museums.",
240
- source: "vellum",
241
- });
242
- });
243
- });
244
-
245
- // ---------------------------------------------------------------------------
246
- // Tests: runReducer — temporary situation extraction
247
- // ---------------------------------------------------------------------------
248
-
249
- describe("runReducer — temporary situation extraction", () => {
250
- beforeEach(() => {
251
- mockSendMessage.mockClear();
252
- providerAvailable = true;
253
- });
254
-
255
- test("extracts time context create operations", async () => {
256
- mockSendMessage.mockResolvedValueOnce(
257
- makeProviderResponse({
258
- timeContexts: [
259
- {
260
- action: "create",
261
- summary: "User traveling to Paris next week",
262
- source: "conversation",
263
- activeFrom: 1700000000000,
264
- activeUntil: 1700604800000,
265
- },
266
- ],
267
- }),
268
- );
269
-
270
- const result = await runReducer(makeInput());
271
-
272
- expect(result.timeContexts).toHaveLength(1);
273
- expect(result.timeContexts[0]).toEqual({
274
- action: "create",
275
- summary: "User traveling to Paris next week",
276
- source: "conversation",
277
- activeFrom: 1700000000000,
278
- activeUntil: 1700604800000,
279
- });
280
- });
281
-
282
- test("extracts time context update operations referencing existing IDs", async () => {
283
- mockSendMessage.mockResolvedValueOnce(
284
- makeProviderResponse({
285
- timeContexts: [
286
- {
287
- action: "update",
288
- id: "tc-existing-1",
289
- summary: "Trip extended to two weeks",
290
- activeUntil: 1701209600000,
291
- },
292
- ],
293
- }),
294
- );
295
-
296
- const result = await runReducer(
297
- makeInput({
298
- existingTimeContexts: [
299
- { id: "tc-existing-1", summary: "User traveling next week" },
300
- ],
301
- }),
302
- );
303
-
304
- expect(result.timeContexts).toHaveLength(1);
305
- expect(result.timeContexts[0]).toEqual({
306
- action: "update",
307
- id: "tc-existing-1",
308
- summary: "Trip extended to two weeks",
309
- activeUntil: 1701209600000,
310
- });
311
- });
312
- });
313
-
314
- // ---------------------------------------------------------------------------
315
- // Tests: runReducer — open-loop creation
316
- // ---------------------------------------------------------------------------
317
-
318
- describe("runReducer — open-loop creation", () => {
319
- beforeEach(() => {
320
- mockSendMessage.mockClear();
321
- providerAvailable = true;
322
- });
323
-
324
- test("extracts open loop create operations", async () => {
325
- mockSendMessage.mockResolvedValueOnce(
326
- makeProviderResponse({
327
- openLoops: [
328
- {
329
- action: "create",
330
- summary: "Book hotel in Paris",
331
- source: "conversation",
332
- dueAt: 1700172800000,
333
- },
334
- ],
335
- }),
336
- );
337
-
338
- const result = await runReducer(makeInput());
339
-
340
- expect(result.openLoops).toHaveLength(1);
341
- expect(result.openLoops[0]).toEqual({
342
- action: "create",
343
- summary: "Book hotel in Paris",
344
- source: "conversation",
345
- dueAt: 1700172800000,
346
- });
347
- });
348
-
349
- test("extracts open loop create without optional dueAt", async () => {
350
- mockSendMessage.mockResolvedValueOnce(
351
- makeProviderResponse({
352
- openLoops: [
353
- {
354
- action: "create",
355
- summary: "Research museums in Paris",
356
- source: "conversation",
357
- },
358
- ],
359
- }),
360
- );
361
-
362
- const result = await runReducer(makeInput());
363
-
364
- expect(result.openLoops).toHaveLength(1);
365
- const op = result.openLoops[0];
366
- expect(op.action).toBe("create");
367
- if (op.action === "create") {
368
- expect(op.summary).toBe("Research museums in Paris");
369
- expect(op.dueAt).toBeUndefined();
370
- }
371
- });
372
- });
373
-
374
- // ---------------------------------------------------------------------------
375
- // Tests: runReducer — explicit resolution
376
- // ---------------------------------------------------------------------------
377
-
378
- describe("runReducer — explicit resolution", () => {
379
- beforeEach(() => {
380
- mockSendMessage.mockClear();
381
- providerAvailable = true;
382
- });
383
-
384
- test("extracts time context resolve operations", async () => {
385
- mockSendMessage.mockResolvedValueOnce(
386
- makeProviderResponse({
387
- timeContexts: [
388
- {
389
- action: "resolve",
390
- id: "tc-old-1",
391
- },
392
- ],
393
- }),
394
- );
395
-
396
- const result = await runReducer(
397
- makeInput({
398
- existingTimeContexts: [
399
- { id: "tc-old-1", summary: "Conference this week" },
400
- ],
401
- }),
402
- );
403
-
404
- expect(result.timeContexts).toHaveLength(1);
405
- expect(result.timeContexts[0]).toEqual({
406
- action: "resolve",
407
- id: "tc-old-1",
408
- });
409
- });
410
-
411
- test("extracts open loop resolve with 'resolved' status", async () => {
412
- mockSendMessage.mockResolvedValueOnce(
413
- makeProviderResponse({
414
- openLoops: [
415
- {
416
- action: "resolve",
417
- id: "ol-done-1",
418
- status: "resolved",
419
- },
420
- ],
421
- }),
422
- );
423
-
424
- const result = await runReducer(
425
- makeInput({
426
- existingOpenLoops: [
427
- { id: "ol-done-1", summary: "Send report to Alice", status: "open" },
428
- ],
429
- }),
430
- );
431
-
432
- expect(result.openLoops).toHaveLength(1);
433
- expect(result.openLoops[0]).toEqual({
434
- action: "resolve",
435
- id: "ol-done-1",
436
- status: "resolved",
437
- });
438
- });
439
-
440
- test("extracts open loop resolve with 'expired' status", async () => {
441
- mockSendMessage.mockResolvedValueOnce(
442
- makeProviderResponse({
443
- openLoops: [
444
- {
445
- action: "resolve",
446
- id: "ol-expired-1",
447
- status: "expired",
448
- },
449
- ],
450
- }),
451
- );
452
-
453
- const result = await runReducer(
454
- makeInput({
455
- existingOpenLoops: [
456
- {
457
- id: "ol-expired-1",
458
- summary: "RSVP deadline passed",
459
- status: "open",
460
- },
461
- ],
462
- }),
463
- );
464
-
465
- expect(result.openLoops).toHaveLength(1);
466
- expect(result.openLoops[0]).toEqual({
467
- action: "resolve",
468
- id: "ol-expired-1",
469
- status: "expired",
470
- });
471
- });
472
- });
473
-
474
- // ---------------------------------------------------------------------------
475
- // Tests: runReducer — provider unavailable / error handling
476
- // ---------------------------------------------------------------------------
477
-
478
- describe("runReducer — error handling", () => {
479
- beforeEach(() => {
480
- mockSendMessage.mockClear();
481
- providerAvailable = true;
482
- });
483
-
484
- test("returns EMPTY_REDUCER_RESULT when no provider is available", async () => {
485
- providerAvailable = false;
486
-
487
- const result = await runReducer(makeInput());
488
-
489
- expect(result).toBe(EMPTY_REDUCER_RESULT);
490
- expect(mockSendMessage).not.toHaveBeenCalled();
491
- });
492
-
493
- test("returns EMPTY_REDUCER_RESULT when provider throws", async () => {
494
- mockSendMessage.mockRejectedValueOnce(new Error("Provider error"));
495
-
496
- const result = await runReducer(makeInput());
497
-
498
- expect(result).toBe(EMPTY_REDUCER_RESULT);
499
- });
500
-
501
- test("returns EMPTY_REDUCER_RESULT when provider returns empty text", async () => {
502
- mockSendMessage.mockResolvedValueOnce({
503
- content: [{ type: "text", text: "" }],
504
- model: "mock-model",
505
- usage: { inputTokens: 10, outputTokens: 0 },
506
- stopReason: "end_turn",
507
- });
508
-
509
- const result = await runReducer(makeInput());
510
-
511
- expect(result).toBe(EMPTY_REDUCER_RESULT);
512
- });
513
-
514
- test("returns EMPTY_REDUCER_RESULT when provider returns invalid JSON", async () => {
515
- mockSendMessage.mockResolvedValueOnce({
516
- content: [{ type: "text", text: "not valid json at all" }],
517
- model: "mock-model",
518
- usage: { inputTokens: 10, outputTokens: 5 },
519
- stopReason: "end_turn",
520
- });
521
-
522
- const result = await runReducer(makeInput());
523
-
524
- expect(result).toBe(EMPTY_REDUCER_RESULT);
525
- });
526
-
527
- test("returns EMPTY_REDUCER_RESULT when provider returns empty object", async () => {
528
- mockSendMessage.mockResolvedValueOnce(makeProviderResponse({}));
529
-
530
- const result = await runReducer(makeInput());
531
-
532
- expect(result).toEqual({
533
- timeContexts: [],
534
- openLoops: [],
535
- archiveObservations: [],
536
- archiveEpisodes: [],
537
- });
538
- expect(result).not.toBe(EMPTY_REDUCER_RESULT);
539
- });
540
- });
541
-
542
- // ---------------------------------------------------------------------------
543
- // Tests: runReducer — provider call arguments
544
- // ---------------------------------------------------------------------------
545
-
546
- describe("runReducer — provider call arguments", () => {
547
- beforeEach(() => {
548
- mockSendMessage.mockClear();
549
- providerAvailable = true;
550
- mockSendMessage.mockResolvedValue(makeProviderResponse({}));
551
- });
552
-
553
- test("sends messages with correct structure", async () => {
554
- await runReducer(makeInput());
555
-
556
- expect(mockSendMessage).toHaveBeenCalledTimes(1);
557
- const [messages] = mockSendMessage.mock.calls[0] as Parameters<
558
- Provider["sendMessage"]
559
- >;
560
-
561
- // Should be a single user message
562
- expect(messages).toHaveLength(1);
563
- expect(messages[0].role).toBe("user");
564
- expect(messages[0].content).toHaveLength(1);
565
- expect(messages[0].content[0].type).toBe("text");
566
- });
567
-
568
- test("does not send any tool definitions", async () => {
569
- await runReducer(makeInput());
570
-
571
- const [, tools] = mockSendMessage.mock.calls[0] as Parameters<
572
- Provider["sendMessage"]
573
- >;
574
- expect(tools).toBeUndefined();
575
- });
576
-
577
- test("sends a system prompt", async () => {
578
- await runReducer(makeInput());
579
-
580
- const [, , systemPrompt] = mockSendMessage.mock.calls[0] as Parameters<
581
- Provider["sendMessage"]
582
- >;
583
- expect(systemPrompt).toBeTruthy();
584
- expect(typeof systemPrompt).toBe("string");
585
- });
586
-
587
- test("uses latency-optimized model intent", async () => {
588
- await runReducer(makeInput());
589
-
590
- const [, , , options] = mockSendMessage.mock.calls[0] as Parameters<
591
- Provider["sendMessage"]
592
- >;
593
- expect(options?.config?.modelIntent).toBe("latency-optimized");
594
- });
595
-
596
- test("passes abort signal to provider", async () => {
597
- await runReducer(makeInput());
598
-
599
- const [, , , options] = mockSendMessage.mock.calls[0] as Parameters<
600
- Provider["sendMessage"]
601
- >;
602
- expect(options?.signal).toBeDefined();
603
- expect(options?.signal).toBeInstanceOf(AbortSignal);
604
- });
605
- });
606
-
607
- // ---------------------------------------------------------------------------
608
- // Tests: runReducer — side-effect-free guarantee
609
- // ---------------------------------------------------------------------------
610
-
611
- describe("runReducer — side-effect-free", () => {
612
- beforeEach(() => {
613
- mockSendMessage.mockClear();
614
- providerAvailable = true;
615
- });
616
-
617
- test("returns typed result without performing any writes", async () => {
618
- const fullResponse = {
619
- timeContexts: [
620
- {
621
- action: "create",
622
- summary: "Paris trip",
623
- source: "conversation",
624
- activeFrom: 1700000000000,
625
- activeUntil: 1700604800000,
626
- },
627
- ],
628
- openLoops: [
629
- {
630
- action: "create",
631
- summary: "Book hotel",
632
- source: "conversation",
633
- },
634
- ],
635
- archiveObservations: [
636
- {
637
- content: "User likes museums",
638
- role: "user",
639
- },
640
- ],
641
- archiveEpisodes: [
642
- {
643
- title: "Trip planning",
644
- summary: "Discussed Paris trip",
645
- },
646
- ],
647
- };
648
-
649
- mockSendMessage.mockResolvedValueOnce(makeProviderResponse(fullResponse));
650
-
651
- const result = await runReducer(makeInput());
652
-
653
- // Verify the result is correctly typed and populated
654
- expect(result.timeContexts).toHaveLength(1);
655
- expect(result.openLoops).toHaveLength(1);
656
- expect(result.archiveObservations).toHaveLength(1);
657
- expect(result.archiveEpisodes).toHaveLength(1);
658
-
659
- // The function only called sendMessage — no other side effects
660
- expect(mockSendMessage).toHaveBeenCalledTimes(1);
661
- });
662
-
663
- test("handles mixed valid and invalid operations gracefully", async () => {
664
- mockSendMessage.mockResolvedValueOnce(
665
- makeProviderResponse({
666
- timeContexts: [
667
- // valid
668
- {
669
- action: "create",
670
- summary: "Valid context",
671
- source: "conversation",
672
- activeFrom: 1000,
673
- activeUntil: 2000,
674
- },
675
- // invalid — missing summary
676
- {
677
- action: "create",
678
- source: "conversation",
679
- activeFrom: 1000,
680
- activeUntil: 2000,
681
- },
682
- ],
683
- openLoops: [
684
- // valid
685
- {
686
- action: "create",
687
- summary: "Valid loop",
688
- source: "conversation",
689
- },
690
- // invalid — missing source
691
- { action: "create", summary: "Bad loop" },
692
- ],
693
- }),
694
- );
695
-
696
- const result = await runReducer(makeInput());
697
-
698
- // Only valid operations should be present
699
- expect(result.timeContexts).toHaveLength(1);
700
- expect(result.timeContexts[0].action).toBe("create");
701
- expect(result.openLoops).toHaveLength(1);
702
- expect(result.openLoops[0].action).toBe("create");
703
- });
704
- });