@talkpilot/core-db 1.1.18 → 1.2.0

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