@peopl-health/nexus 3.15.7 → 3.16.1

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.
@@ -15,6 +15,7 @@ const { ScheduledMessage } = require('../models/agendaMessageModel');
15
15
 
16
16
  const { ensureWhatsAppFormat } = require('../helpers/twilioHelper');
17
17
  const { uploadMediaToS3, getFileExtension } = require('../helpers/mediaHelper');
18
+ const { recordDeliveryAttempt } = require('../helpers/deliveryAttemptHelper');
18
19
 
19
20
  const { MessageProvider } = require('../adapters/MessageProvider');
20
21
  const { TwilioMessageAdapter } = require('./messageAdapters/TwilioMessageAdapter');
@@ -140,18 +141,21 @@ class TwilioProvider extends MessageProvider {
140
141
  };
141
142
 
142
143
  let result;
144
+ const kind = messageParams.contentSid ? 'template' : 'freeform';
143
145
  const chunks = messageParams.body?.length > 1600 && !messageParams.mediaUrl && !messageParams.contentSid
144
146
  ? this._splitMessageAtWordBoundaries(messageParams.body) : null;
145
-
147
+
146
148
  if (chunks) {
147
149
  for (let i = 0; i < chunks.length; i++) {
148
150
  result = await this.twilioClient.messages.create({ ...messageParams, body: chunks[i] });
149
151
  await saveMessage(chunks[i], result);
152
+ await recordDeliveryAttempt({ messageData, twilioResult: result, kind });
150
153
  if (i < chunks.length - 1) await new Promise(r => setTimeout(r, 100));
151
154
  }
152
155
  } else {
153
156
  result = await this.twilioClient.messages.create(messageParams);
154
157
  await saveMessage(messageParams.body || messageData.body, result);
158
+ await recordDeliveryAttempt({ messageData, twilioResult: result, kind });
155
159
  }
156
160
 
