payservedb 9.1.1 → 9.1.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 (216) hide show
  1. package/.env +2 -2
  2. package/ZOHO_INTEGRATION_SCHEMA.md +644 -644
  3. package/index.js +401 -401
  4. package/package.json +17 -17
  5. package/src/models/InvoiceWithholdingTax.js +67 -67
  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/auto_reply_rule.js +68 -68
  19. package/src/models/bankdetails.js +47 -47
  20. package/src/models/billerAddress.js +124 -124
  21. package/src/models/booking_invoice.js +165 -165
  22. package/src/models/bookinganalytics.js +63 -63
  23. package/src/models/bookingconfig.js +45 -45
  24. package/src/models/bookingproperty.js +179 -179
  25. package/src/models/bookingreservation.js +239 -239
  26. package/src/models/bookingrevenuerecord.js +84 -84
  27. package/src/models/budget.js +95 -95
  28. package/src/models/budgetCategory.js +19 -19
  29. package/src/models/campaigns.js +108 -108
  30. package/src/models/cashpayment.js +290 -290
  31. package/src/models/combinedUnits.js +62 -62
  32. package/src/models/combined_invoice.js +424 -424
  33. package/src/models/common_area_electricity.js +38 -38
  34. package/src/models/common_area_generator.js +41 -41
  35. package/src/models/common_area_utility_alert.js +37 -37
  36. package/src/models/common_area_water.js +39 -39
  37. package/src/models/communication_status.js +33 -33
  38. package/src/models/communication_user_opt.js +32 -32
  39. package/src/models/community_guidelines.js +35 -35
  40. package/src/models/company.js +53 -53
  41. package/src/models/coreBaseSettings.js +16 -16
  42. package/src/models/coreInvoiceSettings.js +100 -100
  43. package/src/models/counter_schema.js +21 -21
  44. package/src/models/country_tax.js +42 -42
  45. package/src/models/currency_settings.js +39 -39
  46. package/src/models/customer.js +234 -234
  47. package/src/models/customer_preference.js +52 -52
  48. package/src/models/customer_satisfaction_survey.js +297 -297
  49. package/src/models/customer_surveys.js +139 -139
  50. package/src/models/customer_tickets.js +237 -237
  51. package/src/models/dailyChecklist.js +312 -312
  52. package/src/models/default_payment_details.js +17 -17
  53. package/src/models/deliveryTimeMarks.js +18 -18
  54. package/src/models/document_type.js +19 -19
  55. package/src/models/dutyRosterChecklist.js +250 -250
  56. package/src/models/dutyroster.js +136 -136
  57. package/src/models/email.js +37 -37
  58. package/src/models/email_cc_config.js +48 -48
  59. package/src/models/email_sms_queue.js +61 -61
  60. package/src/models/email_thread.js +35 -35
  61. package/src/models/entry_exit.js +53 -53
  62. package/src/models/expense.js +99 -99
  63. package/src/models/expense_category.js +45 -45
  64. package/src/models/facility.js +76 -76
  65. package/src/models/facilityBillingPrices.js +29 -29
  66. package/src/models/facilityInvoice.js +240 -240
  67. package/src/models/facilityInvoicePayment.js +52 -52
  68. package/src/models/facilityInvoiceRecipient.js +32 -32
  69. package/src/models/facilityWalletTransactionsMetadata.js +236 -236
  70. package/src/models/facility_departements.js +20 -20
  71. package/src/models/facility_etims_config.js +116 -116
  72. package/src/models/facility_payment_details.js +20 -20
  73. package/src/models/facility_rating.js +78 -78
  74. package/src/models/facilityasset.js +25 -25
  75. package/src/models/faq.js +15 -15
  76. package/src/models/gl_account_double_entries.js +25 -25
  77. package/src/models/gl_accounts.js +56 -56
  78. package/src/models/gl_entries.js +49 -49
  79. package/src/models/goodsReceivedNotes.js +115 -115
  80. package/src/models/guard.js +47 -47
  81. package/src/models/handover.js +258 -258
  82. package/src/models/inspection_category.js +38 -38
  83. package/src/models/invoice.js +528 -525
  84. package/src/models/invoiceCreditAdjustment.js +45 -45
  85. package/src/models/invoice_edit_log.js +81 -81
  86. package/src/models/invoice_generation_approval.js +86 -86
  87. package/src/models/invoicing_schedule.js +40 -40
  88. package/src/models/item_inspection.js +96 -96
  89. package/src/models/knowledge_base.js +109 -109
  90. package/src/models/knowledge_base_rating.js +44 -44
  91. package/src/models/leaseagreement.js +243 -243
  92. package/src/models/leasetemplate.js +17 -17
  93. package/src/models/levy.js +212 -212
  94. package/src/models/levy_invoice_settings.js +26 -26
  95. package/src/models/levycontract.js +215 -215
  96. package/src/models/levytype.js +23 -23
  97. package/src/models/maintenance_service_vendor.js +38 -38
  98. package/src/models/maintenance_services.js +17 -17
  99. package/src/models/maintenancerequisition.js +31 -31
  100. package/src/models/master_workplan.js +32 -32
  101. package/src/models/master_workplan_child.js +34 -34
  102. package/src/models/message.js +38 -38
  103. package/src/models/module.js +21 -21
  104. package/src/models/movein_application.js +29 -29
  105. package/src/models/movein_audit_log.js +21 -21
  106. package/src/models/movein_booking.js +33 -33
  107. package/src/models/movein_commission.js +46 -46
  108. package/src/models/movein_conversation.js +25 -25
  109. package/src/models/movein_deal.js +79 -79
  110. package/src/models/movein_handoff_token.js +16 -16
  111. package/src/models/movein_landlord.js +18 -18
  112. package/src/models/movein_landlord_user.js +20 -20
  113. package/src/models/movein_message.js +27 -27
  114. package/src/models/movein_notification.js +27 -27
  115. package/src/models/movein_otp.js +14 -14
  116. package/src/models/movein_payment.js +46 -46
  117. package/src/models/movein_reminder.js +77 -77
  118. package/src/models/movein_reservation.js +31 -31
  119. package/src/models/movein_unit.js +59 -59
  120. package/src/models/movein_user.js +15 -15
  121. package/src/models/movein_viewing_slot.js +21 -21
  122. package/src/models/notification.js +44 -44
  123. package/src/models/paymentTermsMarks.js +19 -19
  124. package/src/models/penalty.js +76 -76
  125. package/src/models/pendingCredentials.js +32 -32
  126. package/src/models/powerMeterCommunicationProtocol.js +17 -17
  127. package/src/models/powerMeterCustomerAccount.js +78 -78
  128. package/src/models/powerMeterCustomerBand.js +14 -14
  129. package/src/models/powerMeterDailyReading.js +30 -30
  130. package/src/models/powerMeterGateways.js +40 -40
  131. package/src/models/powerMeterMonthlyReading.js +34 -34
  132. package/src/models/powerMeterPowerCharges.js +85 -85
  133. package/src/models/powerMeterSettings.js +200 -200
  134. package/src/models/powerMeterSingleDayReading.js +32 -32
  135. package/src/models/powerMeters.js +149 -149
  136. package/src/models/powerMetersManufacturer.js +14 -14
  137. package/src/models/power_invoice.js +359 -359
  138. package/src/models/power_meter_account.js +81 -81
  139. package/src/models/power_meter_command_logs.js +30 -30
  140. package/src/models/power_meter_command_queue.js +33 -33
  141. package/src/models/power_meter_negative_balance.js +44 -44
  142. package/src/models/power_prepaid_credits.js +47 -47
  143. package/src/models/power_prepaid_debits.js +53 -53
  144. package/src/models/power_prepaid_orders.js +78 -78
  145. package/src/models/power_sms_notification.js +26 -26
  146. package/src/models/privacy_policy.js +19 -19
  147. package/src/models/propertyManagerContract.js +556 -556
  148. package/src/models/propertyManagerRevenue.js +195 -195
  149. package/src/models/purchaseOrderInvoice.js +74 -74
  150. package/src/models/purchase_order.js +213 -213
  151. package/src/models/purchase_request.js +110 -110
  152. package/src/models/quickbooks_config.js +52 -52
  153. package/src/models/recipient_group.js +61 -61
  154. package/src/models/recipient_group_member.js +62 -62
  155. package/src/models/refresh_token.js +23 -23
  156. package/src/models/reminder.js +197 -197
  157. package/src/models/report.js +13 -13
  158. package/src/models/resident.js +121 -121
  159. package/src/models/rfq_details.js +131 -131
  160. package/src/models/rfq_response.js +153 -153
  161. package/src/models/service_charge_invoice_upload.js +42 -42
  162. package/src/models/service_charge_payments.js +27 -27
  163. package/src/models/servicerequest.js +55 -55
  164. package/src/models/settings.js +62 -62
  165. package/src/models/short_urls.js +21 -21
  166. package/src/models/smart_meter_daily_consumption.js +44 -44
  167. package/src/models/sms_africastalking.js +20 -20
  168. package/src/models/sms_balance_notification.js +26 -26
  169. package/src/models/sms_meliora.js +20 -20
  170. package/src/models/staff.js +36 -36
  171. package/src/models/stocksandspare.js +161 -161
  172. package/src/models/suppliers.js +79 -79
  173. package/src/models/terms_and_conditions.js +19 -19
  174. package/src/models/tickets.js +186 -186
  175. package/src/models/tickets_category.js +72 -72
  176. package/src/models/unitManagementTemplate.js +44 -44
  177. package/src/models/unitasset.js +25 -25
  178. package/src/models/units.js +130 -130
  179. package/src/models/user.js +186 -186
  180. package/src/models/valueaddedservices.js +21 -21
  181. package/src/models/vas_invoices_upload.js +50 -50
  182. package/src/models/vas_payments.js +24 -24
  183. package/src/models/vasinvoice.js +196 -196
  184. package/src/models/vasvendor.js +52 -52
  185. package/src/models/visitLog.js +95 -95
  186. package/src/models/visitor.js +67 -67
  187. package/src/models/waitlist.js +45 -45
  188. package/src/models/wallet.js +44 -44
  189. package/src/models/wallet_transactions.js +50 -50
  190. package/src/models/water_invoice.js +351 -351
  191. package/src/models/water_meter_Command_Queue.js +33 -33
  192. package/src/models/water_meter_account.js +86 -86
  193. package/src/models/water_meter_billing.js +58 -58
  194. package/src/models/water_meter_combined_accounts.js +92 -92
  195. package/src/models/water_meter_communication.js +17 -17
  196. package/src/models/water_meter_communication_logs.js +39 -39
  197. package/src/models/water_meter_concentrator.js +70 -70
  198. package/src/models/water_meter_daily_history.js +32 -32
  199. package/src/models/water_meter_high_risk.js +36 -36
  200. package/src/models/water_meter_iot_cards.js +34 -34
  201. package/src/models/water_meter_loan_deduction.js +134 -134
  202. package/src/models/water_meter_manufacturer.js +35 -35
  203. package/src/models/water_meter_monthly_history.js +36 -36
  204. package/src/models/water_meter_negative_amounts.js +44 -44
  205. package/src/models/water_meter_settings.js +290 -290
  206. package/src/models/water_meter_single_day_history.js +34 -34
  207. package/src/models/water_meter_size.js +15 -15
  208. package/src/models/water_meters.js +176 -176
  209. package/src/models/water_meters_delivery.js +76 -76
  210. package/src/models/water_prepaid_credit.js +47 -47
  211. package/src/models/water_prepaid_debit.js +50 -50
  212. package/src/models/whatsapp_conversation.js +23 -23
  213. package/src/models/workorder.js +49 -49
  214. package/src/models/zohoAccount.js +453 -453
  215. package/src/models/zohoIntegration.js +262 -262
  216. package/src/models/zohoItem.js +504 -504
