@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,154 +1,154 @@
|
|
|
1
|
-
import type { Collection } from "mongodb";
|
|
2
|
-
import {
|
|
3
|
-
Client,
|
|
4
|
-
getClientsCollection,
|
|
5
|
-
getDb,
|
|
6
|
-
getFlowsCollection,
|
|
7
|
-
ObjectId,
|
|
8
|
-
findSubscriptions,
|
|
9
|
-
} from '../index';
|
|
10
|
-
import {
|
|
11
|
-
ClientPhoneNumberForFlow,
|
|
12
|
-
PhoneNumber,
|
|
13
|
-
PhoneNumberWithFlow,
|
|
14
|
-
PurchasedPhoneProviderPayload
|
|
15
|
-
} from "./phone_numbers.types";
|
|
16
|
-
|
|
17
|
-
export const getPhoneNumbersCollection = (): Collection<PhoneNumber> =>
|
|
18
|
-
getDb().collection<PhoneNumber>("phone_numbers");
|
|
19
|
-
|
|
20
|
-
export const getPhoneDataByPhoneNumber = async (
|
|
21
|
-
phoneNumber: string,
|
|
22
|
-
): Promise<PhoneNumberWithFlow | null> => {
|
|
23
|
-
const phoneCallData = await getPhoneNumbersCollection().findOne({
|
|
24
|
-
phone_number: phoneNumber,
|
|
25
|
-
});
|
|
26
|
-
if (!phoneCallData) {
|
|
27
|
-
throw new Error("PhoneNumber not found");
|
|
28
|
-
}
|
|
29
|
-
const flow = await getFlowsCollection().findOne({
|
|
30
|
-
_id: new ObjectId(phoneCallData.flow_id),
|
|
31
|
-
});
|
|
32
|
-
if (!flow) {
|
|
33
|
-
throw new Error("Flow not found");
|
|
34
|
-
}
|
|
35
|
-
const [subscription] =
|
|
36
|
-
(await findSubscriptions({
|
|
37
|
-
clientId: phoneCallData.client_id,
|
|
38
|
-
isActive: true,
|
|
39
|
-
})) || [];
|
|
40
|
-
|
|
41
|
-
return {
|
|
42
|
-
...phoneCallData,
|
|
43
|
-
flow,
|
|
44
|
-
subscriptionId: subscription?._id?.toString() ?? undefined,
|
|
45
|
-
};
|
|
46
|
-
};
|
|
47
|
-
|
|
48
|
-
export const getClientPrimaryPhoneNumber = async (
|
|
49
|
-
clientId: string,
|
|
50
|
-
): Promise<string | null> => {
|
|
51
|
-
return (
|
|
52
|
-
(
|
|
53
|
-
await getPhoneNumbersCollection().findOne({
|
|
54
|
-
client_id: clientId,
|
|
55
|
-
is_primary: true,
|
|
56
|
-
})
|
|
57
|
-
)?.phone_number ?? null
|
|
58
|
-
);
|
|
59
|
-
};
|
|
60
|
-
|
|
61
|
-
export const getClientPhoneNumber = getClientPrimaryPhoneNumber;
|
|
62
|
-
|
|
63
|
-
export const getClientPhoneData = async (
|
|
64
|
-
clientId: string,
|
|
65
|
-
isPrimary?: boolean,
|
|
66
|
-
): Promise<PhoneNumber | null> => {
|
|
67
|
-
const filter = { client_id: clientId, is_primary: isPrimary !== false };
|
|
68
|
-
const options =
|
|
69
|
-
isPrimary === false ? { sort: { createdAt: -1 } as const } : {};
|
|
70
|
-
return getPhoneNumbersCollection().findOne(filter, options);
|
|
71
|
-
};
|
|
72
|
-
|
|
73
|
-
export const getPhoneNumbersForFlows = async (
|
|
74
|
-
clientId: string,
|
|
75
|
-
): Promise<ClientPhoneNumberForFlow[]> => {
|
|
76
|
-
const docs = await getPhoneNumbersCollection()
|
|
77
|
-
.find({
|
|
78
|
-
client_id: clientId,
|
|
79
|
-
flow_id: { $exists: true, $type: "objectId" },
|
|
80
|
-
})
|
|
81
|
-
.project({ _id: 0, flow_id: 1, phone_number: 1, is_primary: 1 })
|
|
82
|
-
.sort({ createdAt: -1 })
|
|
83
|
-
.toArray();
|
|
84
|
-
|
|
85
|
-
return docs.map((doc) => ({
|
|
86
|
-
flowId: String(doc.flow_id),
|
|
87
|
-
phoneNumber: doc.phone_number,
|
|
88
|
-
isPrimary: doc.is_primary,
|
|
89
|
-
}));
|
|
90
|
-
};
|
|
91
|
-
|
|
92
|
-
export const createPhoneNumberEntity = async (
|
|
93
|
-
phoneNumber: string,
|
|
94
|
-
flowId: string,
|
|
95
|
-
clientId: string,
|
|
96
|
-
): Promise<PhoneNumber> => {
|
|
97
|
-
const existing = await getClientPhoneData(clientId);
|
|
98
|
-
const isPrimary = !existing;
|
|
99
|
-
|
|
100
|
-
await getPhoneNumbersCollection().insertOne({
|
|
101
|
-
phone_number: phoneNumber,
|
|
102
|
-
flow_id: new ObjectId(flowId),
|
|
103
|
-
client_id: clientId,
|
|
104
|
-
is_primary: isPrimary,
|
|
105
|
-
createdAt: new Date(),
|
|
106
|
-
updatedAt: new Date(),
|
|
107
|
-
});
|
|
108
|
-
const phoneNumberData = await getClientPhoneData(clientId, isPrimary);
|
|
109
|
-
if (!phoneNumberData) throw new Error("Failed to create phoneNumber");
|
|
110
|
-
return phoneNumberData;
|
|
111
|
-
};
|
|
112
|
-
|
|
113
|
-
/** API-purchased number; `flowId` may be omitted until the number is bound to a flow. */
|
|
114
|
-
export const createPurchasedPhoneNumber = async (
|
|
115
|
-
phoneNumber: string,
|
|
116
|
-
providerPayload: PurchasedPhoneProviderPayload,
|
|
117
|
-
clientId: string,
|
|
118
|
-
flowId?: string | null,
|
|
119
|
-
): Promise<PhoneNumber> => {
|
|
120
|
-
const existingPhoneForClient = await getClientPhoneData(clientId);
|
|
121
|
-
const isPrimary = !existingPhoneForClient;
|
|
122
|
-
|
|
123
|
-
const now = new Date();
|
|
124
|
-
const trimmedFlowId = flowId?.trim();
|
|
125
|
-
const flowObjectId = trimmedFlowId ? new ObjectId(trimmedFlowId) : undefined;
|
|
126
|
-
|
|
127
|
-
const doc: PhoneNumber = {
|
|
128
|
-
phone_number: phoneNumber,
|
|
129
|
-
client_id: clientId,
|
|
130
|
-
is_primary: isPrimary,
|
|
131
|
-
...providerPayload,
|
|
132
|
-
createdAt: now,
|
|
133
|
-
updatedAt: now,
|
|
134
|
-
};
|
|
135
|
-
|
|
136
|
-
if (flowObjectId) {
|
|
137
|
-
doc.flow_id = flowObjectId;
|
|
138
|
-
}
|
|
139
|
-
|
|
140
|
-
await getPhoneNumbersCollection().insertOne(doc);
|
|
141
|
-
|
|
142
|
-
const created = await getClientPhoneData(clientId, isPrimary);
|
|
143
|
-
if (!created) throw new Error('Failed to create phoneNumber');
|
|
144
|
-
return created;
|
|
145
|
-
};
|
|
146
|
-
|
|
147
|
-
export const findClientByPhoneNumber = async (phoneNumber: string): Promise<Client> => {
|
|
148
|
-
const phoneData = await getPhoneNumbersCollection().findOne({ phone_number: phoneNumber });
|
|
149
|
-
if (!phoneData) throw new Error('Failed to get phone data');
|
|
150
|
-
const clientId = phoneData.client_id;
|
|
151
|
-
const client = await getClientsCollection().findOne({ clientId });
|
|
152
|
-
if (!client) throw new Error("Failed to get client");
|
|
153
|
-
return client;
|
|
154
|
-
};
|
|
1
|
+
import type { Collection } from "mongodb";
|
|
2
|
+
import {
|
|
3
|
+
Client,
|
|
4
|
+
getClientsCollection,
|
|
5
|
+
getDb,
|
|
6
|
+
getFlowsCollection,
|
|
7
|
+
ObjectId,
|
|
8
|
+
findSubscriptions,
|
|
9
|
+
} from '../index';
|
|
10
|
+
import {
|
|
11
|
+
ClientPhoneNumberForFlow,
|
|
12
|
+
PhoneNumber,
|
|
13
|
+
PhoneNumberWithFlow,
|
|
14
|
+
PurchasedPhoneProviderPayload
|
|
15
|
+
} from "./phone_numbers.types";
|
|
16
|
+
|
|
17
|
+
export const getPhoneNumbersCollection = (): Collection<PhoneNumber> =>
|
|
18
|
+
getDb().collection<PhoneNumber>("phone_numbers");
|
|
19
|
+
|
|
20
|
+
export const getPhoneDataByPhoneNumber = async (
|
|
21
|
+
phoneNumber: string,
|
|
22
|
+
): Promise<PhoneNumberWithFlow | null> => {
|
|
23
|
+
const phoneCallData = await getPhoneNumbersCollection().findOne({
|
|
24
|
+
phone_number: phoneNumber,
|
|
25
|
+
});
|
|
26
|
+
if (!phoneCallData) {
|
|
27
|
+
throw new Error("PhoneNumber not found");
|
|
28
|
+
}
|
|
29
|
+
const flow = await getFlowsCollection().findOne({
|
|
30
|
+
_id: new ObjectId(phoneCallData.flow_id),
|
|
31
|
+
});
|
|
32
|
+
if (!flow) {
|
|
33
|
+
throw new Error("Flow not found");
|
|
34
|
+
}
|
|
35
|
+
const [subscription] =
|
|
36
|
+
(await findSubscriptions({
|
|
37
|
+
clientId: phoneCallData.client_id,
|
|
38
|
+
isActive: true,
|
|
39
|
+
})) || [];
|
|
40
|
+
|
|
41
|
+
return {
|
|
42
|
+
...phoneCallData,
|
|
43
|
+
flow,
|
|
44
|
+
subscriptionId: subscription?._id?.toString() ?? undefined,
|
|
45
|
+
};
|
|
46
|
+
};
|
|
47
|
+
|
|
48
|
+
export const getClientPrimaryPhoneNumber = async (
|
|
49
|
+
clientId: string,
|
|
50
|
+
): Promise<string | null> => {
|
|
51
|
+
return (
|
|
52
|
+
(
|
|
53
|
+
await getPhoneNumbersCollection().findOne({
|
|
54
|
+
client_id: clientId,
|
|
55
|
+
is_primary: true,
|
|
56
|
+
})
|
|
57
|
+
)?.phone_number ?? null
|
|
58
|
+
);
|
|
59
|
+
};
|
|
60
|
+
|
|
61
|
+
export const getClientPhoneNumber = getClientPrimaryPhoneNumber;
|
|
62
|
+
|
|
63
|
+
export const getClientPhoneData = async (
|
|
64
|
+
clientId: string,
|
|
65
|
+
isPrimary?: boolean,
|
|
66
|
+
): Promise<PhoneNumber | null> => {
|
|
67
|
+
const filter = { client_id: clientId, is_primary: isPrimary !== false };
|
|
68
|
+
const options =
|
|
69
|
+
isPrimary === false ? { sort: { createdAt: -1 } as const } : {};
|
|
70
|
+
return getPhoneNumbersCollection().findOne(filter, options);
|
|
71
|
+
};
|
|
72
|
+
|
|
73
|
+
export const getPhoneNumbersForFlows = async (
|
|
74
|
+
clientId: string,
|
|
75
|
+
): Promise<ClientPhoneNumberForFlow[]> => {
|
|
76
|
+
const docs = await getPhoneNumbersCollection()
|
|
77
|
+
.find({
|
|
78
|
+
client_id: clientId,
|
|
79
|
+
flow_id: { $exists: true, $type: "objectId" },
|
|
80
|
+
})
|
|
81
|
+
.project({ _id: 0, flow_id: 1, phone_number: 1, is_primary: 1 })
|
|
82
|
+
.sort({ createdAt: -1 })
|
|
83
|
+
.toArray();
|
|
84
|
+
|
|
85
|
+
return docs.map((doc) => ({
|
|
86
|
+
flowId: String(doc.flow_id),
|
|
87
|
+
phoneNumber: doc.phone_number,
|
|
88
|
+
isPrimary: doc.is_primary,
|
|
89
|
+
}));
|
|
90
|
+
};
|
|
91
|
+
|
|
92
|
+
export const createPhoneNumberEntity = async (
|
|
93
|
+
phoneNumber: string,
|
|
94
|
+
flowId: string,
|
|
95
|
+
clientId: string,
|
|
96
|
+
): Promise<PhoneNumber> => {
|
|
97
|
+
const existing = await getClientPhoneData(clientId);
|
|
98
|
+
const isPrimary = !existing;
|
|
99
|
+
|
|
100
|
+
await getPhoneNumbersCollection().insertOne({
|
|
101
|
+
phone_number: phoneNumber,
|
|
102
|
+
flow_id: new ObjectId(flowId),
|
|
103
|
+
client_id: clientId,
|
|
104
|
+
is_primary: isPrimary,
|
|
105
|
+
createdAt: new Date(),
|
|
106
|
+
updatedAt: new Date(),
|
|
107
|
+
});
|
|
108
|
+
const phoneNumberData = await getClientPhoneData(clientId, isPrimary);
|
|
109
|
+
if (!phoneNumberData) throw new Error("Failed to create phoneNumber");
|
|
110
|
+
return phoneNumberData;
|
|
111
|
+
};
|
|
112
|
+
|
|
113
|
+
/** API-purchased number; `flowId` may be omitted until the number is bound to a flow. */
|
|
114
|
+
export const createPurchasedPhoneNumber = async (
|
|
115
|
+
phoneNumber: string,
|
|
116
|
+
providerPayload: PurchasedPhoneProviderPayload,
|
|
117
|
+
clientId: string,
|
|
118
|
+
flowId?: string | null,
|
|
119
|
+
): Promise<PhoneNumber> => {
|
|
120
|
+
const existingPhoneForClient = await getClientPhoneData(clientId);
|
|
121
|
+
const isPrimary = !existingPhoneForClient;
|
|
122
|
+
|
|
123
|
+
const now = new Date();
|
|
124
|
+
const trimmedFlowId = flowId?.trim();
|
|
125
|
+
const flowObjectId = trimmedFlowId ? new ObjectId(trimmedFlowId) : undefined;
|
|
126
|
+
|
|
127
|
+
const doc: PhoneNumber = {
|
|
128
|
+
phone_number: phoneNumber,
|
|
129
|
+
client_id: clientId,
|
|
130
|
+
is_primary: isPrimary,
|
|
131
|
+
...providerPayload,
|
|
132
|
+
createdAt: now,
|
|
133
|
+
updatedAt: now,
|
|
134
|
+
};
|
|
135
|
+
|
|
136
|
+
if (flowObjectId) {
|
|
137
|
+
doc.flow_id = flowObjectId;
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
await getPhoneNumbersCollection().insertOne(doc);
|
|
141
|
+
|
|
142
|
+
const created = await getClientPhoneData(clientId, isPrimary);
|
|
143
|
+
if (!created) throw new Error('Failed to create phoneNumber');
|
|
144
|
+
return created;
|
|
145
|
+
};
|
|
146
|
+
|
|
147
|
+
export const findClientByPhoneNumber = async (phoneNumber: string): Promise<Client> => {
|
|
148
|
+
const phoneData = await getPhoneNumbersCollection().findOne({ phone_number: phoneNumber });
|
|
149
|
+
if (!phoneData) throw new Error('Failed to get phone data');
|
|
150
|
+
const clientId = phoneData.client_id;
|
|
151
|
+
const client = await getClientsCollection().findOne({ clientId });
|
|
152
|
+
if (!client) throw new Error("Failed to get client");
|
|
153
|
+
return client;
|
|
154
|
+
};
|
|
@@ -1,17 +1,17 @@
|
|
|
1
|
-
export const phoneNumbersMongoSchema = {
|
|
2
|
-
bsonType: 'object',
|
|
3
|
-
required: ['client_id', 'phone_number'],
|
|
4
|
-
properties: {
|
|
5
|
-
_id: { bsonType: 'objectId' },
|
|
6
|
-
client_id: { bsonType: 'string' },
|
|
7
|
-
flow_id: { bsonType: ['objectId', 'string'] },
|
|
8
|
-
phone_number: { bsonType: 'string' },
|
|
9
|
-
is_primary: { bsonType: 'bool' },
|
|
10
|
-
provider: { bsonType: 'string' },
|
|
11
|
-
provider_sid: { bsonType: 'string' },
|
|
12
|
-
friendly_name: { bsonType: 'string' },
|
|
13
|
-
createdAt: { bsonType: 'date' },
|
|
14
|
-
updatedAt: { bsonType: 'date' },
|
|
15
|
-
},
|
|
16
|
-
additionalProperties: false,
|
|
17
|
-
} as const;
|
|
1
|
+
export const phoneNumbersMongoSchema = {
|
|
2
|
+
bsonType: 'object',
|
|
3
|
+
required: ['client_id', 'phone_number'],
|
|
4
|
+
properties: {
|
|
5
|
+
_id: { bsonType: 'objectId' },
|
|
6
|
+
client_id: { bsonType: 'string' },
|
|
7
|
+
flow_id: { bsonType: ['objectId', 'string'] },
|
|
8
|
+
phone_number: { bsonType: 'string' },
|
|
9
|
+
is_primary: { bsonType: 'bool' },
|
|
10
|
+
provider: { bsonType: 'string' },
|
|
11
|
+
provider_sid: { bsonType: 'string' },
|
|
12
|
+
friendly_name: { bsonType: 'string' },
|
|
13
|
+
createdAt: { bsonType: 'date' },
|
|
14
|
+
updatedAt: { bsonType: 'date' },
|
|
15
|
+
},
|
|
16
|
+
additionalProperties: false,
|
|
17
|
+
} as const;
|
|
@@ -1,30 +1,30 @@
|
|
|
1
|
-
import { ObjectId, WithId } from 'mongodb';
|
|
2
|
-
import { Flow } from '../flows/flows.types';
|
|
3
|
-
|
|
4
|
-
export type PurchasedPhoneProviderPayload = {
|
|
5
|
-
provider: 'twilio' | 'telnyx';
|
|
6
|
-
provider_sid: string;
|
|
7
|
-
friendly_name?: string;
|
|
8
|
-
};
|
|
9
|
-
|
|
10
|
-
export type PhoneNumber = {
|
|
11
|
-
_id?: ObjectId;
|
|
12
|
-
client_id: string;
|
|
13
|
-
flow_id?: ObjectId;
|
|
14
|
-
phone_number: string;
|
|
15
|
-
subscriptionId?: string;
|
|
16
|
-
is_primary: boolean;
|
|
17
|
-
provider?: 'twilio' | 'telnyx';
|
|
18
|
-
provider_sid?: string;
|
|
19
|
-
friendly_name?: string;
|
|
20
|
-
createdAt: Date;
|
|
21
|
-
updatedAt: Date;
|
|
22
|
-
};
|
|
23
|
-
|
|
24
|
-
export type PhoneNumberWithFlow = WithId<PhoneNumber> & { flow: Flow };
|
|
25
|
-
|
|
26
|
-
export type ClientPhoneNumberForFlow = {
|
|
27
|
-
flowId: string;
|
|
28
|
-
phoneNumber: string;
|
|
29
|
-
isPrimary: boolean;
|
|
30
|
-
};
|
|
1
|
+
import { ObjectId, WithId } from 'mongodb';
|
|
2
|
+
import { Flow } from '../flows/flows.types';
|
|
3
|
+
|
|
4
|
+
export type PurchasedPhoneProviderPayload = {
|
|
5
|
+
provider: 'twilio' | 'telnyx';
|
|
6
|
+
provider_sid: string;
|
|
7
|
+
friendly_name?: string;
|
|
8
|
+
};
|
|
9
|
+
|
|
10
|
+
export type PhoneNumber = {
|
|
11
|
+
_id?: ObjectId;
|
|
12
|
+
client_id: string;
|
|
13
|
+
flow_id?: ObjectId;
|
|
14
|
+
phone_number: string;
|
|
15
|
+
subscriptionId?: string;
|
|
16
|
+
is_primary: boolean;
|
|
17
|
+
provider?: 'twilio' | 'telnyx';
|
|
18
|
+
provider_sid?: string;
|
|
19
|
+
friendly_name?: string;
|
|
20
|
+
createdAt: Date;
|
|
21
|
+
updatedAt: Date;
|
|
22
|
+
};
|
|
23
|
+
|
|
24
|
+
export type PhoneNumberWithFlow = WithId<PhoneNumber> & { flow: Flow };
|
|
25
|
+
|
|
26
|
+
export type ClientPhoneNumberForFlow = {
|
|
27
|
+
flowId: string;
|
|
28
|
+
phoneNumber: string;
|
|
29
|
+
isPrimary: boolean;
|
|
30
|
+
};
|
|
@@ -1,70 +1,70 @@
|
|
|
1
|
-
import {
|
|
2
|
-
createPlanDoc,
|
|
3
|
-
findPlans,
|
|
4
|
-
findPlansByQuery,
|
|
5
|
-
countPlans,
|
|
6
|
-
updatePlanDoc,
|
|
7
|
-
getPlansCollection,
|
|
8
|
-
} from "../plans.getters";
|
|
9
|
-
import { Plan } from "../plans.types";
|
|
10
|
-
|
|
11
|
-
describe("db.plans", () => {
|
|
12
|
-
const createTestPlan = (
|
|
13
|
-
overrides?: Partial<Plan>,
|
|
14
|
-
): Omit<Plan, "createdAt" | "updatedAt"> => ({
|
|
15
|
-
productKey: "basic-500",
|
|
16
|
-
name: "Basic 500",
|
|
17
|
-
monthlyCallQuota: 500,
|
|
18
|
-
currency: "ILS",
|
|
19
|
-
priceMonthlyMinor: 10000,
|
|
20
|
-
isActive: true,
|
|
21
|
-
...overrides,
|
|
22
|
-
});
|
|
23
|
-
|
|
24
|
-
it("should create and find a plan", async () => {
|
|
25
|
-
const planData = createTestPlan({ name: "CreateTest" });
|
|
26
|
-
const created = await createPlanDoc(planData);
|
|
27
|
-
|
|
28
|
-
expect(created._id).toBeDefined();
|
|
29
|
-
expect(created.name).toBe("CreateTest");
|
|
30
|
-
|
|
31
|
-
const found = await findPlans({ _id: created._id });
|
|
32
|
-
expect(found.length).toBe(1);
|
|
33
|
-
expect(found[0].name).toBe("CreateTest");
|
|
34
|
-
});
|
|
35
|
-
|
|
36
|
-
it("should find plans by query with limit", async () => {
|
|
37
|
-
await getPlansCollection().deleteMany({});
|
|
38
|
-
await createPlanDoc(createTestPlan({ productKey: "p1", isActive: true }));
|
|
39
|
-
await createPlanDoc(createTestPlan({ productKey: "p2", isActive: true }));
|
|
40
|
-
await createPlanDoc(createTestPlan({ productKey: "p3", isActive: false }));
|
|
41
|
-
|
|
42
|
-
const activePlans = await findPlansByQuery(
|
|
43
|
-
{ isActive: true },
|
|
44
|
-
{ limit: 1 },
|
|
45
|
-
);
|
|
46
|
-
expect(activePlans.length).toBe(1);
|
|
47
|
-
expect(activePlans[0].isActive).toBe(true);
|
|
48
|
-
});
|
|
49
|
-
|
|
50
|
-
it("should count plans matching query", async () => {
|
|
51
|
-
await getPlansCollection().deleteMany({});
|
|
52
|
-
await createPlanDoc(createTestPlan({ currency: "USD" }));
|
|
53
|
-
await createPlanDoc(createTestPlan({ currency: "USD" }));
|
|
54
|
-
await createPlanDoc(createTestPlan({ currency: "ILS" }));
|
|
55
|
-
|
|
56
|
-
const count = await countPlans({ currency: "USD" });
|
|
57
|
-
expect(count).toBe(2);
|
|
58
|
-
});
|
|
59
|
-
|
|
60
|
-
it("should update a plan", async () => {
|
|
61
|
-
const plan = await createPlanDoc(createTestPlan({ isActive: true }));
|
|
62
|
-
const updated = await updatePlanDoc(plan._id, { isActive: false });
|
|
63
|
-
|
|
64
|
-
expect(updated).not.toBeNull();
|
|
65
|
-
expect(updated?.isActive).toBe(false);
|
|
66
|
-
|
|
67
|
-
const found = await findPlans({ _id: plan._id });
|
|
68
|
-
expect(found[0].isActive).toBe(false);
|
|
69
|
-
});
|
|
70
|
-
});
|
|
1
|
+
import {
|
|
2
|
+
createPlanDoc,
|
|
3
|
+
findPlans,
|
|
4
|
+
findPlansByQuery,
|
|
5
|
+
countPlans,
|
|
6
|
+
updatePlanDoc,
|
|
7
|
+
getPlansCollection,
|
|
8
|
+
} from "../plans.getters";
|
|
9
|
+
import { Plan } from "../plans.types";
|
|
10
|
+
|
|
11
|
+
describe("db.plans", () => {
|
|
12
|
+
const createTestPlan = (
|
|
13
|
+
overrides?: Partial<Plan>,
|
|
14
|
+
): Omit<Plan, "createdAt" | "updatedAt"> => ({
|
|
15
|
+
productKey: "basic-500",
|
|
16
|
+
name: "Basic 500",
|
|
17
|
+
monthlyCallQuota: 500,
|
|
18
|
+
currency: "ILS",
|
|
19
|
+
priceMonthlyMinor: 10000,
|
|
20
|
+
isActive: true,
|
|
21
|
+
...overrides,
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
it("should create and find a plan", async () => {
|
|
25
|
+
const planData = createTestPlan({ name: "CreateTest" });
|
|
26
|
+
const created = await createPlanDoc(planData);
|
|
27
|
+
|
|
28
|
+
expect(created._id).toBeDefined();
|
|
29
|
+
expect(created.name).toBe("CreateTest");
|
|
30
|
+
|
|
31
|
+
const found = await findPlans({ _id: created._id });
|
|
32
|
+
expect(found.length).toBe(1);
|
|
33
|
+
expect(found[0].name).toBe("CreateTest");
|
|
34
|
+
});
|
|
35
|
+
|
|
36
|
+
it("should find plans by query with limit", async () => {
|
|
37
|
+
await getPlansCollection().deleteMany({});
|
|
38
|
+
await createPlanDoc(createTestPlan({ productKey: "p1", isActive: true }));
|
|
39
|
+
await createPlanDoc(createTestPlan({ productKey: "p2", isActive: true }));
|
|
40
|
+
await createPlanDoc(createTestPlan({ productKey: "p3", isActive: false }));
|
|
41
|
+
|
|
42
|
+
const activePlans = await findPlansByQuery(
|
|
43
|
+
{ isActive: true },
|
|
44
|
+
{ limit: 1 },
|
|
45
|
+
);
|
|
46
|
+
expect(activePlans.length).toBe(1);
|
|
47
|
+
expect(activePlans[0].isActive).toBe(true);
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
it("should count plans matching query", async () => {
|
|
51
|
+
await getPlansCollection().deleteMany({});
|
|
52
|
+
await createPlanDoc(createTestPlan({ currency: "USD" }));
|
|
53
|
+
await createPlanDoc(createTestPlan({ currency: "USD" }));
|
|
54
|
+
await createPlanDoc(createTestPlan({ currency: "ILS" }));
|
|
55
|
+
|
|
56
|
+
const count = await countPlans({ currency: "USD" });
|
|
57
|
+
expect(count).toBe(2);
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
it("should update a plan", async () => {
|
|
61
|
+
const plan = await createPlanDoc(createTestPlan({ isActive: true }));
|
|
62
|
+
const updated = await updatePlanDoc(plan._id, { isActive: false });
|
|
63
|
+
|
|
64
|
+
expect(updated).not.toBeNull();
|
|
65
|
+
expect(updated?.isActive).toBe(false);
|
|
66
|
+
|
|
67
|
+
const found = await findPlans({ _id: plan._id });
|
|
68
|
+
expect(found[0].isActive).toBe(false);
|
|
69
|
+
});
|
|
70
|
+
});
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
export * from "./plans.types";
|
|
2
|
-
export * from "./plans.getters";
|
|
1
|
+
export * from "./plans.types";
|
|
2
|
+
export * from "./plans.getters";
|