@talkpilot/core-db 1.3.1 → 1.3.4
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/.claude/settings.local.json +7 -0
- package/README.md +30 -0
- package/dist/municipal/tickets/index.d.ts +2 -1
- package/dist/municipal/tickets/index.d.ts.map +1 -1
- package/dist/municipal/tickets/index.js +1 -0
- package/dist/municipal/tickets/index.js.map +1 -1
- package/dist/municipal/tickets/tickets.constants.d.ts +7 -0
- package/dist/municipal/tickets/tickets.constants.d.ts.map +1 -0
- package/dist/municipal/tickets/tickets.constants.js +10 -0
- package/dist/municipal/tickets/tickets.constants.js.map +1 -0
- package/dist/municipal/tickets/tickets.deprecated.getters.d.ts +12 -0
- package/dist/municipal/tickets/tickets.deprecated.getters.d.ts.map +1 -0
- package/dist/municipal/tickets/tickets.deprecated.getters.js +131 -0
- package/dist/municipal/tickets/tickets.deprecated.getters.js.map +1 -0
- package/dist/municipal/tickets/tickets.getters.d.ts +0 -11
- package/dist/municipal/tickets/tickets.getters.d.ts.map +1 -1
- package/dist/municipal/tickets/tickets.getters.js +0 -128
- package/dist/municipal/tickets/tickets.getters.js.map +1 -1
- package/dist/municipal/tickets/tickets.statistics.aggregation.d.ts +35 -0
- package/dist/municipal/tickets/tickets.statistics.aggregation.d.ts.map +1 -0
- package/dist/municipal/tickets/tickets.statistics.aggregation.js +86 -0
- package/dist/municipal/tickets/tickets.statistics.aggregation.js.map +1 -0
- package/dist/municipal/tickets/tickets.statistics.dates.d.ts +7 -0
- package/dist/municipal/tickets/tickets.statistics.dates.d.ts.map +1 -0
- package/dist/municipal/tickets/tickets.statistics.dates.js +40 -0
- package/dist/municipal/tickets/tickets.statistics.dates.js.map +1 -0
- package/dist/municipal/tickets/tickets.statistics.getters.d.ts +8 -0
- package/dist/municipal/tickets/tickets.statistics.getters.d.ts.map +1 -0
- package/dist/municipal/tickets/tickets.statistics.getters.js +44 -0
- package/dist/municipal/tickets/tickets.statistics.getters.js.map +1 -0
- package/dist/municipal/tickets/tickets.statistics.pipeline.d.ts +53 -0
- package/dist/municipal/tickets/tickets.statistics.pipeline.d.ts.map +1 -0
- package/dist/municipal/tickets/tickets.statistics.pipeline.js +112 -0
- package/dist/municipal/tickets/tickets.statistics.pipeline.js.map +1 -0
- package/dist/municipal/tickets/tickets.statistics.utils.d.ts +7 -0
- package/dist/municipal/tickets/tickets.statistics.utils.d.ts.map +1 -0
- package/dist/municipal/tickets/tickets.statistics.utils.js +40 -0
- package/dist/municipal/tickets/tickets.statistics.utils.js.map +1 -0
- package/dist/municipal/tickets/tickets.types.d.ts +14 -5
- package/dist/municipal/tickets/tickets.types.d.ts.map +1 -1
- package/dist/talkpilot/calls/calls.constants.d.ts +17 -0
- package/dist/talkpilot/calls/calls.constants.d.ts.map +1 -0
- package/dist/talkpilot/calls/calls.constants.js +20 -0
- package/dist/talkpilot/calls/calls.constants.js.map +1 -0
- package/dist/talkpilot/calls/calls.statistics.getters.d.ts +28 -0
- package/dist/talkpilot/calls/calls.statistics.getters.d.ts.map +1 -0
- package/dist/talkpilot/calls/calls.statistics.getters.js +424 -0
- package/dist/talkpilot/calls/calls.statistics.getters.js.map +1 -0
- package/dist/talkpilot/calls/calls.statistics.ticketScope.d.ts +12 -0
- package/dist/talkpilot/calls/calls.statistics.ticketScope.d.ts.map +1 -0
- package/dist/talkpilot/calls/calls.statistics.ticketScope.js +37 -0
- package/dist/talkpilot/calls/calls.statistics.ticketScope.js.map +1 -0
- package/dist/talkpilot/calls/calls.statistics.tickets.d.ts +17 -0
- package/dist/talkpilot/calls/calls.statistics.tickets.d.ts.map +1 -0
- package/dist/talkpilot/calls/calls.statistics.tickets.js +33 -0
- package/dist/talkpilot/calls/calls.statistics.tickets.js.map +1 -0
- package/dist/talkpilot/calls/calls.statistics.types.d.ts +39 -0
- package/dist/talkpilot/calls/calls.statistics.types.d.ts.map +1 -0
- package/dist/talkpilot/calls/calls.statistics.types.js +3 -0
- package/dist/talkpilot/calls/calls.statistics.types.js.map +1 -0
- package/dist/talkpilot/calls/calls.types.d.ts +1 -2
- package/dist/talkpilot/calls/calls.types.d.ts.map +1 -1
- package/dist/talkpilot/calls/calls.types.js +0 -3
- package/dist/talkpilot/calls/calls.types.js.map +1 -1
- package/dist/talkpilot/calls/dashboard/calls.dashboard.d.ts +33 -1
- package/dist/talkpilot/calls/dashboard/calls.dashboard.d.ts.map +1 -1
- package/dist/talkpilot/calls/dashboard/calls.dashboard.js +131 -146
- package/dist/talkpilot/calls/dashboard/calls.dashboard.js.map +1 -1
- package/dist/talkpilot/calls/dashboard/calls.dashboard.types.d.ts +27 -6
- package/dist/talkpilot/calls/dashboard/calls.dashboard.types.d.ts.map +1 -1
- package/dist/talkpilot/calls/index.d.ts +3 -0
- package/dist/talkpilot/calls/index.d.ts.map +1 -1
- package/dist/talkpilot/calls/index.js +3 -0
- package/dist/talkpilot/calls/index.js.map +1 -1
- package/dist/utils/date.utils.d.ts +49 -0
- package/dist/utils/date.utils.d.ts.map +1 -0
- package/dist/utils/date.utils.js +103 -0
- package/dist/utils/date.utils.js.map +1 -0
- package/dist/utils/statistics.aggregation.d.ts +20 -0
- package/dist/utils/statistics.aggregation.d.ts.map +1 -0
- package/dist/utils/statistics.aggregation.js +43 -0
- package/dist/utils/statistics.aggregation.js.map +1 -0
- package/package.json +1 -1
- package/src/municipal/tickets/__tests__/tickets.getters.spec.ts +1 -37
- package/src/municipal/tickets/__tests__/tickets.statistics.spec.ts +51 -0
- package/src/municipal/tickets/index.ts +2 -1
- package/src/municipal/tickets/tickets.constants.ts +8 -0
- package/src/municipal/tickets/tickets.getters.ts +0 -140
- package/src/municipal/tickets/tickets.statistics.aggregation.ts +96 -0
- package/src/municipal/tickets/tickets.statistics.getters.ts +71 -0
- package/src/municipal/tickets/tickets.types.ts +19 -9
- package/src/talkpilot/calls/__tests__/calls.dashboard.spec.ts +8 -111
- package/src/talkpilot/calls/__tests__/calls.statistics.spec.ts +344 -0
- package/src/talkpilot/calls/calls.constants.ts +20 -0
- package/src/talkpilot/calls/calls.statistics.getters.ts +587 -0
- package/src/talkpilot/calls/calls.statistics.types.ts +44 -0
- package/src/talkpilot/calls/calls.types.ts +4 -2
- package/src/talkpilot/calls/dashboard/calls.dashboard.ts +148 -197
- package/src/talkpilot/calls/dashboard/calls.dashboard.types.ts +25 -12
- package/src/talkpilot/calls/index.ts +3 -0
- package/src/talkpilot/clientsConfig/__tests__/clientsConfig.spec.ts +7 -0
- package/src/utils/date.utils.ts +116 -0
|
@@ -0,0 +1,587 @@
|
|
|
1
|
+
import { getCallsCollection } from "./calls.getters";
|
|
2
|
+
import {
|
|
3
|
+
HOURS_PER_DAY,
|
|
4
|
+
STATS_MAX_TIME_MS,
|
|
5
|
+
TREND_MONTH_BUCKET_MIN_MONTHS,
|
|
6
|
+
TREND_WEEK_BUCKET_MIN_MONTHS,
|
|
7
|
+
} from "./calls.constants";
|
|
8
|
+
import {
|
|
9
|
+
addDaysToDateStr,
|
|
10
|
+
DAY_MS,
|
|
11
|
+
dateAtNoonUtc,
|
|
12
|
+
formatYmdInTz,
|
|
13
|
+
getWeekdayInTz,
|
|
14
|
+
getYmdInTz,
|
|
15
|
+
incrementYmd,
|
|
16
|
+
parseYmd,
|
|
17
|
+
toUtcDate,
|
|
18
|
+
ymdToStr,
|
|
19
|
+
} from "../../utils/date.utils";
|
|
20
|
+
import {
|
|
21
|
+
findTicketSubjectRows,
|
|
22
|
+
ticketStatsDateScopeFromYmd,
|
|
23
|
+
} from "../../municipal/tickets/tickets.statistics.getters";
|
|
24
|
+
import type { CityName } from "../../municipal/utils/types";
|
|
25
|
+
import type {
|
|
26
|
+
SubjectItem,
|
|
27
|
+
TicketStatsDateScope,
|
|
28
|
+
} from "../../municipal/tickets/tickets.types";
|
|
29
|
+
import type {
|
|
30
|
+
CallRoutingFilter,
|
|
31
|
+
CallsHourlyBucket,
|
|
32
|
+
CallsRoutingAggregation,
|
|
33
|
+
CallsStatsFilter,
|
|
34
|
+
CallsSummaryAggregation,
|
|
35
|
+
CallsTrendBucket,
|
|
36
|
+
ResolvedTimeBucket,
|
|
37
|
+
} from "./calls.statistics.types";
|
|
38
|
+
|
|
39
|
+
const emptySummary = (): CallsSummaryAggregation => ({
|
|
40
|
+
totalCalls: 0,
|
|
41
|
+
avgDurationSeconds: 0,
|
|
42
|
+
openTickets: 0,
|
|
43
|
+
draftTickets: 0,
|
|
44
|
+
});
|
|
45
|
+
|
|
46
|
+
const emptyRouting = (): CallsRoutingAggregation => ({
|
|
47
|
+
aiHandled: 0,
|
|
48
|
+
transferred: 0,
|
|
49
|
+
hijacked: 0,
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
const isTransferred = { $eq: ["$redirectedCall", true] };
|
|
53
|
+
|
|
54
|
+
const isHijacked = {
|
|
55
|
+
$and: [
|
|
56
|
+
{ $eq: ["$isConferenceCall", true] },
|
|
57
|
+
{ $ne: ["$redirectedCall", true] },
|
|
58
|
+
],
|
|
59
|
+
};
|
|
60
|
+
|
|
61
|
+
const countIf = (condition: Record<string, unknown>) => ({
|
|
62
|
+
$sum: { $cond: [condition, 1, 0] },
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
const needsTicketStatusFilter = (filters?: CallsStatsFilter["ticketStatusFilters"]) => {
|
|
66
|
+
if (!filters?.length) return false;
|
|
67
|
+
const hasOpened = filters.includes("opened");
|
|
68
|
+
const hasNotOpened = filters.includes("not_opened");
|
|
69
|
+
return hasOpened !== hasNotOpened;
|
|
70
|
+
};
|
|
71
|
+
|
|
72
|
+
const routingMatch = (routing?: CallRoutingFilter[]) => {
|
|
73
|
+
if (!routing?.length) return null;
|
|
74
|
+
|
|
75
|
+
const clauses: Record<string, unknown>[] = [];
|
|
76
|
+
if (routing.includes("transferred")) clauses.push({ redirectedCall: true });
|
|
77
|
+
if (routing.includes("hijacked")) {
|
|
78
|
+
clauses.push({ isConferenceCall: true, redirectedCall: { $ne: true } });
|
|
79
|
+
}
|
|
80
|
+
if (routing.includes("ai_only")) {
|
|
81
|
+
clauses.push({
|
|
82
|
+
redirectedCall: { $ne: true },
|
|
83
|
+
$or: [
|
|
84
|
+
{ isConferenceCall: { $ne: true } },
|
|
85
|
+
{ isConferenceCall: { $exists: false } },
|
|
86
|
+
],
|
|
87
|
+
});
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
return clauses.length ? { $or: clauses } : null;
|
|
91
|
+
};
|
|
92
|
+
|
|
93
|
+
const ticketStatusMatch = (filter: CallsStatsFilter) => {
|
|
94
|
+
if (!needsTicketStatusFilter(filter.ticketStatusFilters)) return null;
|
|
95
|
+
|
|
96
|
+
const sids = filter.callSidsWithTickets ?? [];
|
|
97
|
+
const openedOnly = filter.ticketStatusFilters?.includes("opened");
|
|
98
|
+
return { callSid: openedOnly ? { $in: sids } : { $nin: sids } };
|
|
99
|
+
};
|
|
100
|
+
|
|
101
|
+
// "HH:mm" → minute-of-day (0-1439), for the per-call time-of-day filter.
|
|
102
|
+
const parseMinuteOfDay = (time?: string): number | undefined =>
|
|
103
|
+
time == null ? undefined : Number(time.slice(0, 2)) * 60 + Number(time.slice(3, 5));
|
|
104
|
+
|
|
105
|
+
const hourMatch = (filter: CallsStatsFilter): Record<string, unknown> | null => {
|
|
106
|
+
const from = parseMinuteOfDay(filter.hourFrom);
|
|
107
|
+
const to = parseMinuteOfDay(filter.hourTo);
|
|
108
|
+
if (from == null && to == null) return null;
|
|
109
|
+
|
|
110
|
+
// hourTo is exclusive: a call counts when minuteOfDay is in [from, to).
|
|
111
|
+
// from > to → the range wraps past midnight: minuteOfDay >= from OR < to
|
|
112
|
+
if (from != null && to != null && from > to) {
|
|
113
|
+
return { $or: [{ minuteOfDay: { $gte: from } }, { minuteOfDay: { $lt: to } }] };
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
const minuteOfDay: Record<string, unknown> = {};
|
|
117
|
+
if (from != null) minuteOfDay.$gte = from;
|
|
118
|
+
if (to != null) minuteOfDay.$lt = to;
|
|
119
|
+
return { minuteOfDay };
|
|
120
|
+
};
|
|
121
|
+
|
|
122
|
+
const createdAtMatchForFilter = (
|
|
123
|
+
filter: CallsStatsFilter,
|
|
124
|
+
): { createdAt: { $gte: Date; $lte: Date } } => {
|
|
125
|
+
const bounds = ticketStatsDateScopeFromYmd(
|
|
126
|
+
filter.startStr,
|
|
127
|
+
filter.endStr,
|
|
128
|
+
filter.timezone,
|
|
129
|
+
);
|
|
130
|
+
return { createdAt: { $gte: bounds.from!, $lte: bounds.to! } };
|
|
131
|
+
};
|
|
132
|
+
|
|
133
|
+
const buildCallsMatchStages = (filter: CallsStatsFilter): Record<string, unknown>[] => {
|
|
134
|
+
const hourRange = hourMatch(filter);
|
|
135
|
+
const routing = routingMatch(filter.callRouting);
|
|
136
|
+
const ticketMatch = ticketStatusMatch(filter);
|
|
137
|
+
|
|
138
|
+
return [
|
|
139
|
+
{
|
|
140
|
+
$match: {
|
|
141
|
+
clientId: filter.clientId,
|
|
142
|
+
isIncomingCall: true,
|
|
143
|
+
...createdAtMatchForFilter(filter),
|
|
144
|
+
},
|
|
145
|
+
},
|
|
146
|
+
{
|
|
147
|
+
$addFields: {
|
|
148
|
+
dateLocal: {
|
|
149
|
+
$dateToString: {
|
|
150
|
+
format: "%Y-%m-%d",
|
|
151
|
+
date: "$createdAt",
|
|
152
|
+
timezone: filter.timezone,
|
|
153
|
+
},
|
|
154
|
+
},
|
|
155
|
+
hour: { $hour: { date: "$createdAt", timezone: filter.timezone } },
|
|
156
|
+
minuteOfDay: {
|
|
157
|
+
$add: [
|
|
158
|
+
{ $multiply: [{ $hour: { date: "$createdAt", timezone: filter.timezone } }, 60] },
|
|
159
|
+
{ $minute: { date: "$createdAt", timezone: filter.timezone } },
|
|
160
|
+
],
|
|
161
|
+
},
|
|
162
|
+
},
|
|
163
|
+
},
|
|
164
|
+
{ $match: { dateLocal: { $gte: filter.startStr, $lte: filter.endStr } } },
|
|
165
|
+
...(hourRange ? [{ $match: hourRange }] : []),
|
|
166
|
+
...(routing ? [{ $match: routing }] : []),
|
|
167
|
+
...(ticketMatch ? [{ $match: ticketMatch }] : []),
|
|
168
|
+
];
|
|
169
|
+
};
|
|
170
|
+
|
|
171
|
+
const aggregateCalls = (
|
|
172
|
+
filter: CallsStatsFilter,
|
|
173
|
+
...tail: Record<string, unknown>[]
|
|
174
|
+
) =>
|
|
175
|
+
getCallsCollection().aggregate([...buildCallsMatchStages(filter), ...tail], {
|
|
176
|
+
maxTimeMS: STATS_MAX_TIME_MS,
|
|
177
|
+
});
|
|
178
|
+
|
|
179
|
+
const trendBucketExpr = (bucket: ResolvedTimeBucket, timezone: string) =>
|
|
180
|
+
bucket === "day"
|
|
181
|
+
? { bucket: "$dateLocal" }
|
|
182
|
+
: {
|
|
183
|
+
bucket: {
|
|
184
|
+
$dateToString: {
|
|
185
|
+
format: "%Y-%m-%d",
|
|
186
|
+
date: { $dateTrunc: { date: "$createdAt", unit: bucket, timezone } },
|
|
187
|
+
timezone,
|
|
188
|
+
},
|
|
189
|
+
},
|
|
190
|
+
};
|
|
191
|
+
|
|
192
|
+
const mapTrendRow = (row: {
|
|
193
|
+
_id: unknown;
|
|
194
|
+
totalCalls: number;
|
|
195
|
+
openTickets: number;
|
|
196
|
+
}): CallsTrendBucket => ({
|
|
197
|
+
bucket: String(row._id),
|
|
198
|
+
totalCalls: row.totalCalls,
|
|
199
|
+
openTickets: row.openTickets,
|
|
200
|
+
});
|
|
201
|
+
|
|
202
|
+
export const rangeDurationDays = (startStr: string, endStr: string): number => {
|
|
203
|
+
const start = toUtcDate(startStr).getTime();
|
|
204
|
+
const end = toUtcDate(endStr).getTime();
|
|
205
|
+
return Math.floor((end - start) / DAY_MS) + 1;
|
|
206
|
+
};
|
|
207
|
+
|
|
208
|
+
/**
|
|
209
|
+
* Number of distinct calendar months the [startStr, endStr] range spans, inclusive.
|
|
210
|
+
* Counts month boundaries rather than 30-day periods: e.g. Jan 31 -> Feb 1 returns 2.
|
|
211
|
+
*/
|
|
212
|
+
export const rangeDurationMonths = (startStr: string, endStr: string): number => {
|
|
213
|
+
const [sy, sm] = parseYmd(startStr);
|
|
214
|
+
const [ey, em] = parseYmd(endStr);
|
|
215
|
+
return (ey - sy) * 12 + (em - sm) + 1;
|
|
216
|
+
};
|
|
217
|
+
|
|
218
|
+
/**
|
|
219
|
+
* Chooses how the client calls-over-time bar chart groups each column:
|
|
220
|
+
* one bar per day, week, or calendar month. Short ranges stay daily; longer
|
|
221
|
+
* ones switch to weekly or monthly so the chart does not show hundreds of bars.
|
|
222
|
+
*/
|
|
223
|
+
export const resolveTimeBucketFromRange = (
|
|
224
|
+
startStr: string,
|
|
225
|
+
endStr: string,
|
|
226
|
+
): ResolvedTimeBucket => {
|
|
227
|
+
const months = rangeDurationMonths(startStr, endStr);
|
|
228
|
+
if (months > TREND_MONTH_BUCKET_MIN_MONTHS) return "month";
|
|
229
|
+
if (months > TREND_WEEK_BUCKET_MIN_MONTHS) return "week";
|
|
230
|
+
return "day";
|
|
231
|
+
};
|
|
232
|
+
|
|
233
|
+
const weekBucketKeyForDate = (dateStr: string, timezone: string): string => {
|
|
234
|
+
const anchor = dateAtNoonUtc(dateStr);
|
|
235
|
+
const weekStart = new Date(anchor.getTime() - getWeekdayInTz(anchor, timezone) * DAY_MS);
|
|
236
|
+
return formatYmdInTz(weekStart, timezone);
|
|
237
|
+
};
|
|
238
|
+
|
|
239
|
+
const formatWeekBucketRange = (
|
|
240
|
+
weekStartStr: string,
|
|
241
|
+
clipStartStr: string,
|
|
242
|
+
clipEndStr: string,
|
|
243
|
+
): string => {
|
|
244
|
+
const weekEndStr = addDaysToDateStr(weekStartStr, 6);
|
|
245
|
+
const start = weekStartStr < clipStartStr ? clipStartStr : weekStartStr;
|
|
246
|
+
const end = weekEndStr > clipEndStr ? clipEndStr : weekEndStr;
|
|
247
|
+
return `${start}/${end}`;
|
|
248
|
+
};
|
|
249
|
+
|
|
250
|
+
const monthBucketKey = (y: number, m: number): string =>
|
|
251
|
+
`${y}-${String(m).padStart(2, "0")}-01`;
|
|
252
|
+
|
|
253
|
+
const emptyTrendBucket = (bucket: string): CallsTrendBucket => ({
|
|
254
|
+
bucket,
|
|
255
|
+
totalCalls: 0,
|
|
256
|
+
openTickets: 0,
|
|
257
|
+
});
|
|
258
|
+
|
|
259
|
+
const indexTrendRowsByBucket = (rows: CallsTrendBucket[]) =>
|
|
260
|
+
new Map(rows.map((row) => [row.bucket, row]));
|
|
261
|
+
|
|
262
|
+
const fillDailyTrendBuckets = (
|
|
263
|
+
rows: CallsTrendBucket[],
|
|
264
|
+
startStr: string,
|
|
265
|
+
endStr: string,
|
|
266
|
+
): CallsTrendBucket[] => {
|
|
267
|
+
const byBucket = indexTrendRowsByBucket(rows);
|
|
268
|
+
const filled: CallsTrendBucket[] = [];
|
|
269
|
+
|
|
270
|
+
for (let cursor = startStr; cursor <= endStr; cursor = ymdToStr(...incrementYmd(...parseYmd(cursor)))) {
|
|
271
|
+
filled.push(byBucket.get(cursor) ?? emptyTrendBucket(cursor));
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
return filled;
|
|
275
|
+
};
|
|
276
|
+
|
|
277
|
+
const fillMonthlyTrendBuckets = (
|
|
278
|
+
rows: CallsTrendBucket[],
|
|
279
|
+
startStr: string,
|
|
280
|
+
endStr: string,
|
|
281
|
+
timezone: string,
|
|
282
|
+
): CallsTrendBucket[] => {
|
|
283
|
+
const byBucket = indexTrendRowsByBucket(rows);
|
|
284
|
+
const startParts = getYmdInTz(dateAtNoonUtc(startStr), timezone);
|
|
285
|
+
const endParts = getYmdInTz(dateAtNoonUtc(endStr), timezone);
|
|
286
|
+
|
|
287
|
+
const filled: CallsTrendBucket[] = [];
|
|
288
|
+
let { y, m } = startParts;
|
|
289
|
+
|
|
290
|
+
while (y < endParts.y || (y === endParts.y && m <= endParts.m)) {
|
|
291
|
+
const bucket = monthBucketKey(y, m);
|
|
292
|
+
filled.push(byBucket.get(bucket) ?? emptyTrendBucket(bucket));
|
|
293
|
+
m += 1;
|
|
294
|
+
if (m > 12) {
|
|
295
|
+
m = 1;
|
|
296
|
+
y += 1;
|
|
297
|
+
}
|
|
298
|
+
}
|
|
299
|
+
|
|
300
|
+
return filled;
|
|
301
|
+
};
|
|
302
|
+
|
|
303
|
+
const fillWeeklyTrendBuckets = (
|
|
304
|
+
rows: CallsTrendBucket[],
|
|
305
|
+
startStr: string,
|
|
306
|
+
endStr: string,
|
|
307
|
+
timezone: string,
|
|
308
|
+
): CallsTrendBucket[] => {
|
|
309
|
+
const byBucket = indexTrendRowsByBucket(rows);
|
|
310
|
+
const filled: CallsTrendBucket[] = [];
|
|
311
|
+
const seen = new Set<string>();
|
|
312
|
+
|
|
313
|
+
for (let i = 0, days = rangeDurationDays(startStr, endStr); i < days; i += 1) {
|
|
314
|
+
const bucket = formatWeekBucketRange(
|
|
315
|
+
weekBucketKeyForDate(addDaysToDateStr(startStr, i), timezone),
|
|
316
|
+
startStr,
|
|
317
|
+
endStr,
|
|
318
|
+
);
|
|
319
|
+
if (seen.has(bucket)) continue;
|
|
320
|
+
seen.add(bucket);
|
|
321
|
+
filled.push(byBucket.get(bucket) ?? emptyTrendBucket(bucket));
|
|
322
|
+
}
|
|
323
|
+
|
|
324
|
+
return filled;
|
|
325
|
+
};
|
|
326
|
+
|
|
327
|
+
const fillTrendBuckets = (
|
|
328
|
+
rows: CallsTrendBucket[],
|
|
329
|
+
bucket: ResolvedTimeBucket,
|
|
330
|
+
startStr: string,
|
|
331
|
+
endStr: string,
|
|
332
|
+
timezone: string,
|
|
333
|
+
): CallsTrendBucket[] => {
|
|
334
|
+
switch (bucket) {
|
|
335
|
+
case "day":
|
|
336
|
+
return fillDailyTrendBuckets(rows, startStr, endStr);
|
|
337
|
+
case "week":
|
|
338
|
+
return fillWeeklyTrendBuckets(rows, startStr, endStr, timezone);
|
|
339
|
+
case "month":
|
|
340
|
+
return fillMonthlyTrendBuckets(rows, startStr, endStr, timezone);
|
|
341
|
+
default: {
|
|
342
|
+
return fillDailyTrendBuckets(rows, startStr, endStr);
|
|
343
|
+
}
|
|
344
|
+
}
|
|
345
|
+
};
|
|
346
|
+
|
|
347
|
+
const resolveHourRange = (filter: CallsStatsFilter): { hourFrom: number; hourTo: number } => {
|
|
348
|
+
const from = parseMinuteOfDay(filter.hourFrom);
|
|
349
|
+
const to = parseMinuteOfDay(filter.hourTo);
|
|
350
|
+
if (from == null && to == null) return { hourFrom: 0, hourTo: HOURS_PER_DAY - 1 };
|
|
351
|
+
return {
|
|
352
|
+
hourFrom: from == null ? 0 : Math.floor(from / 60),
|
|
353
|
+
// hourTo is exclusive: last populated bucket is the hour holding minute (to - 1).
|
|
354
|
+
hourTo: to == null ? HOURS_PER_DAY - 1 : Math.floor((to - 1) / 60),
|
|
355
|
+
};
|
|
356
|
+
};
|
|
357
|
+
|
|
358
|
+
const roundOneDecimal = (value: number): number => Math.round(value * 10) / 10;
|
|
359
|
+
|
|
360
|
+
const toHourlyDistribution = (
|
|
361
|
+
rows: { _id: unknown; totalCalls: number }[],
|
|
362
|
+
hourRange: { hourFrom: number; hourTo: number },
|
|
363
|
+
daysInRange: number,
|
|
364
|
+
): CallsHourlyBucket[] => {
|
|
365
|
+
const counts = new Map(rows.map((r) => [Number(r._id), r.totalCalls]));
|
|
366
|
+
const { hourFrom, hourTo } = hourRange;
|
|
367
|
+
const buckets: CallsHourlyBucket[] = [];
|
|
368
|
+
|
|
369
|
+
const pushHour = (hour: number) => {
|
|
370
|
+
const totalCalls = counts.get(hour) ?? 0;
|
|
371
|
+
buckets.push({
|
|
372
|
+
hour,
|
|
373
|
+
totalCalls,
|
|
374
|
+
avgCalls: roundOneDecimal(daysInRange > 0 ? totalCalls / daysInRange : 0),
|
|
375
|
+
});
|
|
376
|
+
};
|
|
377
|
+
|
|
378
|
+
if (hourFrom <= hourTo) {
|
|
379
|
+
for (let hour = hourFrom; hour <= hourTo; hour += 1) pushHour(hour);
|
|
380
|
+
} else {
|
|
381
|
+
// Wraps past midnight: hourFrom..23, then 0..hourTo.
|
|
382
|
+
for (let hour = hourFrom; hour < HOURS_PER_DAY; hour += 1) pushHour(hour);
|
|
383
|
+
for (let hour = 0; hour <= hourTo; hour += 1) pushHour(hour);
|
|
384
|
+
}
|
|
385
|
+
|
|
386
|
+
return buckets;
|
|
387
|
+
};
|
|
388
|
+
|
|
389
|
+
export const aggregateCallsSummary = async (
|
|
390
|
+
filter: CallsStatsFilter,
|
|
391
|
+
): Promise<CallsSummaryAggregation> => {
|
|
392
|
+
const hasTickets = Boolean(filter.callSidsWithTickets?.length || filter.callSidsWithDraftTickets?.length);
|
|
393
|
+
const hasStatusFilter = needsTicketStatusFilter(filter.ticketStatusFilters);
|
|
394
|
+
|
|
395
|
+
if (!hasTickets && !hasStatusFilter) {
|
|
396
|
+
const result = await aggregateCalls(filter, {
|
|
397
|
+
$group: { _id: null, totalCalls: { $sum: 1 }, avgDurationSeconds: { $avg: "$callLength" } },
|
|
398
|
+
}).next();
|
|
399
|
+
if (!result) return emptySummary();
|
|
400
|
+
return {
|
|
401
|
+
totalCalls: result.totalCalls,
|
|
402
|
+
avgDurationSeconds: roundOneDecimal(result.avgDurationSeconds ?? 0),
|
|
403
|
+
openTickets: 0,
|
|
404
|
+
draftTickets: 0,
|
|
405
|
+
};
|
|
406
|
+
}
|
|
407
|
+
|
|
408
|
+
// Single pass: project per-call data, handle ticket status filter and counting in JS.
|
|
409
|
+
// Avoids embedding callSidsWithTickets inside any MongoDB pipeline stage.
|
|
410
|
+
const rows = await aggregateCalls(
|
|
411
|
+
{ ...filter, ticketStatusFilters: undefined },
|
|
412
|
+
{ $project: { _id: 0, callSid: 1, callLength: 1 } },
|
|
413
|
+
).toArray();
|
|
414
|
+
|
|
415
|
+
const ticketSet = new Set(filter.callSidsWithTickets ?? []);
|
|
416
|
+
const draftSet = new Set(filter.callSidsWithDraftTickets ?? []);
|
|
417
|
+
const openedOnly = hasStatusFilter && filter.ticketStatusFilters!.includes("opened");
|
|
418
|
+
const notOpenedOnly = hasStatusFilter && filter.ticketStatusFilters!.includes("not_opened");
|
|
419
|
+
|
|
420
|
+
let totalCalls = 0;
|
|
421
|
+
let totalDuration = 0;
|
|
422
|
+
let openTickets = 0;
|
|
423
|
+
let draftTickets = 0;
|
|
424
|
+
|
|
425
|
+
for (const row of rows) {
|
|
426
|
+
const sid = row.callSid as string;
|
|
427
|
+
const hasTicket = ticketSet.has(sid);
|
|
428
|
+
if (openedOnly && !hasTicket) continue;
|
|
429
|
+
if (notOpenedOnly && hasTicket) continue;
|
|
430
|
+
totalCalls += 1;
|
|
431
|
+
totalDuration += (row.callLength as number | undefined) ?? 0;
|
|
432
|
+
if (hasTicket) openTickets += 1;
|
|
433
|
+
if (draftSet.has(sid)) draftTickets += 1;
|
|
434
|
+
}
|
|
435
|
+
|
|
436
|
+
if (!totalCalls) return emptySummary();
|
|
437
|
+
|
|
438
|
+
return {
|
|
439
|
+
totalCalls,
|
|
440
|
+
avgDurationSeconds: roundOneDecimal(totalDuration / totalCalls),
|
|
441
|
+
openTickets,
|
|
442
|
+
draftTickets,
|
|
443
|
+
};
|
|
444
|
+
};
|
|
445
|
+
|
|
446
|
+
export const aggregateCallsTrend = async (
|
|
447
|
+
filter: CallsStatsFilter,
|
|
448
|
+
bucket: ResolvedTimeBucket,
|
|
449
|
+
): Promise<CallsTrendBucket[]> => {
|
|
450
|
+
const ticketSet = filter.callSidsWithTickets?.length
|
|
451
|
+
? new Set(filter.callSidsWithTickets)
|
|
452
|
+
: null;
|
|
453
|
+
|
|
454
|
+
if (ticketSet) {
|
|
455
|
+
const rows = await aggregateCalls(
|
|
456
|
+
{ ...filter, ticketStatusFilters: undefined },
|
|
457
|
+
{ $addFields: trendBucketExpr(bucket, filter.timezone) },
|
|
458
|
+
{ $project: { _id: 0, bucket: 1, callSid: 1 } },
|
|
459
|
+
).toArray();
|
|
460
|
+
|
|
461
|
+
const openedOnly = needsTicketStatusFilter(filter.ticketStatusFilters) && filter.ticketStatusFilters!.includes("opened");
|
|
462
|
+
const notOpenedOnly = needsTicketStatusFilter(filter.ticketStatusFilters) && filter.ticketStatusFilters!.includes("not_opened");
|
|
463
|
+
|
|
464
|
+
const byBucket = new Map<string, { totalCalls: number; openTickets: number }>();
|
|
465
|
+
for (const row of rows) {
|
|
466
|
+
const sid = row.callSid as string;
|
|
467
|
+
const hasTicket = ticketSet.has(sid);
|
|
468
|
+
if (openedOnly && !hasTicket) continue;
|
|
469
|
+
if (notOpenedOnly && hasTicket) continue;
|
|
470
|
+
const label =
|
|
471
|
+
bucket === "week"
|
|
472
|
+
? formatWeekBucketRange(String(row.bucket), filter.startStr, filter.endStr)
|
|
473
|
+
: String(row.bucket);
|
|
474
|
+
const cur = byBucket.get(label) ?? { totalCalls: 0, openTickets: 0 };
|
|
475
|
+
cur.totalCalls += 1;
|
|
476
|
+
if (hasTicket) cur.openTickets += 1;
|
|
477
|
+
byBucket.set(label, cur);
|
|
478
|
+
}
|
|
479
|
+
|
|
480
|
+
const mappedRows = [...byBucket.entries()]
|
|
481
|
+
.map(([label, counts]) => ({ bucket: label, ...counts }));
|
|
482
|
+
|
|
483
|
+
return fillTrendBuckets(mappedRows, bucket, filter.startStr, filter.endStr, filter.timezone);
|
|
484
|
+
}
|
|
485
|
+
|
|
486
|
+
const rows = await aggregateCalls(
|
|
487
|
+
filter,
|
|
488
|
+
{ $addFields: trendBucketExpr(bucket, filter.timezone) },
|
|
489
|
+
{ $group: { _id: "$bucket", totalCalls: { $sum: 1 } } },
|
|
490
|
+
{ $sort: { _id: 1 } },
|
|
491
|
+
).toArray();
|
|
492
|
+
|
|
493
|
+
const mappedRows =
|
|
494
|
+
bucket === "week"
|
|
495
|
+
? rows.map((row) => ({
|
|
496
|
+
bucket: formatWeekBucketRange(String(row._id), filter.startStr, filter.endStr),
|
|
497
|
+
totalCalls: row.totalCalls as number,
|
|
498
|
+
openTickets: 0,
|
|
499
|
+
}))
|
|
500
|
+
: rows.map((row) =>
|
|
501
|
+
mapTrendRow({ _id: row._id, totalCalls: row.totalCalls as number, openTickets: 0 }),
|
|
502
|
+
);
|
|
503
|
+
|
|
504
|
+
return fillTrendBuckets(mappedRows, bucket, filter.startStr, filter.endStr, filter.timezone);
|
|
505
|
+
};
|
|
506
|
+
|
|
507
|
+
export const aggregateCallsHourlyByRange = async (
|
|
508
|
+
filter: CallsStatsFilter,
|
|
509
|
+
): Promise<CallsHourlyBucket[]> => {
|
|
510
|
+
const rows = await aggregateCalls(
|
|
511
|
+
filter,
|
|
512
|
+
{ $group: { _id: "$hour", totalCalls: { $sum: 1 } } },
|
|
513
|
+
{ $sort: { _id: 1 } },
|
|
514
|
+
).toArray();
|
|
515
|
+
|
|
516
|
+
return toHourlyDistribution(
|
|
517
|
+
rows.map((r) => ({ _id: r._id, totalCalls: r.totalCalls as number })),
|
|
518
|
+
resolveHourRange(filter),
|
|
519
|
+
rangeDurationDays(filter.startStr, filter.endStr),
|
|
520
|
+
);
|
|
521
|
+
};
|
|
522
|
+
|
|
523
|
+
export const aggregateCallsRouting = async (
|
|
524
|
+
filter: CallsStatsFilter,
|
|
525
|
+
): Promise<CallsRoutingAggregation> => {
|
|
526
|
+
const result = await aggregateCalls(
|
|
527
|
+
filter,
|
|
528
|
+
{
|
|
529
|
+
$group: {
|
|
530
|
+
_id: null,
|
|
531
|
+
transferred: countIf(isTransferred),
|
|
532
|
+
hijacked: countIf(isHijacked),
|
|
533
|
+
totalCalls: { $sum: 1 },
|
|
534
|
+
},
|
|
535
|
+
},
|
|
536
|
+
{
|
|
537
|
+
$addFields: {
|
|
538
|
+
aiHandled: {
|
|
539
|
+
$subtract: ["$totalCalls", { $add: ["$transferred", "$hijacked"] }],
|
|
540
|
+
},
|
|
541
|
+
},
|
|
542
|
+
},
|
|
543
|
+
).next();
|
|
544
|
+
|
|
545
|
+
if (!result) return emptyRouting();
|
|
546
|
+
|
|
547
|
+
return {
|
|
548
|
+
aiHandled: result.aiHandled,
|
|
549
|
+
transferred: result.transferred,
|
|
550
|
+
hijacked: result.hijacked,
|
|
551
|
+
};
|
|
552
|
+
};
|
|
553
|
+
|
|
554
|
+
export const findFilteredCallSids = async (filter: CallsStatsFilter): Promise<string[]> => {
|
|
555
|
+
const rows = await aggregateCalls(
|
|
556
|
+
filter,
|
|
557
|
+
{ $group: { _id: "$callSid" } },
|
|
558
|
+
).toArray();
|
|
559
|
+
return rows.map((r) => String(r._id)).filter(Boolean);
|
|
560
|
+
};
|
|
561
|
+
|
|
562
|
+
/**
|
|
563
|
+
* "Subjects" panel: ticket count per subject, for calls passing the filters.
|
|
564
|
+
* Calls and tickets are in separate DBs, so we join in code — filtered call IDs
|
|
565
|
+
* from calls, subjects from tickets. Returns the full list sorted most-common
|
|
566
|
+
* first; callers slice if they want a top-N.
|
|
567
|
+
*/
|
|
568
|
+
export const findSubjectsByCityAndDateRange = async (
|
|
569
|
+
cityName: CityName,
|
|
570
|
+
filter: CallsStatsFilter,
|
|
571
|
+
dateScope: TicketStatsDateScope,
|
|
572
|
+
): Promise<SubjectItem[]> => {
|
|
573
|
+
const filteredCallSids = new Set(await findFilteredCallSids(filter));
|
|
574
|
+
if (!filteredCallSids.size) return [];
|
|
575
|
+
|
|
576
|
+
const rows = await findTicketSubjectRows(cityName, dateScope);
|
|
577
|
+
|
|
578
|
+
const counts = new Map<string, number>();
|
|
579
|
+
for (const row of rows) {
|
|
580
|
+
if (!filteredCallSids.has(row.callSid)) continue;
|
|
581
|
+
counts.set(row.subject, (counts.get(row.subject) ?? 0) + 1);
|
|
582
|
+
}
|
|
583
|
+
|
|
584
|
+
return [...counts.entries()]
|
|
585
|
+
.map(([subject, count]) => ({ subject, count }))
|
|
586
|
+
.sort((a, b) => b.count - a.count || a.subject.localeCompare(b.subject));
|
|
587
|
+
};
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
export type ResolvedTimeBucket = "day" | "week" | "month";
|
|
2
|
+
|
|
3
|
+
export type TicketStatusFilter = "opened" | "not_opened";
|
|
4
|
+
export type CallRoutingFilter = "ai_only" | "transferred" | "hijacked";
|
|
5
|
+
|
|
6
|
+
export type CallsStatsFilter = {
|
|
7
|
+
clientId: string;
|
|
8
|
+
startStr: string;
|
|
9
|
+
endStr: string;
|
|
10
|
+
timezone: string;
|
|
11
|
+
hourFrom?: string; // HH:mm
|
|
12
|
+
hourTo?: string; // HH:mm
|
|
13
|
+
callRouting?: CallRoutingFilter[];
|
|
14
|
+
callSidsWithTickets?: string[];
|
|
15
|
+
/** Call SIDs whose municipal ticket matches the draft-ticket definition (see tickets statistics getters). */
|
|
16
|
+
callSidsWithDraftTickets?: string[];
|
|
17
|
+
ticketStatusFilters?: TicketStatusFilter[];
|
|
18
|
+
};
|
|
19
|
+
|
|
20
|
+
export type CallsSummaryAggregation = {
|
|
21
|
+
totalCalls: number;
|
|
22
|
+
avgDurationSeconds: number;
|
|
23
|
+
openTickets: number;
|
|
24
|
+
draftTickets: number;
|
|
25
|
+
};
|
|
26
|
+
|
|
27
|
+
export type CallsTrendBucket = {
|
|
28
|
+
/** Day: YYYY-MM-DD; week: YYYY-MM-DD/YYYY-MM-DD (inclusive, clipped to filter); month: YYYY-MM-01 */
|
|
29
|
+
bucket: string;
|
|
30
|
+
totalCalls: number;
|
|
31
|
+
openTickets: number;
|
|
32
|
+
};
|
|
33
|
+
|
|
34
|
+
export type CallsHourlyBucket = {
|
|
35
|
+
hour: number;
|
|
36
|
+
totalCalls: number;
|
|
37
|
+
avgCalls: number;
|
|
38
|
+
};
|
|
39
|
+
|
|
40
|
+
export type CallsRoutingAggregation = {
|
|
41
|
+
aiHandled: number;
|
|
42
|
+
transferred: number;
|
|
43
|
+
hijacked: number;
|
|
44
|
+
};
|
|
@@ -2,8 +2,10 @@ import { ObjectId, Sort, WithId } from "mongodb";
|
|
|
2
2
|
import { TranscriptionSegment } from "../results";
|
|
3
3
|
import { LeadProperty } from "../leads";
|
|
4
4
|
|
|
5
|
-
|
|
6
|
-
|
|
5
|
+
import {
|
|
6
|
+
CONFERENCE_ROLE_CUSTOMER,
|
|
7
|
+
CONFERENCE_ROLE_SUPERVISOR,
|
|
8
|
+
} from "./calls.constants";
|
|
7
9
|
|
|
8
10
|
export type ConferenceRole =
|
|
9
11
|
| typeof CONFERENCE_ROLE_CUSTOMER
|