@talkpilot/core-db 1.3.3 → 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.
Files changed (150) hide show
  1. package/.claude/settings.local.json +7 -0
  2. package/README.md +30 -0
  3. package/dist/connection.d.ts.map +1 -1
  4. package/dist/connection.js +0 -10
  5. package/dist/connection.js.map +1 -1
  6. package/dist/index.d.ts +0 -3
  7. package/dist/index.d.ts.map +1 -1
  8. package/dist/index.js +1 -7
  9. package/dist/index.js.map +1 -1
  10. package/dist/municipal/tickets/index.d.ts +2 -1
  11. package/dist/municipal/tickets/index.d.ts.map +1 -1
  12. package/dist/municipal/tickets/index.js +1 -0
  13. package/dist/municipal/tickets/index.js.map +1 -1
  14. package/dist/municipal/tickets/tickets.constants.d.ts +7 -0
  15. package/dist/municipal/tickets/tickets.constants.d.ts.map +1 -0
  16. package/dist/municipal/tickets/tickets.constants.js +10 -0
  17. package/dist/municipal/tickets/tickets.constants.js.map +1 -0
  18. package/dist/municipal/tickets/tickets.deprecated.getters.d.ts +12 -0
  19. package/dist/municipal/tickets/tickets.deprecated.getters.d.ts.map +1 -0
  20. package/dist/municipal/tickets/tickets.deprecated.getters.js +131 -0
  21. package/dist/municipal/tickets/tickets.deprecated.getters.js.map +1 -0
  22. package/dist/municipal/tickets/tickets.getters.d.ts +0 -11
  23. package/dist/municipal/tickets/tickets.getters.d.ts.map +1 -1
  24. package/dist/municipal/tickets/tickets.getters.js +0 -128
  25. package/dist/municipal/tickets/tickets.getters.js.map +1 -1
  26. package/dist/municipal/tickets/tickets.statistics.aggregation.d.ts +35 -0
  27. package/dist/municipal/tickets/tickets.statistics.aggregation.d.ts.map +1 -0
  28. package/dist/municipal/tickets/tickets.statistics.aggregation.js +86 -0
  29. package/dist/municipal/tickets/tickets.statistics.aggregation.js.map +1 -0
  30. package/dist/municipal/tickets/tickets.statistics.dates.d.ts +7 -0
  31. package/dist/municipal/tickets/tickets.statistics.dates.d.ts.map +1 -0
  32. package/dist/municipal/tickets/tickets.statistics.dates.js +40 -0
  33. package/dist/municipal/tickets/tickets.statistics.dates.js.map +1 -0
  34. package/dist/municipal/tickets/tickets.statistics.getters.d.ts +8 -0
  35. package/dist/municipal/tickets/tickets.statistics.getters.d.ts.map +1 -0
  36. package/dist/municipal/tickets/tickets.statistics.getters.js +44 -0
  37. package/dist/municipal/tickets/tickets.statistics.getters.js.map +1 -0
  38. package/dist/municipal/tickets/tickets.statistics.pipeline.d.ts +53 -0
  39. package/dist/municipal/tickets/tickets.statistics.pipeline.d.ts.map +1 -0
  40. package/dist/municipal/tickets/tickets.statistics.pipeline.js +112 -0
  41. package/dist/municipal/tickets/tickets.statistics.pipeline.js.map +1 -0
  42. package/dist/municipal/tickets/tickets.statistics.utils.d.ts +7 -0
  43. package/dist/municipal/tickets/tickets.statistics.utils.d.ts.map +1 -0
  44. package/dist/municipal/tickets/tickets.statistics.utils.js +40 -0
  45. package/dist/municipal/tickets/tickets.statistics.utils.js.map +1 -0
  46. package/dist/municipal/tickets/tickets.types.d.ts +14 -5
  47. package/dist/municipal/tickets/tickets.types.d.ts.map +1 -1
  48. package/dist/talkpilot/calls/calls.constants.d.ts +17 -0
  49. package/dist/talkpilot/calls/calls.constants.d.ts.map +1 -0
  50. package/dist/talkpilot/calls/calls.constants.js +20 -0
  51. package/dist/talkpilot/calls/calls.constants.js.map +1 -0
  52. package/dist/talkpilot/calls/calls.statistics.getters.d.ts +28 -0
  53. package/dist/talkpilot/calls/calls.statistics.getters.d.ts.map +1 -0
  54. package/dist/talkpilot/calls/calls.statistics.getters.js +424 -0
  55. package/dist/talkpilot/calls/calls.statistics.getters.js.map +1 -0
  56. package/dist/talkpilot/calls/calls.statistics.ticketScope.d.ts +12 -0
  57. package/dist/talkpilot/calls/calls.statistics.ticketScope.d.ts.map +1 -0
  58. package/dist/talkpilot/calls/calls.statistics.ticketScope.js +37 -0
  59. package/dist/talkpilot/calls/calls.statistics.ticketScope.js.map +1 -0
  60. package/dist/talkpilot/calls/calls.statistics.tickets.d.ts +17 -0
  61. package/dist/talkpilot/calls/calls.statistics.tickets.d.ts.map +1 -0
  62. package/dist/talkpilot/calls/calls.statistics.tickets.js +33 -0
  63. package/dist/talkpilot/calls/calls.statistics.tickets.js.map +1 -0
  64. package/dist/talkpilot/calls/calls.statistics.types.d.ts +39 -0
  65. package/dist/talkpilot/calls/calls.statistics.types.d.ts.map +1 -0
  66. package/dist/{websitalk/scans/scans.types.js → talkpilot/calls/calls.statistics.types.js} +1 -1
  67. package/dist/talkpilot/calls/calls.statistics.types.js.map +1 -0
  68. package/dist/talkpilot/calls/calls.types.d.ts +1 -2
  69. package/dist/talkpilot/calls/calls.types.d.ts.map +1 -1
  70. package/dist/talkpilot/calls/calls.types.js +0 -3
  71. package/dist/talkpilot/calls/calls.types.js.map +1 -1
  72. package/dist/talkpilot/calls/dashboard/calls.dashboard.d.ts +33 -1
  73. package/dist/talkpilot/calls/dashboard/calls.dashboard.d.ts.map +1 -1
  74. package/dist/talkpilot/calls/dashboard/calls.dashboard.js +131 -146
  75. package/dist/talkpilot/calls/dashboard/calls.dashboard.js.map +1 -1
  76. package/dist/talkpilot/calls/dashboard/calls.dashboard.types.d.ts +27 -6
  77. package/dist/talkpilot/calls/dashboard/calls.dashboard.types.d.ts.map +1 -1
  78. package/dist/talkpilot/calls/index.d.ts +3 -0
  79. package/dist/talkpilot/calls/index.d.ts.map +1 -1
  80. package/dist/talkpilot/calls/index.js +3 -0
  81. package/dist/talkpilot/calls/index.js.map +1 -1
  82. package/dist/test-utils/db-utils.d.ts.map +1 -1
  83. package/dist/test-utils/db-utils.js +0 -2
  84. package/dist/test-utils/db-utils.js.map +1 -1
  85. package/dist/test-utils/factories/index.d.ts +0 -1
  86. package/dist/test-utils/factories/index.d.ts.map +1 -1
  87. package/dist/test-utils/factories/index.js +0 -1
  88. package/dist/test-utils/factories/index.js.map +1 -1
  89. package/dist/utils/date.utils.d.ts +49 -0
  90. package/dist/utils/date.utils.d.ts.map +1 -0
  91. package/dist/utils/date.utils.js +103 -0
  92. package/dist/utils/date.utils.js.map +1 -0
  93. package/dist/utils/statistics.aggregation.d.ts +20 -0
  94. package/dist/utils/statistics.aggregation.d.ts.map +1 -0
  95. package/dist/utils/statistics.aggregation.js +43 -0
  96. package/dist/utils/statistics.aggregation.js.map +1 -0
  97. package/package.json +1 -1
  98. package/src/connection.ts +0 -12
  99. package/src/index.ts +0 -9
  100. package/src/municipal/tickets/__tests__/tickets.getters.spec.ts +1 -37
  101. package/src/municipal/tickets/__tests__/tickets.statistics.spec.ts +51 -0
  102. package/src/municipal/tickets/index.ts +2 -1
  103. package/src/municipal/tickets/tickets.constants.ts +8 -0
  104. package/src/municipal/tickets/tickets.getters.ts +0 -140
  105. package/src/municipal/tickets/tickets.statistics.aggregation.ts +96 -0
  106. package/src/municipal/tickets/tickets.statistics.getters.ts +71 -0
  107. package/src/municipal/tickets/tickets.types.ts +19 -9
  108. package/src/talkpilot/calls/__tests__/calls.dashboard.spec.ts +8 -111
  109. package/src/talkpilot/calls/__tests__/calls.statistics.spec.ts +344 -0
  110. package/src/talkpilot/calls/calls.constants.ts +20 -0
  111. package/src/talkpilot/calls/calls.statistics.getters.ts +587 -0
  112. package/src/talkpilot/calls/calls.statistics.types.ts +44 -0
  113. package/src/talkpilot/calls/calls.types.ts +4 -2
  114. package/src/talkpilot/calls/dashboard/calls.dashboard.ts +148 -197
  115. package/src/talkpilot/calls/dashboard/calls.dashboard.types.ts +25 -12
  116. package/src/talkpilot/calls/index.ts +3 -0
  117. package/src/talkpilot/clientsConfig/__tests__/clientsConfig.spec.ts +7 -0
  118. package/src/test-utils/db-utils.ts +1 -3
  119. package/src/test-utils/factories/index.ts +0 -1
  120. package/src/utils/date.utils.ts +116 -0
  121. package/dist/test-utils/factories/websitalk/scans.d.ts +0 -5
  122. package/dist/test-utils/factories/websitalk/scans.d.ts.map +0 -1
  123. package/dist/test-utils/factories/websitalk/scans.js +0 -25
  124. package/dist/test-utils/factories/websitalk/scans.js.map +0 -1
  125. package/dist/websitalk/index.d.ts +0 -7
  126. package/dist/websitalk/index.d.ts.map +0 -1
  127. package/dist/websitalk/index.js +0 -34
  128. package/dist/websitalk/index.js.map +0 -1
  129. package/dist/websitalk/mongodb-client.d.ts +0 -13
  130. package/dist/websitalk/mongodb-client.d.ts.map +0 -1
  131. package/dist/websitalk/mongodb-client.js +0 -56
  132. package/dist/websitalk/mongodb-client.js.map +0 -1
  133. package/dist/websitalk/scans/index.d.ts +0 -3
  134. package/dist/websitalk/scans/index.d.ts.map +0 -1
  135. package/dist/websitalk/scans/index.js +0 -19
  136. package/dist/websitalk/scans/index.js.map +0 -1
  137. package/dist/websitalk/scans/scans.getters.d.ts +0 -12
  138. package/dist/websitalk/scans/scans.getters.d.ts.map +0 -1
  139. package/dist/websitalk/scans/scans.getters.js +0 -74
  140. package/dist/websitalk/scans/scans.getters.js.map +0 -1
  141. package/dist/websitalk/scans/scans.types.d.ts +0 -45
  142. package/dist/websitalk/scans/scans.types.d.ts.map +0 -1
  143. package/dist/websitalk/scans/scans.types.js.map +0 -1
  144. package/src/test-utils/factories/websitalk/scans.ts +0 -23
  145. package/src/websitalk/index.ts +0 -15
  146. package/src/websitalk/mongodb-client.ts +0 -61
  147. package/src/websitalk/scans/__tests__/scans.spec.ts +0 -218
  148. package/src/websitalk/scans/index.ts +0 -2
  149. package/src/websitalk/scans/scans.getters.ts +0 -113
  150. package/src/websitalk/scans/scans.types.ts +0 -53
