@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.
- 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 +2 -1
- package/dist/talkpilot/calls/calls.getters.d.ts.map +1 -1
- package/dist/talkpilot/calls/calls.getters.js +176 -0
- package/dist/talkpilot/calls/calls.getters.js.map +1 -1
- package/dist/talkpilot/calls/calls.types.d.ts +49 -2
- package/dist/talkpilot/calls/calls.types.d.ts.map +1 -1
- package/dist/talkpilot/calls/calls.types.js.map +1 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts +1 -0
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts.map +1 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.js +13 -0
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.js.map +1 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.types.d.ts +2 -0
- 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 +446 -248
- package/src/talkpilot/calls/calls.types.ts +171 -116
- 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 +106 -106
- package/src/talkpilot/clientsConfig/clientsConfig.getters.ts +44 -22
- package/src/talkpilot/clientsConfig/clientsConfig.types.ts +94 -92
- 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,45 +1,45 @@
|
|
|
1
|
-
import { ObjectId } from "mongodb";
|
|
2
|
-
import {
|
|
3
|
-
buildSimpleQuery,
|
|
4
|
-
buildDateRangeFilter,
|
|
5
|
-
} from "../subscriptions.getters.utils";
|
|
6
|
-
import { SimpleSubscriptionFilter } from "../subscriptions.types";
|
|
7
|
-
|
|
8
|
-
describe("subscriptions.getters.utils", () => {
|
|
9
|
-
describe("buildSimpleQuery", () => {
|
|
10
|
-
it("returns empty query when filter is empty", () => {
|
|
11
|
-
const filter: SimpleSubscriptionFilter = {};
|
|
12
|
-
const result = buildSimpleQuery(filter);
|
|
13
|
-
expect(result).toEqual({});
|
|
14
|
-
});
|
|
15
|
-
|
|
16
|
-
it("converts _id string to ObjectId", () => {
|
|
17
|
-
const id = "507f1f77bcf86cd799439011";
|
|
18
|
-
const filter: SimpleSubscriptionFilter = { _id: id };
|
|
19
|
-
const result = buildSimpleQuery(filter);
|
|
20
|
-
expect(result._id).toBeInstanceOf(ObjectId);
|
|
21
|
-
expect((result._id as ObjectId).toHexString()).toBe(id);
|
|
22
|
-
});
|
|
23
|
-
|
|
24
|
-
it("includes valid simple fields", () => {
|
|
25
|
-
const filter: SimpleSubscriptionFilter = {
|
|
26
|
-
clientId: "abc",
|
|
27
|
-
isActive: true,
|
|
28
|
-
};
|
|
29
|
-
const result = buildSimpleQuery(filter);
|
|
30
|
-
expect(result).toEqual({ clientId: "abc", isActive: true });
|
|
31
|
-
});
|
|
32
|
-
});
|
|
33
|
-
|
|
34
|
-
describe("buildDateRangeFilter", () => {
|
|
35
|
-
it("returns a mongo date range filter", () => {
|
|
36
|
-
const from = new Date("2023-01-01");
|
|
37
|
-
const to = new Date("2023-01-02");
|
|
38
|
-
const result = buildDateRangeFilter(from, to);
|
|
39
|
-
expect(result).toEqual({
|
|
40
|
-
$gte: from,
|
|
41
|
-
$lt: to,
|
|
42
|
-
});
|
|
43
|
-
});
|
|
44
|
-
});
|
|
45
|
-
});
|
|
1
|
+
import { ObjectId } from "mongodb";
|
|
2
|
+
import {
|
|
3
|
+
buildSimpleQuery,
|
|
4
|
+
buildDateRangeFilter,
|
|
5
|
+
} from "../subscriptions.getters.utils";
|
|
6
|
+
import { SimpleSubscriptionFilter } from "../subscriptions.types";
|
|
7
|
+
|
|
8
|
+
describe("subscriptions.getters.utils", () => {
|
|
9
|
+
describe("buildSimpleQuery", () => {
|
|
10
|
+
it("returns empty query when filter is empty", () => {
|
|
11
|
+
const filter: SimpleSubscriptionFilter = {};
|
|
12
|
+
const result = buildSimpleQuery(filter);
|
|
13
|
+
expect(result).toEqual({});
|
|
14
|
+
});
|
|
15
|
+
|
|
16
|
+
it("converts _id string to ObjectId", () => {
|
|
17
|
+
const id = "507f1f77bcf86cd799439011";
|
|
18
|
+
const filter: SimpleSubscriptionFilter = { _id: id };
|
|
19
|
+
const result = buildSimpleQuery(filter);
|
|
20
|
+
expect(result._id).toBeInstanceOf(ObjectId);
|
|
21
|
+
expect((result._id as ObjectId).toHexString()).toBe(id);
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
it("includes valid simple fields", () => {
|
|
25
|
+
const filter: SimpleSubscriptionFilter = {
|
|
26
|
+
clientId: "abc",
|
|
27
|
+
isActive: true,
|
|
28
|
+
};
|
|
29
|
+
const result = buildSimpleQuery(filter);
|
|
30
|
+
expect(result).toEqual({ clientId: "abc", isActive: true });
|
|
31
|
+
});
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
describe("buildDateRangeFilter", () => {
|
|
35
|
+
it("returns a mongo date range filter", () => {
|
|
36
|
+
const from = new Date("2023-01-01");
|
|
37
|
+
const to = new Date("2023-01-02");
|
|
38
|
+
const result = buildDateRangeFilter(from, to);
|
|
39
|
+
expect(result).toEqual({
|
|
40
|
+
$gte: from,
|
|
41
|
+
$lt: to,
|
|
42
|
+
});
|
|
43
|
+
});
|
|
44
|
+
});
|
|
45
|
+
});
|
|
@@ -1,3 +1,3 @@
|
|
|
1
|
-
export * from "./subscriptions.types";
|
|
2
|
-
export * from "./subscriptions.getters";
|
|
3
|
-
export * from "./subscriptions.getters.utils";
|
|
1
|
+
export * from "./subscriptions.types";
|
|
2
|
+
export * from "./subscriptions.getters";
|
|
3
|
+
export * from "./subscriptions.getters.utils";
|
|
@@ -1,146 +1,146 @@
|
|
|
1
|
-
import { getDb } from "../index";
|
|
2
|
-
import {
|
|
3
|
-
SimpleSubscriptionFilter,
|
|
4
|
-
Subscription,
|
|
5
|
-
SubscriptionDoc,
|
|
6
|
-
SubscriptionFilter,
|
|
7
|
-
SubscriptionQueryOptions,
|
|
8
|
-
} from "./subscriptions.types";
|
|
9
|
-
import { Filter, ObjectId } from "mongodb";
|
|
10
|
-
import { buildSimpleQuery } from "./subscriptions.getters.utils";
|
|
11
|
-
import { applyQueryOptions } from "../utils/query.utils";
|
|
12
|
-
|
|
13
|
-
export const getSubscriptionsCollection = () => {
|
|
14
|
-
return getDb().collection<Subscription>("subscriptions");
|
|
15
|
-
};
|
|
16
|
-
|
|
17
|
-
/**
|
|
18
|
-
* A simple filter for Subscription: each field may be a value or a MongoDB operator expression
|
|
19
|
-
*/
|
|
20
|
-
|
|
21
|
-
export const findSubscriptions = async (
|
|
22
|
-
filter?: SubscriptionFilter,
|
|
23
|
-
options?: SubscriptionQueryOptions,
|
|
24
|
-
): Promise<SubscriptionDoc[]> => {
|
|
25
|
-
let query: Filter<Subscription> = {};
|
|
26
|
-
|
|
27
|
-
if (filter) {
|
|
28
|
-
const simpleFilter: SimpleSubscriptionFilter = {
|
|
29
|
-
_id: filter._id,
|
|
30
|
-
clientId: filter.clientId,
|
|
31
|
-
planId: filter.planId,
|
|
32
|
-
startDate: filter.startDate,
|
|
33
|
-
endDate: filter.endDate,
|
|
34
|
-
isActive: filter.isActive,
|
|
35
|
-
createdAt: filter.createdAt,
|
|
36
|
-
nextRenewalDate: filter.nextRenewalDate,
|
|
37
|
-
};
|
|
38
|
-
const baseQuery = buildSimpleQuery(simpleFilter);
|
|
39
|
-
if (Object.keys(baseQuery).length > 0) {
|
|
40
|
-
query = baseQuery;
|
|
41
|
-
}
|
|
42
|
-
if (filter.or && filter.or.length > 0) {
|
|
43
|
-
query.$or = filter.or.map(buildSimpleQuery);
|
|
44
|
-
}
|
|
45
|
-
if (filter.and && filter.and.length > 0) {
|
|
46
|
-
query.$and = filter.and.map(buildSimpleQuery);
|
|
47
|
-
}
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
const cursor = getSubscriptionsCollection().find(query);
|
|
51
|
-
return await applyQueryOptions(cursor, options).toArray();
|
|
52
|
-
};
|
|
53
|
-
|
|
54
|
-
export const countSubscriptions = async (
|
|
55
|
-
query: Filter<Subscription>,
|
|
56
|
-
): Promise<number> => {
|
|
57
|
-
return getSubscriptionsCollection().countDocuments(query);
|
|
58
|
-
};
|
|
59
|
-
|
|
60
|
-
export const createSubscriptionDoc = async (
|
|
61
|
-
subscriptionData: Omit<Subscription, "createdAt" | "updatedAt">,
|
|
62
|
-
): Promise<SubscriptionDoc> => {
|
|
63
|
-
const now = new Date();
|
|
64
|
-
const result = await getSubscriptionsCollection().insertOne({
|
|
65
|
-
...subscriptionData,
|
|
66
|
-
createdAt: now,
|
|
67
|
-
updatedAt: now,
|
|
68
|
-
});
|
|
69
|
-
|
|
70
|
-
const subscriptions = await findSubscriptions({ _id: result.insertedId });
|
|
71
|
-
if (!subscriptions[0])
|
|
72
|
-
throw new Error("Failed to retrieve created subscription");
|
|
73
|
-
return subscriptions[0];
|
|
74
|
-
};
|
|
75
|
-
|
|
76
|
-
export const updateSubscriptionDoc = async (
|
|
77
|
-
subscriptionId: ObjectId,
|
|
78
|
-
updates: Partial<Subscription>,
|
|
79
|
-
): Promise<SubscriptionDoc | null> => {
|
|
80
|
-
return await getSubscriptionsCollection().findOneAndUpdate(
|
|
81
|
-
{ _id: subscriptionId },
|
|
82
|
-
{
|
|
83
|
-
$set: {
|
|
84
|
-
...updates,
|
|
85
|
-
updatedAt: new Date(),
|
|
86
|
-
},
|
|
87
|
-
},
|
|
88
|
-
{
|
|
89
|
-
returnDocument: "after",
|
|
90
|
-
},
|
|
91
|
-
);
|
|
92
|
-
};
|
|
93
|
-
|
|
94
|
-
export const incrementSubscriptionUsage = async (
|
|
95
|
-
subscriptionId: ObjectId,
|
|
96
|
-
amount: number = 1,
|
|
97
|
-
): Promise<SubscriptionDoc | null> => {
|
|
98
|
-
const fieldToIncrement = "cycles.$[activeCycle].usage.usedCalls";
|
|
99
|
-
return await getSubscriptionsCollection().findOneAndUpdate(
|
|
100
|
-
{ _id: subscriptionId, "cycles.isActive": true },
|
|
101
|
-
{
|
|
102
|
-
$inc: { [fieldToIncrement]: amount },
|
|
103
|
-
$set: { updatedAt: new Date() },
|
|
104
|
-
},
|
|
105
|
-
{
|
|
106
|
-
arrayFilters: [{ "activeCycle.isActive": true }],
|
|
107
|
-
returnDocument: "after",
|
|
108
|
-
},
|
|
109
|
-
);
|
|
110
|
-
};
|
|
111
|
-
|
|
112
|
-
export const deactivateCurrentSubscriptionCycle = async (
|
|
113
|
-
subscriptionId: ObjectId,
|
|
114
|
-
nextRenewalDate: Date,
|
|
115
|
-
): Promise<void> => {
|
|
116
|
-
await getSubscriptionsCollection().updateOne(
|
|
117
|
-
{ _id: subscriptionId, "cycles.isActive": true },
|
|
118
|
-
{
|
|
119
|
-
$set: {
|
|
120
|
-
"cycles.$[activeCycle].isActive": false,
|
|
121
|
-
nextRenewalDate: nextRenewalDate,
|
|
122
|
-
updatedAt: new Date(),
|
|
123
|
-
},
|
|
124
|
-
},
|
|
125
|
-
{
|
|
126
|
-
arrayFilters: [{ "activeCycle.isActive": true }],
|
|
127
|
-
},
|
|
128
|
-
);
|
|
129
|
-
};
|
|
130
|
-
|
|
131
|
-
export const pushNewSubscriptionCycle = async (
|
|
132
|
-
subscriptionId: ObjectId,
|
|
133
|
-
newCycle: Subscription["cycles"][number],
|
|
134
|
-
): Promise<SubscriptionDoc | null> => {
|
|
135
|
-
return await getSubscriptionsCollection().findOneAndUpdate(
|
|
136
|
-
{ _id: subscriptionId },
|
|
137
|
-
{
|
|
138
|
-
$push: {
|
|
139
|
-
cycles: newCycle,
|
|
140
|
-
},
|
|
141
|
-
},
|
|
142
|
-
{
|
|
143
|
-
returnDocument: "after",
|
|
144
|
-
},
|
|
145
|
-
);
|
|
146
|
-
};
|
|
1
|
+
import { getDb } from "../index";
|
|
2
|
+
import {
|
|
3
|
+
SimpleSubscriptionFilter,
|
|
4
|
+
Subscription,
|
|
5
|
+
SubscriptionDoc,
|
|
6
|
+
SubscriptionFilter,
|
|
7
|
+
SubscriptionQueryOptions,
|
|
8
|
+
} from "./subscriptions.types";
|
|
9
|
+
import { Filter, ObjectId } from "mongodb";
|
|
10
|
+
import { buildSimpleQuery } from "./subscriptions.getters.utils";
|
|
11
|
+
import { applyQueryOptions } from "../utils/query.utils";
|
|
12
|
+
|
|
13
|
+
export const getSubscriptionsCollection = () => {
|
|
14
|
+
return getDb().collection<Subscription>("subscriptions");
|
|
15
|
+
};
|
|
16
|
+
|
|
17
|
+
/**
|
|
18
|
+
* A simple filter for Subscription: each field may be a value or a MongoDB operator expression
|
|
19
|
+
*/
|
|
20
|
+
|
|
21
|
+
export const findSubscriptions = async (
|
|
22
|
+
filter?: SubscriptionFilter,
|
|
23
|
+
options?: SubscriptionQueryOptions,
|
|
24
|
+
): Promise<SubscriptionDoc[]> => {
|
|
25
|
+
let query: Filter<Subscription> = {};
|
|
26
|
+
|
|
27
|
+
if (filter) {
|
|
28
|
+
const simpleFilter: SimpleSubscriptionFilter = {
|
|
29
|
+
_id: filter._id,
|
|
30
|
+
clientId: filter.clientId,
|
|
31
|
+
planId: filter.planId,
|
|
32
|
+
startDate: filter.startDate,
|
|
33
|
+
endDate: filter.endDate,
|
|
34
|
+
isActive: filter.isActive,
|
|
35
|
+
createdAt: filter.createdAt,
|
|
36
|
+
nextRenewalDate: filter.nextRenewalDate,
|
|
37
|
+
};
|
|
38
|
+
const baseQuery = buildSimpleQuery(simpleFilter);
|
|
39
|
+
if (Object.keys(baseQuery).length > 0) {
|
|
40
|
+
query = baseQuery;
|
|
41
|
+
}
|
|
42
|
+
if (filter.or && filter.or.length > 0) {
|
|
43
|
+
query.$or = filter.or.map(buildSimpleQuery);
|
|
44
|
+
}
|
|
45
|
+
if (filter.and && filter.and.length > 0) {
|
|
46
|
+
query.$and = filter.and.map(buildSimpleQuery);
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
const cursor = getSubscriptionsCollection().find(query);
|
|
51
|
+
return await applyQueryOptions(cursor, options).toArray();
|
|
52
|
+
};
|
|
53
|
+
|
|
54
|
+
export const countSubscriptions = async (
|
|
55
|
+
query: Filter<Subscription>,
|
|
56
|
+
): Promise<number> => {
|
|
57
|
+
return getSubscriptionsCollection().countDocuments(query);
|
|
58
|
+
};
|
|
59
|
+
|
|
60
|
+
export const createSubscriptionDoc = async (
|
|
61
|
+
subscriptionData: Omit<Subscription, "createdAt" | "updatedAt">,
|
|
62
|
+
): Promise<SubscriptionDoc> => {
|
|
63
|
+
const now = new Date();
|
|
64
|
+
const result = await getSubscriptionsCollection().insertOne({
|
|
65
|
+
...subscriptionData,
|
|
66
|
+
createdAt: now,
|
|
67
|
+
updatedAt: now,
|
|
68
|
+
});
|
|
69
|
+
|
|
70
|
+
const subscriptions = await findSubscriptions({ _id: result.insertedId });
|
|
71
|
+
if (!subscriptions[0])
|
|
72
|
+
throw new Error("Failed to retrieve created subscription");
|
|
73
|
+
return subscriptions[0];
|
|
74
|
+
};
|
|
75
|
+
|
|
76
|
+
export const updateSubscriptionDoc = async (
|
|
77
|
+
subscriptionId: ObjectId,
|
|
78
|
+
updates: Partial<Subscription>,
|
|
79
|
+
): Promise<SubscriptionDoc | null> => {
|
|
80
|
+
return await getSubscriptionsCollection().findOneAndUpdate(
|
|
81
|
+
{ _id: subscriptionId },
|
|
82
|
+
{
|
|
83
|
+
$set: {
|
|
84
|
+
...updates,
|
|
85
|
+
updatedAt: new Date(),
|
|
86
|
+
},
|
|
87
|
+
},
|
|
88
|
+
{
|
|
89
|
+
returnDocument: "after",
|
|
90
|
+
},
|
|
91
|
+
);
|
|
92
|
+
};
|
|
93
|
+
|
|
94
|
+
export const incrementSubscriptionUsage = async (
|
|
95
|
+
subscriptionId: ObjectId,
|
|
96
|
+
amount: number = 1,
|
|
97
|
+
): Promise<SubscriptionDoc | null> => {
|
|
98
|
+
const fieldToIncrement = "cycles.$[activeCycle].usage.usedCalls";
|
|
99
|
+
return await getSubscriptionsCollection().findOneAndUpdate(
|
|
100
|
+
{ _id: subscriptionId, "cycles.isActive": true },
|
|
101
|
+
{
|
|
102
|
+
$inc: { [fieldToIncrement]: amount },
|
|
103
|
+
$set: { updatedAt: new Date() },
|
|
104
|
+
},
|
|
105
|
+
{
|
|
106
|
+
arrayFilters: [{ "activeCycle.isActive": true }],
|
|
107
|
+
returnDocument: "after",
|
|
108
|
+
},
|
|
109
|
+
);
|
|
110
|
+
};
|
|
111
|
+
|
|
112
|
+
export const deactivateCurrentSubscriptionCycle = async (
|
|
113
|
+
subscriptionId: ObjectId,
|
|
114
|
+
nextRenewalDate: Date,
|
|
115
|
+
): Promise<void> => {
|
|
116
|
+
await getSubscriptionsCollection().updateOne(
|
|
117
|
+
{ _id: subscriptionId, "cycles.isActive": true },
|
|
118
|
+
{
|
|
119
|
+
$set: {
|
|
120
|
+
"cycles.$[activeCycle].isActive": false,
|
|
121
|
+
nextRenewalDate: nextRenewalDate,
|
|
122
|
+
updatedAt: new Date(),
|
|
123
|
+
},
|
|
124
|
+
},
|
|
125
|
+
{
|
|
126
|
+
arrayFilters: [{ "activeCycle.isActive": true }],
|
|
127
|
+
},
|
|
128
|
+
);
|
|
129
|
+
};
|
|
130
|
+
|
|
131
|
+
export const pushNewSubscriptionCycle = async (
|
|
132
|
+
subscriptionId: ObjectId,
|
|
133
|
+
newCycle: Subscription["cycles"][number],
|
|
134
|
+
): Promise<SubscriptionDoc | null> => {
|
|
135
|
+
return await getSubscriptionsCollection().findOneAndUpdate(
|
|
136
|
+
{ _id: subscriptionId },
|
|
137
|
+
{
|
|
138
|
+
$push: {
|
|
139
|
+
cycles: newCycle,
|
|
140
|
+
},
|
|
141
|
+
},
|
|
142
|
+
{
|
|
143
|
+
returnDocument: "after",
|
|
144
|
+
},
|
|
145
|
+
);
|
|
146
|
+
};
|
|
@@ -1,33 +1,33 @@
|
|
|
1
|
-
import { ObjectId, Filter } from "mongodb";
|
|
2
|
-
import {
|
|
3
|
-
Subscription,
|
|
4
|
-
SimpleSubscriptionFilter,
|
|
5
|
-
simpleFields,
|
|
6
|
-
SimpleField,
|
|
7
|
-
} from "../index";
|
|
8
|
-
|
|
9
|
-
export const buildSimpleQuery = (
|
|
10
|
-
filter: SimpleSubscriptionFilter,
|
|
11
|
-
): Filter<Subscription> => {
|
|
12
|
-
const query: Filter<Subscription> = {};
|
|
13
|
-
|
|
14
|
-
if (filter._id) {
|
|
15
|
-
query._id = new ObjectId(filter._id);
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
simpleFields.forEach((field: SimpleField) => {
|
|
19
|
-
const val = filter[field];
|
|
20
|
-
if (val !== undefined) {
|
|
21
|
-
query[field] = val as any;
|
|
22
|
-
}
|
|
23
|
-
});
|
|
24
|
-
|
|
25
|
-
return query;
|
|
26
|
-
};
|
|
27
|
-
|
|
28
|
-
export const buildDateRangeFilter = (from: Date, to: Date) => {
|
|
29
|
-
return {
|
|
30
|
-
$gte: from,
|
|
31
|
-
$lt: to,
|
|
32
|
-
};
|
|
33
|
-
};
|
|
1
|
+
import { ObjectId, Filter } from "mongodb";
|
|
2
|
+
import {
|
|
3
|
+
Subscription,
|
|
4
|
+
SimpleSubscriptionFilter,
|
|
5
|
+
simpleFields,
|
|
6
|
+
SimpleField,
|
|
7
|
+
} from "../index";
|
|
8
|
+
|
|
9
|
+
export const buildSimpleQuery = (
|
|
10
|
+
filter: SimpleSubscriptionFilter,
|
|
11
|
+
): Filter<Subscription> => {
|
|
12
|
+
const query: Filter<Subscription> = {};
|
|
13
|
+
|
|
14
|
+
if (filter._id) {
|
|
15
|
+
query._id = new ObjectId(filter._id);
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
simpleFields.forEach((field: SimpleField) => {
|
|
19
|
+
const val = filter[field];
|
|
20
|
+
if (val !== undefined) {
|
|
21
|
+
query[field] = val as any;
|
|
22
|
+
}
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
return query;
|
|
26
|
+
};
|
|
27
|
+
|
|
28
|
+
export const buildDateRangeFilter = (from: Date, to: Date) => {
|
|
29
|
+
return {
|
|
30
|
+
$gte: from,
|
|
31
|
+
$lt: to,
|
|
32
|
+
};
|
|
33
|
+
};
|
|
@@ -1,66 +1,66 @@
|
|
|
1
|
-
import { Filter, ObjectId, WithId, Sort } from "mongodb";
|
|
2
|
-
import { PlanDoc } from "../plans";
|
|
3
|
-
|
|
4
|
-
export type SubscriptionCycle = {
|
|
5
|
-
usage: {
|
|
6
|
-
usedCalls: number;
|
|
7
|
-
};
|
|
8
|
-
startDate: Date;
|
|
9
|
-
endDate: Date;
|
|
10
|
-
cycleNumber: number;
|
|
11
|
-
isActive: boolean;
|
|
12
|
-
};
|
|
13
|
-
|
|
14
|
-
export type Subscription = {
|
|
15
|
-
clientId: string;
|
|
16
|
-
planId: string;
|
|
17
|
-
planSnapshot: PlanDoc;
|
|
18
|
-
startDate: Date;
|
|
19
|
-
endDate?: Date; // optional, if we want to predefine the end date
|
|
20
|
-
isActive: boolean;
|
|
21
|
-
createdAt: Date;
|
|
22
|
-
updatedAt: Date;
|
|
23
|
-
nextRenewalDate: Date;
|
|
24
|
-
cycles: SubscriptionCycle[];
|
|
25
|
-
};
|
|
26
|
-
|
|
27
|
-
export type SubscriptionDoc = WithId<Subscription>;
|
|
28
|
-
export type Subscriptions = SubscriptionDoc[];
|
|
29
|
-
|
|
30
|
-
// Define which Subscription fields we allow in a simple filter
|
|
31
|
-
export const simpleFields = [
|
|
32
|
-
"clientId",
|
|
33
|
-
"planId",
|
|
34
|
-
"startDate",
|
|
35
|
-
"endDate",
|
|
36
|
-
"isActive",
|
|
37
|
-
"createdAt",
|
|
38
|
-
"nextRenewalDate",
|
|
39
|
-
] as const;
|
|
40
|
-
export type SimpleField = (typeof simpleFields)[number];
|
|
41
|
-
|
|
42
|
-
/**
|
|
43
|
-
* A simple filter for Subscription: each field may be a literal or a MongoDB operator
|
|
44
|
-
*/
|
|
45
|
-
export type SimpleSubscriptionFilter = {
|
|
46
|
-
_id?: ObjectId | string;
|
|
47
|
-
} & {
|
|
48
|
-
[K in keyof Subscription]?: Filter<Subscription>[K];
|
|
49
|
-
};
|
|
50
|
-
|
|
51
|
-
/**
|
|
52
|
-
* Composite filter allowing nested OR/AND arrays
|
|
53
|
-
*/
|
|
54
|
-
export type SubscriptionFilter = SimpleSubscriptionFilter & {
|
|
55
|
-
or?: SimpleSubscriptionFilter[];
|
|
56
|
-
and?: SimpleSubscriptionFilter[];
|
|
57
|
-
};
|
|
58
|
-
|
|
59
|
-
/**
|
|
60
|
-
* Query options for sorting and limiting results
|
|
61
|
-
*/
|
|
62
|
-
export type SubscriptionQueryOptions = {
|
|
63
|
-
sort?: Sort;
|
|
64
|
-
skip?: number;
|
|
65
|
-
limit?: number;
|
|
66
|
-
};
|
|
1
|
+
import { Filter, ObjectId, WithId, Sort } from "mongodb";
|
|
2
|
+
import { PlanDoc } from "../plans";
|
|
3
|
+
|
|
4
|
+
export type SubscriptionCycle = {
|
|
5
|
+
usage: {
|
|
6
|
+
usedCalls: number;
|
|
7
|
+
};
|
|
8
|
+
startDate: Date;
|
|
9
|
+
endDate: Date;
|
|
10
|
+
cycleNumber: number;
|
|
11
|
+
isActive: boolean;
|
|
12
|
+
};
|
|
13
|
+
|
|
14
|
+
export type Subscription = {
|
|
15
|
+
clientId: string;
|
|
16
|
+
planId: string;
|
|
17
|
+
planSnapshot: PlanDoc;
|
|
18
|
+
startDate: Date;
|
|
19
|
+
endDate?: Date; // optional, if we want to predefine the end date
|
|
20
|
+
isActive: boolean;
|
|
21
|
+
createdAt: Date;
|
|
22
|
+
updatedAt: Date;
|
|
23
|
+
nextRenewalDate: Date;
|
|
24
|
+
cycles: SubscriptionCycle[];
|
|
25
|
+
};
|
|
26
|
+
|
|
27
|
+
export type SubscriptionDoc = WithId<Subscription>;
|
|
28
|
+
export type Subscriptions = SubscriptionDoc[];
|
|
29
|
+
|
|
30
|
+
// Define which Subscription fields we allow in a simple filter
|
|
31
|
+
export const simpleFields = [
|
|
32
|
+
"clientId",
|
|
33
|
+
"planId",
|
|
34
|
+
"startDate",
|
|
35
|
+
"endDate",
|
|
36
|
+
"isActive",
|
|
37
|
+
"createdAt",
|
|
38
|
+
"nextRenewalDate",
|
|
39
|
+
] as const;
|
|
40
|
+
export type SimpleField = (typeof simpleFields)[number];
|
|
41
|
+
|
|
42
|
+
/**
|
|
43
|
+
* A simple filter for Subscription: each field may be a literal or a MongoDB operator
|
|
44
|
+
*/
|
|
45
|
+
export type SimpleSubscriptionFilter = {
|
|
46
|
+
_id?: ObjectId | string;
|
|
47
|
+
} & {
|
|
48
|
+
[K in keyof Subscription]?: Filter<Subscription>[K];
|
|
49
|
+
};
|
|
50
|
+
|
|
51
|
+
/**
|
|
52
|
+
* Composite filter allowing nested OR/AND arrays
|
|
53
|
+
*/
|
|
54
|
+
export type SubscriptionFilter = SimpleSubscriptionFilter & {
|
|
55
|
+
or?: SimpleSubscriptionFilter[];
|
|
56
|
+
and?: SimpleSubscriptionFilter[];
|
|
57
|
+
};
|
|
58
|
+
|
|
59
|
+
/**
|
|
60
|
+
* Query options for sorting and limiting results
|
|
61
|
+
*/
|
|
62
|
+
export type SubscriptionQueryOptions = {
|
|
63
|
+
sort?: Sort;
|
|
64
|
+
skip?: number;
|
|
65
|
+
limit?: number;
|
|
66
|
+
};
|