@talkpilot/core-db 1.1.19 → 1.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (130) hide show
  1. package/.cursor/rules/development.mdc +65 -65
  2. package/DEVELOPMENT.md +98 -98
  3. package/README.md +160 -160
  4. package/dist/talkpilot/calls/calls.getters.d.ts +2 -1
  5. package/dist/talkpilot/calls/calls.getters.d.ts.map +1 -1
  6. package/dist/talkpilot/calls/calls.getters.js +176 -0
  7. package/dist/talkpilot/calls/calls.getters.js.map +1 -1
  8. package/dist/talkpilot/calls/calls.types.d.ts +48 -0
  9. package/dist/talkpilot/calls/calls.types.d.ts.map +1 -1
  10. package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts +1 -0
  11. package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts.map +1 -1
  12. package/dist/talkpilot/clientsConfig/clientsConfig.getters.js +13 -0
  13. package/dist/talkpilot/clientsConfig/clientsConfig.getters.js.map +1 -1
  14. package/jest.config.js +19 -19
  15. package/package.json +45 -45
  16. package/src/__tests__/setup.ts +20 -20
  17. package/src/connection.ts +42 -42
  18. package/src/index.ts +16 -16
  19. package/src/municipal/__tests__/validation.spec.ts +62 -62
  20. package/src/municipal/cities/cities.getters.ts +50 -50
  21. package/src/municipal/cities/cities.types.ts +11 -11
  22. package/src/municipal/cities/index.ts +2 -2
  23. package/src/municipal/departmentsSubjects/departmentsSubjects.getters.ts +282 -282
  24. package/src/municipal/departmentsSubjects/departmentsSubjects.types.ts +72 -72
  25. package/src/municipal/departmentsSubjects/index.ts +9 -9
  26. package/src/municipal/index.ts +21 -21
  27. package/src/municipal/mongodb-client.ts +61 -61
  28. package/src/municipal/streets/index.ts +2 -2
  29. package/src/municipal/streets/streets.getters.ts +125 -125
  30. package/src/municipal/streets/streets.types.ts +18 -18
  31. package/src/municipal/systemInstructions/__tests__/getters.spec.ts +113 -113
  32. package/src/municipal/systemInstructions/__tests__/setters.spec.ts +274 -274
  33. package/src/municipal/systemInstructions/index.ts +7 -7
  34. package/src/municipal/systemInstructions/instructions.getters.ts +57 -57
  35. package/src/municipal/systemInstructions/instructions.setters.ts +119 -119
  36. package/src/municipal/systemInstructions/instructions.types.ts +30 -30
  37. package/src/municipal/tickets/__tests__/tickets.getters.spec.ts +66 -66
  38. package/src/municipal/tickets/index.ts +2 -2
  39. package/src/municipal/tickets/tickets.getters.ts +261 -261
  40. package/src/municipal/tickets/tickets.types.ts +43 -43
  41. package/src/municipal/utils/types.ts +11 -11
  42. package/src/talkpilot/__tests__/db.spec.ts +38 -38
  43. package/src/talkpilot/__tests__/mongodb-client.spec.ts +18 -18
  44. package/src/talkpilot/__tests__/validation.spec.ts +68 -68
  45. package/src/talkpilot/agents/__tests__/agents.getters.spec.ts +29 -29
  46. package/src/talkpilot/agents/agents.getters.ts +34 -34
  47. package/src/talkpilot/agents/agents.types.ts +14 -14
  48. package/src/talkpilot/agents/index.ts +2 -2
  49. package/src/talkpilot/backgroundToolResults/__tests__/backgroundToolResults.getters.spec.ts +147 -147
  50. package/src/talkpilot/backgroundToolResults/backgroundToolResults.getters.ts +65 -65
  51. package/src/talkpilot/backgroundToolResults/backgroundToolResults.types.ts +23 -23
  52. package/src/talkpilot/backgroundToolResults/index.ts +2 -2
  53. package/src/talkpilot/calls/__tests__/callStats.utils.spec.ts +128 -128
  54. package/src/talkpilot/calls/__tests__/calls.spec.ts +252 -252
  55. package/src/talkpilot/calls/calls.getters.ts +446 -248
  56. package/src/talkpilot/calls/calls.types.ts +171 -115
  57. package/src/talkpilot/calls/index.ts +2 -2
  58. package/src/talkpilot/clientAudioBuffers/__tests__/clientAudioBuffer.getters.spec.ts +160 -160
  59. package/src/talkpilot/clientAudioBuffers/clientAudioBuffer.getters.ts +117 -117
  60. package/src/talkpilot/clientAudioBuffers/clientsAudioBuffers.types.ts +25 -25
  61. package/src/talkpilot/clientAudioBuffers/index.ts +2 -2
  62. package/src/talkpilot/clients/clients.getters.ts +16 -16
  63. package/src/talkpilot/clients/clients.types.ts +14 -14
  64. package/src/talkpilot/clients/index.ts +2 -2
  65. package/src/talkpilot/clientsConfig/__tests__/clientsConfig.spec.ts +106 -106
  66. package/src/talkpilot/clientsConfig/clientsConfig.getters.ts +44 -22
  67. package/src/talkpilot/clientsConfig/clientsConfig.types.ts +94 -94
  68. package/src/talkpilot/clientsConfig/index.ts +2 -2
  69. package/src/talkpilot/flows/__tests__/flows.schema.spec.ts +67 -67
  70. package/src/talkpilot/flows/flows.getter.ts +14 -14
  71. package/src/talkpilot/flows/flows.schema.ts +153 -153
  72. package/src/talkpilot/flows/flows.types.ts +184 -184
  73. package/src/talkpilot/flows/index.ts +2 -2
  74. package/src/talkpilot/groups/__tests__/groups.spec.ts +90 -90
  75. package/src/talkpilot/groups/__tests__/phone.utils.spec.ts +32 -32
  76. package/src/talkpilot/groups/groups.getters.ts +30 -30
  77. package/src/talkpilot/groups/groups.types.ts +29 -29
  78. package/src/talkpilot/groups/index.ts +3 -3
  79. package/src/talkpilot/groups/phone.utils.ts +46 -46
  80. package/src/talkpilot/index.ts +29 -29
  81. package/src/talkpilot/leads/index.ts +2 -2
  82. package/src/talkpilot/leads/leads.getter.ts +6 -6
  83. package/src/talkpilot/leads/leads.schema.ts +33 -33
  84. package/src/talkpilot/leads/leads.types.ts +20 -20
  85. package/src/talkpilot/mongodb-client.ts +78 -78
  86. package/src/talkpilot/phone_numbers/__tests__/phone_numbers.spec.ts +247 -247
  87. package/src/talkpilot/phone_numbers/index.ts +2 -2
  88. package/src/talkpilot/phone_numbers/phone_numbers.getter.ts +154 -154
  89. package/src/talkpilot/phone_numbers/phone_numbers.schema.ts +17 -17
  90. package/src/talkpilot/phone_numbers/phone_numbers.types.ts +30 -30
  91. package/src/talkpilot/plans/__tests__/plans.spec.ts +70 -70
  92. package/src/talkpilot/plans/index.ts +2 -2
  93. package/src/talkpilot/plans/plans.getters.ts +132 -132
  94. package/src/talkpilot/plans/plans.types.ts +89 -89
  95. package/src/talkpilot/results/index.ts +7 -7
  96. package/src/talkpilot/results/results.getter.ts +35 -35
  97. package/src/talkpilot/results/results.schema.ts +25 -25
  98. package/src/talkpilot/results/results.types.ts +34 -34
  99. package/src/talkpilot/retry_analyze/__tests__/retryAnalyze.getters.spec.ts +156 -156
  100. package/src/talkpilot/retry_analyze/index.ts +2 -2
  101. package/src/talkpilot/retry_analyze/retryAnalyze.getters.ts +75 -75
  102. package/src/talkpilot/retry_analyze/retryAnalyze.types.ts +13 -13
  103. package/src/talkpilot/sessions/__tests__/sessions.spec.ts +147 -147
  104. package/src/talkpilot/sessions/index.ts +2 -2
  105. package/src/talkpilot/sessions/sessions.getter.ts +92 -92
  106. package/src/talkpilot/sessions/sessions.schema.ts +34 -34
  107. package/src/talkpilot/sessions/sessions.types.ts +30 -30
  108. package/src/talkpilot/subscriptions/__tests__/subscriptions.getters.utils.spec.ts +45 -45
  109. package/src/talkpilot/subscriptions/index.ts +3 -3
  110. package/src/talkpilot/subscriptions/subscriptions.getters.ts +146 -146
  111. package/src/talkpilot/subscriptions/subscriptions.getters.utils.ts +33 -33
  112. package/src/talkpilot/subscriptions/subscriptions.types.ts +66 -66
  113. package/src/talkpilot/utils/__tests__/query.utils.spec.ts +49 -49
  114. package/src/talkpilot/utils/query.utils.ts +21 -21
  115. package/src/test-utils/db-utils.ts +24 -24
  116. package/src/test-utils/factories/index.ts +12 -12
  117. package/src/test-utils/factories/municipal/cities.ts +16 -16
  118. package/src/test-utils/factories/municipal/departmentsSubjects.ts +37 -37
  119. package/src/test-utils/factories/municipal/streets.ts +22 -22
  120. package/src/test-utils/factories/municipal/tickets.ts +39 -39
  121. package/src/test-utils/factories/talkpilot/agents.ts +19 -19
  122. package/src/test-utils/factories/talkpilot/calls.ts +37 -37
  123. package/src/test-utils/factories/talkpilot/clientAudioBuffers.ts +20 -20
  124. package/src/test-utils/factories/talkpilot/clientsConfig.ts +18 -18
  125. package/src/test-utils/factories/talkpilot/flows.ts +33 -33
  126. package/src/test-utils/factories/talkpilot/groups.ts +33 -33
  127. package/src/test-utils/factories/talkpilot/phone_numbers.ts +22 -22
  128. package/src/test-utils/factories/talkpilot/sessions.ts +35 -35
  129. package/src/utils/validation.ts +23 -23
  130. package/tsconfig.json +23 -23
