payservedb 8.7.0 → 8.7.2

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 (186) hide show
  1. package/.env +2 -2
  2. package/ZOHO_INTEGRATION_SCHEMA.md +644 -644
  3. package/index.js +319 -319
  4. package/package.json +17 -17
  5. package/src/models/account.js +52 -52
  6. package/src/models/agent_departments.js +59 -59
  7. package/src/models/agent_notifications.js +53 -53
  8. package/src/models/agent_performance.js +127 -127
  9. package/src/models/agent_roles.js +77 -77
  10. package/src/models/agents.js +154 -154
  11. package/src/models/apilog.js +18 -18
  12. package/src/models/approvalsWorkflows.js +49 -49
  13. package/src/models/archivedapilog.js +18 -18
  14. package/src/models/asset.js +92 -92
  15. package/src/models/assetsAssignment.js +64 -64
  16. package/src/models/auditTrail.js +346 -346
  17. package/src/models/bankdetails.js +47 -47
  18. package/src/models/billerAddress.js +124 -124
  19. package/src/models/booking_invoice.js +165 -165
  20. package/src/models/bookinganalytics.js +63 -63
  21. package/src/models/bookingconfig.js +45 -45
  22. package/src/models/bookingproperty.js +179 -179
  23. package/src/models/bookingreservation.js +239 -239
  24. package/src/models/bookingrevenuerecord.js +84 -84
  25. package/src/models/budget.js +95 -95
  26. package/src/models/budgetCategory.js +19 -19
  27. package/src/models/campaigns.js +108 -108
  28. package/src/models/cashpayment.js +291 -291
  29. package/src/models/combinedUnits.js +62 -62
  30. package/src/models/combined_invoice.js +424 -424
  31. package/src/models/common_area_electricity.js +38 -38
  32. package/src/models/common_area_generator.js +41 -41
  33. package/src/models/common_area_utility_alert.js +37 -37
  34. package/src/models/common_area_water.js +39 -39
  35. package/src/models/communication_status.js +33 -33
  36. package/src/models/communication_user_opt.js +32 -32
  37. package/src/models/community_guidelines.js +35 -35
  38. package/src/models/company.js +53 -53
  39. package/src/models/coreBaseSettings.js +16 -16
  40. package/src/models/coreInvoiceSettings.js +100 -100
  41. package/src/models/counter_schema.js +21 -21
  42. package/src/models/country_tax.js +42 -42
  43. package/src/models/currency_settings.js +39 -39
  44. package/src/models/customer.js +214 -214
  45. package/src/models/customer_satisfaction_survey.js +278 -278
  46. package/src/models/customer_surveys.js +139 -139
  47. package/src/models/customer_tickets.js +239 -239
  48. package/src/models/dailyChecklist.js +312 -312
  49. package/src/models/default_payment_details.js +17 -17
  50. package/src/models/deliveryTimeMarks.js +18 -18
  51. package/src/models/document_type.js +19 -19
  52. package/src/models/dutyRosterChecklist.js +250 -250
  53. package/src/models/dutyroster.js +136 -136
  54. package/src/models/email.js +37 -37
  55. package/src/models/email_sms_queue.js +61 -61
  56. package/src/models/entry_exit.js +53 -53
  57. package/src/models/expense.js +99 -99
  58. package/src/models/expense_category.js +45 -45
  59. package/src/models/facility.js +71 -71
  60. package/src/models/facilityBillingPrices.js +29 -29
  61. package/src/models/facilityInvoice.js +240 -240
  62. package/src/models/facilityInvoicePayment.js +52 -52
  63. package/src/models/facilityInvoiceRecipient.js +32 -32
  64. package/src/models/facilityWalletTransactionsMetadata.js +236 -236
  65. package/src/models/facility_departements.js +20 -20
  66. package/src/models/facility_payment_details.js +20 -20
  67. package/src/models/facility_rating.js +78 -78
  68. package/src/models/facilityasset.js +25 -25
  69. package/src/models/faq.js +14 -14
  70. package/src/models/gl_account_double_entries.js +25 -25
  71. package/src/models/gl_accounts.js +56 -56
  72. package/src/models/gl_entries.js +49 -49
  73. package/src/models/goodsReceivedNotes.js +115 -115
  74. package/src/models/guard.js +47 -47
  75. package/src/models/handover.js +258 -258
  76. package/src/models/inspection_category.js +38 -38
  77. package/src/models/invoice.js +480 -480
  78. package/src/models/invoice_generation_approval.js +86 -86
  79. package/src/models/invoicing_schedule.js +40 -40
  80. package/src/models/item_inspection.js +96 -96
  81. package/src/models/knowledge_base.js +109 -109
  82. package/src/models/knowledge_base_rating.js +44 -44
  83. package/src/models/leaseagreement.js +237 -237
  84. package/src/models/leasetemplate.js +17 -17
  85. package/src/models/levy.js +223 -223
  86. package/src/models/levy_invoice_settings.js +26 -26
  87. package/src/models/levycontract.js +210 -210
  88. package/src/models/levytype.js +23 -23
  89. package/src/models/maintenance_service_vendor.js +38 -38
  90. package/src/models/maintenance_services.js +17 -17
  91. package/src/models/maintenancerequisition.js +31 -31
  92. package/src/models/master_workplan.js +32 -32
  93. package/src/models/master_workplan_child.js +34 -34
  94. package/src/models/message.js +38 -38
  95. package/src/models/module.js +21 -21
  96. package/src/models/notification.js +44 -44
  97. package/src/models/paymentTermsMarks.js +19 -19
  98. package/src/models/penalty.js +76 -76
  99. package/src/models/pendingCredentials.js +32 -32
  100. package/src/models/powerMeterCommunicationProtocol.js +17 -17
  101. package/src/models/powerMeterCustomerAccount.js +78 -78
  102. package/src/models/powerMeterCustomerBand.js +14 -14
  103. package/src/models/powerMeterDailyReading.js +30 -30
  104. package/src/models/powerMeterGateways.js +40 -40
  105. package/src/models/powerMeterMonthlyReading.js +34 -34
  106. package/src/models/powerMeterPowerCharges.js +85 -85
  107. package/src/models/powerMeterSettings.js +159 -159
  108. package/src/models/powerMeterSingleDayReading.js +32 -32
  109. package/src/models/powerMeters.js +116 -116
  110. package/src/models/powerMetersManufacturer.js +14 -14
  111. package/src/models/power_meter_account.js +81 -81
  112. package/src/models/power_meter_command_logs.js +30 -30
  113. package/src/models/power_meter_command_queue.js +33 -33
  114. package/src/models/power_meter_negative_balance.js +44 -44
  115. package/src/models/power_prepaid_credits.js +47 -47
  116. package/src/models/power_prepaid_debits.js +53 -53
  117. package/src/models/power_prepaid_orders.js +78 -78
  118. package/src/models/power_sms_notification.js +26 -26
  119. package/src/models/privacy_policy.js +19 -19
  120. package/src/models/propertyManagerContract.js +556 -556
  121. package/src/models/propertyManagerRevenue.js +195 -195
  122. package/src/models/purchaseOrderInvoice.js +74 -74
  123. package/src/models/purchase_order.js +213 -213
  124. package/src/models/purchase_request.js +110 -110
  125. package/src/models/quickbooks_config.js +52 -41
  126. package/src/models/refresh_token.js +23 -23
  127. package/src/models/reminder.js +197 -197
  128. package/src/models/report.js +13 -13
  129. package/src/models/resident.js +121 -121
  130. package/src/models/rfq_details.js +131 -131
  131. package/src/models/rfq_response.js +153 -153
  132. package/src/models/service_charge_invoice_upload.js +42 -42
  133. package/src/models/service_charge_payments.js +27 -27
  134. package/src/models/servicerequest.js +55 -55
  135. package/src/models/settings.js +62 -62
  136. package/src/models/short_urls.js +21 -21
  137. package/src/models/smart_meter_daily_consumption.js +44 -44
  138. package/src/models/sms_africastalking.js +20 -20
  139. package/src/models/sms_balance_notification.js +26 -26
  140. package/src/models/sms_meliora.js +20 -20
  141. package/src/models/staff.js +36 -36
  142. package/src/models/stocksandspare.js +161 -161
  143. package/src/models/suppliers.js +74 -74
  144. package/src/models/terms_and_conditions.js +19 -19
  145. package/src/models/tickets.js +186 -186
  146. package/src/models/tickets_category.js +72 -72
  147. package/src/models/unitManagementTemplate.js +44 -44
  148. package/src/models/unitasset.js +25 -25
  149. package/src/models/units.js +118 -118
  150. package/src/models/user.js +185 -185
  151. package/src/models/valueaddedservices.js +50 -79
  152. package/src/models/vas_invoices_upload.js +50 -50
  153. package/src/models/vas_payments.js +24 -24
  154. package/src/models/vasinvoice.js +192 -192
  155. package/src/models/vasvendor.js +57 -57
  156. package/src/models/visitLog.js +95 -95
  157. package/src/models/visitor.js +67 -67
  158. package/src/models/waitlist.js +45 -45
  159. package/src/models/wallet.js +44 -44
  160. package/src/models/wallet_transactions.js +50 -50
  161. package/src/models/water_invoice.js +351 -351
  162. package/src/models/water_meter_Command_Queue.js +33 -33
  163. package/src/models/water_meter_account.js +86 -86
  164. package/src/models/water_meter_billing.js +58 -58
  165. package/src/models/water_meter_combined_accounts.js +92 -92
  166. package/src/models/water_meter_communication.js +17 -17
  167. package/src/models/water_meter_communication_logs.js +39 -39
  168. package/src/models/water_meter_concentrator.js +70 -70
  169. package/src/models/water_meter_daily_history.js +32 -32
  170. package/src/models/water_meter_high_risk.js +36 -36
  171. package/src/models/water_meter_iot_cards.js +34 -34
  172. package/src/models/water_meter_loan_deduction.js +134 -134
  173. package/src/models/water_meter_manufacturer.js +35 -35
  174. package/src/models/water_meter_monthly_history.js +36 -36
  175. package/src/models/water_meter_negative_amounts.js +44 -44
  176. package/src/models/water_meter_settings.js +290 -290
  177. package/src/models/water_meter_single_day_history.js +34 -34
  178. package/src/models/water_meter_size.js +15 -15
  179. package/src/models/water_meters.js +133 -133
  180. package/src/models/water_meters_delivery.js +76 -76
  181. package/src/models/water_prepaid_credit.js +47 -47
  182. package/src/models/water_prepaid_debit.js +50 -50
  183. package/src/models/workorder.js +49 -49
  184. package/src/models/zohoAccount.js +453 -453
  185. package/src/models/zohoIntegration.js +262 -262
  186. package/src/models/zohoItem.js +504 -504
