@sendly/cli 3.4.0 → 3.5.0
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 +9 -0
- package/dist/lib/api-client.js +62 -1
- package/oclif.manifest.json +41 -14
- package/package.json +2 -4
package/dist/commands/status.js
CHANGED
|
@@ -1,9 +1,9 @@
|
|
|
1
1
|
import { AuthenticatedCommand } from "../lib/base-command.js";
|
|
2
2
|
import { apiClient } from "../lib/api-client.js";
|
|
3
|
-
import { colors, json, isJsonMode, formatRelativeTime
|
|
3
|
+
import { colors, json, isJsonMode, formatRelativeTime } from "../lib/output.js";
|
|
4
4
|
import { getConfigValue } from "../lib/config.js";
|
|
5
5
|
export default class Status extends AuthenticatedCommand {
|
|
6
|
-
static description = "Show account status dashboard with credits, usage, and
|
|
6
|
+
static description = "Show account status dashboard with credits, usage, and capabilities";
|
|
7
7
|
static examples = [
|
|
8
8
|
"<%= config.bin %> status",
|
|
9
9
|
"<%= config.bin %> status --json",
|
|
@@ -12,7 +12,11 @@ export default class Status extends AuthenticatedCommand {
|
|
|
12
12
|
...AuthenticatedCommand.baseFlags,
|
|
13
13
|
};
|
|
14
14
|
async run() {
|
|
15
|
-
// Fetch
|
|
15
|
+
// Fetch comprehensive account status
|
|
16
|
+
const status = await apiClient
|
|
17
|
+
.get("/api/cli/account/status")
|
|
18
|
+
.catch(() => null);
|
|
19
|
+
// Fallback data if new endpoint not available
|
|
16
20
|
const [credits, messages, webhooks, keys] = await Promise.all([
|
|
17
21
|
apiClient
|
|
18
22
|
.get("/api/v1/account/credits")
|
|
@@ -22,91 +26,164 @@ export default class Status extends AuthenticatedCommand {
|
|
|
22
26
|
limit: 5,
|
|
23
27
|
})
|
|
24
28
|
.catch(() => ({ messages: [], total: 0 })),
|
|
25
|
-
apiClient
|
|
26
|
-
.get("/api/v1/webhooks")
|
|
27
|
-
.catch(() => []),
|
|
29
|
+
apiClient.get("/api/v1/webhooks").catch(() => []),
|
|
28
30
|
apiClient
|
|
29
31
|
.get("/api/v1/account/keys")
|
|
30
32
|
.catch(() => ({ keys: [] })),
|
|
31
33
|
]);
|
|
32
|
-
const email = getConfigValue("email") || "Unknown";
|
|
34
|
+
const email = status?.account?.email || getConfigValue("email") || "Unknown";
|
|
33
35
|
const apiMode = getConfigValue("environment") || "test";
|
|
34
36
|
if (isJsonMode()) {
|
|
35
37
|
json({
|
|
36
|
-
account: {
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
credits: {
|
|
38
|
+
account: status?.account || { email, tier: "sandbox" },
|
|
39
|
+
verification: status?.verification || null,
|
|
40
|
+
capabilities: status?.capabilities || { regions: ["sandbox"] },
|
|
41
|
+
credits: status?.credits || {
|
|
41
42
|
balance: credits.balance,
|
|
42
|
-
|
|
43
|
-
available: credits.availableBalance,
|
|
43
|
+
canSendLive: false,
|
|
44
44
|
},
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
45
|
+
keys: status?.keys || {
|
|
46
|
+
hasTestKey: false,
|
|
47
|
+
hasLiveKey: false,
|
|
48
|
+
totalActive: keys.keys?.filter((k) => k.isActive).length || 0,
|
|
49
49
|
},
|
|
50
|
+
nextSteps: status?.nextSteps || [],
|
|
50
51
|
});
|
|
51
52
|
return;
|
|
52
53
|
}
|
|
53
54
|
// Beautiful dashboard output
|
|
54
55
|
console.log();
|
|
55
|
-
console.log(colors.bold(colors.primary(" Sendly Status
|
|
56
|
-
console.log(colors.dim(" " + "─".repeat(
|
|
56
|
+
console.log(colors.bold(colors.primary(" Sendly Account Status")));
|
|
57
|
+
console.log(colors.dim(" " + "─".repeat(50)));
|
|
57
58
|
console.log();
|
|
58
|
-
// Account Section
|
|
59
|
+
// Account & Tier Section
|
|
59
60
|
console.log(colors.bold(" Account"));
|
|
60
|
-
console.log(` ${colors.dim("Email:")}
|
|
61
|
-
|
|
61
|
+
console.log(` ${colors.dim("Email:")} ${email}`);
|
|
62
|
+
// Show tier with color coding
|
|
63
|
+
const tier = status?.account?.tier || "sandbox";
|
|
64
|
+
const tierDisplay = {
|
|
65
|
+
sandbox: colors.warning("Sandbox") + colors.dim(" (test only)"),
|
|
66
|
+
international: colors.primary("International") + colors.dim(" (48 countries)"),
|
|
67
|
+
domestic: colors.success("US & Canada") + colors.dim(" (toll-free)"),
|
|
68
|
+
global: colors.success("Global") + colors.dim(" (full access)"),
|
|
69
|
+
};
|
|
70
|
+
console.log(` ${colors.dim("Tier:")} ${tierDisplay[tier]}`);
|
|
71
|
+
console.log();
|
|
72
|
+
// Verification Section
|
|
73
|
+
if (status?.verification) {
|
|
74
|
+
console.log(colors.bold(" Verification"));
|
|
75
|
+
const v = status.verification;
|
|
76
|
+
// Status with color
|
|
77
|
+
const statusColors = {
|
|
78
|
+
approved: colors.success,
|
|
79
|
+
verified: colors.success,
|
|
80
|
+
pending: colors.warning,
|
|
81
|
+
in_progress: colors.warning,
|
|
82
|
+
rejected: colors.error,
|
|
83
|
+
};
|
|
84
|
+
const statusColor = statusColors[v.status || ""] || colors.dim;
|
|
85
|
+
console.log(` ${colors.dim("Status:")} ${statusColor(v.status || "unknown")}`);
|
|
86
|
+
// Type
|
|
87
|
+
const typeLabels = {
|
|
88
|
+
toll_free: "Toll-Free (US/CA)",
|
|
89
|
+
international: "International",
|
|
90
|
+
both: "Global (US/CA + Intl)",
|
|
91
|
+
};
|
|
92
|
+
console.log(` ${colors.dim("Type:")} ${typeLabels[v.type || ""] || v.type}`);
|
|
93
|
+
// Show sender ID or toll-free number
|
|
94
|
+
if (v.alphaSenderId) {
|
|
95
|
+
console.log(` ${colors.dim("Sender ID:")} ${colors.primary(v.alphaSenderId)}`);
|
|
96
|
+
}
|
|
97
|
+
if (v.tollFreeNumber) {
|
|
98
|
+
console.log(` ${colors.dim("Toll-Free:")} ${colors.primary(v.tollFreeNumber)}`);
|
|
99
|
+
}
|
|
100
|
+
console.log();
|
|
101
|
+
}
|
|
102
|
+
// Capabilities Section
|
|
103
|
+
console.log(colors.bold(" Send Capabilities"));
|
|
104
|
+
const caps = status?.capabilities || {
|
|
105
|
+
canSendSandbox: true,
|
|
106
|
+
canSendInternational: false,
|
|
107
|
+
canSendDomestic: false,
|
|
108
|
+
};
|
|
109
|
+
// Sandbox
|
|
110
|
+
const sandboxIcon = caps.canSendSandbox
|
|
111
|
+
? colors.success("✓")
|
|
112
|
+
: colors.dim("○");
|
|
113
|
+
console.log(` ${sandboxIcon} ${colors.dim("Sandbox")} Test numbers only`);
|
|
114
|
+
// International
|
|
115
|
+
const intlIcon = caps.canSendInternational
|
|
116
|
+
? colors.success("✓")
|
|
117
|
+
: colors.dim("○");
|
|
118
|
+
const intlLabel = caps.canSendInternational
|
|
119
|
+
? "48 countries"
|
|
120
|
+
: colors.dim("Upgrade to unlock");
|
|
121
|
+
console.log(` ${intlIcon} ${colors.dim("International")} ${intlLabel}`);
|
|
122
|
+
// US & Canada
|
|
123
|
+
const usIcon = caps.canSendDomestic ? colors.success("✓") : colors.dim("○");
|
|
124
|
+
const usLabel = caps.canSendDomestic
|
|
125
|
+
? "Two-way messaging"
|
|
126
|
+
: colors.dim("Upgrade to unlock");
|
|
127
|
+
console.log(` ${usIcon} ${colors.dim("US & Canada")} ${usLabel}`);
|
|
62
128
|
console.log();
|
|
63
129
|
// Credits Section
|
|
130
|
+
const creditBalance = status?.credits?.balance ?? credits.balance ?? 0;
|
|
64
131
|
console.log(colors.bold(" Credits"));
|
|
65
|
-
const
|
|
66
|
-
const creditColor = available > 100
|
|
132
|
+
const creditColor = creditBalance > 100
|
|
67
133
|
? colors.success
|
|
68
|
-
:
|
|
134
|
+
: creditBalance > 10
|
|
69
135
|
? colors.warning
|
|
70
136
|
: colors.error;
|
|
71
|
-
console.log(` ${colors.dim("
|
|
137
|
+
console.log(` ${colors.dim("Balance:")} ${creditColor(creditBalance.toLocaleString())} credits`);
|
|
72
138
|
if (credits.reservedBalance > 0) {
|
|
73
|
-
console.log(` ${colors.dim("Reserved:")}
|
|
139
|
+
console.log(` ${colors.dim("Reserved:")} ${colors.warning(credits.reservedBalance.toLocaleString())} credits`);
|
|
74
140
|
}
|
|
75
|
-
console.log(` ${colors.dim("Capacity:")} ~${Math.floor(available).toLocaleString()} SMS (US/CA)`);
|
|
76
141
|
console.log();
|
|
77
|
-
//
|
|
78
|
-
|
|
79
|
-
|
|
142
|
+
// Resources Section
|
|
143
|
+
const activeKeys = status?.keys?.totalActive ??
|
|
144
|
+
keys.keys?.filter((k) => k.isActive).length ??
|
|
145
|
+
0;
|
|
80
146
|
const activeWebhooks = webhooks.filter((w) => w.is_active).length;
|
|
81
|
-
console.log(
|
|
82
|
-
console.log(` ${colors.dim("
|
|
147
|
+
console.log(colors.bold(" Resources"));
|
|
148
|
+
console.log(` ${colors.dim("API Keys:")} ${activeKeys} active`);
|
|
149
|
+
console.log(` ${colors.dim("Webhooks:")} ${activeWebhooks} configured`);
|
|
83
150
|
console.log();
|
|
84
151
|
// Recent Activity
|
|
85
152
|
if (messages.messages && messages.messages.length > 0) {
|
|
86
153
|
console.log(colors.bold(" Recent Messages"));
|
|
87
154
|
messages.messages.slice(0, 3).forEach((msg) => {
|
|
88
|
-
const
|
|
89
|
-
const statusIcon =
|
|
155
|
+
const msgStatus = msg.status || "unknown";
|
|
156
|
+
const statusIcon = msgStatus === "delivered"
|
|
90
157
|
? colors.success("✓")
|
|
91
|
-
:
|
|
158
|
+
: msgStatus === "sent"
|
|
92
159
|
? colors.primary("→")
|
|
93
|
-
:
|
|
160
|
+
: msgStatus === "failed"
|
|
94
161
|
? colors.error("✗")
|
|
95
162
|
: colors.dim("○");
|
|
96
163
|
const to = msg.to || "Unknown";
|
|
97
164
|
const time = msg.createdAt
|
|
98
165
|
? formatRelativeTime(msg.createdAt)
|
|
99
166
|
: "recently";
|
|
100
|
-
console.log(` ${statusIcon} ${colors.dim(to.slice(-4).padStart(8, "•"))} ${colors.dim(
|
|
167
|
+
console.log(` ${statusIcon} ${colors.dim(to.slice(-4).padStart(8, "•"))} ${colors.dim(msgStatus.padEnd(10))} ${colors.dim(time)}`);
|
|
168
|
+
});
|
|
169
|
+
console.log();
|
|
170
|
+
}
|
|
171
|
+
// Next Steps (if any)
|
|
172
|
+
if (status?.nextSteps && status.nextSteps.length > 0) {
|
|
173
|
+
console.log(colors.bold(colors.warning(" Next Steps")));
|
|
174
|
+
status.nextSteps.forEach((step, i) => {
|
|
175
|
+
console.log(` ${colors.warning(`${i + 1}.`)} ${step}`);
|
|
101
176
|
});
|
|
102
177
|
console.log();
|
|
103
178
|
}
|
|
104
179
|
// Quick Actions
|
|
105
180
|
console.log(colors.dim(" Quick Actions"));
|
|
106
181
|
console.log(` ${colors.code("sendly sms send")} Send a message`);
|
|
107
|
-
console.log(` ${colors.code("sendly
|
|
108
|
-
|
|
182
|
+
console.log(` ${colors.code("sendly keys list")} View API keys`);
|
|
183
|
+
if (tier === "sandbox") {
|
|
184
|
+
console.log(` ${colors.code("sendly onboarding")} Upgrade to production`);
|
|
185
|
+
}
|
|
109
186
|
console.log();
|
|
110
187
|
}
|
|
111
188
|
}
|
|
112
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"status.js","sourceRoot":"","sources":["../../src/commands/status.ts"],"names":[],"mappings":"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","sourcesContent":["import { AuthenticatedCommand } from \"../lib/base-command.js\";\nimport { apiClient } from \"../lib/api-client.js\";\nimport {\n  header,\n  keyValue,\n  colors,\n  json,\n  divider,\n  isJsonMode,\n  formatRelativeTime,\n} from \"../lib/output.js\";\nimport { getConfigValue } from \"../lib/config.js\";\n\ninterface StatusResponse {\n  account: {\n    email: string;\n    userId: string;\n    verified: boolean;\n  };\n  credits: {\n    balance: number;\n    reserved: number;\n    available: number;\n  };\n  usage: {\n    messagesSentToday: number;\n    messagesSentThisMonth: number;\n    webhooksConfigured: number;\n    activeApiKeys: number;\n  };\n  limits: {\n    dailyLimit: number;\n    monthlyLimit: number;\n    rateLimit: number;\n  };\n}\n\nexport default class Status extends AuthenticatedCommand {\n  static description =\n    \"Show account status dashboard with credits, usage, and health\";\n\n  static examples = [\n    \"<%= config.bin %> status\",\n    \"<%= config.bin %> status --json\",\n  ];\n\n  static flags = {\n    ...AuthenticatedCommand.baseFlags,\n  };\n\n  async run(): Promise<void> {\n    // Fetch all status data in parallel\n    const [credits, messages, webhooks, keys] = await Promise.all([\n      apiClient\n        .get<{\n          balance: number;\n          reservedBalance: number;\n          availableBalance: number;\n        }>(\"/api/v1/account/credits\")\n        .catch(() => ({ balance: 0, reservedBalance: 0, availableBalance: 0 })),\n      apiClient\n        .get<{ messages: any[]; total: number }>(\"/api/v1/messages\", {\n          limit: 5,\n        })\n        .catch(() => ({ messages: [], total: 0 })),\n      apiClient\n        .get<any[]>(\"/api/v1/webhooks\")\n        .catch(() => []),\n      apiClient\n        .get<{ keys: any[] }>(\"/api/v1/account/keys\")\n        .catch(() => ({ keys: [] })),\n    ]);\n\n    const email = getConfigValue(\"email\") || \"Unknown\";\n    const apiMode = getConfigValue(\"environment\") || \"test\";\n\n    if (isJsonMode()) {\n      json({\n        account: {\n          email,\n          apiMode,\n        },\n        credits: {\n          balance: credits.balance,\n          reserved: credits.reservedBalance,\n          available: credits.availableBalance,\n        },\n        usage: {\n          recentMessages: messages.total,\n          webhooksConfigured: webhooks.length,\n          activeApiKeys: keys.keys?.filter((k: any) => k.isActive).length || 0,\n        },\n      });\n      return;\n    }\n\n    // Beautiful dashboard output\n    console.log();\n    console.log(\n      colors.bold(colors.primary(\"  Sendly Status Dashboard\")),\n    );\n    console.log(colors.dim(\"  \" + \"─\".repeat(40)));\n    console.log();\n\n    // Account Section\n    console.log(colors.bold(\"  Account\"));\n    console.log(`    ${colors.dim(\"Email:\")}        ${email}`);\n    console.log(\n      `    ${colors.dim(\"API Mode:\")}     ${apiMode === \"test\" ? colors.warning(\"test\") + colors.dim(\" (sandbox)\") : colors.success(\"live\") + colors.dim(\" (production)\")}`,\n    );\n    console.log();\n\n    // Credits Section\n    console.log(colors.bold(\"  Credits\"));\n    const available = credits.availableBalance || 0;\n    const creditColor =\n      available > 100\n        ? colors.success\n        : available > 10\n          ? colors.warning\n          : colors.error;\n    console.log(\n      `    ${colors.dim(\"Available:\")}    ${creditColor(available.toLocaleString())} credits`,\n    );\n    if (credits.reservedBalance > 0) {\n      console.log(\n        `    ${colors.dim(\"Reserved:\")}     ${colors.warning(credits.reservedBalance.toLocaleString())} credits`,\n      );\n    }\n    console.log(\n      `    ${colors.dim(\"Capacity:\")}     ~${Math.floor(available).toLocaleString()} SMS (US/CA)`,\n    );\n    console.log();\n\n    // Usage Section\n    console.log(colors.bold(\"  Resources\"));\n    const activeKeys = keys.keys?.filter((k: any) => k.isActive).length || 0;\n    const activeWebhooks = webhooks.filter((w: any) => w.is_active).length;\n    console.log(\n      `    ${colors.dim(\"API Keys:\")}     ${activeKeys} active`,\n    );\n    console.log(\n      `    ${colors.dim(\"Webhooks:\")}     ${activeWebhooks} configured${webhooks.length > activeWebhooks ? colors.dim(` (${webhooks.length - activeWebhooks} paused)`) : \"\"}`,\n    );\n    console.log();\n\n    // Recent Activity\n    if (messages.messages && messages.messages.length > 0) {\n      console.log(colors.bold(\"  Recent Messages\"));\n      messages.messages.slice(0, 3).forEach((msg: any) => {\n        const status = msg.status || \"unknown\";\n        const statusIcon =\n          status === \"delivered\"\n            ? colors.success(\"✓\")\n            : status === \"sent\"\n              ? colors.primary(\"→\")\n              : status === \"failed\"\n                ? colors.error(\"✗\")\n                : colors.dim(\"○\");\n        const to = msg.to || \"Unknown\";\n        const time = msg.createdAt\n          ? formatRelativeTime(msg.createdAt)\n          : \"recently\";\n        console.log(\n          `    ${statusIcon} ${colors.dim(to.slice(-4).padStart(8, \"•\"))} ${colors.dim(status.padEnd(10))} ${colors.dim(time)}`,\n        );\n      });\n      console.log();\n    }\n\n    // Quick Actions\n    console.log(colors.dim(\"  Quick Actions\"));\n    console.log(\n      `    ${colors.code(\"sendly sms send\")}      Send a message`,\n    );\n    console.log(\n      `    ${colors.code(\"sendly credits balance\")} Check credit balance`,\n    );\n    console.log(\n      `    ${colors.code(\"sendly webhooks list\")}  View webhooks`,\n    );\n    console.log();\n  }\n}\n"]}
|
|
189
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"status.js","sourceRoot":"","sources":["../../src/commands/status.ts"],"names":[],"mappings":"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","sourcesContent":["import { AuthenticatedCommand } from \"../lib/base-command.js\";\nimport { apiClient } from \"../lib/api-client.js\";\nimport { colors, json, isJsonMode, formatRelativeTime } from \"../lib/output.js\";\nimport { getConfigValue } from \"../lib/config.js\";\n\ninterface AccountStatusResponse {\n  account: {\n    email: string;\n    tier: \"sandbox\" | \"international\" | \"domestic\" | \"global\";\n    onboardingCompleted: boolean;\n  };\n  verification: {\n    type: string | null;\n    status: string | null;\n    isVerified: boolean;\n    alphaSenderId: string | null;\n    tollFreeNumber: string | null;\n    businessName: string | null;\n  } | null;\n  capabilities: {\n    canSendSandbox: boolean;\n    canSendInternational: boolean;\n    canSendDomestic: boolean;\n    regions: string[];\n  };\n  credits: {\n    balance: number;\n    canSendLive: boolean;\n  };\n  keys: {\n    hasTestKey: boolean;\n    hasLiveKey: boolean;\n    totalActive: number;\n  };\n  nextSteps: string[];\n}\n\nexport default class Status extends AuthenticatedCommand {\n  static description =\n    \"Show account status dashboard with credits, usage, and capabilities\";\n\n  static examples = [\n    \"<%= config.bin %> status\",\n    \"<%= config.bin %> status --json\",\n  ];\n\n  static flags = {\n    ...AuthenticatedCommand.baseFlags,\n  };\n\n  async run(): Promise<void> {\n    // Fetch comprehensive account status\n    const status = await apiClient\n      .get<AccountStatusResponse>(\"/api/cli/account/status\")\n      .catch(() => null);\n\n    // Fallback data if new endpoint not available\n    const [credits, messages, webhooks, keys] = await Promise.all([\n      apiClient\n        .get<{\n          balance: number;\n          reservedBalance: number;\n          availableBalance: number;\n        }>(\"/api/v1/account/credits\")\n        .catch(() => ({ balance: 0, reservedBalance: 0, availableBalance: 0 })),\n      apiClient\n        .get<{ messages: any[]; total: number }>(\"/api/v1/messages\", {\n          limit: 5,\n        })\n        .catch(() => ({ messages: [], total: 0 })),\n      apiClient.get<any[]>(\"/api/v1/webhooks\").catch(() => []),\n      apiClient\n        .get<{ keys: any[] }>(\"/api/v1/account/keys\")\n        .catch(() => ({ keys: [] })),\n    ]);\n\n    const email =\n      status?.account?.email || getConfigValue(\"email\") || \"Unknown\";\n    const apiMode = getConfigValue(\"environment\") || \"test\";\n\n    if (isJsonMode()) {\n      json({\n        account: status?.account || { email, tier: \"sandbox\" },\n        verification: status?.verification || null,\n        capabilities: status?.capabilities || { regions: [\"sandbox\"] },\n        credits: status?.credits || {\n          balance: credits.balance,\n          canSendLive: false,\n        },\n        keys: status?.keys || {\n          hasTestKey: false,\n          hasLiveKey: false,\n          totalActive: keys.keys?.filter((k: any) => k.isActive).length || 0,\n        },\n        nextSteps: status?.nextSteps || [],\n      });\n      return;\n    }\n\n    // Beautiful dashboard output\n    console.log();\n    console.log(colors.bold(colors.primary(\"  Sendly Account Status\")));\n    console.log(colors.dim(\"  \" + \"─\".repeat(50)));\n    console.log();\n\n    // Account & Tier Section\n    console.log(colors.bold(\"  Account\"));\n    console.log(`    ${colors.dim(\"Email:\")}         ${email}`);\n\n    // Show tier with color coding\n    const tier = status?.account?.tier || \"sandbox\";\n    const tierDisplay = {\n      sandbox: colors.warning(\"Sandbox\") + colors.dim(\" (test only)\"),\n      international:\n        colors.primary(\"International\") + colors.dim(\" (48 countries)\"),\n      domestic: colors.success(\"US & Canada\") + colors.dim(\" (toll-free)\"),\n      global: colors.success(\"Global\") + colors.dim(\" (full access)\"),\n    };\n    console.log(`    ${colors.dim(\"Tier:\")}          ${tierDisplay[tier]}`);\n    console.log();\n\n    // Verification Section\n    if (status?.verification) {\n      console.log(colors.bold(\"  Verification\"));\n      const v = status.verification;\n\n      // Status with color\n      const statusColors: Record<string, (s: string) => string> = {\n        approved: colors.success,\n        verified: colors.success,\n        pending: colors.warning,\n        in_progress: colors.warning,\n        rejected: colors.error,\n      };\n      const statusColor = statusColors[v.status || \"\"] || colors.dim;\n      console.log(\n        `    ${colors.dim(\"Status:\")}        ${statusColor(v.status || \"unknown\")}`,\n      );\n\n      // Type\n      const typeLabels: Record<string, string> = {\n        toll_free: \"Toll-Free (US/CA)\",\n        international: \"International\",\n        both: \"Global (US/CA + Intl)\",\n      };\n      console.log(\n        `    ${colors.dim(\"Type:\")}          ${typeLabels[v.type || \"\"] || v.type}`,\n      );\n\n      // Show sender ID or toll-free number\n      if (v.alphaSenderId) {\n        console.log(\n          `    ${colors.dim(\"Sender ID:\")}     ${colors.primary(v.alphaSenderId)}`,\n        );\n      }\n      if (v.tollFreeNumber) {\n        console.log(\n          `    ${colors.dim(\"Toll-Free:\")}     ${colors.primary(v.tollFreeNumber)}`,\n        );\n      }\n      console.log();\n    }\n\n    // Capabilities Section\n    console.log(colors.bold(\"  Send Capabilities\"));\n    const caps = status?.capabilities || {\n      canSendSandbox: true,\n      canSendInternational: false,\n      canSendDomestic: false,\n    };\n\n    // Sandbox\n    const sandboxIcon = caps.canSendSandbox\n      ? colors.success(\"✓\")\n      : colors.dim(\"○\");\n    console.log(\n      `    ${sandboxIcon} ${colors.dim(\"Sandbox\")}        Test numbers only`,\n    );\n\n    // International\n    const intlIcon = caps.canSendInternational\n      ? colors.success(\"✓\")\n      : colors.dim(\"○\");\n    const intlLabel = caps.canSendInternational\n      ? \"48 countries\"\n      : colors.dim(\"Upgrade to unlock\");\n    console.log(`    ${intlIcon} ${colors.dim(\"International\")}  ${intlLabel}`);\n\n    // US & Canada\n    const usIcon = caps.canSendDomestic ? colors.success(\"✓\") : colors.dim(\"○\");\n    const usLabel = caps.canSendDomestic\n      ? \"Two-way messaging\"\n      : colors.dim(\"Upgrade to unlock\");\n    console.log(`    ${usIcon} ${colors.dim(\"US & Canada\")}    ${usLabel}`);\n    console.log();\n\n    // Credits Section\n    const creditBalance = status?.credits?.balance ?? credits.balance ?? 0;\n    console.log(colors.bold(\"  Credits\"));\n    const creditColor =\n      creditBalance > 100\n        ? colors.success\n        : creditBalance > 10\n          ? colors.warning\n          : colors.error;\n    console.log(\n      `    ${colors.dim(\"Balance:\")}       ${creditColor(creditBalance.toLocaleString())} credits`,\n    );\n    if (credits.reservedBalance > 0) {\n      console.log(\n        `    ${colors.dim(\"Reserved:\")}      ${colors.warning(credits.reservedBalance.toLocaleString())} credits`,\n      );\n    }\n    console.log();\n\n    // Resources Section\n    const activeKeys =\n      status?.keys?.totalActive ??\n      keys.keys?.filter((k: any) => k.isActive).length ??\n      0;\n    const activeWebhooks = webhooks.filter((w: any) => w.is_active).length;\n    console.log(colors.bold(\"  Resources\"));\n    console.log(`    ${colors.dim(\"API Keys:\")}      ${activeKeys} active`);\n    console.log(\n      `    ${colors.dim(\"Webhooks:\")}      ${activeWebhooks} configured`,\n    );\n    console.log();\n\n    // Recent Activity\n    if (messages.messages && messages.messages.length > 0) {\n      console.log(colors.bold(\"  Recent Messages\"));\n      messages.messages.slice(0, 3).forEach((msg: any) => {\n        const msgStatus = msg.status || \"unknown\";\n        const statusIcon =\n          msgStatus === \"delivered\"\n            ? colors.success(\"✓\")\n            : msgStatus === \"sent\"\n              ? colors.primary(\"→\")\n              : msgStatus === \"failed\"\n                ? colors.error(\"✗\")\n                : colors.dim(\"○\");\n        const to = msg.to || \"Unknown\";\n        const time = msg.createdAt\n          ? formatRelativeTime(msg.createdAt)\n          : \"recently\";\n        console.log(\n          `    ${statusIcon} ${colors.dim(to.slice(-4).padStart(8, \"•\"))} ${colors.dim(msgStatus.padEnd(10))} ${colors.dim(time)}`,\n        );\n      });\n      console.log();\n    }\n\n    // Next Steps (if any)\n    if (status?.nextSteps && status.nextSteps.length > 0) {\n      console.log(colors.bold(colors.warning(\"  Next Steps\")));\n      status.nextSteps.forEach((step, i) => {\n        console.log(`    ${colors.warning(`${i + 1}.`)} ${step}`);\n      });\n      console.log();\n    }\n\n    // Quick Actions\n    console.log(colors.dim(\"  Quick Actions\"));\n    console.log(`    ${colors.code(\"sendly sms send\")}      Send a message`);\n    console.log(`    ${colors.code(\"sendly keys list\")}     View API keys`);\n    if (tier === \"sandbox\") {\n      console.log(\n        `    ${colors.code(\"sendly onboarding\")}    Upgrade to production`,\n      );\n    }\n    console.log();\n  }\n}\n"]}
|
package/dist/lib/api-client.d.ts
CHANGED
|
@@ -47,6 +47,15 @@ declare class ApiClient {
|
|
|
47
47
|
post<T>(path: string, body?: Record<string, unknown>, requireAuth?: boolean): Promise<T>;
|
|
48
48
|
patch<T>(path: string, body?: Record<string, unknown>, requireAuth?: boolean): Promise<T>;
|
|
49
49
|
delete<T>(path: string, requireAuth?: boolean): Promise<T>;
|
|
50
|
+
/**
|
|
51
|
+
* Upload a file using multipart/form-data
|
|
52
|
+
* Used for batch CSV uploads to Supabase storage
|
|
53
|
+
*/
|
|
54
|
+
uploadFile<T>(path: string, file: {
|
|
55
|
+
buffer: Buffer;
|
|
56
|
+
filename: string;
|
|
57
|
+
mimetype?: string;
|
|
58
|
+
}, requireAuth?: boolean): Promise<T>;
|
|
50
59
|
}
|
|
51
60
|
export declare const apiClient: ApiClient;
|
|
52
61
|
export {};
|
package/dist/lib/api-client.js
CHANGED
|
@@ -175,6 +175,67 @@ class ApiClient {
|
|
|
175
175
|
async delete(path, requireAuth = true) {
|
|
176
176
|
return this.request("DELETE", path, { requireAuth });
|
|
177
177
|
}
|
|
178
|
+
/**
|
|
179
|
+
* Upload a file using multipart/form-data
|
|
180
|
+
* Used for batch CSV uploads to Supabase storage
|
|
181
|
+
*/
|
|
182
|
+
async uploadFile(path, file, requireAuth = true) {
|
|
183
|
+
const maxRetries = getEffectiveValue("maxRetries");
|
|
184
|
+
const timeout = getEffectiveValue("timeout");
|
|
185
|
+
const url = `${this.getBaseUrl()}${path}`;
|
|
186
|
+
// Build multipart form data manually (Node.js compatible)
|
|
187
|
+
const boundary = `----FormBoundary${Date.now()}${Math.random().toString(36).substring(2)}`;
|
|
188
|
+
const mimetype = file.mimetype || "text/csv";
|
|
189
|
+
const header = Buffer.from(`--${boundary}\r\n` +
|
|
190
|
+
`Content-Disposition: form-data; name="file"; filename="${file.filename}"\r\n` +
|
|
191
|
+
`Content-Type: ${mimetype}\r\n\r\n`);
|
|
192
|
+
const footer = Buffer.from(`\r\n--${boundary}--\r\n`);
|
|
193
|
+
const body = Buffer.concat([header, file.buffer, footer]);
|
|
194
|
+
const headers = {
|
|
195
|
+
"Content-Type": `multipart/form-data; boundary=${boundary}`,
|
|
196
|
+
Accept: "application/json",
|
|
197
|
+
"User-Agent": `@sendly/cli/${version}`,
|
|
198
|
+
};
|
|
199
|
+
if (requireAuth) {
|
|
200
|
+
const token = getAuthToken();
|
|
201
|
+
if (!token) {
|
|
202
|
+
throw new AuthenticationError();
|
|
203
|
+
}
|
|
204
|
+
headers["Authorization"] = `Bearer ${token}`;
|
|
205
|
+
}
|
|
206
|
+
let lastError;
|
|
207
|
+
for (let attempt = 0; attempt <= maxRetries; attempt++) {
|
|
208
|
+
try {
|
|
209
|
+
const controller = new AbortController();
|
|
210
|
+
const timeoutId = setTimeout(() => controller.abort(), timeout);
|
|
211
|
+
const response = await fetch(url, {
|
|
212
|
+
method: "POST",
|
|
213
|
+
headers,
|
|
214
|
+
body,
|
|
215
|
+
signal: controller.signal,
|
|
216
|
+
});
|
|
217
|
+
clearTimeout(timeoutId);
|
|
218
|
+
this.updateRateLimitInfo(response.headers);
|
|
219
|
+
const data = await response.json().catch(() => ({}));
|
|
220
|
+
if (!response.ok) {
|
|
221
|
+
this.handleError(response.status, data);
|
|
222
|
+
}
|
|
223
|
+
return data;
|
|
224
|
+
}
|
|
225
|
+
catch (error) {
|
|
226
|
+
lastError = error;
|
|
227
|
+
if (!isRetryableError(error)) {
|
|
228
|
+
throw error;
|
|
229
|
+
}
|
|
230
|
+
if (attempt === maxRetries) {
|
|
231
|
+
throw error;
|
|
232
|
+
}
|
|
233
|
+
const backoffMs = Math.min(1000 * Math.pow(2, attempt), 10000);
|
|
234
|
+
await sleep(backoffMs);
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
throw lastError || new Error("Upload failed");
|
|
238
|
+
}
|
|
178
239
|
}
|
|
179
240
|
export const apiClient = new ApiClient();
|
|
180
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"api-client.js","sourceRoot":"","sources":["../../src/lib/api-client.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAC5C,OAAO,EAAE,YAAY,EAAE,cAAc,EAAE,iBAAiB,EAAE,MAAM,aAAa,CAAC;AAE9E,iCAAiC;AACjC,MAAM,OAAO,GAAG,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;AAC/C,MAAM,EAAE,OAAO,EAAE,GAAG,OAAO,CAAC,oBAAoB,CAAwB,CAAC;AAEzE;;GAEG;AACH,SAAS,KAAK,CAAC,EAAU;IACvB,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,CAAC;AAC3D,CAAC;AAED;;GAEG;AACH,SAAS,gBAAgB,CAAC,KAAc;IACtC,iBAAiB;IACjB,IAAI,KAAK,YAAY,SAAS,IAAI,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;QAClE,OAAO,IAAI,CAAC;IACd,CAAC;IACD,oCAAoC;IACpC,IAAI,KAAK,YAAY,QAAQ,IAAI,KAAK,CAAC,UAAU,IAAI,GAAG,EAAE,CAAC;QACzD,OAAO,IAAI,CAAC;IACd,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC;AAiBD,MAAM,OAAO,QAAS,SAAQ,KAAK;IAExB;IAEA;IACA;IAJT,YACS,IAAY,EACnB,OAAe,EACR,UAAkB,EAClB,OAAiC;QAExC,KAAK,CAAC,OAAO,CAAC,CAAC;QALR,SAAI,GAAJ,IAAI,CAAQ;QAEZ,eAAU,GAAV,UAAU,CAAQ;QAClB,YAAO,GAAP,OAAO,CAA0B;QAGxC,IAAI,CAAC,IAAI,GAAG,UAAU,CAAC;IACzB,CAAC;CACF;AAED,MAAM,OAAO,mBAAoB,SAAQ,QAAQ;IAC/C,YACE,UAAkB,8CAA8C;QAEhE,KAAK,CAAC,sBAAsB,EAAE,OAAO,EAAE,GAAG,CAAC,CAAC;QAC5C,IAAI,CAAC,IAAI,GAAG,qBAAqB,CAAC;IACpC,CAAC;CACF;AAED,MAAM,OAAO,cAAe,SAAQ,QAAQ;IAEjC;IADT,YACS,UAAkB,EACzB,UAAkB,qBAAqB;QAEvC,KAAK,CAAC,qBAAqB,EAAE,OAAO,EAAE,GAAG,CAAC,CAAC;QAHpC,eAAU,GAAV,UAAU,CAAQ;QAIzB,IAAI,CAAC,IAAI,GAAG,gBAAgB,CAAC;IAC/B,CAAC;CACF;AAED,MAAM,OAAO,wBAAyB,SAAQ,QAAQ;IACpD,YAAY,UAAkB,sBAAsB;QAClD,KAAK,CAAC,sBAAsB,EAAE,OAAO,EAAE,GAAG,CAAC,CAAC;QAC5C,IAAI,CAAC,IAAI,GAAG,0BAA0B,CAAC;IACzC,CAAC;CACF;AAED,MAAM,SAAS;IACL,aAAa,CAAiB;IAE9B,UAAU;QAChB,OAAO,cAAc,CAAC,SAAS,CAAC,IAAI,qBAAqB,CAAC;IAC5D,CAAC;IAEO,UAAU,CAAC,cAAuB,IAAI;QAC5C,MAAM,OAAO,GAA2B;YACtC,cAAc,EAAE,kBAAkB;YAClC,MAAM,EAAE,kBAAkB;YAC1B,YAAY,EAAE,eAAe,OAAO,EAAE;SACvC,CAAC;QAEF,IAAI,WAAW,EAAE,CAAC;YAChB,MAAM,KAAK,GAAG,YAAY,EAAE,CAAC;YAC7B,IAAI,CAAC,KAAK,EAAE,CAAC;gBACX,MAAM,IAAI,mBAAmB,EAAE,CAAC;YAClC,CAAC;YACD,OAAO,CAAC,eAAe,CAAC,GAAG,UAAU,KAAK,EAAE,CAAC;QAC/C,CAAC;QAED,OAAO,OAAO,CAAC;IACjB,CAAC;IAED,KAAK,CAAC,OAAO,CACX,MAAc,EACd,IAAY,EACZ,UAII,EAAE;QAEN,MAAM,EAAE,IAAI,EAAE,KAAK,EAAE,WAAW,GAAG,IAAI,EAAE,GAAG,OAAO,CAAC;QACpD,MAAM,UAAU,GAAG,iBAAiB,CAAC,YAAY,CAAC,CAAC;QACnD,MAAM,OAAO,GAAG,iBAAiB,CAAC,SAAS,CAAC,CAAC;QAE7C,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC,GAAG,IAAI,CAAC,UAAU,EAAE,GAAG,IAAI,EAAE,CAAC,CAAC;QACnD,IAAI,KAAK,EAAE,CAAC;YACV,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,EAAE,KAAK,CAAC,EAAE,EAAE;gBAC7C,IAAI,KAAK,KAAK,SAAS,EAAE,CAAC;oBACxB,GAAG,CAAC,YAAY,CAAC,MAAM,CAAC,GAAG,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;gBAC9C,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC;QAED,IAAI,SAA4B,CAAC;QAEjC,KAAK,IAAI,OAAO,GAAG,CAAC,EAAE,OAAO,IAAI,UAAU,EAAE,OAAO,EAAE,EAAE,CAAC;YACvD,IAAI,CAAC;gBACH,MAAM,UAAU,GAAG,IAAI,eAAe,EAAE,CAAC;gBACzC,MAAM,SAAS,GAAG,UAAU,CAAC,GAAG,EAAE,CAAC,UAAU,CAAC,KAAK,EAAE,EAAE,OAAO,CAAC,CAAC;gBAEhE,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,GAAG,CAAC,QAAQ,EAAE,EAAE;oBAC3C,MAAM;oBACN,OAAO,EAAE,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC;oBACrC,IAAI,EAAE,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,SAAS;oBAC7C,MAAM,EAAE,UAAU,CAAC,MAAM;iBAC1B,CAAC,CAAC;gBAEH,YAAY,CAAC,SAAS,CAAC,CAAC;gBAExB,yBAAyB;gBACzB,IAAI,CAAC,mBAAmB,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAE3C,iBAAiB;gBACjB,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;gBAErD,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAE,CAAC;oBACjB,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;gBAC1C,CAAC;gBAED,OAAO,IAAS,CAAC;YACnB,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,SAAS,GAAG,KAAc,CAAC;gBAE3B,uDAAuD;gBACvD,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,EAAE,CAAC;oBAC7B,MAAM,KAAK,CAAC;gBACd,CAAC;gBAED,8BAA8B;gBAC9B,IAAI,OAAO,KAAK,UAAU,EAAE,CAAC;oBAC3B,MAAM,KAAK,CAAC;gBACd,CAAC;gBAED,wCAAwC;gBACxC,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,OAAO,CAAC,EAAE,KAAK,CAAC,CAAC;gBAC/D,MAAM,KAAK,CAAC,SAAS,CAAC,CAAC;YACzB,CAAC;QACH,CAAC;QAED,qDAAqD;QACrD,MAAM,SAAS,IAAI,IAAI,KAAK,CAAC,gBAAgB,CAAC,CAAC;IACjD,CAAC;IAEO,mBAAmB,CAAC,OAAgB;QAC1C,MAAM,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,mBAAmB,CAAC,CAAC;QAC/C,MAAM,SAAS,GAAG,OAAO,CAAC,GAAG,CAAC,uBAAuB,CAAC,CAAC;QACvD,MAAM,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,mBAAmB,CAAC,CAAC;QAE/C,IAAI,KAAK,IAAI,SAAS,IAAI,KAAK,EAAE,CAAC;YAChC,IAAI,CAAC,aAAa,GAAG;gBACnB,KAAK,EAAE,QAAQ,CAAC,KAAK,EAAE,EAAE,CAAC;gBAC1B,SAAS,EAAE,QAAQ,CAAC,SAAS,EAAE,EAAE,CAAC;gBAClC,KAAK,EAAE,QAAQ,CAAC,KAAK,EAAE,EAAE,CAAC;aAC3B,CAAC;QACJ,CAAC;IACH,CAAC;IAEO,WAAW,CAAC,UAAkB,EAAE,IAAS;QAC/C,MAAM,KAAK,GAAG,IAAI,EAAE,KAAK,IAAI,eAAe,CAAC;QAC7C,MAAM,OAAO,GAAG,IAAI,EAAE,OAAO,IAAI,QAAQ,UAAU,EAAE,CAAC;QACtD,MAAM,OAAO,GAAG,IAAI,EAAE,OAAO,CAAC;QAE9B,QAAQ,UAAU,EAAE,CAAC;YACnB,KAAK,GAAG,CAAC;YACT,KAAK,GAAG;gBACN,MAAM,IAAI,mBAAmB,CAAC,OAAO,CAAC,CAAC;YACzC,KAAK,GAAG;gBACN,MAAM,IAAI,wBAAwB,CAAC,OAAO,CAAC,CAAC;YAC9C,KAAK,GAAG;gBACN,MAAM,UAAU,GAAG,IAAI,EAAE,UAAU,IAAI,EAAE,CAAC;gBAC1C,MAAM,IAAI,cAAc,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;YAChD;gBACE,MAAM,IAAI,QAAQ,CAAC,KAAK,EAAE,OAAO,EAAE,UAAU,EAAE,OAAO,CAAC,CAAC;QAC5D,CAAC;IACH,CAAC;IAED,gBAAgB;QACd,OAAO,IAAI,CAAC,aAAa,CAAC;IAC5B,CAAC;IAED,sBAAsB;IACtB,KAAK,CAAC,GAAG,CACP,IAAY,EACZ,KAA6D,EAC7D,cAAuB,IAAI;QAE3B,OAAO,IAAI,CAAC,OAAO,CAAI,KAAK,EAAE,IAAI,EAAE,EAAE,KAAK,EAAE,WAAW,EAAE,CAAC,CAAC;IAC9D,CAAC;IAED,KAAK,CAAC,IAAI,CACR,IAAY,EACZ,IAA8B,EAC9B,cAAuB,IAAI;QAE3B,OAAO,IAAI,CAAC,OAAO,CAAI,MAAM,EAAE,IAAI,EAAE,EAAE,IAAI,EAAE,WAAW,EAAE,CAAC,CAAC;IAC9D,CAAC;IAED,KAAK,CAAC,KAAK,CACT,IAAY,EACZ,IAA8B,EAC9B,cAAuB,IAAI;QAE3B,OAAO,IAAI,CAAC,OAAO,CAAI,OAAO,EAAE,IAAI,EAAE,EAAE,IAAI,EAAE,WAAW,EAAE,CAAC,CAAC;IAC/D,CAAC;IAED,KAAK,CAAC,MAAM,CAAI,IAAY,EAAE,cAAuB,IAAI;QACvD,OAAO,IAAI,CAAC,OAAO,CAAI,QAAQ,EAAE,IAAI,EAAE,EAAE,WAAW,EAAE,CAAC,CAAC;IAC1D,CAAC;CACF;AAED,MAAM,CAAC,MAAM,SAAS,GAAG,IAAI,SAAS,EAAE,CAAC","sourcesContent":["/**\n * API Client for Sendly CLI\n * Handles all HTTP requests to the Sendly API\n */\n\nimport { createRequire } from \"node:module\";\nimport { getAuthToken, getConfigValue, getEffectiveValue } from \"./config.js\";\n\n// Read version from package.json\nconst require = createRequire(import.meta.url);\nconst { version } = require(\"../../package.json\") as { version: string };\n\n/**\n * Sleep for a given number of milliseconds\n */\nfunction sleep(ms: number): Promise<void> {\n  return new Promise((resolve) => setTimeout(resolve, ms));\n}\n\n/**\n * Check if an error is retryable (network errors or 5xx server errors)\n */\nfunction isRetryableError(error: unknown): boolean {\n  // Network errors\n  if (error instanceof TypeError && error.message.includes(\"fetch\")) {\n    return true;\n  }\n  // Server errors (5xx) are retryable\n  if (error instanceof ApiError && error.statusCode >= 500) {\n    return true;\n  }\n  return false;\n}\n\nexport interface ApiResponse<T> {\n  data?: T;\n  error?: {\n    code: string;\n    message: string;\n    details?: Record<string, unknown>;\n  };\n}\n\nexport interface RateLimitInfo {\n  limit: number;\n  remaining: number;\n  reset: number;\n}\n\nexport class ApiError extends Error {\n  constructor(\n    public code: string,\n    message: string,\n    public statusCode: number,\n    public details?: Record<string, unknown>,\n  ) {\n    super(message);\n    this.name = \"ApiError\";\n  }\n}\n\nexport class AuthenticationError extends ApiError {\n  constructor(\n    message: string = \"Not authenticated. Run 'sendly login' first.\",\n  ) {\n    super(\"authentication_error\", message, 401);\n    this.name = \"AuthenticationError\";\n  }\n}\n\nexport class RateLimitError extends ApiError {\n  constructor(\n    public retryAfter: number,\n    message: string = \"Rate limit exceeded\",\n  ) {\n    super(\"rate_limit_exceeded\", message, 429);\n    this.name = \"RateLimitError\";\n  }\n}\n\nexport class InsufficientCreditsError extends ApiError {\n  constructor(message: string = \"Insufficient credits\") {\n    super(\"insufficient_credits\", message, 402);\n    this.name = \"InsufficientCreditsError\";\n  }\n}\n\nclass ApiClient {\n  private rateLimitInfo?: RateLimitInfo;\n\n  private getBaseUrl(): string {\n    return getConfigValue(\"baseUrl\") || \"https://sendly.live\";\n  }\n\n  private getHeaders(requireAuth: boolean = true): Record<string, string> {\n    const headers: Record<string, string> = {\n      \"Content-Type\": \"application/json\",\n      Accept: \"application/json\",\n      \"User-Agent\": `@sendly/cli/${version}`,\n    };\n\n    if (requireAuth) {\n      const token = getAuthToken();\n      if (!token) {\n        throw new AuthenticationError();\n      }\n      headers[\"Authorization\"] = `Bearer ${token}`;\n    }\n\n    return headers;\n  }\n\n  async request<T>(\n    method: string,\n    path: string,\n    options: {\n      body?: Record<string, unknown>;\n      query?: Record<string, string | number | boolean | undefined>;\n      requireAuth?: boolean;\n    } = {},\n  ): Promise<T> {\n    const { body, query, requireAuth = true } = options;\n    const maxRetries = getEffectiveValue(\"maxRetries\");\n    const timeout = getEffectiveValue(\"timeout\");\n\n    const url = new URL(`${this.getBaseUrl()}${path}`);\n    if (query) {\n      Object.entries(query).forEach(([key, value]) => {\n        if (value !== undefined) {\n          url.searchParams.append(key, String(value));\n        }\n      });\n    }\n\n    let lastError: Error | undefined;\n\n    for (let attempt = 0; attempt <= maxRetries; attempt++) {\n      try {\n        const controller = new AbortController();\n        const timeoutId = setTimeout(() => controller.abort(), timeout);\n\n        const response = await fetch(url.toString(), {\n          method,\n          headers: this.getHeaders(requireAuth),\n          body: body ? JSON.stringify(body) : undefined,\n          signal: controller.signal,\n        });\n\n        clearTimeout(timeoutId);\n\n        // Update rate limit info\n        this.updateRateLimitInfo(response.headers);\n\n        // Parse response\n        const data = await response.json().catch(() => ({}));\n\n        if (!response.ok) {\n          this.handleError(response.status, data);\n        }\n\n        return data as T;\n      } catch (error) {\n        lastError = error as Error;\n\n        // Don't retry non-retryable errors (4xx client errors)\n        if (!isRetryableError(error)) {\n          throw error;\n        }\n\n        // Don't retry on last attempt\n        if (attempt === maxRetries) {\n          throw error;\n        }\n\n        // Exponential backoff: 1s, 2s, 4s, etc.\n        const backoffMs = Math.min(1000 * Math.pow(2, attempt), 10000);\n        await sleep(backoffMs);\n      }\n    }\n\n    // Should never reach here, but TypeScript needs this\n    throw lastError || new Error(\"Request failed\");\n  }\n\n  private updateRateLimitInfo(headers: Headers): void {\n    const limit = headers.get(\"X-RateLimit-Limit\");\n    const remaining = headers.get(\"X-RateLimit-Remaining\");\n    const reset = headers.get(\"X-RateLimit-Reset\");\n\n    if (limit && remaining && reset) {\n      this.rateLimitInfo = {\n        limit: parseInt(limit, 10),\n        remaining: parseInt(remaining, 10),\n        reset: parseInt(reset, 10),\n      };\n    }\n  }\n\n  private handleError(statusCode: number, data: any): never {\n    const error = data?.error || \"unknown_error\";\n    const message = data?.message || `HTTP ${statusCode}`;\n    const details = data?.details;\n\n    switch (statusCode) {\n      case 401:\n      case 403:\n        throw new AuthenticationError(message);\n      case 402:\n        throw new InsufficientCreditsError(message);\n      case 429:\n        const retryAfter = data?.retryAfter || 60;\n        throw new RateLimitError(retryAfter, message);\n      default:\n        throw new ApiError(error, message, statusCode, details);\n    }\n  }\n\n  getRateLimitInfo(): RateLimitInfo | undefined {\n    return this.rateLimitInfo;\n  }\n\n  // Convenience methods\n  async get<T>(\n    path: string,\n    query?: Record<string, string | number | boolean | undefined>,\n    requireAuth: boolean = true,\n  ): Promise<T> {\n    return this.request<T>(\"GET\", path, { query, requireAuth });\n  }\n\n  async post<T>(\n    path: string,\n    body?: Record<string, unknown>,\n    requireAuth: boolean = true,\n  ): Promise<T> {\n    return this.request<T>(\"POST\", path, { body, requireAuth });\n  }\n\n  async patch<T>(\n    path: string,\n    body?: Record<string, unknown>,\n    requireAuth: boolean = true,\n  ): Promise<T> {\n    return this.request<T>(\"PATCH\", path, { body, requireAuth });\n  }\n\n  async delete<T>(path: string, requireAuth: boolean = true): Promise<T> {\n    return this.request<T>(\"DELETE\", path, { requireAuth });\n  }\n}\n\nexport const apiClient = new ApiClient();\n"]}
|
|
241
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"api-client.js","sourceRoot":"","sources":["../../src/lib/api-client.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * API Client for Sendly CLI\n * Handles all HTTP requests to the Sendly API\n */\n\nimport { createRequire } from \"node:module\";\nimport { getAuthToken, getConfigValue, getEffectiveValue } from \"./config.js\";\n\n// Read version from package.json\nconst require = createRequire(import.meta.url);\nconst { version } = require(\"../../package.json\") as { version: string };\n\n/**\n * Sleep for a given number of milliseconds\n */\nfunction sleep(ms: number): Promise<void> {\n  return new Promise((resolve) => setTimeout(resolve, ms));\n}\n\n/**\n * Check if an error is retryable (network errors or 5xx server errors)\n */\nfunction isRetryableError(error: unknown): boolean {\n  // Network errors\n  if (error instanceof TypeError && error.message.includes(\"fetch\")) {\n    return true;\n  }\n  // Server errors (5xx) are retryable\n  if (error instanceof ApiError && error.statusCode >= 500) {\n    return true;\n  }\n  return false;\n}\n\nexport interface ApiResponse<T> {\n  data?: T;\n  error?: {\n    code: string;\n    message: string;\n    details?: Record<string, unknown>;\n  };\n}\n\nexport interface RateLimitInfo {\n  limit: number;\n  remaining: number;\n  reset: number;\n}\n\nexport class ApiError extends Error {\n  constructor(\n    public code: string,\n    message: string,\n    public statusCode: number,\n    public details?: Record<string, unknown>,\n  ) {\n    super(message);\n    this.name = \"ApiError\";\n  }\n}\n\nexport class AuthenticationError extends ApiError {\n  constructor(\n    message: string = \"Not authenticated. Run 'sendly login' first.\",\n  ) {\n    super(\"authentication_error\", message, 401);\n    this.name = \"AuthenticationError\";\n  }\n}\n\nexport class RateLimitError extends ApiError {\n  constructor(\n    public retryAfter: number,\n    message: string = \"Rate limit exceeded\",\n  ) {\n    super(\"rate_limit_exceeded\", message, 429);\n    this.name = \"RateLimitError\";\n  }\n}\n\nexport class InsufficientCreditsError extends ApiError {\n  constructor(message: string = \"Insufficient credits\") {\n    super(\"insufficient_credits\", message, 402);\n    this.name = \"InsufficientCreditsError\";\n  }\n}\n\nclass ApiClient {\n  private rateLimitInfo?: RateLimitInfo;\n\n  private getBaseUrl(): string {\n    return getConfigValue(\"baseUrl\") || \"https://sendly.live\";\n  }\n\n  private getHeaders(requireAuth: boolean = true): Record<string, string> {\n    const headers: Record<string, string> = {\n      \"Content-Type\": \"application/json\",\n      Accept: \"application/json\",\n      \"User-Agent\": `@sendly/cli/${version}`,\n    };\n\n    if (requireAuth) {\n      const token = getAuthToken();\n      if (!token) {\n        throw new AuthenticationError();\n      }\n      headers[\"Authorization\"] = `Bearer ${token}`;\n    }\n\n    return headers;\n  }\n\n  async request<T>(\n    method: string,\n    path: string,\n    options: {\n      body?: Record<string, unknown>;\n      query?: Record<string, string | number | boolean | undefined>;\n      requireAuth?: boolean;\n    } = {},\n  ): Promise<T> {\n    const { body, query, requireAuth = true } = options;\n    const maxRetries = getEffectiveValue(\"maxRetries\");\n    const timeout = getEffectiveValue(\"timeout\");\n\n    const url = new URL(`${this.getBaseUrl()}${path}`);\n    if (query) {\n      Object.entries(query).forEach(([key, value]) => {\n        if (value !== undefined) {\n          url.searchParams.append(key, String(value));\n        }\n      });\n    }\n\n    let lastError: Error | undefined;\n\n    for (let attempt = 0; attempt <= maxRetries; attempt++) {\n      try {\n        const controller = new AbortController();\n        const timeoutId = setTimeout(() => controller.abort(), timeout);\n\n        const response = await fetch(url.toString(), {\n          method,\n          headers: this.getHeaders(requireAuth),\n          body: body ? JSON.stringify(body) : undefined,\n          signal: controller.signal,\n        });\n\n        clearTimeout(timeoutId);\n\n        // Update rate limit info\n        this.updateRateLimitInfo(response.headers);\n\n        // Parse response\n        const data = await response.json().catch(() => ({}));\n\n        if (!response.ok) {\n          this.handleError(response.status, data);\n        }\n\n        return data as T;\n      } catch (error) {\n        lastError = error as Error;\n\n        // Don't retry non-retryable errors (4xx client errors)\n        if (!isRetryableError(error)) {\n          throw error;\n        }\n\n        // Don't retry on last attempt\n        if (attempt === maxRetries) {\n          throw error;\n        }\n\n        // Exponential backoff: 1s, 2s, 4s, etc.\n        const backoffMs = Math.min(1000 * Math.pow(2, attempt), 10000);\n        await sleep(backoffMs);\n      }\n    }\n\n    // Should never reach here, but TypeScript needs this\n    throw lastError || new Error(\"Request failed\");\n  }\n\n  private updateRateLimitInfo(headers: Headers): void {\n    const limit = headers.get(\"X-RateLimit-Limit\");\n    const remaining = headers.get(\"X-RateLimit-Remaining\");\n    const reset = headers.get(\"X-RateLimit-Reset\");\n\n    if (limit && remaining && reset) {\n      this.rateLimitInfo = {\n        limit: parseInt(limit, 10),\n        remaining: parseInt(remaining, 10),\n        reset: parseInt(reset, 10),\n      };\n    }\n  }\n\n  private handleError(statusCode: number, data: any): never {\n    const error = data?.error || \"unknown_error\";\n    const message = data?.message || `HTTP ${statusCode}`;\n    const details = data?.details;\n\n    switch (statusCode) {\n      case 401:\n      case 403:\n        throw new AuthenticationError(message);\n      case 402:\n        throw new InsufficientCreditsError(message);\n      case 429:\n        const retryAfter = data?.retryAfter || 60;\n        throw new RateLimitError(retryAfter, message);\n      default:\n        throw new ApiError(error, message, statusCode, details);\n    }\n  }\n\n  getRateLimitInfo(): RateLimitInfo | undefined {\n    return this.rateLimitInfo;\n  }\n\n  // Convenience methods\n  async get<T>(\n    path: string,\n    query?: Record<string, string | number | boolean | undefined>,\n    requireAuth: boolean = true,\n  ): Promise<T> {\n    return this.request<T>(\"GET\", path, { query, requireAuth });\n  }\n\n  async post<T>(\n    path: string,\n    body?: Record<string, unknown>,\n    requireAuth: boolean = true,\n  ): Promise<T> {\n    return this.request<T>(\"POST\", path, { body, requireAuth });\n  }\n\n  async patch<T>(\n    path: string,\n    body?: Record<string, unknown>,\n    requireAuth: boolean = true,\n  ): Promise<T> {\n    return this.request<T>(\"PATCH\", path, { body, requireAuth });\n  }\n\n  async delete<T>(path: string, requireAuth: boolean = true): Promise<T> {\n    return this.request<T>(\"DELETE\", path, { requireAuth });\n  }\n\n  /**\n   * Upload a file using multipart/form-data\n   * Used for batch CSV uploads to Supabase storage\n   */\n  async uploadFile<T>(\n    path: string,\n    file: {\n      buffer: Buffer;\n      filename: string;\n      mimetype?: string;\n    },\n    requireAuth: boolean = true,\n  ): Promise<T> {\n    const maxRetries = getEffectiveValue(\"maxRetries\");\n    const timeout = getEffectiveValue(\"timeout\");\n    const url = `${this.getBaseUrl()}${path}`;\n\n    // Build multipart form data manually (Node.js compatible)\n    const boundary = `----FormBoundary${Date.now()}${Math.random().toString(36).substring(2)}`;\n    const mimetype = file.mimetype || \"text/csv\";\n\n    const header = Buffer.from(\n      `--${boundary}\\r\\n` +\n        `Content-Disposition: form-data; name=\"file\"; filename=\"${file.filename}\"\\r\\n` +\n        `Content-Type: ${mimetype}\\r\\n\\r\\n`,\n    );\n    const footer = Buffer.from(`\\r\\n--${boundary}--\\r\\n`);\n    const body = Buffer.concat([header, file.buffer, footer]);\n\n    const headers: Record<string, string> = {\n      \"Content-Type\": `multipart/form-data; boundary=${boundary}`,\n      Accept: \"application/json\",\n      \"User-Agent\": `@sendly/cli/${version}`,\n    };\n\n    if (requireAuth) {\n      const token = getAuthToken();\n      if (!token) {\n        throw new AuthenticationError();\n      }\n      headers[\"Authorization\"] = `Bearer ${token}`;\n    }\n\n    let lastError: Error | undefined;\n\n    for (let attempt = 0; attempt <= maxRetries; attempt++) {\n      try {\n        const controller = new AbortController();\n        const timeoutId = setTimeout(() => controller.abort(), timeout);\n\n        const response = await fetch(url, {\n          method: \"POST\",\n          headers,\n          body,\n          signal: controller.signal,\n        });\n\n        clearTimeout(timeoutId);\n\n        this.updateRateLimitInfo(response.headers);\n        const data = await response.json().catch(() => ({}));\n\n        if (!response.ok) {\n          this.handleError(response.status, data);\n        }\n\n        return data as T;\n      } catch (error) {\n        lastError = error as Error;\n\n        if (!isRetryableError(error)) {\n          throw error;\n        }\n\n        if (attempt === maxRetries) {\n          throw error;\n        }\n\n        const backoffMs = Math.min(1000 * Math.pow(2, attempt), 10000);\n        await sleep(backoffMs);\n      }\n    }\n\n    throw lastError || new Error(\"Upload failed\");\n  }\n}\n\nexport const apiClient = new ApiClient();\n"]}
|
package/oclif.manifest.json
CHANGED
|
@@ -219,7 +219,7 @@
|
|
|
219
219
|
"status": {
|
|
220
220
|
"aliases": [],
|
|
221
221
|
"args": {},
|
|
222
|
-
"description": "Show account status dashboard with credits, usage, and
|
|
222
|
+
"description": "Show account status dashboard with credits, usage, and capabilities",
|
|
223
223
|
"examples": [
|
|
224
224
|
"<%= config.bin %> status",
|
|
225
225
|
"<%= config.bin %> status --json"
|
|
@@ -748,15 +748,15 @@
|
|
|
748
748
|
"sms:batch": {
|
|
749
749
|
"aliases": [],
|
|
750
750
|
"args": {},
|
|
751
|
-
"description": "Send batch SMS messages",
|
|
751
|
+
"description": "Send batch SMS messages (uploads CSV to cloud for audit trail)",
|
|
752
752
|
"examples": [
|
|
753
|
-
"<%= config.bin %> sms batch --file recipients.
|
|
754
|
-
"<%= config.bin %> sms batch --
|
|
755
|
-
"<%= config.bin %> sms batch --
|
|
756
|
-
"<%= config.bin %> sms batch --file recipients.csv --
|
|
757
|
-
"<%= config.bin %> sms batch --file
|
|
758
|
-
"<%= config.bin %> sms batch --
|
|
759
|
-
"<%= config.bin %> sms batch --
|
|
753
|
+
"<%= config.bin %> sms batch --file recipients.csv",
|
|
754
|
+
"<%= config.bin %> sms batch --file phones.csv --text \"Hello everyone!\"",
|
|
755
|
+
"<%= config.bin %> sms batch --to +15551234567,+15559876543 --text \"Hello!\"",
|
|
756
|
+
"<%= config.bin %> sms batch --file recipients.csv --dry-run",
|
|
757
|
+
"<%= config.bin %> sms batch --file phones.csv --text \"Code: 123\" --type transactional",
|
|
758
|
+
"<%= config.bin %> sms batch --reuse abc123-def456",
|
|
759
|
+
"<%= config.bin %> sms batch --history"
|
|
760
760
|
],
|
|
761
761
|
"flags": {
|
|
762
762
|
"json": {
|
|
@@ -774,9 +774,11 @@
|
|
|
774
774
|
},
|
|
775
775
|
"file": {
|
|
776
776
|
"char": "F",
|
|
777
|
-
"description": "
|
|
777
|
+
"description": "CSV file with phone numbers (and optional message text)",
|
|
778
778
|
"exclusive": [
|
|
779
|
-
"to"
|
|
779
|
+
"to",
|
|
780
|
+
"reuse",
|
|
781
|
+
"history"
|
|
780
782
|
],
|
|
781
783
|
"name": "file",
|
|
782
784
|
"hasDynamicHelp": false,
|
|
@@ -787,7 +789,9 @@
|
|
|
787
789
|
"char": "t",
|
|
788
790
|
"description": "Comma-separated recipient phone numbers (E.164 format)",
|
|
789
791
|
"exclusive": [
|
|
790
|
-
"file"
|
|
792
|
+
"file",
|
|
793
|
+
"reuse",
|
|
794
|
+
"history"
|
|
791
795
|
],
|
|
792
796
|
"name": "to",
|
|
793
797
|
"hasDynamicHelp": false,
|
|
@@ -796,7 +800,7 @@
|
|
|
796
800
|
},
|
|
797
801
|
"text": {
|
|
798
802
|
"char": "m",
|
|
799
|
-
"description": "Message text (
|
|
803
|
+
"description": "Message text (required with --to, optional with --file if CSV has text column)",
|
|
800
804
|
"name": "text",
|
|
801
805
|
"hasDynamicHelp": false,
|
|
802
806
|
"multiple": false,
|
|
@@ -828,6 +832,29 @@
|
|
|
828
832
|
"name": "dry-run",
|
|
829
833
|
"allowNo": false,
|
|
830
834
|
"type": "boolean"
|
|
835
|
+
},
|
|
836
|
+
"reuse": {
|
|
837
|
+
"description": "Re-use a previous batch upload by ID (see --history)",
|
|
838
|
+
"exclusive": [
|
|
839
|
+
"file",
|
|
840
|
+
"to",
|
|
841
|
+
"history"
|
|
842
|
+
],
|
|
843
|
+
"name": "reuse",
|
|
844
|
+
"hasDynamicHelp": false,
|
|
845
|
+
"multiple": false,
|
|
846
|
+
"type": "option"
|
|
847
|
+
},
|
|
848
|
+
"history": {
|
|
849
|
+
"description": "Show recent batch upload history",
|
|
850
|
+
"exclusive": [
|
|
851
|
+
"file",
|
|
852
|
+
"to",
|
|
853
|
+
"reuse"
|
|
854
|
+
],
|
|
855
|
+
"name": "history",
|
|
856
|
+
"allowNo": false,
|
|
857
|
+
"type": "boolean"
|
|
831
858
|
}
|
|
832
859
|
},
|
|
833
860
|
"hasDynamicHelp": false,
|
|
@@ -1762,5 +1789,5 @@
|
|
|
1762
1789
|
]
|
|
1763
1790
|
}
|
|
1764
1791
|
},
|
|
1765
|
-
"version": "3.
|
|
1792
|
+
"version": "3.5.0"
|
|
1766
1793
|
}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@sendly/cli",
|
|
3
|
-
"version": "3.
|
|
3
|
+
"version": "3.5.0",
|
|
4
4
|
"type": "module",
|
|
5
5
|
"description": "Sendly CLI - Send SMS from your terminal",
|
|
6
6
|
"author": "Sendly <support@sendly.live>",
|
|
@@ -34,8 +34,7 @@
|
|
|
34
34
|
"command_not_found": "./dist/hooks/command-not-found"
|
|
35
35
|
},
|
|
36
36
|
"plugins": [
|
|
37
|
-
"@oclif/plugin-help"
|
|
38
|
-
"@oclif/plugin-plugins"
|
|
37
|
+
"@oclif/plugin-help"
|
|
39
38
|
],
|
|
40
39
|
"topicSeparator": " ",
|
|
41
40
|
"topics": {
|
|
@@ -59,7 +58,6 @@
|
|
|
59
58
|
"dependencies": {
|
|
60
59
|
"@oclif/core": "^3.27.0",
|
|
61
60
|
"@oclif/plugin-help": "^6.2.0",
|
|
62
|
-
"@oclif/plugin-plugins": "^5.3.0",
|
|
63
61
|
"chalk": "^5.3.0",
|
|
64
62
|
"cli-table3": "^0.6.5",
|
|
65
63
|
"conf": "^12.0.0",
|