@talkpilot/core-db 1.3.0 → 1.3.3

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 (149) hide show
  1. package/README.md +0 -30
  2. package/dist/connection.d.ts.map +1 -1
  3. package/dist/connection.js +10 -0
  4. package/dist/connection.js.map +1 -1
  5. package/dist/index.d.ts +3 -0
  6. package/dist/index.d.ts.map +1 -1
  7. package/dist/index.js +7 -1
  8. package/dist/index.js.map +1 -1
  9. package/dist/municipal/tickets/index.d.ts +1 -2
  10. package/dist/municipal/tickets/index.d.ts.map +1 -1
  11. package/dist/municipal/tickets/index.js +0 -1
  12. package/dist/municipal/tickets/index.js.map +1 -1
  13. package/dist/municipal/tickets/tickets.getters.d.ts +11 -0
  14. package/dist/municipal/tickets/tickets.getters.d.ts.map +1 -1
  15. package/dist/municipal/tickets/tickets.getters.js +128 -0
  16. package/dist/municipal/tickets/tickets.getters.js.map +1 -1
  17. package/dist/municipal/tickets/tickets.types.d.ts +5 -10
  18. package/dist/municipal/tickets/tickets.types.d.ts.map +1 -1
  19. package/dist/talkpilot/calls/calls.types.d.ts +2 -1
  20. package/dist/talkpilot/calls/calls.types.d.ts.map +1 -1
  21. package/dist/talkpilot/calls/calls.types.js +3 -0
  22. package/dist/talkpilot/calls/calls.types.js.map +1 -1
  23. package/dist/talkpilot/calls/dashboard/calls.dashboard.d.ts +1 -33
  24. package/dist/talkpilot/calls/dashboard/calls.dashboard.d.ts.map +1 -1
  25. package/dist/talkpilot/calls/dashboard/calls.dashboard.js +146 -131
  26. package/dist/talkpilot/calls/dashboard/calls.dashboard.js.map +1 -1
  27. package/dist/talkpilot/calls/dashboard/calls.dashboard.types.d.ts +6 -27
  28. package/dist/talkpilot/calls/dashboard/calls.dashboard.types.d.ts.map +1 -1
  29. package/dist/talkpilot/calls/index.d.ts +0 -3
  30. package/dist/talkpilot/calls/index.d.ts.map +1 -1
  31. package/dist/talkpilot/calls/index.js +0 -3
  32. package/dist/talkpilot/calls/index.js.map +1 -1
  33. package/dist/test-utils/db-utils.d.ts.map +1 -1
  34. package/dist/test-utils/db-utils.js +2 -0
  35. package/dist/test-utils/db-utils.js.map +1 -1
  36. package/dist/test-utils/factories/index.d.ts +1 -0
  37. package/dist/test-utils/factories/index.d.ts.map +1 -1
  38. package/dist/test-utils/factories/index.js +1 -0
  39. package/dist/test-utils/factories/index.js.map +1 -1
  40. package/dist/test-utils/factories/websitalk/scans.d.ts +5 -0
  41. package/dist/test-utils/factories/websitalk/scans.d.ts.map +1 -0
  42. package/dist/test-utils/factories/websitalk/scans.js +25 -0
  43. package/dist/test-utils/factories/websitalk/scans.js.map +1 -0
  44. package/dist/websitalk/index.d.ts +7 -0
  45. package/dist/websitalk/index.d.ts.map +1 -0
  46. package/dist/websitalk/index.js +34 -0
  47. package/dist/websitalk/index.js.map +1 -0
  48. package/dist/websitalk/mongodb-client.d.ts +13 -0
  49. package/dist/websitalk/mongodb-client.d.ts.map +1 -0
  50. package/dist/websitalk/mongodb-client.js +56 -0
  51. package/dist/websitalk/mongodb-client.js.map +1 -0
  52. package/dist/websitalk/scans/index.d.ts +3 -0
  53. package/dist/websitalk/scans/index.d.ts.map +1 -0
  54. package/dist/websitalk/scans/index.js +19 -0
  55. package/dist/websitalk/scans/index.js.map +1 -0
  56. package/dist/websitalk/scans/scans.getters.d.ts +12 -0
  57. package/dist/websitalk/scans/scans.getters.d.ts.map +1 -0
  58. package/dist/websitalk/scans/scans.getters.js +74 -0
  59. package/dist/websitalk/scans/scans.getters.js.map +1 -0
  60. package/dist/websitalk/scans/scans.types.d.ts +45 -0
  61. package/dist/websitalk/scans/scans.types.d.ts.map +1 -0
  62. package/dist/{talkpilot/calls/calls.statistics.types.js → websitalk/scans/scans.types.js} +1 -1
  63. package/dist/websitalk/scans/scans.types.js.map +1 -0
  64. package/package.json +1 -1
  65. package/src/connection.ts +12 -0
  66. package/src/index.ts +9 -0
  67. package/src/municipal/tickets/__tests__/tickets.getters.spec.ts +37 -1
  68. package/src/municipal/tickets/index.ts +1 -2
  69. package/src/municipal/tickets/tickets.getters.ts +140 -0
  70. package/src/municipal/tickets/tickets.types.ts +9 -14
  71. package/src/talkpilot/calls/__tests__/calls.dashboard.spec.ts +111 -8
  72. package/src/talkpilot/calls/calls.types.ts +2 -4
  73. package/src/talkpilot/calls/dashboard/calls.dashboard.ts +197 -148
  74. package/src/talkpilot/calls/dashboard/calls.dashboard.types.ts +12 -25
  75. package/src/talkpilot/calls/index.ts +0 -3
  76. package/src/talkpilot/clientsConfig/__tests__/clientsConfig.spec.ts +0 -7
  77. package/src/test-utils/db-utils.ts +3 -1
  78. package/src/test-utils/factories/index.ts +1 -0
  79. package/src/test-utils/factories/websitalk/scans.ts +23 -0
  80. package/src/websitalk/index.ts +15 -0
  81. package/src/websitalk/mongodb-client.ts +61 -0
  82. package/src/websitalk/scans/__tests__/scans.spec.ts +218 -0
  83. package/src/websitalk/scans/index.ts +2 -0
  84. package/src/websitalk/scans/scans.getters.ts +113 -0
  85. package/src/websitalk/scans/scans.types.ts +53 -0
  86. package/dist/municipal/tickets/tickets.constants.d.ts +0 -7
  87. package/dist/municipal/tickets/tickets.constants.d.ts.map +0 -1
  88. package/dist/municipal/tickets/tickets.constants.js +0 -10
  89. package/dist/municipal/tickets/tickets.constants.js.map +0 -1
  90. package/dist/municipal/tickets/tickets.deprecated.getters.d.ts +0 -12
  91. package/dist/municipal/tickets/tickets.deprecated.getters.d.ts.map +0 -1
  92. package/dist/municipal/tickets/tickets.deprecated.getters.js +0 -131
  93. package/dist/municipal/tickets/tickets.deprecated.getters.js.map +0 -1
  94. package/dist/municipal/tickets/tickets.statistics.aggregation.d.ts +0 -45
  95. package/dist/municipal/tickets/tickets.statistics.aggregation.d.ts.map +0 -1
  96. package/dist/municipal/tickets/tickets.statistics.aggregation.js +0 -98
  97. package/dist/municipal/tickets/tickets.statistics.aggregation.js.map +0 -1
  98. package/dist/municipal/tickets/tickets.statistics.dates.d.ts +0 -7
  99. package/dist/municipal/tickets/tickets.statistics.dates.d.ts.map +0 -1
  100. package/dist/municipal/tickets/tickets.statistics.dates.js +0 -40
  101. package/dist/municipal/tickets/tickets.statistics.dates.js.map +0 -1
  102. package/dist/municipal/tickets/tickets.statistics.getters.d.ts +0 -9
  103. package/dist/municipal/tickets/tickets.statistics.getters.d.ts.map +0 -1
  104. package/dist/municipal/tickets/tickets.statistics.getters.js +0 -55
  105. package/dist/municipal/tickets/tickets.statistics.getters.js.map +0 -1
  106. package/dist/municipal/tickets/tickets.statistics.pipeline.d.ts +0 -53
  107. package/dist/municipal/tickets/tickets.statistics.pipeline.d.ts.map +0 -1
  108. package/dist/municipal/tickets/tickets.statistics.pipeline.js +0 -112
  109. package/dist/municipal/tickets/tickets.statistics.pipeline.js.map +0 -1
  110. package/dist/municipal/tickets/tickets.statistics.utils.d.ts +0 -7
  111. package/dist/municipal/tickets/tickets.statistics.utils.d.ts.map +0 -1
  112. package/dist/municipal/tickets/tickets.statistics.utils.js +0 -40
  113. package/dist/municipal/tickets/tickets.statistics.utils.js.map +0 -1
  114. package/dist/talkpilot/calls/calls.constants.d.ts +0 -17
  115. package/dist/talkpilot/calls/calls.constants.d.ts.map +0 -1
  116. package/dist/talkpilot/calls/calls.constants.js +0 -20
  117. package/dist/talkpilot/calls/calls.constants.js.map +0 -1
  118. package/dist/talkpilot/calls/calls.statistics.getters.d.ts +0 -19
  119. package/dist/talkpilot/calls/calls.statistics.getters.d.ts.map +0 -1
  120. package/dist/talkpilot/calls/calls.statistics.getters.js +0 -375
  121. package/dist/talkpilot/calls/calls.statistics.getters.js.map +0 -1
  122. package/dist/talkpilot/calls/calls.statistics.ticketScope.d.ts +0 -12
  123. package/dist/talkpilot/calls/calls.statistics.ticketScope.d.ts.map +0 -1
  124. package/dist/talkpilot/calls/calls.statistics.ticketScope.js +0 -37
  125. package/dist/talkpilot/calls/calls.statistics.ticketScope.js.map +0 -1
  126. package/dist/talkpilot/calls/calls.statistics.tickets.d.ts +0 -17
  127. package/dist/talkpilot/calls/calls.statistics.tickets.d.ts.map +0 -1
  128. package/dist/talkpilot/calls/calls.statistics.tickets.js +0 -33
  129. package/dist/talkpilot/calls/calls.statistics.tickets.js.map +0 -1
  130. package/dist/talkpilot/calls/calls.statistics.types.d.ts +0 -39
  131. package/dist/talkpilot/calls/calls.statistics.types.d.ts.map +0 -1
  132. package/dist/talkpilot/calls/calls.statistics.types.js.map +0 -1
  133. package/dist/utils/date.utils.d.ts +0 -49
  134. package/dist/utils/date.utils.d.ts.map +0 -1
  135. package/dist/utils/date.utils.js +0 -103
  136. package/dist/utils/date.utils.js.map +0 -1
  137. package/dist/utils/statistics.aggregation.d.ts +0 -20
  138. package/dist/utils/statistics.aggregation.d.ts.map +0 -1
  139. package/dist/utils/statistics.aggregation.js +0 -43
  140. package/dist/utils/statistics.aggregation.js.map +0 -1
  141. package/src/municipal/tickets/__tests__/tickets.statistics.spec.ts +0 -104
  142. package/src/municipal/tickets/tickets.constants.ts +0 -8
  143. package/src/municipal/tickets/tickets.statistics.aggregation.ts +0 -113
  144. package/src/municipal/tickets/tickets.statistics.getters.ts +0 -93
  145. package/src/talkpilot/calls/__tests__/calls.statistics.spec.ts +0 -281
  146. package/src/talkpilot/calls/calls.constants.ts +0 -20
  147. package/src/talkpilot/calls/calls.statistics.getters.ts +0 -525
  148. package/src/talkpilot/calls/calls.statistics.types.ts +0 -44
  149. package/src/utils/date.utils.ts +0 -116
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/websitalk/index.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;AAAA,qCAAwD;AAExD,0CAAwB;AACxB,mDAA0D;AAAjD,wHAAA,sBAAsB,OAAA;AAE/B,IAAI,EAAM,CAAC;AACJ,MAAM,KAAK,GAAG,CAAC,CAAK,EAAE,EAAE;IAC7B,EAAE,GAAG,CAAC,CAAC;AACT,CAAC,CAAC;AAFW,QAAA,KAAK,SAEhB;AAEK,MAAM,KAAK,GAAG,GAAO,EAAE;IAC5B,IAAI,CAAC,EAAE;QAAE,MAAM,IAAI,KAAK,CAAC,iCAAiC,CAAC,CAAC;IAC5D,OAAO,EAAE,CAAC;AACZ,CAAC,CAAC;AAHW,QAAA,KAAK,SAGhB;AACW,QAAA,QAAQ,GAAG,kBAAa,CAAC"}
@@ -0,0 +1,13 @@
1
+ import { Db } from "mongodb";
2
+ declare class WebsitalkMongoDBClient {
3
+ private client;
4
+ private db;
5
+ private readonly defaultDbName;
6
+ connect(uri?: string, dbName?: string): Promise<void>;
7
+ disconnect(): Promise<void>;
8
+ getDb(): Db;
9
+ isConnected(): boolean;
10
+ }
11
+ export declare const websitalkMongodbClient: WebsitalkMongoDBClient;
12
+ export {};
13
+ //# sourceMappingURL=mongodb-client.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mongodb-client.d.ts","sourceRoot":"","sources":["../../src/websitalk/mongodb-client.ts"],"names":[],"mappings":"AAAA,OAAO,EAAe,EAAE,EAAE,MAAM,SAAS,CAAC;AAI1C,cAAM,sBAAsB;IAC1B,OAAO,CAAC,MAAM,CAA4B;IAC1C,OAAO,CAAC,EAAE,CAAmB;IAC7B,OAAO,CAAC,QAAQ,CAAC,aAAa,CAAkB;IAE1C,OAAO,CAAC,GAAG,CAAC,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAuBrD,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IAcjC,KAAK,IAAI,EAAE;IASX,WAAW,IAAI,OAAO;CAGvB;AAED,eAAO,MAAM,sBAAsB,wBAA+B,CAAC"}
@@ -0,0 +1,56 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.websitalkMongodbClient = void 0;
4
+ const mongodb_1 = require("mongodb");
5
+ const index_1 = require("./index");
6
+ const validation_1 = require("../utils/validation");
7
+ class WebsitalkMongoDBClient {
8
+ client = null;
9
+ db = null;
10
+ defaultDbName = "website-talk";
11
+ async connect(uri, dbName) {
12
+ if (this.client) {
13
+ return;
14
+ }
15
+ const mongodbUri = uri || process.env.MONGO_URI || process.env.MONGODB_URI;
16
+ (0, validation_1.validateConfig)("MONGO_URI", mongodbUri);
17
+ (0, validation_1.validateMongoUri)(mongodbUri);
18
+ try {
19
+ this.client = new mongodb_1.MongoClient(mongodbUri);
20
+ await this.client.connect();
21
+ const targetDbName = dbName || process.env.WEBSITALK_DB_NAME || this.defaultDbName;
22
+ this.db = this.client.db(targetDbName);
23
+ (0, index_1.setDb)(this.db);
24
+ console.info(`[core-db] Website Talk MongoDB connected: ${targetDbName}`);
25
+ }
26
+ catch (error) {
27
+ console.error("[core-db] Website Talk connection failed", error);
28
+ throw error;
29
+ }
30
+ }
31
+ async disconnect() {
32
+ if (this.client) {
33
+ try {
34
+ await this.client.close();
35
+ this.client = null;
36
+ this.db = null;
37
+ console.info("Website Talk MongoDB disconnected successfully");
38
+ }
39
+ catch (error) {
40
+ console.error("[core-db] Website Talk disconnection failed", error);
41
+ throw error;
42
+ }
43
+ }
44
+ }
45
+ getDb() {
46
+ if (!this.db) {
47
+ throw new Error("Website Talk database not initialized. Call connect() first.");
48
+ }
49
+ return this.db;
50
+ }
51
+ isConnected() {
52
+ return this.client !== null && this.client !== undefined;
53
+ }
54
+ }
55
+ exports.websitalkMongodbClient = new WebsitalkMongoDBClient();
56
+ //# sourceMappingURL=mongodb-client.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mongodb-client.js","sourceRoot":"","sources":["../../src/websitalk/mongodb-client.ts"],"names":[],"mappings":";;;AAAA,qCAA0C;AAC1C,mCAAgC;AAChC,oDAAuE;AAEvE,MAAM,sBAAsB;IAClB,MAAM,GAAuB,IAAI,CAAC;IAClC,EAAE,GAAc,IAAI,CAAC;IACZ,aAAa,GAAG,cAAc,CAAC;IAEhD,KAAK,CAAC,OAAO,CAAC,GAAY,EAAE,MAAe;QACzC,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YAChB,OAAO;QACT,CAAC;QAED,MAAM,UAAU,GAAG,GAAG,IAAI,OAAO,CAAC,GAAG,CAAC,SAAS,IAAI,OAAO,CAAC,GAAG,CAAC,WAAW,CAAC;QAC3E,IAAA,2BAAc,EAAC,WAAW,EAAE,UAAU,CAAC,CAAC;QACxC,IAAA,6BAAgB,EAAC,UAAW,CAAC,CAAC;QAE9B,IAAI,CAAC;YACH,IAAI,CAAC,MAAM,GAAG,IAAI,qBAAW,CAAC,UAAW,CAAC,CAAC;YAC3C,MAAM,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;YAC5B,MAAM,YAAY,GAChB,MAAM,IAAI,OAAO,CAAC,GAAG,CAAC,iBAAiB,IAAI,IAAI,CAAC,aAAa,CAAC;YAChE,IAAI,CAAC,EAAE,GAAG,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,YAAY,CAAC,CAAC;YACvC,IAAA,aAAK,EAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACf,OAAO,CAAC,IAAI,CAAC,6CAA6C,YAAY,EAAE,CAAC,CAAC;QAC5E,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,0CAA0C,EAAE,KAAK,CAAC,CAAC;YACjE,MAAM,KAAK,CAAC;QACd,CAAC;IACH,CAAC;IAED,KAAK,CAAC,UAAU;QACd,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YAChB,IAAI,CAAC;gBACH,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;gBAC1B,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC;gBACnB,IAAI,CAAC,EAAE,GAAG,IAAI,CAAC;gBACf,OAAO,CAAC,IAAI,CAAC,gDAAgD,CAAC,CAAC;YACjE,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,OAAO,CAAC,KAAK,CAAC,6CAA6C,EAAE,KAAK,CAAC,CAAC;gBACpE,MAAM,KAAK,CAAC;YACd,CAAC;QACH,CAAC;IACH,CAAC;IAED,KAAK;QACH,IAAI,CAAC,IAAI,CAAC,EAAE,EAAE,CAAC;YACb,MAAM,IAAI,KAAK,CACb,8DAA8D,CAC/D,CAAC;QACJ,CAAC;QACD,OAAO,IAAI,CAAC,EAAE,CAAC;IACjB,CAAC;IAED,WAAW;QACT,OAAO,IAAI,CAAC,MAAM,KAAK,IAAI,IAAI,IAAI,CAAC,MAAM,KAAK,SAAS,CAAC;IAC3D,CAAC;CACF;AAEY,QAAA,sBAAsB,GAAG,IAAI,sBAAsB,EAAE,CAAC"}
@@ -0,0 +1,3 @@
1
+ export * from "./scans.getters";
2
+ export * from "./scans.types";
3
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/websitalk/scans/index.ts"],"names":[],"mappings":"AAAA,cAAc,iBAAiB,CAAC;AAChC,cAAc,eAAe,CAAC"}
@@ -0,0 +1,19 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./scans.getters"), exports);
18
+ __exportStar(require("./scans.types"), exports);
19
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/websitalk/scans/index.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;AAAA,kDAAgC;AAChC,gDAA8B"}
@@ -0,0 +1,12 @@
1
+ import { Collection, Filter, ObjectId } from "mongodb";
2
+ import { QueryOptions } from "../../talkpilot/utils/query.utils";
3
+ import type { ActiveScanStatus, Scan, ScanDoc } from "./scans.types";
4
+ export declare const getScansCollection: () => Collection<Scan>;
5
+ export declare const getScansByClientAndDateRange: (clientId: string, startDate: Date, endDate: Date) => Promise<ScanDoc[]>;
6
+ export declare const getScanFieldsByClientAndDateRange: <K extends keyof Scan>(clientId: string, startDate: Date, endDate: Date, fields: K[]) => Promise<Pick<ScanDoc, "_id" | K>[]>;
7
+ export declare const createScanDoc: (scan: Pick<Scan, "clientId" | "baseUrl" | "notifyPhoneNumber">) => Promise<import("mongodb").InsertOneResult<Scan>>;
8
+ export declare const getActiveScanStatus: (clientId: string) => Promise<ActiveScanStatus>;
9
+ export declare const updateScanDoc: (scanId: ObjectId, updates: Partial<Omit<Scan, "clientId" | "createdAt">>) => Promise<ScanDoc | null>;
10
+ export declare const findScansByQuery: (query: Filter<Scan>, options?: QueryOptions) => Promise<ScanDoc[]>;
11
+ export declare const countScans: (query: Filter<Scan>) => Promise<number>;
12
+ //# sourceMappingURL=scans.getters.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scans.getters.d.ts","sourceRoot":"","sources":["../../../src/websitalk/scans/scans.getters.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AAEvD,OAAO,EAEL,YAAY,EACb,MAAM,mCAAmC,CAAC;AAC3C,OAAO,KAAK,EACV,gBAAgB,EAChB,IAAI,EACJ,OAAO,EAER,MAAM,eAAe,CAAC;AAEvB,eAAO,MAAM,kBAAkB,QAAO,UAAU,CAAC,IAAI,CAEpD,CAAC;AAEF,eAAO,MAAM,4BAA4B,GACvC,UAAU,MAAM,EAChB,WAAW,IAAI,EACf,SAAS,IAAI,KACZ,OAAO,CAAC,OAAO,EAAE,CAUnB,CAAC;AAEF,eAAO,MAAM,iCAAiC,GAAI,CAAC,SAAS,MAAM,IAAI,EACpE,UAAU,MAAM,EAChB,WAAW,IAAI,EACf,SAAS,IAAI,EACb,QAAQ,CAAC,EAAE,KACV,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,GAAG,CAAC,CAAC,EAAE,CAkBpC,CAAC;AAEF,eAAO,MAAM,aAAa,GACxB,MAAM,IAAI,CAAC,IAAI,EAAE,UAAU,GAAG,SAAS,GAAG,mBAAmB,CAAC,qDAa/D,CAAC;AAIF,eAAO,MAAM,mBAAmB,GAC9B,UAAU,MAAM,KACf,OAAO,CAAC,gBAAgB,CAW1B,CAAC;AAEF,eAAO,MAAM,aAAa,GACxB,QAAQ,QAAQ,EAChB,SAAS,OAAO,CAAC,IAAI,CAAC,IAAI,EAAE,UAAU,GAAG,WAAW,CAAC,CAAC,KACrD,OAAO,CAAC,OAAO,GAAG,IAAI,CAMxB,CAAC;AAEF,eAAO,MAAM,gBAAgB,GAC3B,OAAO,MAAM,CAAC,IAAI,CAAC,EACnB,UAAU,YAAY,KACrB,OAAO,CAAC,OAAO,EAAE,CAGnB,CAAC;AAEF,eAAO,MAAM,UAAU,GAAU,OAAO,MAAM,CAAC,IAAI,CAAC,KAAG,OAAO,CAAC,MAAM,CAEpE,CAAC"}
@@ -0,0 +1,74 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.countScans = exports.findScansByQuery = exports.updateScanDoc = exports.getActiveScanStatus = exports.createScanDoc = exports.getScanFieldsByClientAndDateRange = exports.getScansByClientAndDateRange = exports.getScansCollection = void 0;
4
+ const index_1 = require("../index");
5
+ const query_utils_1 = require("../../talkpilot/utils/query.utils");
6
+ const getScansCollection = () => {
7
+ return (0, index_1.getDb)().collection("scans");
8
+ };
9
+ exports.getScansCollection = getScansCollection;
10
+ const getScansByClientAndDateRange = (clientId, startDate, endDate) => {
11
+ return (0, exports.getScansCollection)()
12
+ .find({
13
+ clientId,
14
+ createdAt: {
15
+ $gte: startDate,
16
+ $lte: endDate,
17
+ },
18
+ })
19
+ .toArray();
20
+ };
21
+ exports.getScansByClientAndDateRange = getScansByClientAndDateRange;
22
+ const getScanFieldsByClientAndDateRange = (clientId, startDate, endDate, fields) => {
23
+ const projection = fields.reduce((acc, field) => {
24
+ acc[field] = 1;
25
+ return acc;
26
+ }, {});
27
+ return (0, exports.getScansCollection)()
28
+ .find({
29
+ clientId,
30
+ createdAt: {
31
+ $gte: startDate,
32
+ $lte: endDate,
33
+ },
34
+ }, { projection })
35
+ .toArray();
36
+ };
37
+ exports.getScanFieldsByClientAndDateRange = getScanFieldsByClientAndDateRange;
38
+ const createScanDoc = (scan) => {
39
+ return (0, exports.getScansCollection)().insertOne({
40
+ ...scan,
41
+ status: "CHECKING",
42
+ activeTasksCount: 0,
43
+ pagesScanned: 0,
44
+ pagesSkipped: 0,
45
+ skipped: [],
46
+ siteChrome: {},
47
+ createdAt: new Date(),
48
+ updatedAt: new Date(),
49
+ });
50
+ };
51
+ exports.createScanDoc = createScanDoc;
52
+ const FINISHED_SCAN_STATUSES = ["COMPLETED", "FAILED"];
53
+ const getActiveScanStatus = async (clientId) => {
54
+ const latestScan = await (0, exports.getScansCollection)().findOne({ clientId }, { sort: { createdAt: -1 } });
55
+ if (!latestScan || FINISHED_SCAN_STATUSES.includes(latestScan.status)) {
56
+ return { isActive: false };
57
+ }
58
+ return { isActive: true, status: latestScan.status };
59
+ };
60
+ exports.getActiveScanStatus = getActiveScanStatus;
61
+ const updateScanDoc = async (scanId, updates) => {
62
+ return await (0, exports.getScansCollection)().findOneAndUpdate({ _id: scanId }, { $set: { ...updates, updatedAt: new Date() } }, { returnDocument: "after" });
63
+ };
64
+ exports.updateScanDoc = updateScanDoc;
65
+ const findScansByQuery = async (query, options) => {
66
+ const cursor = (0, exports.getScansCollection)().find(query);
67
+ return await (0, query_utils_1.applyQueryOptions)(cursor, options).toArray();
68
+ };
69
+ exports.findScansByQuery = findScansByQuery;
70
+ const countScans = async (query) => {
71
+ return (0, exports.getScansCollection)().countDocuments(query);
72
+ };
73
+ exports.countScans = countScans;
74
+ //# sourceMappingURL=scans.getters.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scans.getters.js","sourceRoot":"","sources":["../../../src/websitalk/scans/scans.getters.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,45 @@
1
+ import { WithId } from "mongodb";
2
+ export type ScanStatus = "CHECKING" | "PREPARING" | "SCANNING" | "PROCESSING" | "COMPLETED" | "FAILED";
3
+ export type ActiveScanStatus = {
4
+ isActive: true;
5
+ status: ScanStatus;
6
+ } | {
7
+ isActive: false;
8
+ };
9
+ export type SiteChromeEntry = {
10
+ url: string;
11
+ urlCanonical?: string | null;
12
+ title?: string | null;
13
+ h1?: string | null;
14
+ language?: string | null;
15
+ sourceType?: string | null;
16
+ status?: string | null;
17
+ context?: string | null;
18
+ contentHash?: string | null;
19
+ errorMessage?: string | null;
20
+ breadcrumbs?: unknown;
21
+ description?: string | null;
22
+ sourceUrl?: string | null;
23
+ fetchedAt?: Date | null;
24
+ createdAt?: Date;
25
+ updatedAt?: Date;
26
+ };
27
+ export type SkippedEntry = {
28
+ path: string;
29
+ reason: string;
30
+ };
31
+ export type Scan = {
32
+ clientId: string;
33
+ baseUrl: string;
34
+ notifyPhoneNumber?: string | null;
35
+ status: ScanStatus;
36
+ activeTasksCount: number;
37
+ pagesScanned: number;
38
+ pagesSkipped: number;
39
+ skipped: SkippedEntry[];
40
+ siteChrome: SiteChromeEntry;
41
+ createdAt: Date;
42
+ updatedAt: Date;
43
+ };
44
+ export type ScanDoc = WithId<Scan>;
45
+ //# sourceMappingURL=scans.types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scans.types.d.ts","sourceRoot":"","sources":["../../../src/websitalk/scans/scans.types.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAEjC,MAAM,MAAM,UAAU,GAClB,UAAU,GACV,WAAW,GACX,UAAU,GACV,YAAY,GACZ,WAAW,GACX,QAAQ,CAAC;AAEb,MAAM,MAAM,gBAAgB,GACxB;IAAE,QAAQ,EAAE,IAAI,CAAC;IAAC,MAAM,EAAE,UAAU,CAAA;CAAE,GACtC;IAAE,QAAQ,EAAE,KAAK,CAAA;CAAE,CAAC;AAExB,MAAM,MAAM,eAAe,GAAG;IAC5B,GAAG,EAAE,MAAM,CAAC;IACZ,YAAY,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC7B,KAAK,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IACtB,EAAE,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IACnB,QAAQ,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IACzB,UAAU,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC3B,MAAM,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IACvB,OAAO,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IACxB,WAAW,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC5B,YAAY,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC7B,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB,WAAW,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC5B,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,SAAS,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC;IACxB,SAAS,CAAC,EAAE,IAAI,CAAC;IACjB,SAAS,CAAC,EAAE,IAAI,CAAC;CAClB,CAAC;AAEF,MAAM,MAAM,YAAY,GAAG;IACzB,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;CAChB,CAAC;AAEF,MAAM,MAAM,IAAI,GAAG;IACjB,QAAQ,EAAE,MAAM,CAAC;IACjB,OAAO,EAAE,MAAM,CAAC;IAChB,iBAAiB,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAClC,MAAM,EAAE,UAAU,CAAC;IACnB,gBAAgB,EAAE,MAAM,CAAC;IACzB,YAAY,EAAE,MAAM,CAAC;IACrB,YAAY,EAAE,MAAM,CAAC;IACrB,OAAO,EAAE,YAAY,EAAE,CAAC;IACxB,UAAU,EAAE,eAAe,CAAC;IAC5B,SAAS,EAAE,IAAI,CAAC;IAChB,SAAS,EAAE,IAAI,CAAC;CACjB,CAAC;AAEF,MAAM,MAAM,OAAO,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC"}
@@ -1,3 +1,3 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- //# sourceMappingURL=calls.statistics.types.js.map
3
+ //# sourceMappingURL=scans.types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scans.types.js","sourceRoot":"","sources":["../../../src/websitalk/scans/scans.types.ts"],"names":[],"mappings":""}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@talkpilot/core-db",
3
- "version": "1.3.0",
3
+ "version": "1.3.3",
4
4
  "description": "Core database package for centralized connections and ORM integration.",
