@talkpilot/core-db 1.2.2 → 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 (100) hide show
  1. package/.cursor/rules/development.mdc +65 -65
  2. package/DEVELOPMENT.md +98 -98
  3. package/README.md +169 -139
  4. package/README_OLD.md +160 -160
  5. package/dist/municipal/tickets/index.d.ts +2 -1
  6. package/dist/municipal/tickets/index.d.ts.map +1 -1
  7. package/dist/municipal/tickets/index.js +1 -0
  8. package/dist/municipal/tickets/index.js.map +1 -1
  9. package/dist/municipal/tickets/tickets.constants.d.ts +7 -0
  10. package/dist/municipal/tickets/tickets.constants.d.ts.map +1 -0
  11. package/dist/municipal/tickets/tickets.constants.js +10 -0
  12. package/dist/municipal/tickets/tickets.constants.js.map +1 -0
  13. package/dist/municipal/tickets/tickets.deprecated.getters.d.ts +12 -0
  14. package/dist/municipal/tickets/tickets.deprecated.getters.d.ts.map +1 -0
  15. package/dist/municipal/tickets/tickets.deprecated.getters.js +131 -0
  16. package/dist/municipal/tickets/tickets.deprecated.getters.js.map +1 -0
  17. package/dist/municipal/tickets/tickets.getters.d.ts +0 -11
  18. package/dist/municipal/tickets/tickets.getters.d.ts.map +1 -1
  19. package/dist/municipal/tickets/tickets.getters.js +0 -128
  20. package/dist/municipal/tickets/tickets.getters.js.map +1 -1
  21. package/dist/municipal/tickets/tickets.statistics.aggregation.d.ts +45 -0
  22. package/dist/municipal/tickets/tickets.statistics.aggregation.d.ts.map +1 -0
  23. package/dist/municipal/tickets/tickets.statistics.aggregation.js +98 -0
  24. package/dist/municipal/tickets/tickets.statistics.aggregation.js.map +1 -0
  25. package/dist/municipal/tickets/tickets.statistics.dates.d.ts +7 -0
  26. package/dist/municipal/tickets/tickets.statistics.dates.d.ts.map +1 -0
  27. package/dist/municipal/tickets/tickets.statistics.dates.js +40 -0
  28. package/dist/municipal/tickets/tickets.statistics.dates.js.map +1 -0
  29. package/dist/municipal/tickets/tickets.statistics.getters.d.ts +9 -0
  30. package/dist/municipal/tickets/tickets.statistics.getters.d.ts.map +1 -0
  31. package/dist/municipal/tickets/tickets.statistics.getters.js +55 -0
  32. package/dist/municipal/tickets/tickets.statistics.getters.js.map +1 -0
  33. package/dist/municipal/tickets/tickets.statistics.pipeline.d.ts +53 -0
  34. package/dist/municipal/tickets/tickets.statistics.pipeline.d.ts.map +1 -0
  35. package/dist/municipal/tickets/tickets.statistics.pipeline.js +112 -0
  36. package/dist/municipal/tickets/tickets.statistics.pipeline.js.map +1 -0
  37. package/dist/municipal/tickets/tickets.statistics.utils.d.ts +7 -0
  38. package/dist/municipal/tickets/tickets.statistics.utils.d.ts.map +1 -0
  39. package/dist/municipal/tickets/tickets.statistics.utils.js +40 -0
  40. package/dist/municipal/tickets/tickets.statistics.utils.js.map +1 -0
  41. package/dist/municipal/tickets/tickets.types.d.ts +10 -5
  42. package/dist/municipal/tickets/tickets.types.d.ts.map +1 -1
  43. package/dist/talkpilot/calls/calls.constants.d.ts +17 -0
  44. package/dist/talkpilot/calls/calls.constants.d.ts.map +1 -0
  45. package/dist/talkpilot/calls/calls.constants.js +20 -0
  46. package/dist/talkpilot/calls/calls.constants.js.map +1 -0
  47. package/dist/talkpilot/calls/calls.statistics.getters.d.ts +19 -0
  48. package/dist/talkpilot/calls/calls.statistics.getters.d.ts.map +1 -0
  49. package/dist/talkpilot/calls/calls.statistics.getters.js +375 -0
  50. package/dist/talkpilot/calls/calls.statistics.getters.js.map +1 -0
  51. package/dist/talkpilot/calls/calls.statistics.ticketScope.d.ts +12 -0
  52. package/dist/talkpilot/calls/calls.statistics.ticketScope.d.ts.map +1 -0
  53. package/dist/talkpilot/calls/calls.statistics.ticketScope.js +37 -0
  54. package/dist/talkpilot/calls/calls.statistics.ticketScope.js.map +1 -0
  55. package/dist/talkpilot/calls/calls.statistics.tickets.d.ts +17 -0
  56. package/dist/talkpilot/calls/calls.statistics.tickets.d.ts.map +1 -0
  57. package/dist/talkpilot/calls/calls.statistics.tickets.js +33 -0
  58. package/dist/talkpilot/calls/calls.statistics.tickets.js.map +1 -0
  59. package/dist/talkpilot/calls/calls.statistics.types.d.ts +39 -0
  60. package/dist/talkpilot/calls/calls.statistics.types.d.ts.map +1 -0
  61. package/dist/talkpilot/calls/calls.statistics.types.js +3 -0
  62. package/dist/talkpilot/calls/calls.statistics.types.js.map +1 -0
  63. package/dist/talkpilot/calls/calls.types.d.ts +3 -3
  64. package/dist/talkpilot/calls/calls.types.d.ts.map +1 -1
  65. package/dist/talkpilot/calls/calls.types.js +0 -3
  66. package/dist/talkpilot/calls/calls.types.js.map +1 -1
  67. package/dist/talkpilot/calls/index.d.ts +3 -0
  68. package/dist/talkpilot/calls/index.d.ts.map +1 -1
  69. package/dist/talkpilot/calls/index.js +3 -0
  70. package/dist/talkpilot/calls/index.js.map +1 -1
  71. package/dist/utils/date.utils.d.ts +49 -0
  72. package/dist/utils/date.utils.d.ts.map +1 -0
  73. package/dist/utils/date.utils.js +103 -0
  74. package/dist/utils/date.utils.js.map +1 -0
  75. package/dist/utils/statistics.aggregation.d.ts +20 -0
  76. package/dist/utils/statistics.aggregation.d.ts.map +1 -0
  77. package/dist/utils/statistics.aggregation.js +43 -0
  78. package/dist/utils/statistics.aggregation.js.map +1 -0
  79. package/jest.config.js +19 -19
  80. package/package.json +46 -46
  81. package/src/municipal/tickets/__tests__/tickets.getters.spec.ts +1 -37
  82. package/src/municipal/tickets/__tests__/tickets.statistics.spec.ts +104 -0
  83. package/src/municipal/tickets/index.ts +2 -1
  84. package/src/municipal/tickets/tickets.constants.ts +8 -0
  85. package/src/municipal/tickets/tickets.getters.ts +0 -140
  86. package/src/municipal/tickets/tickets.statistics.aggregation.ts +113 -0
  87. package/src/municipal/tickets/tickets.statistics.getters.ts +93 -0
  88. package/src/municipal/tickets/tickets.types.ts +14 -9
  89. package/src/talkpilot/calls/__tests__/calls.statistics.spec.ts +281 -0
  90. package/src/talkpilot/calls/calls.constants.ts +20 -0
  91. package/src/talkpilot/calls/calls.statistics.getters.ts +525 -0
  92. package/src/talkpilot/calls/calls.statistics.types.ts +44 -0
  93. package/src/talkpilot/calls/calls.types.ts +6 -3
  94. package/src/talkpilot/calls/index.ts +3 -0
  95. package/src/utils/date.utils.ts +116 -0
  96. package/tsconfig.json +23 -23
  97. package/dist/talkpilot/calls/calls.dashboard.d.ts +0 -3
  98. package/dist/talkpilot/calls/calls.dashboard.d.ts.map +0 -1
  99. package/dist/talkpilot/calls/calls.dashboard.js +0 -191
  100. package/dist/talkpilot/calls/calls.dashboard.js.map +0 -1
