@sendly/cli 3.4.0 ā 3.5.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.
- package/dist/commands/sms/batch.d.ts +31 -2
- package/dist/commands/sms/batch.js +305 -214
- package/dist/commands/status.js +119 -42
- package/dist/lib/api-client.d.ts +13 -0
- package/dist/lib/api-client.js +76 -1
- package/dist/lib/base-command.js +8 -2
- package/oclif.manifest.json +41 -14
- package/package.json +2 -4
|
@@ -1,34 +1,35 @@
|
|
|
1
1
|
import { Flags } from "@oclif/core";
|
|
2
|
-
import { readFileSync } from "fs";
|
|
2
|
+
import { readFileSync, existsSync } from "fs";
|
|
3
|
+
import { basename } from "path";
|
|
3
4
|
import { AuthenticatedCommand } from "../../lib/base-command.js";
|
|
4
5
|
import { apiClient } from "../../lib/api-client.js";
|
|
5
6
|
import { success, error, spinner, colors, json, isJsonMode, } from "../../lib/output.js";
|
|
6
7
|
export default class SmsBatch extends AuthenticatedCommand {
|
|
7
|
-
static description = "Send batch SMS messages";
|
|
8
|
+
static description = "Send batch SMS messages (uploads CSV to cloud for audit trail)";
|
|
8
9
|
static examples = [
|
|
9
|
-
"<%= config.bin %> sms batch --file recipients.
|
|
10
|
-
'<%= config.bin %> sms batch --
|
|
11
|
-
'<%= config.bin %> sms batch --
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
10
|
+
"<%= config.bin %> sms batch --file recipients.csv",
|
|
11
|
+
'<%= config.bin %> sms batch --file phones.csv --text "Hello everyone!"',
|
|
12
|
+
'<%= config.bin %> sms batch --to +15551234567,+15559876543 --text "Hello!"',
|
|
13
|
+
"<%= config.bin %> sms batch --file recipients.csv --dry-run",
|
|
14
|
+
'<%= config.bin %> sms batch --file phones.csv --text "Code: 123" --type transactional',
|
|
15
|
+
"<%= config.bin %> sms batch --reuse abc123-def456",
|
|
16
|
+
"<%= config.bin %> sms batch --history",
|
|
16
17
|
];
|
|
17
18
|
static flags = {
|
|
18
19
|
...AuthenticatedCommand.baseFlags,
|
|
19
20
|
file: Flags.string({
|
|
20
21
|
char: "F",
|
|
21
|
-
description: "
|
|
22
|
-
exclusive: ["to"],
|
|
22
|
+
description: "CSV file with phone numbers (and optional message text)",
|
|
23
|
+
exclusive: ["to", "reuse", "history"],
|
|
23
24
|
}),
|
|
24
25
|
to: Flags.string({
|
|
25
26
|
char: "t",
|
|
26
27
|
description: "Comma-separated recipient phone numbers (E.164 format)",
|
|
27
|
-
exclusive: ["file"],
|
|
28
|
+
exclusive: ["file", "reuse", "history"],
|
|
28
29
|
}),
|
|
29
30
|
text: Flags.string({
|
|
30
31
|
char: "m",
|
|
31
|
-
description: "Message text (
|
|
32
|
+
description: "Message text (required with --to, optional with --file if CSV has text column)",
|
|
32
33
|
}),
|
|
33
34
|
from: Flags.string({
|
|
34
35
|
char: "f",
|
|
@@ -44,44 +45,118 @@ export default class SmsBatch extends AuthenticatedCommand {
|
|
|
44
45
|
description: "Preview batch without sending (validates access, shows cost and compliance breakdown)",
|
|
45
46
|
default: false,
|
|
46
47
|
}),
|
|
48
|
+
reuse: Flags.string({
|
|
49
|
+
description: "Re-use a previous batch upload by ID (see --history)",
|
|
50
|
+
exclusive: ["file", "to", "history"],
|
|
51
|
+
}),
|
|
52
|
+
history: Flags.boolean({
|
|
53
|
+
description: "Show recent batch upload history",
|
|
54
|
+
exclusive: ["file", "to", "reuse"],
|
|
55
|
+
default: false,
|
|
56
|
+
}),
|
|
47
57
|
};
|
|
48
58
|
async run() {
|
|
49
59
|
const { flags } = await this.parse(SmsBatch);
|
|
50
|
-
|
|
51
|
-
|
|
60
|
+
// Handle history subcommand
|
|
61
|
+
if (flags.history) {
|
|
62
|
+
return this.showHistory();
|
|
63
|
+
}
|
|
64
|
+
// Handle reuse subcommand
|
|
65
|
+
if (flags.reuse) {
|
|
66
|
+
return this.reuseBatch(flags.reuse, flags);
|
|
67
|
+
}
|
|
68
|
+
// Handle --to flag (inline recipients, no upload needed)
|
|
69
|
+
if (flags.to) {
|
|
70
|
+
return this.sendInlineRecipients(flags);
|
|
71
|
+
}
|
|
72
|
+
// Handle --file flag (upload to Supabase)
|
|
52
73
|
if (flags.file) {
|
|
53
|
-
|
|
74
|
+
return this.uploadAndSend(flags);
|
|
54
75
|
}
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
76
|
+
error("Either --file, --to, --reuse, or --history is required");
|
|
77
|
+
this.exit(1);
|
|
78
|
+
}
|
|
79
|
+
/**
|
|
80
|
+
* Show batch upload history
|
|
81
|
+
*/
|
|
82
|
+
async showHistory() {
|
|
83
|
+
const spin = spinner("Fetching batch history...").start();
|
|
84
|
+
try {
|
|
85
|
+
const response = await apiClient.get("/api/cli/batch/history", { limit: 15 });
|
|
86
|
+
spin.stop();
|
|
87
|
+
if (isJsonMode()) {
|
|
88
|
+
json(response);
|
|
89
|
+
return;
|
|
90
|
+
}
|
|
91
|
+
if (response.uploads.length === 0) {
|
|
92
|
+
console.log(colors.dim("\nNo batch uploads found.\n"));
|
|
93
|
+
console.log("Upload a CSV file with: sendly sms batch --file your.csv");
|
|
94
|
+
return;
|
|
59
95
|
}
|
|
60
|
-
|
|
96
|
+
console.log(colors.bold("\nš Recent Batch Uploads\n"));
|
|
97
|
+
console.log(colors.dim("ā".repeat(80)));
|
|
98
|
+
for (const upload of response.uploads) {
|
|
99
|
+
const date = new Date(upload.createdAt).toLocaleString();
|
|
100
|
+
const size = this.formatBytes(upload.size);
|
|
101
|
+
const source = upload.metadata?.source === "cli" ? "CLI" : "Web";
|
|
102
|
+
console.log(`${colors.info(upload.id.slice(0, 8))} ${upload.filename}`);
|
|
103
|
+
console.log(colors.dim(` ${date} | ${size} | ${upload.metadata?.validCount || 0} valid recipients | ${source}`));
|
|
104
|
+
}
|
|
105
|
+
console.log(colors.dim("ā".repeat(80)));
|
|
106
|
+
console.log(colors.dim("\nRe-use a batch: sendly sms batch --reuse <id>"));
|
|
61
107
|
}
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
108
|
+
catch (err) {
|
|
109
|
+
spin.stop();
|
|
110
|
+
throw err;
|
|
65
111
|
}
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
112
|
+
}
|
|
113
|
+
/**
|
|
114
|
+
* Re-use a previous batch upload
|
|
115
|
+
*/
|
|
116
|
+
async reuseBatch(uploadId, flags) {
|
|
117
|
+
const spin = spinner("Fetching batch data...").start();
|
|
118
|
+
try {
|
|
119
|
+
const response = await apiClient.get(`/api/cli/batch/reuse/${uploadId}`);
|
|
120
|
+
spin.stop();
|
|
121
|
+
console.log(colors.success(`\nā Loaded batch: ${response.upload.filename}`));
|
|
122
|
+
console.log(colors.dim(` ${response.validation.validCount} valid recipients from ${response.validation.totalRows} rows`));
|
|
123
|
+
// Apply shared text if provided
|
|
124
|
+
let messages = response.recipients;
|
|
125
|
+
if (flags.text) {
|
|
126
|
+
messages = messages.map((r) => ({
|
|
127
|
+
to: r.to,
|
|
128
|
+
text: r.text || flags.text,
|
|
129
|
+
}));
|
|
130
|
+
}
|
|
131
|
+
// Validate all messages have text
|
|
132
|
+
const missingText = messages.filter((m) => !m.text?.trim());
|
|
133
|
+
if (missingText.length > 0) {
|
|
134
|
+
error(`${missingText.length} recipients missing message text. Use --text to provide a default.`);
|
|
135
|
+
this.exit(1);
|
|
136
|
+
}
|
|
137
|
+
// Proceed with preview or send
|
|
138
|
+
await this.previewOrSend(messages, flags, response.upload.id);
|
|
72
139
|
}
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
this.exit(1);
|
|
140
|
+
catch (err) {
|
|
141
|
+
spin.stop();
|
|
142
|
+
throw err;
|
|
77
143
|
}
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
144
|
+
}
|
|
145
|
+
/**
|
|
146
|
+
* Send to inline recipients (--to flag)
|
|
147
|
+
* This path doesn't upload to Supabase since it's just a few numbers
|
|
148
|
+
*/
|
|
149
|
+
async sendInlineRecipients(flags) {
|
|
150
|
+
if (!flags.text) {
|
|
151
|
+
error("--text is required when using --to");
|
|
82
152
|
this.exit(1);
|
|
83
153
|
}
|
|
84
|
-
|
|
154
|
+
const phones = flags.to.split(",").map((p) => p.trim());
|
|
155
|
+
const messages = phones.map((phone) => ({
|
|
156
|
+
to: phone.startsWith("+") ? phone : `+${phone}`,
|
|
157
|
+
text: flags.text,
|
|
158
|
+
}));
|
|
159
|
+
// Validate phone numbers
|
|
85
160
|
for (const msg of messages) {
|
|
86
161
|
if (!/^\+[1-9]\d{1,14}$/.test(msg.to)) {
|
|
87
162
|
error(`Invalid phone number: ${msg.to}`, {
|
|
@@ -89,119 +164,98 @@ export default class SmsBatch extends AuthenticatedCommand {
|
|
|
89
164
|
});
|
|
90
165
|
this.exit(1);
|
|
91
166
|
}
|
|
92
|
-
if (!msg.text?.trim()) {
|
|
93
|
-
error(`Empty message text for ${msg.to}`, {
|
|
94
|
-
hint: "Use --text to provide a shared message for all recipients",
|
|
95
|
-
});
|
|
96
|
-
this.exit(1);
|
|
97
|
-
}
|
|
98
167
|
}
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
}
|
|
142
|
-
}
|
|
143
|
-
// Compliance check results
|
|
144
|
-
if (preview.compliance) {
|
|
145
|
-
console.log(colors.bold("\nš”ļø Compliance Check:\n"));
|
|
146
|
-
console.log(` Message Type: ${preview.compliance.messageType.toUpperCase()}`);
|
|
147
|
-
if (preview.compliance.shaftBlocked > 0) {
|
|
148
|
-
console.log(colors.error(` SHAFT Blocked: ${preview.compliance.shaftBlocked} messages`));
|
|
149
|
-
for (const msg of preview.compliance.shaftBlockedMessages.slice(0, 3)) {
|
|
150
|
-
console.log(colors.dim(` āā ${msg.to}: ${msg.category} (${msg.matchedTerms.join(", ")})`));
|
|
151
|
-
}
|
|
152
|
-
if (preview.compliance.shaftBlockedMessages.length > 3) {
|
|
153
|
-
console.log(colors.dim(` ... and ${preview.compliance.shaftBlockedMessages.length - 3} more`));
|
|
154
|
-
}
|
|
155
|
-
}
|
|
156
|
-
else {
|
|
157
|
-
console.log(colors.success(" SHAFT Check: ā All messages pass content filter"));
|
|
158
|
-
}
|
|
159
|
-
if (preview.compliance.messageType === "marketing") {
|
|
160
|
-
if (preview.compliance.quietHoursRescheduled > 0) {
|
|
161
|
-
console.log(colors.warning(` Quiet Hours: ${preview.compliance.quietHoursRescheduled} messages will be rescheduled`));
|
|
162
|
-
for (const msg of preview.compliance.quietHoursBlockedMessages.slice(0, 3)) {
|
|
163
|
-
const nextTime = msg.nextAllowedTime
|
|
164
|
-
? new Date(msg.nextAllowedTime).toLocaleString()
|
|
165
|
-
: "next available window";
|
|
166
|
-
console.log(colors.dim(` āā ${msg.to}: ${msg.recipientTimezone} ā ${nextTime}`));
|
|
167
|
-
}
|
|
168
|
-
if (preview.compliance.quietHoursBlockedMessages.length > 3) {
|
|
169
|
-
console.log(colors.dim(` ... and ${preview.compliance.quietHoursBlockedMessages.length - 3} more`));
|
|
170
|
-
}
|
|
171
|
-
}
|
|
172
|
-
else {
|
|
173
|
-
console.log(colors.success(" Quiet Hours: ā All recipients within allowed hours"));
|
|
174
|
-
}
|
|
175
|
-
}
|
|
176
|
-
else {
|
|
177
|
-
console.log(colors.dim(" Quiet Hours: Bypassed (transactional message)"));
|
|
178
|
-
}
|
|
179
|
-
}
|
|
180
|
-
// Warnings
|
|
181
|
-
if (preview.warnings.length > 0) {
|
|
182
|
-
console.log(colors.warning("\nā ļø Warnings:"));
|
|
183
|
-
for (const w of preview.warnings) {
|
|
184
|
-
console.log(` ⢠${w}`);
|
|
185
|
-
}
|
|
168
|
+
await this.previewOrSend(messages, flags);
|
|
169
|
+
}
|
|
170
|
+
/**
|
|
171
|
+
* Upload CSV to Supabase and send
|
|
172
|
+
*/
|
|
173
|
+
async uploadAndSend(flags) {
|
|
174
|
+
const filePath = flags.file;
|
|
175
|
+
// Check file exists
|
|
176
|
+
if (!existsSync(filePath)) {
|
|
177
|
+
error(`File not found: ${filePath}`);
|
|
178
|
+
this.exit(1);
|
|
179
|
+
}
|
|
180
|
+
// Validate file extension
|
|
181
|
+
if (!filePath.endsWith(".csv")) {
|
|
182
|
+
error("Only CSV files are supported", {
|
|
183
|
+
hint: "Convert your file to CSV format with columns: phone,message",
|
|
184
|
+
});
|
|
185
|
+
this.exit(1);
|
|
186
|
+
}
|
|
187
|
+
// Read file
|
|
188
|
+
const buffer = readFileSync(filePath);
|
|
189
|
+
const filename = basename(filePath);
|
|
190
|
+
// Check file size (5MB limit)
|
|
191
|
+
if (buffer.length > 5 * 1024 * 1024) {
|
|
192
|
+
error("File too large (max 5MB)", {
|
|
193
|
+
hint: "Split your CSV into smaller files",
|
|
194
|
+
});
|
|
195
|
+
this.exit(1);
|
|
196
|
+
}
|
|
197
|
+
const spin = spinner(`Uploading ${filename}...`).start();
|
|
198
|
+
try {
|
|
199
|
+
// Upload to Supabase via CLI endpoint
|
|
200
|
+
const response = await apiClient.uploadFile("/api/cli/batch/upload", { buffer, filename, mimetype: "text/csv" });
|
|
201
|
+
spin.stop();
|
|
202
|
+
// Show upload summary
|
|
203
|
+
console.log(colors.success(`\nā Uploaded: ${response.upload.filename}`));
|
|
204
|
+
console.log(colors.dim(` Upload ID: ${response.upload.id}`));
|
|
205
|
+
console.log(colors.dim(` ${response.validation.validCount} valid / ${response.validation.invalidCount} invalid / ${response.validation.duplicatesRemoved} duplicates removed`));
|
|
206
|
+
// Show validation errors if any
|
|
207
|
+
if (response.errors.length > 0) {
|
|
208
|
+
console.log(colors.warning(`\nā ļø Validation errors:`));
|
|
209
|
+
for (const err of response.errors.slice(0, 5)) {
|
|
210
|
+
console.log(colors.dim(` Row ${err.row}: ${err.phone} - ${err.error}`));
|
|
186
211
|
}
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
console.log(colors.error(`\nā Blocked Messages (${preview.blockedMessages.length} total):`));
|
|
190
|
-
for (const b of preview.blockedMessages.slice(0, 5)) {
|
|
191
|
-
console.log(` ${b.to}: ${b.reason}`);
|
|
192
|
-
}
|
|
193
|
-
if (preview.blockedMessages.length > 5) {
|
|
194
|
-
console.log(colors.dim(` ... and ${preview.blockedMessages.length - 5} more`));
|
|
195
|
-
}
|
|
212
|
+
if (response.hasMoreErrors) {
|
|
213
|
+
console.log(colors.dim(` ... and ${response.validation.invalidCount - 5} more errors`));
|
|
196
214
|
}
|
|
197
|
-
console.log("\n" + colors.dim("No messages were sent. Remove --dry-run to send."));
|
|
198
|
-
return;
|
|
199
215
|
}
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
216
|
+
if (response.recipients.length === 0) {
|
|
217
|
+
error("No valid recipients found in CSV");
|
|
218
|
+
this.exit(1);
|
|
219
|
+
}
|
|
220
|
+
// Apply shared text if provided
|
|
221
|
+
let messages = response.recipients;
|
|
222
|
+
if (flags.text) {
|
|
223
|
+
messages = messages.map((r) => ({
|
|
224
|
+
to: r.to,
|
|
225
|
+
text: r.text || flags.text,
|
|
226
|
+
}));
|
|
227
|
+
}
|
|
228
|
+
// Validate all messages have text
|
|
229
|
+
const missingText = messages.filter((m) => !m.text?.trim());
|
|
230
|
+
if (missingText.length > 0) {
|
|
231
|
+
error(`${missingText.length} recipients missing message text. Use --text to provide a default.`);
|
|
232
|
+
console.log(colors.dim("\nCSV should have columns: phone,message OR use --text flag"));
|
|
233
|
+
this.exit(1);
|
|
203
234
|
}
|
|
235
|
+
// Proceed with preview or send
|
|
236
|
+
await this.previewOrSend(messages, flags, response.upload.id);
|
|
204
237
|
}
|
|
238
|
+
catch (err) {
|
|
239
|
+
spin.stop();
|
|
240
|
+
throw err;
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
/**
|
|
244
|
+
* Preview or send the batch
|
|
245
|
+
*/
|
|
246
|
+
async previewOrSend(messages, flags, uploadId) {
|
|
247
|
+
// Check batch size
|
|
248
|
+
if (messages.length > 1000) {
|
|
249
|
+
error("Batch size cannot exceed 1000 messages", {
|
|
250
|
+
hint: "Split your messages into smaller batches",
|
|
251
|
+
});
|
|
252
|
+
this.exit(1);
|
|
253
|
+
}
|
|
254
|
+
// Handle dry-run mode
|
|
255
|
+
if (flags["dry-run"]) {
|
|
256
|
+
return this.showPreview(messages, flags);
|
|
257
|
+
}
|
|
258
|
+
// Send the batch
|
|
205
259
|
const spin = spinner(`Sending ${messages.length} messages...`);
|
|
206
260
|
spin.start();
|
|
207
261
|
try {
|
|
@@ -209,6 +263,7 @@ export default class SmsBatch extends AuthenticatedCommand {
|
|
|
209
263
|
messages,
|
|
210
264
|
messageType: flags.type,
|
|
211
265
|
...(flags.from && { from: flags.from }),
|
|
266
|
+
...(uploadId && { uploadId }), // Link to file upload for audit trail
|
|
212
267
|
});
|
|
213
268
|
spin.stop();
|
|
214
269
|
if (isJsonMode()) {
|
|
@@ -262,84 +317,120 @@ export default class SmsBatch extends AuthenticatedCommand {
|
|
|
262
317
|
throw err;
|
|
263
318
|
}
|
|
264
319
|
}
|
|
265
|
-
|
|
320
|
+
/**
|
|
321
|
+
* Show batch preview (dry-run mode)
|
|
322
|
+
*/
|
|
323
|
+
async showPreview(messages, flags) {
|
|
324
|
+
const spin = spinner("Analyzing batch...").start();
|
|
266
325
|
try {
|
|
267
|
-
const
|
|
268
|
-
|
|
269
|
-
if (
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
326
|
+
const preview = await apiClient.post("/api/v1/messages/batch/preview", { messages, messageType: flags.type });
|
|
327
|
+
spin.stop();
|
|
328
|
+
if (isJsonMode()) {
|
|
329
|
+
json(preview);
|
|
330
|
+
return;
|
|
331
|
+
}
|
|
332
|
+
// Show comprehensive preview
|
|
333
|
+
console.log(colors.bold("\nš Batch Preview (Dry Run)\n"));
|
|
334
|
+
// Summary table
|
|
335
|
+
console.log(colors.dim("ā".repeat(50)));
|
|
336
|
+
console.log(`Total messages: ${preview.total}`);
|
|
337
|
+
console.log(`Sendable: ${colors.success(String(preview.sendable))}`);
|
|
338
|
+
console.log(`Blocked: ${preview.blocked > 0 ? colors.error(String(preview.blocked)) : "0"}`);
|
|
339
|
+
console.log(`Duplicates removed: ${preview.duplicates}`);
|
|
340
|
+
console.log(colors.dim("ā".repeat(50)));
|
|
341
|
+
// Credits
|
|
342
|
+
console.log(`\nCredits needed: ${preview.creditsNeeded}`);
|
|
343
|
+
console.log(`Your balance: ${preview.creditBalance}`);
|
|
344
|
+
if (!preview.hasSufficientCredits) {
|
|
345
|
+
console.log(colors.error(`ā ļø Insufficient credits! Need ${preview.creditsNeeded - preview.creditBalance} more.`));
|
|
346
|
+
}
|
|
347
|
+
// Access info
|
|
348
|
+
console.log(`\nAPI Key type: ${preview.keyType.toUpperCase()}`);
|
|
349
|
+
console.log(`Write access: ${preview.hasWriteScope ? "ā" : "ā"}`);
|
|
350
|
+
console.log(`Domestic (US/CA): ${preview.messagingProfile.canSendDomestic ? "ā" : "ā"}`);
|
|
351
|
+
console.log(`International: ${preview.messagingProfile.canSendInternational ? "ā" : "ā"}`);
|
|
352
|
+
// Country breakdown
|
|
353
|
+
const countries = Object.entries(preview.byCountry);
|
|
354
|
+
if (countries.length > 0) {
|
|
355
|
+
console.log(colors.bold("\nš By Country:\n"));
|
|
356
|
+
for (const [country, data] of countries) {
|
|
357
|
+
const status = data.allowed ? colors.success("ā") : colors.error("ā");
|
|
358
|
+
console.log(` ${status} ${country}: ${data.count} msgs, ${data.credits} credits (${data.tier})`);
|
|
359
|
+
if (!data.allowed && data.blockedReason) {
|
|
360
|
+
console.log(colors.dim(` āā ${data.blockedReason}`));
|
|
361
|
+
}
|
|
279
362
|
}
|
|
280
|
-
error("Invalid JSON format", {
|
|
281
|
-
hint: "Expected array of {to, text} objects or {messages: [...]}",
|
|
282
|
-
});
|
|
283
|
-
this.exit(1);
|
|
284
363
|
}
|
|
285
|
-
//
|
|
286
|
-
if (
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
if (phone && (phone.startsWith("+") || /^\d/.test(phone))) {
|
|
310
|
-
messages.push({
|
|
311
|
-
to: phone.startsWith("+") ? phone : `+${phone}`,
|
|
312
|
-
text: parts.length >= 2
|
|
313
|
-
? parts.slice(1).join(",").trim().replace(/"/g, "") ||
|
|
314
|
-
undefined
|
|
315
|
-
: undefined,
|
|
316
|
-
});
|
|
364
|
+
// Compliance check results
|
|
365
|
+
if (preview.compliance) {
|
|
366
|
+
console.log(colors.bold("\nš”ļø Compliance Check:\n"));
|
|
367
|
+
console.log(` Message Type: ${preview.compliance.messageType.toUpperCase()}`);
|
|
368
|
+
if (preview.compliance.shaftBlocked > 0) {
|
|
369
|
+
console.log(colors.error(` SHAFT Blocked: ${preview.compliance.shaftBlocked} messages`));
|
|
370
|
+
for (const msg of preview.compliance.shaftBlockedMessages.slice(0, 3)) {
|
|
371
|
+
console.log(colors.dim(` āā ${msg.to}: ${msg.category} (${msg.matchedTerms.join(", ")})`));
|
|
372
|
+
}
|
|
373
|
+
if (preview.compliance.shaftBlockedMessages.length > 3) {
|
|
374
|
+
console.log(colors.dim(` ... and ${preview.compliance.shaftBlockedMessages.length - 3} more`));
|
|
375
|
+
}
|
|
376
|
+
}
|
|
377
|
+
else {
|
|
378
|
+
console.log(colors.success(" SHAFT Check: ā All messages pass content filter"));
|
|
379
|
+
}
|
|
380
|
+
if (preview.compliance.messageType === "marketing") {
|
|
381
|
+
if (preview.compliance.quietHoursRescheduled > 0) {
|
|
382
|
+
console.log(colors.warning(` Quiet Hours: ${preview.compliance.quietHoursRescheduled} messages will be rescheduled`));
|
|
383
|
+
for (const msg of preview.compliance.quietHoursBlockedMessages.slice(0, 3)) {
|
|
384
|
+
const nextTime = msg.nextAllowedTime
|
|
385
|
+
? new Date(msg.nextAllowedTime).toLocaleString()
|
|
386
|
+
: "next available window";
|
|
387
|
+
console.log(colors.dim(` āā ${msg.to}: ${msg.recipientTimezone} ā ${nextTime}`));
|
|
317
388
|
}
|
|
389
|
+
if (preview.compliance.quietHoursBlockedMessages.length > 3) {
|
|
390
|
+
console.log(colors.dim(` ... and ${preview.compliance.quietHoursBlockedMessages.length - 3} more`));
|
|
391
|
+
}
|
|
392
|
+
}
|
|
393
|
+
else {
|
|
394
|
+
console.log(colors.success(" Quiet Hours: ā All recipients within allowed hours"));
|
|
318
395
|
}
|
|
319
396
|
}
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
hint: "Use .json or .csv file",
|
|
324
|
-
});
|
|
325
|
-
this.exit(1);
|
|
326
|
-
}
|
|
327
|
-
catch (err) {
|
|
328
|
-
if (err.code === "ENOENT") {
|
|
329
|
-
error(`File not found: ${filePath}`);
|
|
397
|
+
else {
|
|
398
|
+
console.log(colors.dim(" Quiet Hours: Bypassed (transactional message)"));
|
|
399
|
+
}
|
|
330
400
|
}
|
|
331
|
-
|
|
332
|
-
|
|
401
|
+
// Warnings
|
|
402
|
+
if (preview.warnings.length > 0) {
|
|
403
|
+
console.log(colors.warning("\nā ļø Warnings:"));
|
|
404
|
+
for (const w of preview.warnings) {
|
|
405
|
+
console.log(` ⢠${w}`);
|
|
406
|
+
}
|
|
333
407
|
}
|
|
334
|
-
|
|
335
|
-
|
|
408
|
+
// Blocked messages (first 5)
|
|
409
|
+
if (preview.blockedMessages.length > 0) {
|
|
410
|
+
console.log(colors.error(`\nā Blocked Messages (${preview.blockedMessages.length} total):`));
|
|
411
|
+
for (const b of preview.blockedMessages.slice(0, 5)) {
|
|
412
|
+
console.log(` ${b.to}: ${b.reason}`);
|
|
413
|
+
}
|
|
414
|
+
if (preview.blockedMessages.length > 5) {
|
|
415
|
+
console.log(colors.dim(` ... and ${preview.blockedMessages.length - 5} more`));
|
|
416
|
+
}
|
|
336
417
|
}
|
|
337
|
-
|
|
418
|
+
console.log("\n" + colors.dim("No messages were sent. Remove --dry-run to send."));
|
|
419
|
+
}
|
|
420
|
+
catch (err) {
|
|
421
|
+
spin.stop();
|
|
422
|
+
throw err;
|
|
338
423
|
}
|
|
339
424
|
}
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
|
|
425
|
+
/**
|
|
426
|
+
* Format bytes to human readable string
|
|
427
|
+
*/
|
|
428
|
+
formatBytes(bytes) {
|
|
429
|
+
if (bytes < 1024)
|
|
430
|
+
return `${bytes} B`;
|
|
431
|
+
if (bytes < 1024 * 1024)
|
|
432
|
+
return `${(bytes / 1024).toFixed(1)} KB`;
|
|
433
|
+
return `${(bytes / (1024 * 1024)).toFixed(1)} MB`;
|
|
343
434
|
}
|
|
344
435
|
}
|
|
345
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"batch.js","sourceRoot":"","sources":["../../../src/commands/sms/batch.ts"],"names":[],"mappings":"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","sourcesContent":["import { Flags } from \"@oclif/core\";\nimport { readFileSync } from \"fs\";\nimport { AuthenticatedCommand } from \"../../lib/base-command.js\";\nimport { apiClient } from \"../../lib/api-client.js\";\nimport {\n  success,\n  error,\n  spinner,\n  colors,\n  json,\n  isJsonMode,\n} from \"../../lib/output.js\";\n\ninterface BatchMessage {\n  to: string;\n  text?: string; // Optional - can be provided via --text flag\n}\n\ninterface BatchResponse {\n  batchId: string;\n  total: number;\n  sent: number;\n  queued: number;\n  failed: number;\n  creditsUsed: number;\n  creditsRefunded: number;\n  status: string;\n  messages?: Array<{\n    index: number;\n    id: string;\n    to: string;\n    status: string;\n    error?: string;\n  }>;\n}\n\ninterface BatchPreviewResponse {\n  total: number;\n  sendable: number;\n  blocked: number;\n  duplicates: number;\n  creditsNeeded: number;\n  creditBalance: number;\n  hasSufficientCredits: boolean;\n  keyType: \"test\" | \"live\";\n  keyScopes: string[];\n  hasWriteScope: boolean;\n  messagingProfile: {\n    id: string | null;\n    canSendDomestic: boolean;\n    canSendInternational: boolean;\n    verificationStatus: string | null;\n    verificationType: string | null;\n  };\n  byCountry: Record<\n    string,\n    {\n      count: number;\n      credits: number;\n      tier: string;\n      allowed: boolean;\n      blockedReason?: string;\n    }\n  >;\n  blockedMessages: Array<{\n    index: number;\n    to: string;\n    reason: string;\n  }>;\n  compliance: {\n    messageType: \"marketing\" | \"transactional\";\n    shaftBlocked: number;\n    quietHoursBlocked: number;\n    quietHoursRescheduled: number;\n    shaftBlockedMessages: Array<{\n      index: number;\n      to: string;\n      category: string;\n      matchedTerms: string[];\n    }>;\n    quietHoursBlockedMessages: Array<{\n      index: number;\n      to: string;\n      recipientTimezone: string;\n      recipientLocalTime: string;\n      nextAllowedTime?: string;\n    }>;\n  };\n  warnings: string[];\n}\n\nexport default class SmsBatch extends AuthenticatedCommand {\n  static description = \"Send batch SMS messages\";\n\n  static examples = [\n    \"<%= config.bin %> sms batch --file recipients.json\",\n    '<%= config.bin %> sms batch --to +15551234567,+15559876543 --text \"Hello everyone!\"',\n    '<%= config.bin %> sms batch --file phones.csv --text \"Your order is ready!\"',\n    '<%= config.bin %> sms batch --file recipients.csv --from \"Sendly\"',\n    \"<%= config.bin %> sms batch --file messages.json --json\",\n    '<%= config.bin %> sms batch --file phones.csv --text \"Hi\" --dry-run',\n    '<%= config.bin %> sms batch --file phones.csv --text \"Your code: 123\" --type transactional',\n  ];\n\n  static flags = {\n    ...AuthenticatedCommand.baseFlags,\n    file: Flags.string({\n      char: \"F\",\n      description: \"JSON file with messages array [{to, text}, ...]\",\n      exclusive: [\"to\"],\n    }),\n    to: Flags.string({\n      char: \"t\",\n      description: \"Comma-separated recipient phone numbers (E.164 format)\",\n      exclusive: [\"file\"],\n    }),\n    text: Flags.string({\n      char: \"m\",\n      description:\n        \"Message text (works with --to or --file for phone-only lists)\",\n    }),\n    from: Flags.string({\n      char: \"f\",\n      description: \"Sender ID or phone number for all messages\",\n    }),\n    type: Flags.string({\n      description:\n        \"Message type: marketing (default) or transactional. Transactional bypasses quiet hours.\",\n      options: [\"marketing\", \"transactional\"],\n      default: \"marketing\",\n    }),\n    \"dry-run\": Flags.boolean({\n      char: \"d\",\n      description:\n        \"Preview batch without sending (validates access, shows cost and compliance breakdown)\",\n      default: false,\n    }),\n  };\n\n  async run(): Promise<void> {\n    const { flags } = await this.parse(SmsBatch);\n\n    let messages: BatchMessage[] = [];\n\n    // Parse messages from file or flags\n    if (flags.file) {\n      messages = this.parseMessagesFromFile(flags.file);\n    } else if (flags.to) {\n      if (!flags.text) {\n        error(\"--text is required when using --to\");\n        this.exit(1);\n      }\n      messages = this.parseMessagesFromFlags(flags.to, flags.text);\n    } else {\n      error(\"Either --file or --to is required\");\n      this.exit(1);\n    }\n\n    // Apply shared text from --text flag to messages without text\n    if (flags.text) {\n      messages = messages.map((msg) => ({\n        to: msg.to,\n        text: msg.text || flags.text,\n      }));\n    }\n\n    // Validate messages\n    if (messages.length === 0) {\n      error(\"No messages to send\");\n      this.exit(1);\n    }\n\n    if (messages.length > 1000) {\n      error(\"Batch size cannot exceed 1000 messages\", {\n        hint: \"Split your messages into smaller batches\",\n      });\n      this.exit(1);\n    }\n\n    // Validate each message\n    for (const msg of messages) {\n      if (!/^\\+[1-9]\\d{1,14}$/.test(msg.to)) {\n        error(`Invalid phone number: ${msg.to}`, {\n          hint: \"Use E.164 format: +15551234567\",\n        });\n        this.exit(1);\n      }\n      if (!msg.text?.trim()) {\n        error(`Empty message text for ${msg.to}`, {\n          hint: \"Use --text to provide a shared message for all recipients\",\n        });\n        this.exit(1);\n      }\n    }\n\n    // Handle dry-run mode\n    if (flags[\"dry-run\"]) {\n      const spin = spinner(\"Analyzing batch...\").start();\n\n      try {\n        const preview = await apiClient.post<BatchPreviewResponse>(\n          \"/api/v1/messages/batch/preview\",\n          { messages, text: flags.text, messageType: flags.type },\n        );\n\n        spin.stop();\n\n        if (isJsonMode()) {\n          json(preview);\n          return;\n        }\n\n        // Show comprehensive preview\n        console.log(colors.bold(\"\\n📊 Batch Preview (Dry Run)\\n\"));\n\n        // Summary table\n        console.log(colors.dim(\"─\".repeat(50)));\n        console.log(`Total messages:     ${preview.total}`);\n        console.log(\n          `Sendable:           ${colors.success(String(preview.sendable))}`,\n        );\n        console.log(\n          `Blocked:            ${preview.blocked > 0 ? colors.error(String(preview.blocked)) : \"0\"}`,\n        );\n        console.log(`Duplicates removed: ${preview.duplicates}`);\n        console.log(colors.dim(\"─\".repeat(50)));\n\n        // Credits\n        console.log(`\\nCredits needed:     ${preview.creditsNeeded}`);\n        console.log(`Your balance:       ${preview.creditBalance}`);\n        if (!preview.hasSufficientCredits) {\n          console.log(\n            colors.error(\n              `⚠️  Insufficient credits! Need ${preview.creditsNeeded - preview.creditBalance} more.`,\n            ),\n          );\n        }\n\n        // Access info\n        console.log(`\\nAPI Key type:       ${preview.keyType.toUpperCase()}`);\n        console.log(`Write access:       ${preview.hasWriteScope ? \"✓\" : \"✗\"}`);\n        console.log(\n          `Domestic (US/CA):   ${preview.messagingProfile.canSendDomestic ? \"✓\" : \"✗\"}`,\n        );\n        console.log(\n          `International:      ${preview.messagingProfile.canSendInternational ? \"✓\" : \"✗\"}`,\n        );\n\n        // Country breakdown\n        const countries = Object.entries(preview.byCountry);\n        if (countries.length > 0) {\n          console.log(colors.bold(\"\\n📍 By Country:\\n\"));\n          for (const [country, data] of countries) {\n            const status = data.allowed\n              ? colors.success(\"✓\")\n              : colors.error(\"✗\");\n            console.log(\n              `  ${status} ${country}: ${data.count} msgs, ${data.credits} credits (${data.tier})`,\n            );\n            if (!data.allowed && data.blockedReason) {\n              console.log(colors.dim(`     └─ ${data.blockedReason}`));\n            }\n          }\n        }\n\n        // Compliance check results\n        if (preview.compliance) {\n          console.log(colors.bold(\"\\n🛡️  Compliance Check:\\n\"));\n          console.log(\n            `  Message Type:      ${preview.compliance.messageType.toUpperCase()}`,\n          );\n\n          if (preview.compliance.shaftBlocked > 0) {\n            console.log(\n              colors.error(\n                `  SHAFT Blocked:     ${preview.compliance.shaftBlocked} messages`,\n              ),\n            );\n            for (const msg of preview.compliance.shaftBlockedMessages.slice(\n              0,\n              3,\n            )) {\n              console.log(\n                colors.dim(\n                  `     └─ ${msg.to}: ${msg.category} (${msg.matchedTerms.join(\", \")})`,\n                ),\n              );\n            }\n            if (preview.compliance.shaftBlockedMessages.length > 3) {\n              console.log(\n                colors.dim(\n                  `     ... and ${preview.compliance.shaftBlockedMessages.length - 3} more`,\n                ),\n              );\n            }\n          } else {\n            console.log(\n              colors.success(\n                \"  SHAFT Check:       ✓ All messages pass content filter\",\n              ),\n            );\n          }\n\n          if (preview.compliance.messageType === \"marketing\") {\n            if (preview.compliance.quietHoursRescheduled > 0) {\n              console.log(\n                colors.warning(\n                  `  Quiet Hours:       ${preview.compliance.quietHoursRescheduled} messages will be rescheduled`,\n                ),\n              );\n              for (const msg of preview.compliance.quietHoursBlockedMessages.slice(\n                0,\n                3,\n              )) {\n                const nextTime = msg.nextAllowedTime\n                  ? new Date(msg.nextAllowedTime).toLocaleString()\n                  : \"next available window\";\n                console.log(\n                  colors.dim(\n                    `     └─ ${msg.to}: ${msg.recipientTimezone} → ${nextTime}`,\n                  ),\n                );\n              }\n              if (preview.compliance.quietHoursBlockedMessages.length > 3) {\n                console.log(\n                  colors.dim(\n                    `     ... and ${preview.compliance.quietHoursBlockedMessages.length - 3} more`,\n                  ),\n                );\n              }\n            } else {\n              console.log(\n                colors.success(\n                  \"  Quiet Hours:       ✓ All recipients within allowed hours\",\n                ),\n              );\n            }\n          } else {\n            console.log(\n              colors.dim(\n                \"  Quiet Hours:       Bypassed (transactional message)\",\n              ),\n            );\n          }\n        }\n\n        // Warnings\n        if (preview.warnings.length > 0) {\n          console.log(colors.warning(\"\\n⚠️  Warnings:\"));\n          for (const w of preview.warnings) {\n            console.log(`   • ${w}`);\n          }\n        }\n\n        // Blocked messages (first 5)\n        if (preview.blockedMessages.length > 0) {\n          console.log(\n            colors.error(\n              `\\n❌ Blocked Messages (${preview.blockedMessages.length} total):`,\n            ),\n          );\n          for (const b of preview.blockedMessages.slice(0, 5)) {\n            console.log(`   ${b.to}: ${b.reason}`);\n          }\n          if (preview.blockedMessages.length > 5) {\n            console.log(\n              colors.dim(\n                `   ... and ${preview.blockedMessages.length - 5} more`,\n              ),\n            );\n          }\n        }\n\n        console.log(\n          \"\\n\" + colors.dim(\"No messages were sent. Remove --dry-run to send.\"),\n        );\n        return;\n      } catch (err) {\n        spin.stop();\n        throw err;\n      }\n    }\n\n    const spin = spinner(`Sending ${messages.length} messages...`);\n    spin.start();\n\n    try {\n      const response = await apiClient.post<BatchResponse>(\n        \"/api/v1/messages/batch\",\n        {\n          messages,\n          messageType: flags.type,\n          ...(flags.from && { from: flags.from }),\n        },\n      );\n\n      spin.stop();\n\n      if (isJsonMode()) {\n        json(response);\n        return;\n      }\n\n      // Determine success level\n      const allSucceeded = response.failed === 0;\n      const allFailed = response.sent === 0 && response.failed > 0;\n      const partialSuccess = response.sent > 0 && response.failed > 0;\n\n      if (allSucceeded) {\n        success(\"Batch sent successfully\", {\n          \"Batch ID\": response.batchId,\n          Total: response.total,\n          Sent: colors.success(String(response.sent)),\n          \"Credits Used\": response.creditsUsed,\n        });\n      } else if (allFailed) {\n        error(\"Batch failed\", {\n          hint: `All ${response.failed} messages failed to send`,\n        });\n        console.log(colors.dim(`  Batch ID: ${response.batchId}`));\n        console.log(\n          colors.dim(`  Credits Refunded: ${response.creditsRefunded}`),\n        );\n      } else if (partialSuccess) {\n        console.log(colors.warning(\"\\n⚠️  Batch completed with errors\\n\"));\n        console.log(`  Batch ID:         ${response.batchId}`);\n        console.log(`  Total:            ${response.total}`);\n        console.log(\n          `  Sent:             ${colors.success(String(response.sent))}`,\n        );\n        console.log(\n          `  Failed:           ${colors.error(String(response.failed))}`,\n        );\n        console.log(`  Credits Used:     ${response.creditsUsed}`);\n        console.log(`  Credits Refunded: ${response.creditsRefunded}`);\n\n        // Show failed messages if available\n        if (response.messages) {\n          const failedMsgs = response.messages.filter(\n            (m) => m.status === \"failed\",\n          );\n          if (failedMsgs.length > 0 && failedMsgs.length <= 5) {\n            console.log(colors.dim(\"\\n  Failed messages:\"));\n            for (const msg of failedMsgs) {\n              console.log(\n                colors.dim(`    ${msg.to}: ${msg.error || \"Unknown error\"}`),\n              );\n            }\n          } else if (failedMsgs.length > 5) {\n            console.log(\n              colors.dim(\n                `\\n  ${failedMsgs.length} messages failed. Use --json for details.`,\n              ),\n            );\n          }\n        }\n      }\n    } catch (err) {\n      spin.stop();\n      throw err;\n    }\n  }\n\n  private parseMessagesFromFile(filePath: string): BatchMessage[] {\n    try {\n      const content = readFileSync(filePath, \"utf-8\");\n\n      // Try JSON first\n      if (filePath.endsWith(\".json\")) {\n        const data = JSON.parse(content);\n        if (Array.isArray(data)) {\n          return data.map((item) => ({\n            to: item.to,\n            text: item.text,\n          }));\n        }\n        if (data.messages && Array.isArray(data.messages)) {\n          return data.messages;\n        }\n        error(\"Invalid JSON format\", {\n          hint: \"Expected array of {to, text} objects or {messages: [...]}\",\n        });\n        this.exit(1);\n      }\n\n      // Try CSV\n      if (filePath.endsWith(\".csv\")) {\n        const lines = content.trim().split(\"\\n\");\n        const messages: BatchMessage[] = [];\n\n        // Improved header detection - check for common header patterns\n        const headerPatterns = [\n          \"to\",\n          \"phone\",\n          \"number\",\n          \"recipient\",\n          \"mobile\",\n          \"cell\",\n        ];\n        const firstLine = lines[0].toLowerCase();\n        const hasHeader = headerPatterns.some((p) => firstLine.includes(p));\n        const startIndex = hasHeader ? 1 : 0;\n\n        for (let i = startIndex; i < lines.length; i++) {\n          const line = lines[i].trim();\n          if (!line) continue; // Skip empty lines\n\n          const parts = line.split(\",\");\n          if (parts.length >= 1) {\n            const phone = parts[0].trim().replace(/\"/g, \"\");\n            // Only add if phone looks valid (starts with + or digit)\n            if (phone && (phone.startsWith(\"+\") || /^\\d/.test(phone))) {\n              messages.push({\n                to: phone.startsWith(\"+\") ? phone : `+${phone}`,\n                text:\n                  parts.length >= 2\n                    ? parts.slice(1).join(\",\").trim().replace(/\"/g, \"\") ||\n                      undefined\n                    : undefined,\n              });\n            }\n          }\n        }\n        return messages;\n      }\n\n      error(\"Unsupported file format\", {\n        hint: \"Use .json or .csv file\",\n      });\n      this.exit(1);\n    } catch (err) {\n      if ((err as NodeJS.ErrnoException).code === \"ENOENT\") {\n        error(`File not found: ${filePath}`);\n      } else if (err instanceof SyntaxError) {\n        error(\"Invalid JSON in file\", { hint: err.message });\n      } else {\n        throw err;\n      }\n      this.exit(1);\n    }\n  }\n\n  private parseMessagesFromFlags(to: string, text: string): BatchMessage[] {\n    const phones = to.split(\",\").map((p) => p.trim());\n    return phones.map((phone) => ({ to: phone, text }));\n  }\n}\n"]}
|
|
436
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"batch.js","sourceRoot":"","sources":["../../../src/commands/sms/batch.ts"],"names":[],"mappings":"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","sourcesContent":["import { Flags } from \"@oclif/core\";\nimport { readFileSync, existsSync } from \"fs\";\nimport { basename } from \"path\";\nimport { AuthenticatedCommand } from \"../../lib/base-command.js\";\nimport { apiClient } from \"../../lib/api-client.js\";\nimport {\n  success,\n  error,\n  spinner,\n  colors,\n  json,\n  isJsonMode,\n} from \"../../lib/output.js\";\n\ninterface BatchUploadResponse {\n  upload: {\n    id: string;\n    filename: string;\n    size: number;\n    createdAt: string;\n  };\n  validation: {\n    totalRows: number;\n    validCount: number;\n    invalidCount: number;\n    duplicatesRemoved: number;\n    hasHeader: boolean;\n  };\n  recipients: Array<{ to: string; text?: string }>;\n  errors: Array<{ row: number; phone: string; error: string }>;\n  hasMoreErrors: boolean;\n}\n\ninterface BatchHistoryResponse {\n  uploads: Array<{\n    id: string;\n    filename: string;\n    size: number;\n    createdAt: string;\n    metadata: {\n      totalRows: number;\n      validCount: number;\n      invalidCount: number;\n      source?: string;\n    };\n  }>;\n}\n\ninterface BatchResponse {\n  batchId: string;\n  total: number;\n  sent: number;\n  queued: number;\n  failed: number;\n  creditsUsed: number;\n  creditsRefunded: number;\n  status: string;\n  messages?: Array<{\n    index: number;\n    id: string;\n    to: string;\n    status: string;\n    error?: string;\n  }>;\n}\n\ninterface BatchPreviewResponse {\n  total: number;\n  sendable: number;\n  blocked: number;\n  duplicates: number;\n  creditsNeeded: number;\n  creditBalance: number;\n  hasSufficientCredits: boolean;\n  keyType: \"test\" | \"live\";\n  keyScopes: string[];\n  hasWriteScope: boolean;\n  messagingProfile: {\n    id: string | null;\n    canSendDomestic: boolean;\n    canSendInternational: boolean;\n    verificationStatus: string | null;\n    verificationType: string | null;\n  };\n  byCountry: Record<\n    string,\n    {\n      count: number;\n      credits: number;\n      tier: string;\n      allowed: boolean;\n      blockedReason?: string;\n    }\n  >;\n  blockedMessages: Array<{\n    index: number;\n    to: string;\n    reason: string;\n  }>;\n  compliance: {\n    messageType: \"marketing\" | \"transactional\";\n    shaftBlocked: number;\n    quietHoursBlocked: number;\n    quietHoursRescheduled: number;\n    shaftBlockedMessages: Array<{\n      index: number;\n      to: string;\n      category: string;\n      matchedTerms: string[];\n    }>;\n    quietHoursBlockedMessages: Array<{\n      index: number;\n      to: string;\n      recipientTimezone: string;\n      recipientLocalTime: string;\n      nextAllowedTime?: string;\n    }>;\n  };\n  warnings: string[];\n}\n\nexport default class SmsBatch extends AuthenticatedCommand {\n  static description =\n    \"Send batch SMS messages (uploads CSV to cloud for audit trail)\";\n\n  static examples = [\n    \"<%= config.bin %> sms batch --file recipients.csv\",\n    '<%= config.bin %> sms batch --file phones.csv --text \"Hello everyone!\"',\n    '<%= config.bin %> sms batch --to +15551234567,+15559876543 --text \"Hello!\"',\n    \"<%= config.bin %> sms batch --file recipients.csv --dry-run\",\n    '<%= config.bin %> sms batch --file phones.csv --text \"Code: 123\" --type transactional',\n    \"<%= config.bin %> sms batch --reuse abc123-def456\",\n    \"<%= config.bin %> sms batch --history\",\n  ];\n\n  static flags = {\n    ...AuthenticatedCommand.baseFlags,\n    file: Flags.string({\n      char: \"F\",\n      description: \"CSV file with phone numbers (and optional message text)\",\n      exclusive: [\"to\", \"reuse\", \"history\"],\n    }),\n    to: Flags.string({\n      char: \"t\",\n      description: \"Comma-separated recipient phone numbers (E.164 format)\",\n      exclusive: [\"file\", \"reuse\", \"history\"],\n    }),\n    text: Flags.string({\n      char: \"m\",\n      description:\n        \"Message text (required with --to, optional with --file if CSV has text column)\",\n    }),\n    from: Flags.string({\n      char: \"f\",\n      description: \"Sender ID or phone number for all messages\",\n    }),\n    type: Flags.string({\n      description:\n        \"Message type: marketing (default) or transactional. Transactional bypasses quiet hours.\",\n      options: [\"marketing\", \"transactional\"],\n      default: \"marketing\",\n    }),\n    \"dry-run\": Flags.boolean({\n      char: \"d\",\n      description:\n        \"Preview batch without sending (validates access, shows cost and compliance breakdown)\",\n      default: false,\n    }),\n    reuse: Flags.string({\n      description: \"Re-use a previous batch upload by ID (see --history)\",\n      exclusive: [\"file\", \"to\", \"history\"],\n    }),\n    history: Flags.boolean({\n      description: \"Show recent batch upload history\",\n      exclusive: [\"file\", \"to\", \"reuse\"],\n      default: false,\n    }),\n  };\n\n  async run(): Promise<void> {\n    const { flags } = await this.parse(SmsBatch);\n\n    // Handle history subcommand\n    if (flags.history) {\n      return this.showHistory();\n    }\n\n    // Handle reuse subcommand\n    if (flags.reuse) {\n      return this.reuseBatch(flags.reuse, flags);\n    }\n\n    // Handle --to flag (inline recipients, no upload needed)\n    if (flags.to) {\n      return this.sendInlineRecipients(flags);\n    }\n\n    // Handle --file flag (upload to Supabase)\n    if (flags.file) {\n      return this.uploadAndSend(flags);\n    }\n\n    error(\"Either --file, --to, --reuse, or --history is required\");\n    this.exit(1);\n  }\n\n  /**\n   * Show batch upload history\n   */\n  private async showHistory(): Promise<void> {\n    const spin = spinner(\"Fetching batch history...\").start();\n\n    try {\n      const response = await apiClient.get<BatchHistoryResponse>(\n        \"/api/cli/batch/history\",\n        { limit: 15 },\n      );\n\n      spin.stop();\n\n      if (isJsonMode()) {\n        json(response);\n        return;\n      }\n\n      if (response.uploads.length === 0) {\n        console.log(colors.dim(\"\\nNo batch uploads found.\\n\"));\n        console.log(\"Upload a CSV file with: sendly sms batch --file your.csv\");\n        return;\n      }\n\n      console.log(colors.bold(\"\\n📁 Recent Batch Uploads\\n\"));\n      console.log(colors.dim(\"─\".repeat(80)));\n\n      for (const upload of response.uploads) {\n        const date = new Date(upload.createdAt).toLocaleString();\n        const size = this.formatBytes(upload.size);\n        const source = upload.metadata?.source === \"cli\" ? \"CLI\" : \"Web\";\n\n        console.log(\n          `${colors.info(upload.id.slice(0, 8))}  ${upload.filename}`,\n        );\n        console.log(\n          colors.dim(\n            `   ${date} | ${size} | ${upload.metadata?.validCount || 0} valid recipients | ${source}`,\n          ),\n        );\n      }\n\n      console.log(colors.dim(\"─\".repeat(80)));\n      console.log(\n        colors.dim(\"\\nRe-use a batch: sendly sms batch --reuse <id>\"),\n      );\n    } catch (err) {\n      spin.stop();\n      throw err;\n    }\n  }\n\n  /**\n   * Re-use a previous batch upload\n   */\n  private async reuseBatch(\n    uploadId: string,\n    flags: { text?: string; from?: string; type?: string; \"dry-run\"?: boolean },\n  ): Promise<void> {\n    const spin = spinner(\"Fetching batch data...\").start();\n\n    try {\n      const response = await apiClient.get<BatchUploadResponse>(\n        `/api/cli/batch/reuse/${uploadId}`,\n      );\n\n      spin.stop();\n\n      console.log(\n        colors.success(`\\n✓ Loaded batch: ${response.upload.filename}`),\n      );\n      console.log(\n        colors.dim(\n          `  ${response.validation.validCount} valid recipients from ${response.validation.totalRows} rows`,\n        ),\n      );\n\n      // Apply shared text if provided\n      let messages = response.recipients;\n      if (flags.text) {\n        messages = messages.map((r) => ({\n          to: r.to,\n          text: r.text || flags.text,\n        }));\n      }\n\n      // Validate all messages have text\n      const missingText = messages.filter((m) => !m.text?.trim());\n      if (missingText.length > 0) {\n        error(\n          `${missingText.length} recipients missing message text. Use --text to provide a default.`,\n        );\n        this.exit(1);\n      }\n\n      // Proceed with preview or send\n      await this.previewOrSend(\n        messages as Array<{ to: string; text: string }>,\n        flags,\n        response.upload.id,\n      );\n    } catch (err) {\n      spin.stop();\n      throw err;\n    }\n  }\n\n  /**\n   * Send to inline recipients (--to flag)\n   * This path doesn't upload to Supabase since it's just a few numbers\n   */\n  private async sendInlineRecipients(flags: {\n    to?: string;\n    text?: string;\n    from?: string;\n    type?: string;\n    \"dry-run\"?: boolean;\n  }): Promise<void> {\n    if (!flags.text) {\n      error(\"--text is required when using --to\");\n      this.exit(1);\n    }\n\n    const phones = flags.to!.split(\",\").map((p) => p.trim());\n    const messages = phones.map((phone) => ({\n      to: phone.startsWith(\"+\") ? phone : `+${phone}`,\n      text: flags.text!,\n    }));\n\n    // Validate phone numbers\n    for (const msg of messages) {\n      if (!/^\\+[1-9]\\d{1,14}$/.test(msg.to)) {\n        error(`Invalid phone number: ${msg.to}`, {\n          hint: \"Use E.164 format: +15551234567\",\n        });\n        this.exit(1);\n      }\n    }\n\n    await this.previewOrSend(messages, flags);\n  }\n\n  /**\n   * Upload CSV to Supabase and send\n   */\n  private async uploadAndSend(flags: {\n    file?: string;\n    text?: string;\n    from?: string;\n    type?: string;\n    \"dry-run\"?: boolean;\n  }): Promise<void> {\n    const filePath = flags.file!;\n\n    // Check file exists\n    if (!existsSync(filePath)) {\n      error(`File not found: ${filePath}`);\n      this.exit(1);\n    }\n\n    // Validate file extension\n    if (!filePath.endsWith(\".csv\")) {\n      error(\"Only CSV files are supported\", {\n        hint: \"Convert your file to CSV format with columns: phone,message\",\n      });\n      this.exit(1);\n    }\n\n    // Read file\n    const buffer = readFileSync(filePath);\n    const filename = basename(filePath);\n\n    // Check file size (5MB limit)\n    if (buffer.length > 5 * 1024 * 1024) {\n      error(\"File too large (max 5MB)\", {\n        hint: \"Split your CSV into smaller files\",\n      });\n      this.exit(1);\n    }\n\n    const spin = spinner(`Uploading ${filename}...`).start();\n\n    try {\n      // Upload to Supabase via CLI endpoint\n      const response = await apiClient.uploadFile<BatchUploadResponse>(\n        \"/api/cli/batch/upload\",\n        { buffer, filename, mimetype: \"text/csv\" },\n      );\n\n      spin.stop();\n\n      // Show upload summary\n      console.log(colors.success(`\\n✓ Uploaded: ${response.upload.filename}`));\n      console.log(colors.dim(`  Upload ID: ${response.upload.id}`));\n      console.log(\n        colors.dim(\n          `  ${response.validation.validCount} valid / ${response.validation.invalidCount} invalid / ${response.validation.duplicatesRemoved} duplicates removed`,\n        ),\n      );\n\n      // Show validation errors if any\n      if (response.errors.length > 0) {\n        console.log(colors.warning(`\\n⚠️  Validation errors:`));\n        for (const err of response.errors.slice(0, 5)) {\n          console.log(\n            colors.dim(`   Row ${err.row}: ${err.phone} - ${err.error}`),\n          );\n        }\n        if (response.hasMoreErrors) {\n          console.log(\n            colors.dim(\n              `   ... and ${response.validation.invalidCount - 5} more errors`,\n            ),\n          );\n        }\n      }\n\n      if (response.recipients.length === 0) {\n        error(\"No valid recipients found in CSV\");\n        this.exit(1);\n      }\n\n      // Apply shared text if provided\n      let messages = response.recipients;\n      if (flags.text) {\n        messages = messages.map((r) => ({\n          to: r.to,\n          text: r.text || flags.text,\n        }));\n      }\n\n      // Validate all messages have text\n      const missingText = messages.filter((m) => !m.text?.trim());\n      if (missingText.length > 0) {\n        error(\n          `${missingText.length} recipients missing message text. Use --text to provide a default.`,\n        );\n        console.log(\n          colors.dim(\n            \"\\nCSV should have columns: phone,message OR use --text flag\",\n          ),\n        );\n        this.exit(1);\n      }\n\n      // Proceed with preview or send\n      await this.previewOrSend(\n        messages as Array<{ to: string; text: string }>,\n        flags,\n        response.upload.id,\n      );\n    } catch (err) {\n      spin.stop();\n      throw err;\n    }\n  }\n\n  /**\n   * Preview or send the batch\n   */\n  private async previewOrSend(\n    messages: Array<{ to: string; text: string }>,\n    flags: { from?: string; type?: string; \"dry-run\"?: boolean },\n    uploadId?: string,\n  ): Promise<void> {\n    // Check batch size\n    if (messages.length > 1000) {\n      error(\"Batch size cannot exceed 1000 messages\", {\n        hint: \"Split your messages into smaller batches\",\n      });\n      this.exit(1);\n    }\n\n    // Handle dry-run mode\n    if (flags[\"dry-run\"]) {\n      return this.showPreview(messages, flags);\n    }\n\n    // Send the batch\n    const spin = spinner(`Sending ${messages.length} messages...`);\n    spin.start();\n\n    try {\n      const response = await apiClient.post<BatchResponse>(\n        \"/api/v1/messages/batch\",\n        {\n          messages,\n          messageType: flags.type,\n          ...(flags.from && { from: flags.from }),\n          ...(uploadId && { uploadId }), // Link to file upload for audit trail\n        },\n      );\n\n      spin.stop();\n\n      if (isJsonMode()) {\n        json(response);\n        return;\n      }\n\n      // Determine success level\n      const allSucceeded = response.failed === 0;\n      const allFailed = response.sent === 0 && response.failed > 0;\n      const partialSuccess = response.sent > 0 && response.failed > 0;\n\n      if (allSucceeded) {\n        success(\"Batch sent successfully\", {\n          \"Batch ID\": response.batchId,\n          Total: response.total,\n          Sent: colors.success(String(response.sent)),\n          \"Credits Used\": response.creditsUsed,\n        });\n      } else if (allFailed) {\n        error(\"Batch failed\", {\n          hint: `All ${response.failed} messages failed to send`,\n        });\n        console.log(colors.dim(`  Batch ID: ${response.batchId}`));\n        console.log(\n          colors.dim(`  Credits Refunded: ${response.creditsRefunded}`),\n        );\n      } else if (partialSuccess) {\n        console.log(colors.warning(\"\\n⚠️  Batch completed with errors\\n\"));\n        console.log(`  Batch ID:         ${response.batchId}`);\n        console.log(`  Total:            ${response.total}`);\n        console.log(\n          `  Sent:             ${colors.success(String(response.sent))}`,\n        );\n        console.log(\n          `  Failed:           ${colors.error(String(response.failed))}`,\n        );\n        console.log(`  Credits Used:     ${response.creditsUsed}`);\n        console.log(`  Credits Refunded: ${response.creditsRefunded}`);\n\n        // Show failed messages if available\n        if (response.messages) {\n          const failedMsgs = response.messages.filter(\n            (m) => m.status === \"failed\",\n          );\n          if (failedMsgs.length > 0 && failedMsgs.length <= 5) {\n            console.log(colors.dim(\"\\n  Failed messages:\"));\n            for (const msg of failedMsgs) {\n              console.log(\n                colors.dim(`    ${msg.to}: ${msg.error || \"Unknown error\"}`),\n              );\n            }\n          } else if (failedMsgs.length > 5) {\n            console.log(\n              colors.dim(\n                `\\n  ${failedMsgs.length} messages failed. Use --json for details.`,\n              ),\n            );\n          }\n        }\n      }\n    } catch (err) {\n      spin.stop();\n      throw err;\n    }\n  }\n\n  /**\n   * Show batch preview (dry-run mode)\n   */\n  private async showPreview(\n    messages: Array<{ to: string; text: string }>,\n    flags: { type?: string },\n  ): Promise<void> {\n    const spin = spinner(\"Analyzing batch...\").start();\n\n    try {\n      const preview = await apiClient.post<BatchPreviewResponse>(\n        \"/api/v1/messages/batch/preview\",\n        { messages, messageType: flags.type },\n      );\n\n      spin.stop();\n\n      if (isJsonMode()) {\n        json(preview);\n        return;\n      }\n\n      // Show comprehensive preview\n      console.log(colors.bold(\"\\n📊 Batch Preview (Dry Run)\\n\"));\n\n      // Summary table\n      console.log(colors.dim(\"─\".repeat(50)));\n      console.log(`Total messages:     ${preview.total}`);\n      console.log(\n        `Sendable:           ${colors.success(String(preview.sendable))}`,\n      );\n      console.log(\n        `Blocked:            ${preview.blocked > 0 ? colors.error(String(preview.blocked)) : \"0\"}`,\n      );\n      console.log(`Duplicates removed: ${preview.duplicates}`);\n      console.log(colors.dim(\"─\".repeat(50)));\n\n      // Credits\n      console.log(`\\nCredits needed:     ${preview.creditsNeeded}`);\n      console.log(`Your balance:       ${preview.creditBalance}`);\n      if (!preview.hasSufficientCredits) {\n        console.log(\n          colors.error(\n            `⚠️  Insufficient credits! Need ${preview.creditsNeeded - preview.creditBalance} more.`,\n          ),\n        );\n      }\n\n      // Access info\n      console.log(`\\nAPI Key type:       ${preview.keyType.toUpperCase()}`);\n      console.log(`Write access:       ${preview.hasWriteScope ? \"✓\" : \"✗\"}`);\n      console.log(\n        `Domestic (US/CA):   ${preview.messagingProfile.canSendDomestic ? \"✓\" : \"✗\"}`,\n      );\n      console.log(\n        `International:      ${preview.messagingProfile.canSendInternational ? \"✓\" : \"✗\"}`,\n      );\n\n      // Country breakdown\n      const countries = Object.entries(preview.byCountry);\n      if (countries.length > 0) {\n        console.log(colors.bold(\"\\n📍 By Country:\\n\"));\n        for (const [country, data] of countries) {\n          const status = data.allowed ? colors.success(\"✓\") : colors.error(\"✗\");\n          console.log(\n            `  ${status} ${country}: ${data.count} msgs, ${data.credits} credits (${data.tier})`,\n          );\n          if (!data.allowed && data.blockedReason) {\n            console.log(colors.dim(`     └─ ${data.blockedReason}`));\n          }\n        }\n      }\n\n      // Compliance check results\n      if (preview.compliance) {\n        console.log(colors.bold(\"\\n🛡️  Compliance Check:\\n\"));\n        console.log(\n          `  Message Type:      ${preview.compliance.messageType.toUpperCase()}`,\n        );\n\n        if (preview.compliance.shaftBlocked > 0) {\n          console.log(\n            colors.error(\n              `  SHAFT Blocked:     ${preview.compliance.shaftBlocked} messages`,\n            ),\n          );\n          for (const msg of preview.compliance.shaftBlockedMessages.slice(\n            0,\n            3,\n          )) {\n            console.log(\n              colors.dim(\n                `     └─ ${msg.to}: ${msg.category} (${msg.matchedTerms.join(\", \")})`,\n              ),\n            );\n          }\n          if (preview.compliance.shaftBlockedMessages.length > 3) {\n            console.log(\n              colors.dim(\n                `     ... and ${preview.compliance.shaftBlockedMessages.length - 3} more`,\n              ),\n            );\n          }\n        } else {\n          console.log(\n            colors.success(\n              \"  SHAFT Check:       ✓ All messages pass content filter\",\n            ),\n          );\n        }\n\n        if (preview.compliance.messageType === \"marketing\") {\n          if (preview.compliance.quietHoursRescheduled > 0) {\n            console.log(\n              colors.warning(\n                `  Quiet Hours:       ${preview.compliance.quietHoursRescheduled} messages will be rescheduled`,\n              ),\n            );\n            for (const msg of preview.compliance.quietHoursBlockedMessages.slice(\n              0,\n              3,\n            )) {\n              const nextTime = msg.nextAllowedTime\n                ? new Date(msg.nextAllowedTime).toLocaleString()\n                : \"next available window\";\n              console.log(\n                colors.dim(\n                  `     └─ ${msg.to}: ${msg.recipientTimezone} → ${nextTime}`,\n                ),\n              );\n            }\n            if (preview.compliance.quietHoursBlockedMessages.length > 3) {\n              console.log(\n                colors.dim(\n                  `     ... and ${preview.compliance.quietHoursBlockedMessages.length - 3} more`,\n                ),\n              );\n            }\n          } else {\n            console.log(\n              colors.success(\n                \"  Quiet Hours:       ✓ All recipients within allowed hours\",\n              ),\n            );\n          }\n        } else {\n          console.log(\n            colors.dim(\"  Quiet Hours:       Bypassed (transactional message)\"),\n          );\n        }\n      }\n\n      // Warnings\n      if (preview.warnings.length > 0) {\n        console.log(colors.warning(\"\\n⚠️  Warnings:\"));\n        for (const w of preview.warnings) {\n          console.log(`   • ${w}`);\n        }\n      }\n\n      // Blocked messages (first 5)\n      if (preview.blockedMessages.length > 0) {\n        console.log(\n          colors.error(\n            `\\n❌ Blocked Messages (${preview.blockedMessages.length} total):`,\n          ),\n        );\n        for (const b of preview.blockedMessages.slice(0, 5)) {\n          console.log(`   ${b.to}: ${b.reason}`);\n        }\n        if (preview.blockedMessages.length > 5) {\n          console.log(\n            colors.dim(`   ... and ${preview.blockedMessages.length - 5} more`),\n          );\n        }\n      }\n\n      console.log(\n        \"\\n\" + colors.dim(\"No messages were sent. Remove --dry-run to send.\"),\n      );\n    } catch (err) {\n      spin.stop();\n      throw err;\n    }\n  }\n\n  /**\n   * Format bytes to human readable string\n   */\n  private formatBytes(bytes: number): string {\n    if (bytes < 1024) return `${bytes} B`;\n    if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`;\n    return `${(bytes / (1024 * 1024)).toFixed(1)} MB`;\n  }\n}\n"]}
|