@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,72 +1,72 @@
|
|
|
1
|
-
import { ObjectId, WithId } from "mongodb";
|
|
2
|
-
import { Request } from "express";
|
|
3
|
-
import { ClientConfigDoc, Products } from "../../talkpilot";
|
|
4
|
-
import { CityName } from "../utils/types";
|
|
5
|
-
|
|
6
|
-
export type DepartmentSubject = {
|
|
7
|
-
_id: ObjectId;
|
|
8
|
-
createdAt: Date;
|
|
9
|
-
updatedAt: Date;
|
|
10
|
-
subjectName: string; // e.g. "אחזקה- איטום"
|
|
11
|
-
subject_id: string; // e.g. "6121"
|
|
12
|
-
sub_subject_name: string; // e.g. "איטום"
|
|
13
|
-
sub_subject_id: string; // e.g. "6123"
|
|
14
|
-
sub_subject_name2?: string; // e.g. "ניקיון באגם" (level 3 - grandchild)
|
|
15
|
-
sub_subject_id2?: string; // e.g. "2898" (level 3 - grandchild)
|
|
16
|
-
descriptions: string[]; // array of guideline / example descriptions
|
|
17
|
-
embedding_text?: string; // the text used to create the embedding
|
|
18
|
-
embedding?: number[]; // length should match embedding_dims (e.g. 1536)
|
|
19
|
-
embedding_model?: string; // e.g. "text-embedding-3-small"
|
|
20
|
-
embedding_dims?: number; // e.g. 1536
|
|
21
|
-
cityName: CityName;
|
|
22
|
-
guidelines?: string | null;
|
|
23
|
-
instructions?: Instruction[];
|
|
24
|
-
communications?: Communication[];
|
|
25
|
-
sla?: number;
|
|
26
|
-
};
|
|
27
|
-
|
|
28
|
-
export interface Instruction {
|
|
29
|
-
id: number;
|
|
30
|
-
instruction: string;
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
export interface Communication {
|
|
34
|
-
id: number;
|
|
35
|
-
type: CommunicationType;
|
|
36
|
-
message_text?: string;
|
|
37
|
-
template_id?: string;
|
|
38
|
-
to?: string;
|
|
39
|
-
is_deleted?: boolean;
|
|
40
|
-
}
|
|
41
|
-
|
|
42
|
-
export type DepartmentSubjectDoc = WithId<DepartmentSubject>;
|
|
43
|
-
|
|
44
|
-
/**
|
|
45
|
-
* Vector search result from MongoDB
|
|
46
|
-
*/
|
|
47
|
-
export type VectorSearchResult = {
|
|
48
|
-
subjectName: string;
|
|
49
|
-
subject_id: number;
|
|
50
|
-
sub_subject_id: string;
|
|
51
|
-
sub_subject_id2?: string | null;
|
|
52
|
-
sub_subject_name: string;
|
|
53
|
-
sub_subject_name2?: string | null;
|
|
54
|
-
descriptions?: string[];
|
|
55
|
-
guidelines?: string | null;
|
|
56
|
-
instructions?: string | null;
|
|
57
|
-
score?: number;
|
|
58
|
-
};
|
|
59
|
-
|
|
60
|
-
export type CommunicationType = "free_text_sms" | "upload_url_sms";
|
|
61
|
-
|
|
62
|
-
/**
|
|
63
|
-
* Extended Express Request type with client configuration
|
|
64
|
-
*/
|
|
65
|
-
export type RequestWithClientConfig = Request & {
|
|
66
|
-
uid: string;
|
|
67
|
-
clientConfig: ClientConfigDoc<Products>;
|
|
68
|
-
isGuest: boolean;
|
|
69
|
-
claims: Record<string, unknown>;
|
|
70
|
-
crmName?: string;
|
|
71
|
-
callSid?: string;
|
|
72
|
-
};
|
|
1
|
+
import { ObjectId, WithId } from "mongodb";
|
|
2
|
+
import { Request } from "express";
|
|
3
|
+
import { ClientConfigDoc, Products } from "../../talkpilot";
|
|
4
|
+
import { CityName } from "../utils/types";
|
|
5
|
+
|
|
6
|
+
export type DepartmentSubject = {
|
|
7
|
+
_id: ObjectId;
|
|
8
|
+
createdAt: Date;
|
|
9
|
+
updatedAt: Date;
|
|
10
|
+
subjectName: string; // e.g. "אחזקה- איטום"
|
|
11
|
+
subject_id: string; // e.g. "6121"
|
|
12
|
+
sub_subject_name: string; // e.g. "איטום"
|
|
13
|
+
sub_subject_id: string; // e.g. "6123"
|
|
14
|
+
sub_subject_name2?: string; // e.g. "ניקיון באגם" (level 3 - grandchild)
|
|
15
|
+
sub_subject_id2?: string; // e.g. "2898" (level 3 - grandchild)
|
|
16
|
+
descriptions: string[]; // array of guideline / example descriptions
|
|
17
|
+
embedding_text?: string; // the text used to create the embedding
|
|
18
|
+
embedding?: number[]; // length should match embedding_dims (e.g. 1536)
|
|
19
|
+
embedding_model?: string; // e.g. "text-embedding-3-small"
|
|
20
|
+
embedding_dims?: number; // e.g. 1536
|
|
21
|
+
cityName: CityName;
|
|
22
|
+
guidelines?: string | null;
|
|
23
|
+
instructions?: Instruction[];
|
|
24
|
+
communications?: Communication[];
|
|
25
|
+
sla?: number;
|
|
26
|
+
};
|
|
27
|
+
|
|
28
|
+
export interface Instruction {
|
|
29
|
+
id: number;
|
|
30
|
+
instruction: string;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
export interface Communication {
|
|
34
|
+
id: number;
|
|
35
|
+
type: CommunicationType;
|
|
36
|
+
message_text?: string;
|
|
37
|
+
template_id?: string;
|
|
38
|
+
to?: string;
|
|
39
|
+
is_deleted?: boolean;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
export type DepartmentSubjectDoc = WithId<DepartmentSubject>;
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* Vector search result from MongoDB
|
|
46
|
+
*/
|
|
47
|
+
export type VectorSearchResult = {
|
|
48
|
+
subjectName: string;
|
|
49
|
+
subject_id: number;
|
|
50
|
+
sub_subject_id: string;
|
|
51
|
+
sub_subject_id2?: string | null;
|
|
52
|
+
sub_subject_name: string;
|
|
53
|
+
sub_subject_name2?: string | null;
|
|
54
|
+
descriptions?: string[];
|
|
55
|
+
guidelines?: string | null;
|
|
56
|
+
instructions?: string | null;
|
|
57
|
+
score?: number;
|
|
58
|
+
};
|
|
59
|
+
|
|
60
|
+
export type CommunicationType = "free_text_sms" | "upload_url_sms";
|
|
61
|
+
|
|
62
|
+
/**
|
|
63
|
+
* Extended Express Request type with client configuration
|
|
64
|
+
*/
|
|
65
|
+
export type RequestWithClientConfig = Request & {
|
|
66
|
+
uid: string;
|
|
67
|
+
clientConfig: ClientConfigDoc<Products>;
|
|
68
|
+
isGuest: boolean;
|
|
69
|
+
claims: Record<string, unknown>;
|
|
70
|
+
crmName?: string;
|
|
71
|
+
callSid?: string;
|
|
72
|
+
};
|
|
@@ -1,9 +1,9 @@
|
|
|
1
|
-
export * from "./departmentsSubjects.getters";
|
|
2
|
-
export type {
|
|
3
|
-
DepartmentSubject,
|
|
4
|
-
DepartmentSubjectDoc,
|
|
5
|
-
VectorSearchResult,
|
|
6
|
-
RequestWithClientConfig,
|
|
7
|
-
Communication,
|
|
8
|
-
Instruction,
|
|
9
|
-
} from "./departmentsSubjects.types";
|
|
1
|
+
export * from "./departmentsSubjects.getters";
|
|
2
|
+
export type {
|
|
3
|
+
DepartmentSubject,
|
|
4
|
+
DepartmentSubjectDoc,
|
|
5
|
+
VectorSearchResult,
|
|
6
|
+
RequestWithClientConfig,
|
|
7
|
+
Communication,
|
|
8
|
+
Instruction,
|
|
9
|
+
} from "./departmentsSubjects.types";
|
package/src/municipal/index.ts
CHANGED
|
@@ -1,21 +1,21 @@
|
|
|
1
|
-
import { Db, ObjectId as MongoObjectId } from "mongodb";
|
|
2
|
-
|
|
3
|
-
export * from "./cities";
|
|
4
|
-
export * from "./streets";
|
|
5
|
-
export * from "./departmentsSubjects";
|
|
6
|
-
export * from "./tickets";
|
|
7
|
-
export * from "./systemInstructions";
|
|
8
|
-
export * from "./utils/types";
|
|
9
|
-
export { CityName } from "./utils/types";
|
|
10
|
-
export { municipalDataMongodbClient } from "./mongodb-client";
|
|
11
|
-
|
|
12
|
-
let db: Db;
|
|
13
|
-
export const setDb = (d: Db) => {
|
|
14
|
-
db = d;
|
|
15
|
-
};
|
|
16
|
-
|
|
17
|
-
export const getDb = (): Db => {
|
|
18
|
-
if (!db) throw new Error("Municipal Data DB not initialised");
|
|
19
|
-
return db;
|
|
20
|
-
};
|
|
21
|
-
export const ObjectId = MongoObjectId;
|
|
1
|
+
import { Db, ObjectId as MongoObjectId } from "mongodb";
|
|
2
|
+
|
|
3
|
+
export * from "./cities";
|
|
4
|
+
export * from "./streets";
|
|
5
|
+
export * from "./departmentsSubjects";
|
|
6
|
+
export * from "./tickets";
|
|
7
|
+
export * from "./systemInstructions";
|
|
8
|
+
export * from "./utils/types";
|
|
9
|
+
export { CityName } from "./utils/types";
|
|
10
|
+
export { municipalDataMongodbClient } from "./mongodb-client";
|
|
11
|
+
|
|
12
|
+
let db: Db;
|
|
13
|
+
export const setDb = (d: Db) => {
|
|
14
|
+
db = d;
|
|
15
|
+
};
|
|
16
|
+
|
|
17
|
+
export const getDb = (): Db => {
|
|
18
|
+
if (!db) throw new Error("Municipal Data DB not initialised");
|
|
19
|
+
return db;
|
|
20
|
+
};
|
|
21
|
+
export const ObjectId = MongoObjectId;
|
|
@@ -1,61 +1,61 @@
|
|
|
1
|
-
import { MongoClient, Db } from "mongodb";
|
|
2
|
-
import { setDb } from "./index";
|
|
3
|
-
import { validateConfig, validateMongoUri } from "../utils/validation";
|
|
4
|
-
|
|
5
|
-
class MunicipalDataMongoDBClient {
|
|
6
|
-
private client: MongoClient | null = null;
|
|
7
|
-
private db: Db | null = null;
|
|
8
|
-
private readonly defaultDbName = "municipal-data";
|
|
9
|
-
|
|
10
|
-
async connect(uri?: string, dbName?: string): Promise<void> {
|
|
11
|
-
if (this.client) {
|
|
12
|
-
return;
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
const mongodbUri = uri || process.env.MONGO_URI || process.env.MONGODB_URI;
|
|
16
|
-
validateConfig("MONGO_URI", mongodbUri);
|
|
17
|
-
validateMongoUri(mongodbUri!);
|
|
18
|
-
|
|
19
|
-
try {
|
|
20
|
-
this.client = new MongoClient(mongodbUri!);
|
|
21
|
-
await this.client.connect();
|
|
22
|
-
const targetDbName =
|
|
23
|
-
dbName || process.env.MUNICIPAL_DB_NAME || this.defaultDbName;
|
|
24
|
-
this.db = this.client.db(targetDbName);
|
|
25
|
-
setDb(this.db);
|
|
26
|
-
console.info(`[core-db] Municipal MongoDB connected: ${targetDbName}`);
|
|
27
|
-
} catch (error) {
|
|
28
|
-
console.error("[core-db] Municipal connection failed", error);
|
|
29
|
-
throw error;
|
|
30
|
-
}
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
async disconnect(): Promise<void> {
|
|
34
|
-
if (this.client) {
|
|
35
|
-
try {
|
|
36
|
-
await this.client.close();
|
|
37
|
-
this.client = null;
|
|
38
|
-
this.db = null;
|
|
39
|
-
console.info("Municipal Data MongoDB disconnected successfully");
|
|
40
|
-
} catch (error) {
|
|
41
|
-
console.error("[core-db] Municipal disconnection failed", error);
|
|
42
|
-
throw error;
|
|
43
|
-
}
|
|
44
|
-
}
|
|
45
|
-
}
|
|
46
|
-
|
|
47
|
-
getDb(): Db {
|
|
48
|
-
if (!this.db) {
|
|
49
|
-
throw new Error(
|
|
50
|
-
"Municipal Data database not initialized. Call connect() first.",
|
|
51
|
-
);
|
|
52
|
-
}
|
|
53
|
-
return this.db;
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
isConnected(): boolean {
|
|
57
|
-
return this.client !== null && this.client !== undefined;
|
|
58
|
-
}
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
export const municipalDataMongodbClient = new MunicipalDataMongoDBClient();
|
|
1
|
+
import { MongoClient, Db } from "mongodb";
|
|
2
|
+
import { setDb } from "./index";
|
|
3
|
+
import { validateConfig, validateMongoUri } from "../utils/validation";
|
|
4
|
+
|
|
5
|
+
class MunicipalDataMongoDBClient {
|
|
6
|
+
private client: MongoClient | null = null;
|
|
7
|
+
private db: Db | null = null;
|
|
8
|
+
private readonly defaultDbName = "municipal-data";
|
|
9
|
+
|
|
10
|
+
async connect(uri?: string, dbName?: string): Promise<void> {
|
|
11
|
+
if (this.client) {
|
|
12
|
+
return;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
const mongodbUri = uri || process.env.MONGO_URI || process.env.MONGODB_URI;
|
|
16
|
+
validateConfig("MONGO_URI", mongodbUri);
|
|
17
|
+
validateMongoUri(mongodbUri!);
|
|
18
|
+
|
|
19
|
+
try {
|
|
20
|
+
this.client = new MongoClient(mongodbUri!);
|
|
21
|
+
await this.client.connect();
|
|
22
|
+
const targetDbName =
|
|
23
|
+
dbName || process.env.MUNICIPAL_DB_NAME || this.defaultDbName;
|
|
24
|
+
this.db = this.client.db(targetDbName);
|
|
25
|
+
setDb(this.db);
|
|
26
|
+
console.info(`[core-db] Municipal MongoDB connected: ${targetDbName}`);
|
|
27
|
+
} catch (error) {
|
|
28
|
+
console.error("[core-db] Municipal connection failed", error);
|
|
29
|
+
throw error;
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
async disconnect(): Promise<void> {
|
|
34
|
+
if (this.client) {
|
|
35
|
+
try {
|
|
36
|
+
await this.client.close();
|
|
37
|
+
this.client = null;
|
|
38
|
+
this.db = null;
|
|
39
|
+
console.info("Municipal Data MongoDB disconnected successfully");
|
|
40
|
+
} catch (error) {
|
|
41
|
+
console.error("[core-db] Municipal disconnection failed", error);
|
|
42
|
+
throw error;
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
getDb(): Db {
|
|
48
|
+
if (!this.db) {
|
|
49
|
+
throw new Error(
|
|
50
|
+
"Municipal Data database not initialized. Call connect() first.",
|
|
51
|
+
);
|
|
52
|
+
}
|
|
53
|
+
return this.db;
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
isConnected(): boolean {
|
|
57
|
+
return this.client !== null && this.client !== undefined;
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
export const municipalDataMongodbClient = new MunicipalDataMongoDBClient();
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
export * from "./streets.getters";
|
|
2
|
-
export type { Street, StreetDoc, StreetHint } from "./streets.types";
|
|
1
|
+
export * from "./streets.getters";
|
|
2
|
+
export type { Street, StreetDoc, StreetHint } from "./streets.types";
|
|
@@ -1,125 +1,125 @@
|
|
|
1
|
-
import { CityName, getDb, ObjectId, Street } from "../index";
|
|
2
|
-
import type { StreetHint } from "./streets.types";
|
|
3
|
-
import { Collection, Filter, ObjectId as MongoObjectId } from "mongodb";
|
|
4
|
-
|
|
5
|
-
export const getStreetsCollection = (): Collection<Street> => {
|
|
6
|
-
return getDb().collection<Street>("streets");
|
|
7
|
-
};
|
|
8
|
-
|
|
9
|
-
export const findStreets = async (
|
|
10
|
-
filter: Filter<Street> = {},
|
|
11
|
-
): Promise<Street[]> => {
|
|
12
|
-
return await getStreetsCollection().find(filter).toArray();
|
|
13
|
-
};
|
|
14
|
-
|
|
15
|
-
export const getStreetById = async (
|
|
16
|
-
streetId: string,
|
|
17
|
-
): Promise<Street | null> => {
|
|
18
|
-
const street = await getStreetsCollection().findOne({
|
|
19
|
-
_id: new ObjectId(streetId),
|
|
20
|
-
});
|
|
21
|
-
return street ? street : null;
|
|
22
|
-
};
|
|
23
|
-
|
|
24
|
-
export const getStreetsByCityName = async (
|
|
25
|
-
cityName: CityName,
|
|
26
|
-
): Promise<Street[]> => {
|
|
27
|
-
return await getStreetsCollection()
|
|
28
|
-
.find({ cityName })
|
|
29
|
-
.sort({ name: 1 })
|
|
30
|
-
.toArray();
|
|
31
|
-
};
|
|
32
|
-
|
|
33
|
-
export const getStreetByStreetId = async (
|
|
34
|
-
streetId: string,
|
|
35
|
-
): Promise<Street | null> => {
|
|
36
|
-
const street = await getStreetsCollection().findOne({ id: streetId });
|
|
37
|
-
return street ? street : null;
|
|
38
|
-
};
|
|
39
|
-
|
|
40
|
-
/** Get street by business id and city. Use this when the client's city is known so the same id in another city is not returned. */
|
|
41
|
-
export const getStreetByStreetIdAndCity = async (
|
|
42
|
-
streetId: string,
|
|
43
|
-
cityName: CityName,
|
|
44
|
-
): Promise<Street | null> => {
|
|
45
|
-
const street = await getStreetsCollection().findOne({
|
|
46
|
-
id: streetId,
|
|
47
|
-
cityName,
|
|
48
|
-
});
|
|
49
|
-
return street ? street : null;
|
|
50
|
-
};
|
|
51
|
-
|
|
52
|
-
export const createStreet = async (
|
|
53
|
-
streetData: Omit<Street, "_id" | "createdAt" | "updatedAt">,
|
|
54
|
-
): Promise<MongoObjectId> => {
|
|
55
|
-
const street: Omit<Street, "_id"> = {
|
|
56
|
-
...streetData,
|
|
57
|
-
createdAt: new Date(),
|
|
58
|
-
updatedAt: new Date(),
|
|
59
|
-
};
|
|
60
|
-
const { insertedId } = await getStreetsCollection().insertOne(
|
|
61
|
-
street as Street,
|
|
62
|
-
);
|
|
63
|
-
return insertedId;
|
|
64
|
-
};
|
|
65
|
-
|
|
66
|
-
export const updateStreet = async (
|
|
67
|
-
streetId: string,
|
|
68
|
-
data: Partial<Omit<Street, "_id" | "createdAt" | "updatedAt">>,
|
|
69
|
-
): Promise<Street | null> => {
|
|
70
|
-
const result = await getStreetsCollection().findOneAndUpdate(
|
|
71
|
-
{ _id: new ObjectId(streetId) },
|
|
72
|
-
{ $set: { ...data, updatedAt: new Date() } },
|
|
73
|
-
{ returnDocument: "after" },
|
|
74
|
-
);
|
|
75
|
-
return result || null;
|
|
76
|
-
};
|
|
77
|
-
|
|
78
|
-
/**
|
|
79
|
-
* Add or update a street hint on an already-fetched street.
|
|
80
|
-
* Only place that creates a new hint id (Date.getTime().toString()). Returns null if hint.id provided but not found, or text empty.
|
|
81
|
-
*/
|
|
82
|
-
export const upsertStreetHintOnStreet = async (
|
|
83
|
-
street: Street,
|
|
84
|
-
hint: Partial<StreetHint> & Pick<StreetHint, "text">,
|
|
85
|
-
): Promise<Street | null> => {
|
|
86
|
-
const text = hint.text.trim();
|
|
87
|
-
if (!text) return null;
|
|
88
|
-
|
|
89
|
-
const hints = street.hints || [];
|
|
90
|
-
let updatedHints: StreetHint[];
|
|
91
|
-
|
|
92
|
-
if (hint.id != null && hint.id !== "") {
|
|
93
|
-
const hintIndex = hints.findIndex((h) => String(h.id) === String(hint.id));
|
|
94
|
-
if (hintIndex === -1) return null;
|
|
95
|
-
updatedHints = [...hints];
|
|
96
|
-
updatedHints[hintIndex] = { id: String(hint.id), text };
|
|
97
|
-
} else {
|
|
98
|
-
updatedHints = [...hints, { id: new Date().getTime().toString(), text }];
|
|
99
|
-
}
|
|
100
|
-
|
|
101
|
-
return updateStreet(street._id.toString(), { hints: updatedHints });
|
|
102
|
-
};
|
|
103
|
-
|
|
104
|
-
/** Remove a hint from a street. streetId is the street business id (e.g. "703"). Updates document updatedAt. */
|
|
105
|
-
export const deleteStreetHint = async (
|
|
106
|
-
streetId: string,
|
|
107
|
-
hintId: string,
|
|
108
|
-
): Promise<Street | null> => {
|
|
109
|
-
const result = await getStreetsCollection().findOneAndUpdate(
|
|
110
|
-
{ id: streetId },
|
|
111
|
-
{ $pull: { hints: { id: hintId } }, $set: { updatedAt: new Date() } },
|
|
112
|
-
{ returnDocument: "after" },
|
|
113
|
-
);
|
|
114
|
-
if (result) {
|
|
115
|
-
console.info("Street hint deleted", { streetId, hintId });
|
|
116
|
-
}
|
|
117
|
-
return result || null;
|
|
118
|
-
};
|
|
119
|
-
|
|
120
|
-
export const deleteStreet = async (streetId: string): Promise<boolean> => {
|
|
121
|
-
const result = await getStreetsCollection().deleteOne({
|
|
122
|
-
_id: new ObjectId(streetId),
|
|
123
|
-
});
|
|
124
|
-
return result.deletedCount > 0;
|
|
125
|
-
};
|
|
1
|
+
import { CityName, getDb, ObjectId, Street } from "../index";
|
|
2
|
+
import type { StreetHint } from "./streets.types";
|
|
3
|
+
import { Collection, Filter, ObjectId as MongoObjectId } from "mongodb";
|
|
4
|
+
|
|
5
|
+
export const getStreetsCollection = (): Collection<Street> => {
|
|
6
|
+
return getDb().collection<Street>("streets");
|
|
7
|
+
};
|
|
8
|
+
|
|
9
|
+
export const findStreets = async (
|
|
10
|
+
filter: Filter<Street> = {},
|
|
11
|
+
): Promise<Street[]> => {
|
|
12
|
+
return await getStreetsCollection().find(filter).toArray();
|
|
13
|
+
};
|
|
14
|
+
|
|
15
|
+
export const getStreetById = async (
|
|
16
|
+
streetId: string,
|
|
17
|
+
): Promise<Street | null> => {
|
|
18
|
+
const street = await getStreetsCollection().findOne({
|
|
19
|
+
_id: new ObjectId(streetId),
|
|
20
|
+
});
|
|
21
|
+
return street ? street : null;
|
|
22
|
+
};
|
|
23
|
+
|
|
24
|
+
export const getStreetsByCityName = async (
|
|
25
|
+
cityName: CityName,
|
|
26
|
+
): Promise<Street[]> => {
|
|
27
|
+
return await getStreetsCollection()
|
|
28
|
+
.find({ cityName })
|
|
29
|
+
.sort({ name: 1 })
|
|
30
|
+
.toArray();
|
|
31
|
+
};
|
|
32
|
+
|
|
33
|
+
export const getStreetByStreetId = async (
|
|
34
|
+
streetId: string,
|
|
35
|
+
): Promise<Street | null> => {
|
|
36
|
+
const street = await getStreetsCollection().findOne({ id: streetId });
|
|
37
|
+
return street ? street : null;
|
|
38
|
+
};
|
|
39
|
+
|
|
40
|
+
/** Get street by business id and city. Use this when the client's city is known so the same id in another city is not returned. */
|
|
41
|
+
export const getStreetByStreetIdAndCity = async (
|
|
42
|
+
streetId: string,
|
|
43
|
+
cityName: CityName,
|
|
44
|
+
): Promise<Street | null> => {
|
|
45
|
+
const street = await getStreetsCollection().findOne({
|
|
46
|
+
id: streetId,
|
|
47
|
+
cityName,
|
|
48
|
+
});
|
|
49
|
+
return street ? street : null;
|
|
50
|
+
};
|
|
51
|
+
|
|
52
|
+
export const createStreet = async (
|
|
53
|
+
streetData: Omit<Street, "_id" | "createdAt" | "updatedAt">,
|
|
54
|
+
): Promise<MongoObjectId> => {
|
|
55
|
+
const street: Omit<Street, "_id"> = {
|
|
56
|
+
...streetData,
|
|
57
|
+
createdAt: new Date(),
|
|
58
|
+
updatedAt: new Date(),
|
|
59
|
+
};
|
|
60
|
+
const { insertedId } = await getStreetsCollection().insertOne(
|
|
61
|
+
street as Street,
|
|
62
|
+
);
|
|
63
|
+
return insertedId;
|
|
64
|
+
};
|
|
65
|
+
|
|
66
|
+
export const updateStreet = async (
|
|
67
|
+
streetId: string,
|
|
68
|
+
data: Partial<Omit<Street, "_id" | "createdAt" | "updatedAt">>,
|
|
69
|
+
): Promise<Street | null> => {
|
|
70
|
+
const result = await getStreetsCollection().findOneAndUpdate(
|
|
71
|
+
{ _id: new ObjectId(streetId) },
|
|
72
|
+
{ $set: { ...data, updatedAt: new Date() } },
|
|
73
|
+
{ returnDocument: "after" },
|
|
74
|
+
);
|
|
75
|
+
return result || null;
|
|
76
|
+
};
|
|
77
|
+
|
|
78
|
+
/**
|
|
79
|
+
* Add or update a street hint on an already-fetched street.
|
|
80
|
+
* Only place that creates a new hint id (Date.getTime().toString()). Returns null if hint.id provided but not found, or text empty.
|
|
81
|
+
*/
|
|
82
|
+
export const upsertStreetHintOnStreet = async (
|
|
83
|
+
street: Street,
|
|
84
|
+
hint: Partial<StreetHint> & Pick<StreetHint, "text">,
|
|
85
|
+
): Promise<Street | null> => {
|
|
86
|
+
const text = hint.text.trim();
|
|
87
|
+
if (!text) return null;
|
|
88
|
+
|
|
89
|
+
const hints = street.hints || [];
|
|
90
|
+
let updatedHints: StreetHint[];
|
|
91
|
+
|
|
92
|
+
if (hint.id != null && hint.id !== "") {
|
|
93
|
+
const hintIndex = hints.findIndex((h) => String(h.id) === String(hint.id));
|
|
94
|
+
if (hintIndex === -1) return null;
|
|
95
|
+
updatedHints = [...hints];
|
|
96
|
+
updatedHints[hintIndex] = { id: String(hint.id), text };
|
|
97
|
+
} else {
|
|
98
|
+
updatedHints = [...hints, { id: new Date().getTime().toString(), text }];
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
return updateStreet(street._id.toString(), { hints: updatedHints });
|
|
102
|
+
};
|
|
103
|
+
|
|
104
|
+
/** Remove a hint from a street. streetId is the street business id (e.g. "703"). Updates document updatedAt. */
|
|
105
|
+
export const deleteStreetHint = async (
|
|
106
|
+
streetId: string,
|
|
107
|
+
hintId: string,
|
|
108
|
+
): Promise<Street | null> => {
|
|
109
|
+
const result = await getStreetsCollection().findOneAndUpdate(
|
|
110
|
+
{ id: streetId },
|
|
111
|
+
{ $pull: { hints: { id: hintId } }, $set: { updatedAt: new Date() } },
|
|
112
|
+
{ returnDocument: "after" },
|
|
113
|
+
);
|
|
114
|
+
if (result) {
|
|
115
|
+
console.info("Street hint deleted", { streetId, hintId });
|
|
116
|
+
}
|
|
117
|
+
return result || null;
|
|
118
|
+
};
|
|
119
|
+
|
|
120
|
+
export const deleteStreet = async (streetId: string): Promise<boolean> => {
|
|
121
|
+
const result = await getStreetsCollection().deleteOne({
|
|
122
|
+
_id: new ObjectId(streetId),
|
|
123
|
+
});
|
|
124
|
+
return result.deletedCount > 0;
|
|
125
|
+
};
|
|
@@ -1,18 +1,18 @@
|
|
|
1
|
-
import { ObjectId, WithId } from "mongodb";
|
|
2
|
-
|
|
3
|
-
export type StreetHint = {
|
|
4
|
-
id: string;
|
|
5
|
-
text: string;
|
|
6
|
-
};
|
|
7
|
-
|
|
8
|
-
export type Street = {
|
|
9
|
-
_id: ObjectId;
|
|
10
|
-
createdAt: Date;
|
|
11
|
-
updatedAt: Date;
|
|
12
|
-
id: string;
|
|
13
|
-
name: string;
|
|
14
|
-
cityName: string;
|
|
15
|
-
hints?: StreetHint[];
|
|
16
|
-
};
|
|
17
|
-
|
|
18
|
-
export type StreetDoc = WithId<Street>;
|
|
1
|
+
import { ObjectId, WithId } from "mongodb";
|
|
2
|
+
|
|
3
|
+
export type StreetHint = {
|
|
4
|
+
id: string;
|
|
5
|
+
text: string;
|
|
6
|
+
};
|
|
7
|
+
|
|
8
|
+
export type Street = {
|
|
9
|
+
_id: ObjectId;
|
|
10
|
+
createdAt: Date;
|
|
11
|
+
updatedAt: Date;
|
|
12
|
+
id: string;
|
|
13
|
+
name: string;
|
|
14
|
+
cityName: string;
|
|
15
|
+
hints?: StreetHint[];
|
|
16
|
+
};
|
|
17
|
+
|
|
18
|
+
export type StreetDoc = WithId<Street>;
|