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