payservedb 8.9.2 → 8.9.3

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