@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.
- package/.cursor/rules/development.mdc +65 -65
- package/DEVELOPMENT.md +98 -98
- package/README.md +160 -160
- package/dist/talkpilot/calls/calls.getters.d.ts +1 -2
- package/dist/talkpilot/calls/calls.getters.d.ts.map +1 -1
- package/dist/talkpilot/calls/calls.getters.js +0 -176
- package/dist/talkpilot/calls/calls.getters.js.map +1 -1
- package/dist/talkpilot/calls/calls.types.d.ts +0 -48
- package/dist/talkpilot/calls/calls.types.d.ts.map +1 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts +0 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts.map +1 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.js +0 -13
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.js.map +1 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.types.d.ts +16 -8
- package/dist/talkpilot/clientsConfig/clientsConfig.types.d.ts.map +1 -1
- package/jest.config.js +19 -19
- package/package.json +45 -45
- package/src/__tests__/setup.ts +20 -20
- package/src/connection.ts +42 -42
- package/src/index.ts +16 -16
- package/src/municipal/__tests__/validation.spec.ts +62 -62
- package/src/municipal/cities/cities.getters.ts +50 -50
- package/src/municipal/cities/cities.types.ts +11 -11
- package/src/municipal/cities/index.ts +2 -2
- package/src/municipal/departmentsSubjects/departmentsSubjects.getters.ts +282 -282
- package/src/municipal/departmentsSubjects/departmentsSubjects.types.ts +72 -72
- package/src/municipal/departmentsSubjects/index.ts +9 -9
- package/src/municipal/index.ts +21 -21
- package/src/municipal/mongodb-client.ts +61 -61
- package/src/municipal/streets/index.ts +2 -2
- package/src/municipal/streets/streets.getters.ts +125 -125
- package/src/municipal/streets/streets.types.ts +18 -18
- package/src/municipal/systemInstructions/__tests__/getters.spec.ts +113 -113
- package/src/municipal/systemInstructions/__tests__/setters.spec.ts +274 -274
- package/src/municipal/systemInstructions/index.ts +7 -7
- package/src/municipal/systemInstructions/instructions.getters.ts +57 -57
- package/src/municipal/systemInstructions/instructions.setters.ts +119 -119
- package/src/municipal/systemInstructions/instructions.types.ts +30 -30
- package/src/municipal/tickets/__tests__/tickets.getters.spec.ts +66 -66
- package/src/municipal/tickets/index.ts +2 -2
- package/src/municipal/tickets/tickets.getters.ts +261 -261
- package/src/municipal/tickets/tickets.types.ts +43 -43
- package/src/municipal/utils/types.ts +11 -11
- package/src/talkpilot/__tests__/db.spec.ts +38 -38
- package/src/talkpilot/__tests__/mongodb-client.spec.ts +18 -18
- package/src/talkpilot/__tests__/validation.spec.ts +68 -68
- package/src/talkpilot/agents/__tests__/agents.getters.spec.ts +29 -29
- package/src/talkpilot/agents/agents.getters.ts +34 -34
- package/src/talkpilot/agents/agents.types.ts +14 -14
- package/src/talkpilot/agents/index.ts +2 -2
- package/src/talkpilot/backgroundToolResults/__tests__/backgroundToolResults.getters.spec.ts +147 -147
- package/src/talkpilot/backgroundToolResults/backgroundToolResults.getters.ts +65 -65
- package/src/talkpilot/backgroundToolResults/backgroundToolResults.types.ts +23 -23
- package/src/talkpilot/backgroundToolResults/index.ts +2 -2
- package/src/talkpilot/calls/__tests__/callStats.utils.spec.ts +128 -128
- package/src/talkpilot/calls/__tests__/calls.spec.ts +252 -252
- package/src/talkpilot/calls/calls.getters.ts +248 -446
- package/src/talkpilot/calls/calls.types.ts +115 -171
- package/src/talkpilot/calls/index.ts +2 -2
- package/src/talkpilot/clientAudioBuffers/__tests__/clientAudioBuffer.getters.spec.ts +160 -160
- package/src/talkpilot/clientAudioBuffers/clientAudioBuffer.getters.ts +117 -117
- package/src/talkpilot/clientAudioBuffers/clientsAudioBuffers.types.ts +25 -25
- package/src/talkpilot/clientAudioBuffers/index.ts +2 -2
- package/src/talkpilot/clients/clients.getters.ts +16 -16
- package/src/talkpilot/clients/clients.types.ts +14 -14
- package/src/talkpilot/clients/index.ts +2 -2
- package/src/talkpilot/clientsConfig/__tests__/clientsConfig.spec.ts +187 -106
- package/src/talkpilot/clientsConfig/clientsConfig.getters.ts +22 -44
- package/src/talkpilot/clientsConfig/clientsConfig.types.ts +119 -94
- package/src/talkpilot/clientsConfig/index.ts +2 -2
- package/src/talkpilot/flows/__tests__/flows.schema.spec.ts +67 -67
- package/src/talkpilot/flows/flows.getter.ts +14 -14
- package/src/talkpilot/flows/flows.schema.ts +153 -153
- package/src/talkpilot/flows/flows.types.ts +184 -184
- package/src/talkpilot/flows/index.ts +2 -2
- package/src/talkpilot/groups/__tests__/groups.spec.ts +90 -90
- package/src/talkpilot/groups/__tests__/phone.utils.spec.ts +32 -32
- package/src/talkpilot/groups/groups.getters.ts +30 -30
- package/src/talkpilot/groups/groups.types.ts +29 -29
- package/src/talkpilot/groups/index.ts +3 -3
- package/src/talkpilot/groups/phone.utils.ts +46 -46
- package/src/talkpilot/index.ts +29 -29
- package/src/talkpilot/leads/index.ts +2 -2
- package/src/talkpilot/leads/leads.getter.ts +6 -6
- package/src/talkpilot/leads/leads.schema.ts +33 -33
- package/src/talkpilot/leads/leads.types.ts +20 -20
- package/src/talkpilot/mongodb-client.ts +78 -78
- package/src/talkpilot/phone_numbers/__tests__/phone_numbers.spec.ts +247 -247
- package/src/talkpilot/phone_numbers/index.ts +2 -2
- package/src/talkpilot/phone_numbers/phone_numbers.getter.ts +154 -154
- package/src/talkpilot/phone_numbers/phone_numbers.schema.ts +17 -17
- package/src/talkpilot/phone_numbers/phone_numbers.types.ts +30 -30
- package/src/talkpilot/plans/__tests__/plans.spec.ts +70 -70
- package/src/talkpilot/plans/index.ts +2 -2
- package/src/talkpilot/plans/plans.getters.ts +132 -132
- package/src/talkpilot/plans/plans.types.ts +89 -89
- package/src/talkpilot/results/index.ts +7 -7
- package/src/talkpilot/results/results.getter.ts +35 -35
- package/src/talkpilot/results/results.schema.ts +25 -25
- package/src/talkpilot/results/results.types.ts +34 -34
- package/src/talkpilot/retry_analyze/__tests__/retryAnalyze.getters.spec.ts +156 -156
- package/src/talkpilot/retry_analyze/index.ts +2 -2
- package/src/talkpilot/retry_analyze/retryAnalyze.getters.ts +75 -75
- package/src/talkpilot/retry_analyze/retryAnalyze.types.ts +13 -13
- package/src/talkpilot/sessions/__tests__/sessions.spec.ts +147 -147
- package/src/talkpilot/sessions/index.ts +2 -2
- package/src/talkpilot/sessions/sessions.getter.ts +92 -92
- package/src/talkpilot/sessions/sessions.schema.ts +34 -34
- package/src/talkpilot/sessions/sessions.types.ts +30 -30
- package/src/talkpilot/subscriptions/__tests__/subscriptions.getters.utils.spec.ts +45 -45
- package/src/talkpilot/subscriptions/index.ts +3 -3
- package/src/talkpilot/subscriptions/subscriptions.getters.ts +146 -146
- package/src/talkpilot/subscriptions/subscriptions.getters.utils.ts +33 -33
- package/src/talkpilot/subscriptions/subscriptions.types.ts +66 -66
- package/src/talkpilot/utils/__tests__/query.utils.spec.ts +49 -49
- package/src/talkpilot/utils/query.utils.ts +21 -21
- package/src/test-utils/db-utils.ts +24 -24
- package/src/test-utils/factories/index.ts +12 -12
- package/src/test-utils/factories/municipal/cities.ts +16 -16
- package/src/test-utils/factories/municipal/departmentsSubjects.ts +37 -37
- package/src/test-utils/factories/municipal/streets.ts +22 -22
- package/src/test-utils/factories/municipal/tickets.ts +39 -39
- package/src/test-utils/factories/talkpilot/agents.ts +19 -19
- package/src/test-utils/factories/talkpilot/calls.ts +37 -37
- package/src/test-utils/factories/talkpilot/clientAudioBuffers.ts +20 -20
- package/src/test-utils/factories/talkpilot/clientsConfig.ts +18 -18
- package/src/test-utils/factories/talkpilot/flows.ts +33 -33
- package/src/test-utils/factories/talkpilot/groups.ts +33 -33
- package/src/test-utils/factories/talkpilot/phone_numbers.ts +22 -22
- package/src/test-utils/factories/talkpilot/sessions.ts +35 -35
- package/src/utils/validation.ts +23 -23
- 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
|
+
};
|