@talkpilot/core-db 1.1.18 → 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 (133) 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 +49 -2
  9. package/dist/talkpilot/calls/calls.types.d.ts.map +1 -1
  10. package/dist/talkpilot/calls/calls.types.js.map +1 -1
  11. package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts +1 -0
  12. package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts.map +1 -1
  13. package/dist/talkpilot/clientsConfig/clientsConfig.getters.js +13 -0
  14. package/dist/talkpilot/clientsConfig/clientsConfig.getters.js.map +1 -1
  15. package/dist/talkpilot/clientsConfig/clientsConfig.types.d.ts +2 -0
  16. package/dist/talkpilot/clientsConfig/clientsConfig.types.d.ts.map +1 -1
  17. package/jest.config.js +19 -19
  18. package/package.json +45 -45
  19. package/src/__tests__/setup.ts +20 -20
  20. package/src/connection.ts +42 -42
  21. package/src/index.ts +16 -16
  22. package/src/municipal/__tests__/validation.spec.ts +62 -62
  23. package/src/municipal/cities/cities.getters.ts +50 -50
  24. package/src/municipal/cities/cities.types.ts +11 -11
  25. package/src/municipal/cities/index.ts +2 -2
  26. package/src/municipal/departmentsSubjects/departmentsSubjects.getters.ts +282 -282
  27. package/src/municipal/departmentsSubjects/departmentsSubjects.types.ts +72 -72
  28. package/src/municipal/departmentsSubjects/index.ts +9 -9
  29. package/src/municipal/index.ts +21 -21
  30. package/src/municipal/mongodb-client.ts +61 -61
  31. package/src/municipal/streets/index.ts +2 -2
  32. package/src/municipal/streets/streets.getters.ts +125 -125
  33. package/src/municipal/streets/streets.types.ts +18 -18
  34. package/src/municipal/systemInstructions/__tests__/getters.spec.ts +113 -113
  35. package/src/municipal/systemInstructions/__tests__/setters.spec.ts +274 -274
  36. package/src/municipal/systemInstructions/index.ts +7 -7
  37. package/src/municipal/systemInstructions/instructions.getters.ts +57 -57
  38. package/src/municipal/systemInstructions/instructions.setters.ts +119 -119
  39. package/src/municipal/systemInstructions/instructions.types.ts +30 -30
  40. package/src/municipal/tickets/__tests__/tickets.getters.spec.ts +66 -66
  41. package/src/municipal/tickets/index.ts +2 -2
  42. package/src/municipal/tickets/tickets.getters.ts +261 -261
  43. package/src/municipal/tickets/tickets.types.ts +43 -43
  44. package/src/municipal/utils/types.ts +11 -11
  45. package/src/talkpilot/__tests__/db.spec.ts +38 -38
  46. package/src/talkpilot/__tests__/mongodb-client.spec.ts +18 -18
  47. package/src/talkpilot/__tests__/validation.spec.ts +68 -68
  48. package/src/talkpilot/agents/__tests__/agents.getters.spec.ts +29 -29
  49. package/src/talkpilot/agents/agents.getters.ts +34 -34
  50. package/src/talkpilot/agents/agents.types.ts +14 -14
  51. package/src/talkpilot/agents/index.ts +2 -2
  52. package/src/talkpilot/backgroundToolResults/__tests__/backgroundToolResults.getters.spec.ts +147 -147
  53. package/src/talkpilot/backgroundToolResults/backgroundToolResults.getters.ts +65 -65
  54. package/src/talkpilot/backgroundToolResults/backgroundToolResults.types.ts +23 -23
  55. package/src/talkpilot/backgroundToolResults/index.ts +2 -2
  56. package/src/talkpilot/calls/__tests__/callStats.utils.spec.ts +128 -128
  57. package/src/talkpilot/calls/__tests__/calls.spec.ts +252 -252
  58. package/src/talkpilot/calls/calls.getters.ts +446 -248
  59. package/src/talkpilot/calls/calls.types.ts +171 -116
  60. package/src/talkpilot/calls/index.ts +2 -2
  61. package/src/talkpilot/clientAudioBuffers/__tests__/clientAudioBuffer.getters.spec.ts +160 -160
  62. package/src/talkpilot/clientAudioBuffers/clientAudioBuffer.getters.ts +117 -117
  63. package/src/talkpilot/clientAudioBuffers/clientsAudioBuffers.types.ts +25 -25
  64. package/src/talkpilot/clientAudioBuffers/index.ts +2 -2
  65. package/src/talkpilot/clients/clients.getters.ts +16 -16
  66. package/src/talkpilot/clients/clients.types.ts +14 -14
  67. package/src/talkpilot/clients/index.ts +2 -2
  68. package/src/talkpilot/clientsConfig/__tests__/clientsConfig.spec.ts +106 -106
  69. package/src/talkpilot/clientsConfig/clientsConfig.getters.ts +44 -22
  70. package/src/talkpilot/clientsConfig/clientsConfig.types.ts +94 -92
  71. package/src/talkpilot/clientsConfig/index.ts +2 -2
  72. package/src/talkpilot/flows/__tests__/flows.schema.spec.ts +67 -67
  73. package/src/talkpilot/flows/flows.getter.ts +14 -14
  74. package/src/talkpilot/flows/flows.schema.ts +153 -153
  75. package/src/talkpilot/flows/flows.types.ts +184 -184
  76. package/src/talkpilot/flows/index.ts +2 -2
  77. package/src/talkpilot/groups/__tests__/groups.spec.ts +90 -90
  78. package/src/talkpilot/groups/__tests__/phone.utils.spec.ts +32 -32
  79. package/src/talkpilot/groups/groups.getters.ts +30 -30
  80. package/src/talkpilot/groups/groups.types.ts +29 -29
  81. package/src/talkpilot/groups/index.ts +3 -3
  82. package/src/talkpilot/groups/phone.utils.ts +46 -46
  83. package/src/talkpilot/index.ts +29 -29
  84. package/src/talkpilot/leads/index.ts +2 -2
  85. package/src/talkpilot/leads/leads.getter.ts +6 -6
  86. package/src/talkpilot/leads/leads.schema.ts +33 -33
  87. package/src/talkpilot/leads/leads.types.ts +20 -20
  88. package/src/talkpilot/mongodb-client.ts +78 -78
  89. package/src/talkpilot/phone_numbers/__tests__/phone_numbers.spec.ts +247 -247
  90. package/src/talkpilot/phone_numbers/index.ts +2 -2
  91. package/src/talkpilot/phone_numbers/phone_numbers.getter.ts +154 -154
  92. package/src/talkpilot/phone_numbers/phone_numbers.schema.ts +17 -17
  93. package/src/talkpilot/phone_numbers/phone_numbers.types.ts +30 -30
  94. package/src/talkpilot/plans/__tests__/plans.spec.ts +70 -70
  95. package/src/talkpilot/plans/index.ts +2 -2
  96. package/src/talkpilot/plans/plans.getters.ts +132 -132
  97. package/src/talkpilot/plans/plans.types.ts +89 -89
  98. package/src/talkpilot/results/index.ts +7 -7
  99. package/src/talkpilot/results/results.getter.ts +35 -35
  100. package/src/talkpilot/results/results.schema.ts +25 -25
  101. package/src/talkpilot/results/results.types.ts +34 -34
  102. package/src/talkpilot/retry_analyze/__tests__/retryAnalyze.getters.spec.ts +156 -156
  103. package/src/talkpilot/retry_analyze/index.ts +2 -2
  104. package/src/talkpilot/retry_analyze/retryAnalyze.getters.ts +75 -75
  105. package/src/talkpilot/retry_analyze/retryAnalyze.types.ts +13 -13
  106. package/src/talkpilot/sessions/__tests__/sessions.spec.ts +147 -147
  107. package/src/talkpilot/sessions/index.ts +2 -2
  108. package/src/talkpilot/sessions/sessions.getter.ts +92 -92
  109. package/src/talkpilot/sessions/sessions.schema.ts +34 -34
  110. package/src/talkpilot/sessions/sessions.types.ts +30 -30
  111. package/src/talkpilot/subscriptions/__tests__/subscriptions.getters.utils.spec.ts +45 -45
  112. package/src/talkpilot/subscriptions/index.ts +3 -3
  113. package/src/talkpilot/subscriptions/subscriptions.getters.ts +146 -146
  114. package/src/talkpilot/subscriptions/subscriptions.getters.utils.ts +33 -33
  115. package/src/talkpilot/subscriptions/subscriptions.types.ts +66 -66
  116. package/src/talkpilot/utils/__tests__/query.utils.spec.ts +49 -49
  117. package/src/talkpilot/utils/query.utils.ts +21 -21
  118. package/src/test-utils/db-utils.ts +24 -24
  119. package/src/test-utils/factories/index.ts +12 -12
  120. package/src/test-utils/factories/municipal/cities.ts +16 -16
  121. package/src/test-utils/factories/municipal/departmentsSubjects.ts +37 -37
  122. package/src/test-utils/factories/municipal/streets.ts +22 -22
  123. package/src/test-utils/factories/municipal/tickets.ts +39 -39
  124. package/src/test-utils/factories/talkpilot/agents.ts +19 -19
  125. package/src/test-utils/factories/talkpilot/calls.ts +37 -37
  126. package/src/test-utils/factories/talkpilot/clientAudioBuffers.ts +20 -20
  127. package/src/test-utils/factories/talkpilot/clientsConfig.ts +18 -18
  128. package/src/test-utils/factories/talkpilot/flows.ts +33 -33
  129. package/src/test-utils/factories/talkpilot/groups.ts +33 -33
  130. package/src/test-utils/factories/talkpilot/phone_numbers.ts +22 -22
  131. package/src/test-utils/factories/talkpilot/sessions.ts +35 -35
  132. package/src/utils/validation.ts +23 -23
  133. 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
+ });