payservedb 9.1.0 → 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,297 +1,297 @@
1
- const mongoose = require('mongoose');
2
-
3
- const customerSatisfactionSurveySchema = new mongoose.Schema({
4
- ticket_id: {
5
- type: mongoose.Schema.Types.ObjectId,
6
- ref: 'CustomerTicket',
7
- required: true,
8
- unique: true
9
- },
10
- ticket_number: {
11
- type: String,
12
- required: true
13
- },
14
- customer_id: {
15
- type: mongoose.Schema.Types.ObjectId,
16
- ref: 'Customer',
17
- required: true
18
- },
19
- facility_id: {
20
- type: mongoose.Schema.Types.ObjectId,
21
- ref: 'Facility',
22
- required: true
23
- },
24
- assigned_agent_id: {
25
- type: mongoose.Schema.Types.ObjectId,
26
- ref: 'User'
27
- },
28
- custom_caller: {
29
- name: {
30
- type: String,
31
- trim: true,
32
- comment: 'Name of the person who reported the ticket (if different from customer)'
33
- },
34
- phone: {
35
- type: String,
36
- trim: true,
37
- comment: 'Phone number of the custom caller'
38
- },
39
- email: {
40
- type: String,
41
- trim: true,
42
- comment: 'Email of the custom caller'
43
- },
44
- relationship: {
45
- type: String,
46
- trim: true,
47
- comment: 'Relationship to customer (e.g., spouse, agent, caretaker)'
48
- }
49
- },
50
- survey_token: {
51
- type: String,
52
- required: true,
53
- unique: true,
54
- index: true
55
- },
56
- // Survey Responses
57
- overall_satisfaction: {
58
- type: Number,
59
- min: 1,
60
- max: 5,
61
- comment: 'Overall satisfaction rating (1-5 stars)'
62
- },
63
- resolution_quality: {
64
- type: Number,
65
- min: 1,
66
- max: 5,
67
- comment: 'Quality of resolution (1-5 stars)'
68
- },
69
- response_time_satisfaction: {
70
- type: Number,
71
- min: 1,
72
- max: 5,
73
- comment: 'Satisfaction with response time (1-5 stars)'
74
- },
75
- agent_professionalism: {
76
- type: Number,
77
- min: 1,
78
- max: 5,
79
- comment: 'Agent professionalism rating (1-5 stars)'
80
- },
81
- agent_knowledge: {
82
- type: Number,
83
- min: 1,
84
- max: 5,
85
- comment: 'Agent knowledge rating (1-5 stars)'
86
- },
87
- communication_clarity: {
88
- type: Number,
89
- min: 1,
90
- max: 5,
91
- comment: 'Communication clarity rating (1-5 stars)'
92
- },
93
- // Additional 0-5 ratings
94
- facility_maintenance: {
95
- type: Number,
96
- min: 0,
97
- max: 5,
98
- comment: 'Rating for overall facility maintenance (0-5)'
99
- },
100
- value_for_money: {
101
- type: Number,
102
- min: 0,
103
- max: 5,
104
- comment: 'Rating for value for money of services (0-5)'
105
- },
106
- ease_of_contact: {
107
- type: Number,
108
- min: 0,
109
- max: 5,
110
- comment: 'How easy it was to reach support (0-5)'
111
- },
112
- // NPS (Net Promoter Score)
113
- would_recommend: {
114
- type: Number,
115
- min: 0,
116
- max: 10,
117
- comment: 'Likelihood to recommend (0-10 scale)'
118
- },
119
- // Open-ended feedback
120
- positive_feedback: {
121
- type: String,
122
- maxlength: 1000,
123
- comment: 'What did we do well?'
124
- },
125
- improvement_feedback: {
126
- type: String,
127
- maxlength: 1000,
128
- comment: 'What could we improve?'
129
- },
130
- additional_comments: {
131
- type: String,
132
- maxlength: 2000
133
- },
134
- // Issue resolution
135
- issue_fully_resolved: {
136
- type: Boolean,
137
- comment: 'Was the issue fully resolved?'
138
- },
139
- issue_resolution_details: {
140
- type: String,
141
- maxlength: 500,
142
- comment: 'If not fully resolved, please explain'
143
- },
144
- // Metadata
145
- status: {
146
- type: String,
147
- enum: ['pending', 'completed', 'expired'],
148
- default: 'pending'
149
- },
150
- sent_at: {
151
- type: Date,
152
- default: Date.now
153
- },
154
- completed_at: {
155
- type: Date
156
- },
157
- expired_at: {
158
- type: Date,
159
- comment: 'Survey expires 7 days after sending'
160
- },
161
- reminder_sent: {
162
- type: Boolean,
163
- default: false
164
- },
165
- reminder_sent_at: {
166
- type: Date
167
- },
168
- ip_address: String,
169
- user_agent: String,
170
- response_time_seconds: {
171
- type: Number,
172
- comment: 'Time taken to complete survey in seconds'
173
- },
174
- created_at: {
175
- type: Date,
176
- default: Date.now
177
- },
178
- updated_at: {
179
- type: Date,
180
- default: Date.now
181
- }
182
- }, {
183
- timestamps: { createdAt: 'created_at', updatedAt: 'updated_at' }
184
- });
185
-
186
- // Indexes for performance
187
- customerSatisfactionSurveySchema.index({ survey_token: 1 });
188
- customerSatisfactionSurveySchema.index({ customer_id: 1, created_at: -1 });
189
- customerSatisfactionSurveySchema.index({ facility_id: 1, status: 1 });
190
- customerSatisfactionSurveySchema.index({ assigned_agent_id: 1, status: 1 });
191
- customerSatisfactionSurveySchema.index({ status: 1, expired_at: 1 });
192
- customerSatisfactionSurveySchema.index({ ticket_id: 1 });
193
-
194
- // Virtual for calculating NPS category
195
- customerSatisfactionSurveySchema.virtual('nps_category').get(function() {
196
- if (this.would_recommend === null || this.would_recommend === undefined) {
197
- return null;
198
- }
199
- if (this.would_recommend >= 9) return 'promoter';
200
- if (this.would_recommend >= 7) return 'passive';
201
- return 'detractor';
202
- });
203
-
204
- // Virtual for calculating average rating
205
- customerSatisfactionSurveySchema.virtual('average_rating').get(function() {
206
- const ratings = [
207
- this.overall_satisfaction,
208
- this.resolution_quality,
209
- this.response_time_satisfaction,
210
- this.agent_professionalism,
211
- this.agent_knowledge,
212
- this.communication_clarity
213
- ].filter(r => r !== null && r !== undefined);
214
-
215
- if (ratings.length === 0) return null;
216
- return (ratings.reduce((sum, r) => sum + r, 0) / ratings.length).toFixed(2);
217
- });
218
-
219
- // Method to check if survey is expired
220
- customerSatisfactionSurveySchema.methods.isExpired = function() {
221
- return this.expired_at && new Date() > this.expired_at;
222
- };
223
-
224
- // Method to mark survey as completed
225
- customerSatisfactionSurveySchema.methods.markCompleted = function() {
226
- this.status = 'completed';
227
- this.completed_at = new Date();
228
- return this.save();
229
- };
230
-
231
- // Static method to get analytics
232
- customerSatisfactionSurveySchema.statics.getAnalytics = async function(filters = {}) {
233
- const match = { status: 'completed' };
234
-
235
- if (filters.facility_id) match.facility_id = mongoose.Types.ObjectId(filters.facility_id);
236
- if (filters.assigned_agent_id) match.assigned_agent_id = mongoose.Types.ObjectId(filters.assigned_agent_id);
237
- if (filters.start_date) match.completed_at = { $gte: new Date(filters.start_date) };
238
- if (filters.end_date) {
239
- match.completed_at = match.completed_at || {};
240
- match.completed_at.$lte = new Date(filters.end_date);
241
- }
242
-
243
- const analytics = await this.aggregate([
244
- { $match: match },
245
- {
246
- $group: {
247
- _id: null,
248
- total_responses: { $sum: 1 },
249
- avg_overall_satisfaction: { $avg: '$overall_satisfaction' },
250
- avg_resolution_quality: { $avg: '$resolution_quality' },
251
- avg_response_time_satisfaction: { $avg: '$response_time_satisfaction' },
252
- avg_agent_professionalism: { $avg: '$agent_professionalism' },
253
- avg_agent_knowledge: { $avg: '$agent_knowledge' },
254
- avg_communication_clarity: { $avg: '$communication_clarity' },
255
- avg_nps: { $avg: '$would_recommend' },
256
- fully_resolved_count: {
257
- $sum: { $cond: ['$issue_fully_resolved', 1, 0] }
258
- },
259
- promoters: {
260
- $sum: { $cond: [{ $gte: ['$would_recommend', 9] }, 1, 0] }
261
- },
262
- passives: {
263
- $sum: { $cond: [{ $and: [{ $gte: ['$would_recommend', 7] }, { $lt: ['$would_recommend', 9] }] }, 1, 0] }
264
- },
265
- detractors: {
266
- $sum: { $cond: [{ $lt: ['$would_recommend', 7] }, 1, 0] }
267
- }
268
- }
269
- }
270
- ]);
271
-
272
- if (analytics.length === 0) {
273
- return {
274
- total_responses: 0,
275
- avg_overall_satisfaction: 0,
276
- nps_score: 0
277
- };
278
- }
279
-
280
- const result = analytics[0];
281
- const total = result.total_responses;
282
-
283
- // Calculate NPS score: (% Promoters - % Detractors)
284
- const nps_score = total > 0
285
- ? Math.round(((result.promoters - result.detractors) / total) * 100)
286
- : 0;
287
-
288
- return {
289
- ...result,
290
- nps_score,
291
- resolution_rate: total > 0
292
- ? Math.round((result.fully_resolved_count / total) * 100)
293
- : 0
294
- };
295
- };
296
-
297
- module.exports = mongoose.model('CustomerSatisfactionSurvey', customerSatisfactionSurveySchema);
1
+ const mongoose = require('mongoose');
2
+
3
+ const customerSatisfactionSurveySchema = new mongoose.Schema({
4
+ ticket_id: {
5
+ type: mongoose.Schema.Types.ObjectId,
6
+ ref: 'CustomerTicket',
7
+ required: true,
8
+ unique: true
9
+ },
10
+ ticket_number: {
11
+ type: String,
12
+ required: true
13
+ },
14
+ customer_id: {
15
+ type: mongoose.Schema.Types.ObjectId,
16
+ ref: 'Customer',
17
+ required: true
18
+ },
19
+ facility_id: {
20
+ type: mongoose.Schema.Types.ObjectId,
21
+ ref: 'Facility',
22
+ required: true
23
+ },
24
+ assigned_agent_id: {
25
+ type: mongoose.Schema.Types.ObjectId,
26
+ ref: 'User'
27
+ },
28
+ custom_caller: {
29
+ name: {
30
+ type: String,
31
+ trim: true,
32
+ comment: 'Name of the person who reported the ticket (if different from customer)'
33
+ },
34
+ phone: {
35
+ type: String,
36
+ trim: true,
37
+ comment: 'Phone number of the custom caller'
38
+ },
39
+ email: {
40
+ type: String,
41
+ trim: true,
42
+ comment: 'Email of the custom caller'
43
+ },
44
+ relationship: {
45
+ type: String,
46
+ trim: true,
47
+ comment: 'Relationship to customer (e.g., spouse, agent, caretaker)'
48
+ }
49
+ },
50
+ survey_token: {
51
+ type: String,
52
+ required: true,
53
+ unique: true,
54
+ index: true
55
+ },
56
+ // Survey Responses
57
+ overall_satisfaction: {
58
+ type: Number,
59
+ min: 1,
60
+ max: 5,
61
+ comment: 'Overall satisfaction rating (1-5 stars)'
62
+ },
63
+ resolution_quality: {
64
+ type: Number,
65
+ min: 1,
66
+ max: 5,
67
+ comment: 'Quality of resolution (1-5 stars)'
68
+ },
69
+ response_time_satisfaction: {
70
+ type: Number,
71
+ min: 1,
72
+ max: 5,
73
+ comment: 'Satisfaction with response time (1-5 stars)'
74
+ },
75
+ agent_professionalism: {
76
+ type: Number,
77
+ min: 1,
78
+ max: 5,
79
+ comment: 'Agent professionalism rating (1-5 stars)'
80
+ },
81
+ agent_knowledge: {
82
+ type: Number,
83
+ min: 1,
84
+ max: 5,
85
+ comment: 'Agent knowledge rating (1-5 stars)'
86
+ },
87
+ communication_clarity: {
88
+ type: Number,
89
+ min: 1,
90
+ max: 5,
91
+ comment: 'Communication clarity rating (1-5 stars)'
92
+ },
93
+ // Additional 0-5 ratings
94
+ facility_maintenance: {
95
+ type: Number,
96
+ min: 0,
97
+ max: 5,
98
+ comment: 'Rating for overall facility maintenance (0-5)'
99
+ },
100
+ value_for_money: {
101
+ type: Number,
102
+ min: 0,
103
+ max: 5,
104
+ comment: 'Rating for value for money of services (0-5)'
105
+ },
106
+ ease_of_contact: {
107
+ type: Number,
108
+ min: 0,
109
+ max: 5,
110
+ comment: 'How easy it was to reach support (0-5)'
111
+ },
112
+ // NPS (Net Promoter Score)
113
+ would_recommend: {
114
+ type: Number,
115
+ min: 0,
116
+ max: 10,
117
+ comment: 'Likelihood to recommend (0-10 scale)'
118
+ },
119
+ // Open-ended feedback
120
+ positive_feedback: {
121
+ type: String,
122
+ maxlength: 1000,
123
+ comment: 'What did we do well?'
124
+ },
125
+ improvement_feedback: {
126
+ type: String,
127
+ maxlength: 1000,
128
+ comment: 'What could we improve?'
129
+ },
130
+ additional_comments: {
131
+ type: String,
132
+ maxlength: 2000
133
+ },
134
+ // Issue resolution
135
+ issue_fully_resolved: {
136
+ type: Boolean,
137
+ comment: 'Was the issue fully resolved?'
138
+ },
139
+ issue_resolution_details: {
140
+ type: String,
141
+ maxlength: 500,
142
+ comment: 'If not fully resolved, please explain'
143
+ },
144
+ // Metadata
145
+ status: {
146
+ type: String,
147
+ enum: ['pending', 'completed', 'expired'],
148
+ default: 'pending'
149
+ },
150
+ sent_at: {
151
+ type: Date,
152
+ default: Date.now
153
+ },
154
+ completed_at: {
155
+ type: Date
156
+ },
157
+ expired_at: {
158
+ type: Date,
159
+ comment: 'Survey expires 7 days after sending'
160
+ },
161
+ reminder_sent: {
162
+ type: Boolean,
163
+ default: false
164
+ },
165
+ reminder_sent_at: {
166
+ type: Date
167
+ },
168
+ ip_address: String,
169
+ user_agent: String,
170
+ response_time_seconds: {
171
+ type: Number,
172
+ comment: 'Time taken to complete survey in seconds'
173
+ },
174
+ created_at: {
175
+ type: Date,
176
+ default: Date.now
177
+ },
178
+ updated_at: {
179
+ type: Date,
180
+ default: Date.now
181
+ }
182
+ }, {
183
+ timestamps: { createdAt: 'created_at', updatedAt: 'updated_at' }
184
+ });
185
+
186
+ // Indexes for performance
187
+ customerSatisfactionSurveySchema.index({ survey_token: 1 });
188
+ customerSatisfactionSurveySchema.index({ customer_id: 1, created_at: -1 });
189
+ customerSatisfactionSurveySchema.index({ facility_id: 1, status: 1 });
190
+ customerSatisfactionSurveySchema.index({ assigned_agent_id: 1, status: 1 });
191
+ customerSatisfactionSurveySchema.index({ status: 1, expired_at: 1 });
192
+ customerSatisfactionSurveySchema.index({ ticket_id: 1 });
193
+
194
+ // Virtual for calculating NPS category
195
+ customerSatisfactionSurveySchema.virtual('nps_category').get(function() {
196
+ if (this.would_recommend === null || this.would_recommend === undefined) {
197
+ return null;
198
+ }
199
+ if (this.would_recommend >= 9) return 'promoter';
200
+ if (this.would_recommend >= 7) return 'passive';
201
+ return 'detractor';
202
+ });
203
+
204
+ // Virtual for calculating average rating
205
+ customerSatisfactionSurveySchema.virtual('average_rating').get(function() {
206
+ const ratings = [
207
+ this.overall_satisfaction,
208
+ this.resolution_quality,
209
+ this.response_time_satisfaction,
210
+ this.agent_professionalism,
211
+ this.agent_knowledge,
212
+ this.communication_clarity
213
+ ].filter(r => r !== null && r !== undefined);
214
+
215
+ if (ratings.length === 0) return null;
216
+ return (ratings.reduce((sum, r) => sum + r, 0) / ratings.length).toFixed(2);
217
+ });
218
+
219
+ // Method to check if survey is expired
220
+ customerSatisfactionSurveySchema.methods.isExpired = function() {
221
+ return this.expired_at && new Date() > this.expired_at;
222
+ };
223
+
224
+ // Method to mark survey as completed
225
+ customerSatisfactionSurveySchema.methods.markCompleted = function() {
226
+ this.status = 'completed';
227
+ this.completed_at = new Date();
228
+ return this.save();
229
+ };
230
+
231
+ // Static method to get analytics
232
+ customerSatisfactionSurveySchema.statics.getAnalytics = async function(filters = {}) {
233
+ const match = { status: 'completed' };
234
+
235
+ if (filters.facility_id) match.facility_id = mongoose.Types.ObjectId(filters.facility_id);
236
+ if (filters.assigned_agent_id) match.assigned_agent_id = mongoose.Types.ObjectId(filters.assigned_agent_id);
237
+ if (filters.start_date) match.completed_at = { $gte: new Date(filters.start_date) };
238
+ if (filters.end_date) {
239
+ match.completed_at = match.completed_at || {};
240
+ match.completed_at.$lte = new Date(filters.end_date);
241
+ }
242
+
243
+ const analytics = await this.aggregate([
244
+ { $match: match },
245
+ {
246
+ $group: {
247
+ _id: null,
248
+ total_responses: { $sum: 1 },
249
+ avg_overall_satisfaction: { $avg: '$overall_satisfaction' },
250
+ avg_resolution_quality: { $avg: '$resolution_quality' },
251
+ avg_response_time_satisfaction: { $avg: '$response_time_satisfaction' },
252
+ avg_agent_professionalism: { $avg: '$agent_professionalism' },
253
+ avg_agent_knowledge: { $avg: '$agent_knowledge' },
254
+ avg_communication_clarity: { $avg: '$communication_clarity' },
255
+ avg_nps: { $avg: '$would_recommend' },
256
+ fully_resolved_count: {
257
+ $sum: { $cond: ['$issue_fully_resolved', 1, 0] }
258
+ },
259
+ promoters: {
260
+ $sum: { $cond: [{ $gte: ['$would_recommend', 9] }, 1, 0] }
261
+ },
262
+ passives: {
263
+ $sum: { $cond: [{ $and: [{ $gte: ['$would_recommend', 7] }, { $lt: ['$would_recommend', 9] }] }, 1, 0] }
264
+ },
265
+ detractors: {
266
+ $sum: { $cond: [{ $lt: ['$would_recommend', 7] }, 1, 0] }
267
+ }
268
+ }
269
+ }
270
+ ]);
271
+
272
+ if (analytics.length === 0) {
273
+ return {
274
+ total_responses: 0,
275
+ avg_overall_satisfaction: 0,
276
+ nps_score: 0
277
+ };
278
+ }
279
+
280
+ const result = analytics[0];
281
+ const total = result.total_responses;
282
+
283
+ // Calculate NPS score: (% Promoters - % Detractors)
284
+ const nps_score = total > 0
285
+ ? Math.round(((result.promoters - result.detractors) / total) * 100)
286
+ : 0;
287
+
288
+ return {
289
+ ...result,
290
+ nps_score,
291
+ resolution_rate: total > 0
292
+ ? Math.round((result.fully_resolved_count / total) * 100)
293
+ : 0
294
+ };
295
+ };
296
+
297
+ module.exports = mongoose.model('CustomerSatisfactionSurvey', customerSatisfactionSurveySchema);