payservedb 8.4.6 → 8.4.8

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 (180) hide show
  1. package/.env +2 -2
  2. package/ZOHO_INTEGRATION_SCHEMA.md +644 -644
  3. package/index.js +312 -312
  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 +173 -173
  23. package/src/models/bookingreservation.js +209 -209
  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 +264 -264
  29. package/src/models/combinedUnits.js +62 -62
  30. package/src/models/common_area_electricity.js +38 -38
  31. package/src/models/common_area_generator.js +41 -41
  32. package/src/models/common_area_utility_alert.js +37 -37
  33. package/src/models/common_area_water.js +39 -39
  34. package/src/models/communication_status.js +33 -33
  35. package/src/models/communication_user_opt.js +32 -32
  36. package/src/models/community_guidelines.js +31 -24
  37. package/src/models/company.js +53 -53
  38. package/src/models/coreBaseSettings.js +16 -16
  39. package/src/models/coreInvoiceSettings.js +100 -100
  40. package/src/models/counter_schema.js +21 -21
  41. package/src/models/country_tax.js +42 -42
  42. package/src/models/currency_settings.js +39 -39
  43. package/src/models/customer.js +210 -210
  44. package/src/models/customer_satisfaction_survey.js +278 -278
  45. package/src/models/customer_surveys.js +139 -139
  46. package/src/models/customer_tickets.js +239 -239
  47. package/src/models/dailyChecklist.js +312 -312
  48. package/src/models/default_payment_details.js +17 -17
  49. package/src/models/deliveryTimeMarks.js +18 -18
  50. package/src/models/document_type.js +19 -19
  51. package/src/models/dutyRosterChecklist.js +250 -250
  52. package/src/models/dutyroster.js +136 -136
  53. package/src/models/email.js +37 -37
  54. package/src/models/email_sms_queue.js +61 -61
  55. package/src/models/entry_exit.js +53 -53
  56. package/src/models/expense.js +99 -99
  57. package/src/models/expense_category.js +45 -45
  58. package/src/models/facility.js +71 -71
  59. package/src/models/facilityBillingPrices.js +29 -29
  60. package/src/models/facilityInvoice.js +240 -240
  61. package/src/models/facilityInvoicePayment.js +52 -52
  62. package/src/models/facilityInvoiceRecipient.js +32 -32
  63. package/src/models/facilityWalletTransactionsMetadata.js +236 -236
  64. package/src/models/facility_departements.js +20 -20
  65. package/src/models/facility_payment_details.js +20 -20
  66. package/src/models/facilityasset.js +25 -25
  67. package/src/models/faq.js +14 -14
  68. package/src/models/gl_account_double_entries.js +25 -25
  69. package/src/models/gl_accounts.js +56 -56
  70. package/src/models/gl_entries.js +49 -49
  71. package/src/models/goodsReceivedNotes.js +115 -115
  72. package/src/models/guard.js +47 -47
  73. package/src/models/handover.js +247 -247
  74. package/src/models/inspection_category.js +38 -38
  75. package/src/models/invoice.js +480 -480
  76. package/src/models/invoicing_schedule.js +41 -36
  77. package/src/models/item_inspection.js +96 -96
  78. package/src/models/knowledge_base.js +109 -109
  79. package/src/models/knowledge_base_rating.js +44 -44
  80. package/src/models/leaseagreement.js +236 -236
  81. package/src/models/leasetemplate.js +17 -17
  82. package/src/models/levy.js +223 -223
  83. package/src/models/levy_invoice_settings.js +26 -26
  84. package/src/models/levycontract.js +177 -177
  85. package/src/models/levytype.js +23 -23
  86. package/src/models/maintenance_service_vendor.js +38 -38
  87. package/src/models/maintenance_services.js +17 -17
  88. package/src/models/maintenancerequisition.js +31 -31
  89. package/src/models/master_workplan.js +32 -32
  90. package/src/models/master_workplan_child.js +34 -34
  91. package/src/models/message.js +38 -38
  92. package/src/models/module.js +21 -21
  93. package/src/models/notification.js +44 -44
  94. package/src/models/paymentTermsMarks.js +19 -19
  95. package/src/models/penalty.js +76 -76
  96. package/src/models/pendingCredentials.js +32 -32
  97. package/src/models/powerMeterCommunicationProtocol.js +17 -17
  98. package/src/models/powerMeterCustomerAccount.js +78 -78
  99. package/src/models/powerMeterCustomerBand.js +14 -14
  100. package/src/models/powerMeterDailyReading.js +30 -30
  101. package/src/models/powerMeterGateways.js +40 -40
  102. package/src/models/powerMeterMonthlyReading.js +34 -34
  103. package/src/models/powerMeterPowerCharges.js +85 -85
  104. package/src/models/powerMeterSettings.js +159 -159
  105. package/src/models/powerMeterSingleDayReading.js +32 -32
  106. package/src/models/powerMeters.js +116 -116
  107. package/src/models/powerMetersManufacturer.js +14 -14
  108. package/src/models/power_meter_account.js +81 -81
  109. package/src/models/power_meter_command_logs.js +30 -30
  110. package/src/models/power_meter_command_queue.js +33 -33
  111. package/src/models/power_meter_negative_balance.js +44 -44
  112. package/src/models/power_prepaid_credits.js +47 -47
  113. package/src/models/power_prepaid_debits.js +53 -53
  114. package/src/models/power_prepaid_orders.js +78 -78
  115. package/src/models/power_sms_notification.js +26 -26
  116. package/src/models/privacy_policy.js +19 -19
  117. package/src/models/propertyManagerContract.js +556 -556
  118. package/src/models/propertyManagerRevenue.js +195 -195
  119. package/src/models/purchaseOrderInvoice.js +74 -74
  120. package/src/models/purchase_order.js +213 -213
  121. package/src/models/purchase_request.js +110 -110
  122. package/src/models/refresh_token.js +23 -23
  123. package/src/models/reminder.js +197 -197
  124. package/src/models/report.js +13 -13
  125. package/src/models/resident.js +121 -121
  126. package/src/models/rfq_details.js +131 -131
  127. package/src/models/rfq_response.js +153 -153
  128. package/src/models/service_charge_invoice_upload.js +42 -42
  129. package/src/models/service_charge_payments.js +27 -27
  130. package/src/models/servicerequest.js +55 -55
  131. package/src/models/settings.js +62 -62
  132. package/src/models/short_urls.js +21 -21
  133. package/src/models/smart_meter_daily_consumption.js +44 -44
  134. package/src/models/sms_africastalking.js +20 -20
  135. package/src/models/sms_balance_notification.js +26 -26
  136. package/src/models/sms_meliora.js +20 -20
  137. package/src/models/staff.js +36 -36
  138. package/src/models/stocksandspare.js +161 -161
  139. package/src/models/suppliers.js +74 -74
  140. package/src/models/terms_and_conditions.js +19 -19
  141. package/src/models/tickets.js +186 -186
  142. package/src/models/tickets_category.js +72 -72
  143. package/src/models/unitManagementTemplate.js +44 -44
  144. package/src/models/unitasset.js +25 -25
  145. package/src/models/units.js +118 -118
  146. package/src/models/user.js +186 -186
  147. package/src/models/valueaddedservices.js +79 -79
  148. package/src/models/vas_invoices_upload.js +50 -50
  149. package/src/models/vas_payments.js +24 -24
  150. package/src/models/vasinvoice.js +192 -192
  151. package/src/models/vasvendor.js +57 -57
  152. package/src/models/visitLog.js +95 -95
  153. package/src/models/visitor.js +67 -67
  154. package/src/models/waitlist.js +45 -45
  155. package/src/models/wallet.js +44 -44
  156. package/src/models/wallet_transactions.js +50 -50
  157. package/src/models/water_invoice.js +351 -351
  158. package/src/models/water_meter_Command_Queue.js +33 -33
  159. package/src/models/water_meter_account.js +87 -87
  160. package/src/models/water_meter_billing.js +58 -58
  161. package/src/models/water_meter_communication.js +17 -17
  162. package/src/models/water_meter_communication_logs.js +39 -39
  163. package/src/models/water_meter_concentrator.js +70 -70
  164. package/src/models/water_meter_daily_history.js +32 -32
  165. package/src/models/water_meter_high_risk.js +36 -36
  166. package/src/models/water_meter_iot_cards.js +34 -34
  167. package/src/models/water_meter_manufacturer.js +35 -35
  168. package/src/models/water_meter_monthly_history.js +36 -36
  169. package/src/models/water_meter_negative_amounts.js +44 -44
  170. package/src/models/water_meter_settings.js +287 -287
  171. package/src/models/water_meter_single_day_history.js +34 -34
  172. package/src/models/water_meter_size.js +15 -15
  173. package/src/models/water_meters.js +133 -133
  174. package/src/models/water_meters_delivery.js +76 -76
  175. package/src/models/water_prepaid_credit.js +47 -47
  176. package/src/models/water_prepaid_debit.js +50 -50
  177. package/src/models/workorder.js +49 -49
  178. package/src/models/zohoAccount.js +453 -453
  179. package/src/models/zohoIntegration.js +262 -262
  180. 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;