@@ -1,184 +1,184 @@
1
- import { ObjectId } from "mongodb";
2
-
3
- /* ---------- Basic helpers ---------- */
4
-
5
- export type Position = { x: number; y: number };
6
- export type HttpMethod = "GET" | "POST" | "PUT" | "PATCH" | "DELETE";
7
-
8
- export type QuestionType = "open" | "multiple" | "table";
9
- export type SelectionType = "single" | "multiple";
10
-
11
- export type Question = {
12
- id: string;
13
- type: QuestionType;
14
- text: string;
15
- repeatAnswer?: boolean;
16
- // For 'multiple' type questions:
17
- answers?: string[] | null;
18
- selectionType?: SelectionType | null; // 'single' = choose one, 'multiple' = choose one or more
19
- // For 'table' type questions:
20
- tableHeader?: string | null;
21
- tableRows?: string[] | null;
22
- tableColumns?: string[] | null; // These are the answer options shared across all rows
23
- // Note: 'table' questions can also have selectionType to indicate if each row allows single or multiple selections
24
- };
25
-
26
- export type NodeKind =
27
- | "start"
28
- | "say"
29
- | "condition"
30
- | "improvise"
31
- | "tool"
32
- | "sendMessage"
33
- | "questionsList"
34
- | "end";
35
- type BaseNode = {
36
- id: string;
37
- type: NodeKind;
38
- text: string;
39
-
40
- name?: string | null;
41
- position: Position;
42
- children?: string[] | null;
43
- successStatus?: string | null;
44
- questions?: Question[] | null;
45
- };
46
-
47
- export type StartNode = BaseNode & { type: "start" };
48
-
49
- export type SayNode = BaseNode & { type: "say" };
50
-
51
- export type ConditionNode = BaseNode & {
52
- type: "condition";
53
- conditions: Array<{
54
- conditionString: string;
55
- nodeType?: string | null;
56
- id?: string | null;
57
- }>;
58
- defaultNodeType?: string | null;
59
- defaultChild?: { id?: string } | null;
60
- };
61
-
62
- export type ImproviseNode = BaseNode & { type: "improvise" };
63
-
64
- export type FlowTool = {
65
- id: string;
66
- name: string;
67
- description?: string;
68
- url: string;
69
- method: HttpMethod;
70
- headers?: Record<string, string>;
71
- parameters: Record<string, unknown>;
72
- bodyFormat?: "json" | "form-data";
73
- preventAudioDuringTool?: boolean;
74
- sendUserAuthToken?: boolean;
75
- runInBackground?: boolean;
76
- backgroundToolType?: "backgroundToolAlways" | "backgroundToolOnce";
77
- backgroundContinuationInstructions?: string;
78
- };
79
-
80
- export type SendMethod = "sms" | "whatsapp";
81
-
82
- export type SendMessageNode = BaseNode & {
83
- type: "sendMessage";
84
- /** Which channel to use. If omitted, decide at runtime (e.g., default to whatsapp). */
85
- send_method: SendMethod;
86
- /** Destination phone (E.164 or templated string like "{{userPhone}}"). */
87
- phone_number: string;
88
- };
89
-
90
- export type ToolNode = BaseNode & {
91
- type: "tool";
92
- name: string;
93
- text: string;
94
- };
95
-
96
- export type QuestionsListNode = BaseNode & {
97
- type: "questionsList";
98
- questions: Question[];
99
- };
100
-
101
- export type EndNode = BaseNode & { type: "end" };
102
-
103
- export type Interaction =
104
- | StartNode
105
- | SayNode
106
- | ConditionNode
107
- | ImproviseNode
108
- | ToolNode
109
- | SendMessageNode
110
- | QuestionsListNode
111
- | EndNode;
112
-
113
- export type Edge = {
114
- id: string;
115
- source: string;
116
- target: string;
117
- label?: string | null;
118
- };
119
-
120
- export type MetaKey = {
121
- key: string;
122
- required?: boolean;
123
- };
124
-
125
- export type UserProperty = {
126
- name: string;
127
- description: string;
128
- };
129
-
130
- export type InterruptionSettings = {
131
- enableInterruptionDetection?: boolean;
132
- interruptionWindowSeconds?: number;
133
- interruptionThresholdSeconds?: number;
134
- interruptionInstruction?: string;
135
- };
136
-
137
- export type SilenceEscalationSettings = {
138
- enableSilenceDetection?: boolean;
139
- firstWarningSilenceSeconds?: number;
140
- firstWarningInstruction?: string;
141
- secondWarningSilenceSeconds?: number;
142
- secondWarningInstruction?: string;
143
- disconnectSilenceSeconds?: number;
144
- silenceTimeoutSeconds?: number;
145
- };
146
-
147
- export type ConversationSettings = {
148
- interruptions?: InterruptionSettings;
149
- silence?: SilenceEscalationSettings;
150
- liveTimeUpdates?: boolean;
151
- };
152
-
153
- /* ---------- The Flow object ---------- */
154
-
155
- export type Flow = {
156
- freeTextContent?: string;
157
- isFreeText?: boolean;
158
- _id?: ObjectId;
159
-
160
- flowName: string;
161
- clientId: string;
162
-
163
- systemInstructions: string;
164
- initialSentence: string;
165
- voice?: string;
166
-
167
- insights: string[];
168
- conversationSettings?: ConversationSettings;
169
-
170
- interactions: Interaction[];
171
-
172
- edges?: Edge[] | null;
173
- metaKeys?: MetaKey[] | null;
174
- userProperties?: UserProperty[] | null;
175
- tools?: FlowTool[];
176
- afterCallEvent?: string;
177
-
178
- // AI provider configuration
179
- aiProvider?: any;
180
-
181
- // Timestamps
182
- createdAt?: Date;
183
- updatedAt?: Date;
184
- };
1
+ import { ObjectId } from "mongodb";
2
+
3
+ /* ---------- Basic helpers ---------- */
4
+
5
+ export type Position = { x: number; y: number };
6
+ export type HttpMethod = "GET" | "POST" | "PUT" | "PATCH" | "DELETE";
7
+
8
+ export type QuestionType = "open" | "multiple" | "table";
9
+ export type SelectionType = "single" | "multiple";
10
+
11
+ export type Question = {
12
+ id: string;
13
+ type: QuestionType;
14
+ text: string;
15
+ repeatAnswer?: boolean;
16
+ // For 'multiple' type questions:
17
+ answers?: string[] | null;
18
+ selectionType?: SelectionType | null; // 'single' = choose one, 'multiple' = choose one or more
19
+ // For 'table' type questions:
20
+ tableHeader?: string | null;
21
+ tableRows?: string[] | null;
22
+ tableColumns?: string[] | null; // These are the answer options shared across all rows
23
+ // Note: 'table' questions can also have selectionType to indicate if each row allows single or multiple selections
24
+ };
25
+
26
+ export type NodeKind =
27
+ | "start"
28
+ | "say"
29
+ | "condition"
30
+ | "improvise"
31
+ | "tool"
32
+ | "sendMessage"
33
+ | "questionsList"
34
+ | "end";
35
+ type BaseNode = {
36
+ id: string;
37
+ type: NodeKind;
38
+ text: string;
39
+
40
+ name?: string | null;
41
+ position: Position;
42
+ children?: string[] | null;
43
+ successStatus?: string | null;
44
+ questions?: Question[] | null;
45
+ };
46
+
47
+ export type StartNode = BaseNode & { type: "start" };
48
+
49
+ export type SayNode = BaseNode & { type: "say" };
50
+
51
+ export type ConditionNode = BaseNode & {
52
+ type: "condition";
53
+ conditions: Array<{
54
+ conditionString: string;
55
+ nodeType?: string | null;
56
+ id?: string | null;
57
+ }>;
58
+ defaultNodeType?: string | null;
59
+ defaultChild?: { id?: string } | null;
60
+ };
61
+
62
+ export type ImproviseNode = BaseNode & { type: "improvise" };
63
+
64
+ export type FlowTool = {
65
+ id: string;
66
+ name: string;
67
+ description?: string;
68
+ url: string;
69
+ method: HttpMethod;
70
+ headers?: Record<string, string>;
71
+ parameters: Record<string, unknown>;
72
+ bodyFormat?: "json" | "form-data";
73
+ preventAudioDuringTool?: boolean;
74
+ sendUserAuthToken?: boolean;
75
+ runInBackground?: boolean;
76
+ backgroundToolType?: "backgroundToolAlways" | "backgroundToolOnce";
77
+ backgroundContinuationInstructions?: string;
78
+ };
79
+
80
+ export type SendMethod = "sms" | "whatsapp";
81
+
82
+ export type SendMessageNode = BaseNode & {
83
+ type: "sendMessage";
84
+ /** Which channel to use. If omitted, decide at runtime (e.g., default to whatsapp). */
85
+ send_method: SendMethod;
86
+ /** Destination phone (E.164 or templated string like "{{userPhone}}"). */
87
+ phone_number: string;
88
+ };
89
+
90
+ export type ToolNode = BaseNode & {
91
+ type: "tool";
92
+ name: string;
93
+ text: string;
94
+ };
95
+
96
+ export type QuestionsListNode = BaseNode & {
97
+ type: "questionsList";
98
+ questions: Question[];
99
+ };
100
+
101
+ export type EndNode = BaseNode & { type: "end" };
102
+
103
+ export type Interaction =
104
+ | StartNode
105
+ | SayNode
106
+ | ConditionNode
107
+ | ImproviseNode
108
+ | ToolNode
109
+ | SendMessageNode
110
+ | QuestionsListNode
111
+ | EndNode;
112
+
113
+ export type Edge = {
114
+ id: string;
115
+ source: string;
116
+ target: string;
117
+ label?: string | null;
118
+ };
119
+
120
+ export type MetaKey = {
121
+ key: string;
122
+ required?: boolean;
123
+ };
124
+
125
+ export type UserProperty = {
126
+ name: string;
127
+ description: string;
128
+ };
129
+
130
+ export type InterruptionSettings = {
131
+ enableInterruptionDetection?: boolean;
132
+ interruptionWindowSeconds?: number;
133
+ interruptionThresholdSeconds?: number;
134
+ interruptionInstruction?: string;
135
+ };
136
+
137
+ export type SilenceEscalationSettings = {
138
+ enableSilenceDetection?: boolean;
139
+ firstWarningSilenceSeconds?: number;
140
+ firstWarningInstruction?: string;
141
+ secondWarningSilenceSeconds?: number;
142
+ secondWarningInstruction?: string;
143
+ disconnectSilenceSeconds?: number;
144
+ silenceTimeoutSeconds?: number;
145
+ };
146
+
147
+ export type ConversationSettings = {
148
+ interruptions?: InterruptionSettings;
149
+ silence?: SilenceEscalationSettings;
150
+ liveTimeUpdates?: boolean;
151
+ };
152
+
153
+ /* ---------- The Flow object ---------- */
154
+
155
+ export type Flow = {
156
+ freeTextContent?: string;
157
+ isFreeText?: boolean;
158
+ _id?: ObjectId;
159
+
160
+ flowName: string;
161
+ clientId: string;
162
+
163
+ systemInstructions: string;
164
+ initialSentence: string;
165
+ voice?: string;
166
+
167
+ insights: string[];
168
+ conversationSettings?: ConversationSettings;
169
+
170
+ interactions: Interaction[];
171
+
172
+ edges?: Edge[] | null;
173
+ metaKeys?: MetaKey[] | null;
174
+ userProperties?: UserProperty[] | null;
175
+ tools?: FlowTool[];
176
+ afterCallEvent?: string;
177
+
178
+ // AI provider configuration
179
+ aiProvider?: any;
180
+
181
+ // Timestamps
182
+ createdAt?: Date;
183
+ updatedAt?: Date;
184
+ };
@@ -1,2 +1,2 @@
1
- export { getFlowsCollection, findFlowById } from "./flows.getter";
2
- export type * from "./flows.types";
1
+ export { getFlowsCollection, findFlowById } from "./flows.getter";
2
+ export type * from "./flows.types";
@@ -1,90 +1,90 @@
1
- import { ObjectId } from "mongodb";
2
- import { faker } from "@faker-js/faker";
3
- import {
4
- getGroupsCollection,
5
- findGroups,
6
- createGroup,
7
- updateGroup,
8
- removeGroup,
9
- } from "../groups.getters";
10
- import { createGroup as createGroupDoc } from "../../../test-utils/factories";
11
-
12
- describe("db.groups", () => {
13
- describe("getGroupsCollection", () => {
14
- it('returns the "groups" collection', () => {
15
- expect(getGroupsCollection().collectionName).toBe("groups");
16
- });
17
- });
18
-
19
- describe("findGroups", () => {
20
- it("returns groups matching the provided filter", async () => {
21
- const clientId = faker.string.uuid();
22
- const g1 = createGroupDoc({ clientId, name: "Group A" });
23
- const g2 = createGroupDoc({ clientId, name: "Group B" });
24
- const gOther = createGroupDoc({
25
- clientId: faker.string.uuid(),
26
- name: "Group C",
27
- });
28
-
29
- await getGroupsCollection().insertMany([g1, g2, gOther]);
30
-
31
- const result = await findGroups({ clientId });
32
- const names = result.map((g) => g.name).sort();
33
-
34
- expect(result).toHaveLength(2);
35
- expect(names).toEqual(["Group A", "Group B"]);
36
- });
37
- });
38
-
39
- describe("createGroup", () => {
40
- it("inserts a group and returns insertedId", async () => {
41
- const toInsert = createGroupDoc({ name: "New Group" });
42
-
43
- const insertedId = await createGroup(toInsert);
44
- expect(insertedId).toBeInstanceOf(ObjectId);
45
-
46
- const fromDb = await getGroupsCollection().findOne({ _id: insertedId });
47
- expect(fromDb?.name).toBe("New Group");
48
- expect(fromDb?.clientId).toBe(toInsert.clientId);
49
- });
50
- });
51
-
52
- describe("updateGroup", () => {
53
- it("updates the group and sets updatedAt, returning the updated document", async () => {
54
- const original = createGroupDoc({
55
- name: "Before",
56
- description: "Old desc",
57
- });
58
- const { insertedId } = await getGroupsCollection().insertOne(original);
59
- const before = await getGroupsCollection().findOne({ _id: insertedId });
60
-
61
- // Add a small delay to ensure updatedAt will be different
62
- await new Promise((resolve) => setTimeout(resolve, 10));
63
-
64
- const res = await updateGroup(
65
- { _id: insertedId },
66
- { name: "After", description: "New desc" },
67
- );
68
-
69
- expect(res?._id).toEqual(insertedId);
70
- expect(res?.name).toBe("After");
71
- expect(res?.description).toBe("New desc");
72
- expect(new Date(res!.updatedAt).getTime()).toBeGreaterThan(
73
- new Date(before!.updatedAt).getTime(),
74
- );
75
- });
76
- });
77
-
78
- describe("removeGroup", () => {
79
- it("removes a group by string id", async () => {
80
- const { insertedId } =
81
- await getGroupsCollection().insertOne(createGroupDoc());
82
-
83
- const result = await removeGroup(insertedId.toHexString());
84
- expect(result.deletedCount).toBe(1);
85
-
86
- const check = await getGroupsCollection().findOne({ _id: insertedId });
87
- expect(check).toBeNull();
88
- });
89
- });
90
- });
1
+ import { ObjectId } from "mongodb";
2
+ import { faker } from "@faker-js/faker";
3
+ import {
4
+ getGroupsCollection,
5
+ findGroups,
6
+ createGroup,
7
+ updateGroup,
8
+ removeGroup,
9
+ } from "../groups.getters";
10
+ import { createGroup as createGroupDoc } from "../../../test-utils/factories";
11
+
12
+ describe("db.groups", () => {
13
+ describe("getGroupsCollection", () => {
14
+ it('returns the "groups" collection', () => {
15
+ expect(getGroupsCollection().collectionName).toBe("groups");
16
+ });
17
+ });
18
+
19
+ describe("findGroups", () => {
20
+ it("returns groups matching the provided filter", async () => {
21
+ const clientId = faker.string.uuid();
22
+ const g1 = createGroupDoc({ clientId, name: "Group A" });
23
+ const g2 = createGroupDoc({ clientId, name: "Group B" });
24
+ const gOther = createGroupDoc({
25
+ clientId: faker.string.uuid(),
26
+ name: "Group C",
27
+ });
28
+
29
+ await getGroupsCollection().insertMany([g1, g2, gOther]);
30
+
31
+ const result = await findGroups({ clientId });
32
+ const names = result.map((g) => g.name).sort();
33
+
34
+ expect(result).toHaveLength(2);
35
+ expect(names).toEqual(["Group A", "Group B"]);
36
+ });
37
+ });
38
+
39
+ describe("createGroup", () => {
40
+ it("inserts a group and returns insertedId", async () => {
41
+ const toInsert = createGroupDoc({ name: "New Group" });
42
+
43
+ const insertedId = await createGroup(toInsert);
44
+ expect(insertedId).toBeInstanceOf(ObjectId);
45
+
46
+ const fromDb = await getGroupsCollection().findOne({ _id: insertedId });
47
+ expect(fromDb?.name).toBe("New Group");
48
+ expect(fromDb?.clientId).toBe(toInsert.clientId);
49
+ });
50
+ });
51
+
52
+ describe("updateGroup", () => {
53
+ it("updates the group and sets updatedAt, returning the updated document", async () => {
54
+ const original = createGroupDoc({
55
+ name: "Before",
56
+ description: "Old desc",
57
+ });
58
+ const { insertedId } = await getGroupsCollection().insertOne(original);
59
+ const before = await getGroupsCollection().findOne({ _id: insertedId });
60
+
61
+ // Add a small delay to ensure updatedAt will be different
62
+ await new Promise((resolve) => setTimeout(resolve, 10));
63
+
64
+ const res = await updateGroup(
65
+ { _id: insertedId },
66
+ { name: "After", description: "New desc" },
67
+ );
68
+
69
+ expect(res?._id).toEqual(insertedId);
70
+ expect(res?.name).toBe("After");
71
+ expect(res?.description).toBe("New desc");
72
+ expect(new Date(res!.updatedAt).getTime()).toBeGreaterThan(
73
+ new Date(before!.updatedAt).getTime(),
74
+ );
75
+ });
76
+ });
77
+
78
+ describe("removeGroup", () => {
79
+ it("removes a group by string id", async () => {
80
+ const { insertedId } =
81
+ await getGroupsCollection().insertOne(createGroupDoc());
82
+
83
+ const result = await removeGroup(insertedId.toHexString());
84
+ expect(result.deletedCount).toBe(1);
85
+
86
+ const check = await getGroupsCollection().findOne({ _id: insertedId });
87
+ expect(check).toBeNull();
88
+ });
89
+ });
90
+ });
@@ -1,32 +1,32 @@
1
- import { enrichPhoneNumber, isValidE164 } from "../phone.utils";
2
-
3
- describe("Phone Utilities", () => {
4
- describe("enrichPhoneNumber", () => {
5
- it("should enrich Israeli number", () => {
6
- const result = enrichPhoneNumber("+972507725874");
7
- expect(result?.e164).toBe("+972507725874");
8
- expect(result?.region).toBeDefined();
9
- expect(result?.countryCallingCode).toBe("972");
10
- });
11
-
12
- it("should enrich US number", () => {
13
- const result = enrichPhoneNumber("+12025551234");
14
- expect(result?.region).toBe("US");
15
- expect(result?.countryCallingCode).toBe("1");
16
- });
17
-
18
- it("should return null for invalid", () => {
19
- expect(enrichPhoneNumber("invalid")).toBeNull();
20
- expect(enrichPhoneNumber("0507725874")).toBeNull();
21
- });
22
- });
23
-
24
- describe("isValidE164", () => {
25
- it("should validate E.164 format", () => {
26
- expect(isValidE164("+972508832983")).toBe(true);
27
- expect(isValidE164("+12025551234")).toBe(true);
28
- expect(isValidE164("0501234567")).toBe(false);
29
- expect(isValidE164("invalid")).toBe(false);
30
- });
31
- });
32
- });
1
+ import { enrichPhoneNumber, isValidE164 } from "../phone.utils";
2
+
3
+ describe("Phone Utilities", () => {
4
+ describe("enrichPhoneNumber", () => {
5
+ it("should enrich Israeli number", () => {
6
+ const result = enrichPhoneNumber("+972507725874");
7
+ expect(result?.e164).toBe("+972507725874");
8
+ expect(result?.region).toBeDefined();
9
+ expect(result?.countryCallingCode).toBe("972");
10
+ });
11
+
12
+ it("should enrich US number", () => {
13
+ const result = enrichPhoneNumber("+12025551234");
14
+ expect(result?.region).toBe("US");
15
+ expect(result?.countryCallingCode).toBe("1");
16
+ });
17
+
18
+ it("should return null for invalid", () => {
19
+ expect(enrichPhoneNumber("invalid")).toBeNull();
20
+ expect(enrichPhoneNumber("0507725874")).toBeNull();
21
+ });
22
+ });
23
+
24
+ describe("isValidE164", () => {
25
+ it("should validate E.164 format", () => {
26
+ expect(isValidE164("+972508832983")).toBe(true);
27
+ expect(isValidE164("+12025551234")).toBe(true);
28
+ expect(isValidE164("0501234567")).toBe(false);
29
+ expect(isValidE164("invalid")).toBe(false);
30
+ });
31
+ });
32
+ });