5
5
  "main": "dist/index.js",
6
6
  "types": "dist/index.d.ts",
package/src/connection.ts CHANGED
@@ -1,5 +1,6 @@
1
1
  import { mongodbClient } from "./talkpilot";
2
2
  import { municipalDataMongodbClient } from "./municipal";
3
+ import { websitalkMongodbClient } from "./websitalk";
3
4
 
4
5
  let isConnected = false;
5
6
 
@@ -15,6 +16,7 @@ export async function ensureDbConnected() {
15
16
  const mongoUri = process.env.MONGO_URI || process.env.MONGODB_URI;
16
17
  const talkpilotUri = mongoUri;
17
18
  const municipalUri = mongoUri;
19
+ const websiteTalkUri = mongoUri;
18
20
 
19
21
  if (!talkpilotUri) {
20
22
  console.warn("[core-db] MONGO_URI is not set");
@@ -38,5 +40,15 @@ export async function ensureDbConnected() {
38
40
  );
39
41
  }
40
42
 
43
+ if (!websiteTalkUri) {
44
+ console.warn("[core-db] MONGO_URI is not set");
45
+ } else {
46
+ const dbName = process.env.WEBSITALK_DB_NAME;
47
+ await websitalkMongodbClient.connect(websiteTalkUri, dbName);
48
+ console.info(
49
+ `[core-db] Connected to Website Talk MongoDB: ${dbName || "website-talk (default)"}`,
50
+ );
51
+ }
52
+
41
53
  isConnected = true;
42
54
  }
