@talkpilot/core-db 1.2.0 → 1.2.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (168) hide show
  1. package/README.md +95 -116
  2. package/README_OLD.md +160 -0
  3. package/dist/talkpilot/calls/calls.dashboard.d.ts +3 -0
  4. package/dist/talkpilot/calls/calls.dashboard.d.ts.map +1 -0
  5. package/dist/talkpilot/calls/calls.dashboard.js +191 -0
  6. package/dist/talkpilot/calls/calls.dashboard.js.map +1 -0
  7. package/dist/talkpilot/calls/calls.getters.d.ts +3 -3
  8. package/dist/talkpilot/calls/calls.getters.d.ts.map +1 -1
  9. package/dist/talkpilot/calls/calls.getters.js +1 -178
  10. package/dist/talkpilot/calls/calls.getters.js.map +1 -1
  11. package/dist/talkpilot/calls/calls.types.d.ts +3 -55
  12. package/dist/talkpilot/calls/calls.types.d.ts.map +1 -1
  13. package/dist/talkpilot/calls/dashboard/calls.dashboard.d.ts +36 -0
  14. package/dist/talkpilot/calls/dashboard/calls.dashboard.d.ts.map +1 -0
  15. package/dist/talkpilot/calls/dashboard/calls.dashboard.js +208 -0
  16. package/dist/talkpilot/calls/dashboard/calls.dashboard.js.map +1 -0
  17. package/dist/talkpilot/calls/dashboard/calls.dashboard.types.d.ts +66 -0
  18. package/dist/talkpilot/calls/dashboard/calls.dashboard.types.d.ts.map +1 -0
  19. package/dist/talkpilot/calls/dashboard/calls.dashboard.types.js +3 -0
  20. package/dist/talkpilot/calls/dashboard/calls.dashboard.types.js.map +1 -0
  21. package/dist/talkpilot/calls/index.d.ts +1 -0
  22. package/dist/talkpilot/calls/index.d.ts.map +1 -1
  23. package/dist/talkpilot/calls/index.js +1 -0
  24. package/dist/talkpilot/calls/index.js.map +1 -1
  25. package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts +2 -2
  26. package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts.map +1 -1
  27. package/dist/talkpilot/clientsConfig/clientsConfig.getters.js +11 -10
  28. package/dist/talkpilot/clientsConfig/clientsConfig.getters.js.map +1 -1
  29. package/dist/talkpilot/clientsConfig/clientsConfig.types.d.ts +20 -9
  30. package/dist/talkpilot/clientsConfig/clientsConfig.types.d.ts.map +1 -1
  31. package/dist/talkpilot/clientsConfig/clientsConfig.types.js +6 -0
  32. package/dist/talkpilot/clientsConfig/clientsConfig.types.js.map +1 -1
  33. package/dist/talkpilot/flows/flows.schema.js +1 -1
  34. package/dist/talkpilot/phone_numbers/index.d.ts +2 -2
  35. package/dist/talkpilot/phone_numbers/phone_numbers.getter.d.ts +1 -1
  36. package/dist/talkpilot/phone_numbers/phone_numbers.getter.d.ts.map +1 -1
  37. package/dist/talkpilot/phone_numbers/phone_numbers.getter.js +5 -3
  38. package/dist/talkpilot/phone_numbers/phone_numbers.getter.js.map +1 -1
  39. package/dist/talkpilot/phone_numbers/phone_numbers.schema.js +12 -12
  40. package/dist/talkpilot/phone_numbers/phone_numbers.types.d.ts +4 -4
  41. package/dist/talkpilot/results/results.getter.d.ts.map +1 -1
  42. package/dist/talkpilot/results/results.getter.js.map +1 -1
  43. package/dist/talkpilot/retry_analyze/retryAnalyze.getters.d.ts.map +1 -1
  44. package/dist/talkpilot/retry_analyze/retryAnalyze.getters.js.map +1 -1
  45. package/dist/utils/shared.types.d.ts +5 -0
  46. package/dist/utils/shared.types.d.ts.map +1 -0
  47. package/dist/utils/shared.types.js +3 -0
  48. package/dist/utils/shared.types.js.map +1 -0
  49. package/package.json +2 -1
  50. package/src/__tests__/setup.ts +20 -20
  51. package/src/connection.ts +42 -42
  52. package/src/index.ts +16 -16
  53. package/src/municipal/__tests__/validation.spec.ts +62 -62
  54. package/src/municipal/cities/cities.getters.ts +50 -50
  55. package/src/municipal/cities/cities.types.ts +11 -11
  56. package/src/municipal/cities/index.ts +2 -2
  57. package/src/municipal/departmentsSubjects/departmentsSubjects.getters.ts +282 -282
  58. package/src/municipal/departmentsSubjects/departmentsSubjects.types.ts +72 -72
  59. package/src/municipal/departmentsSubjects/index.ts +9 -9
  60. package/src/municipal/index.ts +21 -21
  61. package/src/municipal/mongodb-client.ts +61 -61
  62. package/src/municipal/streets/index.ts +2 -2
  63. package/src/municipal/streets/streets.getters.ts +125 -125
  64. package/src/municipal/streets/streets.types.ts +18 -18
  65. package/src/municipal/systemInstructions/__tests__/getters.spec.ts +113 -113
  66. package/src/municipal/systemInstructions/__tests__/setters.spec.ts +274 -274
  67. package/src/municipal/systemInstructions/index.ts +7 -7
  68. package/src/municipal/systemInstructions/instructions.getters.ts +57 -57
  69. package/src/municipal/systemInstructions/instructions.setters.ts +119 -119
  70. package/src/municipal/systemInstructions/instructions.types.ts +30 -30
  71. package/src/municipal/tickets/__tests__/tickets.getters.spec.ts +66 -66
  72. package/src/municipal/tickets/index.ts +2 -2
  73. package/src/municipal/tickets/tickets.getters.ts +261 -261
  74. package/src/municipal/tickets/tickets.types.ts +43 -43
  75. package/src/municipal/utils/types.ts +11 -11
  76. package/src/talkpilot/__tests__/db.spec.ts +38 -38
  77. package/src/talkpilot/__tests__/mongodb-client.spec.ts +18 -18
  78. package/src/talkpilot/__tests__/validation.spec.ts +68 -68
  79. package/src/talkpilot/agents/__tests__/agents.getters.spec.ts +29 -29
  80. package/src/talkpilot/agents/agents.getters.ts +34 -34
  81. package/src/talkpilot/agents/agents.types.ts +14 -14
  82. package/src/talkpilot/agents/index.ts +2 -2
  83. package/src/talkpilot/backgroundToolResults/__tests__/backgroundToolResults.getters.spec.ts +147 -147
  84. package/src/talkpilot/backgroundToolResults/backgroundToolResults.getters.ts +65 -65
  85. package/src/talkpilot/backgroundToolResults/backgroundToolResults.types.ts +23 -23
  86. package/src/talkpilot/backgroundToolResults/index.ts +2 -2
  87. package/src/talkpilot/calls/__tests__/callStats.utils.spec.ts +128 -128
  88. package/src/talkpilot/calls/__tests__/calls.dashboard.spec.ts +46 -0
  89. package/src/talkpilot/calls/__tests__/calls.spec.ts +270 -252
  90. package/src/talkpilot/calls/calls.getters.ts +248 -446
  91. package/src/talkpilot/calls/calls.types.ts +113 -171
  92. package/src/talkpilot/calls/dashboard/calls.dashboard.ts +243 -0
  93. package/src/talkpilot/calls/dashboard/calls.dashboard.types.ts +70 -0
  94. package/src/talkpilot/calls/index.ts +3 -2
  95. package/src/talkpilot/clientAudioBuffers/__tests__/clientAudioBuffer.getters.spec.ts +160 -160
  96. package/src/talkpilot/clientAudioBuffers/clientAudioBuffer.getters.ts +117 -117
  97. package/src/talkpilot/clientAudioBuffers/clientsAudioBuffers.types.ts +25 -25
  98. package/src/talkpilot/clientAudioBuffers/index.ts +2 -2
  99. package/src/talkpilot/clients/clients.getters.ts +16 -16
  100. package/src/talkpilot/clients/clients.types.ts +14 -14
  101. package/src/talkpilot/clients/index.ts +2 -2
  102. package/src/talkpilot/clientsConfig/__tests__/clientsConfig.getters.spec.ts +53 -0
  103. package/src/talkpilot/clientsConfig/__tests__/clientsConfig.spec.ts +197 -106
  104. package/src/talkpilot/clientsConfig/clientsConfig.getters.ts +55 -44
  105. package/src/talkpilot/clientsConfig/clientsConfig.types.ts +127 -94
  106. package/src/talkpilot/clientsConfig/index.ts +2 -2
  107. package/src/talkpilot/flows/__tests__/flows.schema.spec.ts +71 -67
  108. package/src/talkpilot/flows/flows.getter.ts +14 -14
  109. package/src/talkpilot/flows/flows.schema.ts +153 -153
  110. package/src/talkpilot/flows/flows.types.ts +184 -184
  111. package/src/talkpilot/flows/index.ts +2 -2
  112. package/src/talkpilot/groups/__tests__/groups.spec.ts +90 -90
  113. package/src/talkpilot/groups/__tests__/phone.utils.spec.ts +32 -32
  114. package/src/talkpilot/groups/groups.getters.ts +30 -30
  115. package/src/talkpilot/groups/groups.types.ts +29 -29
  116. package/src/talkpilot/groups/index.ts +3 -3
  117. package/src/talkpilot/groups/phone.utils.ts +46 -46
  118. package/src/talkpilot/index.ts +29 -29
  119. package/src/talkpilot/leads/index.ts +2 -2
  120. package/src/talkpilot/leads/leads.getter.ts +6 -6
  121. package/src/talkpilot/leads/leads.schema.ts +33 -33
  122. package/src/talkpilot/leads/leads.types.ts +20 -20
  123. package/src/talkpilot/mongodb-client.ts +78 -78
  124. package/src/talkpilot/phone_numbers/__tests__/phone_numbers.spec.ts +252 -247
  125. package/src/talkpilot/phone_numbers/index.ts +2 -2
  126. package/src/talkpilot/phone_numbers/phone_numbers.getter.ts +158 -154
  127. package/src/talkpilot/phone_numbers/phone_numbers.schema.ts +17 -17
  128. package/src/talkpilot/phone_numbers/phone_numbers.types.ts +30 -30
  129. package/src/talkpilot/plans/__tests__/plans.spec.ts +70 -70
  130. package/src/talkpilot/plans/index.ts +2 -2
  131. package/src/talkpilot/plans/plans.getters.ts +132 -132
  132. package/src/talkpilot/plans/plans.types.ts +89 -89
  133. package/src/talkpilot/results/index.ts +7 -7
  134. package/src/talkpilot/results/results.getter.ts +39 -35
  135. package/src/talkpilot/results/results.schema.ts +25 -25
  136. package/src/talkpilot/results/results.types.ts +34 -34
  137. package/src/talkpilot/retry_analyze/__tests__/retryAnalyze.getters.spec.ts +156 -156
  138. package/src/talkpilot/retry_analyze/index.ts +2 -2
  139. package/src/talkpilot/retry_analyze/retryAnalyze.getters.ts +84 -75
  140. package/src/talkpilot/retry_analyze/retryAnalyze.types.ts +13 -13
  141. package/src/talkpilot/sessions/__tests__/sessions.spec.ts +147 -147
  142. package/src/talkpilot/sessions/index.ts +2 -2
  143. package/src/talkpilot/sessions/sessions.getter.ts +92 -92
  144. package/src/talkpilot/sessions/sessions.schema.ts +34 -34
  145. package/src/talkpilot/sessions/sessions.types.ts +30 -30
  146. package/src/talkpilot/subscriptions/__tests__/subscriptions.getters.utils.spec.ts +45 -45
  147. package/src/talkpilot/subscriptions/index.ts +3 -3
  148. package/src/talkpilot/subscriptions/subscriptions.getters.ts +146 -146
  149. package/src/talkpilot/subscriptions/subscriptions.getters.utils.ts +33 -33
  150. package/src/talkpilot/subscriptions/subscriptions.types.ts +66 -66
  151. package/src/talkpilot/utils/__tests__/query.utils.spec.ts +49 -49
  152. package/src/talkpilot/utils/query.utils.ts +21 -21
  153. package/src/test-utils/db-utils.ts +24 -24
  154. package/src/test-utils/factories/index.ts +12 -12
  155. package/src/test-utils/factories/municipal/cities.ts +16 -16
  156. package/src/test-utils/factories/municipal/departmentsSubjects.ts +37 -37
  157. package/src/test-utils/factories/municipal/streets.ts +22 -22
  158. package/src/test-utils/factories/municipal/tickets.ts +39 -39
  159. package/src/test-utils/factories/talkpilot/agents.ts +19 -19
  160. package/src/test-utils/factories/talkpilot/calls.ts +37 -37
  161. package/src/test-utils/factories/talkpilot/clientAudioBuffers.ts +20 -20
  162. package/src/test-utils/factories/talkpilot/clientsConfig.ts +18 -18
  163. package/src/test-utils/factories/talkpilot/flows.ts +33 -33
  164. package/src/test-utils/factories/talkpilot/groups.ts +33 -33
  165. package/src/test-utils/factories/talkpilot/phone_numbers.ts +22 -22
  166. package/src/test-utils/factories/talkpilot/sessions.ts +35 -35
  167. package/src/utils/shared.types.ts +4 -0
  168. package/src/utils/validation.ts +23 -23
