@talkpilot/core-db 1.2.0 → 1.2.2

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 (168) hide show
  1. package/README.md +95 -116
  2. package/README_OLD.md +160 -0
  3. package/dist/talkpilot/calls/calls.dashboard.d.ts +3 -0
  4. package/dist/talkpilot/calls/calls.dashboard.d.ts.map +1 -0
  5. package/dist/talkpilot/calls/calls.dashboard.js +191 -0
  6. package/dist/talkpilot/calls/calls.dashboard.js.map +1 -0
  7. package/dist/talkpilot/calls/calls.getters.d.ts +3 -3
  8. package/dist/talkpilot/calls/calls.getters.d.ts.map +1 -1
  9. package/dist/talkpilot/calls/calls.getters.js +1 -178
  10. package/dist/talkpilot/calls/calls.getters.js.map +1 -1
  11. package/dist/talkpilot/calls/calls.types.d.ts +3 -55
  12. package/dist/talkpilot/calls/calls.types.d.ts.map +1 -1
  13. package/dist/talkpilot/calls/dashboard/calls.dashboard.d.ts +36 -0
  14. package/dist/talkpilot/calls/dashboard/calls.dashboard.d.ts.map +1 -0
  15. package/dist/talkpilot/calls/dashboard/calls.dashboard.js +208 -0
  16. package/dist/talkpilot/calls/dashboard/calls.dashboard.js.map +1 -0
  17. package/dist/talkpilot/calls/dashboard/calls.dashboard.types.d.ts +66 -0
  18. package/dist/talkpilot/calls/dashboard/calls.dashboard.types.d.ts.map +1 -0
  19. package/dist/talkpilot/calls/dashboard/calls.dashboard.types.js +3 -0
  20. package/dist/talkpilot/calls/dashboard/calls.dashboard.types.js.map +1 -0
  21. package/dist/talkpilot/calls/index.d.ts +1 -0
  22. package/dist/talkpilot/calls/index.d.ts.map +1 -1
  23. package/dist/talkpilot/calls/index.js +1 -0
  24. package/dist/talkpilot/calls/index.js.map +1 -1
  25. package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts +2 -2
  26. package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts.map +1 -1
  27. package/dist/talkpilot/clientsConfig/clientsConfig.getters.js +11 -10
  28. package/dist/talkpilot/clientsConfig/clientsConfig.getters.js.map +1 -1
  29. package/dist/talkpilot/clientsConfig/clientsConfig.types.d.ts +20 -9
  30. package/dist/talkpilot/clientsConfig/clientsConfig.types.d.ts.map +1 -1
  31. package/dist/talkpilot/clientsConfig/clientsConfig.types.js +6 -0
  32. package/dist/talkpilot/clientsConfig/clientsConfig.types.js.map +1 -1
  33. package/dist/talkpilot/flows/flows.schema.js +1 -1
  34. package/dist/talkpilot/phone_numbers/index.d.ts +2 -2
  35. package/dist/talkpilot/phone_numbers/phone_numbers.getter.d.ts +1 -1
  36. package/dist/talkpilot/phone_numbers/phone_numbers.getter.d.ts.map +1 -1
  37. package/dist/talkpilot/phone_numbers/phone_numbers.getter.js +5 -3
  38. package/dist/talkpilot/phone_numbers/phone_numbers.getter.js.map +1 -1
  39. package/dist/talkpilot/phone_numbers/phone_numbers.schema.js +12 -12
  40. package/dist/talkpilot/phone_numbers/phone_numbers.types.d.ts +4 -4
  41. package/dist/talkpilot/results/results.getter.d.ts.map +1 -1
  42. package/dist/talkpilot/results/results.getter.js.map +1 -1
  43. package/dist/talkpilot/retry_analyze/retryAnalyze.getters.d.ts.map +1 -1
  44. package/dist/talkpilot/retry_analyze/retryAnalyze.getters.js.map +1 -1
  45. package/dist/utils/shared.types.d.ts +5 -0
  46. package/dist/utils/shared.types.d.ts.map +1 -0
  47. package/dist/utils/shared.types.js +3 -0
  48. package/dist/utils/shared.types.js.map +1 -0
  49. package/package.json +2 -1
  50. package/src/__tests__/setup.ts +20 -20
  51. package/src/connection.ts +42 -42
  52. package/src/index.ts +16 -16
  53. package/src/municipal/__tests__/validation.spec.ts +62 -62
  54. package/src/municipal/cities/cities.getters.ts +50 -50
  55. package/src/municipal/cities/cities.types.ts +11 -11
  56. package/src/municipal/cities/index.ts +2 -2
  57. package/src/municipal/departmentsSubjects/departmentsSubjects.getters.ts +282 -282
  58. package/src/municipal/departmentsSubjects/departmentsSubjects.types.ts +72 -72
  59. package/src/municipal/departmentsSubjects/index.ts +9 -9
  60. package/src/municipal/index.ts +21 -21
  61. package/src/municipal/mongodb-client.ts +61 -61
  62. package/src/municipal/streets/index.ts +2 -2
  63. package/src/municipal/streets/streets.getters.ts +125 -125
  64. package/src/municipal/streets/streets.types.ts +18 -18
  65. package/src/municipal/systemInstructions/__tests__/getters.spec.ts +113 -113
  66. package/src/municipal/systemInstructions/__tests__/setters.spec.ts +274 -274
  67. package/src/municipal/systemInstructions/index.ts +7 -7
  68. package/src/municipal/systemInstructions/instructions.getters.ts +57 -57
  69. package/src/municipal/systemInstructions/instructions.setters.ts +119 -119
  70. package/src/municipal/systemInstructions/instructions.types.ts +30 -30
  71. package/src/municipal/tickets/__tests__/tickets.getters.spec.ts +66 -66
  72. package/src/municipal/tickets/index.ts +2 -2
  73. package/src/municipal/tickets/tickets.getters.ts +261 -261
  74. package/src/municipal/tickets/tickets.types.ts +43 -43
  75. package/src/municipal/utils/types.ts +11 -11
  76. package/src/talkpilot/__tests__/db.spec.ts +38 -38
  77. package/src/talkpilot/__tests__/mongodb-client.spec.ts +18 -18
  78. package/src/talkpilot/__tests__/validation.spec.ts +68 -68
  79. package/src/talkpilot/agents/__tests__/agents.getters.spec.ts +29 -29
  80. package/src/talkpilot/agents/agents.getters.ts +34 -34
  81. package/src/talkpilot/agents/agents.types.ts +14 -14
  82. package/src/talkpilot/agents/index.ts +2 -2
  83. package/src/talkpilot/backgroundToolResults/__tests__/backgroundToolResults.getters.spec.ts +147 -147
  84. package/src/talkpilot/backgroundToolResults/backgroundToolResults.getters.ts +65 -65
  85. package/src/talkpilot/backgroundToolResults/backgroundToolResults.types.ts +23 -23
  86. package/src/talkpilot/backgroundToolResults/index.ts +2 -2
  87. package/src/talkpilot/calls/__tests__/callStats.utils.spec.ts +128 -128
  88. package/src/talkpilot/calls/__tests__/calls.dashboard.spec.ts +46 -0
  89. package/src/talkpilot/calls/__tests__/calls.spec.ts +270 -252
  90. package/src/talkpilot/calls/calls.getters.ts +248 -446
  91. package/src/talkpilot/calls/calls.types.ts +113 -171
  92. package/src/talkpilot/calls/dashboard/calls.dashboard.ts +243 -0
  93. package/src/talkpilot/calls/dashboard/calls.dashboard.types.ts +70 -0
  94. package/src/talkpilot/calls/index.ts +3 -2
  95. package/src/talkpilot/clientAudioBuffers/__tests__/clientAudioBuffer.getters.spec.ts +160 -160
  96. package/src/talkpilot/clientAudioBuffers/clientAudioBuffer.getters.ts +117 -117
  97. package/src/talkpilot/clientAudioBuffers/clientsAudioBuffers.types.ts +25 -25
  98. package/src/talkpilot/clientAudioBuffers/index.ts +2 -2
  99. package/src/talkpilot/clients/clients.getters.ts +16 -16
  100. package/src/talkpilot/clients/clients.types.ts +14 -14
  101. package/src/talkpilot/clients/index.ts +2 -2
  102. package/src/talkpilot/clientsConfig/__tests__/clientsConfig.getters.spec.ts +53 -0
  103. package/src/talkpilot/clientsConfig/__tests__/clientsConfig.spec.ts +197 -106
  104. package/src/talkpilot/clientsConfig/clientsConfig.getters.ts +55 -44
  105. package/src/talkpilot/clientsConfig/clientsConfig.types.ts +127 -94
  106. package/src/talkpilot/clientsConfig/index.ts +2 -2
  107. package/src/talkpilot/flows/__tests__/flows.schema.spec.ts +71 -67
  108. package/src/talkpilot/flows/flows.getter.ts +14 -14
  109. package/src/talkpilot/flows/flows.schema.ts +153 -153
  110. package/src/talkpilot/flows/flows.types.ts +184 -184
  111. package/src/talkpilot/flows/index.ts +2 -2
  112. package/src/talkpilot/groups/__tests__/groups.spec.ts +90 -90
  113. package/src/talkpilot/groups/__tests__/phone.utils.spec.ts +32 -32
  114. package/src/talkpilot/groups/groups.getters.ts +30 -30
  115. package/src/talkpilot/groups/groups.types.ts +29 -29
  116. package/src/talkpilot/groups/index.ts +3 -3
  117. package/src/talkpilot/groups/phone.utils.ts +46 -46
  118. package/src/talkpilot/index.ts +29 -29
  119. package/src/talkpilot/leads/index.ts +2 -2
  120. package/src/talkpilot/leads/leads.getter.ts +6 -6
  121. package/src/talkpilot/leads/leads.schema.ts +33 -33
  122. package/src/talkpilot/leads/leads.types.ts +20 -20
  123. package/src/talkpilot/mongodb-client.ts +78 -78
  124. package/src/talkpilot/phone_numbers/__tests__/phone_numbers.spec.ts +252 -247
  125. package/src/talkpilot/phone_numbers/index.ts +2 -2
  126. package/src/talkpilot/phone_numbers/phone_numbers.getter.ts +158 -154
  127. package/src/talkpilot/phone_numbers/phone_numbers.schema.ts +17 -17
  128. package/src/talkpilot/phone_numbers/phone_numbers.types.ts +30 -30
  129. package/src/talkpilot/plans/__tests__/plans.spec.ts +70 -70
  130. package/src/talkpilot/plans/index.ts +2 -2
  131. package/src/talkpilot/plans/plans.getters.ts +132 -132
  132. package/src/talkpilot/plans/plans.types.ts +89 -89
  133. package/src/talkpilot/results/index.ts +7 -7
  134. package/src/talkpilot/results/results.getter.ts +39 -35
  135. package/src/talkpilot/results/results.schema.ts +25 -25
  136. package/src/talkpilot/results/results.types.ts +34 -34
  137. package/src/talkpilot/retry_analyze/__tests__/retryAnalyze.getters.spec.ts +156 -156
  138. package/src/talkpilot/retry_analyze/index.ts +2 -2
  139. package/src/talkpilot/retry_analyze/retryAnalyze.getters.ts +84 -75
  140. package/src/talkpilot/retry_analyze/retryAnalyze.types.ts +13 -13
  141. package/src/talkpilot/sessions/__tests__/sessions.spec.ts +147 -147
  142. package/src/talkpilot/sessions/index.ts +2 -2
  143. package/src/talkpilot/sessions/sessions.getter.ts +92 -92
  144. package/src/talkpilot/sessions/sessions.schema.ts +34 -34
  145. package/src/talkpilot/sessions/sessions.types.ts +30 -30
  146. package/src/talkpilot/subscriptions/__tests__/subscriptions.getters.utils.spec.ts +45 -45
  147. package/src/talkpilot/subscriptions/index.ts +3 -3
  148. package/src/talkpilot/subscriptions/subscriptions.getters.ts +146 -146
  149. package/src/talkpilot/subscriptions/subscriptions.getters.utils.ts +33 -33
  150. package/src/talkpilot/subscriptions/subscriptions.types.ts +66 -66
  151. package/src/talkpilot/utils/__tests__/query.utils.spec.ts +49 -49
  152. package/src/talkpilot/utils/query.utils.ts +21 -21
  153. package/src/test-utils/db-utils.ts +24 -24
  154. package/src/test-utils/factories/index.ts +12 -12
  155. package/src/test-utils/factories/municipal/cities.ts +16 -16
  156. package/src/test-utils/factories/municipal/departmentsSubjects.ts +37 -37
  157. package/src/test-utils/factories/municipal/streets.ts +22 -22
  158. package/src/test-utils/factories/municipal/tickets.ts +39 -39
  159. package/src/test-utils/factories/talkpilot/agents.ts +19 -19
  160. package/src/test-utils/factories/talkpilot/calls.ts +37 -37
  161. package/src/test-utils/factories/talkpilot/clientAudioBuffers.ts +20 -20
  162. package/src/test-utils/factories/talkpilot/clientsConfig.ts +18 -18
  163. package/src/test-utils/factories/talkpilot/flows.ts +33 -33
  164. package/src/test-utils/factories/talkpilot/groups.ts +33 -33
  165. package/src/test-utils/factories/talkpilot/phone_numbers.ts +22 -22
  166. package/src/test-utils/factories/talkpilot/sessions.ts +35 -35
  167. package/src/utils/shared.types.ts +4 -0
  168. package/src/utils/validation.ts +23 -23
