payservedb 8.9.0 → 8.9.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 (193) hide show
  1. package/.env +2 -2
  2. package/ZOHO_INTEGRATION_SCHEMA.md +644 -644
  3. package/index.js +327 -326
  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 +484 -480
  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 +223 -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_invoice.js +471 -0
  117. package/src/models/power_meter_account.js +81 -81
  118. package/src/models/power_meter_command_logs.js +30 -30
  119. package/src/models/power_meter_command_queue.js +33 -33
  120. package/src/models/power_meter_negative_balance.js +44 -44
  121. package/src/models/power_prepaid_credits.js +47 -47
  122. package/src/models/power_prepaid_debits.js +53 -53
  123. package/src/models/power_prepaid_orders.js +78 -78
  124. package/src/models/power_sms_notification.js +26 -26
  125. package/src/models/privacy_policy.js +19 -19
  126. package/src/models/propertyManagerContract.js +556 -556
  127. package/src/models/propertyManagerRevenue.js +195 -195
  128. package/src/models/purchaseOrderInvoice.js +74 -74
  129. package/src/models/purchase_order.js +213 -213
  130. package/src/models/purchase_request.js +110 -110
  131. package/src/models/quickbooks_config.js +52 -52
  132. package/src/models/refresh_token.js +23 -23
  133. package/src/models/reminder.js +197 -197
  134. package/src/models/report.js +13 -13
  135. package/src/models/resident.js +121 -121
  136. package/src/models/rfq_details.js +131 -131
  137. package/src/models/rfq_response.js +153 -153
  138. package/src/models/service_charge_invoice_upload.js +42 -42
  139. package/src/models/service_charge_payments.js +27 -27
  140. package/src/models/servicerequest.js +55 -55
  141. package/src/models/settings.js +62 -62
  142. package/src/models/short_urls.js +21 -21
  143. package/src/models/smart_meter_daily_consumption.js +44 -44
  144. package/src/models/sms_africastalking.js +20 -20
  145. package/src/models/sms_balance_notification.js +26 -26
  146. package/src/models/sms_meliora.js +20 -20
  147. package/src/models/staff.js +36 -36
  148. package/src/models/stocksandspare.js +161 -161
  149. package/src/models/suppliers.js +74 -74
  150. package/src/models/terms_and_conditions.js +19 -19
  151. package/src/models/tickets.js +186 -186
  152. package/src/models/tickets_category.js +72 -72
  153. package/src/models/unitManagementTemplate.js +44 -44
  154. package/src/models/unitasset.js +25 -25
  155. package/src/models/units.js +130 -130
  156. package/src/models/user.js +186 -186
  157. package/src/models/valueaddedservices.js +21 -21
  158. package/src/models/vas_invoices_upload.js +50 -50
  159. package/src/models/vas_payments.js +24 -24
  160. package/src/models/vasinvoice.js +192 -192
  161. package/src/models/vasvendor.js +52 -52
  162. package/src/models/visitLog.js +95 -95
  163. package/src/models/visitor.js +67 -67
  164. package/src/models/waitlist.js +45 -45
  165. package/src/models/wallet.js +44 -44
  166. package/src/models/wallet_transactions.js +50 -50
  167. package/src/models/water_invoice.js +351 -351
  168. package/src/models/water_meter_Command_Queue.js +33 -33
  169. package/src/models/water_meter_account.js +86 -86
  170. package/src/models/water_meter_billing.js +58 -58
  171. package/src/models/water_meter_combined_accounts.js +92 -92
  172. package/src/models/water_meter_communication.js +17 -17
  173. package/src/models/water_meter_communication_logs.js +39 -39
  174. package/src/models/water_meter_concentrator.js +70 -70
  175. package/src/models/water_meter_daily_history.js +32 -32
  176. package/src/models/water_meter_high_risk.js +36 -36
  177. package/src/models/water_meter_iot_cards.js +34 -34
  178. package/src/models/water_meter_loan_deduction.js +134 -134
  179. package/src/models/water_meter_manufacturer.js +35 -35
  180. package/src/models/water_meter_monthly_history.js +36 -36
  181. package/src/models/water_meter_negative_amounts.js +44 -44
  182. package/src/models/water_meter_settings.js +290 -290
  183. package/src/models/water_meter_single_day_history.js +34 -34
  184. package/src/models/water_meter_size.js +15 -15
  185. package/src/models/water_meters.js +176 -176
  186. package/src/models/water_meters_delivery.js +76 -76
  187. package/src/models/water_prepaid_credit.js +47 -47
  188. package/src/models/water_prepaid_debit.js +50 -50
  189. package/src/models/whatsapp_conversation.js +23 -23
  190. package/src/models/workorder.js +49 -49
  191. package/src/models/zohoAccount.js +453 -453
  192. package/src/models/zohoIntegration.js +262 -262
  193. 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);