@talkpilot/core-db 1.2.1 → 1.3.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.
Files changed (147) hide show
  1. package/README.md +117 -108
  2. package/README_OLD.md +160 -0
  3. package/dist/municipal/tickets/index.d.ts +2 -1
  4. package/dist/municipal/tickets/index.d.ts.map +1 -1
  5. package/dist/municipal/tickets/index.js +1 -0
  6. package/dist/municipal/tickets/index.js.map +1 -1
  7. package/dist/municipal/tickets/tickets.constants.d.ts +7 -0
  8. package/dist/municipal/tickets/tickets.constants.d.ts.map +1 -0
  9. package/dist/municipal/tickets/tickets.constants.js +10 -0
  10. package/dist/municipal/tickets/tickets.constants.js.map +1 -0
  11. package/dist/municipal/tickets/tickets.deprecated.getters.d.ts +12 -0
  12. package/dist/municipal/tickets/tickets.deprecated.getters.d.ts.map +1 -0
  13. package/dist/municipal/tickets/tickets.deprecated.getters.js +131 -0
  14. package/dist/municipal/tickets/tickets.deprecated.getters.js.map +1 -0
  15. package/dist/municipal/tickets/tickets.getters.d.ts +0 -11
  16. package/dist/municipal/tickets/tickets.getters.d.ts.map +1 -1
  17. package/dist/municipal/tickets/tickets.getters.js +0 -128
  18. package/dist/municipal/tickets/tickets.getters.js.map +1 -1
  19. package/dist/municipal/tickets/tickets.statistics.aggregation.d.ts +45 -0
  20. package/dist/municipal/tickets/tickets.statistics.aggregation.d.ts.map +1 -0
  21. package/dist/municipal/tickets/tickets.statistics.aggregation.js +98 -0
  22. package/dist/municipal/tickets/tickets.statistics.aggregation.js.map +1 -0
  23. package/dist/municipal/tickets/tickets.statistics.dates.d.ts +7 -0
  24. package/dist/municipal/tickets/tickets.statistics.dates.d.ts.map +1 -0
  25. package/dist/municipal/tickets/tickets.statistics.dates.js +40 -0
  26. package/dist/municipal/tickets/tickets.statistics.dates.js.map +1 -0
  27. package/dist/municipal/tickets/tickets.statistics.getters.d.ts +9 -0
  28. package/dist/municipal/tickets/tickets.statistics.getters.d.ts.map +1 -0
  29. package/dist/municipal/tickets/tickets.statistics.getters.js +55 -0
  30. package/dist/municipal/tickets/tickets.statistics.getters.js.map +1 -0
  31. package/dist/municipal/tickets/tickets.statistics.pipeline.d.ts +53 -0
  32. package/dist/municipal/tickets/tickets.statistics.pipeline.d.ts.map +1 -0
  33. package/dist/municipal/tickets/tickets.statistics.pipeline.js +112 -0
  34. package/dist/municipal/tickets/tickets.statistics.pipeline.js.map +1 -0
  35. package/dist/municipal/tickets/tickets.statistics.utils.d.ts +7 -0
  36. package/dist/municipal/tickets/tickets.statistics.utils.d.ts.map +1 -0
  37. package/dist/municipal/tickets/tickets.statistics.utils.js +40 -0
  38. package/dist/municipal/tickets/tickets.statistics.utils.js.map +1 -0
  39. package/dist/municipal/tickets/tickets.types.d.ts +10 -5
  40. package/dist/municipal/tickets/tickets.types.d.ts.map +1 -1
  41. package/dist/talkpilot/calls/calls.constants.d.ts +17 -0
  42. package/dist/talkpilot/calls/calls.constants.d.ts.map +1 -0
  43. package/dist/talkpilot/calls/calls.constants.js +20 -0
  44. package/dist/talkpilot/calls/calls.constants.js.map +1 -0
  45. package/dist/talkpilot/calls/calls.getters.d.ts +3 -2
  46. package/dist/talkpilot/calls/calls.getters.d.ts.map +1 -1
  47. package/dist/talkpilot/calls/calls.getters.js +1 -2
  48. package/dist/talkpilot/calls/calls.getters.js.map +1 -1
  49. package/dist/talkpilot/calls/calls.statistics.getters.d.ts +19 -0
  50. package/dist/talkpilot/calls/calls.statistics.getters.d.ts.map +1 -0
  51. package/dist/talkpilot/calls/calls.statistics.getters.js +375 -0
  52. package/dist/talkpilot/calls/calls.statistics.getters.js.map +1 -0
  53. package/dist/talkpilot/calls/calls.statistics.ticketScope.d.ts +12 -0
  54. package/dist/talkpilot/calls/calls.statistics.ticketScope.d.ts.map +1 -0
  55. package/dist/talkpilot/calls/calls.statistics.ticketScope.js +37 -0
  56. package/dist/talkpilot/calls/calls.statistics.ticketScope.js.map +1 -0
  57. package/dist/talkpilot/calls/calls.statistics.tickets.d.ts +17 -0
  58. package/dist/talkpilot/calls/calls.statistics.tickets.d.ts.map +1 -0
  59. package/dist/talkpilot/calls/calls.statistics.tickets.js +33 -0
  60. package/dist/talkpilot/calls/calls.statistics.tickets.js.map +1 -0
  61. package/dist/talkpilot/calls/calls.statistics.types.d.ts +39 -0
  62. package/dist/talkpilot/calls/calls.statistics.types.d.ts.map +1 -0
  63. package/dist/talkpilot/calls/calls.statistics.types.js +3 -0
  64. package/dist/talkpilot/calls/calls.statistics.types.js.map +1 -0
  65. package/dist/talkpilot/calls/calls.types.d.ts +6 -10
  66. package/dist/talkpilot/calls/calls.types.d.ts.map +1 -1
  67. package/dist/talkpilot/calls/calls.types.js +0 -3
  68. package/dist/talkpilot/calls/calls.types.js.map +1 -1
  69. package/dist/talkpilot/calls/dashboard/calls.dashboard.d.ts +36 -0
  70. package/dist/talkpilot/calls/dashboard/calls.dashboard.d.ts.map +1 -0
  71. package/dist/talkpilot/calls/dashboard/calls.dashboard.js +208 -0
  72. package/dist/talkpilot/calls/dashboard/calls.dashboard.js.map +1 -0
  73. package/dist/talkpilot/calls/dashboard/calls.dashboard.types.d.ts +66 -0
  74. package/dist/talkpilot/calls/dashboard/calls.dashboard.types.d.ts.map +1 -0
  75. package/dist/talkpilot/calls/dashboard/calls.dashboard.types.js +3 -0
  76. package/dist/talkpilot/calls/dashboard/calls.dashboard.types.js.map +1 -0
  77. package/dist/talkpilot/calls/index.d.ts +4 -0
  78. package/dist/talkpilot/calls/index.d.ts.map +1 -1
  79. package/dist/talkpilot/calls/index.js +4 -0
  80. package/dist/talkpilot/calls/index.js.map +1 -1
  81. package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts +2 -1
  82. package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts.map +1 -1
  83. package/dist/talkpilot/clientsConfig/clientsConfig.getters.js +14 -0
  84. package/dist/talkpilot/clientsConfig/clientsConfig.getters.js.map +1 -1
  85. package/dist/talkpilot/clientsConfig/clientsConfig.types.d.ts +4 -1
  86. package/dist/talkpilot/clientsConfig/clientsConfig.types.d.ts.map +1 -1
  87. package/dist/talkpilot/clientsConfig/clientsConfig.types.js +6 -0
  88. package/dist/talkpilot/clientsConfig/clientsConfig.types.js.map +1 -1
  89. package/dist/talkpilot/flows/flows.schema.js +1 -1
  90. package/dist/talkpilot/phone_numbers/index.d.ts +2 -2
  91. package/dist/talkpilot/phone_numbers/phone_numbers.getter.d.ts +1 -1
  92. package/dist/talkpilot/phone_numbers/phone_numbers.getter.d.ts.map +1 -1
  93. package/dist/talkpilot/phone_numbers/phone_numbers.getter.js +5 -3
  94. package/dist/talkpilot/phone_numbers/phone_numbers.getter.js.map +1 -1
  95. package/dist/talkpilot/phone_numbers/phone_numbers.schema.js +12 -12
  96. package/dist/talkpilot/phone_numbers/phone_numbers.types.d.ts +4 -4
  97. package/dist/talkpilot/results/results.getter.d.ts.map +1 -1
  98. package/dist/talkpilot/results/results.getter.js.map +1 -1
  99. package/dist/talkpilot/retry_analyze/retryAnalyze.getters.d.ts.map +1 -1
  100. package/dist/talkpilot/retry_analyze/retryAnalyze.getters.js.map +1 -1
  101. package/dist/utils/date.utils.d.ts +49 -0
  102. package/dist/utils/date.utils.d.ts.map +1 -0
  103. package/dist/utils/date.utils.js +103 -0
  104. package/dist/utils/date.utils.js.map +1 -0
  105. package/dist/utils/shared.types.d.ts +5 -0
  106. package/dist/utils/shared.types.d.ts.map +1 -0
  107. package/dist/utils/shared.types.js +3 -0
  108. package/dist/utils/shared.types.js.map +1 -0
  109. package/dist/utils/statistics.aggregation.d.ts +20 -0
  110. package/dist/utils/statistics.aggregation.d.ts.map +1 -0
  111. package/dist/utils/statistics.aggregation.js +43 -0
  112. package/dist/utils/statistics.aggregation.js.map +1 -0
  113. package/package.json +2 -1
  114. package/src/municipal/tickets/__tests__/tickets.getters.spec.ts +1 -37
  115. package/src/municipal/tickets/__tests__/tickets.statistics.spec.ts +104 -0
  116. package/src/municipal/tickets/index.ts +2 -1
  117. package/src/municipal/tickets/tickets.constants.ts +8 -0
  118. package/src/municipal/tickets/tickets.getters.ts +0 -140
  119. package/src/municipal/tickets/tickets.statistics.aggregation.ts +113 -0
  120. package/src/municipal/tickets/tickets.statistics.getters.ts +93 -0
  121. package/src/municipal/tickets/tickets.types.ts +14 -9
  122. package/src/talkpilot/calls/__tests__/calls.dashboard.spec.ts +46 -0
  123. package/src/talkpilot/calls/__tests__/calls.spec.ts +48 -30
  124. package/src/talkpilot/calls/__tests__/calls.statistics.spec.ts +281 -0
  125. package/src/talkpilot/calls/calls.constants.ts +20 -0
  126. package/src/talkpilot/calls/calls.getters.ts +6 -6
  127. package/src/talkpilot/calls/calls.statistics.getters.ts +525 -0
  128. package/src/talkpilot/calls/calls.statistics.types.ts +44 -0
  129. package/src/talkpilot/calls/calls.types.ts +16 -15
  130. package/src/talkpilot/calls/dashboard/calls.dashboard.ts +243 -0
  131. package/src/talkpilot/calls/dashboard/calls.dashboard.types.ts +70 -0
  132. package/src/talkpilot/calls/index.ts +4 -0
  133. package/src/talkpilot/clientsConfig/__tests__/clientsConfig.getters.spec.ts +53 -0
  134. package/src/talkpilot/clientsConfig/__tests__/clientsConfig.spec.ts +19 -9
  135. package/src/talkpilot/clientsConfig/clientsConfig.getters.ts +34 -1
  136. package/src/talkpilot/clientsConfig/clientsConfig.types.ts +9 -1
  137. package/src/talkpilot/flows/__tests__/flows.schema.spec.ts +6 -2
  138. package/src/talkpilot/flows/flows.schema.ts +1 -1
  139. package/src/talkpilot/phone_numbers/__tests__/phone_numbers.spec.ts +40 -35
  140. package/src/talkpilot/phone_numbers/index.ts +2 -2
  141. package/src/talkpilot/phone_numbers/phone_numbers.getter.ts +10 -6
  142. package/src/talkpilot/phone_numbers/phone_numbers.schema.ts +12 -12
  143. package/src/talkpilot/phone_numbers/phone_numbers.types.ts +4 -4
  144. package/src/talkpilot/results/results.getter.ts +6 -2
  145. package/src/talkpilot/retry_analyze/retryAnalyze.getters.ts +13 -4
  146. package/src/utils/date.utils.ts +116 -0
  147. package/src/utils/shared.types.ts +4 -0