package/src/index.ts CHANGED
@@ -14,3 +14,12 @@ export * from "./municipal/departmentsSubjects";
14
14
  export * from "./municipal/tickets";
15
15
  export * from "./municipal/systemInstructions";
16
16
  export * from "./municipal/utils/types";
17
+
18
+ export {
19
+ websitalkMongodbClient,
20
+ getDb as getWebsitalkDb,
21
+ setDb as setWebsitalkDb,
22
+ ObjectId as WebsitalkObjectId,
23
+ } from "./websitalk";
24
+ export * from "./websitalk/scans/scans.getters";
25
+ export type * from "./websitalk/scans/scans.types";
@@ -1,4 +1,7 @@
1
- import { findTicketsByCallSid } from "../tickets.getters";
1
+ import {
2
+ findTicketsByCallSid,
3
+ getTicketsCountByCityAndDateRange,
4
+ } from "../tickets.getters";
2
5
  import { setDb } from "../../index";
3
6
  import { Db, Collection } from "mongodb";
4
7
  import { createTicket } from "../../../test-utils/factories";
@@ -27,4 +30,37 @@ describe("tickets getters", () => {
27
30
  expect(mockCollection.find).toHaveBeenCalledWith({ callSid: "123" });
28
31
  expect(result[0].callSid).toBe("123");
29
32
  });