@@ -1,446 +1,248 @@
1
- import {
2
- Call,
3
- CallDoc,
4
- CallQueryOptions,
5
- CallUpdateParams,
6
- getDb,
7
- } from "../index";
8
- import type { CallsByHour, CountOpts, DateRange, ToolExecution,HourlyBucket,DailyBucket,CallLengthBuckets,DashboardKpis,DashboardCharts,RecentCall,DashboardStatsResult, DashboardStatsParams } from "./calls.types";
9
- import { Filter, ObjectId } from "mongodb";
10
- import * as process from "node:process";
11
- import { applyQueryOptions } from "../utils/query.utils";
12
-
13
- const TIMEZONE = 'UTC';
14
-
15
- export const getCallsCollection = () => {
16
- return getDb().collection<Call>("calls");
17
- };
18
-
19
- export const getCallByCallSid = (callSid: string) => {
20
- return getCallsCollection().findOne({ callSid });
21
- };
22
-
23
- export const getCallsByPhoneNumber = (phoneNumber: string) => {
24
- return getCallsCollection()
25
- .find({ customerPhoneNumber: phoneNumber })
26
- .toArray();
27
- };
28
-
29
- export const getCallsByClient = (clientId: string) => {
30
- return getCallsCollection().find({ clientId }).toArray();
31
- };
32
-
33
- export const getCallsByFlow = (flowId: ObjectId) => {
34
- return getCallsCollection().find({ flowId }).toArray();
35
- };
36
-
37
- export const createCallDoc = (
38
- call: Omit<Call, "createdAt" | "updatedAt" | "env">,
39
- ) => {
40
- return getCallsCollection().insertOne({
41
- ...call,
42
- createdAt: new Date(),
43
- updatedAt: new Date(),
44
- env: process.env.ENV ?? "unknown",
45
- });
46
- };
47
-
48
- export const updateCallByCallSid = async (
49
- callSid: string,
50
- updates: CallUpdateParams,
51
- ): Promise<CallDoc | null> => {
52
- return await getCallsCollection().findOneAndUpdate(
53
- { callSid: callSid },
54
- {
55
- $set: {
56
- ...updates,
57
- updatedAt: new Date(),
58
- },
59
- },
60
- { returnDocument: "after" },
61
- );
62
- };
63
-
64
- export const pushToolExecution = async (
65
- callSid: string,
66
- execution: ToolExecution
67
- ): Promise<void> => {
68
- await getCallsCollection().updateOne(
69
- { callSid },
70
- { $push: { toolExecutions: execution } }
71
- );
72
- };
73
-
74
- // get calls by client and date range
75
- //client here is the user id
76
- export const getCallsByClientAndDateRange = (
77
- clientId: string,
78
- startDate: Date,
79
- endDate: Date,
80
- ) => {
81
- return getCallsCollection()
82
- .find({
83
- clientId,
84
- createdAt: {
85
- $gte: startDate,
86
- $lte: endDate,
87
- },
88
- })
89
- .toArray();
90
- };
91
-
92
- export const findCallsByQuery = async (
93
- query: Filter<Call>,
94
- options?: CallQueryOptions,
95
- ): Promise<CallDoc[]> => {
96
- const cursor = getCallsCollection().find(query);
97
- return await applyQueryOptions(cursor, options).toArray();
98
- };
99
-
100
- export const countCalls = async (query: Filter<Call>): Promise<number> => {
101
- return getCallsCollection().countDocuments(query);
102
- };
103
-
104
- export async function countCallsByPhoneInRange(
105
- customerPhoneNumber: string,
106
- opts: CountOpts = {},
107
- range: DateRange = {},
108
- ): Promise<number> {
109
- const filter: Partial<Call> & { customerPhoneNumber: string } = {
110
- customerPhoneNumber,
111
- };
112
-
113
- if (opts.isOutgoingCall !== undefined) {
114
- filter.isOutgoingCall = opts.isOutgoingCall;
115
- }
116
- if (opts.isIncomingCall !== undefined) {
117
- filter.isIncomingCall = opts.isIncomingCall;
118
- }
119
-
120
- if (range.since || range.until) {
121
- (filter as Filter<Call>).createdAt = {
122
- ...(range.since ? { $gte: range.since } : {}),
123
- ...(range.until ? { $lt: range.until } : {}),
124
- };
125
- }
126
-
127
- return getCallsCollection().countDocuments(filter as Filter<Call>);
128
- }
129
-
130
- export const range = {
131
- between: (since: Date, until: Date): DateRange => ({ since, until }),
132
- todayToNow: (): DateRange => ({ since: startOfDay(), until: new Date() }),
133
- yesterday: (): DateRange => {
134
- const s = startOfDay(addDays(new Date(), -1));
135
- const e = addDays(s, 1);
136
- return { since: s, until: e };
137
- },
138
- todayFull: (): DateRange => {
139
- const s = startOfDay();
140
- const e = addDays(s, 1);
141
- return { since: s, until: e };
142
- },
143
- startOfDayToNow: (d: Date): DateRange => ({
144
- since: startOfDay(d),
145
- until: new Date(),
146
- }),
147
- };
148
-
149
- export function startOfDay(d: Date = new Date()): Date {
150
- const x = new Date(d);
151
- x.setUTCHours(0, 0, 0, 0);
152
- return x;
153
- }
154
-
155
- export function addDays(d: Date, days: number): Date {
156
- const x = new Date(d);
157
- x.setDate(x.getDate() + days);
158
- return x;
159
- }
160
-
161
- /**
162
- * Aggregate calls stats for a date range (createdAt in [startStr, endStr] in timezone).
163
- * completed = agentHungUp=true or status='completed'.
164
- */
165
- export async function getCallsStatsForDateRange(
166
- clientId: string,
167
- startStr: string,
168
- endStr: string,
169
- timezone: string,
170
- ): Promise<{ count: number; totalLen: number; completed: number }> {
171
- const coll = getCallsCollection();
172
- const out = await coll
173
- .aggregate<{
174
- _id: null;
175
- count: number;
176
- totalLen: number;
177
- completed: number;
178
- }>([
179
- // 1. Restrict to the given client
180
- { $match: { clientId } },
181
- // 2. Derive dateLocal (createdAt as YYYY-MM-DD in timezone) and isCompleted (agentHungUp or status='completed')
182
- {
183
- $addFields: {
184
- dateLocal: {
185
- $dateToString: { format: "%Y-%m-%d", date: "$createdAt", timezone },
186
- },
187
- isCompleted: {
188
- $or: [
189
- { $eq: ["$agentHungUp", true] },
190
- { $eq: ["$status", "completed"] },
191
- ],
192
- },
193
- },
194
- },
195
- // 3. Keep only documents with dateLocal in [startStr, endStr] (inclusive)
196
- { $match: { dateLocal: { $gte: startStr, $lte: endStr } } },
197
- // 4. Single group: count, totalLen, completed
198
- {
199
- $group: {
200
- _id: null,
201
- count: { $sum: 1 },
202
- totalLen: { $sum: "$callLength" },
203
- completed: { $sum: { $cond: ["$isCompleted", 1, 0] } },
204
- },
205
- },
206
- ])
207
- .next();
208
- return {
209
- count: out?.count ?? 0,
210
- totalLen: out?.totalLen ?? 0,
211
- completed: out?.completed ?? 0,
212
- };
213
- }
214
-
215
- /**
216
- * Aggregate calls by hour for a given date (createdAt converted to given timezone). Hour format "HH:mm".
217
- */
218
- export async function getCallsHourlyAggregation(
219
- clientId: string,
220
- dateStr: string,
221
- timezone: string,
222
- ): Promise<CallsByHour[]> {
223
- const coll = getCallsCollection();
224
- const rows = await coll
225
- .aggregate<{ _id: number; calls: number }>([
226
- // 1. Restrict to the given client
227
- { $match: { clientId } },
228
- // 2. Derive dateLocal (createdAt as YYYY-MM-DD in timezone) and hour (0–23 from createdAt in timezone)
229
- {
230
- $addFields: {
231
- dateLocal: {
232
- $dateToString: { format: "%Y-%m-%d", date: "$createdAt", timezone },
233
- },
234
- hour: { $hour: { date: "$createdAt", timezone } },
235
- },
236
- },
237
- // 3. Keep only the requested date
238
- { $match: { dateLocal: dateStr } },
239
- // 4. Group by hour, sum calls per hour
240
- { $group: { _id: "$hour", calls: { $sum: 1 } } },
241
- // 5. Order by hour ascending (0..23)
242
- { $sort: { _id: 1 } },
243
- ])
244
- .toArray();
245
-
246
- return rows.map((r: any) => ({
247
- hour: `${String(r._id).padStart(2, "0")}:00`,
248
- calls: r.calls,
249
- }));
250
- }
251
-
252
- function buildKpisPipeline() {
253
- return [
254
- {
255
- $group: {
256
- _id: null,
257
- totalCalls: { $sum: 1 },
258
- totalDuration: { $sum: '$callLength' },
259
- completedCount: {
260
- $sum: { $cond: [{ $eq: ['$status', 'completed'] }, 1, 0] },
261
- },
262
- failedCount: {
263
- $sum: { $cond: [{ $eq: ['$status', 'failed'] }, 1, 0] },
264
- },
265
- noAnswerCount: {
266
- $sum: { $cond: [{ $eq: ['$status', 'no-answer'] }, 1, 0] },
267
- },
268
- busyCount: {
269
- $sum: { $cond: [{ $eq: ['$status', 'busy'] }, 1, 0] },
270
- },
271
- },
272
- },
273
- ];
274
- }
275
-
276
- function buildDailyDataPipeline(timezone: string) {
277
- return [
278
- {
279
- $group: {
280
- _id: {
281
- $dateToString: { format: '%Y-%m-%d', date: '$createdAt', timezone: timezone },
282
- },
283
- count: { $sum: 1 },
284
- completed: {
285
- $sum: { $cond: [{ $eq: ['$status', 'completed'] }, 1, 0] },
286
- },
287
- },
288
- },
289
- { $sort: { _id: 1 } },
290
- ];
291
- }
292
-
293
- function buildHourlyDataPipeline(timezone: string) {
294
- return [
295
- {
296
- $group: {
297
- _id: {
298
- day: { $dateToString: { format: '%Y-%m-%d', date: '$createdAt', timezone: timezone } },
299
- hour: { $hour: { date: '$createdAt', timezone: timezone } },
300
- },
301
- count: { $sum: 1 },
302
- },
303
- },
304
- ];
305
- }
306
-
307
- function buildCallLengthBucketsPipeline() {
308
- return [
309
- {
310
- $group: {
311
- _id: null,
312
- short: { $sum: { $cond: [{ $lt: ['$callLength', 60] }, 1, 0] } },
313
- medium: {
314
- $sum: {
315
- $cond: [
316
- { $and: [{ $gte: ['$callLength', 60] }, { $lte: ['$callLength', 180] }] },
317
- 1,
318
- 0,
319
- ],
320
- },
321
- },
322
- long: { $sum: { $cond: [{ $gt: ['$callLength', 180] }, 1, 0] } },
323
- },
324
- },
325
- ];
326
- }
327
-
328
- function buildRecentCallsPipeline() {
329
- return [
330
- { $sort: { createdAt: -1 } },
331
- { $limit: 8 },
332
- {
333
- $project: {
334
- callSid: 1,
335
- customerPhoneNumber: 1,
336
- status: 1,
337
- callLength: 1,
338
- createdAt: 1,
339
- summary: 1,
340
- isIncomingCall: 1,
341
- },
342
- },
343
- ];
344
- }
345
-
346
- export async function getDashboardStats(
347
- params: DashboardStatsParams,
348
- ): Promise<DashboardStatsResult> {
349
- const { clientId, startDate, endDate } = params;
350
-
351
- const startDateObj = new Date(startDate);
352
- const endDateObj = new Date(endDate);
353
-
354
- const startStr = startDateObj.toISOString().slice(0, 10);
355
- const endStr = endDateObj.toISOString().slice(0, 10);
356
-
357
- console.info('[core-db] Fetching dashboard stats', { clientId, startStr, endStr });
358
- const pipeline = [
359
- {
360
- $match: {
361
- clientId,
362
- createdAt: { $gte: startDateObj, $lte: endDateObj },
363
- },
364
- },
365
- {
366
- $facet: {
367
- kpis: buildKpisPipeline(),
368
- dailyData: buildDailyDataPipeline(TIMEZONE),
369
- hourlyData: buildHourlyDataPipeline(TIMEZONE),
370
- callLengthBuckets: buildCallLengthBucketsPipeline(),
371
- recentCalls: buildRecentCallsPipeline(),
372
- },
373
- },
374
- ];
375
-
376
- const callsCollection = getCallsCollection();
377
- const [aggregatedResult] = await callsCollection.aggregate(pipeline).toArray();
378
-
379
- const kpiData = aggregatedResult?.kpis?.[0] ?? {
380
- totalCalls: 0,
381
- totalDuration: 0,
382
- completedCount: 0,
383
- failedCount: 0,
384
- noAnswerCount: 0,
385
- busyCount: 0,
386
- };
387
- const dailyDataRaw = aggregatedResult?.dailyData ?? [];
388
- const hourlyDataRaw = aggregatedResult?.hourlyData ?? [];
389
- const callLengthRaw = aggregatedResult?.callLengthBuckets?.[0] ?? { short: 0, medium: 0, long: 0 };
390
- const recentCallsRaw: RecentCall[] = aggregatedResult?.recentCalls ?? [];
391
-
392
- const count = kpiData.totalCalls;
393
- const totalLen = kpiData.totalDuration;
394
- const completed = kpiData.completedCount;
395
-
396
- const kpis: DashboardKpis = {
397
- totalCalls: count,
398
- avgDurationSeconds: count > 0 ? Math.round(totalLen / count) : 0,
399
- timeSavedMinutes: Math.round(totalLen / 60),
400
- successRate: count > 0 ? Math.round((completed / count) * 1000) / 10 : 0,
401
- completedCount: completed,
402
- failedCount: kpiData.failedCount,
403
- noAnswerCount: kpiData.noAnswerCount,
404
- busyCount: kpiData.busyCount,
405
- };
406
-
407
- const volumeData: DailyBucket[] = dailyDataRaw.map((d: any) => ({
408
- date: d._id,
409
- count: d.count,
410
- completed: d.completed,
411
- }));
412
-
413
- const heatmapMap = new Map<string, Map<number, number>>();
414
- for (const bucket of hourlyDataRaw) {
415
- const dayKey = bucket._id.day;
416
- const hour = bucket._id.hour;
417
- if (!heatmapMap.has(dayKey)) heatmapMap.set(dayKey, new Map());
418
- heatmapMap.get(dayKey)!.set(hour, bucket.count);
419
- }
420
-
421
- const toSortedBuckets = (map: Map<number, number>): HourlyBucket[] =>
422
- Array.from(map.entries())
423
- .sort(([a], [b]) => a - b)
424
- .map(([h, c]) => ({ hour: `${String(h).padStart(2, '0')}:00`, calls: c }));
425
-
426
- const heatmap: Record<string, HourlyBucket[]> = {};
427
- for (const [day, hourMap] of Array.from(heatmapMap.entries()).sort()) {
428
- heatmap[day] = toSortedBuckets(hourMap);
429
- }
430
-
431
- return {
432
- kpis,
433
- charts: {
434
- volumeData,
435
- heatmap,
436
- callLengthBuckets: {
437
- short: callLengthRaw.short,
438
- medium: callLengthRaw.medium,
439
- long: callLengthRaw.long,
440
- },
441
- },
442
- recentCalls: recentCallsRaw,
443
- };
444
- }
445
-
446
-
1
+ import {
2
+ Call,
3
+ CallDoc,
4
+ CallQueryOptions,
5
+ CallUpdateParams,
6
+ getDb,
7
+ } from "../index";
8
+ import type { CountOpts, DateRange, ToolExecution } from "./calls.types";
9
+ import { Filter, ObjectId } from "mongodb";
10
+ import * as process from "node:process";
11
+ import { applyQueryOptions } from "../utils/query.utils";
12
+ import { DashboardHeatmapMetric } from "./dashboard/calls.dashboard.types";
13
+
14
+ export const getCallsCollection = () => {
15
+ return getDb().collection<Call>("calls");
16
+ };
17
+
18
+ export const getCallByCallSid = (callSid: string) => {
19
+ return getCallsCollection().findOne({ callSid });
20
+ };
21
+
22
+ export const getCallsByPhoneNumber = (phoneNumber: string) => {
23
+ return getCallsCollection()
24
+ .find({ customerPhoneNumber: phoneNumber })
25
+ .toArray();
26
+ };
27
+
28
+ export const getCallsByClient = (clientId: string) => {
29
+ return getCallsCollection().find({ clientId }).toArray();
30
+ };
31
+
32
+ export const getCallsByFlow = (flowId: ObjectId) => {
33
+ return getCallsCollection().find({ flowId }).toArray();
34
+ };
35
+
36
+ export const createCallDoc = (
37
+ call: Omit<Call, "createdAt" | "updatedAt" | "env">,
38
+ ) => {
39
+ return getCallsCollection().insertOne({
40
+ ...call,
41
+ createdAt: new Date(),
42
+ updatedAt: new Date(),
43
+ env: process.env.ENV ?? "unknown",
44
+ });
45
+ };
46
+
47
+ export const updateCallByCallSid = async (
48
+ callSid: string,
49
+ updates: CallUpdateParams,
50
+ ): Promise<CallDoc | null> => {
51
+ return await getCallsCollection().findOneAndUpdate(
52
+ { callSid: callSid },
53
+ {
54
+ $set: {
55
+ ...updates,
56
+ updatedAt: new Date(),
57
+ },
58
+ },
59
+ { returnDocument: "after" },
60
+ );
61
+ };
62
+
63
+ export const pushToolExecution = async (
64
+ callSid: string,
65
+ execution: ToolExecution,
66
+ ): Promise<void> => {
67
+ await getCallsCollection().updateOne(
68
+ { callSid },
69
+ { $push: { toolExecutions: execution } },
70
+ );
71
+ };
72
+
73
+ // get calls by client and date range
74
+ //client here is the user id
75
+ export const getCallsByClientAndDateRange = (
76
+ clientId: string,
77
+ startDate: Date,
78
+ endDate: Date,
79
+ ) => {
80
+ return getCallsCollection()
81
+ .find({
82
+ clientId,
83
+ createdAt: {
84
+ $gte: startDate,
85
+ $lte: endDate,
86
+ },
87
+ })
88
+ .toArray();
89
+ };
90
+
91
+ export const findCallsByQuery = async (
92
+ query: Filter<Call>,
93
+ options?: CallQueryOptions,
94
+ ): Promise<CallDoc[]> => {
95
+ const cursor = getCallsCollection().find(query);
96
+ return await applyQueryOptions(cursor, options).toArray();
97
+ };
98
+
99
+ export const countCalls = async (query: Filter<Call>): Promise<number> => {
100
+ return getCallsCollection().countDocuments(query);
101
+ };
102
+
103
+ export async function countCallsByPhoneInRange(
104
+ customerPhoneNumber: string,
105
+ opts: CountOpts = {},
106
+ range: DateRange = {},
107
+ ): Promise<number> {
108
+ const filter: Partial<Call> & { customerPhoneNumber: string } = {
109
+ customerPhoneNumber,
110
+ };
111
+
112
+ if (opts.isOutgoingCall !== undefined) {
113
+ filter.isOutgoingCall = opts.isOutgoingCall;
114
+ }
115
+ if (opts.isIncomingCall !== undefined) {
116
+ filter.isIncomingCall = opts.isIncomingCall;
117
+ }
118
+
119
+ if (range.since || range.until) {
120
+ (filter as Filter<Call>).createdAt = {
121
+ ...(range.since ? { $gte: range.since } : {}),
122
+ ...(range.until ? { $lt: range.until } : {}),
123
+ };
124
+ }
125
+
126
+ return getCallsCollection().countDocuments(filter as Filter<Call>);
127
+ }
128
+
129
+ export const range = {
130
+ between: (since: Date, until: Date): DateRange => ({ since, until }),
131
+ todayToNow: (): DateRange => ({ since: startOfDay(), until: new Date() }),
132
+ yesterday: (): DateRange => {
133
+ const s = startOfDay(addDays(new Date(), -1));
134
+ const e = addDays(s, 1);
135
+ return { since: s, until: e };
136
+ },
137
+ todayFull: (): DateRange => {
138
+ const s = startOfDay();
139
+ const e = addDays(s, 1);
140
+ return { since: s, until: e };
141
+ },
142
+ startOfDayToNow: (d: Date): DateRange => ({
143
+ since: startOfDay(d),
144
+ until: new Date(),
145
+ }),
146
+ };
147
+
148
+ export function startOfDay(d: Date = new Date()): Date {
149
+ const x = new Date(d);
150
+ x.setUTCHours(0, 0, 0, 0);
151
+ return x;
152
+ }
153
+
154
+ export function addDays(d: Date, days: number): Date {
155
+ const x = new Date(d);
156
+ x.setDate(x.getDate() + days);
157
+ return x;
158
+ }
159
+
160
+ /**
161
+ * Aggregate calls stats for a date range (createdAt in [startStr, endStr] in timezone).
162
+ * completed = agentHungUp=true or status='completed'.
163
+ */
164
+ export async function getCallsStatsForDateRange(
165
+ clientId: string,
166
+ startStr: string,
167
+ endStr: string,
168
+ timezone: string,
169
+ ): Promise<{ count: number; totalLen: number; completed: number }> {
170
+ const coll = getCallsCollection();
171
+ const out = await coll
172
+ .aggregate<{
173
+ _id: null;
174
+ count: number;
175
+ totalLen: number;
176
+ completed: number;
177
+ }>([
178
+ // 1. Restrict to the given client
179
+ { $match: { clientId } },
180
+ // 2. Derive dateLocal (createdAt as YYYY-MM-DD in timezone) and isCompleted (agentHungUp or status='completed')
181
+ {
182
+ $addFields: {
183
+ dateLocal: {
184
+ $dateToString: { format: "%Y-%m-%d", date: "$createdAt", timezone },
185
+ },
186
+ isCompleted: {
187
+ $or: [
188
+ { $eq: ["$agentHungUp", true] },
189
+ { $eq: ["$status", "completed"] },
190
+ ],
191
+ },
192
+ },
193
+ },
194
+ // 3. Keep only documents with dateLocal in [startStr, endStr] (inclusive)
195
+ { $match: { dateLocal: { $gte: startStr, $lte: endStr } } },
196
+ // 4. Single group: count, totalLen, completed
197
+ {
198
+ $group: {
199
+ _id: null,
200
+ count: { $sum: 1 },
201
+ totalLen: { $sum: "$callLength" },
202
+ completed: { $sum: { $cond: ["$isCompleted", 1, 0] } },
203
+ },
204
+ },
205
+ ])
206
+ .next();
207
+ return {
208
+ count: out?.count ?? 0,
209
+ totalLen: out?.totalLen ?? 0,
210
+ completed: out?.completed ?? 0,
211
+ };
212
+ }
213
+
214
+ /**
215
+ * Aggregate calls by hour for a given date (createdAt converted to given timezone). Hour format "HH:mm".
216
+ */
217
+ export async function getCallsHourlyAggregation(
218
+ clientId: string,
219
+ dateStr: string,
220
+ timezone: string,
221
+ ): Promise<DashboardHeatmapMetric[]> {
222
+ const coll = getCallsCollection();
223
+ const rows = await coll
224
+ .aggregate<{ _id: number; calls: number }>([
225
+ // 1. Restrict to the given client
226
+ { $match: { clientId } },
227
+ {
228
+ $addFields: {
229
+ dateLocal: {
230
+ $dateToString: { format: "%Y-%m-%d", date: "$createdAt", timezone },
231
+ },
232
+ hour: { $hour: { date: "$createdAt", timezone } },
233
+ },
234
+ },
235
+ // 3. Keep only the requested date
236
+ { $match: { dateLocal: dateStr } },
237
+ // 4. Group by hour, sum calls per hour
238
+ { $group: { _id: "$hour", calls: { $sum: 1 } } },
239
+ // 5. Order by hour ascending (0..23)
240
+ { $sort: { _id: 1 } },
241
+ ])
242
+ .toArray();
243
+
244
+ return rows.map((r: any) => ({
245
+ hour: r._id,
246
+ calls: r.calls,
247
+ }));
248
+ }