payservedb 8.9.8 → 9.0.0

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 (196) hide show
  1. package/.env +2 -2
  2. package/ZOHO_INTEGRATION_SCHEMA.md +644 -644
  3. package/index.js +330 -328
  4. package/package.json +17 -17
  5. package/src/models/InvoiceWithholdingTax.js +68 -0
  6. package/src/models/account.js +52 -52
  7. package/src/models/agent_departments.js +59 -59
  8. package/src/models/agent_notifications.js +53 -53
  9. package/src/models/agent_performance.js +127 -127
  10. package/src/models/agent_roles.js +77 -77
  11. package/src/models/agents.js +154 -154
  12. package/src/models/apilog.js +18 -18
  13. package/src/models/approvalsWorkflows.js +49 -49
  14. package/src/models/archivedapilog.js +18 -18
  15. package/src/models/asset.js +92 -92
  16. package/src/models/assetsAssignment.js +64 -64
  17. package/src/models/auditTrail.js +346 -346
  18. package/src/models/bankdetails.js +47 -47
  19. package/src/models/billerAddress.js +124 -124
  20. package/src/models/booking_invoice.js +165 -165
  21. package/src/models/bookinganalytics.js +63 -63
  22. package/src/models/bookingconfig.js +45 -45
  23. package/src/models/bookingproperty.js +179 -179
  24. package/src/models/bookingreservation.js +239 -239
  25. package/src/models/bookingrevenuerecord.js +84 -84
  26. package/src/models/budget.js +95 -95
  27. package/src/models/budgetCategory.js +19 -19
  28. package/src/models/campaigns.js +108 -108
  29. package/src/models/cashpayment.js +290 -290
  30. package/src/models/combinedUnits.js +62 -62
  31. package/src/models/combined_invoice.js +424 -424
  32. package/src/models/common_area_electricity.js +38 -38
  33. package/src/models/common_area_generator.js +41 -41
  34. package/src/models/common_area_utility_alert.js +37 -37
  35. package/src/models/common_area_water.js +39 -39
  36. package/src/models/communication_status.js +33 -33
  37. package/src/models/communication_user_opt.js +32 -32
  38. package/src/models/community_guidelines.js +35 -35
  39. package/src/models/company.js +53 -53
  40. package/src/models/coreBaseSettings.js +16 -16
  41. package/src/models/coreInvoiceSettings.js +100 -100
  42. package/src/models/counter_schema.js +21 -21
  43. package/src/models/country_tax.js +42 -42
  44. package/src/models/currency_settings.js +39 -39
  45. package/src/models/customer.js +214 -214
  46. package/src/models/customer_preference.js +52 -52
  47. package/src/models/customer_satisfaction_survey.js +297 -297
  48. package/src/models/customer_surveys.js +139 -139
  49. package/src/models/customer_tickets.js +237 -237
  50. package/src/models/dailyChecklist.js +312 -312
  51. package/src/models/default_payment_details.js +17 -17
  52. package/src/models/deliveryTimeMarks.js +18 -18
  53. package/src/models/document_type.js +19 -19
  54. package/src/models/dutyRosterChecklist.js +250 -250
  55. package/src/models/dutyroster.js +136 -136
  56. package/src/models/email.js +37 -37
  57. package/src/models/email_sms_queue.js +61 -61
  58. package/src/models/email_thread.js +35 -35
  59. package/src/models/entry_exit.js +53 -53
  60. package/src/models/expense.js +99 -99
  61. package/src/models/expense_category.js +45 -45
  62. package/src/models/facility.js +76 -76
  63. package/src/models/facilityBillingPrices.js +29 -29
  64. package/src/models/facilityInvoice.js +240 -240
  65. package/src/models/facilityInvoicePayment.js +52 -52
  66. package/src/models/facilityInvoiceRecipient.js +32 -32
  67. package/src/models/facilityWalletTransactionsMetadata.js +236 -236
  68. package/src/models/facility_departements.js +20 -20
  69. package/src/models/facility_etims_config.js +117 -0
  70. package/src/models/facility_payment_details.js +20 -20
  71. package/src/models/facility_rating.js +78 -78
  72. package/src/models/facilityasset.js +25 -25
  73. package/src/models/faq.js +14 -14
  74. package/src/models/gl_account_double_entries.js +25 -25
  75. package/src/models/gl_accounts.js +56 -56
  76. package/src/models/gl_entries.js +49 -49
  77. package/src/models/goodsReceivedNotes.js +115 -115
  78. package/src/models/guard.js +47 -47
  79. package/src/models/handover.js +258 -258
  80. package/src/models/inspection_category.js +38 -38
  81. package/src/models/invoice.js +514 -514
  82. package/src/models/invoice_edit_log.js +81 -81
  83. package/src/models/invoice_generation_approval.js +86 -86
  84. package/src/models/invoicing_schedule.js +40 -40
  85. package/src/models/item_inspection.js +96 -96
  86. package/src/models/knowledge_base.js +109 -109
  87. package/src/models/knowledge_base_rating.js +44 -44
  88. package/src/models/leaseagreement.js +243 -243
  89. package/src/models/leasetemplate.js +17 -17
  90. package/src/models/levy.js +212 -212
  91. package/src/models/levy_invoice_settings.js +26 -26
  92. package/src/models/levycontract.js +215 -215
  93. package/src/models/levytype.js +23 -23
  94. package/src/models/maintenance_service_vendor.js +38 -38
  95. package/src/models/maintenance_services.js +17 -17
  96. package/src/models/maintenancerequisition.js +31 -31
  97. package/src/models/master_workplan.js +32 -32
  98. package/src/models/master_workplan_child.js +34 -34
  99. package/src/models/message.js +38 -38
  100. package/src/models/module.js +21 -21
  101. package/src/models/movein_application.js +29 -29
  102. package/src/models/movein_landlord.js +18 -18
  103. package/src/models/movein_user.js +15 -15
  104. package/src/models/notification.js +44 -44
  105. package/src/models/paymentTermsMarks.js +19 -19
  106. package/src/models/penalty.js +76 -76
  107. package/src/models/pendingCredentials.js +32 -32
  108. package/src/models/powerMeterCommunicationProtocol.js +17 -17
  109. package/src/models/powerMeterCustomerAccount.js +78 -78
  110. package/src/models/powerMeterCustomerBand.js +14 -14
  111. package/src/models/powerMeterDailyReading.js +30 -30
  112. package/src/models/powerMeterGateways.js +40 -40
  113. package/src/models/powerMeterMonthlyReading.js +34 -34
  114. package/src/models/powerMeterPowerCharges.js +85 -85
  115. package/src/models/powerMeterSettings.js +159 -159
  116. package/src/models/powerMeterSingleDayReading.js +32 -32
  117. package/src/models/powerMeters.js +116 -116
  118. package/src/models/powerMetersManufacturer.js +14 -14
  119. package/src/models/power_invoice.js +470 -470
  120. package/src/models/power_meter_account.js +81 -81
  121. package/src/models/power_meter_command_logs.js +30 -30
  122. package/src/models/power_meter_command_queue.js +33 -33
  123. package/src/models/power_meter_negative_balance.js +44 -44
  124. package/src/models/power_prepaid_credits.js +47 -47
  125. package/src/models/power_prepaid_debits.js +53 -53
  126. package/src/models/power_prepaid_orders.js +78 -78
  127. package/src/models/power_sms_notification.js +26 -26
  128. package/src/models/privacy_policy.js +19 -19
  129. package/src/models/propertyManagerContract.js +556 -556
  130. package/src/models/propertyManagerRevenue.js +195 -195
  131. package/src/models/purchaseOrderInvoice.js +74 -74
  132. package/src/models/purchase_order.js +213 -213
  133. package/src/models/purchase_request.js +110 -110
  134. package/src/models/quickbooks_config.js +52 -52
  135. package/src/models/refresh_token.js +23 -23
  136. package/src/models/reminder.js +197 -197
  137. package/src/models/report.js +13 -13
  138. package/src/models/resident.js +121 -121
  139. package/src/models/rfq_details.js +131 -131
  140. package/src/models/rfq_response.js +153 -153
  141. package/src/models/service_charge_invoice_upload.js +42 -42
  142. package/src/models/service_charge_payments.js +27 -27
  143. package/src/models/servicerequest.js +55 -55
  144. package/src/models/settings.js +62 -62
  145. package/src/models/short_urls.js +21 -21
  146. package/src/models/smart_meter_daily_consumption.js +44 -44
  147. package/src/models/sms_africastalking.js +20 -20
  148. package/src/models/sms_balance_notification.js +26 -26
  149. package/src/models/sms_meliora.js +20 -20
  150. package/src/models/staff.js +36 -36
  151. package/src/models/stocksandspare.js +161 -161
  152. package/src/models/suppliers.js +74 -74
  153. package/src/models/terms_and_conditions.js +19 -19
  154. package/src/models/tickets.js +186 -186
  155. package/src/models/tickets_category.js +72 -72
  156. package/src/models/unitManagementTemplate.js +44 -44
  157. package/src/models/unitasset.js +25 -25
  158. package/src/models/units.js +130 -130
  159. package/src/models/user.js +186 -186
  160. package/src/models/valueaddedservices.js +21 -21
  161. package/src/models/vas_invoices_upload.js +50 -50
  162. package/src/models/vas_payments.js +24 -24
  163. package/src/models/vasinvoice.js +196 -192
  164. package/src/models/vasvendor.js +52 -52
  165. package/src/models/visitLog.js +95 -95
  166. package/src/models/visitor.js +67 -67
  167. package/src/models/waitlist.js +45 -45
  168. package/src/models/wallet.js +44 -44
  169. package/src/models/wallet_transactions.js +50 -50
  170. package/src/models/water_invoice.js +351 -351
  171. package/src/models/water_meter_Command_Queue.js +33 -33
  172. package/src/models/water_meter_account.js +86 -86
  173. package/src/models/water_meter_billing.js +58 -58
  174. package/src/models/water_meter_combined_accounts.js +92 -92
  175. package/src/models/water_meter_communication.js +17 -17
  176. package/src/models/water_meter_communication_logs.js +39 -39
  177. package/src/models/water_meter_concentrator.js +70 -70
  178. package/src/models/water_meter_daily_history.js +32 -32
  179. package/src/models/water_meter_high_risk.js +36 -36
  180. package/src/models/water_meter_iot_cards.js +34 -34
  181. package/src/models/water_meter_loan_deduction.js +134 -134
  182. package/src/models/water_meter_manufacturer.js +35 -35
  183. package/src/models/water_meter_monthly_history.js +36 -36
  184. package/src/models/water_meter_negative_amounts.js +44 -44
  185. package/src/models/water_meter_settings.js +290 -290
  186. package/src/models/water_meter_single_day_history.js +34 -34
  187. package/src/models/water_meter_size.js +15 -15
  188. package/src/models/water_meters.js +176 -176
  189. package/src/models/water_meters_delivery.js +76 -76
  190. package/src/models/water_prepaid_credit.js +47 -47
  191. package/src/models/water_prepaid_debit.js +50 -50
  192. package/src/models/whatsapp_conversation.js +23 -23
  193. package/src/models/workorder.js +49 -49
  194. package/src/models/zohoAccount.js +453 -453
  195. package/src/models/zohoIntegration.js +262 -262
  196. package/src/models/zohoItem.js +504 -504
