@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,78 +1,78 @@
1
- import { MongoClient, Db } from "mongodb";
2
- import { setDb } from "./index";
3
- import { validateConfig, validateMongoUri } from "../utils/validation";
4
-
5
- class MongoDBClient {
6
- private client: MongoClient | null = null;
7
- private db: Db | null = null;
8
-
9
- private extractDatabaseName(uri: string): string | null {
10
- try {
11
- const url = new URL(uri);
12
- const dbName = url.pathname.slice(1);
13
- return dbName || null;
14
- } catch {
15
- return null;
16
- }
17
- }
18
-
19
- async connect(uri?: string, databaseName?: string): Promise<void> {
20
- if (this.client) {
21
- return;
22
- }
23
-
24
- const mongodbUri = uri || process.env.MONGO_URI || process.env.MONGODB_URI;
25
- validateConfig("MONGO_URI", mongodbUri);
26
- validateMongoUri(mongodbUri!);
27
-
28
- try {
29
- this.client = new MongoClient(mongodbUri!);
30
- await this.client.connect();
31
-
32
- const dbNameFromUri = this.extractDatabaseName(mongodbUri!);
33
- const dbName =
34
- databaseName || process.env.TALKPILOT_DB_NAME || dbNameFromUri;
35
-
36
- if (!dbName) {
37
- throw new Error(
38
- "[core-db] Database name not specified. Please pass databaseName to connect(), " +
39
- "add it to the URI, or set TALKPILOT_DB_NAME.",
40
- );
41
- }
42
-
43
- this.db = this.client.db(dbName);
44
- setDb(this.db);
45
- console.info(`[core-db] TalkPilot MongoDB connected: ${dbName}`);
46
- } catch (error) {
47
- console.error("[core-db] TalkPilot connection failed", error);
48
- throw error;
49
- }
50
- }
51
-
52
- async disconnect(): Promise<void> {
53
- if (this.client) {
54
- try {
55
- await this.client.close();
56
- this.client = null;
57
- this.db = null;
58
- console.info("MongoDB disconnected successfully");
59
- } catch (error) {
60
- console.error("[core-db] Disconnection failed", error);
61
- throw error;
62
- }
63
- }
64
- }
65
-
66
- getDb(): Db {
67
- if (!this.db) {
68
- throw new Error("Database not initialized. Call connect() first.");
69
- }
70
- return this.db;
71
- }
72
-
73
- isConnected(): boolean {
74
- return this.client !== null && this.client !== undefined;
75
- }
76
- }
77
-
78
- export const mongodbClient = new MongoDBClient();
1
+ import { MongoClient, Db } from "mongodb";
2
+ import { setDb } from "./index";
3
+ import { validateConfig, validateMongoUri } from "../utils/validation";
4
+
5
+ class MongoDBClient {
6
+ private client: MongoClient | null = null;
7
+ private db: Db | null = null;
8
+
9
+ private extractDatabaseName(uri: string): string | null {
10
+ try {
11
+ const url = new URL(uri);
12
+ const dbName = url.pathname.slice(1);
13
+ return dbName || null;
14
+ } catch {
15
+ return null;
16
+ }
17
+ }
18
+
19
+ async connect(uri?: string, databaseName?: string): Promise<void> {
20
+ if (this.client) {
21
+ return;
22
+ }
23
+
24
+ const mongodbUri = uri || process.env.MONGO_URI || process.env.MONGODB_URI;
25
+ validateConfig("MONGO_URI", mongodbUri);
26
+ validateMongoUri(mongodbUri!);
27
+
28
+ try {
29
+ this.client = new MongoClient(mongodbUri!);
30
+ await this.client.connect();
31
+
32
+ const dbNameFromUri = this.extractDatabaseName(mongodbUri!);
33
+ const dbName =
34
+ databaseName || process.env.TALKPILOT_DB_NAME || dbNameFromUri;
35
+
36
+ if (!dbName) {
37
+ throw new Error(
38
+ "[core-db] Database name not specified. Please pass databaseName to connect(), " +
39
+ "add it to the URI, or set TALKPILOT_DB_NAME.",
40
+ );
41
+ }
42
+
43
+ this.db = this.client.db(dbName);
44
+ setDb(this.db);
45
+ console.info(`[core-db] TalkPilot MongoDB connected: ${dbName}`);
46
+ } catch (error) {
47
+ console.error("[core-db] TalkPilot connection failed", error);
48
+ throw error;
49
+ }
50
+ }
51
+
52
+ async disconnect(): Promise<void> {
53
+ if (this.client) {
54
+ try {
55
+ await this.client.close();
56
+ this.client = null;
57
+ this.db = null;
58
+ console.info("MongoDB disconnected successfully");
59
+ } catch (error) {
60
+ console.error("[core-db] Disconnection failed", error);
61
+ throw error;
62
+ }
63
+ }
64
+ }
65
+
66
+ getDb(): Db {
67
+ if (!this.db) {
68
+ throw new Error("Database not initialized. Call connect() first.");
69
+ }
70
+ return this.db;
71
+ }
72
+
73
+ isConnected(): boolean {
74
+ return this.client !== null && this.client !== undefined;
75
+ }
76
+ }
77
+
78
+ export const mongodbClient = new MongoDBClient();
@@ -1,247 +1,247 @@
1
- import {
2
- getPhoneDataByPhoneNumber,
3
- getPhoneNumbersCollection,
4
- getPhoneNumbersForFlows,
5
- getClientPhoneNumber,
6
- getClientPhoneData,
7
- createPhoneNumberEntity,
8
- createPurchasedPhoneNumber,
9
- } from '../phone_numbers.getter';
10
- import { getFlowsCollection } from '../../flows/flows.getter';
11
- import { createFlow, createPhoneNumber } from '../../../test-utils/factories';
12
- import { ObjectId } from 'mongodb';
13
-
14
- describe('db.phoneNumbers', () => {
15
- describe('getPhoneDataByPhoneNumber', () => {
16
- it('return phone number data with flow', async () => {
17
- const flow = createFlow({
18
- clientId: "test-client-id",
19
- conversationSettings: {
20
- interruptions: {
21
- enableInterruptionDetection: true,
22
- interruptionWindowSeconds: 60,
23
- interruptionThresholdSeconds: 10,
24
- interruptionInstruction:
25
- "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.",
26
- },
27
- silence: {
28
- enableSilenceDetection: true,
29
- firstWarningSilenceSeconds: 5,
30
- firstWarningInstruction:
31
- "I have not heard you for a few seconds. If you are still on the line, please say something so we can continue.",
32
- secondWarningSilenceSeconds: 10,
33
- secondWarningInstruction:
34
- "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.",
35
- disconnectSilenceSeconds: 30,
36
- },
37
- },
38
- });
39
- const phoneData = createPhoneNumber({
40
- flow_id: flow._id,
41
- client_id: "test-client-id",
42
- });
43
-
44
- await getFlowsCollection().insertOne(flow);
45
- await getPhoneNumbersCollection().insertOne(phoneData);
46
-
47
- const result = await getPhoneDataByPhoneNumber(phoneData.phone_number);
48
-
49
- expect(result).toBeDefined();
50
- expect(result).toMatchObject({
51
- client_id: phoneData.client_id,
52
- phone_number: phoneData.phone_number,
53
- flow_id: flow._id,
54
- });
55
- // Specifically check if the flow was joined correctly
56
- expect(result?.flow.flowName).toBe(flow.flowName);
57
- expect(
58
- result?.flow.conversationSettings?.silence?.firstWarningSilenceSeconds,
59
- ).toBe(5);
60
- });
61
- });
62
-
63
- describe("getClientPhoneNumber", () => {
64
- it("return phone number by client id", async () => {
65
- const flow = createFlow();
66
- const phoneData = createPhoneNumber({
67
- flow_id: flow._id,
68
- is_primary: true,
69
- });
70
-
71
- await getFlowsCollection().insertOne(flow);
72
- await getPhoneNumbersCollection().insertOne(phoneData);
73
-
74
- const result = await getClientPhoneNumber(phoneData.client_id);
75
- expect(result).toBe(phoneData.phone_number);
76
- });
77
- });
78
-
79
- describe("getClientPhoneData", () => {
80
- it("returns primary phone data by default", async () => {
81
- const clientId = "multiPhoneClient";
82
- const phone1 = createPhoneNumber({
83
- client_id: clientId,
84
- is_primary: false,
85
- phone_number: "1",
86
- createdAt: new Date("2023-01-01"),
87
- });
88
- const phone2 = createPhoneNumber({
89
- client_id: clientId,
90
- is_primary: true,
91
- phone_number: "2",
92
- createdAt: new Date("2023-01-02"),
93
- });
94
-
95
- await getPhoneNumbersCollection().insertOne(phone1);
96
- await getPhoneNumbersCollection().insertOne(phone2);
97
-
98
- const result = await getClientPhoneData(clientId);
99
- expect(result?.phone_number).toBe("2");
100
- });
101
-
102
- it("returns most recent non-primary phone data if isPrimary is false", async () => {
103
- const clientId = "multiPhoneClient2";
104
- const phone1 = createPhoneNumber({
105
- client_id: clientId,
106
- is_primary: false,
107
- phone_number: "1",
108
- createdAt: new Date("2023-01-01"),
109
- });
110
- const phone2 = createPhoneNumber({
111
- client_id: clientId,
112
- is_primary: false,
113
- phone_number: "2",
114
- createdAt: new Date("2023-01-02"),
115
- });
116
- const phone3 = createPhoneNumber({
117
- client_id: clientId,
118
- is_primary: true,
119
- phone_number: "3",
120
- createdAt: new Date("2023-01-03"),
121
- });
122
-
123
- await getPhoneNumbersCollection().insertOne(phone1);
124
- await getPhoneNumbersCollection().insertOne(phone2);
125
- await getPhoneNumbersCollection().insertOne(phone3);
126
-
127
- const result = await getClientPhoneData(clientId, false);
128
- expect(result?.phone_number).toBe("2");
129
- });
130
- });
131
-
132
- describe('getPhoneNumbersForFlows', () => {
133
- it('returns all phone numbers for the client with flow ObjectIds, newest first', async () => {
134
- const clientId = 'flowsClient';
135
- const flow1 = createFlow({ clientId });
136
- const flow2 = createFlow({ clientId });
137
- await getFlowsCollection().insertMany([flow1, flow2]);
138
-
139
- const older = createPhoneNumber({
140
- client_id: clientId,
141
- flow_id: flow1._id,
142
- phone_number: '+100',
143
- is_primary: true,
144
- createdAt: new Date('2023-01-01'),
145
- });
146
- const newer = createPhoneNumber({
147
- client_id: clientId,
148
- flow_id: flow2._id,
149
- phone_number: '+200',
150
- is_primary: false,
151
- createdAt: new Date('2023-06-01'),
152
- });
153
-
154
- await getPhoneNumbersCollection().insertMany([older, newer]);
155
-
156
- const result = await getPhoneNumbersForFlows(clientId);
157
-
158
- expect(result).toHaveLength(2);
159
- expect(result[0]).toEqual({
160
- flowId: String(flow2._id),
161
- phoneNumber: '+200',
162
- isPrimary: false,
163
- });
164
- expect(result[1]).toEqual({
165
- flowId: String(flow1._id),
166
- phoneNumber: '+100',
167
- isPrimary: true,
168
- });
169
- });
170
-
171
- it('returns an empty array when the client has no phone numbers', async () => {
172
- const result = await getPhoneNumbersForFlows('noSuchClient');
173
- expect(result).toEqual([]);
174
- });
175
-
176
- it('does not include phone numbers for other clients', async () => {
177
- const clientA = 'clientA';
178
- const clientB = 'clientB';
179
- const flow = createFlow();
180
- await getFlowsCollection().insertOne(flow);
181
- const phone = createPhoneNumber({ client_id: clientA, flow_id: flow._id });
182
- await getPhoneNumbersCollection().insertOne(phone);
183
-
184
- const result = await getPhoneNumbersForFlows(clientB);
185
- expect(result).toEqual([]);
186
- });
187
- });
188
-
189
- describe('createPhoneNumberEntity', () => {
190
- it('creates first phone number as primary', async () => {
191
- const clientId = 'newClient';
192
- const flowId = new ObjectId().toHexString();
193
- const phoneNumber = "+123456789";
194
-
195
- const result = await createPhoneNumberEntity(
196
- phoneNumber,
197
- flowId,
198
- clientId,
199
- );
200
- expect(result.phone_number).toBe(phoneNumber);
201
- expect(result.is_primary).toBe(true);
202
- });
203
-
204
- it("creates subsequent phone number as non-primary", async () => {
205
- const clientId = "existingClient";
206
- const flowId = new ObjectId().toHexString();
207
-
208
- // Create first one
209
- await createPhoneNumberEntity("+111", flowId, clientId);
210
-
211
- // Create second one
212
- const result = await createPhoneNumberEntity("+222", flowId, clientId);
213
- expect(result.phone_number).toBe("+222");
214
- expect(result.is_primary).toBe(false);
215
- });
216
- });
217
-
218
- describe('createPurchasedPhoneNumber', () => {
219
- it('persists a phone_numbers document with provider payload and flow', async () => {
220
- const clientId = 'purchasedClientDb';
221
- const flowId = new ObjectId().toHexString();
222
- const phoneNumber = '+15551234567';
223
- const providerPayload = {
224
- provider: 'twilio' as const,
225
- provider_sid: 'PNxxxxxxxx',
226
- };
227
-
228
- await createPurchasedPhoneNumber(
229
- phoneNumber,
230
- providerPayload,
231
- clientId,
232
- flowId,
233
- );
234
-
235
- // Re-read from Mongo to confirm insert succeeded (not only the function return value).
236
- const stored = await getPhoneNumbersCollection().findOne({ phone_number: phoneNumber });
237
- expect(stored).not.toBeNull();
238
- expect(stored).toMatchObject({
239
- phone_number: phoneNumber,
240
- client_id: clientId,
241
- provider: 'twilio',
242
- provider_sid: 'PNxxxxxxxx',
243
- flow_id: new ObjectId(flowId),
244
- });
245
- });
246
- });
247
- });
1
+ import {
2
+ getPhoneDataByPhoneNumber,
3
+ getPhoneNumbersCollection,
4
+ getPhoneNumbersForFlows,
5
+ getClientPhoneNumber,
6
+ getClientPhoneData,
7
+ createPhoneNumberEntity,
8
+ createPurchasedPhoneNumber,
9
+ } from '../phone_numbers.getter';
10
+ import { getFlowsCollection } from '../../flows/flows.getter';
11
+ import { createFlow, createPhoneNumber } from '../../../test-utils/factories';
12
+ import { ObjectId } from 'mongodb';
13
+
14
+ describe('db.phoneNumbers', () => {
15
+ describe('getPhoneDataByPhoneNumber', () => {
16
+ it('return phone number data with flow', async () => {
17
+ const flow = createFlow({
18
+ clientId: "test-client-id",
19
+ conversationSettings: {
20
+ interruptions: {
21
+ enableInterruptionDetection: true,
22
+ interruptionWindowSeconds: 60,
23
+ interruptionThresholdSeconds: 10,
24
+ interruptionInstruction:
25
+ "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.",
26
+ },
27
+ silence: {
28
+ enableSilenceDetection: true,
29
+ firstWarningSilenceSeconds: 5,
30
+ firstWarningInstruction:
31
+ "I have not heard you for a few seconds. If you are still on the line, please say something so we can continue.",
32
+ secondWarningSilenceSeconds: 10,
33
+ secondWarningInstruction:
34
+ "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.",
35
+ disconnectSilenceSeconds: 30,
36
+ },
37
+ },
38
+ });
39
+ const phoneData = createPhoneNumber({
40
+ flow_id: flow._id,
41
+ client_id: "test-client-id",
42
+ });
43
+
44
+ await getFlowsCollection().insertOne(flow);
45
+ await getPhoneNumbersCollection().insertOne(phoneData);
46
+
47
+ const result = await getPhoneDataByPhoneNumber(phoneData.phone_number);
48
+
49
+ expect(result).toBeDefined();
50
+ expect(result).toMatchObject({
51
+ client_id: phoneData.client_id,
52
+ phone_number: phoneData.phone_number,
53
+ flow_id: flow._id,
54
+ });
55
+ // Specifically check if the flow was joined correctly
56
+ expect(result?.flow.flowName).toBe(flow.flowName);
57
+ expect(
58
+ result?.flow.conversationSettings?.silence?.firstWarningSilenceSeconds,
59
+ ).toBe(5);
60
+ });
61
+ });
62
+
63
+ describe("getClientPhoneNumber", () => {
64
+ it("return phone number by client id", async () => {
65
+ const flow = createFlow();
66
+ const phoneData = createPhoneNumber({
67
+ flow_id: flow._id,
68
+ is_primary: true,
69
+ });
70
+
71
+ await getFlowsCollection().insertOne(flow);
72
+ await getPhoneNumbersCollection().insertOne(phoneData);
73
+
74
+ const result = await getClientPhoneNumber(phoneData.client_id);
75
+ expect(result).toBe(phoneData.phone_number);
76
+ });
77
+ });
78
+
79
+ describe("getClientPhoneData", () => {
80
+ it("returns primary phone data by default", async () => {
81
+ const clientId = "multiPhoneClient";
82
+ const phone1 = createPhoneNumber({
83
+ client_id: clientId,
84
+ is_primary: false,
85
+ phone_number: "1",
86
+ createdAt: new Date("2023-01-01"),
87
+ });
88
+ const phone2 = createPhoneNumber({
89
+ client_id: clientId,
90
+ is_primary: true,
91
+ phone_number: "2",
92
+ createdAt: new Date("2023-01-02"),
93
+ });
94
+
95
+ await getPhoneNumbersCollection().insertOne(phone1);
96
+ await getPhoneNumbersCollection().insertOne(phone2);
97
+
98
+ const result = await getClientPhoneData(clientId);
99
+ expect(result?.phone_number).toBe("2");
100
+ });
101
+
102
+ it("returns most recent non-primary phone data if isPrimary is false", async () => {
103
+ const clientId = "multiPhoneClient2";
104
+ const phone1 = createPhoneNumber({
105
+ client_id: clientId,
106
+ is_primary: false,
107
+ phone_number: "1",
108
+ createdAt: new Date("2023-01-01"),
109
+ });
110
+ const phone2 = createPhoneNumber({
111
+ client_id: clientId,
112
+ is_primary: false,
113
+ phone_number: "2",
114
+ createdAt: new Date("2023-01-02"),
115
+ });
116
+ const phone3 = createPhoneNumber({
117
+ client_id: clientId,
118
+ is_primary: true,
119
+ phone_number: "3",
120
+ createdAt: new Date("2023-01-03"),
121
+ });
122
+
123
+ await getPhoneNumbersCollection().insertOne(phone1);
124
+ await getPhoneNumbersCollection().insertOne(phone2);
125
+ await getPhoneNumbersCollection().insertOne(phone3);
126
+
127
+ const result = await getClientPhoneData(clientId, false);
128
+ expect(result?.phone_number).toBe("2");
129
+ });
130
+ });
131
+
132
+ describe('getPhoneNumbersForFlows', () => {
133
+ it('returns all phone numbers for the client with flow ObjectIds, newest first', async () => {
134
+ const clientId = 'flowsClient';
135
+ const flow1 = createFlow({ clientId });
136
+ const flow2 = createFlow({ clientId });
137
+ await getFlowsCollection().insertMany([flow1, flow2]);
138
+
139
+ const older = createPhoneNumber({
140
+ client_id: clientId,
141
+ flow_id: flow1._id,
142
+ phone_number: '+100',
143
+ is_primary: true,
144
+ createdAt: new Date('2023-01-01'),
145
+ });
146
+ const newer = createPhoneNumber({
147
+ client_id: clientId,
148
+ flow_id: flow2._id,
149
+ phone_number: '+200',
150
+ is_primary: false,
151
+ createdAt: new Date('2023-06-01'),
152
+ });
153
+
154
+ await getPhoneNumbersCollection().insertMany([older, newer]);
155
+
156
+ const result = await getPhoneNumbersForFlows(clientId);
157
+
158
+ expect(result).toHaveLength(2);
159
+ expect(result[0]).toEqual({
160
+ flowId: String(flow2._id),
161
+ phoneNumber: '+200',
162
+ isPrimary: false,
163
+ });
164
+ expect(result[1]).toEqual({
165
+ flowId: String(flow1._id),
166
+ phoneNumber: '+100',
167
+ isPrimary: true,
168
+ });
169
+ });
170
+
171
+ it('returns an empty array when the client has no phone numbers', async () => {
172
+ const result = await getPhoneNumbersForFlows('noSuchClient');
173
+ expect(result).toEqual([]);
174
+ });
175
+
176
+ it('does not include phone numbers for other clients', async () => {
177
+ const clientA = 'clientA';
178
+ const clientB = 'clientB';
179
+ const flow = createFlow();
180
+ await getFlowsCollection().insertOne(flow);
181
+ const phone = createPhoneNumber({ client_id: clientA, flow_id: flow._id });
182
+ await getPhoneNumbersCollection().insertOne(phone);
183
+
184
+ const result = await getPhoneNumbersForFlows(clientB);
185
+ expect(result).toEqual([]);
186
+ });
187
+ });
188
+
189
+ describe('createPhoneNumberEntity', () => {
190
+ it('creates first phone number as primary', async () => {
191
+ const clientId = 'newClient';
192
+ const flowId = new ObjectId().toHexString();
193
+ const phoneNumber = "+123456789";
194
+
195
+ const result = await createPhoneNumberEntity(
196
+ phoneNumber,
197
+ flowId,
198
+ clientId,
199
+ );
200
+ expect(result.phone_number).toBe(phoneNumber);
201
+ expect(result.is_primary).toBe(true);
202
+ });
203
+
204
+ it("creates subsequent phone number as non-primary", async () => {
205
+ const clientId = "existingClient";
206
+ const flowId = new ObjectId().toHexString();
207
+
208
+ // Create first one
209
+ await createPhoneNumberEntity("+111", flowId, clientId);
210
+
211
+ // Create second one
212
+ const result = await createPhoneNumberEntity("+222", flowId, clientId);
213
+ expect(result.phone_number).toBe("+222");
214
+ expect(result.is_primary).toBe(false);
215
+ });
216
+ });
217
+
218
+ describe('createPurchasedPhoneNumber', () => {
219
+ it('persists a phone_numbers document with provider payload and flow', async () => {
220
+ const clientId = 'purchasedClientDb';
221
+ const flowId = new ObjectId().toHexString();
222
+ const phoneNumber = '+15551234567';
223
+ const providerPayload = {
224
+ provider: 'twilio' as const,
225
+ provider_sid: 'PNxxxxxxxx',
226
+ };
227
+
228
+ await createPurchasedPhoneNumber(
229
+ phoneNumber,
230
+ providerPayload,
231
+ clientId,
232
+ flowId,
233
+ );
234
+
235
+ // Re-read from Mongo to confirm insert succeeded (not only the function return value).
236
+ const stored = await getPhoneNumbersCollection().findOne({ phone_number: phoneNumber });
237
+ expect(stored).not.toBeNull();
238
+ expect(stored).toMatchObject({
239
+ phone_number: phoneNumber,
240
+ client_id: clientId,
241
+ provider: 'twilio',
242
+ provider_sid: 'PNxxxxxxxx',
243
+ flow_id: new ObjectId(flowId),
244
+ });
245
+ });
246
+ });
247
+ });
@@ -1,2 +1,2 @@
1
- export * from './phone_numbers.getter';
2
- export type * from './phone_numbers.types';
1
+ export * from './phone_numbers.getter';
2
+ export type * from './phone_numbers.types';