33
+
34
+ it("getTicketsCountByCityAndDateRange should not count tickets without callSid", async () => {
35
+ const aggregateNext = jest.fn().mockResolvedValue({ n: 1 });
36
+ mockCollection = {
37
+ ...mockCollection,
38
+ aggregate: jest.fn().mockReturnValue({
39
+ next: aggregateNext,
40
+ }),
41
+ };
42
+ mockDb = {
43
+ collection: jest.fn().mockReturnValue(mockCollection),
44
+ };
45
+ setDb(mockDb as Db);
46
+
47
+ const result = await getTicketsCountByCityAndDateRange(
48
+ "ashdod",
49
+ "2026-01-01",
50
+ "2026-01-31",
51
+ "Asia/Jerusalem",
52
+ );
53
+
54
+ expect(mockCollection.aggregate).toHaveBeenCalledWith(
55
+ expect.arrayContaining([
56
+ {
57
+ $match: {
58
+ cityName: "ashdod",
59
+ callSid: { $exists: true, $nin: [null, ""] },
60
+ },
61
+ },
62
+ ]),
63
+ );
64
+ expect(result).toBe(1);
65
+ });
30
66
  });
@@ -1,3 +1,2 @@
1
1
  export * from "./tickets.getters";
2
- export * from "./tickets.statistics.getters";
3
- export type { Ticket, TicketDoc, SubjectItem, TicketStatsDateScope } from "./tickets.types";
2
+ export type { Ticket, TicketDoc, SubjectStatsItem } from "./tickets.types";
@@ -1,4 +1,5 @@
1
1
  import { CityName, getDb, ObjectId, Ticket } from "../index";
