payservedb 7.9.3 → 7.9.5

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 (160) hide show
  1. package/.env +2 -2
  2. package/.idea/material_theme_project_new.xml +11 -11
  3. package/.idea/modules.xml +7 -7
  4. package/.idea/psdb.iml +11 -11
  5. package/.idea/vcs.xml +5 -5
  6. package/index.js +287 -289
  7. package/package.json +17 -17
  8. package/src/models/account.js +35 -35
  9. package/src/models/apilog.js +18 -18
  10. package/src/models/approvalsWorkflows.js +49 -49
  11. package/src/models/archivedapilog.js +18 -18
  12. package/src/models/asset.js +92 -92
  13. package/src/models/assetsAssignment.js +64 -64
  14. package/src/models/auditTrail.js +346 -346
  15. package/src/models/bankdetails.js +43 -43
  16. package/src/models/billerAddress.js +124 -124
  17. package/src/models/booking_invoice.js +151 -151
  18. package/src/models/bookinganalytics.js +63 -63
  19. package/src/models/bookingconfig.js +45 -45
  20. package/src/models/bookingproperty.js +122 -122
  21. package/src/models/bookingreservation.js +192 -192
  22. package/src/models/bookingrevenuerecord.js +84 -84
  23. package/src/models/budget.js +95 -95
  24. package/src/models/budgetCategory.js +19 -19
  25. package/src/models/campaigns.js +94 -94
  26. package/src/models/cashpayment.js +264 -264
  27. package/src/models/combinedUnits.js +62 -62
  28. package/src/models/common_area_electricity.js +38 -38
  29. package/src/models/common_area_generator.js +41 -41
  30. package/src/models/common_area_utility_alert.js +37 -37
  31. package/src/models/common_area_water.js +39 -39
  32. package/src/models/communication_status.js +33 -33
  33. package/src/models/company.js +53 -53
  34. package/src/models/coreBaseSettings.js +16 -16
  35. package/src/models/coreInvoiceSettings.js +100 -100
  36. package/src/models/country_tax.js +42 -42
  37. package/src/models/currency_settings.js +39 -39
  38. package/src/models/customer.js +208 -208
  39. package/src/models/dailyChecklist.js +312 -312
  40. package/src/models/default_payment_details.js +17 -17
  41. package/src/models/deliveryTimeMarks.js +18 -18
  42. package/src/models/dutyRosterChecklist.js +250 -250
  43. package/src/models/dutyroster.js +136 -136
  44. package/src/models/email.js +37 -37
  45. package/src/models/email_sms_queue.js +61 -61
  46. package/src/models/entry_exit.js +53 -53
  47. package/src/models/expense.js +99 -99
  48. package/src/models/expense_category.js +45 -45
  49. package/src/models/facility.js +62 -62
  50. package/src/models/facilityInvoice.js +223 -223
  51. package/src/models/facilityInvoiceRecipient.js +32 -32
  52. package/src/models/facilityWalletTransactionsMetadata.js +236 -236
  53. package/src/models/facility_departements.js +20 -20
  54. package/src/models/facility_payment_details.js +20 -20
  55. package/src/models/facilityasset.js +25 -25
  56. package/src/models/faq.js +18 -18
  57. package/src/models/gl_account_double_entries.js +25 -25
  58. package/src/models/gl_accounts.js +56 -56
  59. package/src/models/gl_entries.js +49 -49
  60. package/src/models/goodsReceivedNotes.js +115 -115
  61. package/src/models/guard.js +47 -47
  62. package/src/models/handover.js +246 -246
  63. package/src/models/invoice.js +336 -336
  64. package/src/models/item_inspection.js +67 -67
  65. package/src/models/leaseagreement.js +226 -226
  66. package/src/models/leasetemplate.js +17 -17
  67. package/src/models/levy.js +206 -206
  68. package/src/models/levy_invoice_settings.js +26 -26
  69. package/src/models/levycontract.js +168 -168
  70. package/src/models/levytype.js +23 -23
  71. package/src/models/maintenance_service_vendor.js +38 -38
  72. package/src/models/maintenance_services.js +17 -17
  73. package/src/models/maintenancerequisition.js +31 -31
  74. package/src/models/master_workplan.js +32 -32
  75. package/src/models/master_workplan_child.js +34 -34
  76. package/src/models/message.js +38 -38
  77. package/src/models/module.js +21 -21
  78. package/src/models/notification.js +44 -44
  79. package/src/models/paymentTermsMarks.js +19 -19
  80. package/src/models/penalty.js +76 -76
  81. package/src/models/pendingCredentials.js +32 -32
  82. package/src/models/powerMeterCommunicationProtocol.js +17 -17
  83. package/src/models/powerMeterCustomerAccount.js +78 -78
  84. package/src/models/powerMeterCustomerBand.js +14 -14
  85. package/src/models/powerMeterDailyReading.js +30 -30
  86. package/src/models/powerMeterGateways.js +40 -40
  87. package/src/models/powerMeterMonthlyReading.js +34 -34
  88. package/src/models/powerMeterPowerCharges.js +85 -85
  89. package/src/models/powerMeterSettings.js +159 -159
  90. package/src/models/powerMeterSingleDayReading.js +32 -32
  91. package/src/models/powerMeters.js +116 -116
  92. package/src/models/powerMetersManufacturer.js +14 -14
  93. package/src/models/power_meter_account.js +81 -81
  94. package/src/models/power_meter_command_logs.js +30 -30
  95. package/src/models/power_meter_command_queue.js +33 -33
  96. package/src/models/power_meter_negative_balance.js +44 -44
  97. package/src/models/power_prepaid_credits.js +47 -47
  98. package/src/models/power_prepaid_debits.js +50 -50
  99. package/src/models/power_prepaid_orders.js +78 -78
  100. package/src/models/power_sms_notification.js +26 -26
  101. package/src/models/propertyManagerContract.js +556 -556
  102. package/src/models/propertyManagerRevenue.js +195 -195
  103. package/src/models/purchaseOrderInvoice.js +74 -74
  104. package/src/models/purchase_order.js +213 -213
  105. package/src/models/purchase_request.js +110 -110
  106. package/src/models/refresh_token.js +23 -23
  107. package/src/models/reminder.js +197 -197
  108. package/src/models/report.js +13 -13
  109. package/src/models/resident.js +121 -121
  110. package/src/models/rfq_details.js +131 -131
  111. package/src/models/rfq_response.js +153 -153
  112. package/src/models/service_charge_invoice_upload.js +42 -42
  113. package/src/models/service_charge_payments.js +27 -27
  114. package/src/models/servicerequest.js +55 -55
  115. package/src/models/settings.js +62 -62
  116. package/src/models/smart_meter_daily_consumption.js +44 -44
  117. package/src/models/sms_africastalking.js +20 -20
  118. package/src/models/sms_balance_notification.js +26 -26
  119. package/src/models/sms_meliora.js +20 -20
  120. package/src/models/staff.js +36 -36
  121. package/src/models/stocksandspare.js +161 -161
  122. package/src/models/suppliers.js +74 -74
  123. package/src/models/tickets.js +121 -121
  124. package/src/models/unitManagementTemplate.js +44 -44
  125. package/src/models/unitasset.js +25 -25
  126. package/src/models/units.js +117 -117
  127. package/src/models/user.js +186 -186
  128. package/src/models/valueaddedservices.js +79 -79
  129. package/src/models/vas_invoices_upload.js +50 -50
  130. package/src/models/vas_payments.js +24 -24
  131. package/src/models/vasinvoice.js +192 -192
  132. package/src/models/vasvendor.js +57 -57
  133. package/src/models/visitLog.js +95 -95
  134. package/src/models/visitor.js +67 -67
  135. package/src/models/waitlist.js +45 -45
  136. package/src/models/wallet.js +44 -44
  137. package/src/models/wallet_transactions.js +50 -50
  138. package/src/models/water_invoice.js +351 -351
  139. package/src/models/water_meter_Command_Queue.js +33 -33
  140. package/src/models/water_meter_account.js +82 -82
  141. package/src/models/water_meter_billing.js +58 -58
  142. package/src/models/water_meter_communication.js +17 -17
  143. package/src/models/water_meter_communication_logs.js +39 -39
  144. package/src/models/water_meter_concentrator.js +66 -66
  145. package/src/models/water_meter_daily_history.js +32 -32
  146. package/src/models/water_meter_high_risk.js +36 -36
  147. package/src/models/water_meter_iot_cards.js +34 -34
  148. package/src/models/water_meter_manufacturer.js +35 -35
  149. package/src/models/water_meter_monthly_history.js +36 -36
  150. package/src/models/water_meter_negative_amounts.js +44 -44
  151. package/src/models/water_meter_settings.js +276 -261
  152. package/src/models/water_meter_single_day_history.js +34 -34
  153. package/src/models/water_meter_size.js +15 -15
  154. package/src/models/water_meters.js +133 -133
  155. package/src/models/water_meters_delivery.js +76 -76
  156. package/src/models/water_prepaid_credit.js +47 -47
  157. package/src/models/water_prepaid_debit.js +50 -50
  158. package/src/models/workorder.js +49 -49
  159. package/src/models/facilityBillingPrices.js +0 -30
  160. package/src/models/facilityInvoicePayment.js +0 -47
