@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.
- 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 +2 -1
- package/dist/talkpilot/calls/calls.getters.d.ts.map +1 -1
- package/dist/talkpilot/calls/calls.getters.js +176 -0
- package/dist/talkpilot/calls/calls.getters.js.map +1 -1
- package/dist/talkpilot/calls/calls.types.d.ts +48 -0
- package/dist/talkpilot/calls/calls.types.d.ts.map +1 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts +1 -0
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts.map +1 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.js +13 -0
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.js.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 +446 -248
- package/src/talkpilot/calls/calls.types.ts +171 -115
- 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 +106 -106
- package/src/talkpilot/clientsConfig/clientsConfig.getters.ts +44 -22
- package/src/talkpilot/clientsConfig/clientsConfig.types.ts +94 -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,147 +1,147 @@
|
|
|
1
|
-
import {
|
|
2
|
-
findSessionOfIncomingCall,
|
|
3
|
-
getSessionsCollection,
|
|
4
|
-
} from "../sessions.getter";
|
|
5
|
-
import { getFlowsCollection } from "../../flows";
|
|
6
|
-
import { getPhoneNumbersCollection } from "../../phone_numbers";
|
|
7
|
-
import {
|
|
8
|
-
createFlow,
|
|
9
|
-
createPhoneNumber,
|
|
10
|
-
createSession,
|
|
11
|
-
} from "../../../test-utils/factories";
|
|
12
|
-
import { faker } from "@faker-js/faker";
|
|
13
|
-
import { ObjectId } from "mongodb";
|
|
14
|
-
|
|
15
|
-
describe("sessions", () => {
|
|
16
|
-
describe("findSessionOfIncomingCall()", () => {
|
|
17
|
-
it("returns session by incoming phone number", async () => {
|
|
18
|
-
const to = "+972500000000";
|
|
19
|
-
const from = "+972540000000";
|
|
20
|
-
const flow = createFlow({
|
|
21
|
-
conversationSettings: {
|
|
22
|
-
interruptions: {
|
|
23
|
-
enableInterruptionDetection: true,
|
|
24
|
-
interruptionWindowSeconds: 60,
|
|
25
|
-
interruptionThresholdSeconds: 10,
|
|
26
|
-
interruptionInstruction:
|
|
27
|
-
"I am hearing background noise that makes it hard for me to focus on what you are saying. Please try to move to a quieter place or reduce the noise so I can better assist you.",
|
|
28
|
-
},
|
|
29
|
-
silence: {
|
|
30
|
-
enableSilenceDetection: true,
|
|
31
|
-
firstWarningSilenceSeconds: 5,
|
|
32
|
-
firstWarningInstruction:
|
|
33
|
-
"I have not heard you for a few seconds. If you are still on the line, please say something so we can continue.",
|
|
34
|
-
secondWarningSilenceSeconds: 10,
|
|
35
|
-
secondWarningInstruction:
|
|
36
|
-
"I still have not heard anything from you. If you do not respond in the next few seconds, I will have to end this call.",
|
|
37
|
-
disconnectSilenceSeconds: 30,
|
|
38
|
-
},
|
|
39
|
-
},
|
|
40
|
-
});
|
|
41
|
-
const phoneNumberData = createPhoneNumber({
|
|
42
|
-
phone_number: to,
|
|
43
|
-
flow_id: flow._id,
|
|
44
|
-
});
|
|
45
|
-
const json = { [faker.lorem.word()]: faker.lorem.sentence() };
|
|
46
|
-
|
|
47
|
-
const session = createSession({
|
|
48
|
-
flow_id: flow._id,
|
|
49
|
-
phone_numbers: [{ phoneNumber: from, gender: "male", name: "yakov" }],
|
|
50
|
-
json,
|
|
51
|
-
});
|
|
52
|
-
|
|
53
|
-
await getFlowsCollection().insertOne(flow);
|
|
54
|
-
await getPhoneNumbersCollection().insertOne(phoneNumberData);
|
|
55
|
-
await getSessionsCollection().insertOne(session);
|
|
56
|
-
|
|
57
|
-
const result = await findSessionOfIncomingCall(from, to);
|
|
58
|
-
expect(result).not.toBeNull();
|
|
59
|
-
expect(result?.phone_numbers[0].phoneNumber).toEqual(from);
|
|
60
|
-
expect(result?.flow_id).toEqual(flow._id);
|
|
61
|
-
});
|
|
62
|
-
|
|
63
|
-
it("works with different phone number formats (with/without plus)", async () => {
|
|
64
|
-
const toInDB = "972500000000";
|
|
65
|
-
const fromInDB = "+972540000000";
|
|
66
|
-
const flow = createFlow();
|
|
67
|
-
const phoneNumberData = createPhoneNumber({
|
|
68
|
-
phone_number: toInDB,
|
|
69
|
-
flow_id: flow._id,
|
|
70
|
-
});
|
|
71
|
-
|
|
72
|
-
const session = createSession({
|
|
73
|
-
flow_id: flow._id,
|
|
74
|
-
phone_numbers: [
|
|
75
|
-
{ phoneNumber: fromInDB, gender: "female", name: "sara" },
|
|
76
|
-
],
|
|
77
|
-
});
|
|
78
|
-
|
|
79
|
-
await getFlowsCollection().insertOne(flow);
|
|
80
|
-
await getPhoneNumbersCollection().insertOne(phoneNumberData);
|
|
81
|
-
await getSessionsCollection().insertOne(session);
|
|
82
|
-
|
|
83
|
-
// Search with from WITH plus (matches + in DB), to WITH plus (matches without + in DB)
|
|
84
|
-
const result = await findSessionOfIncomingCall(
|
|
85
|
-
"+972540000000",
|
|
86
|
-
"+972500000000",
|
|
87
|
-
);
|
|
88
|
-
expect(result).not.toBeNull();
|
|
89
|
-
expect(result?.phone_numbers[0].phoneNumber).toEqual(fromInDB);
|
|
90
|
-
});
|
|
91
|
-
|
|
92
|
-
it("works when flow_id is string in phone_numbers and ObjectId in session", async () => {
|
|
93
|
-
const to = "+972501111111";
|
|
94
|
-
const from = "+972541111111";
|
|
95
|
-
const flowId = new ObjectId();
|
|
96
|
-
|
|
97
|
-
// Store flow_id as string in phone_numbers
|
|
98
|
-
const phoneNumberData = createPhoneNumber({
|
|
99
|
-
phone_number: to,
|
|
100
|
-
flow_id: flowId.toHexString() as any,
|
|
101
|
-
});
|
|
102
|
-
|
|
103
|
-
const session = createSession({
|
|
104
|
-
flow_id: flowId, // stored as ObjectId
|
|
105
|
-
phone_numbers: [{ phoneNumber: from, gender: "male", name: "isaac" }],
|
|
106
|
-
});
|
|
107
|
-
|
|
108
|
-
await getPhoneNumbersCollection().insertOne(phoneNumberData);
|
|
109
|
-
await getSessionsCollection().insertOne(session);
|
|
110
|
-
|
|
111
|
-
const result = await findSessionOfIncomingCall(from, to);
|
|
112
|
-
expect(result).not.toBeNull();
|
|
113
|
-
expect(result?.flow_id?.toString()).toEqual(flowId.toHexString());
|
|
114
|
-
});
|
|
115
|
-
|
|
116
|
-
it("works when flow_id is string in session", async () => {
|
|
117
|
-
const to = "+972502222222";
|
|
118
|
-
const from = "+972542222222";
|
|
119
|
-
const flowId = new ObjectId();
|
|
120
|
-
|
|
121
|
-
const phoneNumberData = createPhoneNumber({
|
|
122
|
-
phone_number: to,
|
|
123
|
-
flow_id: flowId,
|
|
124
|
-
});
|
|
125
|
-
|
|
126
|
-
const session = createSession({
|
|
127
|
-
flow_id: flowId.toHexString(), // stored as string
|
|
128
|
-
phone_numbers: [{ phoneNumber: from, gender: "female", name: "rivka" }],
|
|
129
|
-
});
|
|
130
|
-
|
|
131
|
-
await getPhoneNumbersCollection().insertOne(phoneNumberData);
|
|
132
|
-
await getSessionsCollection().insertOne(session);
|
|
133
|
-
|
|
134
|
-
const result = await findSessionOfIncomingCall(from, to);
|
|
135
|
-
expect(result).not.toBeNull();
|
|
136
|
-
expect(result?.flow_id).toEqual(flowId.toHexString());
|
|
137
|
-
});
|
|
138
|
-
|
|
139
|
-
it("returns null if receiver phone number is not found", async () => {
|
|
140
|
-
const result = await findSessionOfIncomingCall(
|
|
141
|
-
"any-from",
|
|
142
|
-
"non-existent-to",
|
|
143
|
-
);
|
|
144
|
-
expect(result).toBeNull();
|
|
145
|
-
});
|
|
146
|
-
});
|
|
147
|
-
});
|
|
1
|
+
import {
|
|
2
|
+
findSessionOfIncomingCall,
|
|
3
|
+
getSessionsCollection,
|
|
4
|
+
} from "../sessions.getter";
|
|
5
|
+
import { getFlowsCollection } from "../../flows";
|
|
6
|
+
import { getPhoneNumbersCollection } from "../../phone_numbers";
|
|
7
|
+
import {
|
|
8
|
+
createFlow,
|
|
9
|
+
createPhoneNumber,
|
|
10
|
+
createSession,
|
|
11
|
+
} from "../../../test-utils/factories";
|
|
12
|
+
import { faker } from "@faker-js/faker";
|
|
13
|
+
import { ObjectId } from "mongodb";
|
|
14
|
+
|
|
15
|
+
describe("sessions", () => {
|
|
16
|
+
describe("findSessionOfIncomingCall()", () => {
|
|
17
|
+
it("returns session by incoming phone number", async () => {
|
|
18
|
+
const to = "+972500000000";
|
|
19
|
+
const from = "+972540000000";
|
|
20
|
+
const flow = createFlow({
|
|
21
|
+
conversationSettings: {
|
|
22
|
+
interruptions: {
|
|
23
|
+
enableInterruptionDetection: true,
|
|
24
|
+
interruptionWindowSeconds: 60,
|
|
25
|
+
interruptionThresholdSeconds: 10,
|
|
26
|
+
interruptionInstruction:
|
|
27
|
+
"I am hearing background noise that makes it hard for me to focus on what you are saying. Please try to move to a quieter place or reduce the noise so I can better assist you.",
|
|
28
|
+
},
|
|
29
|
+
silence: {
|
|
30
|
+
enableSilenceDetection: true,
|
|
31
|
+
firstWarningSilenceSeconds: 5,
|
|
32
|
+
firstWarningInstruction:
|
|
33
|
+
"I have not heard you for a few seconds. If you are still on the line, please say something so we can continue.",
|
|
34
|
+
secondWarningSilenceSeconds: 10,
|
|
35
|
+
secondWarningInstruction:
|
|
36
|
+
"I still have not heard anything from you. If you do not respond in the next few seconds, I will have to end this call.",
|
|
37
|
+
disconnectSilenceSeconds: 30,
|
|
38
|
+
},
|
|
39
|
+
},
|
|
40
|
+
});
|
|
41
|
+
const phoneNumberData = createPhoneNumber({
|
|
42
|
+
phone_number: to,
|
|
43
|
+
flow_id: flow._id,
|
|
44
|
+
});
|
|
45
|
+
const json = { [faker.lorem.word()]: faker.lorem.sentence() };
|
|
46
|
+
|
|
47
|
+
const session = createSession({
|
|
48
|
+
flow_id: flow._id,
|
|
49
|
+
phone_numbers: [{ phoneNumber: from, gender: "male", name: "yakov" }],
|
|
50
|
+
json,
|
|
51
|
+
});
|
|
52
|
+
|
|
53
|
+
await getFlowsCollection().insertOne(flow);
|
|
54
|
+
await getPhoneNumbersCollection().insertOne(phoneNumberData);
|
|
55
|
+
await getSessionsCollection().insertOne(session);
|
|
56
|
+
|
|
57
|
+
const result = await findSessionOfIncomingCall(from, to);
|
|
58
|
+
expect(result).not.toBeNull();
|
|
59
|
+
expect(result?.phone_numbers[0].phoneNumber).toEqual(from);
|
|
60
|
+
expect(result?.flow_id).toEqual(flow._id);
|
|
61
|
+
});
|
|
62
|
+
|
|
63
|
+
it("works with different phone number formats (with/without plus)", async () => {
|
|
64
|
+
const toInDB = "972500000000";
|
|
65
|
+
const fromInDB = "+972540000000";
|
|
66
|
+
const flow = createFlow();
|
|
67
|
+
const phoneNumberData = createPhoneNumber({
|
|
68
|
+
phone_number: toInDB,
|
|
69
|
+
flow_id: flow._id,
|
|
70
|
+
});
|
|
71
|
+
|
|
72
|
+
const session = createSession({
|
|
73
|
+
flow_id: flow._id,
|
|
74
|
+
phone_numbers: [
|
|
75
|
+
{ phoneNumber: fromInDB, gender: "female", name: "sara" },
|
|
76
|
+
],
|
|
77
|
+
});
|
|
78
|
+
|
|
79
|
+
await getFlowsCollection().insertOne(flow);
|
|
80
|
+
await getPhoneNumbersCollection().insertOne(phoneNumberData);
|
|
81
|
+
await getSessionsCollection().insertOne(session);
|
|
82
|
+
|
|
83
|
+
// Search with from WITH plus (matches + in DB), to WITH plus (matches without + in DB)
|
|
84
|
+
const result = await findSessionOfIncomingCall(
|
|
85
|
+
"+972540000000",
|
|
86
|
+
"+972500000000",
|
|
87
|
+
);
|
|
88
|
+
expect(result).not.toBeNull();
|
|
89
|
+
expect(result?.phone_numbers[0].phoneNumber).toEqual(fromInDB);
|
|
90
|
+
});
|
|
91
|
+
|
|
92
|
+
it("works when flow_id is string in phone_numbers and ObjectId in session", async () => {
|
|
93
|
+
const to = "+972501111111";
|
|
94
|
+
const from = "+972541111111";
|
|
95
|
+
const flowId = new ObjectId();
|
|
96
|
+
|
|
97
|
+
// Store flow_id as string in phone_numbers
|
|
98
|
+
const phoneNumberData = createPhoneNumber({
|
|
99
|
+
phone_number: to,
|
|
100
|
+
flow_id: flowId.toHexString() as any,
|
|
101
|
+
});
|
|
102
|
+
|
|
103
|
+
const session = createSession({
|
|
104
|
+
flow_id: flowId, // stored as ObjectId
|
|
105
|
+
phone_numbers: [{ phoneNumber: from, gender: "male", name: "isaac" }],
|
|
106
|
+
});
|
|
107
|
+
|
|
108
|
+
await getPhoneNumbersCollection().insertOne(phoneNumberData);
|
|
109
|
+
await getSessionsCollection().insertOne(session);
|
|
110
|
+
|
|
111
|
+
const result = await findSessionOfIncomingCall(from, to);
|
|
112
|
+
expect(result).not.toBeNull();
|
|
113
|
+
expect(result?.flow_id?.toString()).toEqual(flowId.toHexString());
|
|
114
|
+
});
|
|
115
|
+
|
|
116
|
+
it("works when flow_id is string in session", async () => {
|
|
117
|
+
const to = "+972502222222";
|
|
118
|
+
const from = "+972542222222";
|
|
119
|
+
const flowId = new ObjectId();
|
|
120
|
+
|
|
121
|
+
const phoneNumberData = createPhoneNumber({
|
|
122
|
+
phone_number: to,
|
|
123
|
+
flow_id: flowId,
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
const session = createSession({
|
|
127
|
+
flow_id: flowId.toHexString(), // stored as string
|
|
128
|
+
phone_numbers: [{ phoneNumber: from, gender: "female", name: "rivka" }],
|
|
129
|
+
});
|
|
130
|
+
|
|
131
|
+
await getPhoneNumbersCollection().insertOne(phoneNumberData);
|
|
132
|
+
await getSessionsCollection().insertOne(session);
|
|
133
|
+
|
|
134
|
+
const result = await findSessionOfIncomingCall(from, to);
|
|
135
|
+
expect(result).not.toBeNull();
|
|
136
|
+
expect(result?.flow_id).toEqual(flowId.toHexString());
|
|
137
|
+
});
|
|
138
|
+
|
|
139
|
+
it("returns null if receiver phone number is not found", async () => {
|
|
140
|
+
const result = await findSessionOfIncomingCall(
|
|
141
|
+
"any-from",
|
|
142
|
+
"non-existent-to",
|
|
143
|
+
);
|
|
144
|
+
expect(result).toBeNull();
|
|
145
|
+
});
|
|
146
|
+
});
|
|
147
|
+
});
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
export * from "./sessions.getter";
|
|
2
|
-
export * from "./sessions.types";
|
|
1
|
+
export * from "./sessions.getter";
|
|
2
|
+
export * from "./sessions.types";
|
|
@@ -1,92 +1,92 @@
|
|
|
1
|
-
import { ObjectId } from "mongodb";
|
|
2
|
-
import type { Collection, WithId } from "mongodb";
|
|
3
|
-
import { getDb, getPhoneNumbersCollection } from "../index";
|
|
4
|
-
import type { Session } from "./sessions.types";
|
|
5
|
-
|
|
6
|
-
export const getSessionsCollection = (): Collection<Session> =>
|
|
7
|
-
getDb().collection<Session>("sessions");
|
|
8
|
-
|
|
9
|
-
const buildPhoneCandidates = (phone: string): string[] => {
|
|
10
|
-
const raw = (phone ?? "").trim();
|
|
11
|
-
if (!raw) return [];
|
|
12
|
-
|
|
13
|
-
const noPlus = raw.startsWith("+") ? raw.slice(1) : raw;
|
|
14
|
-
const digitsOnly = raw.replace(/\D/g, "");
|
|
15
|
-
const digitsOnlyNoPlus = noPlus.replace(/\D/g, "");
|
|
16
|
-
|
|
17
|
-
return Array.from(
|
|
18
|
-
new Set([raw, noPlus, digitsOnly, digitsOnlyNoPlus].filter(Boolean)),
|
|
19
|
-
);
|
|
20
|
-
};
|
|
21
|
-
|
|
22
|
-
const buildFlowIdCandidates = (flowId: unknown): Array<string | ObjectId> => {
|
|
23
|
-
if (!flowId) return [];
|
|
24
|
-
|
|
25
|
-
// Some sessions store flow_id as string, others as ObjectId.
|
|
26
|
-
if (flowId instanceof ObjectId) {
|
|
27
|
-
return [flowId, flowId.toString()];
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
const asString = String(flowId);
|
|
31
|
-
if (!asString) return [];
|
|
32
|
-
|
|
33
|
-
if (ObjectId.isValid(asString)) {
|
|
34
|
-
return [new ObjectId(asString), asString];
|
|
35
|
-
}
|
|
36
|
-
return [asString];
|
|
37
|
-
};
|
|
38
|
-
|
|
39
|
-
export const findSessionById = (
|
|
40
|
-
sessionId: ObjectId,
|
|
41
|
-
clientId: string,
|
|
42
|
-
): Promise<WithId<Session> | null> => {
|
|
43
|
-
return getSessionsCollection().findOne({
|
|
44
|
-
_id: sessionId,
|
|
45
|
-
clientId: clientId,
|
|
46
|
-
});
|
|
47
|
-
};
|
|
48
|
-
|
|
49
|
-
export const findSessionOfIncomingCall = async (from: string, to: string) => {
|
|
50
|
-
const toCandidates = buildPhoneCandidates(to);
|
|
51
|
-
const fromCandidates = buildPhoneCandidates(from);
|
|
52
|
-
|
|
53
|
-
const receiverPhoneData = await getPhoneNumbersCollection().findOne({
|
|
54
|
-
phone_number: { $in: toCandidates },
|
|
55
|
-
});
|
|
56
|
-
if (!receiverPhoneData) {
|
|
57
|
-
return null;
|
|
58
|
-
}
|
|
59
|
-
|
|
60
|
-
const flowIdCandidates = buildFlowIdCandidates(receiverPhoneData.flow_id);
|
|
61
|
-
|
|
62
|
-
const [session] = await getSessionsCollection()
|
|
63
|
-
.aggregate<WithId<Session>>([
|
|
64
|
-
{
|
|
65
|
-
$match: {
|
|
66
|
-
flow_id: { $in: flowIdCandidates },
|
|
67
|
-
phone_numbers: {
|
|
68
|
-
$elemMatch: { phoneNumber: { $in: fromCandidates } },
|
|
69
|
-
},
|
|
70
|
-
},
|
|
71
|
-
},
|
|
72
|
-
{
|
|
73
|
-
$addFields: {
|
|
74
|
-
phone_numbers: {
|
|
75
|
-
$filter: {
|
|
76
|
-
input: "$phone_numbers",
|
|
77
|
-
as: "pn",
|
|
78
|
-
cond: { $in: ["$$pn.phoneNumber", fromCandidates] },
|
|
79
|
-
},
|
|
80
|
-
},
|
|
81
|
-
},
|
|
82
|
-
},
|
|
83
|
-
{ $limit: 1 },
|
|
84
|
-
])
|
|
85
|
-
.toArray();
|
|
86
|
-
|
|
87
|
-
if (!session) {
|
|
88
|
-
console.info("No session found for incoming call");
|
|
89
|
-
return null;
|
|
90
|
-
}
|
|
91
|
-
return session;
|
|
92
|
-
};
|
|
1
|
+
import { ObjectId } from "mongodb";
|
|
2
|
+
import type { Collection, WithId } from "mongodb";
|
|
3
|
+
import { getDb, getPhoneNumbersCollection } from "../index";
|
|
4
|
+
import type { Session } from "./sessions.types";
|
|
5
|
+
|
|
6
|
+
export const getSessionsCollection = (): Collection<Session> =>
|
|
7
|
+
getDb().collection<Session>("sessions");
|
|
8
|
+
|
|
9
|
+
const buildPhoneCandidates = (phone: string): string[] => {
|
|
10
|
+
const raw = (phone ?? "").trim();
|
|
11
|
+
if (!raw) return [];
|
|
12
|
+
|
|
13
|
+
const noPlus = raw.startsWith("+") ? raw.slice(1) : raw;
|
|
14
|
+
const digitsOnly = raw.replace(/\D/g, "");
|
|
15
|
+
const digitsOnlyNoPlus = noPlus.replace(/\D/g, "");
|
|
16
|
+
|
|
17
|
+
return Array.from(
|
|
18
|
+
new Set([raw, noPlus, digitsOnly, digitsOnlyNoPlus].filter(Boolean)),
|
|
19
|
+
);
|
|
20
|
+
};
|
|
21
|
+
|
|
22
|
+
const buildFlowIdCandidates = (flowId: unknown): Array<string | ObjectId> => {
|
|
23
|
+
if (!flowId) return [];
|
|
24
|
+
|
|
25
|
+
// Some sessions store flow_id as string, others as ObjectId.
|
|
26
|
+
if (flowId instanceof ObjectId) {
|
|
27
|
+
return [flowId, flowId.toString()];
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
const asString = String(flowId);
|
|
31
|
+
if (!asString) return [];
|
|
32
|
+
|
|
33
|
+
if (ObjectId.isValid(asString)) {
|
|
34
|
+
return [new ObjectId(asString), asString];
|
|
35
|
+
}
|
|
36
|
+
return [asString];
|
|
37
|
+
};
|
|
38
|
+
|
|
39
|
+
export const findSessionById = (
|
|
40
|
+
sessionId: ObjectId,
|
|
41
|
+
clientId: string,
|
|
42
|
+
): Promise<WithId<Session> | null> => {
|
|
43
|
+
return getSessionsCollection().findOne({
|
|
44
|
+
_id: sessionId,
|
|
45
|
+
clientId: clientId,
|
|
46
|
+
});
|
|
47
|
+
};
|
|
48
|
+
|
|
49
|
+
export const findSessionOfIncomingCall = async (from: string, to: string) => {
|
|
50
|
+
const toCandidates = buildPhoneCandidates(to);
|
|
51
|
+
const fromCandidates = buildPhoneCandidates(from);
|
|
52
|
+
|
|
53
|
+
const receiverPhoneData = await getPhoneNumbersCollection().findOne({
|
|
54
|
+
phone_number: { $in: toCandidates },
|
|
55
|
+
});
|
|
56
|
+
if (!receiverPhoneData) {
|
|
57
|
+
return null;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
const flowIdCandidates = buildFlowIdCandidates(receiverPhoneData.flow_id);
|
|
61
|
+
|
|
62
|
+
const [session] = await getSessionsCollection()
|
|
63
|
+
.aggregate<WithId<Session>>([
|
|
64
|
+
{
|
|
65
|
+
$match: {
|
|
66
|
+
flow_id: { $in: flowIdCandidates },
|
|
67
|
+
phone_numbers: {
|
|
68
|
+
$elemMatch: { phoneNumber: { $in: fromCandidates } },
|
|
69
|
+
},
|
|
70
|
+
},
|
|
71
|
+
},
|
|
72
|
+
{
|
|
73
|
+
$addFields: {
|
|
74
|
+
phone_numbers: {
|
|
75
|
+
$filter: {
|
|
76
|
+
input: "$phone_numbers",
|
|
77
|
+
as: "pn",
|
|
78
|
+
cond: { $in: ["$$pn.phoneNumber", fromCandidates] },
|
|
79
|
+
},
|
|
80
|
+
},
|
|
81
|
+
},
|
|
82
|
+
},
|
|
83
|
+
{ $limit: 1 },
|
|
84
|
+
])
|
|
85
|
+
.toArray();
|
|
86
|
+
|
|
87
|
+
if (!session) {
|
|
88
|
+
console.info("No session found for incoming call");
|
|
89
|
+
return null;
|
|
90
|
+
}
|
|
91
|
+
return session;
|
|
92
|
+
};
|
|
@@ -1,34 +1,34 @@
|
|
|
1
|
-
export const sessionMongoSchema = {
|
|
2
|
-
bsonType: "object",
|
|
3
|
-
required: ["session_name"],
|
|
4
|
-
properties: {
|
|
5
|
-
_id: { bsonType: "objectId" },
|
|
6
|
-
session_name: { bsonType: "string" },
|
|
7
|
-
company_name: { bsonType: ["string", "null"] },
|
|
8
|
-
flow_id: { bsonType: ["objectId", "string", "null"] },
|
|
9
|
-
metaKeys: {
|
|
10
|
-
bsonType: ["array", "null"],
|
|
11
|
-
items: {
|
|
12
|
-
bsonType: "object",
|
|
13
|
-
required: ["key"],
|
|
14
|
-
properties: {
|
|
15
|
-
key: { bsonType: "string" },
|
|
16
|
-
value: { bsonType: "string" },
|
|
17
|
-
},
|
|
18
|
-
},
|
|
19
|
-
},
|
|
20
|
-
phone_numbers: {
|
|
21
|
-
bsonType: "array",
|
|
22
|
-
items: {
|
|
23
|
-
bsonType: "object",
|
|
24
|
-
required: ["name", "phoneNumber"],
|
|
25
|
-
properties: {
|
|
26
|
-
name: { bsonType: "string" },
|
|
27
|
-
phoneNumber: { bsonType: "string" },
|
|
28
|
-
},
|
|
29
|
-
},
|
|
30
|
-
},
|
|
31
|
-
voice: { bsonType: ["string", "null"] },
|
|
32
|
-
},
|
|
33
|
-
additionalProperties: false,
|
|
34
|
-
} as const;
|
|
1
|
+
export const sessionMongoSchema = {
|
|
2
|
+
bsonType: "object",
|
|
3
|
+
required: ["session_name"],
|
|
4
|
+
properties: {
|
|
5
|
+
_id: { bsonType: "objectId" },
|
|
6
|
+
session_name: { bsonType: "string" },
|
|
7
|
+
company_name: { bsonType: ["string", "null"] },
|
|
8
|
+
flow_id: { bsonType: ["objectId", "string", "null"] },
|
|
9
|
+
metaKeys: {
|
|
10
|
+
bsonType: ["array", "null"],
|
|
11
|
+
items: {
|
|
12
|
+
bsonType: "object",
|
|
13
|
+
required: ["key"],
|
|
14
|
+
properties: {
|
|
15
|
+
key: { bsonType: "string" },
|
|
16
|
+
value: { bsonType: "string" },
|
|
17
|
+
},
|
|
18
|
+
},
|
|
19
|
+
},
|
|
20
|
+
phone_numbers: {
|
|
21
|
+
bsonType: "array",
|
|
22
|
+
items: {
|
|
23
|
+
bsonType: "object",
|
|
24
|
+
required: ["name", "phoneNumber"],
|
|
25
|
+
properties: {
|
|
26
|
+
name: { bsonType: "string" },
|
|
27
|
+
phoneNumber: { bsonType: "string" },
|
|
28
|
+
},
|
|
29
|
+
},
|
|
30
|
+
},
|
|
31
|
+
voice: { bsonType: ["string", "null"] },
|
|
32
|
+
},
|
|
33
|
+
additionalProperties: false,
|
|
34
|
+
} as const;
|
|
@@ -1,30 +1,30 @@
|
|
|
1
|
-
import { ObjectId } from "mongodb";
|
|
2
|
-
|
|
3
|
-
export type Session = {
|
|
4
|
-
_id?: ObjectId;
|
|
5
|
-
|
|
6
|
-
session_name: string;
|
|
7
|
-
company_name?: string | null;
|
|
8
|
-
flow_id?: ObjectId | string | null;
|
|
9
|
-
clientId: string;
|
|
10
|
-
|
|
11
|
-
metaKeys?: Array<{
|
|
12
|
-
key: string;
|
|
13
|
-
value?: string;
|
|
14
|
-
}> | null;
|
|
15
|
-
|
|
16
|
-
phone_numbers: Array<{
|
|
17
|
-
name: string;
|
|
18
|
-
phoneNumber: string;
|
|
19
|
-
gender: "male" | "female";
|
|
20
|
-
}>;
|
|
21
|
-
|
|
22
|
-
voice?: string | null;
|
|
23
|
-
json: Record<string, any> | null;
|
|
24
|
-
|
|
25
|
-
// Timestamps
|
|
26
|
-
createdAt?: Date;
|
|
27
|
-
updatedAt?: Date;
|
|
28
|
-
};
|
|
29
|
-
|
|
30
|
-
export type SessionDoc = Session & { _id: ObjectId };
|
|
1
|
+
import { ObjectId } from "mongodb";
|
|
2
|
+
|
|
3
|
+
export type Session = {
|
|
4
|
+
_id?: ObjectId;
|
|
5
|
+
|
|
6
|
+
session_name: string;
|
|
7
|
+
company_name?: string | null;
|
|
8
|
+
flow_id?: ObjectId | string | null;
|
|
9
|
+
clientId: string;
|
|
10
|
+
|
|
11
|
+
metaKeys?: Array<{
|
|
12
|
+
key: string;
|
|
13
|
+
value?: string;
|
|
14
|
+
}> | null;
|
|
15
|
+
|
|
16
|
+
phone_numbers: Array<{
|
|
17
|
+
name: string;
|
|
18
|
+
phoneNumber: string;
|
|
19
|
+
gender: "male" | "female";
|
|
20
|
+
}>;
|
|
21
|
+
|
|
22
|
+
voice?: string | null;
|
|
23
|
+
json: Record<string, any> | null;
|
|
24
|
+
|
|
25
|
+
// Timestamps
|
|
26
|
+
createdAt?: Date;
|
|
27
|
+
updatedAt?: Date;
|
|
28
|
+
};
|
|
29
|
+
|
|
30
|
+
export type SessionDoc = Session & { _id: ObjectId };
|