@talkpilot/core-db 1.3.0 → 1.3.1
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/README.md +0 -30
- package/dist/municipal/tickets/index.d.ts +1 -2
- package/dist/municipal/tickets/index.d.ts.map +1 -1
- package/dist/municipal/tickets/index.js +0 -1
- package/dist/municipal/tickets/index.js.map +1 -1
- package/dist/municipal/tickets/tickets.getters.d.ts +11 -0
- package/dist/municipal/tickets/tickets.getters.d.ts.map +1 -1
- package/dist/municipal/tickets/tickets.getters.js +128 -0
- package/dist/municipal/tickets/tickets.getters.js.map +1 -1
- package/dist/municipal/tickets/tickets.types.d.ts +5 -10
- package/dist/municipal/tickets/tickets.types.d.ts.map +1 -1
- package/dist/talkpilot/calls/calls.types.d.ts +2 -1
- package/dist/talkpilot/calls/calls.types.d.ts.map +1 -1
- package/dist/talkpilot/calls/calls.types.js +3 -0
- package/dist/talkpilot/calls/calls.types.js.map +1 -1
- package/dist/talkpilot/calls/dashboard/calls.dashboard.d.ts +1 -33
- package/dist/talkpilot/calls/dashboard/calls.dashboard.d.ts.map +1 -1
- package/dist/talkpilot/calls/dashboard/calls.dashboard.js +146 -131
- package/dist/talkpilot/calls/dashboard/calls.dashboard.js.map +1 -1
- package/dist/talkpilot/calls/dashboard/calls.dashboard.types.d.ts +6 -27
- package/dist/talkpilot/calls/dashboard/calls.dashboard.types.d.ts.map +1 -1
- package/dist/talkpilot/calls/index.d.ts +0 -3
- package/dist/talkpilot/calls/index.d.ts.map +1 -1
- package/dist/talkpilot/calls/index.js +0 -3
- package/dist/talkpilot/calls/index.js.map +1 -1
- package/package.json +1 -1
- package/src/municipal/tickets/__tests__/tickets.getters.spec.ts +37 -1
- package/src/municipal/tickets/index.ts +1 -2
- package/src/municipal/tickets/tickets.getters.ts +140 -0
- package/src/municipal/tickets/tickets.types.ts +9 -14
- package/src/talkpilot/calls/__tests__/calls.dashboard.spec.ts +111 -8
- package/src/talkpilot/calls/calls.types.ts +2 -4
- package/src/talkpilot/calls/dashboard/calls.dashboard.ts +197 -148
- package/src/talkpilot/calls/dashboard/calls.dashboard.types.ts +12 -25
- package/src/talkpilot/calls/index.ts +0 -3
- package/src/talkpilot/clientsConfig/__tests__/clientsConfig.spec.ts +0 -7
- package/dist/municipal/tickets/tickets.constants.d.ts +0 -7
- package/dist/municipal/tickets/tickets.constants.d.ts.map +0 -1
- package/dist/municipal/tickets/tickets.constants.js +0 -10
- package/dist/municipal/tickets/tickets.constants.js.map +0 -1
- package/dist/municipal/tickets/tickets.deprecated.getters.d.ts +0 -12
- package/dist/municipal/tickets/tickets.deprecated.getters.d.ts.map +0 -1
- package/dist/municipal/tickets/tickets.deprecated.getters.js +0 -131
- package/dist/municipal/tickets/tickets.deprecated.getters.js.map +0 -1
- package/dist/municipal/tickets/tickets.statistics.aggregation.d.ts +0 -45
- package/dist/municipal/tickets/tickets.statistics.aggregation.d.ts.map +0 -1
- package/dist/municipal/tickets/tickets.statistics.aggregation.js +0 -98
- package/dist/municipal/tickets/tickets.statistics.aggregation.js.map +0 -1
- package/dist/municipal/tickets/tickets.statistics.dates.d.ts +0 -7
- package/dist/municipal/tickets/tickets.statistics.dates.d.ts.map +0 -1
- package/dist/municipal/tickets/tickets.statistics.dates.js +0 -40
- package/dist/municipal/tickets/tickets.statistics.dates.js.map +0 -1
- package/dist/municipal/tickets/tickets.statistics.getters.d.ts +0 -9
- package/dist/municipal/tickets/tickets.statistics.getters.d.ts.map +0 -1
- package/dist/municipal/tickets/tickets.statistics.getters.js +0 -55
- package/dist/municipal/tickets/tickets.statistics.getters.js.map +0 -1
- package/dist/municipal/tickets/tickets.statistics.pipeline.d.ts +0 -53
- package/dist/municipal/tickets/tickets.statistics.pipeline.d.ts.map +0 -1
- package/dist/municipal/tickets/tickets.statistics.pipeline.js +0 -112
- package/dist/municipal/tickets/tickets.statistics.pipeline.js.map +0 -1
- package/dist/municipal/tickets/tickets.statistics.utils.d.ts +0 -7
- package/dist/municipal/tickets/tickets.statistics.utils.d.ts.map +0 -1
- package/dist/municipal/tickets/tickets.statistics.utils.js +0 -40
- package/dist/municipal/tickets/tickets.statistics.utils.js.map +0 -1
- package/dist/talkpilot/calls/calls.constants.d.ts +0 -17
- package/dist/talkpilot/calls/calls.constants.d.ts.map +0 -1
- package/dist/talkpilot/calls/calls.constants.js +0 -20
- package/dist/talkpilot/calls/calls.constants.js.map +0 -1
- package/dist/talkpilot/calls/calls.statistics.getters.d.ts +0 -19
- package/dist/talkpilot/calls/calls.statistics.getters.d.ts.map +0 -1
- package/dist/talkpilot/calls/calls.statistics.getters.js +0 -375
- package/dist/talkpilot/calls/calls.statistics.getters.js.map +0 -1
- package/dist/talkpilot/calls/calls.statistics.ticketScope.d.ts +0 -12
- package/dist/talkpilot/calls/calls.statistics.ticketScope.d.ts.map +0 -1
- package/dist/talkpilot/calls/calls.statistics.ticketScope.js +0 -37
- package/dist/talkpilot/calls/calls.statistics.ticketScope.js.map +0 -1
- package/dist/talkpilot/calls/calls.statistics.tickets.d.ts +0 -17
- package/dist/talkpilot/calls/calls.statistics.tickets.d.ts.map +0 -1
- package/dist/talkpilot/calls/calls.statistics.tickets.js +0 -33
- package/dist/talkpilot/calls/calls.statistics.tickets.js.map +0 -1
- package/dist/talkpilot/calls/calls.statistics.types.d.ts +0 -39
- package/dist/talkpilot/calls/calls.statistics.types.d.ts.map +0 -1
- package/dist/talkpilot/calls/calls.statistics.types.js +0 -3
- package/dist/talkpilot/calls/calls.statistics.types.js.map +0 -1
- package/dist/utils/date.utils.d.ts +0 -49
- package/dist/utils/date.utils.d.ts.map +0 -1
- package/dist/utils/date.utils.js +0 -103
- package/dist/utils/date.utils.js.map +0 -1
- package/dist/utils/statistics.aggregation.d.ts +0 -20
- package/dist/utils/statistics.aggregation.d.ts.map +0 -1
- package/dist/utils/statistics.aggregation.js +0 -43
- package/dist/utils/statistics.aggregation.js.map +0 -1
- package/src/municipal/tickets/__tests__/tickets.statistics.spec.ts +0 -104
- package/src/municipal/tickets/tickets.constants.ts +0 -8
- package/src/municipal/tickets/tickets.statistics.aggregation.ts +0 -113
- package/src/municipal/tickets/tickets.statistics.getters.ts +0 -93
- package/src/talkpilot/calls/__tests__/calls.statistics.spec.ts +0 -281
- package/src/talkpilot/calls/calls.constants.ts +0 -20
- package/src/talkpilot/calls/calls.statistics.getters.ts +0 -525
- package/src/talkpilot/calls/calls.statistics.types.ts +0 -44
- package/src/utils/date.utils.ts +0 -116
|
@@ -1,33 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.needsTicketStatusFilter = exports.wantsTicketMetrics = exports.ticketSidsSetAmongCallSids = exports.countOpenAndDraftAmongCallSids = exports.resolveTicketStatusCallSids = exports.resolveCallsStatsTicketScope = exports.ticketStatsDateScopeForFilter = void 0;
|
|
4
|
-
const clientsConfig_getters_1 = require("../clientsConfig/clientsConfig.getters");
|
|
5
|
-
const tickets_statistics_getters_1 = require("../../municipal/tickets/tickets.statistics.getters");
|
|
6
|
-
const needsTicketStatusFilter = (filters) => {
|
|
7
|
-
if (!filters?.length)
|
|
8
|
-
return false;
|
|
9
|
-
const hasOpened = filters.includes("opened");
|
|
10
|
-
const hasNotOpened = filters.includes("not_opened");
|
|
11
|
-
return hasOpened !== hasNotOpened;
|
|
12
|
-
};
|
|
13
|
-
exports.needsTicketStatusFilter = needsTicketStatusFilter;
|
|
14
|
-
const ticketStatsDateScopeForFilter = (filter) => (0, tickets_statistics_getters_1.ticketStatsDateScopeFromYmd)(filter.startStr, filter.endStr, filter.timezone);
|
|
15
|
-
exports.ticketStatsDateScopeForFilter = ticketStatsDateScopeForFilter;
|
|
16
|
-
const resolveCallsStatsTicketScope = async (filter) => {
|
|
17
|
-
const city = filter.cityName ??
|
|
18
|
-
(await (0, clientsConfig_getters_1.getClientConfig)(filter.clientId))?.products?.municipal?.cityName;
|
|
19
|
-
if (!city)
|
|
20
|
-
return undefined;
|
|
21
|
-
return { city: city, dateScope: (0, exports.ticketStatsDateScopeForFilter)(filter) };
|
|
22
|
-
};
|
|
23
|
-
exports.resolveCallsStatsTicketScope = resolveCallsStatsTicketScope;
|
|
24
|
-
/** Call SIDs with a ticket among filtered calls (for opened / not_opened $match). */
|
|
25
|
-
const resolveTicketStatusCallSids = async (filter, scope, filteredCallSids) => (0, tickets_statistics_getters_1.findCallSidsWithTicketsAmong)(scope.city, filteredCallSids, scope.dateScope);
|
|
26
|
-
exports.resolveTicketStatusCallSids = resolveTicketStatusCallSids;
|
|
27
|
-
const countOpenAndDraftAmongCallSids = (scope, callSids) => (0, tickets_statistics_getters_1.countTicketCallsAmongCallSids)(scope.city, callSids, scope.dateScope);
|
|
28
|
-
exports.countOpenAndDraftAmongCallSids = countOpenAndDraftAmongCallSids;
|
|
29
|
-
const ticketSidsSetAmongCallSids = async (scope, callSids) => new Set(await (0, tickets_statistics_getters_1.findCallSidsWithTicketsAmong)(scope.city, callSids, scope.dateScope));
|
|
30
|
-
exports.ticketSidsSetAmongCallSids = ticketSidsSetAmongCallSids;
|
|
31
|
-
const wantsTicketMetrics = (filter) => needsTicketStatusFilter(filter.ticketStatusFilters) || Boolean(filter.cityName);
|
|
32
|
-
exports.wantsTicketMetrics = wantsTicketMetrics;
|
|
33
|
-
//# sourceMappingURL=calls.statistics.tickets.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"calls.statistics.tickets.js","sourceRoot":"","sources":["../../../src/talkpilot/calls/calls.statistics.tickets.ts"],"names":[],"mappings":";;;AAAA,kFAAyE;AAEzE,mGAI4D;AAS5D,MAAM,uBAAuB,GAAG,CAAC,OAAiD,EAAE,EAAE;IACpF,IAAI,CAAC,OAAO,EAAE,MAAM;QAAE,OAAO,KAAK,CAAC;IACnC,MAAM,SAAS,GAAG,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;IAC7C,MAAM,YAAY,GAAG,OAAO,CAAC,QAAQ,CAAC,YAAY,CAAC,CAAC;IACpD,OAAO,SAAS,KAAK,YAAY,CAAC;AACpC,CAAC,CAAC;AA2CO,0DAAuB;AAzCzB,MAAM,6BAA6B,GAAG,CAC3C,MAAwB,EACF,EAAE,CACxB,IAAA,wDAA2B,EAAC,MAAM,CAAC,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,QAAQ,CAAC,CAAC;AAHlE,QAAA,6BAA6B,iCAGqC;AAExE,MAAM,4BAA4B,GAAG,KAAK,EAC/C,MAAwB,EACoB,EAAE;IAC9C,MAAM,IAAI,GACR,MAAM,CAAC,QAAQ;QACf,CAAC,MAAM,IAAA,uCAAe,EAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,EAAE,QAAQ,EAAE,SAAS,EAAE,QAAQ,CAAC;IAC1E,IAAI,CAAC,IAAI;QAAE,OAAO,SAAS,CAAC;IAC5B,OAAO,EAAE,IAAI,EAAE,IAAgB,EAAE,SAAS,EAAE,IAAA,qCAA6B,EAAC,MAAM,CAAC,EAAE,CAAC;AACtF,CAAC,CAAC;AARW,QAAA,4BAA4B,gCAQvC;AAEF,qFAAqF;AAC9E,MAAM,2BAA2B,GAAG,KAAK,EAC9C,MAAwB,EACxB,KAA4B,EAC5B,gBAA0B,EACP,EAAE,CACrB,IAAA,yDAA4B,EAC1B,KAAK,CAAC,IAAI,EACV,gBAAgB,EAChB,KAAK,CAAC,SAAS,CAChB,CAAC;AATS,QAAA,2BAA2B,+BASpC;AAEG,MAAM,8BAA8B,GAAG,CAC5C,KAA4B,EAC5B,QAAkB,EAClB,EAAE,CAAC,IAAA,0DAA6B,EAAC,KAAK,CAAC,IAAI,EAAE,QAAQ,EAAE,KAAK,CAAC,SAAS,CAAC,CAAC;AAH7D,QAAA,8BAA8B,kCAG+B;AAEnE,MAAM,0BAA0B,GAAG,KAAK,EAC7C,KAA4B,EAC5B,QAAkB,EACI,EAAE,CACxB,IAAI,GAAG,CAAC,MAAM,IAAA,yDAA4B,EAAC,KAAK,CAAC,IAAI,EAAE,QAAQ,EAAE,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC;AAJxE,QAAA,0BAA0B,8BAI8C;AAE9E,MAAM,kBAAkB,GAAG,CAAC,MAAwB,EAAW,EAAE,CACtE,uBAAuB,CAAC,MAAM,CAAC,mBAAmB,CAAC,IAAI,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;AADrE,QAAA,kBAAkB,sBACmD"}
|
|
@@ -1,39 +0,0 @@
|
|
|
1
|
-
export type ResolvedTimeBucket = "day" | "week" | "month";
|
|
2
|
-
export type TicketStatusFilter = "opened" | "not_opened";
|
|
3
|
-
export type CallRoutingFilter = "ai_only" | "transferred" | "hijacked";
|
|
4
|
-
export type CallsStatsFilter = {
|
|
5
|
-
clientId: string;
|
|
6
|
-
startStr: string;
|
|
7
|
-
endStr: string;
|
|
8
|
-
timezone: string;
|
|
9
|
-
hourFrom?: number;
|
|
10
|
-
hourTo?: number;
|
|
11
|
-
callRouting?: CallRoutingFilter[];
|
|
12
|
-
callSidsWithTickets?: string[];
|
|
13
|
-
/** Call SIDs whose municipal ticket matches the draft-ticket definition (see tickets statistics getters). */
|
|
14
|
-
callSidsWithDraftTickets?: string[];
|
|
15
|
-
ticketStatusFilters?: TicketStatusFilter[];
|
|
16
|
-
};
|
|
17
|
-
export type CallsSummaryAggregation = {
|
|
18
|
-
totalCalls: number;
|
|
19
|
-
avgDurationSeconds: number;
|
|
20
|
-
openTickets: number;
|
|
21
|
-
draftTickets: number;
|
|
22
|
-
};
|
|
23
|
-
export type CallsTrendBucket = {
|
|
24
|
-
/** Day: YYYY-MM-DD; week: YYYY-MM-DD/YYYY-MM-DD (inclusive, clipped to filter); month: YYYY-MM-01 */
|
|
25
|
-
bucket: string;
|
|
26
|
-
totalCalls: number;
|
|
27
|
-
openTickets: number;
|
|
28
|
-
};
|
|
29
|
-
export type CallsHourlyBucket = {
|
|
30
|
-
hour: number;
|
|
31
|
-
totalCalls: number;
|
|
32
|
-
avgCalls: number;
|
|
33
|
-
};
|
|
34
|
-
export type CallsRoutingAggregation = {
|
|
35
|
-
aiHandled: number;
|
|
36
|
-
transferred: number;
|
|
37
|
-
hijacked: number;
|
|
38
|
-
};
|
|
39
|
-
//# sourceMappingURL=calls.statistics.types.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"calls.statistics.types.d.ts","sourceRoot":"","sources":["../../../src/talkpilot/calls/calls.statistics.types.ts"],"names":[],"mappings":"AAAA,MAAM,MAAM,kBAAkB,GAAG,KAAK,GAAG,MAAM,GAAG,OAAO,CAAC;AAE1D,MAAM,MAAM,kBAAkB,GAAG,QAAQ,GAAG,YAAY,CAAC;AACzD,MAAM,MAAM,iBAAiB,GAAG,SAAS,GAAG,aAAa,GAAG,UAAU,CAAC;AAEvE,MAAM,MAAM,gBAAgB,GAAG;IAC7B,QAAQ,EAAE,MAAM,CAAC;IACjB,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE,MAAM,CAAC;IACf,QAAQ,EAAE,MAAM,CAAC;IACjB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,WAAW,CAAC,EAAE,iBAAiB,EAAE,CAAC;IAClC,mBAAmB,CAAC,EAAE,MAAM,EAAE,CAAC;IAC/B,6GAA6G;IAC7G,wBAAwB,CAAC,EAAE,MAAM,EAAE,CAAC;IACpC,mBAAmB,CAAC,EAAE,kBAAkB,EAAE,CAAC;CAC5C,CAAC;AAEF,MAAM,MAAM,uBAAuB,GAAG;IACpC,UAAU,EAAE,MAAM,CAAC;IACnB,kBAAkB,EAAE,MAAM,CAAC;IAC3B,WAAW,EAAE,MAAM,CAAC;IACpB,YAAY,EAAE,MAAM,CAAC;CACtB,CAAC;AAEF,MAAM,MAAM,gBAAgB,GAAG;IAC7B,qGAAqG;IACrG,MAAM,EAAE,MAAM,CAAC;IACf,UAAU,EAAE,MAAM,CAAC;IACnB,WAAW,EAAE,MAAM,CAAC;CACrB,CAAC;AAEF,MAAM,MAAM,iBAAiB,GAAG;IAC9B,IAAI,EAAE,MAAM,CAAC;IACb,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;CAClB,CAAC;AAEF,MAAM,MAAM,uBAAuB,GAAG;IACpC,SAAS,EAAE,MAAM,CAAC;IAClB,WAAW,EAAE,MAAM,CAAC;IACpB,QAAQ,EAAE,MAAM,CAAC;CAClB,CAAC"}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"calls.statistics.types.js","sourceRoot":"","sources":["../../../src/talkpilot/calls/calls.statistics.types.ts"],"names":[],"mappings":""}
|
|
@@ -1,49 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Shared timezone-aware date helpers for statistics aggregations.
|
|
3
|
-
*/
|
|
4
|
-
/** Milliseconds in a single day. */
|
|
5
|
-
export declare const DAY_MS: number;
|
|
6
|
-
/** Maps a short weekday name (as produced by Intl) to its index (Sun = 0). */
|
|
7
|
-
export declare const WEEKDAY: Record<string, number>;
|
|
8
|
-
/** Parses a `YYYY-MM-DD` string into `[year, month, day]` numbers. */
|
|
9
|
-
export declare const parseYmd: (dateStr: string) => [number, number, number];
|
|
10
|
-
/** Builds a zero-padded `YYYY-MM-DD` string from numeric parts. */
|
|
11
|
-
export declare const ymdToStr: (y: number, m: number, d: number) => string;
|
|
12
|
-
/** Parses a `YYYY-MM-DD` string as midnight UTC. */
|
|
13
|
-
export declare const toUtcDate: (dateStr: string) => Date;
|
|
14
|
-
/** Formats a date as a `YYYY-MM-DD` string in UTC. */
|
|
15
|
-
export declare const formatDate: (date: Date) => string;
|
|
16
|
-
/** Adds `days` to a `YYYY-MM-DD` string, returning the resulting `YYYY-MM-DD`. */
|
|
17
|
-
export declare const addDaysToDateStr: (dateStr: string, days: number) => string;
|
|
18
|
-
/** Next civil calendar day (proleptic Gregorian). Avoids DST 24h steps that can stall iteration. */
|
|
19
|
-
export declare const incrementYmd: (y: number, m: number, d: number) => [number, number, number];
|
|
20
|
-
/**
|
|
21
|
-
* Formats a date as the calendar day in the given timezone.
|
|
22
|
-
* `en-CA` yields ISO-8601 `YYYY-MM-DD`, which is lexicographically sortable
|
|
23
|
-
* (string compare == chronological compare) and matches MongoDB's `%Y-%m-%d`.
|
|
24
|
-
*/
|
|
25
|
-
export declare const formatYmdInTz: (date: Date, timezone: string) => string;
|
|
26
|
-
/**
|
|
27
|
-
* Returns the calendar day in the given timezone as numeric parts.
|
|
28
|
-
* Reads parts by `type` (not string order), so `en-US` is fine here.
|
|
29
|
-
*/
|
|
30
|
-
export declare const getYmdInTz: (date: Date, timezone: string) => {
|
|
31
|
-
y: number;
|
|
32
|
-
m: number;
|
|
33
|
-
d: number;
|
|
34
|
-
};
|
|
35
|
-
/** Weekday index (Sun = 0) of a date in the given timezone. */
|
|
36
|
-
export declare const getWeekdayInTz: (date: Date, timezone: string) => number;
|
|
37
|
-
/**
|
|
38
|
-
* Returns the given `YYYY-MM-DD` day at noon UTC — a stable anchor inside the
|
|
39
|
-
* calendar day, far from midnight so timezone/DST shifts can't cross a date boundary.
|
|
40
|
-
*/
|
|
41
|
-
export declare const dateAtNoonUtc: (dateStr: string) => Date;
|
|
42
|
-
/**
|
|
43
|
-
* UTC instant at the start of the given `YYYY-MM-DD` calendar day in `timezone`.
|
|
44
|
-
* Binary-searches the boundary so it stays correct across DST transitions.
|
|
45
|
-
*/
|
|
46
|
-
export declare const startOfCalendarDayInTz: (dateStr: string, timezone: string) => Date;
|
|
47
|
-
/** UTC instant at the last millisecond of the given calendar day in `timezone`. */
|
|
48
|
-
export declare const endOfCalendarDayInTz: (dateStr: string, timezone: string) => Date;
|
|
49
|
-
//# sourceMappingURL=date.utils.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"date.utils.d.ts","sourceRoot":"","sources":["../../src/utils/date.utils.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,oCAAoC;AACpC,eAAO,MAAM,MAAM,QAAsB,CAAC;AAE1C,8EAA8E;AAC9E,eAAO,MAAM,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAQ1C,CAAC;AAEF,sEAAsE;AACtE,eAAO,MAAM,QAAQ,GAAI,SAAS,MAAM,KAAG,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,CAGjE,CAAC;AAEF,mEAAmE;AACnE,eAAO,MAAM,QAAQ,GAAI,GAAG,MAAM,EAAE,GAAG,MAAM,EAAE,GAAG,MAAM,KAAG,MACS,CAAC;AAErE,oDAAoD;AACpD,eAAO,MAAM,SAAS,GAAI,SAAS,MAAM,KAAG,IACN,CAAC;AAEvC,sDAAsD;AACtD,eAAO,MAAM,UAAU,GAAI,MAAM,IAAI,KAAG,MAAyC,CAAC;AAElF,kFAAkF;AAClF,eAAO,MAAM,gBAAgB,GAAI,SAAS,MAAM,EAAE,MAAM,MAAM,KAAG,MACG,CAAC;AAErE,oGAAoG;AACpG,eAAO,MAAM,YAAY,GACvB,GAAG,MAAM,EACT,GAAG,MAAM,EACT,GAAG,MAAM,KACR,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,CAGzB,CAAC;AAEF;;;;GAIG;AACH,eAAO,MAAM,aAAa,GAAI,MAAM,IAAI,EAAE,UAAU,MAAM,KAAG,MACU,CAAC;AAExE;;;GAGG;AACH,eAAO,MAAM,UAAU,GAAI,MAAM,IAAI,EAAE,UAAU,MAAM;;;;CAYtD,CAAC;AAEF,+DAA+D;AAC/D,eAAO,MAAM,cAAc,GAAI,MAAM,IAAI,EAAE,UAAU,MAAM,KAAG,MAKtD,CAAC;AAET;;;GAGG;AACH,eAAO,MAAM,aAAa,GAAI,SAAS,MAAM,KAAG,IAG/C,CAAC;AAEF;;;GAGG;AACH,eAAO,MAAM,sBAAsB,GAAI,SAAS,MAAM,EAAE,UAAU,MAAM,KAAG,IAY1E,CAAC;AAEF,mFAAmF;AACnF,eAAO,MAAM,oBAAoB,GAAI,SAAS,MAAM,EAAE,UAAU,MAAM,KAAG,IAIxE,CAAC"}
|
package/dist/utils/date.utils.js
DELETED
|
@@ -1,103 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
/**
|
|
3
|
-
* Shared timezone-aware date helpers for statistics aggregations.
|
|
4
|
-
*/
|
|
5
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
-
exports.endOfCalendarDayInTz = exports.startOfCalendarDayInTz = exports.dateAtNoonUtc = exports.getWeekdayInTz = exports.getYmdInTz = exports.formatYmdInTz = exports.incrementYmd = exports.addDaysToDateStr = exports.formatDate = exports.toUtcDate = exports.ymdToStr = exports.parseYmd = exports.WEEKDAY = exports.DAY_MS = void 0;
|
|
7
|
-
/** Milliseconds in a single day. */
|
|
8
|
-
exports.DAY_MS = 24 * 60 * 60 * 1000;
|
|
9
|
-
/** Maps a short weekday name (as produced by Intl) to its index (Sun = 0). */
|
|
10
|
-
exports.WEEKDAY = {
|
|
11
|
-
Sun: 0,
|
|
12
|
-
Mon: 1,
|
|
13
|
-
Tue: 2,
|
|
14
|
-
Wed: 3,
|
|
15
|
-
Thu: 4,
|
|
16
|
-
Fri: 5,
|
|
17
|
-
Sat: 6,
|
|
18
|
-
};
|
|
19
|
-
/** Parses a `YYYY-MM-DD` string into `[year, month, day]` numbers. */
|
|
20
|
-
const parseYmd = (dateStr) => {
|
|
21
|
-
const [y, m, d] = dateStr.split("-").map(Number);
|
|
22
|
-
return [y, m, d];
|
|
23
|
-
};
|
|
24
|
-
exports.parseYmd = parseYmd;
|
|
25
|
-
/** Builds a zero-padded `YYYY-MM-DD` string from numeric parts. */
|
|
26
|
-
const ymdToStr = (y, m, d) => `${y}-${String(m).padStart(2, "0")}-${String(d).padStart(2, "0")}`;
|
|
27
|
-
exports.ymdToStr = ymdToStr;
|
|
28
|
-
/** Parses a `YYYY-MM-DD` string as midnight UTC. */
|
|
29
|
-
const toUtcDate = (dateStr) => new Date(`${dateStr}T00:00:00.000Z`);
|
|
30
|
-
exports.toUtcDate = toUtcDate;
|
|
31
|
-
/** Formats a date as a `YYYY-MM-DD` string in UTC. */
|
|
32
|
-
const formatDate = (date) => date.toISOString().slice(0, 10);
|
|
33
|
-
exports.formatDate = formatDate;
|
|
34
|
-
/** Adds `days` to a `YYYY-MM-DD` string, returning the resulting `YYYY-MM-DD`. */
|
|
35
|
-
const addDaysToDateStr = (dateStr, days) => (0, exports.formatDate)(new Date((0, exports.toUtcDate)(dateStr).getTime() + days * exports.DAY_MS));
|
|
36
|
-
exports.addDaysToDateStr = addDaysToDateStr;
|
|
37
|
-
/** Next civil calendar day (proleptic Gregorian). Avoids DST 24h steps that can stall iteration. */
|
|
38
|
-
const incrementYmd = (y, m, d) => {
|
|
39
|
-
const next = new Date(Date.UTC(y, m - 1, d + 1));
|
|
40
|
-
return [next.getUTCFullYear(), next.getUTCMonth() + 1, next.getUTCDate()];
|
|
41
|
-
};
|
|
42
|
-
exports.incrementYmd = incrementYmd;
|
|
43
|
-
/**
|
|
44
|
-
* Formats a date as the calendar day in the given timezone.
|
|
45
|
-
* `en-CA` yields ISO-8601 `YYYY-MM-DD`, which is lexicographically sortable
|
|
46
|
-
* (string compare == chronological compare) and matches MongoDB's `%Y-%m-%d`.
|
|
47
|
-
*/
|
|
48
|
-
const formatYmdInTz = (date, timezone) => new Intl.DateTimeFormat("en-CA", { timeZone: timezone }).format(date);
|
|
49
|
-
exports.formatYmdInTz = formatYmdInTz;
|
|
50
|
-
/**
|
|
51
|
-
* Returns the calendar day in the given timezone as numeric parts.
|
|
52
|
-
* Reads parts by `type` (not string order), so `en-US` is fine here.
|
|
53
|
-
*/
|
|
54
|
-
const getYmdInTz = (date, timezone) => {
|
|
55
|
-
const parts = Object.fromEntries(new Intl.DateTimeFormat("en-US", {
|
|
56
|
-
timeZone: timezone,
|
|
57
|
-
year: "numeric",
|
|
58
|
-
month: "numeric",
|
|
59
|
-
day: "numeric",
|
|
60
|
-
})
|
|
61
|
-
.formatToParts(date)
|
|
62
|
-
.map((p) => [p.type, Number(p.value)]));
|
|
63
|
-
return { y: parts.year, m: parts.month, d: parts.day };
|
|
64
|
-
};
|
|
65
|
-
exports.getYmdInTz = getYmdInTz;
|
|
66
|
-
/** Weekday index (Sun = 0) of a date in the given timezone. */
|
|
67
|
-
const getWeekdayInTz = (date, timezone) => exports.WEEKDAY[new Intl.DateTimeFormat("en-US", { timeZone: timezone, weekday: "short" }).format(date)] ?? 0;
|
|
68
|
-
exports.getWeekdayInTz = getWeekdayInTz;
|
|
69
|
-
/**
|
|
70
|
-
* Returns the given `YYYY-MM-DD` day at noon UTC — a stable anchor inside the
|
|
71
|
-
* calendar day, far from midnight so timezone/DST shifts can't cross a date boundary.
|
|
72
|
-
*/
|
|
73
|
-
const dateAtNoonUtc = (dateStr) => {
|
|
74
|
-
const [y, m, d] = (0, exports.parseYmd)(dateStr);
|
|
75
|
-
return new Date(Date.UTC(y, m - 1, d, 12, 0, 0));
|
|
76
|
-
};
|
|
77
|
-
exports.dateAtNoonUtc = dateAtNoonUtc;
|
|
78
|
-
/**
|
|
79
|
-
* UTC instant at the start of the given `YYYY-MM-DD` calendar day in `timezone`.
|
|
80
|
-
* Binary-searches the boundary so it stays correct across DST transitions.
|
|
81
|
-
*/
|
|
82
|
-
const startOfCalendarDayInTz = (dateStr, timezone) => {
|
|
83
|
-
const anchor = (0, exports.dateAtNoonUtc)(dateStr).getTime();
|
|
84
|
-
let low = anchor - 2 * exports.DAY_MS;
|
|
85
|
-
let high = anchor + exports.DAY_MS;
|
|
86
|
-
while (high - low > 1 /* ms */) {
|
|
87
|
-
const mid = Math.floor((low + high) / 2);
|
|
88
|
-
if ((0, exports.formatYmdInTz)(new Date(mid), timezone) >= dateStr)
|
|
89
|
-
high = mid;
|
|
90
|
-
else
|
|
91
|
-
low = mid;
|
|
92
|
-
}
|
|
93
|
-
return new Date(high);
|
|
94
|
-
};
|
|
95
|
-
exports.startOfCalendarDayInTz = startOfCalendarDayInTz;
|
|
96
|
-
/** UTC instant at the last millisecond of the given calendar day in `timezone`. */
|
|
97
|
-
const endOfCalendarDayInTz = (dateStr, timezone) => {
|
|
98
|
-
const [y, m, d] = (0, exports.parseYmd)(dateStr);
|
|
99
|
-
const nextDay = new Date(Date.UTC(y, m - 1, d + 1)).toISOString().slice(0, 10);
|
|
100
|
-
return new Date((0, exports.startOfCalendarDayInTz)(nextDay, timezone).getTime() - 1);
|
|
101
|
-
};
|
|
102
|
-
exports.endOfCalendarDayInTz = endOfCalendarDayInTz;
|
|
103
|
-
//# sourceMappingURL=date.utils.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"date.utils.js","sourceRoot":"","sources":["../../src/utils/date.utils.ts"],"names":[],"mappings":"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"}
|
|
@@ -1,20 +0,0 @@
|
|
|
1
|
-
/** Default server-side cap for statistics aggregations (30s). */
|
|
2
|
-
export declare const DEFAULT_STATS_MAX_TIME_MS = 30000;
|
|
3
|
-
/** Log a warning when SID arrays exceed this size (BSON / pipeline pressure). */
|
|
4
|
-
export declare const MAX_RECOMMENDED_SID_ARRAY_SIZE = 10000;
|
|
5
|
-
export type StatisticsAggregationOptions = {
|
|
6
|
-
maxTimeMS?: number;
|
|
7
|
-
};
|
|
8
|
-
export declare class StatisticsTimeoutError extends Error {
|
|
9
|
-
constructor(maxTimeMS: number);
|
|
10
|
-
}
|
|
11
|
-
export declare const resolveMaxTimeMS: (options?: StatisticsAggregationOptions) => number;
|
|
12
|
-
/** Builds a constant lookup object for O(1) membership checks via $getField. */
|
|
13
|
-
export declare const buildCallSidLookup: (callSids: string[]) => Record<string, true>;
|
|
14
|
-
/**
|
|
15
|
-
* $expr that is true when `$callSid` is in `callSids`.
|
|
16
|
-
* Prefer over `$in` for large arrays embedded in aggregation pipelines.
|
|
17
|
-
*/
|
|
18
|
-
export declare const callSidInSetExpr: (callSids: string[]) => Record<string, unknown>;
|
|
19
|
-
export declare const warnIfLargeSidArray: (label: string, callSids: string[]) => void;
|
|
20
|
-
//# sourceMappingURL=statistics.aggregation.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"statistics.aggregation.d.ts","sourceRoot":"","sources":["../../src/utils/statistics.aggregation.ts"],"names":[],"mappings":"AAAA,iEAAiE;AACjE,eAAO,MAAM,yBAAyB,QAAS,CAAC;AAEhD,iFAAiF;AACjF,eAAO,MAAM,8BAA8B,QAAS,CAAC;AAErD,MAAM,MAAM,4BAA4B,GAAG;IACzC,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB,CAAC;AAEF,qBAAa,sBAAuB,SAAQ,KAAK;gBACnC,SAAS,EAAE,MAAM;CAI9B;AAED,eAAO,MAAM,gBAAgB,GAAI,UAAU,4BAA4B,KAAG,MACzB,CAAC;AAElD,gFAAgF;AAChF,eAAO,MAAM,kBAAkB,GAAI,UAAU,MAAM,EAAE,KAAG,MAAM,CAAC,MAAM,EAAE,IAAI,CACK,CAAC;AAEjF;;;GAGG;AACH,eAAO,MAAM,gBAAgB,GAAI,UAAU,MAAM,EAAE,KAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAW1E,CAAC;AAEH,eAAO,MAAM,mBAAmB,GAAI,OAAO,MAAM,EAAE,UAAU,MAAM,EAAE,KAAG,IAMvE,CAAC"}
|
|
@@ -1,43 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.warnIfLargeSidArray = exports.callSidInSetExpr = exports.buildCallSidLookup = exports.resolveMaxTimeMS = exports.StatisticsTimeoutError = exports.MAX_RECOMMENDED_SID_ARRAY_SIZE = exports.DEFAULT_STATS_MAX_TIME_MS = void 0;
|
|
4
|
-
/** Default server-side cap for statistics aggregations (30s). */
|
|
5
|
-
exports.DEFAULT_STATS_MAX_TIME_MS = 30_000;
|
|
6
|
-
/** Log a warning when SID arrays exceed this size (BSON / pipeline pressure). */
|
|
7
|
-
exports.MAX_RECOMMENDED_SID_ARRAY_SIZE = 10_000;
|
|
8
|
-
class StatisticsTimeoutError extends Error {
|
|
9
|
-
constructor(maxTimeMS) {
|
|
10
|
-
super(`Statistics aggregation exceeded maxTimeMS (${maxTimeMS})`);
|
|
11
|
-
this.name = "StatisticsTimeoutError";
|
|
12
|
-
}
|
|
13
|
-
}
|
|
14
|
-
exports.StatisticsTimeoutError = StatisticsTimeoutError;
|
|
15
|
-
const resolveMaxTimeMS = (options) => options?.maxTimeMS ?? exports.DEFAULT_STATS_MAX_TIME_MS;
|
|
16
|
-
exports.resolveMaxTimeMS = resolveMaxTimeMS;
|
|
17
|
-
/** Builds a constant lookup object for O(1) membership checks via $getField. */
|
|
18
|
-
const buildCallSidLookup = (callSids) => Object.fromEntries(callSids.map((sid) => [sid, true]));
|
|
19
|
-
exports.buildCallSidLookup = buildCallSidLookup;
|
|
20
|
-
/**
|
|
21
|
-
* $expr that is true when `$callSid` is in `callSids`.
|
|
22
|
-
* Prefer over `$in` for large arrays embedded in aggregation pipelines.
|
|
23
|
-
*/
|
|
24
|
-
const callSidInSetExpr = (callSids) => ({
|
|
25
|
-
$eq: [
|
|
26
|
-
{
|
|
27
|
-
$getField: {
|
|
28
|
-
field: "$callSid",
|
|
29
|
-
input: (0, exports.buildCallSidLookup)(callSids),
|
|
30
|
-
default: false,
|
|
31
|
-
},
|
|
32
|
-
},
|
|
33
|
-
true,
|
|
34
|
-
],
|
|
35
|
-
});
|
|
36
|
-
exports.callSidInSetExpr = callSidInSetExpr;
|
|
37
|
-
const warnIfLargeSidArray = (label, callSids) => {
|
|
38
|
-
if (callSids.length > exports.MAX_RECOMMENDED_SID_ARRAY_SIZE) {
|
|
39
|
-
console.warn(`[core-db statistics] ${label}: ${callSids.length} call SIDs exceeds recommended max (${exports.MAX_RECOMMENDED_SID_ARRAY_SIZE}). Narrow the date range or cache SIDs once per request.`);
|
|
40
|
-
}
|
|
41
|
-
};
|
|
42
|
-
exports.warnIfLargeSidArray = warnIfLargeSidArray;
|
|
43
|
-
//# sourceMappingURL=statistics.aggregation.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"statistics.aggregation.js","sourceRoot":"","sources":["../../src/utils/statistics.aggregation.ts"],"names":[],"mappings":";;;AAAA,iEAAiE;AACpD,QAAA,yBAAyB,GAAG,MAAM,CAAC;AAEhD,iFAAiF;AACpE,QAAA,8BAA8B,GAAG,MAAM,CAAC;AAMrD,MAAa,sBAAuB,SAAQ,KAAK;IAC/C,YAAY,SAAiB;QAC3B,KAAK,CAAC,8CAA8C,SAAS,GAAG,CAAC,CAAC;QAClE,IAAI,CAAC,IAAI,GAAG,wBAAwB,CAAC;IACvC,CAAC;CACF;AALD,wDAKC;AAEM,MAAM,gBAAgB,GAAG,CAAC,OAAsC,EAAU,EAAE,CACjF,OAAO,EAAE,SAAS,IAAI,iCAAyB,CAAC;AADrC,QAAA,gBAAgB,oBACqB;AAElD,gFAAgF;AACzE,MAAM,kBAAkB,GAAG,CAAC,QAAkB,EAAwB,EAAE,CAC7E,MAAM,CAAC,WAAW,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC,CAAyB,CAAC;AADpE,QAAA,kBAAkB,sBACkD;AAEjF;;;GAGG;AACI,MAAM,gBAAgB,GAAG,CAAC,QAAkB,EAA2B,EAAE,CAAC,CAAC;IAChF,GAAG,EAAE;QACH;YACE,SAAS,EAAE;gBACT,KAAK,EAAE,UAAU;gBACjB,KAAK,EAAE,IAAA,0BAAkB,EAAC,QAAQ,CAAC;gBACnC,OAAO,EAAE,KAAK;aACf;SACF;QACD,IAAI;KACL;CACF,CAAC,CAAC;AAXU,QAAA,gBAAgB,oBAW1B;AAEI,MAAM,mBAAmB,GAAG,CAAC,KAAa,EAAE,QAAkB,EAAQ,EAAE;IAC7E,IAAI,QAAQ,CAAC,MAAM,GAAG,sCAA8B,EAAE,CAAC;QACrD,OAAO,CAAC,IAAI,CACV,wBAAwB,KAAK,KAAK,QAAQ,CAAC,MAAM,uCAAuC,sCAA8B,0DAA0D,CACjL,CAAC;IACJ,CAAC;AACH,CAAC,CAAC;AANW,QAAA,mBAAmB,uBAM9B"}
|
|
@@ -1,104 +0,0 @@
|
|
|
1
|
-
import {
|
|
2
|
-
findCallSidsWithDraftTicketsByCity,
|
|
3
|
-
findCallSidsWithTicketsByCity,
|
|
4
|
-
findSubjectsByCityAndDateRange,
|
|
5
|
-
findSubjectsByCallSids,
|
|
6
|
-
getDepartmentsSubjectsCollection,
|
|
7
|
-
getTicketsCollection,
|
|
8
|
-
ticketStatsDateScopeFromYmd,
|
|
9
|
-
} from "../../../index";
|
|
10
|
-
import { createDepartmentSubject, createTicket } from "../../../test-utils/factories";
|
|
11
|
-
|
|
12
|
-
const CITY = "tests" as const;
|
|
13
|
-
const TZ = "Asia/Jerusalem";
|
|
14
|
-
const mayScope = () => ticketStatsDateScopeFromYmd("2026-05-01", "2026-05-31", TZ);
|
|
15
|
-
|
|
16
|
-
describe("tickets statistics getters", () => {
|
|
17
|
-
it("should scope callSid lookups to the provided createdAt range", async () => {
|
|
18
|
-
await getTicketsCollection().insertMany([
|
|
19
|
-
createTicket({
|
|
20
|
-
cityName: CITY,
|
|
21
|
-
callSid: "CA-in-range",
|
|
22
|
-
createdAt: new Date("2026-05-15T10:00:00.000Z"),
|
|
23
|
-
updatedAt: new Date("2026-05-15T10:00:00.000Z"),
|
|
24
|
-
}),
|
|
25
|
-
createTicket({
|
|
26
|
-
cityName: CITY,
|
|
27
|
-
callSid: "CA-out-of-range",
|
|
28
|
-
createdAt: new Date("2024-01-01T10:00:00.000Z"),
|
|
29
|
-
updatedAt: new Date("2024-01-01T10:00:00.000Z"),
|
|
30
|
-
}),
|
|
31
|
-
]);
|
|
32
|
-
expect(await findCallSidsWithTicketsByCity(CITY, mayScope())).toEqual(["CA-in-range"]);
|
|
33
|
-
});
|
|
34
|
-
|
|
35
|
-
it("should treat whitespace-only event_subject_id as draft", async () => {
|
|
36
|
-
await getTicketsCollection().insertMany([
|
|
37
|
-
createTicket({
|
|
38
|
-
cityName: CITY,
|
|
39
|
-
callSid: "CA-draft-whitespace",
|
|
40
|
-
createdAt: new Date("2026-05-15T10:00:00.000Z"),
|
|
41
|
-
updatedAt: new Date("2026-05-15T10:00:00.000Z"),
|
|
42
|
-
externalCallFields: { event_subject_id: " \t " },
|
|
43
|
-
}),
|
|
44
|
-
createTicket({
|
|
45
|
-
cityName: CITY,
|
|
46
|
-
callSid: "CA-not-draft",
|
|
47
|
-
createdAt: new Date("2026-05-16T10:00:00.000Z"),
|
|
48
|
-
updatedAt: new Date("2026-05-16T10:00:00.000Z"),
|
|
49
|
-
externalCallFields: { event_subject_id: "100" },
|
|
50
|
-
}),
|
|
51
|
-
]);
|
|
52
|
-
expect(await findCallSidsWithDraftTicketsByCity(CITY, mayScope())).toEqual(["CA-draft-whitespace"]);
|
|
53
|
-
});
|
|
54
|
-
|
|
55
|
-
it("should aggregate subjects by city and createdAt date scope", async () => {
|
|
56
|
-
await getDepartmentsSubjectsCollection().insertOne(
|
|
57
|
-
createDepartmentSubject({ cityName: CITY, subject_id: "200", subjectName: "Roads" }),
|
|
58
|
-
);
|
|
59
|
-
await getTicketsCollection().insertMany([
|
|
60
|
-
createTicket({
|
|
61
|
-
cityName: CITY,
|
|
62
|
-
callSid: "CA-subject-in-range",
|
|
63
|
-
createdAt: new Date("2026-05-15T10:00:00.000Z"),
|
|
64
|
-
updatedAt: new Date("2026-05-15T10:00:00.000Z"),
|
|
65
|
-
externalCallFields: { event_subject_id: "200" },
|
|
66
|
-
}),
|
|
67
|
-
createTicket({
|
|
68
|
-
cityName: CITY,
|
|
69
|
-
callSid: "CA-subject-out-of-range",
|
|
70
|
-
createdAt: new Date("2024-01-01T10:00:00.000Z"),
|
|
71
|
-
updatedAt: new Date("2024-01-01T10:00:00.000Z"),
|
|
72
|
-
externalCallFields: { event_subject_id: "200" },
|
|
73
|
-
}),
|
|
74
|
-
]);
|
|
75
|
-
expect(await findSubjectsByCityAndDateRange(CITY, mayScope())).toEqual([
|
|
76
|
-
{ subject: "Roads", count: 1 },
|
|
77
|
-
]);
|
|
78
|
-
});
|
|
79
|
-
|
|
80
|
-
it("should exclude tickets outside dateScope when aggregating subjects by callSid", async () => {
|
|
81
|
-
await getDepartmentsSubjectsCollection().insertOne(
|
|
82
|
-
createDepartmentSubject({ cityName: CITY, subject_id: "300", subjectName: "Parks" }),
|
|
83
|
-
);
|
|
84
|
-
await getTicketsCollection().insertMany([
|
|
85
|
-
createTicket({
|
|
86
|
-
cityName: CITY,
|
|
87
|
-
callSid: "CA-shared-sid",
|
|
88
|
-
createdAt: new Date("2026-05-15T10:00:00.000Z"),
|
|
89
|
-
updatedAt: new Date("2026-05-15T10:00:00.000Z"),
|
|
90
|
-
externalCallFields: { event_subject_id: "300" },
|
|
91
|
-
}),
|
|
92
|
-
createTicket({
|
|
93
|
-
cityName: CITY,
|
|
94
|
-
callSid: "CA-shared-sid",
|
|
95
|
-
createdAt: new Date("2024-01-01T10:00:00.000Z"),
|
|
96
|
-
updatedAt: new Date("2024-01-01T10:00:00.000Z"),
|
|
97
|
-
externalCallFields: { event_subject_id: "300" },
|
|
98
|
-
}),
|
|
99
|
-
]);
|
|
100
|
-
expect(await findSubjectsByCallSids(CITY, ["CA-shared-sid"], undefined, mayScope())).toEqual([
|
|
101
|
-
{ subject: "Parks", count: 1 },
|
|
102
|
-
]);
|
|
103
|
-
});
|
|
104
|
-
});
|
|
@@ -1,8 +0,0 @@
|
|
|
1
|
-
/** Label used when a ticket has no resolvable department/subject. */
|
|
2
|
-
export const UNCLASSIFIED = "ללא מחלקה";
|
|
3
|
-
|
|
4
|
-
/** Max execution time (ms) for ticket statistics aggregations. */
|
|
5
|
-
export const STATS_MAX_TIME_MS = 30_000;
|
|
6
|
-
|
|
7
|
-
/** Default look-back window (days) for ticket statistics date ranges. */
|
|
8
|
-
export const DEFAULT_LOOKBACK_DAYS = 30;
|
|
@@ -1,113 +0,0 @@
|
|
|
1
|
-
import { CityName } from "../utils/types";
|
|
2
|
-
import { STATS_MAX_TIME_MS, UNCLASSIFIED } from "./tickets.constants";
|
|
3
|
-
import { getTicketsCollection } from "./tickets.getters";
|
|
4
|
-
import type { TicketStatsDateRange } from "./tickets.types";
|
|
5
|
-
|
|
6
|
-
const effectiveSubjectIdExpr = {
|
|
7
|
-
$ifNull: [
|
|
8
|
-
"$externalCallFields.event_subject_id",
|
|
9
|
-
{
|
|
10
|
-
$ifNull: [
|
|
11
|
-
"$externalCallFields.event_sub_subject_id",
|
|
12
|
-
"$externalCallFields.event_sub_subject_id2",
|
|
13
|
-
],
|
|
14
|
-
},
|
|
15
|
-
],
|
|
16
|
-
};
|
|
17
|
-
|
|
18
|
-
const departmentsSubjectsLookup = (cityName: CityName) => ({
|
|
19
|
-
$lookup: {
|
|
20
|
-
from: "departmentsSubjects",
|
|
21
|
-
let: { sid: "$effectiveSubjectId", c: cityName },
|
|
22
|
-
pipeline: [
|
|
23
|
-
{
|
|
24
|
-
$match: {
|
|
25
|
-
$expr: {
|
|
26
|
-
$and: [
|
|
27
|
-
{ $eq: ["$cityName", "$$c"] },
|
|
28
|
-
{
|
|
29
|
-
$or: [
|
|
30
|
-
{ $eq: ["$subject_id", { $ifNull: ["$$sid", ""] }] },
|
|
31
|
-
{ $eq: ["$sub_subject_id", { $ifNull: ["$$sid", ""] }] },
|
|
32
|
-
{ $eq: ["$sub_subject_id2", { $ifNull: ["$$sid", ""] }] },
|
|
33
|
-
],
|
|
34
|
-
},
|
|
35
|
-
],
|
|
36
|
-
},
|
|
37
|
-
},
|
|
38
|
-
},
|
|
39
|
-
{ $limit: 1 },
|
|
40
|
-
],
|
|
41
|
-
as: "subj",
|
|
42
|
-
},
|
|
43
|
-
});
|
|
44
|
-
|
|
45
|
-
const subjectLabelExpr = {
|
|
46
|
-
$cond: {
|
|
47
|
-
if: { $gt: [{ $size: "$subj" }, 0] },
|
|
48
|
-
then: {
|
|
49
|
-
$ifNull: [{ $arrayElemAt: ["$subj.subjectName", 0] }, UNCLASSIFIED],
|
|
50
|
-
},
|
|
51
|
-
else: UNCLASSIFIED,
|
|
52
|
-
},
|
|
53
|
-
};
|
|
54
|
-
|
|
55
|
-
const ticketDateMatch = (dateRange: TicketStatsDateRange) => ({
|
|
56
|
-
createdAt: { $gte: dateRange.from, $lte: dateRange.to },
|
|
57
|
-
});
|
|
58
|
-
|
|
59
|
-
export const isDraftSubjectIdExpr = {
|
|
60
|
-
$or: [
|
|
61
|
-
{ $eq: [{ $type: "$externalCallFields.event_subject_id" }, "missing"] },
|
|
62
|
-
{ $eq: ["$externalCallFields.event_subject_id", null] },
|
|
63
|
-
{
|
|
64
|
-
$regexMatch: {
|
|
65
|
-
input: { $ifNull: ["$externalCallFields.event_subject_id", ""] },
|
|
66
|
-
regex: /^\s*$/,
|
|
67
|
-
},
|
|
68
|
-
},
|
|
69
|
-
],
|
|
70
|
-
};
|
|
71
|
-
|
|
72
|
-
export const ticketsWithCallSidMatch = (
|
|
73
|
-
cityName: CityName,
|
|
74
|
-
dateRange: TicketStatsDateRange,
|
|
75
|
-
) => ({
|
|
76
|
-
cityName,
|
|
77
|
-
callSid: { $exists: true, $nin: [null, ""] },
|
|
78
|
-
...ticketDateMatch(dateRange),
|
|
79
|
-
});
|
|
80
|
-
|
|
81
|
-
export const ticketsByCallSidsMatch = (
|
|
82
|
-
cityName: CityName,
|
|
83
|
-
callSids: string[],
|
|
84
|
-
dateRange?: TicketStatsDateRange,
|
|
85
|
-
) => ({
|
|
86
|
-
cityName,
|
|
87
|
-
callSid: { $in: callSids },
|
|
88
|
-
...(dateRange ? ticketDateMatch(dateRange) : {}),
|
|
89
|
-
});
|
|
90
|
-
|
|
91
|
-
export const subjectGroupStages = (
|
|
92
|
-
cityName: CityName,
|
|
93
|
-
limit?: number,
|
|
94
|
-
): Record<string, unknown>[] => {
|
|
95
|
-
const shouldLimit = typeof limit === "number" && limit > 0;
|
|
96
|
-
return [
|
|
97
|
-
{ $addFields: { effectiveSubjectId: effectiveSubjectIdExpr } },
|
|
98
|
-
departmentsSubjectsLookup(cityName),
|
|
99
|
-
{ $addFields: { subject: subjectLabelExpr } },
|
|
100
|
-
{ $group: { _id: "$subject", count: { $sum: 1 } } },
|
|
101
|
-
{ $sort: { count: -1 } },
|
|
102
|
-
...(shouldLimit ? [{ $limit: limit }] : []),
|
|
103
|
-
];
|
|
104
|
-
};
|
|
105
|
-
|
|
106
|
-
export const runTicketStatsAggregate = async <T>(
|
|
107
|
-
pipeline: Record<string, unknown>[],
|
|
108
|
-
): Promise<T[]> => {
|
|
109
|
-
const rows = await getTicketsCollection()
|
|
110
|
-
.aggregate(pipeline, { maxTimeMS: STATS_MAX_TIME_MS })
|
|
111
|
-
.toArray();
|
|
112
|
-
return rows as T[];
|
|
113
|
-
};
|
|
@@ -1,93 +0,0 @@
|
|
|
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
|
-
subjectGroupStages,
|
|
12
|
-
ticketsByCallSidsMatch,
|
|
13
|
-
ticketsWithCallSidMatch,
|
|
14
|
-
} from "./tickets.statistics.aggregation";
|
|
15
|
-
import type {
|
|
16
|
-
SubjectItem,
|
|
17
|
-
TicketStatsDateRange,
|
|
18
|
-
TicketStatsDateScope,
|
|
19
|
-
} from "./tickets.types";
|
|
20
|
-
|
|
21
|
-
export const resolveTicketStatsDateRange = (
|
|
22
|
-
dateScope?: TicketStatsDateScope,
|
|
23
|
-
): TicketStatsDateRange => {
|
|
24
|
-
const to = dateScope?.to ?? new Date();
|
|
25
|
-
const from =
|
|
26
|
-
dateScope?.from ?? new Date(to.getTime() - DEFAULT_LOOKBACK_DAYS * DAY_MS);
|
|
27
|
-
return { from, to };
|
|
28
|
-
};
|
|
29
|
-
|
|
30
|
-
export const ticketStatsDateScopeFromYmd = (
|
|
31
|
-
startStr: string,
|
|
32
|
-
endStr: string,
|
|
33
|
-
timezone: string,
|
|
34
|
-
): TicketStatsDateScope => ({
|
|
35
|
-
from: startOfCalendarDayInTz(startStr, timezone),
|
|
36
|
-
to: endOfCalendarDayInTz(endStr, timezone),
|
|
37
|
-
});
|
|
38
|
-
|
|
39
|
-
export const findCallSidsWithTicketsByCity = async (
|
|
40
|
-
cityName: CityName,
|
|
41
|
-
dateScope?: TicketStatsDateScope,
|
|
42
|
-
): Promise<string[]> => {
|
|
43
|
-
const rows = await runTicketStatsAggregate<{ _id: unknown }>([
|
|
44
|
-
{ $match: ticketsWithCallSidMatch(cityName, resolveTicketStatsDateRange(dateScope)) },
|
|
45
|
-
{ $group: { _id: "$callSid" } },
|
|
46
|
-
]);
|
|
47
|
-
return rows.map((r) => String(r._id)).filter(Boolean);
|
|
48
|
-
};
|
|
49
|
-
|
|
50
|
-
export const findCallSidsWithDraftTicketsByCity = async (
|
|
51
|
-
cityName: CityName,
|
|
52
|
-
dateScope?: TicketStatsDateScope,
|
|
53
|
-
): Promise<string[]> => {
|
|
54
|
-
const rows = await runTicketStatsAggregate<{ _id: unknown }>([
|
|
55
|
-
{ $match: ticketsWithCallSidMatch(cityName, resolveTicketStatsDateRange(dateScope)) },
|
|
56
|
-
{ $match: { $expr: isDraftSubjectIdExpr } },
|
|
57
|
-
{ $group: { _id: "$callSid" } },
|
|
58
|
-
]);
|
|
59
|
-
return rows.map((r) => String(r._id)).filter(Boolean);
|
|
60
|
-
};
|
|
61
|
-
|
|
62
|
-
export const findSubjectsByCityAndDateRange = async (
|
|
63
|
-
cityName: CityName,
|
|
64
|
-
dateScope?: TicketStatsDateScope,
|
|
65
|
-
limit?: number,
|
|
66
|
-
): Promise<SubjectItem[]> => {
|
|
67
|
-
const rows = await runTicketStatsAggregate<{ _id: unknown; count: number }>([
|
|
68
|
-
{ $match: ticketsWithCallSidMatch(cityName, resolveTicketStatsDateRange(dateScope)) },
|
|
69
|
-
...subjectGroupStages(cityName, limit),
|
|
70
|
-
]);
|
|
71
|
-
return rows.map((r) => ({ subject: String(r._id), count: r.count }));
|
|
72
|
-
};
|
|
73
|
-
|
|
74
|
-
export const findSubjectsByCallSids = async (
|
|
75
|
-
cityName: CityName,
|
|
76
|
-
callSids: string[],
|
|
77
|
-
limit?: number,
|
|
78
|
-
dateScope?: TicketStatsDateScope,
|
|
79
|
-
): Promise<SubjectItem[]> => {
|
|
80
|
-
if (!callSids.length) return [];
|
|
81
|
-
|
|
82
|
-
const rows = await runTicketStatsAggregate<{ _id: unknown; count: number }>([
|
|
83
|
-
{
|
|
84
|
-
$match: ticketsByCallSidsMatch(
|
|
85
|
-
cityName,
|
|
86
|
-
callSids,
|
|
87
|
-
dateScope ? resolveTicketStatsDateRange(dateScope) : undefined,
|
|
88
|
-
),
|
|
89
|
-
},
|
|
90
|
-
...subjectGroupStages(cityName, limit),
|
|
91
|
-
]);
|
|
92
|
-
return rows.map((r) => ({ subject: String(r._id), count: r.count }));
|
|
93
|
-
};
|