@@ -1,471 +1,471 @@
1
- const mongoose = require('mongoose');
2
-
3
- const powerInvoiceSchema = new mongoose.Schema(
4
- {
5
- // ── Basic Identifiers ──────────────────────────────────────────────────────
6
- invoiceNumber: {
7
- type: String,
8
- required: true,
9
- unique: true,
10
- },
11
- accountNumber: {
12
- type: String,
13
- required: true,
14
- },
15
- unitName: {
16
- type: String,
17
- required: true,
18
- },
19
- yearMonth: {
20
- type: String,
21
- required: true,
22
- // Format: "YYYY-MM" e.g. "2025-03"
23
- },
24
- facilityId: {
25
- type: mongoose.Schema.Types.ObjectId,
26
- required: true,
27
- },
28
- customerId: {
29
- type: mongoose.Schema.Types.ObjectId,
30
- required: true,
31
- },
32
- billingType: {
33
- type: String,
34
- enum: ['postpaid'],
35
- default: 'postpaid',
36
- },
37
- balanceBroughtForward: {
38
- type: Number,
39
- default: 0,
40
- // Positive = arrears carried forward; Negative = credit from overpayment
41
- },
42
-
43
- // ── Payment Methods ────────────────────────────────────────────────────────
44
- paymentMethods: {
45
- mobilePayment: {
46
- status: {
47
- type: Boolean,
48
- default: false,
49
- },
50
- paymentId: {
51
- type: mongoose.Schema.Types.ObjectId,
52
- ref: 'FacilityPaymentDetails',
53
- },
54
- },
55
- bankPayment: {
56
- status: {
57
- type: Boolean,
58
- default: false,
59
- },
60
- paymentId: {
61
- type: mongoose.Schema.Types.ObjectId,
62
- ref: 'BankDetails',
63
- },
64
- },
65
- cashPayment: {
66
- status: {
67
- type: Boolean,
68
- default: false,
69
- },
70
- },
71
- },
72
- paymentMethod: {
73
- type: String,
74
- enum: ['mobile', 'bank', 'cash', 'mixed'],
75
- default: null,
76
- },
77
-
78
- // ── Biller Address ─────────────────────────────────────────────────────────
79
- billerAddress: {
80
- name: {
81
- type: String,
82
- required: [true, 'Biller name is required'],
83
- trim: true,
84
- minlength: [1, 'Biller name must be at least 1 character long'],
85
- },
86
- email: {
87
- type: String,
88
- trim: true,
89
- lowercase: true,
90
- validate: {
91
- validator: function (v) {
92
- return !v || /^\w+([\.-]?\w+)*@\w+([\.-]?\w+)*(\.\w{2,3})+$/.test(v);
93
- },
94
- message: 'Please enter a valid email address',
95
- },
96
- },
97
- phone: {
98
- type: String,
99
- required: [true, 'Biller phone is required'],
100
- trim: true,
101
- },
102
- address: {
103
- type: String,
104
- required: [true, 'Biller address is required'],
105
- trim: true,
106
- },
107
- city: {
108
- type: String,
109
- trim: true,
110
- },
111
- },
112
-
113
- // ── Dates ──────────────────────────────────────────────────────────────────
114
- dateIssued: {
115
- type: Date,
116
- default: Date.now,
117
- },
118
- dueDate: {
119
- type: Date,
120
- required: true,
121
- },
122
-
123
- // ── Customer Info (denormalised snapshot) ──────────────────────────────────
124
- CustomerInfo: {
125
- fullName: {
126
- type: String,
127
- required: true,
128
- trim: true,
129
- },
130
- },
131
-
132
- // ── Meter & Consumption ────────────────────────────────────────────────────
133
- meterNumber: {
134
- type: String,
135
- required: true,
136
- },
137
- tariff: {
138
- type: String,
139
- required: true,
140
- // e.g. "Domestic Lifeline", "Domestic Ordinary 1", "Domestic Ordinary 2",
141
- // "Commercial Low Voltage", "Industrial Low Voltage"
142
- },
143
- meterReadings: {
144
- previousReading: {
145
- type: Number,
146
- required: true,
147
- },
148
- currentReading: {
149
- type: Number,
150
- required: true,
151
- },
152
- usage: {
153
- type: Number,
154
- required: true,
155
- // kWh consumed = currentReading - previousReading
156
- },
157
- },
158
- consumptionPeriod: {
159
- startDate: {
160
- type: Date,
161
- required: true,
162
- },
163
- endDate: {
164
- type: Date,
165
- required: true,
166
- },
167
- },
168
-
169
- // ── Power Charges ──────────────────────────────────────────────────────────
170
- // All monetary values are in the invoice currency (default KES).
171
- powerCharges: {
172
- tariff: {
173
- type: String,
174
- // Mirror of the top-level tariff field; stored here for invoice snapshot
175
- },
176
- yearMonth: {
177
- type: String,
178
- // Mirror of the top-level yearMonth; stored for historical rate lookups
179
- },
180
-
181
- // Core energy charge based on units consumed and applicable tariff block
182
- energyCharge: {
183
- type: Number,
184
- required: true,
185
- default: 0,
186
- },
187
-
188
- // Fuel Cost Charge (FCC) – passed through from EPRA-approved fuel levy
189
- fuelCostCharge: {
190
- type: Number,
191
- default: 0,
192
- },
193
-
194
- // Foreign Exchange Fluctuation Adjustment
195
- forexAdjustment: {
196
- type: Number,
197
- default: 0,
198
- },
199
-
200
- // Inflation Adjustment Charge
201
- inflationAdjustment: {
202
- type: Number,
203
- default: 0,
204
- },
205
-
206
- // Water Resource Management Levy (WRML)
207
- waterResourceManagementLevy: {
208
- type: Number,
209
- default: 0,
210
- },
211
-
212
- // Energy Regulatory Levy (ERL)
213
- energyRegulatoryLevy: {
214
- type: Number,
215
- default: 0,
216
- },
217
-
218
- // Rural Electrification & Renewable Energy Levy (REL)
219
- ruralElectrificationLevy: {
220
- type: Number,
221
- default: 0,
222
- },
223
-
224
- // Power Factor Surcharge (applied to commercial/industrial customers only)
225
- powerFactorSurcharge: {
226
- type: Number,
227
- default: 0,
228
- },
229
-
230
- // Value Added Tax amount
231
- valueAddedTax: {
232
- type: Number,
233
- default: 0,
234
- },
235
-
236
- // VAT rate applied (%) – captured at time of billing; typically 16
237
- vatPercentage: {
238
- type: Number,
239
- default: 16,
240
- },
241
-
242
- // Rural Electrification Programme levy percentage (%) – typically 5
243
- repPercentage: {
244
- type: Number,
245
- default: 5,
246
- },
247
-
248
- // Grand total = sum of all charges above
249
- totalCharge: {
250
- type: Number,
251
- required: true,
252
- },
253
- },
254
-
255
- // ── Payment & Reconciliation ───────────────────────────────────────────────
256
- amountPaid: {
257
- type: Number,
258
- default: 0,
259
- },
260
-
261
- // ── Invoice Note & Status ──────────────────────────────────────────────────
262
- invoiceNote: {
263
- type: String,
264
- default: 'Payment is due within 10 days',
265
- },
266
- status: {
267
- type: String,
268
- required: true,
269
- enum: ['Pending', 'Paid', 'Partially Paid', 'Cancelled', 'Overdue', 'Unpaid'],
270
- default: 'Unpaid',
271
- },
272
-
273
- // ── View / Read Status
274
- viewStatus: {
275
- isOpened: {
276
- type: Boolean,
277
- default: false,
278
- },
279
- openedAt: {
280
- type: Date,
281
- default: null,
282
- },
283
- },
284
-
285
- // ── Currency
286
- currency: {
287
- id: {
288
- type: mongoose.Schema.Types.ObjectId,
289
- },
290
- name: {
291
- type: String,
292
- default: 'Kenyan Shilling',
293
- },
294
- code: {
295
- type: String,
296
- default: 'KES',
297
- },
298
- symbol: {
299
- type: String,
300
- default: 'KSh',
301
- },
302
- },
303
-
304
- // ── Notification Tracking
305
- notificationsSent: {
306
- type: {
307
- sms: {
308
- type: Boolean,
309
- default: false,
310
- },
311
- email: {
312
- type: Boolean,
313
- default: false,
314
- },
315
- sentAt: {
316
- type: Date,
317
- default: null,
318
- },
319
- lastAttempt: {
320
- type: Date,
321
- default: null,
322
- },
323
- attempts: {
324
- type: Number,
325
- default: 0,
326
- },
327
- smsDetails: {
328
- type: {
329
- success: Boolean,
330
- error: String,
331
- sentAt: Date,
332
- phoneNumber: String,
333
- },
334
- default: null,
335
- },
336
- emailDetails: {
337
- type: {
338
- success: Boolean,
339
- error: String,
340
- sentAt: Date,
341
- emailAddress: String,
342
- },
343
- default: null,
344
- },
345
- },
346
- default: {
347
- sms: false,
348
- email: false,
349
- sentAt: null,
350
- lastAttempt: null,
351
- attempts: 0,
352
- smsDetails: null,
353
- emailDetails: null,
354
- },
355
- },
356
-
357
- // Reconciliation History
358
- reconciliationHistory: [
359
- {
360
- date: {
361
- type: Date,
362
- default: Date.now,
363
- },
364
- amount: {
365
- type: Number,
366
- required: true,
367
- },
368
- type: {
369
- type: String,
370
- default: 'Manual',
371
- // e.g. 'Manual', 'M-Pesa', 'Bank Transfer', 'Cash'
372
- },
373
- paymentReference: {
374
- type: String,
375
- },
376
- paymentCompletion: {
377
- type: String,
378
- default: 'Completed',
379
- },
380
- sourceInvoice: {
381
- type: String,
382
- },
383
- destinationInvoice: {
384
- type: String,
385
- },
386
- notes: {
387
- type: String,
388
- },
389
- remainingBalance: {
390
- type: Number,
391
- },
392
- },
393
- ],
394
- },
395
- {
396
- timestamps: true,
397
- }
398
- );
399
-
400
- // Pre-save middleware
401
-
402
- powerInvoiceSchema.pre('save', function (next) {
403
- // 1. Auto-mark overdue
404
- if (this.isModified('dueDate') || this.isNew) {
405
- const today = new Date();
406
- if (this.dueDate < today && this.status === 'Pending') {
407
- this.status = 'Overdue';
408
- }
409
- }
410
-
411
- // 2. Sync powerCharges snapshot fields with top-level fields
412
- if (this.isNew || this.isModified('tariff')) {
413
- if (this.powerCharges && !this.powerCharges.tariff) {
414
- this.powerCharges.tariff = this.tariff;
415
- }
416
- }
417
- if (this.isNew || this.isModified('yearMonth')) {
418
- if (this.powerCharges && !this.powerCharges.yearMonth) {
419
- this.powerCharges.yearMonth = this.yearMonth;
420
- }
421
- }
422
-
423
- // 3. Update payment method flags from reconciliation history
424
- if (this.isModified('reconciliationHistory') || this.isNew) {
425
- if (this.reconciliationHistory && this.reconciliationHistory.length > 0) {
426
- const latestPayment = this.reconciliationHistory[this.reconciliationHistory.length - 1];
427
- const paymentType = latestPayment.type?.toLowerCase();
428
-
429
- if (!this.paymentMethods) {
430
- this.paymentMethods = {
431
- mobilePayment: { status: false },
432
- bankPayment: { status: false },
433
- cashPayment: { status: false },
434
- };
435
- }
436
-
437
- if (
438
- paymentType &&
439
- (paymentType.includes('mobile') ||
440
- paymentType.includes('mpesa') ||
441
- paymentType.includes('m-pesa'))
442
- ) {
443
- this.paymentMethods.mobilePayment.status = true;
444
- this.paymentMethod = 'mobile';
445
- } else if (paymentType && paymentType.includes('bank')) {
446
- this.paymentMethods.bankPayment.status = true;
447
- this.paymentMethod = 'bank';
448
- } else if (paymentType && paymentType.includes('cash')) {
449
- this.paymentMethods.cashPayment.status = true;
450
- this.paymentMethod = 'cash';
451
- } else {
452
- // Default: treat unspecified / manual payments as cash
453
- this.paymentMethods.cashPayment.status = true;
454
- this.paymentMethod = 'cash';
455
- }
456
- }
457
- }
458
-
459
- next();
460
- });
461
-
462
- // Indexes
463
- powerInvoiceSchema.index({ facilityId: 1, yearMonth: -1 });
464
- powerInvoiceSchema.index({ customerId: 1 });
465
- powerInvoiceSchema.index({ status: 1 });
466
- powerInvoiceSchema.index({ meterNumber: 1 });
467
- powerInvoiceSchema.index({ accountNumber: 1 });
468
-
469
- const PowerInvoice = mongoose.model('PowerInvoice', powerInvoiceSchema);
470
-
1
+ const mongoose = require('mongoose');
2
+
3
+ const powerInvoiceSchema = new mongoose.Schema(
4
+ {
5
+ // ── Basic Identifiers ──────────────────────────────────────────────────────
6
+ invoiceNumber: {
7
+ type: String,
8
+ required: true,
9
+ unique: true,
10
+ },
11
+ accountNumber: {
12
+ type: String,
13
+ required: true,
14
+ },
15
+ unitName: {
16
+ type: String,
17
+ required: true,
18
+ },
19
+ yearMonth: {
20
+ type: String,
21
+ required: true,
22
+ // Format: "YYYY-MM" e.g. "2025-03"
23
+ },
24
+ facilityId: {
25
+ type: mongoose.Schema.Types.ObjectId,
26
+ required: true,
27
+ },
28
+ customerId: {
29
+ type: mongoose.Schema.Types.ObjectId,
30
+ required: true,
31
+ },
32
+ billingType: {
33
+ type: String,
34
+ enum: ['postpaid'],
35
+ default: 'postpaid',
36
+ },
37
+ balanceBroughtForward: {
38
+ type: Number,
39
+ default: 0,
40
+ // Positive = arrears carried forward; Negative = credit from overpayment
41
+ },
42
+
43
+ // ── Payment Methods ────────────────────────────────────────────────────────
44
+ paymentMethods: {
45
+ mobilePayment: {
46
+ status: {
47
+ type: Boolean,
48
+ default: false,
49
+ },
50
+ paymentId: {
51
+ type: mongoose.Schema.Types.ObjectId,
52
+ ref: 'FacilityPaymentDetails',
53
+ },
54
+ },
55
+ bankPayment: {
56
+ status: {
57
+ type: Boolean,
58
+ default: false,
59
+ },
60
+ paymentId: {
61
+ type: mongoose.Schema.Types.ObjectId,
62
+ ref: 'BankDetails',
63
+ },
64
+ },
65
+ cashPayment: {
66
+ status: {
67
+ type: Boolean,
68
+ default: false,
69
+ },
70
+ },
71
+ },
72
+ paymentMethod: {
73
+ type: String,
74
+ enum: ['mobile', 'bank', 'cash', 'mixed'],
75
+ default: null,
76
+ },
77
+
78
+ // ── Biller Address ─────────────────────────────────────────────────────────
79
+ billerAddress: {
80
+ name: {
81
+ type: String,
82
+ required: [true, 'Biller name is required'],
83
+ trim: true,
84
+ minlength: [1, 'Biller name must be at least 1 character long'],
85
+ },
86
+ email: {
87
+ type: String,
88
+ trim: true,
89
+ lowercase: true,
90
+ validate: {
91
+ validator: function (v) {
92
+ return !v || /^\w+([\.-]?\w+)*@\w+([\.-]?\w+)*(\.\w{2,3})+$/.test(v);
93
+ },
94
+ message: 'Please enter a valid email address',
95
+ },
96
+ },
97
+ phone: {
98
+ type: String,
99
+ required: [true, 'Biller phone is required'],
100
+ trim: true,
101
+ },
102
+ address: {
103
+ type: String,
104
+ required: [true, 'Biller address is required'],
105
+ trim: true,
106
+ },
107
+ city: {
108
+ type: String,
109
+ trim: true,
110
+ },
111
+ },
112
+
113
+ // ── Dates ──────────────────────────────────────────────────────────────────
114
+ dateIssued: {
115
+ type: Date,
116
+ default: Date.now,
117
+ },
118
+ dueDate: {
119
+ type: Date,
120
+ required: true,
121
+ },
122
+
123
+ // ── Customer Info (denormalised snapshot) ──────────────────────────────────
124
+ CustomerInfo: {
125
+ fullName: {
126
+ type: String,
127
+ required: true,
128
+ trim: true,
129
+ },
130
+ },
131
+
132
+ // ── Meter & Consumption ────────────────────────────────────────────────────
133
+ meterNumber: {
134
+ type: String,
135
+ required: true,
136
+ },
137
+ tariff: {
138
+ type: String,
139
+ required: true,
140
+ // e.g. "Domestic Lifeline", "Domestic Ordinary 1", "Domestic Ordinary 2",
141
+ // "Commercial Low Voltage", "Industrial Low Voltage"
142
+ },
143
+ meterReadings: {
144
+ previousReading: {
145
+ type: Number,
146
+ required: true,
147
+ },
148
+ currentReading: {
149
+ type: Number,
150
+ required: true,
151
+ },
152
+ usage: {
153
+ type: Number,
154
+ required: true,
155
+ // kWh consumed = currentReading - previousReading
156
+ },
157
+ },
158
+ consumptionPeriod: {
159
+ startDate: {
160
+ type: Date,
161
+ required: true,
162
+ },
163
+ endDate: {
164
+ type: Date,
165
+ required: true,
166
+ },
167
+ },
168
+
169
+ // ── Power Charges ──────────────────────────────────────────────────────────
170
+ // All monetary values are in the invoice currency (default KES).
171
+ powerCharges: {
172
+ tariff: {
173
+ type: String,
174
+ // Mirror of the top-level tariff field; stored here for invoice snapshot
175
+ },
176
+ yearMonth: {
177
+ type: String,
178
+ // Mirror of the top-level yearMonth; stored for historical rate lookups
179
+ },
180
+
181
+ // Core energy charge based on units consumed and applicable tariff block
182
+ energyCharge: {
183
+ type: Number,
184
+ required: true,
185
+ default: 0,
186
+ },
187
+
188
+ // Fuel Cost Charge (FCC) – passed through from EPRA-approved fuel levy
189
+ fuelCostCharge: {
190
+ type: Number,
191
+ default: 0,
192
+ },
193
+
194
+ // Foreign Exchange Fluctuation Adjustment
195
+ forexAdjustment: {
196
+ type: Number,
197
+ default: 0,
198
+ },
199
+
200
+ // Inflation Adjustment Charge
201
+ inflationAdjustment: {
202
+ type: Number,
203
+ default: 0,
204
+ },
205
+
206
+ // Water Resource Management Levy (WRML)
207
+ waterResourceManagementLevy: {
208
+ type: Number,
209
+ default: 0,
210
+ },
211
+
212
+ // Energy Regulatory Levy (ERL)
213
+ energyRegulatoryLevy: {
214
+ type: Number,
215
+ default: 0,
216
+ },
217
+
218
+ // Rural Electrification & Renewable Energy Levy (REL)
219
+ ruralElectrificationLevy: {
220
+ type: Number,
221
+ default: 0,
222
+ },
223
+
224
+ // Power Factor Surcharge (applied to commercial/industrial customers only)
225
+ powerFactorSurcharge: {
226
+ type: Number,
227
+ default: 0,
228
+ },
229
+
230
+ // Value Added Tax amount
231
+ valueAddedTax: {
232
+ type: Number,
233
+ default: 0,
234
+ },
235
+
236
+ // VAT rate applied (%) – captured at time of billing; typically 16
237
+ vatPercentage: {
238
+ type: Number,
239
+ default: 16,
240
+ },
241
+
242
+ // Rural Electrification Programme levy percentage (%) – typically 5
243
+ repPercentage: {
244
+ type: Number,
245
+ default: 5,
246
+ },
247
+
248
+ // Grand total = sum of all charges above
249
+ totalCharge: {
250
+ type: Number,
251
+ required: true,
252
+ },
253
+ },
254
+
255
+ // ── Payment & Reconciliation ───────────────────────────────────────────────
256
+ amountPaid: {
257
+ type: Number,
258
+ default: 0,
259
+ },
260
+
261
+ // ── Invoice Note & Status ──────────────────────────────────────────────────
262
+ invoiceNote: {
263
+ type: String,
264
+ default: 'Payment is due within 10 days',
265
+ },
266
+ status: {
267
+ type: String,
268
+ required: true,
269
+ enum: ['Pending', 'Paid', 'Partially Paid', 'Cancelled', 'Overdue', 'Unpaid'],
270
+ default: 'Unpaid',
271
+ },
272
+
273
+ // ── View / Read Status
274
+ viewStatus: {
275
+ isOpened: {
276
+ type: Boolean,
277
+ default: false,
278
+ },
279
+ openedAt: {
280
+ type: Date,
281
+ default: null,
282
+ },
283
+ },
284
+
285
+ // ── Currency
286
+ currency: {
287
+ id: {
288
+ type: mongoose.Schema.Types.ObjectId,
289
+ },
290
+ name: {
291
+ type: String,
292
+ default: 'Kenyan Shilling',
293
+ },
294
+ code: {
295
+ type: String,
296
+ default: 'KES',
297
+ },
298
+ symbol: {
299
+ type: String,
300
+ default: 'KSh',
301
+ },
302
+ },
303
+
304
+ // ── Notification Tracking
305
+ notificationsSent: {
306
+ type: {
307
+ sms: {
308
+ type: Boolean,
309
+ default: false,
310
+ },
311
+ email: {
312
+ type: Boolean,
313
+ default: false,
314
+ },
315
+ sentAt: {
316
+ type: Date,
317
+ default: null,
318
+ },
319
+ lastAttempt: {
320
+ type: Date,
321
+ default: null,
322
+ },
323
+ attempts: {
324
+ type: Number,
325
+ default: 0,
326
+ },
327
+ smsDetails: {
328
+ type: {
329
+ success: Boolean,
330
+ error: String,
331
+ sentAt: Date,
332
+ phoneNumber: String,
333
+ },
334
+ default: null,
335
+ },
336
+ emailDetails: {
337
+ type: {
338
+ success: Boolean,
339
+ error: String,
340
+ sentAt: Date,
341
+ emailAddress: String,
342
+ },
343
+ default: null,
344
+ },
345
+ },
346
+ default: {
347
+ sms: false,
348
+ email: false,
349
+ sentAt: null,
350
+ lastAttempt: null,
351
+ attempts: 0,
352
+ smsDetails: null,
353
+ emailDetails: null,
354
+ },
355
+ },
356
+
357
+ // Reconciliation History
358
+ reconciliationHistory: [
359
+ {
360
+ date: {
361
+ type: Date,
362
+ default: Date.now,
363
+ },
364
+ amount: {
365
+ type: Number,
366
+ required: true,
367
+ },
368
+ type: {
369
+ type: String,
370
+ default: 'Manual',
371
+ // e.g. 'Manual', 'M-Pesa', 'Bank Transfer', 'Cash'
372
+ },
373
+ paymentReference: {
374
+ type: String,
375
+ },
376
+ paymentCompletion: {
377
+ type: String,
378
+ default: 'Completed',
379
+ },
380
+ sourceInvoice: {
381
+ type: String,
382
+ },
383
+ destinationInvoice: {
384
+ type: String,
385
+ },
386
+ notes: {
387
+ type: String,
388
+ },
389
+ remainingBalance: {
390
+ type: Number,
391
+ },
392
+ },
393
+ ],
394
+ },
395
+ {
396
+ timestamps: true,
397
+ }
398
+ );
399
+
400
+ // Pre-save middleware
401
+
402
+ powerInvoiceSchema.pre('save', function (next) {
403
+ // 1. Auto-mark overdue
404
+ if (this.isModified('dueDate') || this.isNew) {
405
+ const today = new Date();
406
+ if (this.dueDate < today && this.status === 'Pending') {
407
+ this.status = 'Overdue';
408
+ }
409
+ }
410
+
411
+ // 2. Sync powerCharges snapshot fields with top-level fields
412
+ if (this.isNew || this.isModified('tariff')) {
413
+ if (this.powerCharges && !this.powerCharges.tariff) {
414
+ this.powerCharges.tariff = this.tariff;
415
+ }
416
+ }
417
+ if (this.isNew || this.isModified('yearMonth')) {
418
+ if (this.powerCharges && !this.powerCharges.yearMonth) {
419
+ this.powerCharges.yearMonth = this.yearMonth;
420
+ }
421
+ }
422
+
423
+ // 3. Update payment method flags from reconciliation history
424
+ if (this.isModified('reconciliationHistory') || this.isNew) {
425
+ if (this.reconciliationHistory && this.reconciliationHistory.length > 0) {
426
+ const latestPayment = this.reconciliationHistory[this.reconciliationHistory.length - 1];
427
+ const paymentType = latestPayment.type?.toLowerCase();
428
+
429
+ if (!this.paymentMethods) {
430
+ this.paymentMethods = {
431
+ mobilePayment: { status: false },
432
+ bankPayment: { status: false },
433
+ cashPayment: { status: false },
434
+ };
435
+ }
436
+
437
+ if (
438
+ paymentType &&
439
+ (paymentType.includes('mobile') ||
440
+ paymentType.includes('mpesa') ||
441
+ paymentType.includes('m-pesa'))
442
+ ) {
443
+ this.paymentMethods.mobilePayment.status = true;
444
+ this.paymentMethod = 'mobile';
445
+ } else if (paymentType && paymentType.includes('bank')) {
446
+ this.paymentMethods.bankPayment.status = true;
447
+ this.paymentMethod = 'bank';
448
+ } else if (paymentType && paymentType.includes('cash')) {
449
+ this.paymentMethods.cashPayment.status = true;
450
+ this.paymentMethod = 'cash';
451
+ } else {
452
+ // Default: treat unspecified / manual payments as cash
453
+ this.paymentMethods.cashPayment.status = true;
454
+ this.paymentMethod = 'cash';
455
+ }
456
+ }
457
+ }
458
+
459
+ next();
460
+ });
461
+
462
+ // Indexes
463
+ powerInvoiceSchema.index({ facilityId: 1, yearMonth: -1 });
464
+ powerInvoiceSchema.index({ customerId: 1 });
465
+ powerInvoiceSchema.index({ status: 1 });
466
+ powerInvoiceSchema.index({ meterNumber: 1 });
467
+ powerInvoiceSchema.index({ accountNumber: 1 });
468
+
469
+ const PowerInvoice = mongoose.model('PowerInvoice', powerInvoiceSchema);
470
+
471
471
  module.exports = PowerInvoice;