@@ -0,0 +1,116 @@
1
+ /**
2
+ * Shared timezone-aware date helpers for statistics aggregations.
3
+ */
4
+
5
+ /** Milliseconds in a single day. */
6
+ export const DAY_MS = 24 * 60 * 60 * 1000;
7
+
8
+ /** Maps a short weekday name (as produced by Intl) to its index (Sun = 0). */
9
+ export const WEEKDAY: Record<string, number> = {
10
+ Sun: 0,
11
+ Mon: 1,
12
+ Tue: 2,
13
+ Wed: 3,
14
+ Thu: 4,
15
+ Fri: 5,
16
+ Sat: 6,
17
+ };
18
+
19
+ /** Parses a `YYYY-MM-DD` string into `[year, month, day]` numbers. */
20
+ export const parseYmd = (dateStr: string): [number, number, number] => {
21
+ const [y, m, d] = dateStr.split("-").map(Number);
22
+ return [y, m, d];
23
+ };
24
+
25
+ /** Builds a zero-padded `YYYY-MM-DD` string from numeric parts. */
26
+ export const ymdToStr = (y: number, m: number, d: number): string =>
27
+ `${y}-${String(m).padStart(2, "0")}-${String(d).padStart(2, "0")}`;
28
+
29
+ /** Parses a `YYYY-MM-DD` string as midnight UTC. */
30
+ export const toUtcDate = (dateStr: string): Date =>
31
+ new Date(`${dateStr}T00:00:00.000Z`);
32
+
33
+ /** Formats a date as a `YYYY-MM-DD` string in UTC. */
34
+ export const formatDate = (date: Date): string => date.toISOString().slice(0, 10);
35
+
36
+ /** Adds `days` to a `YYYY-MM-DD` string, returning the resulting `YYYY-MM-DD`. */
37
+ export const addDaysToDateStr = (dateStr: string, days: number): string =>
38
+ formatDate(new Date(toUtcDate(dateStr).getTime() + days * DAY_MS));
39
+
40
+ /** Next civil calendar day (proleptic Gregorian). Avoids DST 24h steps that can stall iteration. */
41
+ export const incrementYmd = (
42
+ y: number,
43
+ m: number,
44
+ d: number,
45
+ ): [number, number, number] => {
46
+ const next = new Date(Date.UTC(y, m - 1, d + 1));
47
+ return [next.getUTCFullYear(), next.getUTCMonth() + 1, next.getUTCDate()];
48
+ };
49
+
50
+ /**
51
+ * Formats a date as the calendar day in the given timezone.
52
+ * `en-CA` yields ISO-8601 `YYYY-MM-DD`, which is lexicographically sortable
53
+ * (string compare == chronological compare) and matches MongoDB's `%Y-%m-%d`.
54
+ */
55
+ export const formatYmdInTz = (date: Date, timezone: string): string =>
56
+ new Intl.DateTimeFormat("en-CA", { timeZone: timezone }).format(date);
57
+
58
+ /**
59
+ * Returns the calendar day in the given timezone as numeric parts.
60
+ * Reads parts by `type` (not string order), so `en-US` is fine here.
61
+ */
62
+ export const getYmdInTz = (date: Date, timezone: string) => {
63
+ const parts = Object.fromEntries(
64
+ new Intl.DateTimeFormat("en-US", {
65
+ timeZone: timezone,
66
+ year: "numeric",
67
+ month: "numeric",
68
+ day: "numeric",
69
+ })
70
+ .formatToParts(date)
71
+ .map((p) => [p.type, Number(p.value)]),
72
+ );
73
+ return { y: parts.year, m: parts.month, d: parts.day };
74
+ };
75
+
76
+ /** Weekday index (Sun = 0) of a date in the given timezone. */
77
+ export const getWeekdayInTz = (date: Date, timezone: string): number =>
78
+ WEEKDAY[
79
+ new Intl.DateTimeFormat("en-US", { timeZone: timezone, weekday: "short" }).format(
80
+ date,
81
+ )
82
+ ] ?? 0;
83
+
84
+ /**
85
+ * Returns the given `YYYY-MM-DD` day at noon UTC — a stable anchor inside the
86
+ * calendar day, far from midnight so timezone/DST shifts can't cross a date boundary.
87
+ */
88
+ export const dateAtNoonUtc = (dateStr: string): Date => {
89
+ const [y, m, d] = parseYmd(dateStr);
90
+ return new Date(Date.UTC(y, m - 1, d, 12, 0, 0));
91
+ };
92
+
93
+ /**
94
+ * UTC instant at the start of the given `YYYY-MM-DD` calendar day in `timezone`.
95
+ * Binary-searches the boundary so it stays correct across DST transitions.
96
+ */
97
+ export const startOfCalendarDayInTz = (dateStr: string, timezone: string): Date => {
98
+ const anchor = dateAtNoonUtc(dateStr).getTime();
99
+ let low = anchor - 2 * DAY_MS;
100
+ let high = anchor + DAY_MS;
101
+
102
+ while (high - low > 1 /* ms */) {
103
+ const mid = Math.floor((low + high) / 2);
104
+ if (formatYmdInTz(new Date(mid), timezone) >= dateStr) high = mid;
105
+ else low = mid;
106
+ }
107
+
108
+ return new Date(high);
109
+ };
110
+
111
+ /** UTC instant at the last millisecond of the given calendar day in `timezone`. */
112
+ export const endOfCalendarDayInTz = (dateStr: string, timezone: string): Date => {
113
+ const [y, m, d] = parseYmd(dateStr);
114
+ const nextDay = new Date(Date.UTC(y, m - 1, d + 1)).toISOString().slice(0, 10);
115
+ return new Date(startOfCalendarDayInTz(nextDay, timezone).getTime() - 1);
116
+ };
package/tsconfig.json CHANGED
@@ -1,23 +1,23 @@
1
- {
2
- "compilerOptions": {
3
- "target": "ES2022",
4
- "module": "CommonJS",
5
- "moduleResolution": "node",
6
- "lib": ["ES2022"],
7
- "declaration": true,
8
- "declarationMap": true,
9
- "sourceMap": true,
10
- "outDir": "dist",
11
- "rootDir": "src",
12
- "strict": true,
13
- "esModuleInterop": true,
14
- "forceConsistentCasingInFileNames": true,
15
- "skipLibCheck": true,
16
- "resolveJsonModule": true,
17
- "experimentalDecorators": true,
18
- "emitDecoratorMetadata": true,
19
- "baseUrl": "."
20
- },
21
- "include": ["src/**/*"],
22
- "exclude": ["node_modules", "dist", "test", "**/*.test.ts", "**/__tests__/*", "**/*.spec.ts"]
23
- }
1
+ {
2
+ "compilerOptions": {
3
+ "target": "ES2022",
4
+ "module": "CommonJS",
5
+ "moduleResolution": "node",
6
+ "lib": ["ES2022"],
7
+ "declaration": true,
8
+ "declarationMap": true,
9
+ "sourceMap": true,
10
+ "outDir": "dist",
11
+ "rootDir": "src",
12
+ "strict": true,
13
+ "esModuleInterop": true,
14
+ "forceConsistentCasingInFileNames": true,
15
+ "skipLibCheck": true,
16
+ "resolveJsonModule": true,
17
+ "experimentalDecorators": true,
18
+ "emitDecoratorMetadata": true,
19
+ "baseUrl": "."
20
+ },
21
+ "include": ["src/**/*"],
22
+ "exclude": ["node_modules", "dist", "test", "**/*.test.ts", "**/__tests__/*", "**/*.spec.ts"]
23
+ }
@@ -1,3 +0,0 @@
1
- import { DashboardStatsParams, DashboardStatsResult } from "./calls.types";
2
- export declare function getDashboardStats(params: DashboardStatsParams): Promise<DashboardStatsResult>;
3
- //# sourceMappingURL=calls.dashboard.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"calls.dashboard.d.ts","sourceRoot":"","sources":["../../../src/talkpilot/calls/calls.dashboard.ts"],"names":[],"mappings":"AAEA,OAAO,EAA8B,oBAAoB,EAAE,oBAAoB,EAAiD,MAAM,eAAe,CAAC;AAmGtJ,wBAAsB,iBAAiB,CACrC,MAAM,EAAE,oBAAoB,GAC3B,OAAO,CAAC,oBAAoB,CAAC,CAuG/B"}
@@ -1,191 +0,0 @@
1
- "use strict";
2
- var __importDefault = (this && this.__importDefault) || function (mod) {
3
- return (mod && mod.__esModule) ? mod : { "default": mod };
4
- };
5
- Object.defineProperty(exports, "__esModule", { value: true });
6
- exports.getDashboardStats = getDashboardStats;
7
- const dayjs_1 = __importDefault(require("dayjs"));
8
- const calls_getters_1 = require("./calls.getters");
9
- const clientsConfig_1 = require("../clientsConfig");
10
- const utc_1 = __importDefault(require("dayjs/plugin/utc"));
11
- const timezone_1 = __importDefault(require("dayjs/plugin/timezone"));
12
- dayjs_1.default.extend(utc_1.default);
13
- dayjs_1.default.extend(timezone_1.default);
14
- function buildKpisPipeline() {
15
- return [
16
- {
17
- $group: {
18
- _id: null,
19
- totalCalls: { $sum: 1 },
20
- totalDuration: { $sum: "$callLength" },
21
- completedCount: {
22
- $sum: { $cond: [{ $eq: ["$status", "completed"] }, 1, 0] },
23
- },
24
- failedCount: {
25
- $sum: { $cond: [{ $eq: ["$status", "failed"] }, 1, 0] },
26
- },
27
- noAnswerCount: {
28
- $sum: { $cond: [{ $eq: ["$status", "no-answer"] }, 1, 0] },
29
- },
30
- busyCount: {
31
- $sum: { $cond: [{ $eq: ["$status", "busy"] }, 1, 0] },
32
- },
33
- },
34
- },
35
- ];
36
- }
37
- function buildDailyDataPipeline(timezone) {
38
- return [
39
- {
40
- $group: {
41
- _id: {
42
- $dateToString: {
43
- format: "%Y-%m-%d",
44
- date: "$createdAt",
45
- timezone: timezone,
46
- },
47
- },
48
- count: { $sum: 1 },
49
- completed: {
50
- $sum: { $cond: [{ $eq: ["$status", "completed"] }, 1, 0] },
51
- },
52
- },
53
- },
54
- { $sort: { _id: 1 } },
55
- ];
56
- }
57
- function buildHourlyDataPipeline(timezone) {
58
- return [
59
- {
60
- $group: {
61
- _id: {
62
- day: {
63
- $dateToString: {
64
- format: "%Y-%m-%d",
65
- date: "$createdAt",
66
- timezone: timezone,
67
- },
68
- },
69
- hour: { $hour: { date: "$createdAt", timezone: timezone } },
70
- },
71
- count: { $sum: 1 },
72
- },
73
- },
74
- ];
75
- }
76
- function buildCallLengthBucketsPipeline() {
77
- return [
78
- {
79
- $group: {
80
- _id: null,
81
- short: { $sum: { $cond: [{ $lt: ["$callLength", 60] }, 1, 0] } },
82
- medium: {
83
- $sum: {
84
- $cond: [
85
- {
86
- $and: [
87
- { $gte: ["$callLength", 60] },
88
- { $lte: ["$callLength", 180] },
89
- ],
90
- },
91
- 1,
92
- 0,
93
- ],
94
- },
95
- },
96
- long: { $sum: { $cond: [{ $gt: ["$callLength", 180] }, 1, 0] } },
97
- },
98
- },
99
- ];
100
- }
101
- async function getDashboardStats(params) {
102
- const { clientId, startDate, endDate } = params;
103
- const clientConfig = await (0, clientsConfig_1.getClientConfig)(clientId);
104
- const timezone = clientConfig?.timezone ?? "UTC";
105
- const startDateObj = dayjs_1.default.tz(startDate, timezone).startOf("day").toDate();
106
- const endDateObj = dayjs_1.default.tz(endDate, timezone).endOf("day").toDate();
107
- const pipeline = [
108
- {
109
- $match: {
110
- clientId,
111
- createdAt: { $gte: startDateObj, $lte: endDateObj },
112
- },
113
- },
114
- {
115
- $facet: {
116
- kpis: buildKpisPipeline(),
117
- dailyData: buildDailyDataPipeline(timezone),
118
- hourlyData: buildHourlyDataPipeline(timezone),
119
- callLengthBuckets: buildCallLengthBucketsPipeline(),
120
- },
121
- },
122
- ];
123
- const callsCollection = (0, calls_getters_1.getCallsCollection)();
124
- const [aggregatedResult] = await callsCollection
125
- .aggregate(pipeline)
126
- .toArray();
127
- const kpiData = aggregatedResult?.kpis?.[0] ?? {
128
- totalCalls: 0,
129
- totalDuration: 0,
130
- completedCount: 0,
131
- failedCount: 0,
132
- noAnswerCount: 0,
133
- busyCount: 0,
134
- };
135
- const dailyDataRaw = aggregatedResult?.dailyData ?? [];
136
- const hourlyDataRaw = aggregatedResult?.hourlyData ?? [];
137
- const callLengthRaw = aggregatedResult?.callLengthBuckets?.[0] ?? {
138
- short: 0,
139
- medium: 0,
140
- long: 0,
141
- };
142
- const count = kpiData.totalCalls;
143
- const totalLen = kpiData.totalDuration;
144
- const completed = kpiData.completedCount;
145
- const kpis = {
146
- totalCalls: count,
147
- avgDurationSeconds: count > 0 ? Math.round(totalLen / count) : 0,
148
- timeSavedMinutes: Math.round(totalLen / 60),
149
- successRate: count > 0 ? Math.round((completed / count) * 1000) / 10 : 0,
150
- completedCount: completed,
151
- failedCount: kpiData.failedCount,
152
- noAnswerCount: kpiData.noAnswerCount,
153
- busyCount: kpiData.busyCount,
154
- };
155
- const volumeData = dailyDataRaw.map((d) => ({
156
- date: d._id,
157
- count: d.count,
158
- completed: d.completed,
159
- }));
160
- const heatmapMap = new Map();
161
- for (const bucket of hourlyDataRaw) {
162
- const dayKey = bucket._id.day;
163
- const hour = bucket._id.hour;
164
- if (!heatmapMap.has(dayKey))
165
- heatmapMap.set(dayKey, new Map());
166
- heatmapMap.get(dayKey).set(hour, bucket.count);
167
- }
168
- const toSortedBuckets = (map) => Array.from(map.entries())
169
- .sort(([a], [b]) => a - b)
170
- .map(([h, c]) => ({
171
- hour: `${String(h).padStart(2, "0")}:00`,
172
- calls: c,
173
- }));
174
- const heatmap = {};
175
- for (const [day, hourMap] of Array.from(heatmapMap.entries()).sort()) {
176
- heatmap[day] = toSortedBuckets(hourMap);
177
- }
178
- return {
179
- kpis,
180
- charts: {
181
- volumeData,
182
- heatmap,
183
- callLengthBuckets: {
184
- short: callLengthRaw.short,
185
- medium: callLengthRaw.medium,
186
- long: callLengthRaw.long,
187
- },
188
- },
189
- };
190
- }
191
- //# sourceMappingURL=calls.dashboard.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"calls.dashboard.js","sourceRoot":"","sources":["../../../src/talkpilot/calls/calls.dashboard.ts"],"names":[],"mappings":"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"}