@@ -0,0 +1,71 @@
1
+ import { CityName } from "../utils/types";
2
+ import {
3
+ DAY_MS,
4
+ endOfCalendarDayInTz,
5
+ startOfCalendarDayInTz,
6
+ } from "../../utils/date.utils";
7
+ import { DEFAULT_LOOKBACK_DAYS } from "./tickets.constants";
8
+ import {
9
+ isDraftSubjectIdExpr,
10
+ runTicketStatsAggregate,
11
+ subjectResolutionStages,
12
+ ticketsWithCallSidMatch,
13
+ } from "./tickets.statistics.aggregation";
14
+ import type {
15
+ TicketStatsDateRange,
16
+ TicketStatsDateScope,
17
+ TicketSubjectRow,
18
+ } from "./tickets.types";
19
+
20
+ export const resolveTicketStatsDateRange = (
21
+ dateScope?: TicketStatsDateScope,
22
+ ): TicketStatsDateRange => {
23
+ const to = dateScope?.to ?? new Date();
24
+ const from =
25
+ dateScope?.from ?? new Date(to.getTime() - DEFAULT_LOOKBACK_DAYS * DAY_MS);
26
+ return { from, to };
27
+ };
28
+
29
+ export const ticketStatsDateScopeFromYmd = (
30
+ startStr: string,
31
+ endStr: string,
32
+ timezone: string,
33
+ ): TicketStatsDateScope => ({
34
+ from: startOfCalendarDayInTz(startStr, timezone),
35
+ to: endOfCalendarDayInTz(endStr, timezone),
36
+ });
37
+
38
+ export const findCallSidsWithTicketsByCity = async (
39
+ cityName: CityName,
40
+ dateScope?: TicketStatsDateScope,
41
+ ): Promise<string[]> => {
42
+ const rows = await runTicketStatsAggregate<{ _id: unknown }>([
43
+ { $match: ticketsWithCallSidMatch(cityName, resolveTicketStatsDateRange(dateScope)) },
44
+ { $group: { _id: "$callSid" } },
45
+ ]);
46
+ return rows.map((r) => String(r._id)).filter(Boolean);
47
+ };
48
+
49
+ export const findCallSidsWithDraftTicketsByCity = async (
50
+ cityName: CityName,
51
+ dateScope?: TicketStatsDateScope,
52
+ ): Promise<string[]> => {
53
+ const rows = await runTicketStatsAggregate<{ _id: unknown }>([
54
+ { $match: ticketsWithCallSidMatch(cityName, resolveTicketStatsDateRange(dateScope)) },
55
+ { $match: { $expr: isDraftSubjectIdExpr } },
56
+ { $group: { _id: "$callSid" } },
57
+ ]);
58
+ return rows.map((r) => String(r._id)).filter(Boolean);
59
+ };
60
+
61
+ export const findTicketSubjectRows = async (
62
+ cityName: CityName,
63
+ dateScope?: TicketStatsDateScope,
64
+ ): Promise<TicketSubjectRow[]> => {
65
+ const rows = await runTicketStatsAggregate<{ callSid: unknown; subject: unknown }>([
66
+ { $match: ticketsWithCallSidMatch(cityName, resolveTicketStatsDateRange(dateScope)) },
67
+ ...subjectResolutionStages(cityName),
68
+ { $project: { _id: 0, callSid: 1, subject: 1 } },
69
+ ]);
70
+ return rows.map((r) => ({ callSid: String(r.callSid), subject: String(r.subject) }));
71
+ };
@@ -9,7 +9,6 @@ export type Ticket = {
9
9
  callSid?: string;
10
10
  cityName: CityName;
11
11
  externalCallFields: {
12
- // Request fields
13
12
  first_name?: string;
14
13
  last_name?: string;
15
14
  event_description?: string;
@@ -18,10 +17,9 @@ export type Ticket = {
18
17
  event_subject_id?: string;
19
18
  event_sub_subject_id?: string;
20
19
  event_sub_subject_id2?: string | null;
21
- // Response fields (only if external call was successful)
22
20
  call_number?: string;
23
- status?: number; // Response status from Bina API (1 = success)
24
- error?: string; // Error message from Bina API if any
21
+ status?: number;
22
+ error?: string;
25
23
  image1?: string;
26
24
  image2?: string;
27
25
  image3?: string;
@@ -34,10 +32,22 @@ export type Ticket = {
34
32
 
35
33
  export type TicketDoc = WithId<Ticket>;
36
34
 
37
- /** Call count by department (subject). */
38
- export type SubjectStatsItem = {
39
- subject_name: string;
40
- subject_id: string;
35
+ export type SubjectItem = {
36
+ subject: string;
41
37
  count: number;
42
- percentage: number;
38
+ };
39
+
40
+ export type TicketSubjectRow = {
41
+ callSid: string;
42
+ subject: string;
43
+ };
44
+
45
+ export type TicketStatsDateScope = {
46
+ from?: Date;
47
+ to?: Date;
48
+ };
49
+
50
+ export type TicketStatsDateRange = {
51
+ from: Date;
52
+ to: Date;
43
53
  };
@@ -25,8 +25,8 @@ describe("getDashboardStats", () => {
25
25
 
26
26
  const params = {
27
27
  clientId,
28
- startDate: "2026-05-01T00:00:00.000Z",
29
- endDate: "2026-05-31T23:59:59.999Z",
28
+ startDate: "2026-05-01",
29
+ endDate: "2026-05-31",
30
30
  };
31
31
 
32
32
  const result = await getDashboardStats(params);
@@ -35,115 +35,12 @@ describe("getDashboardStats", () => {
35
35
  expect(result.kpis.totalCalls).toBe(0);
36
36
  expect(result.kpis.completedCount).toBe(0);
37
37
  expect(result.kpis.avgDurationSeconds).toBe(0);
38
- expect(result.kpis.busyCount).toBe(0);
39
- expect(result.kpis.timeSavedMinutes).toBe(0);
40
- // ~30 day range -> daily buckets, gap-filled with zeros across the whole range
41
- expect(result.charts.volumeGranularity).toBe("day");
42
- expect(result.charts.volumeData.length).toBeGreaterThan(0);
43
- expect(result.charts.volumeData.every((d) => d.completed === 0)).toBe(true);
44
- expect(result.charts.volumeData[0].date).toBe("2026-05-01");
45
- expect(
46
- result.charts.volumeData[result.charts.volumeData.length - 1].date,
47
- ).toBe("2026-05-31");
48
- });
49
-
50
- it("should pick a bucket size that keeps the volume chart readable across range lengths", async () => {
51
- const clientId = "client-dash-456";
52
-
53
- await getClientsConfigCollection().insertOne({
54
- clientId,
55
- timezone: "UTC",
56
- products: {},
57
- } as any);
58
-
59
- const oneDay = await getDashboardStats({
60
- clientId,
61
- startDate: "2026-05-01T00:00:00.000Z",
62
- endDate: "2026-05-01T23:59:59.999Z",
63
- });
64
- expect(oneDay.charts.volumeGranularity).toBe("hour");
65
-
66
- const threeWeeks = await getDashboardStats({
67
- clientId,
68
- startDate: "2026-05-01T00:00:00.000Z",
69
- endDate: "2026-05-21T23:59:59.999Z",
70
- });
71
- expect(threeWeeks.charts.volumeGranularity).toBe("day");
72
-
73
- const fourMonths = await getDashboardStats({
74
- clientId,
75
- startDate: "2026-01-01T00:00:00.000Z",
76
- endDate: "2026-04-30T23:59:59.999Z",
77
- });
78
- expect(fourMonths.charts.volumeGranularity).toBe("week");
79
-
80
- const eighteenMonths = await getDashboardStats({
81
- clientId,
82
- startDate: "2025-01-01T00:00:00.000Z",
83
- endDate: "2026-06-30T23:59:59.999Z",
84
- });
85
- expect(eighteenMonths.charts.volumeGranularity).toBe("month");
86
-
87
- const threeYears = await getDashboardStats({
88
- clientId,
89
- startDate: "2023-01-01T00:00:00.000Z",
90
- endDate: "2026-01-01T23:59:59.999Z",
91
- });
92
- expect(threeYears.charts.volumeGranularity).toBe("quarter");
93
-
94
- const fiveYears = await getDashboardStats({
95
- clientId,
96
- startDate: "2021-01-01T00:00:00.000Z",
97
- endDate: "2026-01-01T23:59:59.999Z",
98
- });
99
- expect(fiveYears.charts.volumeGranularity).toBe("year");
100
- });
101
-
102
- it("should fill gaps with zero-count buckets so the chart has no missing points", async () => {
103
- const clientId = "client-dash-789";
104
-
105
- await getClientsConfigCollection().insertOne({
106
- clientId,
107
- timezone: "UTC",
108
- products: {},
109
- } as any);
110
-
111
- // Sparse data: only two of the day's hours have calls.
112
- await getCallsCollection().insertMany([
113
- {
114
- clientId,
115
- status: "completed",
116
- callLength: 60,
117
- createdAt: new Date("2026-05-01T09:15:00Z"),
118
- },
119
- {
120
- clientId,
121
- status: "completed",
122
- callLength: 90,
123
- createdAt: new Date("2026-05-01T18:45:00Z"),
124
- },
125
- ] as any);
126
-
127
- const result = await getDashboardStats({
128
- clientId,
129
- startDate: "2026-05-01T00:00:00.000Z",
130
- endDate: "2026-05-01T23:59:59.999Z",
38
+ expect(result.charts.volumeData).toEqual([]);
39
+ expect(result.charts.heatmap).toEqual({});
40
+ expect(result.charts.callLengthBuckets).toEqual({
41
+ short: 0,
42
+ medium: 0,
43
+ long: 0,
131
44
  });
132
-
133
- expect(result.charts.volumeGranularity).toBe("hour");
134
- // A full day of hourly buckets, with no gaps, sorted chronologically.
135
- expect(result.charts.volumeData).toHaveLength(24);
136
- expect(result.charts.volumeData[0].date).toBe("2026-05-01T00:00");
137
- expect(result.charts.volumeData[23].date).toBe("2026-05-01T23:00");
138
-
139
- const populated = result.charts.volumeData.filter((d) => d.completed > 0);
140
- expect(populated).toHaveLength(2);
141
- expect(populated.map((d) => d.date)).toEqual([
142
- "2026-05-01T09:00",
143
- "2026-05-01T18:00",
144
- ]);
145
-
146
- const empty = result.charts.volumeData.filter((d) => d.completed === 0);
147
- expect(empty).toHaveLength(22);
148
45
  });
149
46
  });
@@ -0,0 +1,344 @@
1
+ import {
2
+ aggregateCallsHourlyByRange,
3
+ aggregateCallsRouting,
4
+ aggregateCallsSummary,
5
+ aggregateCallsTrend,
6
+ findCallSidsWithDraftTicketsByCity,
7
+ findCallSidsWithTicketsByCity,
8
+ getCallsCollection,
9
+ getDepartmentsSubjectsCollection,
10
+ getTicketsCollection,
11
+ findSubjectsByCityAndDateRange,
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 aggregate subjects only for calls passing the filter", async () => {
132
+ await getDepartmentsSubjectsCollection().insertMany([
133
+ createDepartmentSubject({ cityName: CITY, subject_id: "100", subjectName: "Water" }),
134
+ createDepartmentSubject({ cityName: CITY, subject_id: "200", subjectName: "Roads" }),
135
+ createDepartmentSubject({ cityName: CITY, subject_id: "300", subjectName: "Lighting" }),
136
+ ]);
137
+ await getTicketsCollection().insertMany([
138
+ createTicket({
139
+ cityName: CITY,
140
+ callSid: "CA-transferred",
141
+ createdAt: at("2026-05-11T14:00:00.000Z"),
142
+ updatedAt: at("2026-05-11T14:00:00.000Z"),
143
+ externalCallFields: { event_subject_id: "200" },
144
+ }),
145
+ createTicket({
146
+ cityName: CITY,
147
+ callSid: "CA-hijacked",
148
+ createdAt: at("2026-05-12T09:00:00.000Z"),
149
+ updatedAt: at("2026-05-12T09:00:00.000Z"),
150
+ externalCallFields: { event_subject_id: "300" },
151
+ }),
152
+ ]);
153
+
154
+ const statsFilter = filter();
155
+ expect(await findSubjectsByCityAndDateRange(CITY, statsFilter, ticketDateScope(statsFilter))).toEqual([
156
+ { subject: "Lighting", count: 1 },
157
+ { subject: "Roads", count: 1 },
158
+ { subject: "Water", count: 1 },
159
+ ]);
160
+
161
+ const transferredOnly = filter({ callRouting: ["transferred"] });
162
+ expect(
163
+ await findSubjectsByCityAndDateRange(CITY, transferredOnly, ticketDateScope(transferredOnly)),
164
+ ).toEqual([{ subject: "Roads", count: 1 }]);
165
+ });
166
+
167
+ it("should aggregate subjects over a wide date range", async () => {
168
+ await getDepartmentsSubjectsCollection().insertMany([
169
+ createDepartmentSubject({ cityName: CITY, subject_id: "100", subjectName: "Water" }),
170
+ createDepartmentSubject({ cityName: CITY, subject_id: "200", subjectName: "Roads" }),
171
+ ]);
172
+ await incoming("CA-old-roads", "2024-01-01T10:00:00.000Z");
173
+ await getTicketsCollection().insertOne(
174
+ createTicket({
175
+ cityName: CITY,
176
+ callSid: "CA-old-roads",
177
+ createdAt: at("2024-01-01T10:00:00.000Z"),
178
+ updatedAt: at("2024-01-01T10:00:00.000Z"),
179
+ externalCallFields: { event_subject_id: "200" },
180
+ }),
181
+ );
182
+
183
+ const statsFilter = filter({ startStr: "2024-01-01", endStr: "2026-12-31" });
184
+ expect(await findSubjectsByCityAndDateRange(CITY, statsFilter, ticketDateScope(statsFilter))).toEqual([
185
+ { subject: "Roads", count: 1 },
186
+ { subject: "Water", count: 1 },
187
+ ]);
188
+ });
189
+
190
+ it("should aggregate hourly buckets with averages and optional hour window", async () => {
191
+ const statsFilter = filter();
192
+ const daysInRange = rangeDurationDays(statsFilter.startStr, statsFilter.endStr);
193
+ const full = await aggregateCallsHourlyByRange(statsFilter);
194
+
195
+ expect(full).toHaveLength(24);
196
+ expect(full[0]).toEqual({ hour: 0, totalCalls: 0, avgCalls: 0 });
197
+ expect(full.reduce((sum, h) => sum + h.totalCalls, 0)).toBe(3);
198
+ expect(full.find((h) => h.hour === 12)).toEqual({
199
+ hour: 12,
200
+ totalCalls: 1,
201
+ avgCalls: Math.round((1 / daysInRange) * 10) / 10,
202
+ });
203
+
204
+ expect(
205
+ (await aggregateCallsHourlyByRange(filter({ startStr: "2026-05-12", endStr: "2026-05-12" }))).find(
206
+ (h) => h.hour === 12,
207
+ ),
208
+ ).toEqual({ hour: 12, totalCalls: 1, avgCalls: 1 });
209
+
210
+ const avgForOneCall = Math.round((1 / daysInRange) * 10) / 10;
211
+ expect(await aggregateCallsHourlyByRange(filter({ hourFrom: "10:00", hourTo: "13:00" }))).toEqual([
212
+ { hour: 10, totalCalls: 0, avgCalls: 0 },
213
+ { hour: 11, totalCalls: 0, avgCalls: 0 },
214
+ { hour: 12, totalCalls: 1, avgCalls: avgForOneCall },
215
+ ]);
216
+
217
+ // hourTo is exclusive: ending at "12:00" stops before the 12:00 call, last bucket is 11.
218
+ expect(await aggregateCallsHourlyByRange(filter({ hourFrom: "10:00", hourTo: "12:00" }))).toEqual([
219
+ { hour: 10, totalCalls: 0, avgCalls: 0 },
220
+ { hour: 11, totalCalls: 0, avgCalls: 0 },
221
+ ]);
222
+ });
223
+
224
+ it("should emit a wrap-around hour window and count calls on both sides of midnight", async () => {
225
+ // 2026-05-12 local: 23:30 (20:30Z) and 01:00 (2026-05-12 22:00Z) straddle midnight;
226
+ // CA-two-am at 02:00 local (2026-05-11 23:00Z) sits exactly on the exclusive end.
227
+ await incoming("CA-late", "2026-05-12T20:30:00.000Z");
228
+ await incoming("CA-early", "2026-05-12T22:00:00.000Z");
229
+ await incoming("CA-two-am", "2026-05-11T23:00:00.000Z");
230
+ const oneDay = { startStr: "2026-05-12", endStr: "2026-05-13" } as const;
231
+
232
+ const buckets = await aggregateCallsHourlyByRange(
233
+ filter({ ...oneDay, hourFrom: "22:00", hourTo: "02:00" }),
234
+ );
235
+
236
+ // hourTo "02:00" is exclusive: last bucket is 1, and the 02:00 call is not counted.
237
+ expect(buckets.map((b) => b.hour)).toEqual([22, 23, 0, 1]);
238
+ expect(buckets.find((b) => b.hour === 23)?.totalCalls).toBe(1);
239
+ expect(buckets.find((b) => b.hour === 1)?.totalCalls).toBe(1);
240
+ expect(buckets.reduce((sum, b) => sum + b.totalCalls, 0)).toBe(2);
241
+ });
242
+
243
+ it("should fill day trend buckets with zeros", async () => {
244
+ expect(await aggregateCallsTrend(filter({ startStr: "2026-05-10", endStr: "2026-05-14" }), "day")).toEqual([
245
+ { bucket: "2026-05-10", totalCalls: 1, openTickets: 0 },
246
+ { bucket: "2026-05-11", totalCalls: 1, openTickets: 0 },
247
+ { bucket: "2026-05-12", totalCalls: 1, openTickets: 0 },
248
+ { bucket: "2026-05-13", totalCalls: 0, openTickets: 0 },
249
+ { bucket: "2026-05-14", totalCalls: 0, openTickets: 0 },
250
+ ]);
251
+ });
252
+
253
+ it("should bucket calls by calendar day in filter timezone", async () => {
254
+ await incoming("CA-jerusalem-midnight", "2026-05-09T21:00:00.000Z", { callLength: 45 });
255
+ expect(await aggregateCallsTrend(filter({ startStr: "2026-05-10", endStr: "2026-05-12" }), "day")).toEqual([
256
+ { bucket: "2026-05-10", totalCalls: 2, openTickets: 0 },
257
+ { bucket: "2026-05-11", totalCalls: 1, openTickets: 0 },
258
+ { bucket: "2026-05-12", totalCalls: 1, openTickets: 0 },
259
+ ]);
260
+ });
261
+
262
+ it("should pick month, week, or day bucket from calendar-month span", () => {
263
+ expect(rangeDurationMonths("2025-06-01", "2026-05-31")).toBe(12);
264
+ expect(resolveTimeBucketFromRange("2025-06-01", "2026-05-31")).toBe("month");
265
+ expect(resolveTimeBucketFromRange("2026-01-01", "2026-06-30")).toBe("week");
266
+ expect(resolveTimeBucketFromRange("2026-04-01", "2026-06-30")).toBe("day");
267
+ });
268
+
269
+ it("should fill month trend buckets with zeros", async () => {
270
+ await incoming("CA-feb", "2026-02-15T10:00:00.000Z");
271
+ await incoming("CA-apr", "2026-04-15T10:00:00.000Z");
272
+ const month = await aggregateCallsTrend(filter({ startStr: "2026-01-01", endStr: "2026-06-30" }), "month");
273
+ expect(month.map((row) => row.bucket)).toEqual([
274
+ "2026-01-01",
275
+ "2026-02-01",
276
+ "2026-03-01",
277
+ "2026-04-01",
278
+ "2026-05-01",
279
+ "2026-06-01",
280
+ ]);
281
+ expect(month.find((row) => row.bucket === "2026-02-01")).toMatchObject({ totalCalls: 1, openTickets: 0 });
282
+ expect(month.find((row) => row.bucket === "2026-05-01")).toMatchObject({ totalCalls: 3, openTickets: 0 });
283
+ });
284
+
285
+ it("should fill week trend buckets with zeros", async () => {
286
+ await incoming("CA-week-2", "2026-05-18T10:00:00.000Z");
287
+ expect(
288
+ await aggregateCallsTrend(filter({ startStr: "2026-05-04", endStr: "2026-05-24" }), "week"),
289
+ ).toEqual([
290
+ { bucket: "2026-05-04/2026-05-09", totalCalls: 0, openTickets: 0 },
291
+ { bucket: "2026-05-10/2026-05-16", totalCalls: 3, openTickets: 0 },
292
+ { bucket: "2026-05-17/2026-05-23", totalCalls: 1, openTickets: 0 },
293
+ { bucket: "2026-05-24/2026-05-24", totalCalls: 0, openTickets: 0 },
294
+ ]);
295
+ });
296
+
297
+ it("should complete every day bucket across a 12-month range without hanging (DST regression)", async () => {
298
+ const startStr = "2025-07-02";
299
+ const endStr = "2026-06-02";
300
+ const startedAt = Date.now();
301
+ const trend = await aggregateCallsTrend(filter({ startStr, endStr }), "day");
302
+ const elapsedMs = Date.now() - startedAt;
303
+
304
+ expect(trend).toHaveLength(rangeDurationDays(startStr, endStr));
305
+ expect(trend[0]).toEqual({ bucket: startStr, totalCalls: 0, openTickets: 0 });
306
+ expect(trend[trend.length - 1].bucket).toBe(endStr);
307
+ expect(elapsedMs).toBeLessThan(5000);
308
+ });
309
+
310
+ it("should count draft tickets and distinct draft call sids", async () => {
311
+ const descriptionDraft = createTicket({
312
+ cityName: CITY,
313
+ callSid: "CA-hijacked",
314
+ createdAt: at("2026-05-12T09:00:00.000Z"),
315
+ updatedAt: at("2026-05-12T09:00:00.000Z"),
316
+ });
317
+ descriptionDraft.externalCallFields = { event_description: "draft ticket" };
318
+ await getTicketsCollection().insertMany([
319
+ createTicket({
320
+ cityName: CITY,
321
+ callSid: "CA-transferred",
322
+ createdAt: at("2026-05-11T14:00:00.000Z"),
323
+ updatedAt: at("2026-05-11T14:00:00.000Z"),
324
+ externalCallFields: { event_subject_id: "100" },
325
+ }),
326
+ descriptionDraft,
327
+ createTicket({
328
+ cityName: CITY,
329
+ callSid: "CA-whitespace-draft",
330
+ createdAt: at("2026-05-12T10:00:00.000Z"),
331
+ updatedAt: at("2026-05-12T10:00:00.000Z"),
332
+ externalCallFields: { event_subject_id: " " },
333
+ }),
334
+ ]);
335
+
336
+ const statsFilter = filter();
337
+ const dateScope = ticketDateScope(statsFilter);
338
+ const callSidsWithDraftTickets = await findCallSidsWithDraftTicketsByCity(CITY, dateScope);
339
+ const summary = await summaryWithTicketSids();
340
+
341
+ expect(callSidsWithDraftTickets.sort()).toEqual(["CA-hijacked", "CA-whitespace-draft"].sort());
342
+ expect(summary).toMatchObject({ openTickets: 3, draftTickets: 1 });
343
+ });
344
+ });
@@ -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;