@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,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 };