@sendly/cli 3.4.0 → 3.5.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/commands/sms/batch.d.ts +31 -2
- package/dist/commands/sms/batch.js +305 -214
- package/dist/commands/status.js +119 -42
- package/dist/lib/api-client.d.ts +13 -0
- package/dist/lib/api-client.js +76 -1
- package/dist/lib/base-command.js +8 -2
- package/oclif.manifest.json +41 -14
- package/package.json +2 -4
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
|
@@ -24,6 +24,10 @@ export declare class ApiError extends Error {
|
|
|
24
24
|
export declare class AuthenticationError extends ApiError {
|
|
25
25
|
constructor(message?: string);
|
|
26
26
|
}
|
|
27
|
+
export declare class ApiKeyRequiredError extends ApiError {
|
|
28
|
+
hint: string;
|
|
29
|
+
constructor(message?: string, hint?: string);
|
|
30
|
+
}
|
|
27
31
|
export declare class RateLimitError extends ApiError {
|
|
28
32
|
retryAfter: number;
|
|
29
33
|
constructor(retryAfter: number, message?: string);
|
|
@@ -47,6 +51,15 @@ declare class ApiClient {
|
|
|
47
51
|
post<T>(path: string, body?: Record<string, unknown>, requireAuth?: boolean): Promise<T>;
|
|
48
52
|
patch<T>(path: string, body?: Record<string, unknown>, requireAuth?: boolean): Promise<T>;
|
|
49
53
|
delete<T>(path: string, requireAuth?: boolean): Promise<T>;
|
|
54
|
+
/**
|
|
55
|
+
* Upload a file using multipart/form-data
|
|
56
|
+
* Used for batch CSV uploads to Supabase storage
|
|
57
|
+
*/
|
|
58
|
+
uploadFile<T>(path: string, file: {
|
|
59
|
+
buffer: Buffer;
|
|
60
|
+
filename: string;
|
|
61
|
+
mimetype?: string;
|
|
62
|
+
}, requireAuth?: boolean): Promise<T>;
|
|
50
63
|
}
|
|
51
64
|
export declare const apiClient: ApiClient;
|
|
52
65
|
export {};
|
package/dist/lib/api-client.js
CHANGED
|
@@ -45,6 +45,14 @@ export class AuthenticationError extends ApiError {
|
|
|
45
45
|
this.name = "AuthenticationError";
|
|
46
46
|
}
|
|
47
47
|
}
|
|
48
|
+
export class ApiKeyRequiredError extends ApiError {
|
|
49
|
+
hint;
|
|
50
|
+
constructor(message = "API key required for this operation.", hint = "Set SENDLY_API_KEY environment variable or create a key with: sendly keys create --type test") {
|
|
51
|
+
super("api_key_required", message, 401);
|
|
52
|
+
this.hint = hint;
|
|
53
|
+
this.name = "ApiKeyRequiredError";
|
|
54
|
+
}
|
|
55
|
+
}
|
|
48
56
|
export class RateLimitError extends ApiError {
|
|
49
57
|
retryAfter;
|
|
50
58
|
constructor(retryAfter, message = "Rate limit exceeded") {
|
|
@@ -149,6 +157,12 @@ class ApiClient {
|
|
|
149
157
|
switch (statusCode) {
|
|
150
158
|
case 401:
|
|
151
159
|
case 403:
|
|
160
|
+
// Detect if this is an API key required error vs general auth error
|
|
161
|
+
if (error === "invalid_api_key" ||
|
|
162
|
+
error === "api_key_required" ||
|
|
163
|
+
message?.toLowerCase().includes("api key")) {
|
|
164
|
+
throw new ApiKeyRequiredError("API key required for sending messages", "Set SENDLY_API_KEY environment variable or create a key with:\n sendly keys create --type test");
|
|
165
|
+
}
|
|
152
166
|
throw new AuthenticationError(message);
|
|
153
167
|
case 402:
|
|
154
168
|
throw new InsufficientCreditsError(message);
|
|
@@ -175,6 +189,67 @@ class ApiClient {
|
|
|
175
189
|
async delete(path, requireAuth = true) {
|
|
176
190
|
return this.request("DELETE", path, { requireAuth });
|
|
177
191
|
}
|
|
192
|
+
/**
|
|
193
|
+
* Upload a file using multipart/form-data
|
|
194
|
+
* Used for batch CSV uploads to Supabase storage
|
|
195
|
+
*/
|
|
196
|
+
async uploadFile(path, file, requireAuth = true) {
|
|
197
|
+
const maxRetries = getEffectiveValue("maxRetries");
|
|
198
|
+
const timeout = getEffectiveValue("timeout");
|
|
199
|
+
const url = `${this.getBaseUrl()}${path}`;
|
|
200
|
+
// Build multipart form data manually (Node.js compatible)
|
|
201
|
+
const boundary = `----FormBoundary${Date.now()}${Math.random().toString(36).substring(2)}`;
|
|
202
|
+
const mimetype = file.mimetype || "text/csv";
|
|
203
|
+
const header = Buffer.from(`--${boundary}\r\n` +
|
|
204
|
+
`Content-Disposition: form-data; name="file"; filename="${file.filename}"\r\n` +
|
|
205
|
+
`Content-Type: ${mimetype}\r\n\r\n`);
|
|
206
|
+
const footer = Buffer.from(`\r\n--${boundary}--\r\n`);
|
|
207
|
+
const body = Buffer.concat([header, file.buffer, footer]);
|
|
208
|
+
const headers = {
|
|
209
|
+
"Content-Type": `multipart/form-data; boundary=${boundary}`,
|
|
210
|
+
Accept: "application/json",
|
|
211
|
+
"User-Agent": `@sendly/cli/${version}`,
|
|
212
|
+
};
|
|
213
|
+
if (requireAuth) {
|
|
214
|
+
const token = getAuthToken();
|
|
215
|
+
if (!token) {
|
|
216
|
+
throw new AuthenticationError();
|
|
217
|
+
}
|
|
218
|
+
headers["Authorization"] = `Bearer ${token}`;
|
|
219
|
+
}
|
|
220
|
+
let lastError;
|
|
221
|
+
for (let attempt = 0; attempt <= maxRetries; attempt++) {
|
|
222
|
+
try {
|
|
223
|
+
const controller = new AbortController();
|
|
224
|
+
const timeoutId = setTimeout(() => controller.abort(), timeout);
|
|
225
|
+
const response = await fetch(url, {
|
|
226
|
+
method: "POST",
|
|
227
|
+
headers,
|
|
228
|
+
body,
|
|
229
|
+
signal: controller.signal,
|
|
230
|
+
});
|
|
231
|
+
clearTimeout(timeoutId);
|
|
232
|
+
this.updateRateLimitInfo(response.headers);
|
|
233
|
+
const data = await response.json().catch(() => ({}));
|
|
234
|
+
if (!response.ok) {
|
|
235
|
+
this.handleError(response.status, data);
|
|
236
|
+
}
|
|
237
|
+
return data;
|
|
238
|
+
}
|
|
239
|
+
catch (error) {
|
|
240
|
+
lastError = error;
|
|
241
|
+
if (!isRetryableError(error)) {
|
|
242
|
+
throw error;
|
|
243
|
+
}
|
|
244
|
+
if (attempt === maxRetries) {
|
|
245
|
+
throw error;
|
|
246
|
+
}
|
|
247
|
+
const backoffMs = Math.min(1000 * Math.pow(2, attempt), 10000);
|
|
248
|
+
await sleep(backoffMs);
|
|
249
|
+
}
|
|
250
|
+
}
|
|
251
|
+
throw lastError || new Error("Upload failed");
|
|
252
|
+
}
|
|
178
253
|
}
|
|
179
254
|
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":"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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\nexport const apiClient = new ApiClient();\n"]}
|
|
255
|
+
//# 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 ApiKeyRequiredError extends ApiError {\n  constructor(\n    message: string = \"API key required for this operation.\",\n    public hint: string = \"Set SENDLY_API_KEY environment variable or create a key with: sendly keys create --type test\",\n  ) {\n    super(\"api_key_required\", message, 401);\n    this.name = \"ApiKeyRequiredError\";\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        // Detect if this is an API key required error vs general auth error\n        if (\n          error === \"invalid_api_key\" ||\n          error === \"api_key_required\" ||\n          message?.toLowerCase().includes(\"api key\")\n        ) {\n          throw new ApiKeyRequiredError(\n            \"API key required for sending messages\",\n            \"Set SENDLY_API_KEY environment variable or create a key with:\\n  sendly keys create --type test\",\n          );\n        }\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/dist/lib/base-command.js
CHANGED
|
@@ -5,7 +5,7 @@
|
|
|
5
5
|
import { Command, Flags } from "@oclif/core";
|
|
6
6
|
import { setOutputFormat, setQuietMode, error } from "./output.js";
|
|
7
7
|
import { isAuthenticated } from "./config.js";
|
|
8
|
-
import { ApiError, AuthenticationError } from "./api-client.js";
|
|
8
|
+
import { ApiError, AuthenticationError, ApiKeyRequiredError, } from "./api-client.js";
|
|
9
9
|
export class BaseCommand extends Command {
|
|
10
10
|
static baseFlags = {
|
|
11
11
|
json: Flags.boolean({
|
|
@@ -29,6 +29,12 @@ export class BaseCommand extends Command {
|
|
|
29
29
|
}
|
|
30
30
|
}
|
|
31
31
|
async catch(err) {
|
|
32
|
+
if (err instanceof ApiKeyRequiredError) {
|
|
33
|
+
error(err.message, {
|
|
34
|
+
hint: err.hint,
|
|
35
|
+
});
|
|
36
|
+
this.exit(1);
|
|
37
|
+
}
|
|
32
38
|
if (err instanceof AuthenticationError) {
|
|
33
39
|
error("Not authenticated", {
|
|
34
40
|
hint: "Run 'sendly login' to authenticate",
|
|
@@ -57,4 +63,4 @@ export class AuthenticatedCommand extends BaseCommand {
|
|
|
57
63
|
this.requireAuth();
|
|
58
64
|
}
|
|
59
65
|
}
|
|
60
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
66
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFzZS1jb21tYW5kLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2xpYi9iYXNlLWNvbW1hbmQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7OztHQUdHO0FBRUgsT0FBTyxFQUFFLE9BQU8sRUFBRSxLQUFLLEVBQUUsTUFBTSxhQUFhLENBQUM7QUFDN0MsT0FBTyxFQUFFLGVBQWUsRUFBRSxZQUFZLEVBQUUsS0FBSyxFQUFFLE1BQU0sYUFBYSxDQUFDO0FBQ25FLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSxhQUFhLENBQUM7QUFDOUMsT0FBTyxFQUNMLFFBQVEsRUFDUixtQkFBbUIsRUFDbkIsbUJBQW1CLEdBQ3BCLE1BQU0saUJBQWlCLENBQUM7QUFFekIsTUFBTSxPQUFnQixXQUFZLFNBQVEsT0FBTztJQUMvQyxNQUFNLENBQUMsU0FBUyxHQUFHO1FBQ2pCLElBQUksRUFBRSxLQUFLLENBQUMsT0FBTyxDQUFDO1lBQ2xCLFdBQVcsRUFBRSx1QkFBdUI7WUFDcEMsT0FBTyxFQUFFLEtBQUs7U0FDZixDQUFDO1FBQ0YsS0FBSyxFQUFFLEtBQUssQ0FBQyxPQUFPLENBQUM7WUFDbkIsSUFBSSxFQUFFLEdBQUc7WUFDVCxXQUFXLEVBQUUsZ0JBQWdCO1lBQzdCLE9BQU8sRUFBRSxLQUFLO1NBQ2YsQ0FBQztLQUNILENBQUM7SUFFUSxLQUFLLENBQUMsSUFBSTtRQUNsQixNQUFNLEtBQUssQ0FBQyxJQUFJLEVBQUUsQ0FBQztRQUNuQixNQUFNLEVBQUUsS0FBSyxFQUFFLEdBQUcsTUFBTSxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxXQUFpQyxDQUFDLENBQUM7UUFFM0UsSUFBSSxLQUFLLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDZixlQUFlLENBQUMsTUFBTSxDQUFDLENBQUM7UUFDMUIsQ0FBQztRQUNELElBQUksS0FBSyxDQUFDLEtBQUssRUFBRSxDQUFDO1lBQ2hCLFlBQVksQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUNyQixDQUFDO0lBQ0gsQ0FBQztJQUVTLEtBQUssQ0FBQyxLQUFLLENBQUMsR0FBVTtRQUM5QixJQUFJLEdBQUcsWUFBWSxtQkFBbUIsRUFBRSxDQUFDO1lBQ3ZDLEtBQUssQ0FBQyxHQUFHLENBQUMsT0FBTyxFQUFFO2dCQUNqQixJQUFJLEVBQUUsR0FBRyxDQUFDLElBQUk7YUFDZixDQUFDLENBQUM7WUFDSCxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ2YsQ0FBQztRQUVELElBQUksR0FBRyxZQUFZLG1CQUFtQixFQUFFLENBQUM7WUFDdkMsS0FBSyxDQUFDLG1CQUFtQixFQUFFO2dCQUN6QixJQUFJLEVBQUUsb0NBQW9DO2FBQzNDLENBQUMsQ0FBQztZQUNILElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDZixDQUFDO1FBRUQsSUFBSSxHQUFHLFlBQVksUUFBUSxFQUFFLENBQUM7WUFDNUIsS0FBSyxDQUFDLEdBQUcsQ0FBQyxPQUFPLEVBQUU7Z0JBQ2pCLElBQUksRUFBRSxHQUFHLENBQUMsSUFBSTtnQkFDZCxHQUFHLENBQUMsR0FBRyxDQUFDLE9BQU8sSUFBSSxFQUFFLENBQUM7YUFDdkIsQ0FBQyxDQUFDO1lBQ0gsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNmLENBQUM7UUFFRCxLQUFLLENBQUMsR0FBRyxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBQ25CLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDZixDQUFDO0lBRVMsV0FBVztRQUNuQixJQUFJLENBQUMsZUFBZSxFQUFFLEVBQUUsQ0FBQztZQUN2QixNQUFNLElBQUksbUJBQW1CLEVBQUUsQ0FBQztRQUNsQyxDQUFDO0lBQ0gsQ0FBQzs7QUFHSCxNQUFNLE9BQWdCLG9CQUFxQixTQUFRLFdBQVc7SUFDbEQsS0FBSyxDQUFDLElBQUk7UUFDbEIsTUFBTSxLQUFLLENBQUMsSUFBSSxFQUFFLENBQUM7UUFDbkIsSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDO0lBQ3JCLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQmFzZSBDb21tYW5kIGNsYXNzIGZvciBhbGwgU2VuZGx5IENMSSBjb21tYW5kc1xuICogUHJvdmlkZXMgY29tbW9uIGZ1bmN0aW9uYWxpdHkgYW5kIGZsYWdzXG4gKi9cblxuaW1wb3J0IHsgQ29tbWFuZCwgRmxhZ3MgfSBmcm9tIFwiQG9jbGlmL2NvcmVcIjtcbmltcG9ydCB7IHNldE91dHB1dEZvcm1hdCwgc2V0UXVpZXRNb2RlLCBlcnJvciB9IGZyb20gXCIuL291dHB1dC5qc1wiO1xuaW1wb3J0IHsgaXNBdXRoZW50aWNhdGVkIH0gZnJvbSBcIi4vY29uZmlnLmpzXCI7XG5pbXBvcnQge1xuICBBcGlFcnJvcixcbiAgQXV0aGVudGljYXRpb25FcnJvcixcbiAgQXBpS2V5UmVxdWlyZWRFcnJvcixcbn0gZnJvbSBcIi4vYXBpLWNsaWVudC5qc1wiO1xuXG5leHBvcnQgYWJzdHJhY3QgY2xhc3MgQmFzZUNvbW1hbmQgZXh0ZW5kcyBDb21tYW5kIHtcbiAgc3RhdGljIGJhc2VGbGFncyA9IHtcbiAgICBqc29uOiBGbGFncy5ib29sZWFuKHtcbiAgICAgIGRlc2NyaXB0aW9uOiBcIk91dHB1dCBpbiBKU09OIGZvcm1hdFwiLFxuICAgICAgZGVmYXVsdDogZmFsc2UsXG4gICAgfSksXG4gICAgcXVpZXQ6IEZsYWdzLmJvb2xlYW4oe1xuICAgICAgY2hhcjogXCJxXCIsXG4gICAgICBkZXNjcmlwdGlvbjogXCJNaW5pbWFsIG91dHB1dFwiLFxuICAgICAgZGVmYXVsdDogZmFsc2UsXG4gICAgfSksXG4gIH07XG5cbiAgcHJvdGVjdGVkIGFzeW5jIGluaXQoKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgYXdhaXQgc3VwZXIuaW5pdCgpO1xuICAgIGNvbnN0IHsgZmxhZ3MgfSA9IGF3YWl0IHRoaXMucGFyc2UodGhpcy5jb25zdHJ1Y3RvciBhcyB0eXBlb2YgQmFzZUNvbW1hbmQpO1xuXG4gICAgaWYgKGZsYWdzLmpzb24pIHtcbiAgICAgIHNldE91dHB1dEZvcm1hdChcImpzb25cIik7XG4gICAgfVxuICAgIGlmIChmbGFncy5xdWlldCkge1xuICAgICAgc2V0UXVpZXRNb2RlKHRydWUpO1xuICAgIH1cbiAgfVxuXG4gIHByb3RlY3RlZCBhc3luYyBjYXRjaChlcnI6IEVycm9yKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgaWYgKGVyciBpbnN0YW5jZW9mIEFwaUtleVJlcXVpcmVkRXJyb3IpIHtcbiAgICAgIGVycm9yKGVyci5tZXNzYWdlLCB7XG4gICAgICAgIGhpbnQ6IGVyci5oaW50LFxuICAgICAgfSk7XG4gICAgICB0aGlzLmV4aXQoMSk7XG4gICAgfVxuXG4gICAgaWYgKGVyciBpbnN0YW5jZW9mIEF1dGhlbnRpY2F0aW9uRXJyb3IpIHtcbiAgICAgIGVycm9yKFwiTm90IGF1dGhlbnRpY2F0ZWRcIiwge1xuICAgICAgICBoaW50OiBcIlJ1biAnc2VuZGx5IGxvZ2luJyB0byBhdXRoZW50aWNhdGVcIixcbiAgICAgIH0pO1xuICAgICAgdGhpcy5leGl0KDEpO1xuICAgIH1cblxuICAgIGlmIChlcnIgaW5zdGFuY2VvZiBBcGlFcnJvcikge1xuICAgICAgZXJyb3IoZXJyLm1lc3NhZ2UsIHtcbiAgICAgICAgY29kZTogZXJyLmNvZGUsXG4gICAgICAgIC4uLihlcnIuZGV0YWlscyB8fCB7fSksXG4gICAgICB9KTtcbiAgICAgIHRoaXMuZXhpdCgxKTtcbiAgICB9XG5cbiAgICBlcnJvcihlcnIubWVzc2FnZSk7XG4gICAgdGhpcy5leGl0KDEpO1xuICB9XG5cbiAgcHJvdGVjdGVkIHJlcXVpcmVBdXRoKCk6IHZvaWQge1xuICAgIGlmICghaXNBdXRoZW50aWNhdGVkKCkpIHtcbiAgICAgIHRocm93IG5ldyBBdXRoZW50aWNhdGlvbkVycm9yKCk7XG4gICAgfVxuICB9XG59XG5cbmV4cG9ydCBhYnN0cmFjdCBjbGFzcyBBdXRoZW50aWNhdGVkQ29tbWFuZCBleHRlbmRzIEJhc2VDb21tYW5kIHtcbiAgcHJvdGVjdGVkIGFzeW5jIGluaXQoKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgYXdhaXQgc3VwZXIuaW5pdCgpO1xuICAgIHRoaXMucmVxdWlyZUF1dGgoKTtcbiAgfVxufVxuIl19
|