@@ -0,0 +1,281 @@
1
+ import {
2
+ aggregateCallsHourlyByRange,
3
+ aggregateCallsRouting,
4
+ aggregateCallsSummary,
5
+ aggregateCallsTrend,
6
+ findCallSidsWithDraftTicketsByCity,
7
+ findCallSidsWithTicketsByCity,
8
+ getCallsCollection,
9
+ getDepartmentsSubjectsCollection,
10
+ getTicketsCollection,
11
+ findSubjectsByCallSids,
12
+ rangeDurationDays,
13
+ rangeDurationMonths,
14
+ resolveTimeBucketFromRange,
15
+ ticketStatsDateScopeFromYmd,
16
+ } from "../../../index";
17
+ import type { Call } from "../calls.types";
18
+ import type { CallsStatsFilter } from "../calls.statistics.types";
19
+ import {
20
+ createDepartmentSubject,
21
+ createIncomingCallDoc,
22
+ createOutGoingCallDoc,
23
+ createTicket,
24
+ } from "../../../test-utils/factories";
25
+ const CLIENT_ID = "stats-client";
26
+ const CITY = "tests" as const;
27
+ const TZ = "Asia/Jerusalem";
28
+ const at = (iso: string) => new Date(iso);
29
+
30
+ const insertCall = (call: Call) =>
31
+ getCallsCollection().insertOne({ ...call, env: call.env ?? "test" });
32
+
33
+ const incoming = (callSid: string, createdAt: string, extra?: Partial<Call>) =>
34
+ insertCall(
35
+ createIncomingCallDoc({
36
+ clientId: CLIENT_ID,
37
+ callSid,
38
+ createdAt: at(createdAt),
39
+ updatedAt: at(extra?.updatedAt?.toISOString() ?? createdAt),
40
+ ...extra,
41
+ }),
42
+ );
43
+
44
+ const filter = (overrides?: Partial<CallsStatsFilter>): CallsStatsFilter => ({
45
+ clientId: CLIENT_ID,
46
+ startStr: "2026-05-01",
47
+ endStr: "2026-05-31",
48
+ timezone: TZ,
49
+ callSidsWithTickets: [],
50
+ callSidsWithDraftTickets: [],
51
+ ...overrides,
52
+ });
53
+
54
+ const ticketDateScope = (statsFilter: CallsStatsFilter) =>
55
+ ticketStatsDateScopeFromYmd(statsFilter.startStr, statsFilter.endStr, statsFilter.timezone);
56
+
57
+ const summaryWithTicketSids = async (overrides?: Partial<CallsStatsFilter>) => {
58
+ const statsFilter = filter(overrides);
59
+ const dateScope = ticketDateScope(statsFilter);
60
+ const callSidsWithTickets = await findCallSidsWithTicketsByCity(CITY, dateScope);
61
+ const callSidsWithDraftTickets = await findCallSidsWithDraftTicketsByCity(CITY, dateScope);
62
+ return aggregateCallsSummary(filter({ callSidsWithTickets, callSidsWithDraftTickets, ...overrides }));
63
+ };
64
+
65
+ describe("calls statistics getters", () => {
66
+ beforeEach(async () => {
67
+ await incoming("CA-with-ticket", "2026-05-10T10:00:00.000Z", {
68
+ callLength: 120,
69
+ updatedAt: at("2026-05-10T10:02:00.000Z"),
70
+ });
71
+ await incoming("CA-transferred", "2026-05-11T14:00:00.000Z", {
72
+ callLength: 60,
73
+ redirectedCall: true,
74
+ updatedAt: at("2026-05-11T14:01:00.000Z"),
75
+ });
76
+ await incoming("CA-hijacked", "2026-05-12T09:00:00.000Z", {
77
+ callLength: 90,
78
+ isConferenceCall: true,
79
+ redirectedCall: false,
80
+ updatedAt: at("2026-05-12T09:01:30.000Z"),
81
+ });
82
+ await insertCall(
83
+ createOutGoingCallDoc({
84
+ clientId: CLIENT_ID,
85
+ callSid: "CA-outgoing",
86
+ createdAt: at("2026-05-10T11:00:00.000Z"),
87
+ updatedAt: at("2026-05-10T11:01:00.000Z"),
88
+ }),
89
+ );
90
+ await getTicketsCollection().insertOne(
91
+ createTicket({
92
+ cityName: CITY,
93
+ callSid: "CA-with-ticket",
94
+ createdAt: at("2026-05-10T10:00:00.000Z"),
95
+ updatedAt: at("2026-05-10T10:00:00.000Z"),
96
+ externalCallFields: { event_subject_id: "100" },
97
+ }),
98
+ );
99
+ });
100
+
101
+ it("should exclude calls outside the filter date range from summary", async () => {
102
+ await incoming("CA-old", "2024-01-01T10:00:00.000Z", { callLength: 999 });
103
+ const summary = await aggregateCallsSummary(filter());
104
+ expect(summary).toMatchObject({ totalCalls: 3, avgDurationSeconds: 90 });
105
+ });
106
+
107
+ it("should aggregate summary with ticket sids and opened-ticket filter", async () => {
108
+ expect(await summaryWithTicketSids()).toEqual({
109
+ totalCalls: 3,
110
+ openTickets: 1,
111
+ draftTickets: 0,
112
+ avgDurationSeconds: 90,
113
+ });
114
+
115
+ const statsFilter = filter();
116
+ const callSidsWithTickets = await findCallSidsWithTicketsByCity(CITY, ticketDateScope(statsFilter));
117
+ const openedOnly = await aggregateCallsSummary(
118
+ filter({ callSidsWithTickets, ticketStatusFilters: ["opened"] }),
119
+ );
120
+ expect(openedOnly).toMatchObject({ totalCalls: 1, openTickets: 1 });
121
+ });
122
+
123
+ it("should classify routing into mutually exclusive buckets", async () => {
124
+ expect(await aggregateCallsRouting(filter())).toEqual({
125
+ transferred: 1,
126
+ hijacked: 1,
127
+ aiHandled: 1,
128
+ });
129
+ });
130
+
131
+ it("should resolve subjects by callSid with optional limit", async () => {
132
+ await getDepartmentsSubjectsCollection().insertOne(
133
+ createDepartmentSubject({ cityName: CITY, subject_id: "100", subjectName: "Water" }),
134
+ );
135
+ expect(await findSubjectsByCallSids(CITY, ["CA-with-ticket"])).toEqual([
136
+ { subject: "Water", count: 1 },
137
+ ]);
138
+
139
+ await getDepartmentsSubjectsCollection().insertMany([
140
+ createDepartmentSubject({ cityName: CITY, subject_id: "200", subjectName: "Roads" }),
141
+ createDepartmentSubject({ cityName: CITY, subject_id: "300", subjectName: "Lighting" }),
142
+ ]);
143
+ await getTicketsCollection().insertMany([
144
+ createTicket({ cityName: CITY, callSid: "CA-subject-1", externalCallFields: { event_subject_id: "200" } }),
145
+ createTicket({ cityName: CITY, callSid: "CA-subject-2", externalCallFields: { event_subject_id: "300" } }),
146
+ ]);
147
+ const sids = ["CA-subject-1", "CA-subject-2"];
148
+ expect(await findSubjectsByCallSids(CITY, sids)).toHaveLength(2);
149
+ expect(await findSubjectsByCallSids(CITY, sids, 1)).toHaveLength(1);
150
+ });
151
+
152
+ it("should aggregate hourly buckets with averages and optional hour window", async () => {
153
+ const statsFilter = filter();
154
+ const daysInRange = rangeDurationDays(statsFilter.startStr, statsFilter.endStr);
155
+ const full = await aggregateCallsHourlyByRange(statsFilter);
156
+
157
+ expect(full).toHaveLength(24);
158
+ expect(full[0]).toEqual({ hour: 0, totalCalls: 0, avgCalls: 0 });
159
+ expect(full.reduce((sum, h) => sum + h.totalCalls, 0)).toBe(3);
160
+ expect(full.find((h) => h.hour === 12)).toEqual({
161
+ hour: 12,
162
+ totalCalls: 1,
163
+ avgCalls: Math.round((1 / daysInRange) * 10) / 10,
164
+ });
165
+
166
+ expect(
167
+ (await aggregateCallsHourlyByRange(filter({ startStr: "2026-05-12", endStr: "2026-05-12" }))).find(
168
+ (h) => h.hour === 12,
169
+ ),
170
+ ).toEqual({ hour: 12, totalCalls: 1, avgCalls: 1 });
171
+
172
+ const avgForOneCall = Math.round((1 / daysInRange) * 10) / 10;
173
+ expect(await aggregateCallsHourlyByRange(filter({ hourFrom: 10, hourTo: 12 }))).toEqual([
174
+ { hour: 10, totalCalls: 0, avgCalls: 0 },
175
+ { hour: 11, totalCalls: 0, avgCalls: 0 },
176
+ { hour: 12, totalCalls: 1, avgCalls: avgForOneCall },
177
+ ]);
178
+ });
179
+
180
+ it("should fill day trend buckets with zeros", async () => {
181
+ expect(await aggregateCallsTrend(filter({ startStr: "2026-05-10", endStr: "2026-05-14" }), "day")).toEqual([
182
+ { bucket: "2026-05-10", totalCalls: 1, openTickets: 0 },
183
+ { bucket: "2026-05-11", totalCalls: 1, openTickets: 0 },
184
+ { bucket: "2026-05-12", totalCalls: 1, openTickets: 0 },
185
+ { bucket: "2026-05-13", totalCalls: 0, openTickets: 0 },
186
+ { bucket: "2026-05-14", totalCalls: 0, openTickets: 0 },
187
+ ]);
188
+ });
189
+
190
+ it("should bucket calls by calendar day in filter timezone", async () => {
191
+ await incoming("CA-jerusalem-midnight", "2026-05-09T21:00:00.000Z", { callLength: 45 });
192
+ expect(await aggregateCallsTrend(filter({ startStr: "2026-05-10", endStr: "2026-05-12" }), "day")).toEqual([
193
+ { bucket: "2026-05-10", totalCalls: 2, openTickets: 0 },
194
+ { bucket: "2026-05-11", totalCalls: 1, openTickets: 0 },
195
+ { bucket: "2026-05-12", totalCalls: 1, openTickets: 0 },
196
+ ]);
197
+ });
198
+
199
+ it("should pick month, week, or day bucket from calendar-month span", () => {
200
+ expect(rangeDurationMonths("2025-06-01", "2026-05-31")).toBe(12);
201
+ expect(resolveTimeBucketFromRange("2025-06-01", "2026-05-31")).toBe("month");
202
+ expect(resolveTimeBucketFromRange("2026-01-01", "2026-06-30")).toBe("week");
203
+ expect(resolveTimeBucketFromRange("2026-04-01", "2026-06-30")).toBe("day");
204
+ });
205
+
206
+ it("should fill month trend buckets with zeros", async () => {
207
+ await incoming("CA-feb", "2026-02-15T10:00:00.000Z");
208
+ await incoming("CA-apr", "2026-04-15T10:00:00.000Z");
209
+ const month = await aggregateCallsTrend(filter({ startStr: "2026-01-01", endStr: "2026-06-30" }), "month");
210
+ expect(month.map((row) => row.bucket)).toEqual([
211
+ "2026-01-01",
212
+ "2026-02-01",
213
+ "2026-03-01",
214
+ "2026-04-01",
215
+ "2026-05-01",
216
+ "2026-06-01",
217
+ ]);
218
+ expect(month.find((row) => row.bucket === "2026-02-01")).toMatchObject({ totalCalls: 1, openTickets: 0 });
219
+ expect(month.find((row) => row.bucket === "2026-05-01")).toMatchObject({ totalCalls: 3, openTickets: 0 });
220
+ });
221
+
222
+ it("should fill week trend buckets with zeros", async () => {
223
+ await incoming("CA-week-2", "2026-05-18T10:00:00.000Z");
224
+ expect(
225
+ await aggregateCallsTrend(filter({ startStr: "2026-05-04", endStr: "2026-05-24" }), "week"),
226
+ ).toEqual([
227
+ { bucket: "2026-05-04/2026-05-09", totalCalls: 0, openTickets: 0 },
228
+ { bucket: "2026-05-10/2026-05-16", totalCalls: 3, openTickets: 0 },
229
+ { bucket: "2026-05-17/2026-05-23", totalCalls: 1, openTickets: 0 },
230
+ { bucket: "2026-05-24/2026-05-24", totalCalls: 0, openTickets: 0 },
231
+ ]);
232
+ });
233
+
234
+ it("should complete every day bucket across a 12-month range without hanging (DST regression)", async () => {
235
+ const startStr = "2025-07-02";
236
+ const endStr = "2026-06-02";
237
+ const startedAt = Date.now();
238
+ const trend = await aggregateCallsTrend(filter({ startStr, endStr }), "day");
239
+ const elapsedMs = Date.now() - startedAt;
240
+
241
+ expect(trend).toHaveLength(rangeDurationDays(startStr, endStr));
242
+ expect(trend[0]).toEqual({ bucket: startStr, totalCalls: 0, openTickets: 0 });
243
+ expect(trend[trend.length - 1].bucket).toBe(endStr);
244
+ expect(elapsedMs).toBeLessThan(5000);
245
+ });
246
+
247
+ it("should count draft tickets and distinct draft call sids", async () => {
248
+ const descriptionDraft = createTicket({
249
+ cityName: CITY,
250
+ callSid: "CA-hijacked",
251
+ createdAt: at("2026-05-12T09:00:00.000Z"),
252
+ updatedAt: at("2026-05-12T09:00:00.000Z"),
253
+ });
254
+ descriptionDraft.externalCallFields = { event_description: "draft ticket" };
255
+ await getTicketsCollection().insertMany([
256
+ createTicket({
257
+ cityName: CITY,
258
+ callSid: "CA-transferred",
259
+ createdAt: at("2026-05-11T14:00:00.000Z"),
260
+ updatedAt: at("2026-05-11T14:00:00.000Z"),
261
+ externalCallFields: { event_subject_id: "100" },
262
+ }),
263
+ descriptionDraft,
264
+ createTicket({
265
+ cityName: CITY,
266
+ callSid: "CA-whitespace-draft",
267
+ createdAt: at("2026-05-12T10:00:00.000Z"),
268
+ updatedAt: at("2026-05-12T10:00:00.000Z"),
269
+ externalCallFields: { event_subject_id: " " },
270
+ }),
271
+ ]);
272
+
273
+ const statsFilter = filter();
274
+ const dateScope = ticketDateScope(statsFilter);
275
+ const callSidsWithDraftTickets = await findCallSidsWithDraftTicketsByCity(CITY, dateScope);
276
+ const summary = await summaryWithTicketSids();
277
+
278
+ expect(callSidsWithDraftTickets.sort()).toEqual(["CA-hijacked", "CA-whitespace-draft"].sort());
279
+ expect(summary).toMatchObject({ openTickets: 3, draftTickets: 1 });
280
+ });
281
+ });
@@ -0,0 +1,20 @@
1
+ /** Customer role within a conference call. */
2
+ export const CONFERENCE_ROLE_CUSTOMER = "customer" as const;
3
+
4
+ /** Supervisor role within a conference call. */
5
+ export const CONFERENCE_ROLE_SUPERVISOR = "supervisor" as const;
6
+
7
+ /** Hours in a single day. */
8
+ export const HOURS_PER_DAY = 24;
9
+
10
+ /** Max execution time (ms) for calls statistics aggregations. */
11
+ export const STATS_MAX_TIME_MS = 30_000;
12
+
13
+ /**
14
+ * When the client calls bar chart should group by week or month instead of day:
15
+ * > MONTH threshold -> one bar per month
16
+ * > WEEK threshold -> one bar per week
17
+ * otherwise -> one bar per day
18
+ */
19
+ export const TREND_MONTH_BUCKET_MIN_MONTHS = 6;
20
+ export const TREND_WEEK_BUCKET_MIN_MONTHS = 3;
@@ -5,10 +5,11 @@ import {
5
5
  CallUpdateParams,
6
6
  getDb,
7
7
  } from "../index";