@@ -1,2 +1,2 @@
1
- export * from "./clients.getters";
2
- export * from "./clients.types";
1
+ export * from "./clients.getters";
2
+ export * from "./clients.types";
@@ -0,0 +1,53 @@
1
+ import { describe, it, expect, beforeEach, afterEach } from "@jest/globals";
2
+ import {
3
+ getClientsConfigCollection,
4
+ updateProductConfig,
5
+ } from "../clientsConfig.getters";
6
+
7
+ describe("updateProductConfig", () => {
8
+ beforeEach(async () => {
9
+ await getClientsConfigCollection().deleteMany({});
10
+ });
11
+
12
+ afterEach(async () => {
13
+ await getClientsConfigCollection().deleteMany({});
14
+ });
15
+
16
+ it("should update partial fields using correct dot notation", async () => {
17
+ const clientId = "client-123";
18
+ const collection = getClientsConfigCollection();
19
+
20
+ await collection.insertOne({
21
+ clientId,
22
+ products: {
23
+ websiteTalk: {
24
+ language: "English",
25
+ companyName: "OldCorp",
26
+ defaultBaseUrl: "https://old-url.com",
27
+ logoUrl: "",
28
+ timezone: "UTC",
29
+ },
30
+ },
31
+ } as any);
32
+
33
+ const updates = { language: "Hebrew", companyName: "TestCorp" };
34
+ await updateProductConfig(clientId, "websiteTalk", updates);
35
+
36
+ const updatedDoc = await collection.findOne({ clientId });
37
+
38
+ expect(updatedDoc).toBeDefined();
39
+ expect(updatedDoc?.products.websiteTalk?.language).toBe("Hebrew");
40
+ expect(updatedDoc?.products.websiteTalk?.companyName).toBe("TestCorp");
41
+ expect(updatedDoc?.products.websiteTalk?.defaultBaseUrl).toBe(
42
+ "https://old-url.com",
43
+ );
44
+ });
45
+
46
+ it("should throw an error if no client config is found", async () => {
47
+ await expect(
48
+ updateProductConfig("invalid-client", "websiteTalk", {
49
+ language: "English",
50
+ }),
51
+ ).rejects.toThrow("No client config found for clientId: invalid-client");
52
+ });
53
+ });
@@ -1,106 +1,197 @@
1
- import {
2
- createClientConfigDoc,
3
- getClientConfig,
4
- } from "../clientsConfig.getters";
5
- import { createClientConfig } from "../../../test-utils/factories";
6
-
7
- describe("db.clientsConfig", () => {
8
- describe("getClientConfig", () => {
9
- it("should return client config by clientId", async () => {
10
- const clientId = "test-client-id";
11
- const clientConfig = createClientConfig({
12
- clientId,
13
- language: "Hebrew",
14
- products: { feature1: true, feature2: false } as any,
15
- });
16
-
17
- await createClientConfigDoc(clientConfig);
18
-
19
- const result = await getClientConfig(clientId);
20
-
21
- expect(result).toMatchObject({
22
- clientId,
23
- language: "Hebrew",
24
- products: { feature1: true, feature2: false },
25
- });
26
- });
27
-
28
- it("should support websiteTalk product config", async () => {
29
- const clientId = "websiteTalk-test";
30
- const clientConfig = createClientConfig({
31
- clientId,
32
- products: {
33
- websiteTalk: {
34
- logoUrl: "https://cdn.example.com/branding/acme/logo.png",
35
- companyName: "Acme Ltd",
36
- timezone: "Asia/Jerusalem",
37
- language: "he",
38
- defaultBaseUrl: "https://acme.example",
39
- scheduler: {
40
- type: "cron",
41
- schedule: "0 9 * * 1-5"
42
- },
43
- },
44
- },
45
- });
46
-
47
- await createClientConfigDoc(clientConfig);
48
-
49
- const result = await getClientConfig(clientId);
50
-
51
- expect(result).toMatchObject({
52
- clientId,
53
- products: {
54
- websiteTalk: {
55
- logoUrl: "https://cdn.example.com/branding/acme/logo.png",
56
- companyName: "Acme Ltd",
57
- timezone: "Asia/Jerusalem",
58
- language: "he",
59
- defaultBaseUrl: "https://acme.example",
60
- scheduler: {
61
- type: "cron",
62
- },
63
- },
64
- },
65
- });
66
- });
67
-
68
- it("should return null for non-existent clientId", async () => {
69
- const result = await getClientConfig("non-existent-client");
70
- expect(result).toBeNull();
71
- });
72
-
73
- it("should support communications config", async () => {
74
- const clientId = "communications-test";
75
- const clientConfig = createClientConfig({
76
- clientId,
77
- communications: {
78
- sendGrid: {
79
- emailTo: ["test@example.com"],
80
- templateId: "temp-123",
81
- senderId: "sender-123",
82
- },
83
- sms: {},
84
- whatsapp: {},
85
- },
86
- });
87
-
88
- await createClientConfigDoc(clientConfig);
89
-
90
- const result = await getClientConfig(clientId);
91
-
92
- expect(result).toMatchObject({
93
- clientId,
94
- communications: {
95
- sendGrid: {
96
- emailTo: ["test@example.com"],
97
- templateId: "temp-123",
98
- senderId: "sender-123",
99
- },
100
- sms: {},
101
- whatsapp: {},
102
- },
103
- });
104
- });
105
- });
106
- });
1
+ import {
2
+ createClientConfigDoc,
3
+ getClientConfig,
4
+ } from "../clientsConfig.getters";
5
+ import { createClientConfig } from "../../../test-utils/factories";
6
+
7
+ describe("db.clientsConfig", () => {
8
+ describe("getClientConfig", () => {
9
+ it("should return client config by clientId", async () => {
10
+ const clientId = "test-client-id";
11
+ const clientConfig = createClientConfig({
12
+ clientId,
13
+ language: "Hebrew",
14
+ products: { feature1: true, feature2: false } as any,
15
+ });
16
+
17
+ await createClientConfigDoc(clientConfig);
18
+
19
+ const result = await getClientConfig(clientId);
20
+
21
+ expect(result).toMatchObject({
22
+ clientId,
23
+ language: "Hebrew",
24
+ products: { feature1: true, feature2: false },
25
+ });
26
+ });
27
+
28
+ it("should support websiteTalk product config", async () => {
29
+ const clientId = "websiteTalk-test";
30
+ const clientConfig = createClientConfig({
31
+ clientId,
32
+ products: {
33
+ websiteTalk: {
34
+ logoUrl: "https://cdn.example.com/branding/acme/logo.png",
35
+ companyName: "Acme Ltd",
36
+ timezone: "Asia/Jerusalem",
37
+ language: "he",
38
+ defaultBaseUrl: "https://acme.example",
39
+ scheduler: {
40
+ type: "cron",
41
+ schedule: "0 9 * * 1-5",
42
+ },
43
+ },
44
+ },
45
+ });
46
+
47
+ await createClientConfigDoc(clientConfig);
48
+
49
+ const result = await getClientConfig(clientId);
50
+
51
+ expect(result).toMatchObject({
52
+ clientId,
53
+ products: {
54
+ websiteTalk: {
55
+ logoUrl: "https://cdn.example.com/branding/acme/logo.png",
56
+ companyName: "Acme Ltd",
57
+ timezone: "Asia/Jerusalem",
58
+ language: "he",
59
+ defaultBaseUrl: "https://acme.example",
60
+ scheduler: {
61
+ type: "cron",
62
+ },
63
+ },
64
+ },
65
+ });
66
+ });
67
+
68
+ it("should return null for non-existent clientId", async () => {
69
+ const result = await getClientConfig("non-existent-client");
70
+ expect(result).toBeNull();
71
+ });
72
+
73
+ it("given moked_106 with both toolsPrompts when saved and retrieved then both prompts are persisted", async () => {
74
+ // Given
75
+ const clientId = "tools-prompts-both";
76
+ const clientConfig = createClientConfig({
77
+ clientId,
78
+ products: {
79
+ municipal: {
80
+ cityName: "ashdod",
81
+ timezone: "Asia/Jerusalem",
82
+ moked_106: {
83
+ toolsPrompts: {
84
+ findSubjectsSystemPrompt: "Custom subjects prompt",
85
+ findStreetsSystemPrompt: "Custom streets prompt",
86
+ },
87
+ },
88
+ },
89
+ },
90
+ });
91
+
92
+ // When
93
+ await createClientConfigDoc(clientConfig);
94
+ const result = await getClientConfig(clientId);
95
+
96
+ // Then
97
+ expect(result?.products.municipal?.moked_106?.toolsPrompts).toMatchObject(
98
+ {
99
+ findSubjectsSystemPrompt: "Custom subjects prompt",
100
+ findStreetsSystemPrompt: "Custom streets prompt",
101
+ },
102
+ );
103
+ });
104
+
105
+ it("given moked_106 with findSubjectsSystemPrompt only when saved and retrieved then findStreetsSystemPrompt is undefined", async () => {
106
+ // Given
107
+ const clientId = "tools-prompts-subjects-only";
108
+ const clientConfig = createClientConfig({
109
+ clientId,
110
+ products: {
111
+ municipal: {
112
+ cityName: "ashdod",
113
+ timezone: "Asia/Jerusalem",
114
+ moked_106: {
115
+ toolsPrompts: {
116
+ findSubjectsSystemPrompt: "Subjects only prompt",
117
+ },
118
+ },
119
+ },
120
+ },
121
+ });
122
+
123
+ // When
124
+ await createClientConfigDoc(clientConfig);
125
+ const result = await getClientConfig(clientId);
126
+
127
+ // Then
128
+ expect(
129
+ result?.products.municipal?.moked_106?.toolsPrompts
130
+ ?.findSubjectsSystemPrompt,
131
+ ).toBe("Subjects only prompt");
132
+ expect(
133
+ result?.products.municipal?.moked_106?.toolsPrompts
134
+ ?.findStreetsSystemPrompt,
135
+ ).toBeUndefined();
136
+ });
137
+
138
+ it("given moked_106 without toolsPrompts when saved and retrieved then toolsPrompts is undefined", async () => {
139
+ // Given
140
+ const clientId = "tools-prompts-absent";
141
+ const clientConfig = createClientConfig({
142
+ clientId,
143
+ products: {
144
+ municipal: {
145
+ cityName: "ashdod",
146
+ timezone: "Asia/Jerusalem",
147
+ moked_106: {
148
+ avoidSaveTickets: true,
149
+ },
150
+ },
151
+ },
152
+ });
153
+
154
+ // When
155
+ await createClientConfigDoc(clientConfig);
156
+ const result = await getClientConfig(clientId);
157
+
158
+ // Then
159
+ expect(
160
+ result?.products.municipal?.moked_106?.toolsPrompts,
161
+ ).toBeUndefined();
162
+ });
163
+
164
+ it("should support communications config", async () => {
165
+ const clientId = "communications-test";
166
+ const clientConfig = createClientConfig({
167
+ clientId,
168
+ communications: {
169
+ sendGrid: {
170
+ emailTo: ["test@example.com"],
171
+ templateId: "temp-123",
172
+ senderId: "sender-123",
173
+ },
174
+ sms: {},
175
+ whatsapp: {},
176
+ },
177
+ });
178
+
179
+ await createClientConfigDoc(clientConfig);
180
+
181
+ const result = await getClientConfig(clientId);
182
+
183
+ expect(result).toMatchObject({
184
+ clientId,
185
+ communications: {
186
+ sendGrid: {
187
+ emailTo: ["test@example.com"],
188
+ templateId: "temp-123",
189
+ senderId: "sender-123",
190
+ },
191
+ sms: {},
192
+ whatsapp: {},
193
+ },
194
+ });
195
+ });
196
+ });
197
+ });
@@ -1,44 +1,55 @@
1
- import { ClientConfigDoc, getDb, findClientByPhoneNumber } from "../index";
2
- import { Collection } from "mongodb";
3
-
4
- export const getClientsConfigCollection = (): Collection<ClientConfigDoc> => {
5
- return getDb().collection<ClientConfigDoc>("clientsConfig");
6
- };
7
-
8
- export const getClientConfig = (clientId: string) => {
9
- return getClientsConfigCollection().findOne({ clientId });
10
- };
11
-
12
- export const getClientConfigByPhone = async (phone: string) => {
13
- const client = await findClientByPhoneNumber(phone);
14
- if (!client) return null;
15
- return getClientConfig(client.clientId);
16
- };
17
-
18
- export const createClientConfigDoc = async (
19
- clientConfig: ClientConfigDoc,
20
- ): Promise<void> => {
21
- await getClientsConfigCollection().insertOne(clientConfig);
22
- };
23
-
24
- export async function updateWebsitalkDefaultBaseUrl(
25
- clientId: string,
26
- defaultBaseUrl: string,
27
- ): Promise<void> {
28
- try {
29
- const result = await getClientsConfigCollection().updateOne(
30
- { clientId },
31
- { $set: { "products.websiteTalk.defaultBaseUrl": defaultBaseUrl } },
32
- );
33
-
34
- if (result.matchedCount === 0) {
35
- throw new Error(`No client config found for clientId: ${clientId}`);
36
- }
37
- } catch (error) {
38
- console.error(
39
- `[core-db] Error updating defaultBaseUrl for client: ${clientId}`,
40
- error
41
- );
42
- throw error;
43
- }
44
- }
1
+ import {
2
+ ClientConfigDoc,
3
+ getDb,
4
+ findClientByPhoneNumber,
5
+ Products,
6
+ KNOWN_PRODUCT_KEYS,
7
+ KnownProductKey,
8
+ } from "../index";
9
+ import { Collection } from "mongodb";
10
+
11
+ export const getClientsConfigCollection = (): Collection<ClientConfigDoc> => {
12
+ return getDb().collection<ClientConfigDoc>("clientsConfig");
13
+ };
14
+
15
+ export const getClientConfig = (clientId: string) => {
16
+ return getClientsConfigCollection().findOne({ clientId });
17
+ };
18
+
19
+ export const getClientConfigByPhone = async (phone: string) => {
20
+ const client = await findClientByPhoneNumber(phone);
21
+ if (!client) return null;
22
+ return getClientConfig(client.clientId);
23
+ };
24
+
25
+ export const createClientConfigDoc = async (
26
+ clientConfig: ClientConfigDoc,
27
+ ): Promise<void> => {
28
+ await getClientsConfigCollection().insertOne(clientConfig);
29
+ };
30
+
31
+ export async function updateProductConfig<K extends KnownProductKey>(
32
+ clientId: string,
33
+ productKey: K,
34
+ updates: Partial<Products[K]>,
35
+ ): Promise<void> {
36
+ if (!KNOWN_PRODUCT_KEYS.includes(productKey as KnownProductKey)) {
37
+ throw new Error(
38
+ `Invalid product key: "${productKey}". Must be one of: ${KNOWN_PRODUCT_KEYS.join(", ")}`,
39
+ );
40
+ }
41
+
42
+ const setOperations: Record<string, unknown> = {};
43
+ for (const [key, value] of Object.entries(updates)) {
44
+ setOperations[`products.${productKey}.${key}`] = value;
45
+ }
46
+
47
+ const result = await getClientsConfigCollection().updateOne(
48
+ { clientId },
49
+ { $set: setOperations },
50
+ );
51
+
52
+ if (result.matchedCount === 0) {
53
+ throw new Error(`No client config found for clientId: ${clientId}`);
54
+ }
55
+ }