2
+ import type { SubjectStatsItem } from "./tickets.types";
2
3
  import { Collection, Filter, ObjectId as MongoObjectId } from "mongodb";
3
4
 
4
5
  /**
@@ -67,6 +68,145 @@ export const deleteTicket = async (ticketId: string): Promise<boolean> => {
67
68
  return result.deletedCount > 0;
68
69
  };
69
70
 
71
+ /**
72
+ * Count tickets by city and date range (createdAt converted to given timezone).
73
+ * Used as "open" tickets count when status is not available.
74
+ */
75
+ export async function getTicketsCountByCityAndDateRange(
76
+ cityName: string,
77
+ startStr: string,
78
+ endStr: string,
79
+ timezone: string,
80
+ ): Promise<number> {
81
+ const doc = await getTicketsCollection()
82
+ .aggregate<{ n: number }>([
83
+ { $match: { cityName, callSid: { $exists: true, $nin: [null, ""] } } },
84
+ {
85
+ $addFields: {
86
+ dateLocal: {
87
+ $dateToString: { format: "%Y-%m-%d", date: "$createdAt", timezone },
88
+ },
89
+ },
90
+ },
91
+ { $match: { dateLocal: { $gte: startStr, $lte: endStr } } },
92
+ { $count: "n" },
93
+ ])
94
+ .next();
95
+ return doc?.n ?? 0;
96
+ }
97
+
98
+ /**
99
+ * Count of tickets by department (subject) from departmentsSubjects. Different departments (subject_id) are grouped, and sub-subjects are unified under their department.
100
+ * Date filter: createdAt in [startStr, endStr] (in the given timezone). Fallback to "Unclassified" when no match is found in the lookup.
101
+ */
102
+ export async function getTicketsSubjectStats(
103
+ cityName: string,
104
+ startStr: string,
105
+ endStr: string,
106
+ timezone: string,
107
+ ): Promise<SubjectStatsItem[]> {
108
+ const coll = getTicketsCollection();
109
+ const rows = await coll
110
+ .aggregate<{ _id: string; subject: string; count: number }>([
111
+ { $match: { cityName } },
112
+ {
113
+ $addFields: {
114
+ dateLocal: {
115
+ $dateToString: { format: "%Y-%m-%d", date: "$createdAt", timezone },
116
+ },
117
+ },
118
+ },
119
+ { $match: { dateLocal: { $gte: startStr, $lte: endStr } } },
120
+ {
121
+ $addFields: {
122
+ effectiveSubjectId: {
123
+ $ifNull: [
124
+ "$externalCallFields.event_subject_id",
125
+ {
126
+ $ifNull: [
127
+ "$externalCallFields.event_sub_subject_id",
128
+ "$externalCallFields.event_sub_subject_id2",
129
+ ],
130
+ },
131
+ ],
132
+ },
133
+ },
134
+ },
135
+ {
136
+ $lookup: {
137
+ from: "departmentsSubjects",
138
+ let: { sid: "$effectiveSubjectId", c: cityName },
139
+ pipeline: [
140
+ {
141
+ $match: {
142
+ $expr: {
143
+ $and: [
144
+ { $eq: ["$cityName", "$$c"] },
145
+ {
146
+ $or: [
147
+ { $eq: ["$subject_id", { $ifNull: ["$$sid", ""] }] },
148
+ {
149
+ $eq: ["$sub_subject_id", { $ifNull: ["$$sid", ""] }],
150
+ },
151
+ {
152
+ $eq: ["$sub_subject_id2", { $ifNull: ["$$sid", ""] }],
153
+ },
154
+ ],
155
+ },
156
+ ],
157
+ },
158
+ },
159
+ },
160
+ { $limit: 1 },
161
+ ],
162
+ as: "subj",
163
+ },
164
+ },
165
+ {
166
+ $addFields: {
167
+ subject_id: {
168
+ $cond: {
169
+ if: { $gt: [{ $size: "$subj" }, 0] },
170
+ then: {
171
+ $ifNull: [{ $arrayElemAt: ["$subj.subject_id", 0] }, ""],
172
+ },
173
+ else: "",
174
+ },
175
+ },
176
+ subject: {
177
+ $cond: {
178
+ if: { $gt: [{ $size: "$subj" }, 0] },
179
+ then: {
180
+ $ifNull: [
181
+ { $arrayElemAt: ["$subj.subjectName", 0] },
182
+ "Unclassified",
183
+ ],
184
+ },
185
+ else: "Unclassified",
186
+ },
187
+ },
188
+ },
189
+ },
190
+ {
191
+ $group: {
192
+ _id: "$subject_id",
193
+ subject: { $first: "$subject" },
194
+ count: { $sum: 1 },
195
+ },
196
+ },
197
+ { $sort: { count: -1 } },
198
+ ])
199
+ .toArray();
200
+
201
+ const total = rows.reduce((s: number, r: any) => s + r.count, 0);
202
+ return rows.map((r: any) => ({
203
+ subject_name: r.subject,
204
+ subject_id: r._id,
205
+ count: r.count,
206
+ percentage: total > 0 ? Math.round((r.count / total) * 100) : 0,
207
+ }));
208
+ }
209
+
70
210
  export const findTicketByQuery = async (query: Partial<Ticket>) => {
71
211
  return await getTicketsCollection().findOne(query);
72
212
  };