@@ -1,346 +1,346 @@
1
- const mongoose = require("mongoose");
2
-
3
- // User details sub-schema (from JWT token)
4
- const userDetailsSchema = new mongoose.Schema(
5
- {
6
- userId: {
7
- type: String,
8
- required: true,
9
- },
10
- email: {
11
- type: String,
12
- required: true,
13
- },
14
- fullName: {
15
- type: String,
16
- required: true,
17
- },
18
- role: {
19
- type: String,
20
- required: true,
21
- },
22
- phoneNumber: {
23
- type: String,
24
- required: false,
25
- },
26
- facilityId: {
27
- type: String,
28
- required: false,
29
- },
30
- permissions: {
31
- type: mongoose.Schema.Types.Mixed,
32
- required: false,
33
- },
34
- tokenExpiry: {
35
- type: Date,
36
- required: false,
37
- },
38
- tokenIssued: {
39
- type: Date,
40
- required: false,
41
- },
42
- },
43
- { _id: false },
44
- );
45
-
46
- // Main audit trail schema
47
- const auditTrailSchema = new mongoose.Schema(
48
- {
49
- // Basic request information
50
- timestamp: {
51
- type: Date,
52
- required: true,
53
- default: Date.now,
54
- },
55
- method: {
56
- type: String,
57
- enum: ["GET", "POST", "PUT", "PATCH", "DELETE", "OPTIONS", "HEAD"],
58
- required: true,
59
- },
60
- url: {
61
- type: String,
62
- required: true,
63
- },
64
- ip_address: {
65
- type: String,
66
- required: true,
67
- },
68
- platform: {
69
- type: String,
70
- required: true,
71
- },
72
-
73
- // Authorization and user information
74
- is_authorized: {
75
- type: Boolean,
76
- required: true,
77
- default: false,
78
- },
79
- user_details: {
80
- type: userDetailsSchema,
81
- required: false,
82
- },
83
-
84
- // Device and browser information
85
- browser: {
86
- type: String,
87
- required: true,
88
- default: "unknown",
89
- },
90
- operating_system: {
91
- type: String,
92
- required: true,
93
- default: "unknown",
94
- },
95
- device_type: {
96
- type: String,
97
- enum: ["desktop", "mobile", "tablet", "unknown"],
98
- required: true,
99
- default: "unknown",
100
- },
101
- user_agent: {
102
- type: String,
103
- required: false,
104
- },
105
-
106
- // Request data
107
- request_data: {
108
- type: mongoose.Schema.Types.Mixed,
109
- required: false,
110
- },
111
- query_params: {
112
- type: mongoose.Schema.Types.Mixed,
113
- required: false,
114
- default: {},
115
- },
116
- route_params: {
117
- type: mongoose.Schema.Types.Mixed,
118
- required: false,
119
- default: {},
120
- },
121
-
122
- // Activity information
123
- activity: {
124
- type: String,
125
- required: true,
126
- },
127
-
128
- // response status
129
- response_status: {
130
- type: String,
131
- required: true,
132
- enum: ["success", "error", "unknown"],
133
- default: "unknown",
134
- },
135
-
136
- // Method-specific data
137
- previous_data: {
138
- type: mongoose.Schema.Types.Mixed,
139
- required: false,
140
- },
141
- deleted_data: {
142
- type: mongoose.Schema.Types.Mixed,
143
- required: false,
144
- },
145
-
146
- // Custom additional data
147
- custom_data: {
148
- type: mongoose.Schema.Types.Mixed,
149
- required: false,
150
- },
151
-
152
- // Request tracking
153
- request_id: {
154
- type: String,
155
- required: false,
156
- },
157
- },
158
- {
159
- timestamps: true,
160
- },
161
- );
162
-
163
- // Pre-save middleware for data validation and cleanup
164
- auditTrailSchema.pre("save", function (next) {
165
- // Ensure timestamp is set
166
- if (!this.timestamp) {
167
- this.timestamp = new Date();
168
- }
169
-
170
- // Validate that authorized requests have user details
171
- if (this.is_authorized && !this.user_details) {
172
- console.warn("Authorized request without user details detected");
173
- }
174
-
175
- // Ensure query_params and route_params are objects
176
- if (!this.query_params || typeof this.query_params !== "object") {
177
- this.query_params = {};
178
- }
179
- if (!this.route_params || typeof this.route_params !== "object") {
180
- this.route_params = {};
181
- }
182
-
183
- // Set browser to unknown if not provided
184
- if (!this.browser) {
185
- this.browser = "unknown";
186
- }
187
-
188
- // Set operating_system to unknown if not provided
189
- if (!this.operating_system) {
190
- this.operating_system = "unknown";
191
- }
192
-
193
- next();
194
- });
195
-
196
- // Static methods for common queries
197
- auditTrailSchema.statics.findByUser = function (userId, startDate, endDate) {
198
- const query = { "user_details.userId": userId };
199
- if (startDate || endDate) {
200
- query.timestamp = {};
201
- if (startDate) query.timestamp.$gte = startDate;
202
- if (endDate) query.timestamp.$lte = endDate;
203
- }
204
- return this.find(query).sort({ timestamp: -1 });
205
- };
206
-
207
- auditTrailSchema.statics.findByActivity = function (activity, limit = 100) {
208
- return this.find({ activity: new RegExp(activity, "i") })
209
- .sort({ timestamp: -1 })
210
- .limit(limit);
211
- };
212
-
213
- auditTrailSchema.statics.findByFacility = function (
214
- facilityId,
215
- startDate,
216
- endDate,
217
- ) {
218
- const query = { "route_params.facilityId": facilityId };
219
- if (startDate || endDate) {
220
- query.timestamp = {};
221
- if (startDate) query.timestamp.$gte = startDate;
222
- if (endDate) query.timestamp.$lte = endDate;
223
- }
224
- return this.find(query).sort({ timestamp: -1 });
225
- };
226
-
227
- auditTrailSchema.statics.findByAsset = function (assetId, startDate, endDate) {
228
- const query = { "route_params.assetId": assetId };
229
- if (startDate || endDate) {
230
- query.timestamp = {};
231
- if (startDate) query.timestamp.$gte = startDate;
232
- if (endDate) query.timestamp.$lte = endDate;
233
- }
234
- return this.find(query).sort({ timestamp: -1 });
235
- };
236
-
237
- auditTrailSchema.statics.findByIP = function (ipAddress, startDate, endDate) {
238
- const query = { ip_address: ipAddress };
239
- if (startDate || endDate) {
240
- query.timestamp = {};
241
- if (startDate) query.timestamp.$gte = startDate;
242
- if (endDate) query.timestamp.$lte = endDate;
243
- }
244
- return this.find(query).sort({ timestamp: -1 });
245
- };
246
-
247
- auditTrailSchema.statics.findByMethod = function (method, startDate, endDate) {
248
- const query = { method: method.toUpperCase() };
249
- if (startDate || endDate) {
250
- query.timestamp = {};
251
- if (startDate) query.timestamp.$gte = startDate;
252
- if (endDate) query.timestamp.$lte = endDate;
253
- }
254
- return this.find(query).sort({ timestamp: -1 });
255
- };
256
-
257
- // Instance methods
258
- auditTrailSchema.methods.getActivityType = function () {
259
- if (this.activity.includes("Create")) return "CREATE";
260
- if (this.activity.includes("Update")) return "UPDATE";
261
- if (this.activity.includes("Delete")) return "DELETE";
262
- if (this.activity.includes("Login")) return "AUTH";
263
- return "OTHER";
264
- };
265
-
266
- auditTrailSchema.methods.isSuccessful = function () {
267
- return (
268
- this.activity.includes("Success") ||
269
- (!this.activity.includes("Failed") &&
270
- !this.activity.includes("Error") &&
271
- !this.activity.includes("Not Found"))
272
- );
273
- };
274
-
275
- auditTrailSchema.methods.hasDataChanges = function () {
276
- return !!(this.previous_data || this.deleted_data);
277
- };
278
-
279
- // Create indexes for efficient queries
280
- auditTrailSchema.index({ timestamp: -1 }); // Recent activities first
281
- auditTrailSchema.index({ "user_details.userId": 1, timestamp: -1 }); // User activity history
282
- auditTrailSchema.index({ "user_details.email": 1, timestamp: -1 }); // User by email
283
- auditTrailSchema.index({ ip_address: 1, timestamp: -1 }); // IP-based queries
284
- auditTrailSchema.index({ method: 1, timestamp: -1 }); // Method-based queries
285
- auditTrailSchema.index({ activity: 1, timestamp: -1 }); // Activity-based queries
286
- auditTrailSchema.index({ platform: 1, timestamp: -1 }); // Platform-based queries
287
- auditTrailSchema.index({ "route_params.facilityId": 1, timestamp: -1 }); // Facility-based queries
288
- auditTrailSchema.index({ "route_params.assetId": 1, timestamp: -1 }); // Asset-based queries
289
- auditTrailSchema.index({ is_authorized: 1, timestamp: -1 }); // Authorization status
290
- auditTrailSchema.index({ device_type: 1, timestamp: -1 }); // Device type queries
291
- auditTrailSchema.index({ browser: 1, timestamp: -1 }); // Browser-based queries
292
-
293
- // Compound indexes for common query patterns
294
- auditTrailSchema.index({
295
- "user_details.userId": 1,
296
- activity: 1,
297
- timestamp: -1,
298
- }); // User + activity
299
- auditTrailSchema.index({
300
- "route_params.facilityId": 1,
301
- method: 1,
302
- timestamp: -1,
303
- }); // Facility + method
304
- auditTrailSchema.index({
305
- ip_address: 1,
306
- "user_details.userId": 1,
307
- timestamp: -1,
308
- }); // IP + user correlation
309
-
310
- // Text search index for searching activities and user details
311
- auditTrailSchema.index({
312
- activity: "text",
313
- "user_details.email": "text",
314
- "user_details.fullName": "text",
315
- url: "text",
316
- });
317
-
318
- // Activity type constants
319
- auditTrailSchema.statics.ACTIVITY_TYPES = {
320
- CREATE: "CREATE",
321
- UPDATE: "UPDATE",
322
- DELETE: "DELETE",
323
- AUTH: "AUTH",
324
- OTHER: "OTHER",
325
- };
326
-
327
- // Method constants
328
- auditTrailSchema.statics.METHODS = {
329
- GET: "GET",
330
- POST: "POST",
331
- PUT: "PUT",
332
- PATCH: "PATCH",
333
- DELETE: "DELETE",
334
- OPTIONS: "OPTIONS",
335
- HEAD: "HEAD",
336
- };
337
-
338
- // Device type constants
339
- auditTrailSchema.statics.DEVICE_TYPES = {
340
- DESKTOP: "desktop",
341
- MOBILE: "mobile",
342
- TABLET: "tablet",
343
- UNKNOWN: "unknown",
344
- };
345
-
346
- module.exports = mongoose.model("AuditTrail", auditTrailSchema);
1
+ const mongoose = require("mongoose");
2
+
3
+ // User details sub-schema (from JWT token)
4
+ const userDetailsSchema = new mongoose.Schema(
5
+ {
6
+ userId: {
7
+ type: String,
8
+ required: true,
9
+ },
10
+ email: {
11
+ type: String,
12
+ required: true,
13
+ },
14
+ fullName: {
15
+ type: String,
16
+ required: true,
17
+ },
18
+ role: {
19
+ type: String,
20
+ required: true,
21
+ },
22
+ phoneNumber: {
23
+ type: String,
24
+ required: false,
25
+ },
26
+ facilityId: {
27
+ type: String,
28
+ required: false,
29
+ },
30
+ permissions: {
31
+ type: mongoose.Schema.Types.Mixed,
32
+ required: false,
33
+ },
34
+ tokenExpiry: {
35
+ type: Date,
36
+ required: false,
37
+ },
38
+ tokenIssued: {
39
+ type: Date,
40
+ required: false,
41
+ },
42
+ },
43
+ { _id: false },
44
+ );
45
+
46
+ // Main audit trail schema
47
+ const auditTrailSchema = new mongoose.Schema(
48
+ {
49
+ // Basic request information
50
+ timestamp: {
51
+ type: Date,
52
+ required: true,
53
+ default: Date.now,
54
+ },
55
+ method: {
56
+ type: String,
57
+ enum: ["GET", "POST", "PUT", "PATCH", "DELETE", "OPTIONS", "HEAD"],
58
+ required: true,
59
+ },
60
+ url: {
61
+ type: String,
62
+ required: true,
63
+ },
64
+ ip_address: {
65
+ type: String,
66
+ required: true,
67
+ },
68
+ platform: {
69
+ type: String,
70
+ required: true,
71
+ },
72
+
73
+ // Authorization and user information
74
+ is_authorized: {
75
+ type: Boolean,
76
+ required: true,
77
+ default: false,
78
+ },
79
+ user_details: {
80
+ type: userDetailsSchema,
81
+ required: false,
82
+ },
83
+
84
+ // Device and browser information
85
+ browser: {
86
+ type: String,
87
+ required: true,
88
+ default: "unknown",
89
+ },
90
+ operating_system: {
91
+ type: String,
92
+ required: true,
93
+ default: "unknown",
94
+ },
95
+ device_type: {
96
+ type: String,
97
+ enum: ["desktop", "mobile", "tablet", "unknown"],
98
+ required: true,
99
+ default: "unknown",
100
+ },
101
+ user_agent: {
102
+ type: String,
103
+ required: false,
104
+ },
105
+
106
+ // Request data
107
+ request_data: {
108
+ type: mongoose.Schema.Types.Mixed,
109
+ required: false,
110
+ },
111
+ query_params: {
112
+ type: mongoose.Schema.Types.Mixed,
113
+ required: false,
114
+ default: {},
115
+ },
116
+ route_params: {
117
+ type: mongoose.Schema.Types.Mixed,
118
+ required: false,
119
+ default: {},
120
+ },
121
+
122
+ // Activity information
123
+ activity: {
124
+ type: String,
125
+ required: true,
126
+ },
127
+
128
+ // response status
129
+ response_status: {
130
+ type: String,
131
+ required: true,
132
+ enum: ["success", "error", "unknown"],
133
+ default: "unknown",
134
+ },
135
+
136
+ // Method-specific data
137
+ previous_data: {
138
+ type: mongoose.Schema.Types.Mixed,
139
+ required: false,
140
+ },
141
+ deleted_data: {
142
+ type: mongoose.Schema.Types.Mixed,
143
+ required: false,
144
+ },
145
+
146
+ // Custom additional data
147
+ custom_data: {
148
+ type: mongoose.Schema.Types.Mixed,
149
+ required: false,
150
+ },
151
+
152
+ // Request tracking
153
+ request_id: {
154
+ type: String,
155
+ required: false,
156
+ },
157
+ },
158
+ {
159
+ timestamps: true,
160
+ },
161
+ );
162
+
163
+ // Pre-save middleware for data validation and cleanup
164
+ auditTrailSchema.pre("save", function (next) {
165
+ // Ensure timestamp is set
166
+ if (!this.timestamp) {
167
+ this.timestamp = new Date();
168
+ }
169
+
170
+ // Validate that authorized requests have user details
171
+ if (this.is_authorized && !this.user_details) {
172
+ console.warn("Authorized request without user details detected");
173
+ }
174
+
175
+ // Ensure query_params and route_params are objects
176
+ if (!this.query_params || typeof this.query_params !== "object") {
177
+ this.query_params = {};
178
+ }
179
+ if (!this.route_params || typeof this.route_params !== "object") {
180
+ this.route_params = {};
181
+ }
182
+
183
+ // Set browser to unknown if not provided
184
+ if (!this.browser) {
185
+ this.browser = "unknown";
186
+ }
187
+
188
+ // Set operating_system to unknown if not provided
189
+ if (!this.operating_system) {
190
+ this.operating_system = "unknown";
191
+ }
192
+
193
+ next();
194
+ });
195
+
196
+ // Static methods for common queries
197
+ auditTrailSchema.statics.findByUser = function (userId, startDate, endDate) {
198
+ const query = { "user_details.userId": userId };
199
+ if (startDate || endDate) {
200
+ query.timestamp = {};
201
+ if (startDate) query.timestamp.$gte = startDate;
202
+ if (endDate) query.timestamp.$lte = endDate;
203
+ }
204
+ return this.find(query).sort({ timestamp: -1 });
205
+ };
206
+
207
+ auditTrailSchema.statics.findByActivity = function (activity, limit = 100) {
208
+ return this.find({ activity: new RegExp(activity, "i") })
209
+ .sort({ timestamp: -1 })
210
+ .limit(limit);
211
+ };
212
+
213
+ auditTrailSchema.statics.findByFacility = function (
214
+ facilityId,
215
+ startDate,
216
+ endDate,
217
+ ) {
218
+ const query = { "route_params.facilityId": facilityId };
219
+ if (startDate || endDate) {
220
+ query.timestamp = {};
221
+ if (startDate) query.timestamp.$gte = startDate;
222
+ if (endDate) query.timestamp.$lte = endDate;
223
+ }
224
+ return this.find(query).sort({ timestamp: -1 });
225
+ };
226
+
227
+ auditTrailSchema.statics.findByAsset = function (assetId, startDate, endDate) {
228
+ const query = { "route_params.assetId": assetId };
229
+ if (startDate || endDate) {
230
+ query.timestamp = {};
231
+ if (startDate) query.timestamp.$gte = startDate;
232
+ if (endDate) query.timestamp.$lte = endDate;
233
+ }
234
+ return this.find(query).sort({ timestamp: -1 });
235
+ };
236
+
237
+ auditTrailSchema.statics.findByIP = function (ipAddress, startDate, endDate) {
238
+ const query = { ip_address: ipAddress };
239
+ if (startDate || endDate) {
240
+ query.timestamp = {};
241
+ if (startDate) query.timestamp.$gte = startDate;
242
+ if (endDate) query.timestamp.$lte = endDate;
243
+ }
244
+ return this.find(query).sort({ timestamp: -1 });
245
+ };
246
+
247
+ auditTrailSchema.statics.findByMethod = function (method, startDate, endDate) {
248
+ const query = { method: method.toUpperCase() };
249
+ if (startDate || endDate) {
250
+ query.timestamp = {};
251
+ if (startDate) query.timestamp.$gte = startDate;
252
+ if (endDate) query.timestamp.$lte = endDate;
253
+ }
254
+ return this.find(query).sort({ timestamp: -1 });
255
+ };
256
+
257
+ // Instance methods
258
+ auditTrailSchema.methods.getActivityType = function () {
259
+ if (this.activity.includes("Create")) return "CREATE";
260
+ if (this.activity.includes("Update")) return "UPDATE";
261
+ if (this.activity.includes("Delete")) return "DELETE";
262
+ if (this.activity.includes("Login")) return "AUTH";
263
+ return "OTHER";
264
+ };
265
+
266
+ auditTrailSchema.methods.isSuccessful = function () {
267
+ return (
268
+ this.activity.includes("Success") ||
269
+ (!this.activity.includes("Failed") &&
270
+ !this.activity.includes("Error") &&
271
+ !this.activity.includes("Not Found"))
272
+ );
273
+ };
274
+
275
+ auditTrailSchema.methods.hasDataChanges = function () {
276
+ return !!(this.previous_data || this.deleted_data);
277
+ };
278
+
279
+ // Create indexes for efficient queries
280
+ auditTrailSchema.index({ timestamp: -1 }); // Recent activities first
281
+ auditTrailSchema.index({ "user_details.userId": 1, timestamp: -1 }); // User activity history
282
+ auditTrailSchema.index({ "user_details.email": 1, timestamp: -1 }); // User by email
283
+ auditTrailSchema.index({ ip_address: 1, timestamp: -1 }); // IP-based queries
284
+ auditTrailSchema.index({ method: 1, timestamp: -1 }); // Method-based queries
285
+ auditTrailSchema.index({ activity: 1, timestamp: -1 }); // Activity-based queries
286
+ auditTrailSchema.index({ platform: 1, timestamp: -1 }); // Platform-based queries
287
+ auditTrailSchema.index({ "route_params.facilityId": 1, timestamp: -1 }); // Facility-based queries
288
+ auditTrailSchema.index({ "route_params.assetId": 1, timestamp: -1 }); // Asset-based queries
289
+ auditTrailSchema.index({ is_authorized: 1, timestamp: -1 }); // Authorization status
290
+ auditTrailSchema.index({ device_type: 1, timestamp: -1 }); // Device type queries
291
+ auditTrailSchema.index({ browser: 1, timestamp: -1 }); // Browser-based queries
292
+
293
+ // Compound indexes for common query patterns
294
+ auditTrailSchema.index({
295
+ "user_details.userId": 1,
296
+ activity: 1,
297
+ timestamp: -1,
298
+ }); // User + activity
299
+ auditTrailSchema.index({
300
+ "route_params.facilityId": 1,
301
+ method: 1,
302
+ timestamp: -1,
303
+ }); // Facility + method
304
+ auditTrailSchema.index({
305
+ ip_address: 1,
306
+ "user_details.userId": 1,
307
+ timestamp: -1,
308
+ }); // IP + user correlation
309
+
310
+ // Text search index for searching activities and user details
311
+ auditTrailSchema.index({
312
+ activity: "text",
313
+ "user_details.email": "text",
314
+ "user_details.fullName": "text",
315
+ url: "text",
316
+ });
317
+
318
+ // Activity type constants
319
+ auditTrailSchema.statics.ACTIVITY_TYPES = {
320
+ CREATE: "CREATE",
321
+ UPDATE: "UPDATE",
322
+ DELETE: "DELETE",
323
+ AUTH: "AUTH",
324
+ OTHER: "OTHER",
325
+ };
326
+
327
+ // Method constants
328
+ auditTrailSchema.statics.METHODS = {
329
+ GET: "GET",
330
+ POST: "POST",
331
+ PUT: "PUT",
332
+ PATCH: "PATCH",
333
+ DELETE: "DELETE",
334
+ OPTIONS: "OPTIONS",
335
+ HEAD: "HEAD",
336
+ };
337
+
338
+ // Device type constants
339
+ auditTrailSchema.statics.DEVICE_TYPES = {
340
+ DESKTOP: "desktop",
341
+ MOBILE: "mobile",
342
+ TABLET: "tablet",
343
+ UNKNOWN: "unknown",
344
+ };
345
+
346
+ module.exports = mongoose.model("AuditTrail", auditTrailSchema);