@@ -1,360 +1,360 @@
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
- // Unit reference — stored directly from PowerMeterAccount.unitId / PowerMeters.unitId
20
- unitId: {
21
- type: mongoose.Schema.Types.ObjectId,
22
- ref: 'Unit',
23
- default: null,
24
- },
25
- yearMonth: {
26
- type: String,
27
- required: true,
28
- // Format: "YYYY-MM" e.g. "2025-03"
29
- },
30
- facilityId: {
31
- type: mongoose.Schema.Types.ObjectId,
32
- required: true,
33
- },
34
- customerId: {
35
- type: mongoose.Schema.Types.ObjectId,
36
- required: true,
37
- },
38
- billingType: {
39
- type: String,
40
- enum: ['postpaid'],
41
- default: 'postpaid',
42
- },
43
- balanceBroughtForward: {
44
- type: Number,
45
- default: 0,
46
- // Positive = arrears carried forward; Negative = credit from overpayment
47
- },
48
-
49
- // ── Biller Address ─────────────────────────────────────────────────────────
50
- // Reference to the BillerAddress document configured in PowerMeterSettings.
51
- // The sub-document below is a denormalised snapshot captured at invoice time.
52
- billerAddressId: {
53
- type: mongoose.Schema.Types.ObjectId,
54
- ref: 'BillerAddress',
55
- default: null,
56
- },
57
- billerAddress: {
58
- name: {
59
- type: String,
60
- required: [true, 'Biller name is required'],
61
- trim: true,
62
- minlength: [1, 'Biller name must be at least 1 character long'],
63
- },
64
- email: {
65
- type: String,
66
- trim: true,
67
- lowercase: true,
68
- validate: {
69
- validator: function (v) {
70
- return !v || /^\w+([\.-]?\w+)*@\w+([\.-]?\w+)*(\.\w{2,3})+$/.test(v);
71
- },
72
- message: 'Please enter a valid email address',
73
- },
74
- },
75
- phone: {
76
- type: String,
77
- required: [true, 'Biller phone is required'],
78
- trim: true,
79
- },
80
- address: {
81
- type: String,
82
- required: [true, 'Biller address is required'],
83
- trim: true,
84
- },
85
- city: {
86
- type: String,
87
- trim: true,
88
- },
89
- },
90
-
91
- // ── Payment Methods ────────────────────────────────────────────────────────
92
- // paymentMethods stores the IDs configured in PowerMeterSettings so the
93
- // invoice knows which accounts/paybills are available for payment.
94
- // paymentMethod (singular) records how this invoice was actually settled.
95
- paymentMethods: {
96
- mobilePayment: {
97
- status: {
98
- type: Boolean,
99
- default: false,
100
- },
101
- // References the FacilityPaymentDetails (M-Pesa paybill) configured in settings
102
- paymentId: {
103
- type: mongoose.Schema.Types.ObjectId,
104
- ref: 'FacilityPaymentDetails',
105
- default: null,
106
- },
107
- },
108
- bankPayment: {
109
- status: {
110
- type: Boolean,
111
- default: false,
112
- },
113
- // References the BankDetails configured in settings
114
- paymentId: {
115
- type: mongoose.Schema.Types.ObjectId,
116
- ref: 'BankDetails',
117
- default: null,
118
- },
119
- },
120
- cashPayment: {
121
- status: {
122
- type: Boolean,
123
- default: false,
124
- },
125
- },
126
- },
127
- // How this invoice was actually paid (set during reconciliation)
128
- paymentMethod: {
129
- type: String,
130
- enum: ['mobile', 'bank', 'cash', 'mixed'],
131
- default: null,
132
- },
133
-
134
- // ── Dates ──────────────────────────────────────────────────────────────────
135
- dateIssued: {
136
- type: Date,
137
- default: Date.now,
138
- },
139
- dueDate: {
140
- type: Date,
141
- required: true,
142
- },
143
-
144
- // ── Customer Info (denormalised snapshot) ──────────────────────────────────
145
- CustomerInfo: {
146
- fullName: {
147
- type: String,
148
- required: true,
149
- trim: true,
150
- },
151
- },
152
-
153
- // ── Meter & Consumption ────────────────────────────────────────────────────
154
- meterNumber: {
155
- type: String,
156
- required: true,
157
- },
158
- tariff: {
159
- type: String,
160
- required: true,
161
- // e.g. "Domestic Lifeline", "Domestic Ordinary 1", "Domestic Ordinary 2",
162
- // "Commercial Low Voltage", "Industrial Low Voltage"
163
- },
164
- meterReadings: {
165
- previousReading: {
166
- type: Number,
167
- required: true,
168
- },
169
- currentReading: {
170
- type: Number,
171
- required: true,
172
- },
173
- usage: {
174
- type: Number,
175
- required: true,
176
- // kWh consumed = currentReading - previousReading
177
- },
178
- },
179
- consumptionPeriod: {
180
- startDate: {
181
- type: Date,
182
- required: true,
183
- },
184
- endDate: {
185
- type: Date,
186
- required: true,
187
- },
188
- },
189
-
190
- // ── Power Charges ──────────────────────────────────────────────────────────
191
- powerCharges: {
192
- tariff: { type: String },
193
- yearMonth: { type: String },
194
- energyCharge: { type: Number, required: true, default: 0 },
195
- fuelCostCharge: { type: Number, default: 0 },
196
- forexAdjustment: { type: Number, default: 0 },
197
- inflationAdjustment: { type: Number, default: 0 },
198
- waterResourceManagementLevy: { type: Number, default: 0 },
199
- energyRegulatoryLevy: { type: Number, default: 0 },
200
- ruralElectrificationLevy: { type: Number, default: 0 },
201
- powerFactorSurcharge: { type: Number, default: 0 },
202
- valueAddedTax: { type: Number, default: 0 },
203
- vatPercentage: { type: Number, default: 16 },
204
- repPercentage: { type: Number, default: 5 },
205
- totalCharge: { type: Number, required: true },
206
- },
207
-
208
- // ── Payment & Reconciliation ───────────────────────────────────────────────
209
- amountPaid: {
210
- type: Number,
211
- default: 0,
212
- },
213
-
214
- // ── Invoice Note & Status ──────────────────────────────────────────────────
215
- invoiceNote: {
216
- type: String,
217
- default: 'Payment is due within 10 days',
218
- },
219
- status: {
220
- type: String,
221
- required: true,
222
- enum: ['Pending', 'Paid', 'Partially Paid', 'Cancelled', 'Overdue', 'Unpaid'],
223
- default: 'Unpaid',
224
- },
225
-
226
- // ── View / Read Status ─────────────────────────────────────────────────────
227
- viewStatus: {
228
- isOpened: { type: Boolean, default: false },
229
- openedAt: { type: Date, default: null },
230
- },
231
-
232
- // ── Currency ───────────────────────────────────────────────────────────────
233
- currency: {
234
- id: { type: mongoose.Schema.Types.ObjectId },
235
- name: { type: String, default: 'Kenyan Shilling' },
236
- code: { type: String, default: 'KES' },
237
- symbol: { type: String, default: 'KSh' },
238
- },
239
-
240
- // ── Notification Tracking ──────────────────────────────────────────────────
241
- notificationsSent: {
242
- type: {
243
- sms: { type: Boolean, default: false },
244
- email: { type: Boolean, default: false },
245
- sentAt: { type: Date, default: null },
246
- lastAttempt: { type: Date, default: null },
247
- attempts: { type: Number, default: 0 },
248
- smsDetails: {
249
- type: {
250
- success: Boolean,
251
- error: String,
252
- sentAt: Date,
253
- phoneNumber: String,
254
- },
255
- default: null,
256
- },
257
- emailDetails: {
258
- type: {
259
- success: Boolean,
260
- error: String,
261
- sentAt: Date,
262
- emailAddress: String,
263
- },
264
- default: null,
265
- },
266
- },
267
- default: {
268
- sms: false,
269
- email: false,
270
- sentAt: null,
271
- lastAttempt: null,
272
- attempts: 0,
273
- smsDetails: null,
274
- emailDetails: null,
275
- },
276
- },
277
-
278
- // ── Reconciliation History ─────────────────────────────────────────────────
279
- reconciliationHistory: [
280
- {
281
- date: { type: Date, default: Date.now },
282
- amount: { type: Number, required: true },
283
- type: { type: String, default: 'Manual' },
284
- paymentReference: { type: String },
285
- paymentCompletion: { type: String, default: 'Completed' },
286
- sourceInvoice: { type: String },
287
- destinationInvoice: { type: String },
288
- notes: { type: String },
289
- remainingBalance: { type: Number },
290
- },
291
- ],
292
- },
293
- {
294
- timestamps: true,
295
- }
296
- );
297
-
298
- // ── Pre-save middleware ────────────────────────────────────────────────────────
299
-
300
- powerInvoiceSchema.pre('save', function (next) {
301
- // 1. Auto-mark overdue
302
- if (this.isModified('dueDate') || this.isNew) {
303
- const today = new Date();
304
- if (this.dueDate < today && this.status === 'Pending') {
305
- this.status = 'Overdue';
306
- }
307
- }
308
-
309
- // 2. Sync powerCharges snapshot fields with top-level fields
310
- if ((this.isNew || this.isModified('tariff')) && this.powerCharges && !this.powerCharges.tariff) {
311
- this.powerCharges.tariff = this.tariff;
312
- }
313
- if ((this.isNew || this.isModified('yearMonth')) && this.powerCharges && !this.powerCharges.yearMonth) {
314
- this.powerCharges.yearMonth = this.yearMonth;
315
- }
316
-
317
- // 3. Update payment method flags from reconciliation history
318
- if (this.isModified('reconciliationHistory') || this.isNew) {
319
- if (this.reconciliationHistory && this.reconciliationHistory.length > 0) {
320
- const latestPayment = this.reconciliationHistory[this.reconciliationHistory.length - 1];
321
- const paymentType = latestPayment.type?.toLowerCase();
322
-
323
- if (!this.paymentMethods) {
324
- this.paymentMethods = {
325
- mobilePayment: { status: false },
326
- bankPayment: { status: false },
327
- cashPayment: { status: false },
328
- };
329
- }
330
-
331
- if (paymentType && (paymentType.includes('mobile') || paymentType.includes('mpesa') || paymentType.includes('m-pesa'))) {
332
- this.paymentMethods.mobilePayment.status = true;
333
- this.paymentMethod = 'mobile';
334
- } else if (paymentType && paymentType.includes('bank')) {
335
- this.paymentMethods.bankPayment.status = true;
336
- this.paymentMethod = 'bank';
337
- } else if (paymentType && paymentType.includes('cash')) {
338
- this.paymentMethods.cashPayment.status = true;
339
- this.paymentMethod = 'cash';
340
- } else {
341
- this.paymentMethods.cashPayment.status = true;
342
- this.paymentMethod = 'cash';
343
- }
344
- }
345
- }
346
-
347
- next();
348
- });
349
-
350
- // ── Indexes ────────────────────────────────────────────────────────────────────
351
- powerInvoiceSchema.index({ facilityId: 1, yearMonth: -1 });
352
- powerInvoiceSchema.index({ customerId: 1 });
353
- powerInvoiceSchema.index({ status: 1 });
354
- powerInvoiceSchema.index({ meterNumber: 1 });
355
- powerInvoiceSchema.index({ accountNumber: 1 });
356
- powerInvoiceSchema.index({ unitId: 1 });
357
-
358
- const PowerInvoice = mongoose.model('PowerInvoice', powerInvoiceSchema);
359
-
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
+ // Unit reference — stored directly from PowerMeterAccount.unitId / PowerMeters.unitId
20
+ unitId: {
21
+ type: mongoose.Schema.Types.ObjectId,
22
+ ref: 'Unit',
23
+ default: null,
24
+ },
25
+ yearMonth: {
26
+ type: String,
27
+ required: true,
28
+ // Format: "YYYY-MM" e.g. "2025-03"
29
+ },
30
+ facilityId: {
31
+ type: mongoose.Schema.Types.ObjectId,
32
+ required: true,
33
+ },
34
+ customerId: {
35
+ type: mongoose.Schema.Types.ObjectId,
36
+ required: true,
37
+ },
38
+ billingType: {
39
+ type: String,
40
+ enum: ['postpaid'],
41
+ default: 'postpaid',
42
+ },
43
+ balanceBroughtForward: {
44
+ type: Number,
45
+ default: 0,
46
+ // Positive = arrears carried forward; Negative = credit from overpayment
47
+ },
48
+
49
+ // ── Biller Address ─────────────────────────────────────────────────────────
50
+ // Reference to the BillerAddress document configured in PowerMeterSettings.
51
+ // The sub-document below is a denormalised snapshot captured at invoice time.
52
+ billerAddressId: {
53
+ type: mongoose.Schema.Types.ObjectId,
54
+ ref: 'BillerAddress',
55
+ default: null,
56
+ },
57
+ billerAddress: {
58
+ name: {
59
+ type: String,
60
+ required: [true, 'Biller name is required'],
61
+ trim: true,
62
+ minlength: [1, 'Biller name must be at least 1 character long'],
63
+ },
64
+ email: {
65
+ type: String,
66
+ trim: true,
67
+ lowercase: true,
68
+ validate: {
69
+ validator: function (v) {
70
+ return !v || /^\w+([\.-]?\w+)*@\w+([\.-]?\w+)*(\.\w{2,3})+$/.test(v);
71
+ },
72
+ message: 'Please enter a valid email address',
73
+ },
74
+ },
75
+ phone: {
76
+ type: String,
77
+ required: [true, 'Biller phone is required'],
78
+ trim: true,
79
+ },
80
+ address: {
81
+ type: String,
82
+ required: [true, 'Biller address is required'],
83
+ trim: true,
84
+ },
85
+ city: {
86
+ type: String,
87
+ trim: true,
88
+ },
89
+ },
90
+
91
+ // ── Payment Methods ────────────────────────────────────────────────────────
92
+ // paymentMethods stores the IDs configured in PowerMeterSettings so the
93
+ // invoice knows which accounts/paybills are available for payment.
94
+ // paymentMethod (singular) records how this invoice was actually settled.
95
+ paymentMethods: {
96
+ mobilePayment: {
97
+ status: {
98
+ type: Boolean,
99
+ default: false,
100
+ },
101
+ // References the FacilityPaymentDetails (M-Pesa paybill) configured in settings
102
+ paymentId: {
103
+ type: mongoose.Schema.Types.ObjectId,
104
+ ref: 'FacilityPaymentDetails',
105
+ default: null,
106
+ },
107
+ },
108
+ bankPayment: {
109
+ status: {
110
+ type: Boolean,
111
+ default: false,
112
+ },
113
+ // References the BankDetails configured in settings
114
+ paymentId: {
115
+ type: mongoose.Schema.Types.ObjectId,
116
+ ref: 'BankDetails',
117
+ default: null,
118
+ },
119
+ },
120
+ cashPayment: {
121
+ status: {
122
+ type: Boolean,
123
+ default: false,
124
+ },
125
+ },
126
+ },
127
+ // How this invoice was actually paid (set during reconciliation)
128
+ paymentMethod: {
129
+ type: String,
130
+ enum: ['mobile', 'bank', 'cash', 'mixed'],
131
+ default: null,
132
+ },
133
+
134
+ // ── Dates ──────────────────────────────────────────────────────────────────
135
+ dateIssued: {
136
+ type: Date,
137
+ default: Date.now,
138
+ },
139
+ dueDate: {
140
+ type: Date,
141
+ required: true,
142
+ },
143
+
144
+ // ── Customer Info (denormalised snapshot) ──────────────────────────────────
145
+ CustomerInfo: {
146
+ fullName: {
147
+ type: String,
148
+ required: true,
149
+ trim: true,
150
+ },
151
+ },
152
+
153
+ // ── Meter & Consumption ────────────────────────────────────────────────────
154
+ meterNumber: {
155
+ type: String,
156
+ required: true,
157
+ },
158
+ tariff: {
159
+ type: String,
160
+ required: true,
161
+ // e.g. "Domestic Lifeline", "Domestic Ordinary 1", "Domestic Ordinary 2",
162
+ // "Commercial Low Voltage", "Industrial Low Voltage"
163
+ },
164
+ meterReadings: {
165
+ previousReading: {
166
+ type: Number,
167
+ required: true,
168
+ },
169
+ currentReading: {
170
+ type: Number,
171
+ required: true,
172
+ },
173
+ usage: {
174
+ type: Number,
175
+ required: true,
176
+ // kWh consumed = currentReading - previousReading
177
+ },
178
+ },
179
+ consumptionPeriod: {
180
+ startDate: {
181
+ type: Date,
182
+ required: true,
183
+ },
184
+ endDate: {
185
+ type: Date,
186
+ required: true,
187
+ },
188
+ },
189
+
190
+ // ── Power Charges ──────────────────────────────────────────────────────────
191
+ powerCharges: {
192
+ tariff: { type: String },
193
+ yearMonth: { type: String },
194
+ energyCharge: { type: Number, required: true, default: 0 },
195
+ fuelCostCharge: { type: Number, default: 0 },
196
+ forexAdjustment: { type: Number, default: 0 },
197
+ inflationAdjustment: { type: Number, default: 0 },
198
+ waterResourceManagementLevy: { type: Number, default: 0 },
199
+ energyRegulatoryLevy: { type: Number, default: 0 },
200
+ ruralElectrificationLevy: { type: Number, default: 0 },
201
+ powerFactorSurcharge: { type: Number, default: 0 },
202
+ valueAddedTax: { type: Number, default: 0 },
203
+ vatPercentage: { type: Number, default: 16 },
204
+ repPercentage: { type: Number, default: 5 },
205
+ totalCharge: { type: Number, required: true },
206
+ },
207
+
208
+ // ── Payment & Reconciliation ───────────────────────────────────────────────
209
+ amountPaid: {
210
+ type: Number,
211
+ default: 0,
212
+ },
213
+
214
+ // ── Invoice Note & Status ──────────────────────────────────────────────────
215
+ invoiceNote: {
216
+ type: String,
217
+ default: 'Payment is due within 10 days',
218
+ },
219
+ status: {
220
+ type: String,
221
+ required: true,
222
+ enum: ['Pending', 'Paid', 'Partially Paid', 'Cancelled', 'Overdue', 'Unpaid'],
223
+ default: 'Unpaid',
224
+ },
225
+
226
+ // ── View / Read Status ─────────────────────────────────────────────────────
227
+ viewStatus: {
228
+ isOpened: { type: Boolean, default: false },
229
+ openedAt: { type: Date, default: null },
230
+ },
231
+
232
+ // ── Currency ───────────────────────────────────────────────────────────────
233
+ currency: {
234
+ id: { type: mongoose.Schema.Types.ObjectId },
235
+ name: { type: String, default: 'Kenyan Shilling' },
236
+ code: { type: String, default: 'KES' },
237
+ symbol: { type: String, default: 'KSh' },
238
+ },
239
+
240
+ // ── Notification Tracking ──────────────────────────────────────────────────
241
+ notificationsSent: {
242
+ type: {
243
+ sms: { type: Boolean, default: false },
244
+ email: { type: Boolean, default: false },
245
+ sentAt: { type: Date, default: null },
246
+ lastAttempt: { type: Date, default: null },
247
+ attempts: { type: Number, default: 0 },
248
+ smsDetails: {
249
+ type: {
250
+ success: Boolean,
251
+ error: String,
252
+ sentAt: Date,
253
+ phoneNumber: String,
254
+ },
255
+ default: null,
256
+ },
257
+ emailDetails: {
258
+ type: {
259
+ success: Boolean,
260
+ error: String,
261
+ sentAt: Date,
262
+ emailAddress: String,
263
+ },
264
+ default: null,
265
+ },
266
+ },
267
+ default: {
268
+ sms: false,
269
+ email: false,
270
+ sentAt: null,
271
+ lastAttempt: null,
272
+ attempts: 0,
273
+ smsDetails: null,
274
+ emailDetails: null,
275
+ },
276
+ },
277
+
278
+ // ── Reconciliation History ─────────────────────────────────────────────────
279
+ reconciliationHistory: [
280
+ {
281
+ date: { type: Date, default: Date.now },
282
+ amount: { type: Number, required: true },
283
+ type: { type: String, default: 'Manual' },
284
+ paymentReference: { type: String },
285
+ paymentCompletion: { type: String, default: 'Completed' },
286
+ sourceInvoice: { type: String },
287
+ destinationInvoice: { type: String },
288
+ notes: { type: String },
289
+ remainingBalance: { type: Number },
290
+ },
291
+ ],
292
+ },
293
+ {
294
+ timestamps: true,
295
+ }
296
+ );
297
+
298
+ // ── Pre-save middleware ────────────────────────────────────────────────────────
299
+
300
+ powerInvoiceSchema.pre('save', function (next) {
301
+ // 1. Auto-mark overdue
302
+ if (this.isModified('dueDate') || this.isNew) {
303
+ const today = new Date();
304
+ if (this.dueDate < today && this.status === 'Pending') {
305
+ this.status = 'Overdue';
306
+ }
307
+ }
308
+
309
+ // 2. Sync powerCharges snapshot fields with top-level fields
310
+ if ((this.isNew || this.isModified('tariff')) && this.powerCharges && !this.powerCharges.tariff) {
311
+ this.powerCharges.tariff = this.tariff;
312
+ }
313
+ if ((this.isNew || this.isModified('yearMonth')) && this.powerCharges && !this.powerCharges.yearMonth) {
314
+ this.powerCharges.yearMonth = this.yearMonth;
315
+ }
316
+
317
+ // 3. Update payment method flags from reconciliation history
318
+ if (this.isModified('reconciliationHistory') || this.isNew) {
319
+ if (this.reconciliationHistory && this.reconciliationHistory.length > 0) {
320
+ const latestPayment = this.reconciliationHistory[this.reconciliationHistory.length - 1];
321
+ const paymentType = latestPayment.type?.toLowerCase();
322
+
323
+ if (!this.paymentMethods) {
324
+ this.paymentMethods = {
325
+ mobilePayment: { status: false },
326
+ bankPayment: { status: false },
327
+ cashPayment: { status: false },
328
+ };
329
+ }
330
+
331
+ if (paymentType && (paymentType.includes('mobile') || paymentType.includes('mpesa') || paymentType.includes('m-pesa'))) {
332
+ this.paymentMethods.mobilePayment.status = true;
333
+ this.paymentMethod = 'mobile';
334
+ } else if (paymentType && paymentType.includes('bank')) {
335
+ this.paymentMethods.bankPayment.status = true;
336
+ this.paymentMethod = 'bank';
337
+ } else if (paymentType && paymentType.includes('cash')) {
338
+ this.paymentMethods.cashPayment.status = true;
339
+ this.paymentMethod = 'cash';
340
+ } else {
341
+ this.paymentMethods.cashPayment.status = true;
342
+ this.paymentMethod = 'cash';
343
+ }
344
+ }
345
+ }
346
+
347
+ next();
348
+ });
349
+
350
+ // ── Indexes ────────────────────────────────────────────────────────────────────
351
+ powerInvoiceSchema.index({ facilityId: 1, yearMonth: -1 });
352
+ powerInvoiceSchema.index({ customerId: 1 });
353
+ powerInvoiceSchema.index({ status: 1 });
354
+ powerInvoiceSchema.index({ meterNumber: 1 });
355
+ powerInvoiceSchema.index({ accountNumber: 1 });
356
+ powerInvoiceSchema.index({ unitId: 1 });
357
+
358
+ const PowerInvoice = mongoose.model('PowerInvoice', powerInvoiceSchema);
359
+
360
360
  module.exports = PowerInvoice;