@talkpilot/core-db 1.2.0 → 1.2.1
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.
- package/.cursor/rules/development.mdc +65 -65
- package/DEVELOPMENT.md +98 -98
- package/README.md +160 -160
- package/dist/talkpilot/calls/calls.getters.d.ts +1 -2
- package/dist/talkpilot/calls/calls.getters.d.ts.map +1 -1
- package/dist/talkpilot/calls/calls.getters.js +0 -176
- package/dist/talkpilot/calls/calls.getters.js.map +1 -1
- package/dist/talkpilot/calls/calls.types.d.ts +0 -48
- package/dist/talkpilot/calls/calls.types.d.ts.map +1 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts +0 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts.map +1 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.js +0 -13
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.js.map +1 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.types.d.ts +16 -8
- package/dist/talkpilot/clientsConfig/clientsConfig.types.d.ts.map +1 -1
- package/jest.config.js +19 -19
- package/package.json +45 -45
- package/src/__tests__/setup.ts +20 -20
- package/src/connection.ts +42 -42
- package/src/index.ts +16 -16
- package/src/municipal/__tests__/validation.spec.ts +62 -62
- package/src/municipal/cities/cities.getters.ts +50 -50
- package/src/municipal/cities/cities.types.ts +11 -11
- package/src/municipal/cities/index.ts +2 -2
- package/src/municipal/departmentsSubjects/departmentsSubjects.getters.ts +282 -282
- package/src/municipal/departmentsSubjects/departmentsSubjects.types.ts +72 -72
- package/src/municipal/departmentsSubjects/index.ts +9 -9
- package/src/municipal/index.ts +21 -21
- package/src/municipal/mongodb-client.ts +61 -61
- package/src/municipal/streets/index.ts +2 -2
- package/src/municipal/streets/streets.getters.ts +125 -125
- package/src/municipal/streets/streets.types.ts +18 -18
- package/src/municipal/systemInstructions/__tests__/getters.spec.ts +113 -113
- package/src/municipal/systemInstructions/__tests__/setters.spec.ts +274 -274
- package/src/municipal/systemInstructions/index.ts +7 -7
- package/src/municipal/systemInstructions/instructions.getters.ts +57 -57
- package/src/municipal/systemInstructions/instructions.setters.ts +119 -119
- package/src/municipal/systemInstructions/instructions.types.ts +30 -30
- package/src/municipal/tickets/__tests__/tickets.getters.spec.ts +66 -66
- package/src/municipal/tickets/index.ts +2 -2
- package/src/municipal/tickets/tickets.getters.ts +261 -261
- package/src/municipal/tickets/tickets.types.ts +43 -43
- package/src/municipal/utils/types.ts +11 -11
- package/src/talkpilot/__tests__/db.spec.ts +38 -38
- package/src/talkpilot/__tests__/mongodb-client.spec.ts +18 -18
- package/src/talkpilot/__tests__/validation.spec.ts +68 -68
- package/src/talkpilot/agents/__tests__/agents.getters.spec.ts +29 -29
- package/src/talkpilot/agents/agents.getters.ts +34 -34
- package/src/talkpilot/agents/agents.types.ts +14 -14
- package/src/talkpilot/agents/index.ts +2 -2
- package/src/talkpilot/backgroundToolResults/__tests__/backgroundToolResults.getters.spec.ts +147 -147
- package/src/talkpilot/backgroundToolResults/backgroundToolResults.getters.ts +65 -65
- package/src/talkpilot/backgroundToolResults/backgroundToolResults.types.ts +23 -23
- package/src/talkpilot/backgroundToolResults/index.ts +2 -2
- package/src/talkpilot/calls/__tests__/callStats.utils.spec.ts +128 -128
- package/src/talkpilot/calls/__tests__/calls.spec.ts +252 -252
- package/src/talkpilot/calls/calls.getters.ts +248 -446
- package/src/talkpilot/calls/calls.types.ts +115 -171
- package/src/talkpilot/calls/index.ts +2 -2
- package/src/talkpilot/clientAudioBuffers/__tests__/clientAudioBuffer.getters.spec.ts +160 -160
- package/src/talkpilot/clientAudioBuffers/clientAudioBuffer.getters.ts +117 -117
- package/src/talkpilot/clientAudioBuffers/clientsAudioBuffers.types.ts +25 -25
- package/src/talkpilot/clientAudioBuffers/index.ts +2 -2
- package/src/talkpilot/clients/clients.getters.ts +16 -16
- package/src/talkpilot/clients/clients.types.ts +14 -14
- package/src/talkpilot/clients/index.ts +2 -2
- package/src/talkpilot/clientsConfig/__tests__/clientsConfig.spec.ts +187 -106
- package/src/talkpilot/clientsConfig/clientsConfig.getters.ts +22 -44
- package/src/talkpilot/clientsConfig/clientsConfig.types.ts +119 -94
- package/src/talkpilot/clientsConfig/index.ts +2 -2
- package/src/talkpilot/flows/__tests__/flows.schema.spec.ts +67 -67
- package/src/talkpilot/flows/flows.getter.ts +14 -14
- package/src/talkpilot/flows/flows.schema.ts +153 -153
- package/src/talkpilot/flows/flows.types.ts +184 -184
- package/src/talkpilot/flows/index.ts +2 -2
- package/src/talkpilot/groups/__tests__/groups.spec.ts +90 -90
- package/src/talkpilot/groups/__tests__/phone.utils.spec.ts +32 -32
- package/src/talkpilot/groups/groups.getters.ts +30 -30
- package/src/talkpilot/groups/groups.types.ts +29 -29
- package/src/talkpilot/groups/index.ts +3 -3
- package/src/talkpilot/groups/phone.utils.ts +46 -46
- package/src/talkpilot/index.ts +29 -29
- package/src/talkpilot/leads/index.ts +2 -2
- package/src/talkpilot/leads/leads.getter.ts +6 -6
- package/src/talkpilot/leads/leads.schema.ts +33 -33
- package/src/talkpilot/leads/leads.types.ts +20 -20
- package/src/talkpilot/mongodb-client.ts +78 -78
- package/src/talkpilot/phone_numbers/__tests__/phone_numbers.spec.ts +247 -247
- package/src/talkpilot/phone_numbers/index.ts +2 -2
- package/src/talkpilot/phone_numbers/phone_numbers.getter.ts +154 -154
- package/src/talkpilot/phone_numbers/phone_numbers.schema.ts +17 -17
- package/src/talkpilot/phone_numbers/phone_numbers.types.ts +30 -30
- package/src/talkpilot/plans/__tests__/plans.spec.ts +70 -70
- package/src/talkpilot/plans/index.ts +2 -2
- package/src/talkpilot/plans/plans.getters.ts +132 -132
- package/src/talkpilot/plans/plans.types.ts +89 -89
- package/src/talkpilot/results/index.ts +7 -7
- package/src/talkpilot/results/results.getter.ts +35 -35
- package/src/talkpilot/results/results.schema.ts +25 -25
- package/src/talkpilot/results/results.types.ts +34 -34
- package/src/talkpilot/retry_analyze/__tests__/retryAnalyze.getters.spec.ts +156 -156
- package/src/talkpilot/retry_analyze/index.ts +2 -2
- package/src/talkpilot/retry_analyze/retryAnalyze.getters.ts +75 -75
- package/src/talkpilot/retry_analyze/retryAnalyze.types.ts +13 -13
- package/src/talkpilot/sessions/__tests__/sessions.spec.ts +147 -147
- package/src/talkpilot/sessions/index.ts +2 -2
- package/src/talkpilot/sessions/sessions.getter.ts +92 -92
- package/src/talkpilot/sessions/sessions.schema.ts +34 -34
- package/src/talkpilot/sessions/sessions.types.ts +30 -30
- package/src/talkpilot/subscriptions/__tests__/subscriptions.getters.utils.spec.ts +45 -45
- package/src/talkpilot/subscriptions/index.ts +3 -3
- package/src/talkpilot/subscriptions/subscriptions.getters.ts +146 -146
- package/src/talkpilot/subscriptions/subscriptions.getters.utils.ts +33 -33
- package/src/talkpilot/subscriptions/subscriptions.types.ts +66 -66
- package/src/talkpilot/utils/__tests__/query.utils.spec.ts +49 -49
- package/src/talkpilot/utils/query.utils.ts +21 -21
- package/src/test-utils/db-utils.ts +24 -24
- package/src/test-utils/factories/index.ts +12 -12
- package/src/test-utils/factories/municipal/cities.ts +16 -16
- package/src/test-utils/factories/municipal/departmentsSubjects.ts +37 -37
- package/src/test-utils/factories/municipal/streets.ts +22 -22
- package/src/test-utils/factories/municipal/tickets.ts +39 -39
- package/src/test-utils/factories/talkpilot/agents.ts +19 -19
- package/src/test-utils/factories/talkpilot/calls.ts +37 -37
- package/src/test-utils/factories/talkpilot/clientAudioBuffers.ts +20 -20
- package/src/test-utils/factories/talkpilot/clientsConfig.ts +18 -18
- package/src/test-utils/factories/talkpilot/flows.ts +33 -33
- package/src/test-utils/factories/talkpilot/groups.ts +33 -33
- package/src/test-utils/factories/talkpilot/phone_numbers.ts +22 -22
- package/src/test-utils/factories/talkpilot/sessions.ts +35 -35
- package/src/utils/validation.ts +23 -23
- 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
|
+
});
|