@@ -9,6 +9,7 @@ export type Ticket = {
9
9
  callSid?: string;
10
10
  cityName: CityName;
11
11
  externalCallFields: {
12
+ // Request fields
12
13
  first_name?: string;
13
14
  last_name?: string;
14
15
  event_description?: string;
@@ -17,9 +18,10 @@ export type Ticket = {
17
18
  event_subject_id?: string;
18
19
  event_sub_subject_id?: string;
19
20
  event_sub_subject_id2?: string | null;
21
+ // Response fields (only if external call was successful)
20
22
  call_number?: string;
21
- status?: number;
22
- error?: string;
23
+ status?: number; // Response status from Bina API (1 = success)
24
+ error?: string; // Error message from Bina API if any
23
25
  image1?: string;
24
26
  image2?: string;
25
27
  image3?: string;
@@ -32,17 +34,10 @@ export type Ticket = {
32
34
 
33
35
  export type TicketDoc = WithId<Ticket>;
34
36
 
35
- export type SubjectItem = {
36
- subject: string;
37
+ /** Call count by department (subject). */
38
+ export type SubjectStatsItem = {
39
+ subject_name: string;
40
+ subject_id: string;
37
41
  count: number;
38
- };
39
-
40
- export type TicketStatsDateScope = {
41
- from?: Date;
42
- to?: Date;
43
- };
44
-
45
- export type TicketStatsDateRange = {
46
- from: Date;
47
- to: Date;
42
+ percentage: number;
48
43
  };