8
- import type { CallsByHour, CountOpts, DateRange, ToolExecution } from "./calls.types";
8
+ import type { CountOpts, DateRange, ToolExecution } from "./calls.types";
9
9
  import { Filter, ObjectId } from "mongodb";
10
10
  import * as process from "node:process";
11
11
  import { applyQueryOptions } from "../utils/query.utils";
12
+ import { DashboardHeatmapMetric } from "./dashboard/calls.dashboard.types";
12
13
 
13
14
  export const getCallsCollection = () => {
14
15
  return getDb().collection<Call>("calls");
@@ -61,11 +62,11 @@ export const updateCallByCallSid = async (
61
62
 
62
63
  export const pushToolExecution = async (
63
64
  callSid: string,
64
- execution: ToolExecution
65
+ execution: ToolExecution,
65
66
  ): Promise<void> => {
66
67
  await getCallsCollection().updateOne(
67
68
  { callSid },
68
- { $push: { toolExecutions: execution } }
69
+ { $push: { toolExecutions: execution } },
69
70
  );
70
71
  };
71
72
 
@@ -217,13 +218,12 @@ export async function getCallsHourlyAggregation(
217
218
  clientId: string,
218
219
  dateStr: string,
219
220
  timezone: string,
220
- ): Promise<CallsByHour[]> {
221
+ ): Promise<DashboardHeatmapMetric[]> {
221
222
  const coll = getCallsCollection();
222
223
  const rows = await coll
223
224
  .aggregate<{ _id: number; calls: number }>([
224
225
  // 1. Restrict to the given client
225
226
  { $match: { clientId } },
226
- // 2. Derive dateLocal (createdAt as YYYY-MM-DD in timezone) and hour (0–23 from createdAt in timezone)
227
227
  {
228
228
  $addFields: {
229
229
  dateLocal: {
@@ -242,7 +242,7 @@ export async function getCallsHourlyAggregation(
242
242
  .toArray();
243
243
 
244
244
  return rows.map((r: any) => ({
245
- hour: `${String(r._id).padStart(2, "0")}:00`,
245
+ hour: r._id,
246
246
  calls: r.calls,
247
247
  }));
248
248
  }