@talkpilot/core-db 1.1.19 → 1.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (130) hide show
  1. package/.cursor/rules/development.mdc +65 -65
  2. package/DEVELOPMENT.md +98 -98
  3. package/README.md +160 -160
  4. package/dist/talkpilot/calls/calls.getters.d.ts +2 -1
  5. package/dist/talkpilot/calls/calls.getters.d.ts.map +1 -1
  6. package/dist/talkpilot/calls/calls.getters.js +176 -0
  7. package/dist/talkpilot/calls/calls.getters.js.map +1 -1
  8. package/dist/talkpilot/calls/calls.types.d.ts +48 -0
  9. package/dist/talkpilot/calls/calls.types.d.ts.map +1 -1
  10. package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts +1 -0
  11. package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts.map +1 -1
  12. package/dist/talkpilot/clientsConfig/clientsConfig.getters.js +13 -0
  13. package/dist/talkpilot/clientsConfig/clientsConfig.getters.js.map +1 -1
  14. package/jest.config.js +19 -19
  15. package/package.json +45 -45
  16. package/src/__tests__/setup.ts +20 -20
  17. package/src/connection.ts +42 -42
  18. package/src/index.ts +16 -16
  19. package/src/municipal/__tests__/validation.spec.ts +62 -62
  20. package/src/municipal/cities/cities.getters.ts +50 -50
  21. package/src/municipal/cities/cities.types.ts +11 -11
  22. package/src/municipal/cities/index.ts +2 -2
  23. package/src/municipal/departmentsSubjects/departmentsSubjects.getters.ts +282 -282
  24. package/src/municipal/departmentsSubjects/departmentsSubjects.types.ts +72 -72
  25. package/src/municipal/departmentsSubjects/index.ts +9 -9
  26. package/src/municipal/index.ts +21 -21
  27. package/src/municipal/mongodb-client.ts +61 -61
  28. package/src/municipal/streets/index.ts +2 -2
  29. package/src/municipal/streets/streets.getters.ts +125 -125
  30. package/src/municipal/streets/streets.types.ts +18 -18
  31. package/src/municipal/systemInstructions/__tests__/getters.spec.ts +113 -113
  32. package/src/municipal/systemInstructions/__tests__/setters.spec.ts +274 -274
  33. package/src/municipal/systemInstructions/index.ts +7 -7
  34. package/src/municipal/systemInstructions/instructions.getters.ts +57 -57
  35. package/src/municipal/systemInstructions/instructions.setters.ts +119 -119
  36. package/src/municipal/systemInstructions/instructions.types.ts +30 -30
  37. package/src/municipal/tickets/__tests__/tickets.getters.spec.ts +66 -66
  38. package/src/municipal/tickets/index.ts +2 -2
  39. package/src/municipal/tickets/tickets.getters.ts +261 -261
  40. package/src/municipal/tickets/tickets.types.ts +43 -43
  41. package/src/municipal/utils/types.ts +11 -11
  42. package/src/talkpilot/__tests__/db.spec.ts +38 -38
  43. package/src/talkpilot/__tests__/mongodb-client.spec.ts +18 -18
  44. package/src/talkpilot/__tests__/validation.spec.ts +68 -68
  45. package/src/talkpilot/agents/__tests__/agents.getters.spec.ts +29 -29
  46. package/src/talkpilot/agents/agents.getters.ts +34 -34
  47. package/src/talkpilot/agents/agents.types.ts +14 -14
  48. package/src/talkpilot/agents/index.ts +2 -2
  49. package/src/talkpilot/backgroundToolResults/__tests__/backgroundToolResults.getters.spec.ts +147 -147
  50. package/src/talkpilot/backgroundToolResults/backgroundToolResults.getters.ts +65 -65
  51. package/src/talkpilot/backgroundToolResults/backgroundToolResults.types.ts +23 -23
  52. package/src/talkpilot/backgroundToolResults/index.ts +2 -2
  53. package/src/talkpilot/calls/__tests__/callStats.utils.spec.ts +128 -128
  54. package/src/talkpilot/calls/__tests__/calls.spec.ts +252 -252
  55. package/src/talkpilot/calls/calls.getters.ts +446 -248
  56. package/src/talkpilot/calls/calls.types.ts +171 -115
  57. package/src/talkpilot/calls/index.ts +2 -2
  58. package/src/talkpilot/clientAudioBuffers/__tests__/clientAudioBuffer.getters.spec.ts +160 -160
  59. package/src/talkpilot/clientAudioBuffers/clientAudioBuffer.getters.ts +117 -117
  60. package/src/talkpilot/clientAudioBuffers/clientsAudioBuffers.types.ts +25 -25
  61. package/src/talkpilot/clientAudioBuffers/index.ts +2 -2
  62. package/src/talkpilot/clients/clients.getters.ts +16 -16
  63. package/src/talkpilot/clients/clients.types.ts +14 -14
  64. package/src/talkpilot/clients/index.ts +2 -2
  65. package/src/talkpilot/clientsConfig/__tests__/clientsConfig.spec.ts +106 -106
  66. package/src/talkpilot/clientsConfig/clientsConfig.getters.ts +44 -22
  67. package/src/talkpilot/clientsConfig/clientsConfig.types.ts +94 -94
  68. package/src/talkpilot/clientsConfig/index.ts +2 -2
  69. package/src/talkpilot/flows/__tests__/flows.schema.spec.ts +67 -67
  70. package/src/talkpilot/flows/flows.getter.ts +14 -14
  71. package/src/talkpilot/flows/flows.schema.ts +153 -153
  72. package/src/talkpilot/flows/flows.types.ts +184 -184
  73. package/src/talkpilot/flows/index.ts +2 -2
  74. package/src/talkpilot/groups/__tests__/groups.spec.ts +90 -90
  75. package/src/talkpilot/groups/__tests__/phone.utils.spec.ts +32 -32
  76. package/src/talkpilot/groups/groups.getters.ts +30 -30
  77. package/src/talkpilot/groups/groups.types.ts +29 -29
  78. package/src/talkpilot/groups/index.ts +3 -3
  79. package/src/talkpilot/groups/phone.utils.ts +46 -46
  80. package/src/talkpilot/index.ts +29 -29
  81. package/src/talkpilot/leads/index.ts +2 -2
  82. package/src/talkpilot/leads/leads.getter.ts +6 -6
  83. package/src/talkpilot/leads/leads.schema.ts +33 -33
  84. package/src/talkpilot/leads/leads.types.ts +20 -20
  85. package/src/talkpilot/mongodb-client.ts +78 -78
  86. package/src/talkpilot/phone_numbers/__tests__/phone_numbers.spec.ts +247 -247
  87. package/src/talkpilot/phone_numbers/index.ts +2 -2
  88. package/src/talkpilot/phone_numbers/phone_numbers.getter.ts +154 -154
  89. package/src/talkpilot/phone_numbers/phone_numbers.schema.ts +17 -17
  90. package/src/talkpilot/phone_numbers/phone_numbers.types.ts +30 -30
  91. package/src/talkpilot/plans/__tests__/plans.spec.ts +70 -70
  92. package/src/talkpilot/plans/index.ts +2 -2
  93. package/src/talkpilot/plans/plans.getters.ts +132 -132
  94. package/src/talkpilot/plans/plans.types.ts +89 -89
  95. package/src/talkpilot/results/index.ts +7 -7
  96. package/src/talkpilot/results/results.getter.ts +35 -35
  97. package/src/talkpilot/results/results.schema.ts +25 -25
  98. package/src/talkpilot/results/results.types.ts +34 -34
  99. package/src/talkpilot/retry_analyze/__tests__/retryAnalyze.getters.spec.ts +156 -156
  100. package/src/talkpilot/retry_analyze/index.ts +2 -2
  101. package/src/talkpilot/retry_analyze/retryAnalyze.getters.ts +75 -75
  102. package/src/talkpilot/retry_analyze/retryAnalyze.types.ts +13 -13
  103. package/src/talkpilot/sessions/__tests__/sessions.spec.ts +147 -147
  104. package/src/talkpilot/sessions/index.ts +2 -2
  105. package/src/talkpilot/sessions/sessions.getter.ts +92 -92
  106. package/src/talkpilot/sessions/sessions.schema.ts +34 -34
  107. package/src/talkpilot/sessions/sessions.types.ts +30 -30
  108. package/src/talkpilot/subscriptions/__tests__/subscriptions.getters.utils.spec.ts +45 -45
  109. package/src/talkpilot/subscriptions/index.ts +3 -3
  110. package/src/talkpilot/subscriptions/subscriptions.getters.ts +146 -146
  111. package/src/talkpilot/subscriptions/subscriptions.getters.utils.ts +33 -33
  112. package/src/talkpilot/subscriptions/subscriptions.types.ts +66 -66
  113. package/src/talkpilot/utils/__tests__/query.utils.spec.ts +49 -49
  114. package/src/talkpilot/utils/query.utils.ts +21 -21
  115. package/src/test-utils/db-utils.ts +24 -24
  116. package/src/test-utils/factories/index.ts +12 -12
  117. package/src/test-utils/factories/municipal/cities.ts +16 -16
  118. package/src/test-utils/factories/municipal/departmentsSubjects.ts +37 -37
  119. package/src/test-utils/factories/municipal/streets.ts +22 -22
  120. package/src/test-utils/factories/municipal/tickets.ts +39 -39
  121. package/src/test-utils/factories/talkpilot/agents.ts +19 -19
  122. package/src/test-utils/factories/talkpilot/calls.ts +37 -37
  123. package/src/test-utils/factories/talkpilot/clientAudioBuffers.ts +20 -20
  124. package/src/test-utils/factories/talkpilot/clientsConfig.ts +18 -18
  125. package/src/test-utils/factories/talkpilot/flows.ts +33 -33
  126. package/src/test-utils/factories/talkpilot/groups.ts +33 -33
  127. package/src/test-utils/factories/talkpilot/phone_numbers.ts +22 -22
  128. package/src/test-utils/factories/talkpilot/sessions.ts +35 -35
  129. package/src/utils/validation.ts +23 -23
  130. package/tsconfig.json +23 -23