@@ -1,480 +1,480 @@
1
- const mongoose = require("mongoose");
2
-
3
- const invoiceSchema = new mongoose.Schema(
4
- {
5
- invoiceNumber: {
6
- type: String,
7
- required: true,
8
- unique: true,
9
- },
10
- accountNumber: {
11
- type: String,
12
- required: true,
13
- },
14
- client: {
15
- clientId: {
16
- type: mongoose.Schema.Types.ObjectId,
17
- ref: "Customer",
18
- required: true,
19
- },
20
- firstName: {
21
- type: String,
22
- required: true,
23
- },
24
- lastName: {
25
- type: String,
26
- required: true,
27
- },
28
- },
29
- facility: {
30
- id: {
31
- type: mongoose.Schema.Types.ObjectId,
32
- ref: "Facility",
33
- required: true,
34
- },
35
- name: {
36
- type: String,
37
- required: true,
38
- },
39
- },
40
- unit: {
41
- id: { type: mongoose.Schema.Types.ObjectId, ref: "Unit", required: true },
42
- name: { type: String, required: true },
43
- },
44
- currency: {
45
- id: {
46
- type: mongoose.Schema.Types.ObjectId,
47
- ref: "Currency",
48
- required: true,
49
- },
50
- name: {
51
- type: String,
52
- required: true,
53
- },
54
- code: {
55
- type: String,
56
- required: true,
57
- uppercase: true,
58
- minlength: 3,
59
- maxlength: 3,
60
- },
61
- },
62
- items: [
63
- {
64
- description: { type: String, required: true },
65
- quantity: { type: Number, required: true, min: 1 },
66
- unitPrice: { type: Number, required: true, min: 0 },
67
- },
68
- ],
69
- subTotal: {
70
- type: Number,
71
- required: true,
72
- },
73
- tax: {
74
- type: Number,
75
- required: true,
76
- },
77
- totalAmount: {
78
- type: Number,
79
- required: true,
80
- },
81
- amountPaid: {
82
- type: Number,
83
- default: 0,
84
- min: 0,
85
- },
86
- // Mark as deprecated, keep for backward compatibility
87
- overpay: {
88
- type: Number,
89
- default: 0,
90
- min: 0,
91
- deprecated: true,
92
- },
93
- issueDate: {
94
- type: Date,
95
- required: true,
96
- },
97
- dueDate: {
98
- type: Date,
99
- required: true,
100
- },
101
- status: {
102
- type: String,
103
- required: true,
104
- enum: [
105
- "Unpaid",
106
- "Pending",
107
- "Paid",
108
- "Overdue",
109
- "Cancelled",
110
- "Partially Paid",
111
- "Void",
112
- ],
113
- },
114
- penalty: {
115
- type: Number,
116
- default: 0,
117
- },
118
- whatFor: {
119
- invoiceType: { type: String, required: true },
120
- description: { type: String },
121
- },
122
- invoiceNote: {
123
- type: String,
124
- default: null,
125
- },
126
- balanceBroughtForward: {
127
- type: Number,
128
- default: 0,
129
- },
130
- // NEW FIELD 1: Year-Month for easy filtering
131
- yearMonth: {
132
- type: String,
133
- match: /^\d{4}-\d{2}$/, // Validates format like "2025-08"
134
- index: true,
135
- },
136
- // NEW FIELD 2: Simple notification tracking
137
- notificationsSent: {
138
- sms: {
139
- type: Boolean,
140
- default: false,
141
- },
142
- email: {
143
- type: Boolean,
144
- default: false,
145
- },
146
- attempts: {
147
- type: Number,
148
- default: 0,
149
- },
150
- },
151
- voidMetadata: {
152
- voidedBy: {
153
- userId: { type: mongoose.Schema.Types.ObjectId, ref: "User" },
154
- name: { type: String },
155
- role: { type: String },
156
- },
157
- voidedAt: { type: Date },
158
- reason: { type: String },
159
- },
160
- cancelMetadata: {
161
- cancelledBy: {
162
- userId: { type: mongoose.Schema.Types.ObjectId, ref: "User" },
163
- name: { type: String },
164
- role: { type: String },
165
- },
166
- cancelledAt: { type: Date },
167
- reason: { type: String },
168
- },
169
- lastReminderSent: Date,
170
- reminderHistory: [
171
- {
172
- sentAt: Date,
173
- reminderId: mongoose.Schema.Types.ObjectId,
174
- notificationTypes: [String],
175
- },
176
- ],
177
- reconciliationHistory: [
178
- {
179
- date: { type: Date, required: true },
180
- amount: { type: Number, required: true },
181
- type: {
182
- type: String,
183
- enum: [
184
- "payment",
185
- "cash",
186
- "cheque",
187
- "bank-transfer",
188
- "mpesa-transfer",
189
- "overpay-transfer",
190
- "balance-deduction",
191
- "overpay-received",
192
- "credit-forward",
193
- "debit-forward",
194
- ],
195
- required: true,
196
- },
197
- sourceInvoice: String,
198
- destinationInvoice: String,
199
- paymentReference: String,
200
- paymentCompletion: String,
201
- remainingBalance: Number,
202
- notes: String,
203
- exchangeRate: {
204
- type: Number,
205
- default: 1, // For cross-currency reconciliations
206
- },
207
- originalCurrency: {
208
- code: String, // Original currency code if different from invoice currency
209
- amount: Number, // Amount in original currency
210
- },
211
- },
212
- ],
213
- paymentDetails: {
214
- paymentStatus: { type: String, required: true },
215
- paymentMethod: { type: String },
216
- paymentDate: { type: Date },
217
- transactionId: { type: String },
218
- },
219
- // New field to track when an invoice has been viewed
220
- viewStatus: {
221
- isOpened: { type: Boolean, default: false },
222
- openedAt: { type: Date, default: null },
223
- openedBy: {
224
- facilityId: { type: mongoose.Schema.Types.ObjectId, ref: "Facility" },
225
- userId: { type: mongoose.Schema.Types.ObjectId, ref: "User" },
226
- userRole: { type: String },
227
- },
228
- viewHistory: [
229
- {
230
- viewedAt: { type: Date, required: true },
231
- facilityId: { type: mongoose.Schema.Types.ObjectId, ref: "Facility" },
232
- userId: { type: mongoose.Schema.Types.ObjectId, ref: "User" },
233
- },
234
- ],
235
- },
236
- invoiceUrl: {
237
- type: String,
238
- default: null,
239
- },
240
- // eTims tax integration fields
241
- ticketUrl: {
242
- type: String,
243
- default: null,
244
- description: "QR code URL from eTims/KRA for tax invoice verification",
245
- },
246
- txsync: {
247
- status: {
248
- type: String,
249
- enum: ["pending", "synced", "failed", "not-applicable"],
250
- default: "not-applicable",
251
- },
252
- syncedAt: {
253
- type: Date,
254
- default: null,
255
- },
256
- etimsInvoiceNo: {
257
- type: String,
258
- default: null,
259
- },
260
- sdcId: {
261
- type: String,
262
- default: null,
263
- },
264
- receiptNo: {
265
- type: String,
266
- default: null,
267
- },
268
- errorMessage: {
269
- type: String,
270
- default: null,
271
- },
272
- attempts: {
273
- type: Number,
274
- default: 0,
275
- },
276
- lastAttemptAt: {
277
- type: Date,
278
- default: null,
279
- },
280
- },
281
- // New fields for double entry accounts
282
- invoiceDoubleEntryAccount: {
283
- type: mongoose.Schema.Types.ObjectId,
284
- ref: "GLAccountDoubleEntries",
285
- required: false,
286
- },
287
- paymentDoubleEntryAccount: {
288
- type: mongoose.Schema.Types.ObjectId,
289
- ref: "GLAccountDoubleEntries",
290
- required: false,
291
- },
292
- // GL account details for invoice entries
293
- accountdebitedData: {
294
- amount: { type: Number },
295
- description: { type: String },
296
- isActive: { type: Boolean, default: true },
297
- },
298
- accountcreditedData: {
299
- amount: { type: Number },
300
- description: { type: String },
301
- isActive: { type: Boolean, default: true },
302
- },
303
- },
304
- {
305
- timestamps: true,
306
- },
307
- );
308
-
309
- // Add indexes for frequently queried fields
310
- invoiceSchema.index({ accountNumber: 1 });
311
- invoiceSchema.index({ status: 1 });
312
- invoiceSchema.index({ "client.clientId": 1, status: 1 });
313
- invoiceSchema.index({ "reconciliationHistory.paymentReference": 1 });
314
- invoiceSchema.index({ issueDate: -1 });
315
- invoiceSchema.index({ "currency.code": 1 }); // Add index for currency code
316
- invoiceSchema.index({ "currency.id": 1 }); // Add index for currency ID
317
- invoiceSchema.index({ "currency.code": 1, "client.clientId": 1, status: 1 }); // Compound index for currency-based queries
318
- invoiceSchema.index({ "client.clientId": 1, balanceBroughtForward: 1 }); // Add index for finding invoices with credit balances
319
- invoiceSchema.index({ "viewStatus.isOpened": 1 }); // Add index for view status
320
- invoiceSchema.index({ "viewStatus.openedBy.facilityId": 1 }); // Add index for facility view tracking
321
- // Add index for double entry accounts
322
- invoiceSchema.index({ invoiceDoubleEntryAccount: 1 });
323
- invoiceSchema.index({ paymentDoubleEntryAccount: 1 });
324
- // NEW INDEXES for new fields
325
- invoiceSchema.index({ yearMonth: 1 }); // For filtering by year-month
326
- invoiceSchema.index({ yearMonth: 1, status: 1 }); // Combined index for monthly reports
327
- invoiceSchema.index({ "notificationsSent.sms": 1 }); // For finding SMS sent/not sent
328
- invoiceSchema.index({ "notificationsSent.email": 1 }); // For finding email sent/not sent
329
- // eTims tax sync indexes
330
- invoiceSchema.index({ "txsync.status": 1 }); // For finding invoices by sync status
331
- invoiceSchema.index({ "txsync.etimsInvoiceNo": 1 }); // For looking up by eTims invoice number
332
-
333
- // Add virtual field for calculating balance
334
- invoiceSchema.virtual("calculatedBalance").get(function () {
335
- const baseBalance = this.totalAmount - (this.amountPaid || 0);
336
-
337
- // Add positive balanceBroughtForward (customer owes money)
338
- if (this.balanceBroughtForward > 0) {
339
- return baseBalance + this.balanceBroughtForward;
340
- }
341
-
342
- // Subtract negative balanceBroughtForward (credit)
343
- return baseBalance;
344
- });
345
-
346
- // Add virtual field for credit balance
347
- invoiceSchema.virtual("creditBalance").get(function () {
348
- return this.balanceBroughtForward < 0
349
- ? Math.abs(this.balanceBroughtForward)
350
- : 0;
351
- });
352
-
353
- // Getter for compatible overpay field
354
- invoiceSchema.virtual("effectiveOverpay").get(function () {
355
- return this.balanceBroughtForward < 0
356
- ? Math.abs(this.balanceBroughtForward)
357
- : 0;
358
- });
359
-
360
- // Add virtual populate for invoice double entry account
361
- invoiceSchema.virtual("invoiceDoubleEntry", {
362
- ref: "GLAccountDoubleEntries",
363
- localField: "invoiceDoubleEntryAccount",
364
- foreignField: "_id",
365
- justOne: true,
366
- });
367
-
368
- // Add virtual populate for payment double entry account
369
- invoiceSchema.virtual("paymentDoubleEntry", {
370
- ref: "GLAccountDoubleEntries",
371
- localField: "paymentDoubleEntryAccount",
372
- foreignField: "_id",
373
- justOne: true,
374
- });
375
-
376
- // Add method for currency conversion if needed
377
- invoiceSchema.methods.convertAmount = function (
378
- amount,
379
- fromCurrency,
380
- toCurrency,
381
- exchangeRate,
382
- ) {
383
- if (fromCurrency === toCurrency) {
384
- return amount;
385
- }
386
- return amount * exchangeRate;
387
- };
388
-
389
- // Add static method to find invoices by currency
390
- invoiceSchema.statics.findByCurrency = function (currencyCode) {
391
- return this.find({ "currency.code": currencyCode.toUpperCase() });
392
- };
393
-
394
- // Add static method to find invoices with credit balance
395
- invoiceSchema.statics.findWithCreditBalance = function (clientId) {
396
- return this.find({
397
- "client.clientId": clientId,
398
- balanceBroughtForward: { $lt: 0 },
399
- }).sort({ updatedAt: -1 });
400
- };
401
-
402
- // Add static method to calculate totals by currency
403
- invoiceSchema.statics.calculateTotalsByCurrency = function (query = {}) {
404
- return this.aggregate([
405
- { $match: query },
406
- {
407
- $group: {
408
- _id: "$currency.code",
409
- totalAmount: { $sum: "$totalAmount" },
410
- totalPaid: { $sum: "$amountPaid" },
411
- count: { $sum: 1 },
412
- },
413
- },
414
- ]);
415
- };
416
-
417
- // New static method to find all unviewed invoices
418
- invoiceSchema.statics.findUnviewedInvoices = function (facilityId) {
419
- return this.find({
420
- "facility.id": facilityId,
421
- "viewStatus.isOpened": false,
422
- });
423
- };
424
-
425
- // NEW STATIC METHODS for new fields
426
- // Find invoices by year-month
427
- invoiceSchema.statics.findByYearMonth = function (yearMonth) {
428
- return this.find({ yearMonth: yearMonth });
429
- };
430
-
431
- // Find invoices where notifications haven't been sent
432
- invoiceSchema.statics.findPendingNotifications = function (facilityId) {
433
- return this.find({
434
- "facility.id": facilityId,
435
- status: { $in: ["Unpaid", "Overdue"] },
436
- $or: [
437
- { "notificationsSent.sms": false },
438
- { "notificationsSent.email": false },
439
- ],
440
- });
441
- };
442
-
443
- // Find invoices pending eTims sync
444
- invoiceSchema.statics.findPendingTaxSync = function (facilityId) {
445
- return this.find({
446
- "facility.id": facilityId,
447
- "txsync.status": "pending",
448
- });
449
- };
450
-
451
- // Find invoices with failed eTims sync
452
- invoiceSchema.statics.findFailedTaxSync = function (facilityId) {
453
- return this.find({
454
- "facility.id": facilityId,
455
- "txsync.status": "failed",
456
- });
457
- };
458
-
459
- // Pre-save middleware to ensure overpay and balanceBroughtForward stay in sync during transition
460
- invoiceSchema.pre("save", function (next) {
461
- // If balanceBroughtForward is negative (credit), sync with overpay for backwards compatibility
462
- if (this.balanceBroughtForward < 0) {
463
- this.overpay = Math.abs(this.balanceBroughtForward);
464
- } else {
465
- this.overpay = 0; // No overpay if there's no negative balance
466
- }
467
-
468
- // NEW: Auto-generate yearMonth from issueDate if not provided
469
- if (!this.yearMonth && this.issueDate) {
470
- const year = this.issueDate.getFullYear();
471
- const month = String(this.issueDate.getMonth() + 1).padStart(2, "0");
472
- this.yearMonth = `${year}-${month}`;
473
- }
474
-
475
- next();
476
- });
477
-
478
- const Invoice = mongoose.model("Invoice", invoiceSchema);
479
-
480
- module.exports = Invoice;
1
+ const mongoose = require("mongoose");
2
+
3
+ const invoiceSchema = new mongoose.Schema(
4
+ {
5
+ invoiceNumber: {
6
+ type: String,
7
+ required: true,
8
+ unique: true,
9
+ },
10
+ accountNumber: {
11
+ type: String,
12
+ required: true,
13
+ },
14
+ client: {
15
+ clientId: {
16
+ type: mongoose.Schema.Types.ObjectId,
17
+ ref: "Customer",
18
+ required: true,
19
+ },
20
+ firstName: {
21
+ type: String,
22
+ required: true,
23
+ },
24
+ lastName: {
25
+ type: String,
26
+ required: true,
27
+ },
28
+ },
29
+ facility: {
30
+ id: {
31
+ type: mongoose.Schema.Types.ObjectId,
32
+ ref: "Facility",
33
+ required: true,
34
+ },
35
+ name: {
36
+ type: String,
37
+ required: true,
38
+ },
39
+ },
40
+ unit: {
41
+ id: { type: mongoose.Schema.Types.ObjectId, ref: "Unit", required: true },
42
+ name: { type: String, required: true },
43
+ },
44
+ currency: {
45
+ id: {
46
+ type: mongoose.Schema.Types.ObjectId,
47
+ ref: "Currency",
48
+ required: true,
49
+ },
50
+ name: {
51
+ type: String,
52
+ required: true,
53
+ },
54
+ code: {
55
+ type: String,
56
+ required: true,
57
+ uppercase: true,
58
+ minlength: 3,
59
+ maxlength: 3,
60
+ },
61
+ },
62
+ items: [
63
+ {
64
+ description: { type: String, required: true },
65
+ quantity: { type: Number, required: true, min: 1 },
66
+ unitPrice: { type: Number, required: true, min: 0 },
67
+ },
68
+ ],
69
+ subTotal: {
70
+ type: Number,
71
+ required: true,
72
+ },
73
+ tax: {
74
+ type: Number,
75
+ required: true,
76
+ },
77
+ totalAmount: {
78
+ type: Number,
79
+ required: true,
80
+ },
81
+ amountPaid: {
82
+ type: Number,
83
+ default: 0,
84
+ min: 0,
85
+ },
86
+ // Mark as deprecated, keep for backward compatibility
87
+ overpay: {
88
+ type: Number,
89
+ default: 0,
90
+ min: 0,
91
+ deprecated: true,
92
+ },
93
+ issueDate: {
94
+ type: Date,
95
+ required: true,
96
+ },
97
+ dueDate: {
98
+ type: Date,
99
+ required: true,
100
+ },
101
+ status: {
102
+ type: String,
103
+ required: true,
104
+ enum: [
105
+ "Unpaid",
106
+ "Pending",
107
+ "Paid",
108
+ "Overdue",
109
+ "Cancelled",
110
+ "Partially Paid",
111
+ "Void",
112
+ ],
113
+ },
114
+ penalty: {
115
+ type: Number,
116
+ default: 0,
117
+ },
118
+ whatFor: {
119
+ invoiceType: { type: String, required: true },
120
+ description: { type: String },
121
+ },
122
+ invoiceNote: {
123
+ type: String,
124
+ default: null,
125
+ },
126
+ balanceBroughtForward: {
127
+ type: Number,
128
+ default: 0,
129
+ },
130
+ // NEW FIELD 1: Year-Month for easy filtering
131
+ yearMonth: {
132
+ type: String,
133
+ match: /^\d{4}-\d{2}$/, // Validates format like "2025-08"
134
+ index: true,
135
+ },
136
+ // NEW FIELD 2: Simple notification tracking
137
+ notificationsSent: {
138
+ sms: {
139
+ type: Boolean,
140
+ default: false,
141
+ },
142
+ email: {
143
+ type: Boolean,
144
+ default: false,
145
+ },
146
+ attempts: {
147
+ type: Number,
148
+ default: 0,
149
+ },
150
+ },
151
+ voidMetadata: {
152
+ voidedBy: {
153
+ userId: { type: mongoose.Schema.Types.ObjectId, ref: "User" },
154
+ name: { type: String },
155
+ role: { type: String },
156
+ },
157
+ voidedAt: { type: Date },
158
+ reason: { type: String },
159
+ },
160
+ cancelMetadata: {
161
+ cancelledBy: {
162
+ userId: { type: mongoose.Schema.Types.ObjectId, ref: "User" },
163
+ name: { type: String },
164
+ role: { type: String },
165
+ },
166
+ cancelledAt: { type: Date },
167
+ reason: { type: String },
168
+ },
169
+ lastReminderSent: Date,
170
+ reminderHistory: [
171
+ {
172
+ sentAt: Date,
173
+ reminderId: mongoose.Schema.Types.ObjectId,
174
+ notificationTypes: [String],
175
+ },
176
+ ],
177
+ reconciliationHistory: [
178
+ {
179
+ date: { type: Date, required: true },
180
+ amount: { type: Number, required: true },
181
+ type: {
182
+ type: String,
183
+ enum: [
184
+ "payment",
185
+ "cash",
186
+ "cheque",
187
+ "bank-transfer",
188
+ "mpesa-transfer",
189
+ "overpay-transfer",
190
+ "balance-deduction",
191
+ "overpay-received",
192
+ "credit-forward",
193
+ "debit-forward",
194
+ ],
195
+ required: true,
196
+ },
197
+ sourceInvoice: String,
198
+ destinationInvoice: String,
199
+ paymentReference: String,
200
+ paymentCompletion: String,
201
+ remainingBalance: Number,
202
+ notes: String,
203
+ exchangeRate: {
204
+ type: Number,
205
+ default: 1, // For cross-currency reconciliations
206
+ },
207
+ originalCurrency: {
208
+ code: String, // Original currency code if different from invoice currency
209
+ amount: Number, // Amount in original currency
210
+ },
211
+ },
212
+ ],
213
+ paymentDetails: {
214
+ paymentStatus: { type: String, required: true },
215
+ paymentMethod: { type: String },
216
+ paymentDate: { type: Date },
217
+ transactionId: { type: String },
218
+ },
219
+ // New field to track when an invoice has been viewed
220
+ viewStatus: {
221
+ isOpened: { type: Boolean, default: false },
222
+ openedAt: { type: Date, default: null },
223
+ openedBy: {
224
+ facilityId: { type: mongoose.Schema.Types.ObjectId, ref: "Facility" },
225
+ userId: { type: mongoose.Schema.Types.ObjectId, ref: "User" },
226
+ userRole: { type: String },
227
+ },
228
+ viewHistory: [
229
+ {
230
+ viewedAt: { type: Date, required: true },
231
+ facilityId: { type: mongoose.Schema.Types.ObjectId, ref: "Facility" },
232
+ userId: { type: mongoose.Schema.Types.ObjectId, ref: "User" },
233
+ },
234
+ ],
235
+ },
236
+ invoiceUrl: {
237
+ type: String,
238
+ default: null,
239
+ },
240
+ // eTims tax integration fields
241
+ ticketUrl: {
242
+ type: String,
243
+ default: null,
244
+ description: "QR code URL from eTims/KRA for tax invoice verification",
245
+ },
246
+ txsync: {
247
+ status: {
248
+ type: String,
249
+ enum: ["pending", "synced", "failed", "not-applicable"],
250
+ default: "not-applicable",
251
+ },
252
+ syncedAt: {
253
+ type: Date,
254
+ default: null,
255
+ },
256
+ etimsInvoiceNo: {
257
+ type: String,
258
+ default: null,
259
+ },
260
+ sdcId: {
261
+ type: String,
262
+ default: null,
263
+ },
264
+ receiptNo: {
265
+ type: String,
266
+ default: null,
267
+ },
268
+ errorMessage: {
269
+ type: String,
270
+ default: null,
271
+ },
272
+ attempts: {
273
+ type: Number,
274
+ default: 0,
275
+ },
276
+ lastAttemptAt: {
277
+ type: Date,
278
+ default: null,
279
+ },
280
+ },
281
+ // New fields for double entry accounts
282
+ invoiceDoubleEntryAccount: {
283
+ type: mongoose.Schema.Types.ObjectId,
284
+ ref: "GLAccountDoubleEntries",
285
+ required: false,
286
+ },
287
+ paymentDoubleEntryAccount: {
288
+ type: mongoose.Schema.Types.ObjectId,
289
+ ref: "GLAccountDoubleEntries",
290
+ required: false,
291
+ },
292
+ // GL account details for invoice entries
293
+ accountdebitedData: {
294
+ amount: { type: Number },
295
+ description: { type: String },
296
+ isActive: { type: Boolean, default: true },
297
+ },
298
+ accountcreditedData: {
299
+ amount: { type: Number },
300
+ description: { type: String },
301
+ isActive: { type: Boolean, default: true },
302
+ },
303
+ },
304
+ {
305
+ timestamps: true,
306
+ },
307
+ );
308
+
309
+ // Add indexes for frequently queried fields
310
+ invoiceSchema.index({ accountNumber: 1 });
311
+ invoiceSchema.index({ status: 1 });
312
+ invoiceSchema.index({ "client.clientId": 1, status: 1 });
313
+ invoiceSchema.index({ "reconciliationHistory.paymentReference": 1 });
314
+ invoiceSchema.index({ issueDate: -1 });
315
+ invoiceSchema.index({ "currency.code": 1 }); // Add index for currency code
316
+ invoiceSchema.index({ "currency.id": 1 }); // Add index for currency ID
317
+ invoiceSchema.index({ "currency.code": 1, "client.clientId": 1, status: 1 }); // Compound index for currency-based queries
318
+ invoiceSchema.index({ "client.clientId": 1, balanceBroughtForward: 1 }); // Add index for finding invoices with credit balances
319
+ invoiceSchema.index({ "viewStatus.isOpened": 1 }); // Add index for view status
320
+ invoiceSchema.index({ "viewStatus.openedBy.facilityId": 1 }); // Add index for facility view tracking
321
+ // Add index for double entry accounts
322
+ invoiceSchema.index({ invoiceDoubleEntryAccount: 1 });
323
+ invoiceSchema.index({ paymentDoubleEntryAccount: 1 });
324
+ // NEW INDEXES for new fields
325
+ invoiceSchema.index({ yearMonth: 1 }); // For filtering by year-month
326
+ invoiceSchema.index({ yearMonth: 1, status: 1 }); // Combined index for monthly reports
327
+ invoiceSchema.index({ "notificationsSent.sms": 1 }); // For finding SMS sent/not sent
328
+ invoiceSchema.index({ "notificationsSent.email": 1 }); // For finding email sent/not sent
329
+ // eTims tax sync indexes
330
+ invoiceSchema.index({ "txsync.status": 1 }); // For finding invoices by sync status
331
+ invoiceSchema.index({ "txsync.etimsInvoiceNo": 1 }); // For looking up by eTims invoice number
332
+
333
+ // Add virtual field for calculating balance
334
+ invoiceSchema.virtual("calculatedBalance").get(function () {
335
+ const baseBalance = this.totalAmount - (this.amountPaid || 0);
336
+
337
+ // Add positive balanceBroughtForward (customer owes money)
338
+ if (this.balanceBroughtForward > 0) {
339
+ return baseBalance + this.balanceBroughtForward;
340
+ }
341
+
342
+ // Subtract negative balanceBroughtForward (credit)
343
+ return baseBalance;
344
+ });
345
+
346
+ // Add virtual field for credit balance
347
+ invoiceSchema.virtual("creditBalance").get(function () {
348
+ return this.balanceBroughtForward < 0
349
+ ? Math.abs(this.balanceBroughtForward)
350
+ : 0;
351
+ });
352
+
353
+ // Getter for compatible overpay field
354
+ invoiceSchema.virtual("effectiveOverpay").get(function () {
355
+ return this.balanceBroughtForward < 0
356
+ ? Math.abs(this.balanceBroughtForward)
357
+ : 0;
358
+ });
359
+
360
+ // Add virtual populate for invoice double entry account
361
+ invoiceSchema.virtual("invoiceDoubleEntry", {
362
+ ref: "GLAccountDoubleEntries",
363
+ localField: "invoiceDoubleEntryAccount",
364
+ foreignField: "_id",
365
+ justOne: true,
366
+ });
367
+
368
+ // Add virtual populate for payment double entry account
369
+ invoiceSchema.virtual("paymentDoubleEntry", {
370
+ ref: "GLAccountDoubleEntries",
371
+ localField: "paymentDoubleEntryAccount",
372
+ foreignField: "_id",
373
+ justOne: true,
374
+ });
375
+
376
+ // Add method for currency conversion if needed
377
+ invoiceSchema.methods.convertAmount = function (
378
+ amount,
379
+ fromCurrency,
380
+ toCurrency,
381
+ exchangeRate,
382
+ ) {
383
+ if (fromCurrency === toCurrency) {
384
+ return amount;
385
+ }
386
+ return amount * exchangeRate;
387
+ };
388
+
389
+ // Add static method to find invoices by currency
390
+ invoiceSchema.statics.findByCurrency = function (currencyCode) {
391
+ return this.find({ "currency.code": currencyCode.toUpperCase() });
392
+ };
393
+
394
+ // Add static method to find invoices with credit balance
395
+ invoiceSchema.statics.findWithCreditBalance = function (clientId) {
396
+ return this.find({
397
+ "client.clientId": clientId,
398
+ balanceBroughtForward: { $lt: 0 },
399
+ }).sort({ updatedAt: -1 });
400
+ };
401
+
402
+ // Add static method to calculate totals by currency
403
+ invoiceSchema.statics.calculateTotalsByCurrency = function (query = {}) {
404
+ return this.aggregate([
405
+ { $match: query },
406
+ {
407
+ $group: {
408
+ _id: "$currency.code",
409
+ totalAmount: { $sum: "$totalAmount" },
410
+ totalPaid: { $sum: "$amountPaid" },
411
+ count: { $sum: 1 },
412
+ },
413
+ },
414
+ ]);
415
+ };
416
+
417
+ // New static method to find all unviewed invoices
418
+ invoiceSchema.statics.findUnviewedInvoices = function (facilityId) {
419
+ return this.find({
420
+ "facility.id": facilityId,
421
+ "viewStatus.isOpened": false,
422
+ });
423
+ };
424
+
425
+ // NEW STATIC METHODS for new fields
426
+ // Find invoices by year-month
427
+ invoiceSchema.statics.findByYearMonth = function (yearMonth) {
428
+ return this.find({ yearMonth: yearMonth });
429
+ };
430
+
431
+ // Find invoices where notifications haven't been sent
432
+ invoiceSchema.statics.findPendingNotifications = function (facilityId) {
433
+ return this.find({
434
+ "facility.id": facilityId,
435
+ status: { $in: ["Unpaid", "Overdue"] },
436
+ $or: [
437
+ { "notificationsSent.sms": false },
438
+ { "notificationsSent.email": false },
439
+ ],
440
+ });
441
+ };
442
+
443
+ // Find invoices pending eTims sync
444
+ invoiceSchema.statics.findPendingTaxSync = function (facilityId) {
445
+ return this.find({
446
+ "facility.id": facilityId,
447
+ "txsync.status": "pending",
448
+ });
449
+ };
450
+
451
+ // Find invoices with failed eTims sync
452
+ invoiceSchema.statics.findFailedTaxSync = function (facilityId) {
453
+ return this.find({
454
+ "facility.id": facilityId,
455
+ "txsync.status": "failed",
456
+ });
457
+ };
458
+
459
+ // Pre-save middleware to ensure overpay and balanceBroughtForward stay in sync during transition
460
+ invoiceSchema.pre("save", function (next) {
461
+ // If balanceBroughtForward is negative (credit), sync with overpay for backwards compatibility
462
+ if (this.balanceBroughtForward < 0) {
463
+ this.overpay = Math.abs(this.balanceBroughtForward);
464
+ } else {
465
+ this.overpay = 0; // No overpay if there's no negative balance
466
+ }
467
+
468
+ // NEW: Auto-generate yearMonth from issueDate if not provided
469
+ if (!this.yearMonth && this.issueDate) {
470
+ const year = this.issueDate.getFullYear();
471
+ const month = String(this.issueDate.getMonth() + 1).padStart(2, "0");
472
+ this.yearMonth = `${year}-${month}`;
473
+ }
474
+
475
+ next();
476
+ });
477
+
478
+ const Invoice = mongoose.model("Invoice", invoiceSchema);
479
+
480
+ module.exports = Invoice;