157
161
  return {
@@ -0,0 +1,46 @@
1
+ const { logger } = require('../utils/logger');
2
+
3
+ const { Message } = require('../models/messageModel');
4
+ const { DeliveryAttempt } = require('../models/deliveryAttemptModel');
5
+
6
+ async function recordDeliveryAttempt({ messageData, twilioResult, kind }) {
7
+ if (messageData?._skipStorage) return null;
8
+ if (!twilioResult?.sid) return null;
9
+
10
+ try {
11
+ const msgDoc = await Message.findOne({ message_id: twilioResult.sid }, '_id').lean();
12
+ if (!msgDoc) {
13
+ logger.warn('[deliveryAttemptHelper] Message not found for SID; skipping attempt record', {
14
+ twilioSid: twilioResult.sid
15
+ });
16
+ return null;
17
+ }
18
+
19
+ const status = twilioResult.status?.toLowerCase() || 'sent';
20
+
21
+ const attempt = await DeliveryAttempt.create({
22
+ messageId: msgDoc._id,
23
+ kind,
24
+ twilioSid: twilioResult.sid,
25
+ contentSid: messageData?.contentSid || null,
26
+ status,
27
+ completedAt: new Date()
28
+ });
29
+
30
+ await Message.updateOne(
31
+ { _id: msgDoc._id },
32
+ { $set: { 'statusInfo.latestDeliveryStatus': status } }
33
+ );
34
+
35
+ return attempt;
36
+ } catch (error) {
37
+ logger.error('[deliveryAttemptHelper] Failed to record delivery attempt', {
38
+ twilioSid: twilioResult?.sid,
39
+ kind,
40
+ error: error.message
41
+ });
42
+ return null;
43
+ }
44
+ }
45
+
46
+ module.exports = { recordDeliveryAttempt };
@@ -13,6 +13,7 @@ async function updateMessageStatus(messageSid, status, errorCode = null, errorMe
13
13
  try {
14
14
  const updateData = {
15
15
  'statusInfo.status': status,
16
+ 'statusInfo.latestDeliveryStatus': status,
16
17
  'statusInfo.updatedAt': new Date(),
17
18
  ...(errorCode && { 'statusInfo.errorCode': errorCode }),
18
19
  ...(errorMessage && { 'statusInfo.errorMessage': errorMessage })
@@ -0,0 +1,41 @@
1
+ const mongoose = require('mongoose');
2
+
3
+ const KIND_VALUES = ['freeform', 'template', 'recovery_template'];
4
+ const STATUS_VALUES = [null, 'queued', 'sending', 'sent', 'delivered', 'undelivered', 'failed', 'read'];
5
+
6
+ const deliveryAttemptSchema = new mongoose.Schema({
7
+ messageId: {
8
+ type: mongoose.Schema.Types.ObjectId,
9
+ ref: 'Message',
10
+ required: true
11
+ },
12
+ kind: {
13
+ type: String,
14
+ enum: KIND_VALUES,
15
+ required: true
16
+ },
17
+ twilioSid: { type: String, default: null },
18
+ contentSid: { type: String, default: null },
19
+ status: {
20
+ type: String,
21
+ enum: STATUS_VALUES,
22
+ default: null
23
+ },
24
+ errorCode: { type: String, default: null },
25
+ errorMessage: { type: String, default: null },
26
+ attemptedAt: { type: Date, default: Date.now, required: true },
27
+ completedAt: { type: Date, default: null },
28
+ raw: { type: mongoose.Schema.Types.Mixed, default: null }
29
+ }, { timestamps: true });
30
+
31
+ deliveryAttemptSchema.index({ messageId: 1, attemptedAt: 1 }, { name: 'message_attempt_history_idx' });
32
+ deliveryAttemptSchema.index({ kind: 1, status: 1, attemptedAt: -1 }, { name: 'analytics_idx' });
33
+ deliveryAttemptSchema.index({ twilioSid: 1 }, { name: 'twilio_sid_idx', sparse: true });
34
+
35
+ const DeliveryAttempt = mongoose.model('DeliveryAttempt', deliveryAttemptSchema);
36
+
37
+ module.exports = {
38
+ DeliveryAttempt,
39
+ DELIVERY_ATTEMPT_KINDS: KIND_VALUES,
40
+ DELIVERY_ATTEMPT_STATUSES: STATUS_VALUES
41
+ };
@@ -8,6 +8,7 @@ const { logger } = require('../utils/logger');
8
8
  const { getClinicalContext } = require('../helpers/patientInformationHelper');
9
9
 
10
10
  const { updateRecordByFilter } = require('../services/airtableService');
11
+ const { DELIVERY_ATTEMPT_STATUSES } = require('./deliveryAttemptModel');
11
12
 
12
13
  const { Thread } = require('./threadModel');
13
14
 
@@ -75,7 +76,7 @@ const messageSchema = new mongoose.Schema({
75
76
  statusInfo: {
76
77
  status: {
77
78
  type: String,
78
- enum: ['queued', 'sending', 'sent', 'delivered', 'undelivered', 'failed', 'read', null],
79
+ enum: DELIVERY_ATTEMPT_STATUSES,
79
80
  default: null
80
81
  },
81
82
  errorCode: { type: String, default: null },
@@ -85,7 +86,12 @@ const messageSchema = new mongoose.Schema({
85
86
  recoveryStartedAt: { type: Date, default: null },
86
87
  recoverySentAt: { type: Date, default: null },
87
88
  recoveryMessageId: { type: String, default: null },
88
- recoveryRejectedAt: { type: Date, default: null }
89
+ recoveryRejectedAt: { type: Date, default: null },
90
+ latestDeliveryStatus: {
91
+ type: String,
92
+ enum: DELIVERY_ATTEMPT_STATUSES,
93
+ default: null
94
+ }
89
95
  },
90
96
  prompt: { type: Object, default: null },
91
97
  preset: { type: Object, default: null },
@@ -106,6 +112,7 @@ messageSchema.index({ nombre_whatsapp: 1, createdAt: -1 }, { name: 'nombre_whats
106
112
  messageSchema.index({ createdAt: -1 }, { name: 'global_sort_idx' });
107
113
 
108
114
  messageSchema.index({ 'statusInfo.recoveryMessageId': 1 }, { name: 'recovery_message_id_idx', sparse: true });
115
+ messageSchema.index({ message_id: 1 }, { name: 'message_id_idx', sparse: true });
109
116
 
110
117
  messageSchema.index({ triggeredBy: 1, createdAt: -1 }, { name: 'triggered_by_idx', sparse: true });
111
118
 
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@peopl-health/nexus",
3
- "version": "3.15.7",
3
+ "version": "3.16.1",
4
4
  "description": "Core messaging and assistant library for WhatsApp communication platforms",
5
5
  "keywords": [
6
6
  "whatsapp",