@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,282 +1,282 @@
1
- import { CityName, DepartmentSubject, getDb } from "../index";
2
- import { Collection, Filter } from "mongodb";
3
- import { VectorSearchResult } from "./departmentsSubjects.types";
4
-
5
- export const getDepartmentsSubjectsCollection =
6
- (): Collection<DepartmentSubject> => {
7
- return getDb().collection<DepartmentSubject>("departmentsSubjects");
8
- };
9
-
10
- /**
11
- * Get all departments subjects for a given city.
12
- * No DB sort; tree build does not depend on order. Callers that need alphabetical order
13
- * sort in memory (e.g. sortTreeAlphabetically in full-tree-subjects.service).
14
- * @param cityName - City name to filter subjects
15
- * @param withEmbeddings - Whether to include embedding and embedding_model fields (default: false)
16
- * @returns Array of all department subjects for the city
17
- */
18
- export const getAllDepartmentsSubjectsByCity = async (
19
- cityName: CityName,
20
- withEmbeddings: boolean = false,
21
- ): Promise<DepartmentSubject[]> => {
22
- const collection = getDepartmentsSubjectsCollection();
23
- const query = collection.find({ cityName });
24
-
25
- if (!withEmbeddings) {
26
- return (await query
27
- .project({ embedding: 0, embedding_model: 0 })
28
- .toArray()) as DepartmentSubject[];
29
- }
30
-
31
- return await query.toArray();
32
- };
33
-
34
- /**
35
- * Perform vector search on departmentsSubjects collection
36
- * @param queryVector - The embedding vector to search with
37
- * @param cityName - City name to filter results
38
- * @param limit - Maximum number of results (default: 5)
39
- * @param numCandidates - Number of candidates to consider (default: 800)
40
- * @returns Array of vector search results
41
- */
42
- export const vectorSearchDepartmentsSubjects = async (
43
- queryVector: number[],
44
- cityName: CityName,
45
- limit: number = 5,
46
- numCandidates: number = 800,
47
- ): Promise<VectorSearchResult[]> => {
48
- const collection = getDepartmentsSubjectsCollection();
49
- const indexName = "subjectTreesIndex";
50
-
51
- return await collection
52
- .aggregate<VectorSearchResult>([
53
- {
54
- $vectorSearch: {
55
- index: indexName,
56
- path: "embedding",
57
- queryVector,
58
- numCandidates,
59
- limit,
60
- filter: { cityName },
61
- },
62
- },
63
- {
64
- $project: {
65
- _id: 0,
66
- subjectName: 1,
67
- subject_id: { $toInt: "$subject_id" }, // Convert string to number
68
- sub_subject_name: 1,
69
- sub_subject_id: 1,
70
- descriptions: 1,
71
- guidelines: 1,
72
- score: { $meta: "vectorSearchScore" },
73
- },
74
- },
75
- ])
76
- .toArray();
77
- };
78
-
79
- export const addDepartmentSubjectInstruction = async (
80
- filter: Filter<DepartmentSubject>,
81
- instructionText: string,
82
- ): Promise<DepartmentSubject | null> => {
83
- const newInstruction = {
84
- id: Date.now(),
85
- instruction: instructionText,
86
- };
87
-
88
- const result = await getDepartmentsSubjectsCollection().findOneAndUpdate(
89
- filter,
90
- {
91
- $push: { instructions: newInstruction },
92
- $set: { updatedAt: new Date() },
93
- },
94
- { returnDocument: "after" },
95
- );
96
-
97
- return result || null;
98
- };
99
-
100
- export const updateDepartmentSubjectInstruction = async (
101
- filter: Filter<DepartmentSubject>,
102
- instructionId: number,
103
- newInstructionText: string,
104
- ): Promise<DepartmentSubject | null> => {
105
- const result = await getDepartmentsSubjectsCollection().findOneAndUpdate(
106
- { ...filter, "instructions.id": instructionId },
107
- {
108
- $set: {
109
- "instructions.$.instruction": newInstructionText,
110
- updatedAt: new Date(),
111
- },
112
- },
113
- { returnDocument: "after" },
114
- );
115
-
116
- return result || null;
117
- };
118
-
119
- export const removeDepartmentSubjectInstruction = async (
120
- filter: Filter<DepartmentSubject>,
121
- instructionId: number,
122
- ): Promise<DepartmentSubject | null> => {
123
- const result = await getDepartmentsSubjectsCollection().findOneAndUpdate(
124
- filter,
125
- {
126
- $pull: { instructions: { id: instructionId } },
127
- $set: { updatedAt: new Date() },
128
- },
129
- { returnDocument: "after" },
130
- );
131
-
132
- return result || null;
133
- };
134
-
135
- export const addDepartmentSubjectCommunication = async (
136
- filter: Filter<DepartmentSubject>,
137
- communication: {
138
- type: "free_text_sms" | "upload_url_sms";
139
- message_text?: string;
140
- template_id?: string;
141
- to?: string;
142
- },
143
- ): Promise<DepartmentSubject | null> => {
144
- const newCommunication = {
145
- id: Date.now(),
146
- type: communication.type,
147
- ...(communication.message_text !== undefined && {
148
- message_text: communication.message_text,
149
- }),
150
- is_deleted: false,
151
- ...(communication.template_id !== undefined && {
152
- template_id: communication.template_id,
153
- }),
154
- ...(communication.to !== undefined && { to: communication.to }),
155
- };
156
-
157
- const result = await getDepartmentsSubjectsCollection().findOneAndUpdate(
158
- filter,
159
- {
160
- $push: { communications: newCommunication },
161
- $set: { updatedAt: new Date() },
162
- },
163
- { returnDocument: "after" },
164
- );
165
-
166
- return result || null;
167
- };
168
-
169
- export const updateDepartmentSubjectCommunication = async (
170
- filter: Filter<DepartmentSubject>,
171
- communicationId: number,
172
- updates: {
173
- type?: "free_text_sms" | "upload_url_sms";
174
- message_text?: string | null;
175
- template_id?: string | null;
176
- to?: string | null;
177
- is_deleted?: boolean | null;
178
- },
179
- ): Promise<DepartmentSubject | null> => {
180
- const setFields: Record<string, unknown> = { updatedAt: new Date() };
181
- if (updates.type !== undefined)
182
- setFields["communications.$.type"] = updates.type;
183
- if (updates.message_text !== undefined)
184
- setFields["communications.$.message_text"] = updates.message_text;
185
- if (updates.template_id !== undefined)
186
- setFields["communications.$.template_id"] = updates.template_id;
187
- if (updates.to !== undefined) setFields["communications.$.to"] = updates.to;
188
- if (updates.is_deleted !== undefined)
189
- setFields["communications.$.is_deleted"] = updates.is_deleted;
190
-
191
- const result = await getDepartmentsSubjectsCollection().findOneAndUpdate(
192
- { ...filter, "communications.id": communicationId },
193
- { $set: setFields },
194
- { returnDocument: "after" },
195
- );
196
-
197
- return result || null;
198
- };
199
-
200
- export const removeDepartmentSubjectCommunication = async (
201
- filter: Filter<DepartmentSubject>,
202
- communicationId: number,
203
- ): Promise<DepartmentSubject | null> => {
204
- const result = await getDepartmentsSubjectsCollection().findOneAndUpdate(
205
- filter,
206
- {
207
- $pull: { communications: { id: communicationId } },
208
- $set: { updatedAt: new Date() },
209
- },
210
- { returnDocument: "after" },
211
- );
212
-
213
- return result || null;
214
- };
215
-
216
- /**
217
- * Get department subject by subject_id, sub_subject_id, and cityName
218
- * @param subjectId - Subject ID (from event_subject_id)
219
- * @param subSubjectId - Sub-subject ID (from event_sub_subject_id, optional)
220
- * @param cityName - City name to filter results
221
- * @returns DepartmentSubject document or null if not found
222
- */
223
- export const getDepartmentSubjectBySubjectIds = async (
224
- subjectId: string,
225
- subSubjectId: string,
226
- cityName: CityName,
227
- ): Promise<DepartmentSubject | null> => {
228
- const collection = getDepartmentsSubjectsCollection();
229
-
230
- const filter: Filter<DepartmentSubject> = {
231
- subject_id: subjectId,
232
- cityName,
233
- };
234
-
235
- if (subSubjectId) {
236
- filter.sub_subject_id = subSubjectId;
237
- }
238
-
239
- const result = await collection.findOne(filter);
240
-
241
- return result || null;
242
- };
243
-
244
- /**
245
- * Update or create guidelines field for a department subject
246
- * @param filter - MongoDB filter to find the document
247
- * @param guidelines - Guidelines text to set
248
- * @returns Updated DepartmentSubject document or null if not found
249
- */
250
- export const updateDepartmentSubjectGuidelines = async (
251
- filter: Filter<DepartmentSubject>,
252
- guidelines: string,
253
- ): Promise<DepartmentSubject | null> => {
254
- // $set will create the field if it doesn't exist
255
- const result = await getDepartmentsSubjectsCollection().findOneAndUpdate(
256
- filter,
257
- { $set: { guidelines, updatedAt: new Date() } },
258
- { returnDocument: "after" },
259
- );
260
- return result || null;
261
- };
262
-
263
- /**
264
- * Clear guidelines field from a department subject document (sets it to null)
265
- * @param filter - MongoDB filter to find the document
266
- * @returns Updated DepartmentSubject document or null if not found
267
- */
268
- export const deleteDepartmentSubjectGuidelines = async (
269
- filter: Filter<DepartmentSubject>,
270
- ): Promise<DepartmentSubject | null> => {
271
- // $set clears the field by setting it to null (keeps the field in the document)
272
- const result = await getDepartmentsSubjectsCollection().findOneAndUpdate(
273
- filter,
274
- { $set: { guidelines: null, updatedAt: new Date() } },
275
- { returnDocument: "after" },
276
- );
277
- return result || null;
278
- };
279
-
280
- export const findSubjectByQuery = async (query: Partial<DepartmentSubject>) => {
281
- return await getDepartmentsSubjectsCollection().findOne(query);
282
- };
1
+ import { CityName, DepartmentSubject, getDb } from "../index";
2
+ import { Collection, Filter } from "mongodb";
3
+ import { VectorSearchResult } from "./departmentsSubjects.types";
4
+
5
+ export const getDepartmentsSubjectsCollection =
6
+ (): Collection<DepartmentSubject> => {
7
+ return getDb().collection<DepartmentSubject>("departmentsSubjects");
8
+ };
9
+
10
+ /**
11
+ * Get all departments subjects for a given city.
12
+ * No DB sort; tree build does not depend on order. Callers that need alphabetical order
13
+ * sort in memory (e.g. sortTreeAlphabetically in full-tree-subjects.service).
14
+ * @param cityName - City name to filter subjects
15
+ * @param withEmbeddings - Whether to include embedding and embedding_model fields (default: false)
16
+ * @returns Array of all department subjects for the city
17
+ */
18
+ export const getAllDepartmentsSubjectsByCity = async (
19
+ cityName: CityName,
20
+ withEmbeddings: boolean = false,
21
+ ): Promise<DepartmentSubject[]> => {
22
+ const collection = getDepartmentsSubjectsCollection();
23
+ const query = collection.find({ cityName });
24
+
25
+ if (!withEmbeddings) {
26
+ return (await query
27
+ .project({ embedding: 0, embedding_model: 0 })
28
+ .toArray()) as DepartmentSubject[];
29
+ }
30
+
31
+ return await query.toArray();
32
+ };
33
+
34
+ /**
35
+ * Perform vector search on departmentsSubjects collection
36
+ * @param queryVector - The embedding vector to search with
37
+ * @param cityName - City name to filter results
38
+ * @param limit - Maximum number of results (default: 5)
39
+ * @param numCandidates - Number of candidates to consider (default: 800)
40
+ * @returns Array of vector search results
41
+ */
42
+ export const vectorSearchDepartmentsSubjects = async (
43
+ queryVector: number[],
44
+ cityName: CityName,
45
+ limit: number = 5,
46
+ numCandidates: number = 800,
47
+ ): Promise<VectorSearchResult[]> => {
48
+ const collection = getDepartmentsSubjectsCollection();
49
+ const indexName = "subjectTreesIndex";
50
+
51
+ return await collection
52
+ .aggregate<VectorSearchResult>([
53
+ {
54
+ $vectorSearch: {
55
+ index: indexName,
56
+ path: "embedding",
57
+ queryVector,
58
+ numCandidates,
59
+ limit,
60
+ filter: { cityName },
61
+ },
62
+ },
63
+ {
64
+ $project: {
65
+ _id: 0,
66
+ subjectName: 1,
67
+ subject_id: { $toInt: "$subject_id" }, // Convert string to number
68
+ sub_subject_name: 1,
69
+ sub_subject_id: 1,
70
+ descriptions: 1,
71
+ guidelines: 1,
72
+ score: { $meta: "vectorSearchScore" },
73
+ },
74
+ },
75
+ ])
76
+ .toArray();
77
+ };
78
+
79
+ export const addDepartmentSubjectInstruction = async (
80
+ filter: Filter<DepartmentSubject>,
81
+ instructionText: string,
82
+ ): Promise<DepartmentSubject | null> => {
83
+ const newInstruction = {
84
+ id: Date.now(),
85
+ instruction: instructionText,
86
+ };
87
+
88
+ const result = await getDepartmentsSubjectsCollection().findOneAndUpdate(
89
+ filter,
90
+ {
91
+ $push: { instructions: newInstruction },
92
+ $set: { updatedAt: new Date() },
93
+ },
94
+ { returnDocument: "after" },
95
+ );
96
+
97
+ return result || null;
98
+ };
99
+
100
+ export const updateDepartmentSubjectInstruction = async (
101
+ filter: Filter<DepartmentSubject>,
102
+ instructionId: number,
103
+ newInstructionText: string,
104
+ ): Promise<DepartmentSubject | null> => {
105
+ const result = await getDepartmentsSubjectsCollection().findOneAndUpdate(
106
+ { ...filter, "instructions.id": instructionId },
107
+ {
108
+ $set: {
109
+ "instructions.$.instruction": newInstructionText,
110
+ updatedAt: new Date(),
111
+ },
112
+ },
113
+ { returnDocument: "after" },
114
+ );
115
+
116
+ return result || null;
117
+ };
118
+
119
+ export const removeDepartmentSubjectInstruction = async (
120
+ filter: Filter<DepartmentSubject>,
121
+ instructionId: number,
122
+ ): Promise<DepartmentSubject | null> => {
123
+ const result = await getDepartmentsSubjectsCollection().findOneAndUpdate(
124
+ filter,
125
+ {
126
+ $pull: { instructions: { id: instructionId } },
127
+ $set: { updatedAt: new Date() },
128
+ },
129
+ { returnDocument: "after" },
130
+ );
131
+
132
+ return result || null;
133
+ };
134
+
135
+ export const addDepartmentSubjectCommunication = async (
136
+ filter: Filter<DepartmentSubject>,
137
+ communication: {
138
+ type: "free_text_sms" | "upload_url_sms";
139
+ message_text?: string;
140
+ template_id?: string;
141
+ to?: string;
142
+ },
143
+ ): Promise<DepartmentSubject | null> => {
144
+ const newCommunication = {
145
+ id: Date.now(),
146
+ type: communication.type,
147
+ ...(communication.message_text !== undefined && {
148
+ message_text: communication.message_text,
149
+ }),
150
+ is_deleted: false,
151
+ ...(communication.template_id !== undefined && {
152
+ template_id: communication.template_id,
153
+ }),
154
+ ...(communication.to !== undefined && { to: communication.to }),
155
+ };
156
+
157
+ const result = await getDepartmentsSubjectsCollection().findOneAndUpdate(
158
+ filter,
159
+ {
160
+ $push: { communications: newCommunication },
161
+ $set: { updatedAt: new Date() },
162
+ },
163
+ { returnDocument: "after" },
164
+ );
165
+
166
+ return result || null;
167
+ };
168
+
169
+ export const updateDepartmentSubjectCommunication = async (
170
+ filter: Filter<DepartmentSubject>,
171
+ communicationId: number,
172
+ updates: {
173
+ type?: "free_text_sms" | "upload_url_sms";
174
+ message_text?: string | null;
175
+ template_id?: string | null;
176
+ to?: string | null;
177
+ is_deleted?: boolean | null;
178
+ },
179
+ ): Promise<DepartmentSubject | null> => {
180
+ const setFields: Record<string, unknown> = { updatedAt: new Date() };
181
+ if (updates.type !== undefined)
182
+ setFields["communications.$.type"] = updates.type;
183
+ if (updates.message_text !== undefined)
184
+ setFields["communications.$.message_text"] = updates.message_text;
185
+ if (updates.template_id !== undefined)
186
+ setFields["communications.$.template_id"] = updates.template_id;
187
+ if (updates.to !== undefined) setFields["communications.$.to"] = updates.to;
188
+ if (updates.is_deleted !== undefined)
189
+ setFields["communications.$.is_deleted"] = updates.is_deleted;
190
+
191
+ const result = await getDepartmentsSubjectsCollection().findOneAndUpdate(
192
+ { ...filter, "communications.id": communicationId },
193
+ { $set: setFields },
194
+ { returnDocument: "after" },
195
+ );
196
+
197
+ return result || null;
198
+ };
199
+
200
+ export const removeDepartmentSubjectCommunication = async (
201
+ filter: Filter<DepartmentSubject>,
202
+ communicationId: number,
203
+ ): Promise<DepartmentSubject | null> => {
204
+ const result = await getDepartmentsSubjectsCollection().findOneAndUpdate(
205
+ filter,
206
+ {
207
+ $pull: { communications: { id: communicationId } },
208
+ $set: { updatedAt: new Date() },
209
+ },
210
+ { returnDocument: "after" },
211
+ );
212
+
213
+ return result || null;
214
+ };
215
+
216
+ /**
217
+ * Get department subject by subject_id, sub_subject_id, and cityName
218
+ * @param subjectId - Subject ID (from event_subject_id)
219
+ * @param subSubjectId - Sub-subject ID (from event_sub_subject_id, optional)
220
+ * @param cityName - City name to filter results
221
+ * @returns DepartmentSubject document or null if not found
222
+ */
223
+ export const getDepartmentSubjectBySubjectIds = async (
224
+ subjectId: string,
225
+ subSubjectId: string,
226
+ cityName: CityName,
227
+ ): Promise<DepartmentSubject | null> => {
228
+ const collection = getDepartmentsSubjectsCollection();
229
+
230
+ const filter: Filter<DepartmentSubject> = {
231
+ subject_id: subjectId,
232
+ cityName,
233
+ };
234
+
235
+ if (subSubjectId) {
236
+ filter.sub_subject_id = subSubjectId;
237
+ }
238
+
239
+ const result = await collection.findOne(filter);
240
+
241
+ return result || null;
242
+ };
243
+
244
+ /**
245
+ * Update or create guidelines field for a department subject
246
+ * @param filter - MongoDB filter to find the document
247
+ * @param guidelines - Guidelines text to set
248
+ * @returns Updated DepartmentSubject document or null if not found
249
+ */
250
+ export const updateDepartmentSubjectGuidelines = async (
251
+ filter: Filter<DepartmentSubject>,
252
+ guidelines: string,
253
+ ): Promise<DepartmentSubject | null> => {
254
+ // $set will create the field if it doesn't exist
255
+ const result = await getDepartmentsSubjectsCollection().findOneAndUpdate(
256
+ filter,
257
+ { $set: { guidelines, updatedAt: new Date() } },
258
+ { returnDocument: "after" },
259
+ );
260
+ return result || null;
261
+ };
262
+
263
+ /**
264
+ * Clear guidelines field from a department subject document (sets it to null)
265
+ * @param filter - MongoDB filter to find the document
266
+ * @returns Updated DepartmentSubject document or null if not found
267
+ */
268
+ export const deleteDepartmentSubjectGuidelines = async (
269
+ filter: Filter<DepartmentSubject>,
270
+ ): Promise<DepartmentSubject | null> => {
271
+ // $set clears the field by setting it to null (keeps the field in the document)
272
+ const result = await getDepartmentsSubjectsCollection().findOneAndUpdate(
273
+ filter,
274
+ { $set: { guidelines: null, updatedAt: new Date() } },
275
+ { returnDocument: "after" },
276
+ );
277
+ return result || null;
278
+ };
279
+
280
+ export const findSubjectByQuery = async (query: Partial<DepartmentSubject>) => {
281
+ return await getDepartmentsSubjectsCollection().findOne(query);
282
+ };