@@ -1,252 +1,252 @@
1
- import {
2
- createCallDoc,
3
- getCallsByFlow,
4
- getCallsByClient,
5
- getCallsByPhoneNumber,
6
- getCallByCallSid,
7
- findCallsByQuery,
8
- countCalls,
9
- getCallsByClientAndDateRange,
10
- getCallsCollection,
11
- pushToolExecution,
12
- } from "../calls.getters";
13
- import type { ToolExecution } from '../calls.types';
14
- import { ObjectId } from "mongodb";
15
- import { createOutGoingCallDoc } from "../../../test-utils/factories";
16
-
17
- describe("db.calls", () => {
18
- it("should return calls by flow", async () => {
19
- const flowId = new ObjectId();
20
- const call1 = createOutGoingCallDoc({ flowId });
21
- const call2 = createOutGoingCallDoc({ flowId });
22
-
23
- await createCallDoc(call1);
24
- await createCallDoc(call2);
25
-
26
- const result = await getCallsByFlow(flowId);
27
-
28
- expect(result.length).toBe(2);
29
- expect(result).toMatchObject([
30
- { callSid: call1.callSid },
31
- { callSid: call2.callSid },
32
- ]);
33
- });
34
-
35
- it("should get call by callSid", async () => {
36
- const call = createOutGoingCallDoc();
37
- await createCallDoc(call);
38
-
39
- const result = await getCallByCallSid(call.callSid);
40
-
41
- expect(result).toBeDefined();
42
- expect(result?.callSid).toBe(call.callSid);
43
- });
44
-
45
- it("should get calls by client", async () => {
46
- const call = createOutGoingCallDoc();
47
- await createCallDoc(call);
48
-
49
- const result = await getCallsByClient(call.clientId);
50
-
51
- expect(result.some((c) => c.callSid === call.callSid)).toBe(true);
52
- });
53
-
54
- it("should get calls by phone number", async () => {
55
- const call = createOutGoingCallDoc();
56
- await createCallDoc(call);
57
-
58
- const result = await getCallsByPhoneNumber(call.customerPhoneNumber);
59
-
60
- expect(result.some((c) => c.callSid === call.callSid)).toBe(true);
61
- });
62
-
63
- describe("findCallsByQuery()", () => {
64
- it("should find calls by status with limit and sort", async () => {
65
- const clientId = "client123";
66
- const call1 = createOutGoingCallDoc({ clientId, status: "completed" });
67
- const call2 = createOutGoingCallDoc({ clientId, status: "completed" });
68
- const call3 = createOutGoingCallDoc({ clientId, status: "busy" });
69
-
70
- // We need to control createdAt to test sort. We'll insert directly.
71
- await getCallsCollection().insertOne({
72
- ...call1,
73
- createdAt: new Date("2023-01-01"),
74
- updatedAt: new Date(),
75
- env: "test",
76
- });
77
- await getCallsCollection().insertOne({
78
- ...call2,
79
- createdAt: new Date("2023-01-02"),
80
- updatedAt: new Date(),
81
- env: "test",
82
- });
83
- await getCallsCollection().insertOne({
84
- ...call3,
85
- createdAt: new Date(),
86
- updatedAt: new Date(),
87
- env: "test",
88
- });
89
-
90
- const result = await findCallsByQuery(
91
- { clientId, status: "completed" },
92
- { sort: { createdAt: -1 }, limit: 1 },
93
- );
94
-
95
- expect(result.length).toBe(1);
96
- expect(result[0].callSid).toBe(call2.callSid);
97
- });
98
- });
99
-
100
- describe("countCalls()", () => {
101
- it("should count calls matching query", async () => {
102
- const clientId = "countClient";
103
- await createCallDoc(
104
- createOutGoingCallDoc({ clientId, status: "completed" }),
105
- );
106
- await createCallDoc(
107
- createOutGoingCallDoc({ clientId, status: "completed" }),
108
- );
109
- await createCallDoc(
110
- createOutGoingCallDoc({ clientId, status: "failed" }),
111
- );
112
-
113
- const count = await countCalls({ clientId, status: "completed" });
114
- expect(count).toBe(2);
115
- });
116
- });
117
-
118
- describe('pushToolExecution()', () => {
119
- let callSid: string;
120
-
121
- beforeEach(async () => {
122
- const call = createOutGoingCallDoc();
123
- await createCallDoc(call);
124
- callSid = call.callSid;
125
- });
126
-
127
- const makeHttpExecution = (overrides?: Partial<ToolExecution>): ToolExecution => ({
128
- toolName: 'sendSms',
129
- executedAt: new Date(),
130
- durationMs: 120,
131
- args: { to: '+1234567890', message: 'hello' },
132
- meta: { kind: 'http', url: 'https://api.example.com/sms', method: 'POST' },
133
- status: 'success',
134
- httpStatus: 200,
135
- ...overrides,
136
- });
137
-
138
- it('should append an execution to a call with none', async () => {
139
- const execution = makeHttpExecution();
140
- await pushToolExecution(callSid, execution);
141
-
142
- const result = await getCallByCallSid(callSid);
143
- expect(result?.toolExecutions).toHaveLength(1);
144
- expect(result?.toolExecutions?.[0]).toMatchObject({
145
- toolName: 'sendSms',
146
- status: 'success',
147
- httpStatus: 200,
148
- });
149
- });
150
-
151
- it('should maintain insertion order across multiple pushes', async () => {
152
- const first = makeHttpExecution({ toolName: 'first' });
153
- const second = makeHttpExecution({ toolName: 'second' });
154
- const third = makeHttpExecution({ toolName: 'third' });
155
-
156
- await pushToolExecution(callSid, first);
157
- await pushToolExecution(callSid, second);
158
- await pushToolExecution(callSid, third);
159
-
160
- const result = await getCallByCallSid(callSid);
161
- expect(result?.toolExecutions?.map(e => e.toolName)).toEqual(['first', 'second', 'third']);
162
- });
163
-
164
- it('should store an internal tool execution', async () => {
165
- const execution: ToolExecution = {
166
- toolName: 'endFlow',
167
- executedAt: new Date(),
168
- durationMs: 5,
169
- args: {},
170
- meta: { kind: 'internal' },
171
- status: 'success',
172
- };
173
-
174
- await pushToolExecution(callSid, execution);
175
-
176
- const result = await getCallByCallSid(callSid);
177
- expect(result?.toolExecutions?.[0]).toMatchObject({
178
- toolName: 'endFlow',
179
- meta: { kind: 'internal' },
180
- });
181
- });
182
-
183
- it('should store redacted args for sensitive tools', async () => {
184
- const execution = makeHttpExecution({ args: { _redacted: true } });
185
- await pushToolExecution(callSid, execution);
186
-
187
- const result = await getCallByCallSid(callSid);
188
- expect(result?.toolExecutions?.[0].args).toEqual({ _redacted: true });
189
- });
190
-
191
- it('should store an error execution', async () => {
192
- const execution = makeHttpExecution({ status: 'error', httpStatus: 500 });
193
- await pushToolExecution(callSid, execution);
194
-
195
- const result = await getCallByCallSid(callSid);
196
- expect(result?.toolExecutions?.[0]).toMatchObject({ status: 'error', httpStatus: 500 });
197
- });
198
-
199
- it('should be a no-op for an unknown callSid', async () => {
200
- await expect(pushToolExecution('nonexistent-sid', makeHttpExecution())).resolves.not.toThrow();
201
- });
202
-
203
- it('should store the response body', async () => {
204
- const execution = makeHttpExecution({
205
- response: { userId: '123', status: 'sent' },
206
- });
207
- await pushToolExecution(callSid, execution);
208
-
209
- const result = await getCallByCallSid(callSid);
210
- expect(result?.toolExecutions?.[0].response).toEqual({ userId: '123', status: 'sent' });
211
- });
212
- });
213
-
214
- describe("getCallsByClientAndDateRange()", () => {
215
- it("should return calls within date range", async () => {
216
- const clientId = "dateRangeClient";
217
- const startDate = new Date("2023-05-01");
218
- const endDate = new Date("2023-05-31");
219
-
220
- const callInside = createOutGoingCallDoc({ clientId });
221
- const callBefore = createOutGoingCallDoc({ clientId });
222
- const callAfter = createOutGoingCallDoc({ clientId });
223
-
224
- await getCallsCollection().insertOne({
225
- ...callInside,
226
- createdAt: new Date("2023-05-15"),
227
- updatedAt: new Date(),
228
- env: "test",
229
- });
230
- await getCallsCollection().insertOne({
231
- ...callBefore,
232
- createdAt: new Date("2023-04-30"),
233
- updatedAt: new Date(),
234
- env: "test",
235
- });
236
- await getCallsCollection().insertOne({
237
- ...callAfter,
238
- createdAt: new Date("2023-06-01"),
239
- updatedAt: new Date(),
240
- env: "test",
241
- });
242
-
243
- const result = await getCallsByClientAndDateRange(
244
- clientId,
245
- startDate,
246
- endDate,
247
- );
248
- expect(result.length).toBe(1);
249
- expect(result[0].callSid).toBe(callInside.callSid);
250
- });
251
- });
252
- });
1
+ import {
2
+ createCallDoc,
3
+ getCallsByFlow,
4
+ getCallsByClient,
5
+ getCallsByPhoneNumber,
6
+ getCallByCallSid,
7
+ findCallsByQuery,
8
+ countCalls,
9
+ getCallsByClientAndDateRange,
10
+ getCallsCollection,
11
+ pushToolExecution,
12
+ } from "../calls.getters";
13
+ import type { ToolExecution } from '../calls.types';
14
+ import { ObjectId } from "mongodb";
15
+ import { createOutGoingCallDoc } from "../../../test-utils/factories";
16
+
17
+ describe("db.calls", () => {
18
+ it("should return calls by flow", async () => {
19
+ const flowId = new ObjectId();
20
+ const call1 = createOutGoingCallDoc({ flowId });
21
+ const call2 = createOutGoingCallDoc({ flowId });
22
+
23
+ await createCallDoc(call1);
24
+ await createCallDoc(call2);
25
+
26
+ const result = await getCallsByFlow(flowId);
27
+
28
+ expect(result.length).toBe(2);
29
+ expect(result).toMatchObject([
30
+ { callSid: call1.callSid },
31
+ { callSid: call2.callSid },
32
+ ]);
33
+ });
34
+
35
+ it("should get call by callSid", async () => {
36
+ const call = createOutGoingCallDoc();
37
+ await createCallDoc(call);
38
+
39
+ const result = await getCallByCallSid(call.callSid);
40
+
41
+ expect(result).toBeDefined();
42
+ expect(result?.callSid).toBe(call.callSid);
43
+ });
44
+
45
+ it("should get calls by client", async () => {
46
+ const call = createOutGoingCallDoc();
47
+ await createCallDoc(call);
48
+
49
+ const result = await getCallsByClient(call.clientId);
50
+
51
+ expect(result.some((c) => c.callSid === call.callSid)).toBe(true);
52
+ });
53
+
54
+ it("should get calls by phone number", async () => {
55
+ const call = createOutGoingCallDoc();
56
+ await createCallDoc(call);
57
+
58
+ const result = await getCallsByPhoneNumber(call.customerPhoneNumber);
59
+
60
+ expect(result.some((c) => c.callSid === call.callSid)).toBe(true);
61
+ });
62
+
63
+ describe("findCallsByQuery()", () => {
64
+ it("should find calls by status with limit and sort", async () => {
65
+ const clientId = "client123";
66
+ const call1 = createOutGoingCallDoc({ clientId, status: "completed" });
67
+ const call2 = createOutGoingCallDoc({ clientId, status: "completed" });
68
+ const call3 = createOutGoingCallDoc({ clientId, status: "busy" });
69
+
70
+ // We need to control createdAt to test sort. We'll insert directly.
71
+ await getCallsCollection().insertOne({
72
+ ...call1,
73
+ createdAt: new Date("2023-01-01"),
74
+ updatedAt: new Date(),
75
+ env: "test",
76
+ });
77
+ await getCallsCollection().insertOne({
78
+ ...call2,
79
+ createdAt: new Date("2023-01-02"),
80
+ updatedAt: new Date(),
81
+ env: "test",
82
+ });
83
+ await getCallsCollection().insertOne({
84
+ ...call3,
85
+ createdAt: new Date(),
86
+ updatedAt: new Date(),
87
+ env: "test",
88
+ });
89
+
90
+ const result = await findCallsByQuery(
91
+ { clientId, status: "completed" },
92
+ { sort: { createdAt: -1 }, limit: 1 },
93
+ );
94
+
95
+ expect(result.length).toBe(1);
96
+ expect(result[0].callSid).toBe(call2.callSid);
97
+ });
98
+ });
99
+
100
+ describe("countCalls()", () => {
101
+ it("should count calls matching query", async () => {
102
+ const clientId = "countClient";
103
+ await createCallDoc(
104
+ createOutGoingCallDoc({ clientId, status: "completed" }),
105
+ );
106
+ await createCallDoc(
107
+ createOutGoingCallDoc({ clientId, status: "completed" }),
108
+ );
109
+ await createCallDoc(
110
+ createOutGoingCallDoc({ clientId, status: "failed" }),
111
+ );
112
+
113
+ const count = await countCalls({ clientId, status: "completed" });
114
+ expect(count).toBe(2);
115
+ });
116
+ });
117
+
118
+ describe('pushToolExecution()', () => {
119
+ let callSid: string;
120
+
121
+ beforeEach(async () => {
122
+ const call = createOutGoingCallDoc();
123
+ await createCallDoc(call);
124
+ callSid = call.callSid;
125
+ });
126
+
127
+ const makeHttpExecution = (overrides?: Partial<ToolExecution>): ToolExecution => ({
128
+ toolName: 'sendSms',
129
+ executedAt: new Date(),
130
+ durationMs: 120,
131
+ args: { to: '+1234567890', message: 'hello' },
132
+ meta: { kind: 'http', url: 'https://api.example.com/sms', method: 'POST' },
133
+ status: 'success',
134
+ httpStatus: 200,
135
+ ...overrides,
136
+ });
137
+
138
+ it('should append an execution to a call with none', async () => {
139
+ const execution = makeHttpExecution();
140
+ await pushToolExecution(callSid, execution);
141
+
142
+ const result = await getCallByCallSid(callSid);
143
+ expect(result?.toolExecutions).toHaveLength(1);
144
+ expect(result?.toolExecutions?.[0]).toMatchObject({
145
+ toolName: 'sendSms',
146
+ status: 'success',
147
+ httpStatus: 200,
148
+ });
149
+ });
150
+
151
+ it('should maintain insertion order across multiple pushes', async () => {
152
+ const first = makeHttpExecution({ toolName: 'first' });
153
+ const second = makeHttpExecution({ toolName: 'second' });
154
+ const third = makeHttpExecution({ toolName: 'third' });
155
+
156
+ await pushToolExecution(callSid, first);
157
+ await pushToolExecution(callSid, second);
158
+ await pushToolExecution(callSid, third);
159
+
160
+ const result = await getCallByCallSid(callSid);
161
+ expect(result?.toolExecutions?.map(e => e.toolName)).toEqual(['first', 'second', 'third']);
162
+ });
163
+
164
+ it('should store an internal tool execution', async () => {
165
+ const execution: ToolExecution = {
166
+ toolName: 'endFlow',
167
+ executedAt: new Date(),
168
+ durationMs: 5,
169
+ args: {},
170
+ meta: { kind: 'internal' },
171
+ status: 'success',
172
+ };
173
+
174
+ await pushToolExecution(callSid, execution);
175
+
176
+ const result = await getCallByCallSid(callSid);
177
+ expect(result?.toolExecutions?.[0]).toMatchObject({
178
+ toolName: 'endFlow',
179
+ meta: { kind: 'internal' },
180
+ });
181
+ });
182
+
183
+ it('should store redacted args for sensitive tools', async () => {
184
+ const execution = makeHttpExecution({ args: { _redacted: true } });
185
+ await pushToolExecution(callSid, execution);
186
+
187
+ const result = await getCallByCallSid(callSid);
188
+ expect(result?.toolExecutions?.[0].args).toEqual({ _redacted: true });
189
+ });
190
+
191
+ it('should store an error execution', async () => {
192
+ const execution = makeHttpExecution({ status: 'error', httpStatus: 500 });
193
+ await pushToolExecution(callSid, execution);
194
+
195
+ const result = await getCallByCallSid(callSid);
196
+ expect(result?.toolExecutions?.[0]).toMatchObject({ status: 'error', httpStatus: 500 });
197
+ });
198
+
199
+ it('should be a no-op for an unknown callSid', async () => {
200
+ await expect(pushToolExecution('nonexistent-sid', makeHttpExecution())).resolves.not.toThrow();
201
+ });
202
+
203
+ it('should store the response body', async () => {
204
+ const execution = makeHttpExecution({
205
+ response: { userId: '123', status: 'sent' },
206
+ });
207
+ await pushToolExecution(callSid, execution);
208
+
209
+ const result = await getCallByCallSid(callSid);
210
+ expect(result?.toolExecutions?.[0].response).toEqual({ userId: '123', status: 'sent' });
211
+ });
212
+ });
213
+
214
+ describe("getCallsByClientAndDateRange()", () => {
215
+ it("should return calls within date range", async () => {
216
+ const clientId = "dateRangeClient";
217
+ const startDate = new Date("2023-05-01");
218
+ const endDate = new Date("2023-05-31");
219
+
220
+ const callInside = createOutGoingCallDoc({ clientId });
221
+ const callBefore = createOutGoingCallDoc({ clientId });
222
+ const callAfter = createOutGoingCallDoc({ clientId });
223
+
224
+ await getCallsCollection().insertOne({
225
+ ...callInside,
226
+ createdAt: new Date("2023-05-15"),
227
+ updatedAt: new Date(),
228
+ env: "test",
229
+ });
230
+ await getCallsCollection().insertOne({
231
+ ...callBefore,
232
+ createdAt: new Date("2023-04-30"),
233
+ updatedAt: new Date(),
234
+ env: "test",
235
+ });
236
+ await getCallsCollection().insertOne({
237
+ ...callAfter,
238
+ createdAt: new Date("2023-06-01"),
239
+ updatedAt: new Date(),
240
+ env: "test",
241
+ });
242
+
243
+ const result = await getCallsByClientAndDateRange(
244
+ clientId,
245
+ startDate,
246
+ endDate,
247
+ );
248
+ expect(result.length).toBe(1);
249
+ expect(result[0].callSid).toBe(callInside.callSid);
250
+ });
251
+ });
252
+ });