@sendly/cli 3.3.1 ā 3.3.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.
- package/dist/commands/sms/batch.d.ts +1 -0
- package/dist/commands/sms/batch.js +47 -3
- package/dist/commands/sms/schedule.d.ts +1 -0
- package/dist/commands/sms/schedule.js +8 -1
- package/dist/commands/sms/send.d.ts +1 -0
- package/dist/commands/sms/send.js +8 -1
- package/oclif.manifest.json +105 -66
- package/package.json +1 -1
|
@@ -7,6 +7,7 @@ export default class SmsBatch extends AuthenticatedCommand {
|
|
|
7
7
|
to: import("@oclif/core/lib/interfaces/parser.js").OptionFlag<string | undefined, import("@oclif/core/lib/interfaces/parser.js").CustomOptions>;
|
|
8
8
|
text: import("@oclif/core/lib/interfaces/parser.js").OptionFlag<string | undefined, import("@oclif/core/lib/interfaces/parser.js").CustomOptions>;
|
|
9
9
|
from: import("@oclif/core/lib/interfaces/parser.js").OptionFlag<string | undefined, import("@oclif/core/lib/interfaces/parser.js").CustomOptions>;
|
|
10
|
+
type: import("@oclif/core/lib/interfaces/parser.js").OptionFlag<string, import("@oclif/core/lib/interfaces/parser.js").CustomOptions>;
|
|
10
11
|
"dry-run": import("@oclif/core/lib/interfaces/parser.js").BooleanFlag<boolean>;
|
|
11
12
|
json: import("@oclif/core/lib/interfaces/parser.js").BooleanFlag<boolean>;
|
|
12
13
|
quiet: import("@oclif/core/lib/interfaces/parser.js").BooleanFlag<boolean>;
|
|
@@ -12,6 +12,7 @@ export default class SmsBatch extends AuthenticatedCommand {
|
|
|
12
12
|
'<%= config.bin %> sms batch --file recipients.csv --from "Sendly"',
|
|
13
13
|
"<%= config.bin %> sms batch --file messages.json --json",
|
|
14
14
|
'<%= config.bin %> sms batch --file phones.csv --text "Hi" --dry-run',
|
|
15
|
+
'<%= config.bin %> sms batch --file phones.csv --text "Your code: 123" --type transactional',
|
|
15
16
|
];
|
|
16
17
|
static flags = {
|
|
17
18
|
...AuthenticatedCommand.baseFlags,
|
|
@@ -33,9 +34,14 @@ export default class SmsBatch extends AuthenticatedCommand {
|
|
|
33
34
|
char: "f",
|
|
34
35
|
description: "Sender ID or phone number for all messages",
|
|
35
36
|
}),
|
|
37
|
+
type: Flags.string({
|
|
38
|
+
description: "Message type: marketing (default) or transactional. Transactional bypasses quiet hours.",
|
|
39
|
+
options: ["marketing", "transactional"],
|
|
40
|
+
default: "marketing",
|
|
41
|
+
}),
|
|
36
42
|
"dry-run": Flags.boolean({
|
|
37
43
|
char: "d",
|
|
38
|
-
description: "Preview batch without sending (validates access, shows cost breakdown)",
|
|
44
|
+
description: "Preview batch without sending (validates access, shows cost and compliance breakdown)",
|
|
39
45
|
default: false,
|
|
40
46
|
}),
|
|
41
47
|
};
|
|
@@ -94,7 +100,7 @@ export default class SmsBatch extends AuthenticatedCommand {
|
|
|
94
100
|
if (flags["dry-run"]) {
|
|
95
101
|
const spin = spinner("Analyzing batch...").start();
|
|
96
102
|
try {
|
|
97
|
-
const preview = await apiClient.post("/api/v1/messages/batch/preview", { messages, text: flags.text });
|
|
103
|
+
const preview = await apiClient.post("/api/v1/messages/batch/preview", { messages, text: flags.text, messageType: flags.type });
|
|
98
104
|
spin.stop();
|
|
99
105
|
if (isJsonMode()) {
|
|
100
106
|
json(preview);
|
|
@@ -134,6 +140,43 @@ export default class SmsBatch extends AuthenticatedCommand {
|
|
|
134
140
|
}
|
|
135
141
|
}
|
|
136
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
|
+
}
|
|
137
180
|
// Warnings
|
|
138
181
|
if (preview.warnings.length > 0) {
|
|
139
182
|
console.log(colors.warning("\nā ļø Warnings:"));
|
|
@@ -164,6 +207,7 @@ export default class SmsBatch extends AuthenticatedCommand {
|
|
|
164
207
|
try {
|
|
165
208
|
const response = await apiClient.post("/api/v1/messages/batch", {
|
|
166
209
|
messages,
|
|
210
|
+
messageType: flags.type,
|
|
167
211
|
...(flags.from && { from: flags.from }),
|
|
168
212
|
});
|
|
169
213
|
spin.stop();
|
|
@@ -298,4 +342,4 @@ export default class SmsBatch extends AuthenticatedCommand {
|
|
|
298
342
|
return phones.map((phone) => ({ to: phone, text }));
|
|
299
343
|
}
|
|
300
344
|
}
|
|
301
|
-
//# 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  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  ];\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    \"dry-run\": Flags.boolean({\n      char: \"d\",\n      description:\n        \"Preview batch without sending (validates access, shows cost 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 },\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        // 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          ...(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"]}
|
|
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"]}
|
|
@@ -7,6 +7,7 @@ export default class SmsSchedule extends AuthenticatedCommand {
|
|
|
7
7
|
text: import("@oclif/core/lib/interfaces/parser.js").OptionFlag<string, import("@oclif/core/lib/interfaces/parser.js").CustomOptions>;
|
|
8
8
|
at: import("@oclif/core/lib/interfaces/parser.js").OptionFlag<string, import("@oclif/core/lib/interfaces/parser.js").CustomOptions>;
|
|
9
9
|
from: import("@oclif/core/lib/interfaces/parser.js").OptionFlag<string | undefined, import("@oclif/core/lib/interfaces/parser.js").CustomOptions>;
|
|
10
|
+
type: import("@oclif/core/lib/interfaces/parser.js").OptionFlag<string, import("@oclif/core/lib/interfaces/parser.js").CustomOptions>;
|
|
10
11
|
json: import("@oclif/core/lib/interfaces/parser.js").BooleanFlag<boolean>;
|
|
11
12
|
quiet: import("@oclif/core/lib/interfaces/parser.js").BooleanFlag<boolean>;
|
|
12
13
|
};
|
|
@@ -7,6 +7,7 @@ export default class SmsSchedule extends AuthenticatedCommand {
|
|
|
7
7
|
static examples = [
|
|
8
8
|
'<%= config.bin %> sms schedule --to +15551234567 --text "Reminder!" --at "2025-01-20T10:00:00Z"',
|
|
9
9
|
'<%= config.bin %> sms schedule --to +15551234567 --text "Meeting in 1 hour" --at "2025-01-15T14:00:00Z" --from "Sendly"',
|
|
10
|
+
'<%= config.bin %> sms schedule --to +15551234567 --text "Your code: 123456" --at "2025-01-20T10:00:00Z" --type transactional',
|
|
10
11
|
'<%= config.bin %> sms schedule --to +15551234567 --text "Hello!" --at "2025-01-20T10:00:00Z" --json',
|
|
11
12
|
];
|
|
12
13
|
static flags = {
|
|
@@ -30,6 +31,11 @@ export default class SmsSchedule extends AuthenticatedCommand {
|
|
|
30
31
|
char: "f",
|
|
31
32
|
description: "Sender ID or phone number",
|
|
32
33
|
}),
|
|
34
|
+
type: Flags.string({
|
|
35
|
+
description: "Message type: marketing (default) or transactional. Transactional messages bypass quiet hours.",
|
|
36
|
+
options: ["marketing", "transactional"],
|
|
37
|
+
default: "marketing",
|
|
38
|
+
}),
|
|
33
39
|
};
|
|
34
40
|
async run() {
|
|
35
41
|
const { flags } = await this.parse(SmsSchedule);
|
|
@@ -72,6 +78,7 @@ export default class SmsSchedule extends AuthenticatedCommand {
|
|
|
72
78
|
to: flags.to,
|
|
73
79
|
text: flags.text,
|
|
74
80
|
scheduledAt: flags.at,
|
|
81
|
+
messageType: flags.type,
|
|
75
82
|
...(flags.from && { from: flags.from }),
|
|
76
83
|
});
|
|
77
84
|
spin.stop();
|
|
@@ -93,4 +100,4 @@ export default class SmsSchedule extends AuthenticatedCommand {
|
|
|
93
100
|
}
|
|
94
101
|
}
|
|
95
102
|
}
|
|
96
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
103
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"schedule.js","sourceRoot":"","sources":["../../../src/commands/sms/schedule.ts"],"names":[],"mappings":"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","sourcesContent":["import { Flags } from \"@oclif/core\";\nimport { AuthenticatedCommand } from \"../../lib/base-command.js\";\nimport { apiClient } from \"../../lib/api-client.js\";\nimport {\n  success,\n  error,\n  spinner,\n  colors,\n  formatStatus,\n  json,\n  isJsonMode,\n} from \"../../lib/output.js\";\n\ninterface ScheduledMessageResponse {\n  id: string;\n  to: string;\n  from?: string;\n  text: string;\n  status: string;\n  scheduledAt: string;\n  createdAt: string;\n}\n\nexport default class SmsSchedule extends AuthenticatedCommand {\n  static description = \"Schedule an SMS message for future delivery\";\n\n  static examples = [\n    '<%= config.bin %> sms schedule --to +15551234567 --text \"Reminder!\" --at \"2025-01-20T10:00:00Z\"',\n    '<%= config.bin %> sms schedule --to +15551234567 --text \"Meeting in 1 hour\" --at \"2025-01-15T14:00:00Z\" --from \"Sendly\"',\n    '<%= config.bin %> sms schedule --to +15551234567 --text \"Your code: 123456\" --at \"2025-01-20T10:00:00Z\" --type transactional',\n    '<%= config.bin %> sms schedule --to +15551234567 --text \"Hello!\" --at \"2025-01-20T10:00:00Z\" --json',\n  ];\n\n  static flags = {\n    ...AuthenticatedCommand.baseFlags,\n    to: Flags.string({\n      char: \"t\",\n      description: \"Recipient phone number (E.164 format)\",\n      required: true,\n    }),\n    text: Flags.string({\n      char: \"m\",\n      description: \"Message text\",\n      required: true,\n    }),\n    at: Flags.string({\n      char: \"a\",\n      description:\n        \"Scheduled time (ISO 8601 format, e.g., 2025-01-20T10:00:00Z)\",\n      required: true,\n    }),\n    from: Flags.string({\n      char: \"f\",\n      description: \"Sender ID or phone number\",\n    }),\n    type: Flags.string({\n      description:\n        \"Message type: marketing (default) or transactional. Transactional messages bypass quiet hours.\",\n      options: [\"marketing\", \"transactional\"],\n      default: \"marketing\",\n    }),\n  };\n\n  async run(): Promise<void> {\n    const { flags } = await this.parse(SmsSchedule);\n\n    // Validate phone number format\n    if (!/^\\+[1-9]\\d{1,14}$/.test(flags.to)) {\n      error(\"Invalid phone number format\", {\n        hint: \"Use E.164 format: +15551234567\",\n      });\n      this.exit(1);\n    }\n\n    // Validate message text\n    if (!flags.text.trim()) {\n      error(\"Message text cannot be empty\");\n      this.exit(1);\n    }\n\n    // Validate scheduled time\n    const scheduledDate = new Date(flags.at);\n    if (isNaN(scheduledDate.getTime())) {\n      error(\"Invalid scheduled time format\", {\n        hint: \"Use ISO 8601 format: 2025-01-20T10:00:00Z\",\n      });\n      this.exit(1);\n    }\n\n    // Check scheduling time constraints (Telnyx requires 5 min - 5 days)\n    const FIVE_MINUTES = 5 * 60 * 1000;\n    const FIVE_DAYS = 5 * 24 * 60 * 60 * 1000;\n    const timeUntilSend = scheduledDate.getTime() - Date.now();\n\n    if (timeUntilSend < FIVE_MINUTES) {\n      error(\"Scheduled time must be at least 5 minutes from now\");\n      this.exit(1);\n    }\n\n    if (timeUntilSend > FIVE_DAYS) {\n      error(\"Scheduled time must be within 5 days\");\n      this.exit(1);\n    }\n\n    const spin = spinner(\"Scheduling message...\");\n    spin.start();\n\n    try {\n      const response = await apiClient.post<ScheduledMessageResponse>(\n        \"/api/v1/messages/schedule\",\n        {\n          to: flags.to,\n          text: flags.text,\n          scheduledAt: flags.at,\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      const formattedTime = new Date(response.scheduledAt).toLocaleString();\n\n      success(\"Message scheduled\", {\n        ID: response.id,\n        To: response.to,\n        Status: formatStatus(response.status),\n        \"Scheduled For\": colors.code(formattedTime),\n      });\n    } catch (err) {\n      spin.stop();\n      throw err;\n    }\n  }\n}\n"]}
|
|
@@ -6,6 +6,7 @@ export default class SmsSend extends AuthenticatedCommand {
|
|
|
6
6
|
to: import("@oclif/core/lib/interfaces/parser.js").OptionFlag<string, import("@oclif/core/lib/interfaces/parser.js").CustomOptions>;
|
|
7
7
|
text: import("@oclif/core/lib/interfaces/parser.js").OptionFlag<string, import("@oclif/core/lib/interfaces/parser.js").CustomOptions>;
|
|
8
8
|
from: import("@oclif/core/lib/interfaces/parser.js").OptionFlag<string | undefined, import("@oclif/core/lib/interfaces/parser.js").CustomOptions>;
|
|
9
|
+
type: import("@oclif/core/lib/interfaces/parser.js").OptionFlag<string, import("@oclif/core/lib/interfaces/parser.js").CustomOptions>;
|
|
9
10
|
json: import("@oclif/core/lib/interfaces/parser.js").BooleanFlag<boolean>;
|
|
10
11
|
quiet: import("@oclif/core/lib/interfaces/parser.js").BooleanFlag<boolean>;
|
|
11
12
|
};
|
|
@@ -7,6 +7,7 @@ export default class SmsSend extends AuthenticatedCommand {
|
|
|
7
7
|
static examples = [
|
|
8
8
|
'<%= config.bin %> sms send --to +15551234567 --text "Hello!"',
|
|
9
9
|
'<%= config.bin %> sms send --to +15551234567 --text "Hello!" --from "Sendly"',
|
|
10
|
+
'<%= config.bin %> sms send --to +15551234567 --text "Hello!" --type transactional',
|
|
10
11
|
'<%= config.bin %> sms send --to +15551234567 --text "Hello!" --json',
|
|
11
12
|
];
|
|
12
13
|
static flags = {
|
|
@@ -25,6 +26,11 @@ export default class SmsSend extends AuthenticatedCommand {
|
|
|
25
26
|
char: "f",
|
|
26
27
|
description: "Sender ID or phone number",
|
|
27
28
|
}),
|
|
29
|
+
type: Flags.string({
|
|
30
|
+
description: "Message type: marketing (default) or transactional",
|
|
31
|
+
options: ["marketing", "transactional"],
|
|
32
|
+
default: "marketing",
|
|
33
|
+
}),
|
|
28
34
|
};
|
|
29
35
|
async run() {
|
|
30
36
|
const { flags } = await this.parse(SmsSend);
|
|
@@ -46,6 +52,7 @@ export default class SmsSend extends AuthenticatedCommand {
|
|
|
46
52
|
const response = await apiClient.post("/api/v1/messages", {
|
|
47
53
|
to: flags.to,
|
|
48
54
|
text: flags.text,
|
|
55
|
+
messageType: flags.type,
|
|
49
56
|
...(flags.from && { from: flags.from }),
|
|
50
57
|
});
|
|
51
58
|
spin.stop();
|
|
@@ -67,4 +74,4 @@ export default class SmsSend extends AuthenticatedCommand {
|
|
|
67
74
|
}
|
|
68
75
|
}
|
|
69
76
|
}
|
|
70
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
77
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VuZC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9jb21tYW5kcy9zbXMvc2VuZC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsS0FBSyxFQUFFLE1BQU0sYUFBYSxDQUFDO0FBQ3BDLE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBQ2pFLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUNwRCxPQUFPLEVBQ0wsT0FBTyxFQUNQLEtBQUssRUFDTCxPQUFPLEVBRVAsWUFBWSxFQUNaLGFBQWEsRUFDYixJQUFJLEVBQ0osVUFBVSxHQUNYLE1BQU0scUJBQXFCLENBQUM7QUFhN0IsTUFBTSxDQUFDLE9BQU8sT0FBTyxPQUFRLFNBQVEsb0JBQW9CO0lBQ3ZELE1BQU0sQ0FBQyxXQUFXLEdBQUcscUJBQXFCLENBQUM7SUFFM0MsTUFBTSxDQUFDLFFBQVEsR0FBRztRQUNoQiw4REFBOEQ7UUFDOUQsOEVBQThFO1FBQzlFLG1GQUFtRjtRQUNuRixxRUFBcUU7S0FDdEUsQ0FBQztJQUVGLE1BQU0sQ0FBQyxLQUFLLEdBQUc7UUFDYixHQUFHLG9CQUFvQixDQUFDLFNBQVM7UUFDakMsRUFBRSxFQUFFLEtBQUssQ0FBQyxNQUFNLENBQUM7WUFDZixJQUFJLEVBQUUsR0FBRztZQUNULFdBQVcsRUFBRSx1Q0FBdUM7WUFDcEQsUUFBUSxFQUFFLElBQUk7U0FDZixDQUFDO1FBQ0YsSUFBSSxFQUFFLEtBQUssQ0FBQyxNQUFNLENBQUM7WUFDakIsSUFBSSxFQUFFLEdBQUc7WUFDVCxXQUFXLEVBQUUsY0FBYztZQUMzQixRQUFRLEVBQUUsSUFBSTtTQUNmLENBQUM7UUFDRixJQUFJLEVBQUUsS0FBSyxDQUFDLE1BQU0sQ0FBQztZQUNqQixJQUFJLEVBQUUsR0FBRztZQUNULFdBQVcsRUFBRSwyQkFBMkI7U0FDekMsQ0FBQztRQUNGLElBQUksRUFBRSxLQUFLLENBQUMsTUFBTSxDQUFDO1lBQ2pCLFdBQVcsRUFBRSxvREFBb0Q7WUFDakUsT0FBTyxFQUFFLENBQUMsV0FBVyxFQUFFLGVBQWUsQ0FBQztZQUN2QyxPQUFPLEVBQUUsV0FBVztTQUNyQixDQUFDO0tBQ0gsQ0FBQztJQUVGLEtBQUssQ0FBQyxHQUFHO1FBQ1AsTUFBTSxFQUFFLEtBQUssRUFBRSxHQUFHLE1BQU0sSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUU1QywrQkFBK0I7UUFDL0IsSUFBSSxDQUFDLG1CQUFtQixDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQztZQUN4QyxLQUFLLENBQUMsNkJBQTZCLEVBQUU7Z0JBQ25DLElBQUksRUFBRSxnQ0FBZ0M7YUFDdkMsQ0FBQyxDQUFDO1lBQ0gsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNmLENBQUM7UUFFRCx3QkFBd0I7UUFDeEIsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLEVBQUUsQ0FBQztZQUN2QixLQUFLLENBQUMsOEJBQThCLENBQUMsQ0FBQztZQUN0QyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ2YsQ0FBQztRQUVELE1BQU0sSUFBSSxHQUFHLE9BQU8sQ0FBQyxvQkFBb0IsQ0FBQyxDQUFDO1FBQzNDLElBQUksQ0FBQyxLQUFLLEVBQUUsQ0FBQztRQUViLElBQUksQ0FBQztZQUNILE1BQU0sUUFBUSxHQUFHLE1BQU0sU0FBUyxDQUFDLElBQUksQ0FDbkMsa0JBQWtCLEVBQ2xCO2dCQUNFLEVBQUUsRUFBRSxLQUFLLENBQUMsRUFBRTtnQkFDWixJQUFJLEVBQUUsS0FBSyxDQUFDLElBQUk7Z0JBQ2hCLFdBQVcsRUFBRSxLQUFLLENBQUMsSUFBSTtnQkFDdkIsR0FBRyxDQUFDLEtBQUssQ0FBQyxJQUFJLElBQUksRUFBRSxJQUFJLEVBQUUsS0FBSyxDQUFDLElBQUksRUFBRSxDQUFDO2FBQ3hDLENBQ0YsQ0FBQztZQUVGLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQztZQUVaLElBQUksVUFBVSxFQUFFLEVBQUUsQ0FBQztnQkFDakIsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDO2dCQUNmLE9BQU87WUFDVCxDQUFDO1lBRUQsT0FBTyxDQUFDLGNBQWMsRUFBRTtnQkFDdEIsRUFBRSxFQUFFLFFBQVEsQ0FBQyxFQUFFO2dCQUNmLEVBQUUsRUFBRSxRQUFRLENBQUMsRUFBRTtnQkFDZixNQUFNLEVBQUUsWUFBWSxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUM7Z0JBQ3JDLFFBQVEsRUFBRSxRQUFRLENBQUMsUUFBUTtnQkFDM0IsT0FBTyxFQUFFLGFBQWEsQ0FBQyxRQUFRLENBQUMsV0FBVyxDQUFDO2FBQzdDLENBQUMsQ0FBQztRQUNMLENBQUM7UUFBQyxPQUFPLEdBQUcsRUFBRSxDQUFDO1lBQ2IsSUFBSSxDQUFDLElBQUksRUFBRSxDQUFDO1lBQ1osTUFBTSxHQUFHLENBQUM7UUFDWixDQUFDO0lBQ0gsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEZsYWdzIH0gZnJvbSBcIkBvY2xpZi9jb3JlXCI7XG5pbXBvcnQgeyBBdXRoZW50aWNhdGVkQ29tbWFuZCB9IGZyb20gXCIuLi8uLi9saWIvYmFzZS1jb21tYW5kLmpzXCI7XG5pbXBvcnQgeyBhcGlDbGllbnQgfSBmcm9tIFwiLi4vLi4vbGliL2FwaS1jbGllbnQuanNcIjtcbmltcG9ydCB7XG4gIHN1Y2Nlc3MsXG4gIGVycm9yLFxuICBzcGlubmVyLFxuICBjb2xvcnMsXG4gIGZvcm1hdFN0YXR1cyxcbiAgZm9ybWF0Q3JlZGl0cyxcbiAganNvbixcbiAgaXNKc29uTW9kZSxcbn0gZnJvbSBcIi4uLy4uL2xpYi9vdXRwdXQuanNcIjtcblxuaW50ZXJmYWNlIFNlbmRNZXNzYWdlUmVzcG9uc2Uge1xuICBpZDogc3RyaW5nO1xuICB0bzogc3RyaW5nO1xuICBmcm9tOiBzdHJpbmc7XG4gIHRleHQ6IHN0cmluZztcbiAgc3RhdHVzOiBzdHJpbmc7XG4gIHNlZ21lbnRzOiBudW1iZXI7XG4gIGNyZWRpdHNVc2VkOiBudW1iZXI7XG4gIGNyZWF0ZWRBdDogc3RyaW5nO1xufVxuXG5leHBvcnQgZGVmYXVsdCBjbGFzcyBTbXNTZW5kIGV4dGVuZHMgQXV0aGVudGljYXRlZENvbW1hbmQge1xuICBzdGF0aWMgZGVzY3JpcHRpb24gPSBcIlNlbmQgYW4gU01TIG1lc3NhZ2VcIjtcblxuICBzdGF0aWMgZXhhbXBsZXMgPSBbXG4gICAgJzwlPSBjb25maWcuYmluICU+IHNtcyBzZW5kIC0tdG8gKzE1NTUxMjM0NTY3IC0tdGV4dCBcIkhlbGxvIVwiJyxcbiAgICAnPCU9IGNvbmZpZy5iaW4gJT4gc21zIHNlbmQgLS10byArMTU1NTEyMzQ1NjcgLS10ZXh0IFwiSGVsbG8hXCIgLS1mcm9tIFwiU2VuZGx5XCInLFxuICAgICc8JT0gY29uZmlnLmJpbiAlPiBzbXMgc2VuZCAtLXRvICsxNTU1MTIzNDU2NyAtLXRleHQgXCJIZWxsbyFcIiAtLXR5cGUgdHJhbnNhY3Rpb25hbCcsXG4gICAgJzwlPSBjb25maWcuYmluICU+IHNtcyBzZW5kIC0tdG8gKzE1NTUxMjM0NTY3IC0tdGV4dCBcIkhlbGxvIVwiIC0tanNvbicsXG4gIF07XG5cbiAgc3RhdGljIGZsYWdzID0ge1xuICAgIC4uLkF1dGhlbnRpY2F0ZWRDb21tYW5kLmJhc2VGbGFncyxcbiAgICB0bzogRmxhZ3Muc3RyaW5nKHtcbiAgICAgIGNoYXI6IFwidFwiLFxuICAgICAgZGVzY3JpcHRpb246IFwiUmVjaXBpZW50IHBob25lIG51bWJlciAoRS4xNjQgZm9ybWF0KVwiLFxuICAgICAgcmVxdWlyZWQ6IHRydWUsXG4gICAgfSksXG4gICAgdGV4dDogRmxhZ3Muc3RyaW5nKHtcbiAgICAgIGNoYXI6IFwibVwiLFxuICAgICAgZGVzY3JpcHRpb246IFwiTWVzc2FnZSB0ZXh0XCIsXG4gICAgICByZXF1aXJlZDogdHJ1ZSxcbiAgICB9KSxcbiAgICBmcm9tOiBGbGFncy5zdHJpbmcoe1xuICAgICAgY2hhcjogXCJmXCIsXG4gICAgICBkZXNjcmlwdGlvbjogXCJTZW5kZXIgSUQgb3IgcGhvbmUgbnVtYmVyXCIsXG4gICAgfSksXG4gICAgdHlwZTogRmxhZ3Muc3RyaW5nKHtcbiAgICAgIGRlc2NyaXB0aW9uOiBcIk1lc3NhZ2UgdHlwZTogbWFya2V0aW5nIChkZWZhdWx0KSBvciB0cmFuc2FjdGlvbmFsXCIsXG4gICAgICBvcHRpb25zOiBbXCJtYXJrZXRpbmdcIiwgXCJ0cmFuc2FjdGlvbmFsXCJdLFxuICAgICAgZGVmYXVsdDogXCJtYXJrZXRpbmdcIixcbiAgICB9KSxcbiAgfTtcblxuICBhc3luYyBydW4oKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgY29uc3QgeyBmbGFncyB9ID0gYXdhaXQgdGhpcy5wYXJzZShTbXNTZW5kKTtcblxuICAgIC8vIFZhbGlkYXRlIHBob25lIG51bWJlciBmb3JtYXRcbiAgICBpZiAoIS9eXFwrWzEtOV1cXGR7MSwxNH0kLy50ZXN0KGZsYWdzLnRvKSkge1xuICAgICAgZXJyb3IoXCJJbnZhbGlkIHBob25lIG51bWJlciBmb3JtYXRcIiwge1xuICAgICAgICBoaW50OiBcIlVzZSBFLjE2NCBmb3JtYXQ6ICsxNTU1MTIzNDU2N1wiLFxuICAgICAgfSk7XG4gICAgICB0aGlzLmV4aXQoMSk7XG4gICAgfVxuXG4gICAgLy8gVmFsaWRhdGUgbWVzc2FnZSB0ZXh0XG4gICAgaWYgKCFmbGFncy50ZXh0LnRyaW0oKSkge1xuICAgICAgZXJyb3IoXCJNZXNzYWdlIHRleHQgY2Fubm90IGJlIGVtcHR5XCIpO1xuICAgICAgdGhpcy5leGl0KDEpO1xuICAgIH1cblxuICAgIGNvbnN0IHNwaW4gPSBzcGlubmVyKFwiU2VuZGluZyBtZXNzYWdlLi4uXCIpO1xuICAgIHNwaW4uc3RhcnQoKTtcblxuICAgIHRyeSB7XG4gICAgICBjb25zdCByZXNwb25zZSA9IGF3YWl0IGFwaUNsaWVudC5wb3N0PFNlbmRNZXNzYWdlUmVzcG9uc2U+KFxuICAgICAgICBcIi9hcGkvdjEvbWVzc2FnZXNcIixcbiAgICAgICAge1xuICAgICAgICAgIHRvOiBmbGFncy50byxcbiAgICAgICAgICB0ZXh0OiBmbGFncy50ZXh0LFxuICAgICAgICAgIG1lc3NhZ2VUeXBlOiBmbGFncy50eXBlLFxuICAgICAgICAgIC4uLihmbGFncy5mcm9tICYmIHsgZnJvbTogZmxhZ3MuZnJvbSB9KSxcbiAgICAgICAgfSxcbiAgICAgICk7XG5cbiAgICAgIHNwaW4uc3RvcCgpO1xuXG4gICAgICBpZiAoaXNKc29uTW9kZSgpKSB7XG4gICAgICAgIGpzb24ocmVzcG9uc2UpO1xuICAgICAgICByZXR1cm47XG4gICAgICB9XG5cbiAgICAgIHN1Y2Nlc3MoXCJNZXNzYWdlIHNlbnRcIiwge1xuICAgICAgICBJRDogcmVzcG9uc2UuaWQsXG4gICAgICAgIFRvOiByZXNwb25zZS50byxcbiAgICAgICAgU3RhdHVzOiBmb3JtYXRTdGF0dXMocmVzcG9uc2Uuc3RhdHVzKSxcbiAgICAgICAgU2VnbWVudHM6IHJlc3BvbnNlLnNlZ21lbnRzLFxuICAgICAgICBDcmVkaXRzOiBmb3JtYXRDcmVkaXRzKHJlc3BvbnNlLmNyZWRpdHNVc2VkKSxcbiAgICAgIH0pO1xuICAgIH0gY2F0Y2ggKGVycikge1xuICAgICAgc3Bpbi5zdG9wKCk7XG4gICAgICB0aHJvdyBlcnI7XG4gICAgfVxuICB9XG59XG4iXX0=
|
package/oclif.manifest.json
CHANGED
|
@@ -682,69 +682,6 @@
|
|
|
682
682
|
"revoke.js"
|
|
683
683
|
]
|
|
684
684
|
},
|
|
685
|
-
"logs:tail": {
|
|
686
|
-
"aliases": [],
|
|
687
|
-
"args": {},
|
|
688
|
-
"description": "Tail logs in real-time (like stripe logs tail)",
|
|
689
|
-
"examples": [
|
|
690
|
-
"<%= config.bin %> logs tail",
|
|
691
|
-
"<%= config.bin %> logs tail --status failed",
|
|
692
|
-
"<%= config.bin %> logs tail --since 1h"
|
|
693
|
-
],
|
|
694
|
-
"flags": {
|
|
695
|
-
"json": {
|
|
696
|
-
"description": "Output in JSON format",
|
|
697
|
-
"name": "json",
|
|
698
|
-
"allowNo": false,
|
|
699
|
-
"type": "boolean"
|
|
700
|
-
},
|
|
701
|
-
"quiet": {
|
|
702
|
-
"char": "q",
|
|
703
|
-
"description": "Minimal output",
|
|
704
|
-
"name": "quiet",
|
|
705
|
-
"allowNo": false,
|
|
706
|
-
"type": "boolean"
|
|
707
|
-
},
|
|
708
|
-
"status": {
|
|
709
|
-
"char": "s",
|
|
710
|
-
"description": "Filter by status (sent, delivered, failed)",
|
|
711
|
-
"name": "status",
|
|
712
|
-
"hasDynamicHelp": false,
|
|
713
|
-
"multiple": false,
|
|
714
|
-
"type": "option"
|
|
715
|
-
},
|
|
716
|
-
"since": {
|
|
717
|
-
"description": "Show logs since (e.g., 1h, 30m, 1d)",
|
|
718
|
-
"name": "since",
|
|
719
|
-
"default": "1h",
|
|
720
|
-
"hasDynamicHelp": false,
|
|
721
|
-
"multiple": false,
|
|
722
|
-
"type": "option"
|
|
723
|
-
},
|
|
724
|
-
"type": {
|
|
725
|
-
"char": "t",
|
|
726
|
-
"description": "Filter by type (message, api_call, webhook)",
|
|
727
|
-
"name": "type",
|
|
728
|
-
"hasDynamicHelp": false,
|
|
729
|
-
"multiple": false,
|
|
730
|
-
"type": "option"
|
|
731
|
-
}
|
|
732
|
-
},
|
|
733
|
-
"hasDynamicHelp": false,
|
|
734
|
-
"hiddenAliases": [],
|
|
735
|
-
"id": "logs:tail",
|
|
736
|
-
"pluginAlias": "@sendly/cli",
|
|
737
|
-
"pluginName": "@sendly/cli",
|
|
738
|
-
"pluginType": "core",
|
|
739
|
-
"strict": true,
|
|
740
|
-
"isESM": true,
|
|
741
|
-
"relativePath": [
|
|
742
|
-
"dist",
|
|
743
|
-
"commands",
|
|
744
|
-
"logs",
|
|
745
|
-
"tail.js"
|
|
746
|
-
]
|
|
747
|
-
},
|
|
748
685
|
"sms:batch": {
|
|
749
686
|
"aliases": [],
|
|
750
687
|
"args": {},
|
|
@@ -755,7 +692,8 @@
|
|
|
755
692
|
"<%= config.bin %> sms batch --file phones.csv --text \"Your order is ready!\"",
|
|
756
693
|
"<%= config.bin %> sms batch --file recipients.csv --from \"Sendly\"",
|
|
757
694
|
"<%= config.bin %> sms batch --file messages.json --json",
|
|
758
|
-
"<%= config.bin %> sms batch --file phones.csv --text \"Hi\" --dry-run"
|
|
695
|
+
"<%= config.bin %> sms batch --file phones.csv --text \"Hi\" --dry-run",
|
|
696
|
+
"<%= config.bin %> sms batch --file phones.csv --text \"Your code: 123\" --type transactional"
|
|
759
697
|
],
|
|
760
698
|
"flags": {
|
|
761
699
|
"json": {
|
|
@@ -809,9 +747,21 @@
|
|
|
809
747
|
"multiple": false,
|
|
810
748
|
"type": "option"
|
|
811
749
|
},
|
|
750
|
+
"type": {
|
|
751
|
+
"description": "Message type: marketing (default) or transactional. Transactional bypasses quiet hours.",
|
|
752
|
+
"name": "type",
|
|
753
|
+
"default": "marketing",
|
|
754
|
+
"hasDynamicHelp": false,
|
|
755
|
+
"multiple": false,
|
|
756
|
+
"options": [
|
|
757
|
+
"marketing",
|
|
758
|
+
"transactional"
|
|
759
|
+
],
|
|
760
|
+
"type": "option"
|
|
761
|
+
},
|
|
812
762
|
"dry-run": {
|
|
813
763
|
"char": "d",
|
|
814
|
-
"description": "Preview batch without sending (validates access, shows cost breakdown)",
|
|
764
|
+
"description": "Preview batch without sending (validates access, shows cost and compliance breakdown)",
|
|
815
765
|
"name": "dry-run",
|
|
816
766
|
"allowNo": false,
|
|
817
767
|
"type": "boolean"
|
|
@@ -1007,6 +957,7 @@
|
|
|
1007
957
|
"examples": [
|
|
1008
958
|
"<%= config.bin %> sms schedule --to +15551234567 --text \"Reminder!\" --at \"2025-01-20T10:00:00Z\"",
|
|
1009
959
|
"<%= config.bin %> sms schedule --to +15551234567 --text \"Meeting in 1 hour\" --at \"2025-01-15T14:00:00Z\" --from \"Sendly\"",
|
|
960
|
+
"<%= config.bin %> sms schedule --to +15551234567 --text \"Your code: 123456\" --at \"2025-01-20T10:00:00Z\" --type transactional",
|
|
1010
961
|
"<%= config.bin %> sms schedule --to +15551234567 --text \"Hello!\" --at \"2025-01-20T10:00:00Z\" --json"
|
|
1011
962
|
],
|
|
1012
963
|
"flags": {
|
|
@@ -1057,6 +1008,18 @@
|
|
|
1057
1008
|
"hasDynamicHelp": false,
|
|
1058
1009
|
"multiple": false,
|
|
1059
1010
|
"type": "option"
|
|
1011
|
+
},
|
|
1012
|
+
"type": {
|
|
1013
|
+
"description": "Message type: marketing (default) or transactional. Transactional messages bypass quiet hours.",
|
|
1014
|
+
"name": "type",
|
|
1015
|
+
"default": "marketing",
|
|
1016
|
+
"hasDynamicHelp": false,
|
|
1017
|
+
"multiple": false,
|
|
1018
|
+
"options": [
|
|
1019
|
+
"marketing",
|
|
1020
|
+
"transactional"
|
|
1021
|
+
],
|
|
1022
|
+
"type": "option"
|
|
1060
1023
|
}
|
|
1061
1024
|
},
|
|
1062
1025
|
"hasDynamicHelp": false,
|
|
@@ -1138,6 +1101,7 @@
|
|
|
1138
1101
|
"examples": [
|
|
1139
1102
|
"<%= config.bin %> sms send --to +15551234567 --text \"Hello!\"",
|
|
1140
1103
|
"<%= config.bin %> sms send --to +15551234567 --text \"Hello!\" --from \"Sendly\"",
|
|
1104
|
+
"<%= config.bin %> sms send --to +15551234567 --text \"Hello!\" --type transactional",
|
|
1141
1105
|
"<%= config.bin %> sms send --to +15551234567 --text \"Hello!\" --json"
|
|
1142
1106
|
],
|
|
1143
1107
|
"flags": {
|
|
@@ -1179,6 +1143,18 @@
|
|
|
1179
1143
|
"hasDynamicHelp": false,
|
|
1180
1144
|
"multiple": false,
|
|
1181
1145
|
"type": "option"
|
|
1146
|
+
},
|
|
1147
|
+
"type": {
|
|
1148
|
+
"description": "Message type: marketing (default) or transactional",
|
|
1149
|
+
"name": "type",
|
|
1150
|
+
"default": "marketing",
|
|
1151
|
+
"hasDynamicHelp": false,
|
|
1152
|
+
"multiple": false,
|
|
1153
|
+
"options": [
|
|
1154
|
+
"marketing",
|
|
1155
|
+
"transactional"
|
|
1156
|
+
],
|
|
1157
|
+
"type": "option"
|
|
1182
1158
|
}
|
|
1183
1159
|
},
|
|
1184
1160
|
"hasDynamicHelp": false,
|
|
@@ -1196,6 +1172,69 @@
|
|
|
1196
1172
|
"send.js"
|
|
1197
1173
|
]
|
|
1198
1174
|
},
|
|
1175
|
+
"logs:tail": {
|
|
1176
|
+
"aliases": [],
|
|
1177
|
+
"args": {},
|
|
1178
|
+
"description": "Tail logs in real-time (like stripe logs tail)",
|
|
1179
|
+
"examples": [
|
|
1180
|
+
"<%= config.bin %> logs tail",
|
|
1181
|
+
"<%= config.bin %> logs tail --status failed",
|
|
1182
|
+
"<%= config.bin %> logs tail --since 1h"
|
|
1183
|
+
],
|
|
1184
|
+
"flags": {
|
|
1185
|
+
"json": {
|
|
1186
|
+
"description": "Output in JSON format",
|
|
1187
|
+
"name": "json",
|
|
1188
|
+
"allowNo": false,
|
|
1189
|
+
"type": "boolean"
|
|
1190
|
+
},
|
|
1191
|
+
"quiet": {
|
|
1192
|
+
"char": "q",
|
|
1193
|
+
"description": "Minimal output",
|
|
1194
|
+
"name": "quiet",
|
|
1195
|
+
"allowNo": false,
|
|
1196
|
+
"type": "boolean"
|
|
1197
|
+
},
|
|
1198
|
+
"status": {
|
|
1199
|
+
"char": "s",
|
|
1200
|
+
"description": "Filter by status (sent, delivered, failed)",
|
|
1201
|
+
"name": "status",
|
|
1202
|
+
"hasDynamicHelp": false,
|
|
1203
|
+
"multiple": false,
|
|
1204
|
+
"type": "option"
|
|
1205
|
+
},
|
|
1206
|
+
"since": {
|
|
1207
|
+
"description": "Show logs since (e.g., 1h, 30m, 1d)",
|
|
1208
|
+
"name": "since",
|
|
1209
|
+
"default": "1h",
|
|
1210
|
+
"hasDynamicHelp": false,
|
|
1211
|
+
"multiple": false,
|
|
1212
|
+
"type": "option"
|
|
1213
|
+
},
|
|
1214
|
+
"type": {
|
|
1215
|
+
"char": "t",
|
|
1216
|
+
"description": "Filter by type (message, api_call, webhook)",
|
|
1217
|
+
"name": "type",
|
|
1218
|
+
"hasDynamicHelp": false,
|
|
1219
|
+
"multiple": false,
|
|
1220
|
+
"type": "option"
|
|
1221
|
+
}
|
|
1222
|
+
},
|
|
1223
|
+
"hasDynamicHelp": false,
|
|
1224
|
+
"hiddenAliases": [],
|
|
1225
|
+
"id": "logs:tail",
|
|
1226
|
+
"pluginAlias": "@sendly/cli",
|
|
1227
|
+
"pluginName": "@sendly/cli",
|
|
1228
|
+
"pluginType": "core",
|
|
1229
|
+
"strict": true,
|
|
1230
|
+
"isESM": true,
|
|
1231
|
+
"relativePath": [
|
|
1232
|
+
"dist",
|
|
1233
|
+
"commands",
|
|
1234
|
+
"logs",
|
|
1235
|
+
"tail.js"
|
|
1236
|
+
]
|
|
1237
|
+
},
|
|
1199
1238
|
"webhooks:create": {
|
|
1200
1239
|
"aliases": [],
|
|
1201
1240
|
"args": {},
|
|
@@ -1723,5 +1762,5 @@
|
|
|
1723
1762
|
]
|
|
1724
1763
|
}
|
|
1725
1764
|
},
|
|
1726
|
-
"version": "3.3.
|
|
1765
|
+
"version": "3.3.2"
|
|
1727
1766
|
}
|