@sendly/cli 1.0.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.
Files changed (60) hide show
  1. package/README.md +323 -0
  2. package/bin/run.js +5 -0
  3. package/dist/commands/config/get.d.ts +13 -0
  4. package/dist/commands/config/get.js +38 -0
  5. package/dist/commands/config/list.d.ts +10 -0
  6. package/dist/commands/config/list.js +45 -0
  7. package/dist/commands/config/set.d.ts +14 -0
  8. package/dist/commands/config/set.js +67 -0
  9. package/dist/commands/credits/balance.d.ts +10 -0
  10. package/dist/commands/credits/balance.js +38 -0
  11. package/dist/commands/credits/history.d.ts +11 -0
  12. package/dist/commands/credits/history.js +88 -0
  13. package/dist/commands/doctor.d.ts +23 -0
  14. package/dist/commands/doctor.js +336 -0
  15. package/dist/commands/keys/create.d.ts +12 -0
  16. package/dist/commands/keys/create.js +47 -0
  17. package/dist/commands/keys/list.d.ts +10 -0
  18. package/dist/commands/keys/list.js +65 -0
  19. package/dist/commands/keys/revoke.d.ts +15 -0
  20. package/dist/commands/keys/revoke.js +68 -0
  21. package/dist/commands/login.d.ts +12 -0
  22. package/dist/commands/login.js +114 -0
  23. package/dist/commands/logout.d.ts +10 -0
  24. package/dist/commands/logout.js +20 -0
  25. package/dist/commands/logs/tail.d.ts +17 -0
  26. package/dist/commands/logs/tail.js +183 -0
  27. package/dist/commands/sms/batch.d.ts +16 -0
  28. package/dist/commands/sms/batch.js +163 -0
  29. package/dist/commands/sms/cancel.d.ts +13 -0
  30. package/dist/commands/sms/cancel.js +46 -0
  31. package/dist/commands/sms/get.d.ts +13 -0
  32. package/dist/commands/sms/get.js +51 -0
  33. package/dist/commands/sms/list.d.ts +12 -0
  34. package/dist/commands/sms/list.js +79 -0
  35. package/dist/commands/sms/schedule.d.ts +14 -0
  36. package/dist/commands/sms/schedule.js +91 -0
  37. package/dist/commands/sms/scheduled.d.ts +12 -0
  38. package/dist/commands/sms/scheduled.js +82 -0
  39. package/dist/commands/sms/send.d.ts +13 -0
  40. package/dist/commands/sms/send.js +70 -0
  41. package/dist/commands/webhooks/list.d.ts +10 -0
  42. package/dist/commands/webhooks/list.js +80 -0
  43. package/dist/commands/webhooks/listen.d.ts +20 -0
  44. package/dist/commands/webhooks/listen.js +202 -0
  45. package/dist/commands/whoami.d.ts +10 -0
  46. package/dist/commands/whoami.js +51 -0
  47. package/dist/index.d.ts +26 -0
  48. package/dist/index.js +27 -0
  49. package/dist/lib/api-client.d.ts +52 -0
  50. package/dist/lib/api-client.js +129 -0
  51. package/dist/lib/auth.d.ts +52 -0
  52. package/dist/lib/auth.js +171 -0
  53. package/dist/lib/base-command.d.ts +17 -0
  54. package/dist/lib/base-command.js +60 -0
  55. package/dist/lib/config.d.ts +54 -0
  56. package/dist/lib/config.js +182 -0
  57. package/dist/lib/output.d.ts +43 -0
  58. package/dist/lib/output.js +222 -0
  59. package/oclif.manifest.json +1147 -0
  60. package/package.json +98 -0
@@ -0,0 +1,182 @@
1
+ /**
2
+ * CLI Configuration Management
3
+ * Stores user preferences and credentials in ~/.sendly/
4
+ *
5
+ * Environment Variables (take precedence over config file):
6
+ * - SENDLY_API_KEY: API key for authentication
7
+ * - SENDLY_BASE_URL: Custom API endpoint
8
+ * - SENDLY_OUTPUT_FORMAT: Default output format (human/json)
9
+ * - SENDLY_NO_COLOR: Disable colored output (any value)
10
+ * - SENDLY_TIMEOUT: Request timeout in ms (default: 30000)
11
+ * - SENDLY_MAX_RETRIES: Max retry attempts (default: 3)
12
+ * - CI: Auto-detect CI mode (disables interactive prompts)
13
+ */
14
+ import Conf from "conf";
15
+ import * as fs from "node:fs";
16
+ import * as path from "node:path";
17
+ import * as os from "node:os";
18
+ /**
19
+ * Check if running in CI environment
20
+ */
21
+ export function isCI() {
22
+ return !!(process.env.CI ||
23
+ process.env.CONTINUOUS_INTEGRATION ||
24
+ process.env.GITHUB_ACTIONS ||
25
+ process.env.GITLAB_CI ||
26
+ process.env.CIRCLECI ||
27
+ process.env.TRAVIS ||
28
+ process.env.BUILDKITE);
29
+ }
30
+ /**
31
+ * Check if color output is disabled
32
+ */
33
+ export function isColorDisabled() {
34
+ return !!(process.env.SENDLY_NO_COLOR ||
35
+ process.env.NO_COLOR ||
36
+ process.env.TERM === "dumb");
37
+ }
38
+ const CONFIG_DIR = path.join(os.homedir(), ".sendly");
39
+ const CONFIG_FILE = "config.json";
40
+ // Ensure config directory exists
41
+ if (!fs.existsSync(CONFIG_DIR)) {
42
+ fs.mkdirSync(CONFIG_DIR, { recursive: true, mode: 0o700 });
43
+ }
44
+ const config = new Conf({
45
+ projectName: "sendly",
46
+ cwd: CONFIG_DIR,
47
+ configName: "config",
48
+ defaults: {
49
+ environment: "test",
50
+ baseUrl: "https://sendly.live",
51
+ defaultFormat: "human",
52
+ colorEnabled: true,
53
+ timeout: 30000,
54
+ maxRetries: 3,
55
+ },
56
+ // Encrypt sensitive data
57
+ encryptionKey: process.env.SENDLY_CONFIG_KEY || "sendly-cli-default-key-v1",
58
+ });
59
+ /**
60
+ * Get effective config value with environment variable override
61
+ * Priority: env var > config file > default
62
+ */
63
+ export function getEffectiveValue(key) {
64
+ // Environment variable overrides
65
+ switch (key) {
66
+ case "apiKey":
67
+ if (process.env.SENDLY_API_KEY) {
68
+ return process.env.SENDLY_API_KEY;
69
+ }
70
+ break;
71
+ case "baseUrl":
72
+ if (process.env.SENDLY_BASE_URL) {
73
+ return process.env.SENDLY_BASE_URL;
74
+ }
75
+ break;
76
+ case "defaultFormat":
77
+ if (process.env.SENDLY_OUTPUT_FORMAT) {
78
+ const format = process.env.SENDLY_OUTPUT_FORMAT.toLowerCase();
79
+ if (format === "json" || format === "human") {
80
+ return format;
81
+ }
82
+ }
83
+ break;
84
+ case "colorEnabled":
85
+ if (isColorDisabled()) {
86
+ return false;
87
+ }
88
+ break;
89
+ case "timeout":
90
+ if (process.env.SENDLY_TIMEOUT) {
91
+ const timeout = parseInt(process.env.SENDLY_TIMEOUT, 10);
92
+ if (!isNaN(timeout) && timeout > 0) {
93
+ return timeout;
94
+ }
95
+ }
96
+ break;
97
+ case "maxRetries":
98
+ if (process.env.SENDLY_MAX_RETRIES) {
99
+ const retries = parseInt(process.env.SENDLY_MAX_RETRIES, 10);
100
+ if (!isNaN(retries) && retries >= 0) {
101
+ return retries;
102
+ }
103
+ }
104
+ break;
105
+ }
106
+ // Fall back to config file value
107
+ return config.get(key);
108
+ }
109
+ export function getConfig() {
110
+ return config.store;
111
+ }
112
+ export function setConfig(key, value) {
113
+ config.set(key, value);
114
+ }
115
+ export function getConfigValue(key) {
116
+ return config.get(key);
117
+ }
118
+ export function clearConfig() {
119
+ config.clear();
120
+ }
121
+ export function clearAuth() {
122
+ config.delete("apiKey");
123
+ config.delete("accessToken");
124
+ config.delete("refreshToken");
125
+ config.delete("tokenExpiresAt");
126
+ config.delete("userId");
127
+ config.delete("email");
128
+ }
129
+ export function isAuthenticated() {
130
+ // Check env var first
131
+ if (process.env.SENDLY_API_KEY)
132
+ return true;
133
+ const apiKey = config.get("apiKey");
134
+ const accessToken = config.get("accessToken");
135
+ return !!(apiKey || accessToken);
136
+ }
137
+ export function getAuthToken() {
138
+ // Environment variable takes highest precedence
139
+ if (process.env.SENDLY_API_KEY) {
140
+ return process.env.SENDLY_API_KEY;
141
+ }
142
+ // Then stored API key
143
+ const apiKey = config.get("apiKey");
144
+ if (apiKey)
145
+ return apiKey;
146
+ // Finally, access token (if not expired)
147
+ const accessToken = config.get("accessToken");
148
+ const expiresAt = config.get("tokenExpiresAt");
149
+ if (accessToken && expiresAt && Date.now() < expiresAt) {
150
+ return accessToken;
151
+ }
152
+ return undefined;
153
+ }
154
+ export function setApiKey(apiKey) {
155
+ // Validate API key format
156
+ if (!/^sk_(test|live)_v1_[a-zA-Z0-9_-]+$/.test(apiKey)) {
157
+ throw new Error("Invalid API key format. Expected sk_test_v1_xxx or sk_live_v1_xxx");
158
+ }
159
+ config.set("apiKey", apiKey);
160
+ // Set environment based on key type
161
+ if (apiKey.startsWith("sk_test_")) {
162
+ config.set("environment", "test");
163
+ }
164
+ else {
165
+ config.set("environment", "live");
166
+ }
167
+ }
168
+ export function setAuthTokens(accessToken, refreshToken, expiresIn, userId, email) {
169
+ config.set("accessToken", accessToken);
170
+ config.set("refreshToken", refreshToken);
171
+ config.set("tokenExpiresAt", Date.now() + expiresIn * 1000);
172
+ config.set("userId", userId);
173
+ config.set("email", email);
174
+ }
175
+ export function getConfigPath() {
176
+ return path.join(CONFIG_DIR, CONFIG_FILE);
177
+ }
178
+ export function getConfigDir() {
179
+ return CONFIG_DIR;
180
+ }
181
+ export { config };
182
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"config.js","sourceRoot":"","sources":["../../src/lib/config.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * CLI Configuration Management\n * Stores user preferences and credentials in ~/.sendly/\n *\n * Environment Variables (take precedence over config file):\n * - SENDLY_API_KEY: API key for authentication\n * - SENDLY_BASE_URL: Custom API endpoint\n * - SENDLY_OUTPUT_FORMAT: Default output format (human/json)\n * - SENDLY_NO_COLOR: Disable colored output (any value)\n * - SENDLY_TIMEOUT: Request timeout in ms (default: 30000)\n * - SENDLY_MAX_RETRIES: Max retry attempts (default: 3)\n * - CI: Auto-detect CI mode (disables interactive prompts)\n */\n\nimport Conf from \"conf\";\nimport * as fs from \"node:fs\";\nimport * as path from \"node:path\";\nimport * as os from \"node:os\";\n\nexport interface SendlyConfig {\n  // Authentication\n  apiKey?: string;\n  accessToken?: string;\n  refreshToken?: string;\n  tokenExpiresAt?: number;\n  userId?: string;\n  email?: string;\n\n  // Environment\n  environment: \"test\" | \"live\";\n  baseUrl: string;\n\n  // Preferences\n  defaultFormat: \"human\" | \"json\";\n  colorEnabled: boolean;\n\n  // Network\n  timeout: number;\n  maxRetries: number;\n}\n\n/**\n * Check if running in CI environment\n */\nexport function isCI(): boolean {\n  return !!(\n    process.env.CI ||\n    process.env.CONTINUOUS_INTEGRATION ||\n    process.env.GITHUB_ACTIONS ||\n    process.env.GITLAB_CI ||\n    process.env.CIRCLECI ||\n    process.env.TRAVIS ||\n    process.env.BUILDKITE\n  );\n}\n\n/**\n * Check if color output is disabled\n */\nexport function isColorDisabled(): boolean {\n  return !!(\n    process.env.SENDLY_NO_COLOR ||\n    process.env.NO_COLOR ||\n    process.env.TERM === \"dumb\"\n  );\n}\n\nconst CONFIG_DIR = path.join(os.homedir(), \".sendly\");\nconst CONFIG_FILE = \"config.json\";\n\n// Ensure config directory exists\nif (!fs.existsSync(CONFIG_DIR)) {\n  fs.mkdirSync(CONFIG_DIR, { recursive: true, mode: 0o700 });\n}\n\nconst config = new Conf<SendlyConfig>({\n  projectName: \"sendly\",\n  cwd: CONFIG_DIR,\n  configName: \"config\",\n  defaults: {\n    environment: \"test\",\n    baseUrl: \"https://sendly.live\",\n    defaultFormat: \"human\",\n    colorEnabled: true,\n    timeout: 30000,\n    maxRetries: 3,\n  },\n  // Encrypt sensitive data\n  encryptionKey: process.env.SENDLY_CONFIG_KEY || \"sendly-cli-default-key-v1\",\n});\n\n/**\n * Get effective config value with environment variable override\n * Priority: env var > config file > default\n */\nexport function getEffectiveValue<K extends keyof SendlyConfig>(\n  key: K,\n): SendlyConfig[K] {\n  // Environment variable overrides\n  switch (key) {\n    case \"apiKey\":\n      if (process.env.SENDLY_API_KEY) {\n        return process.env.SENDLY_API_KEY as SendlyConfig[K];\n      }\n      break;\n    case \"baseUrl\":\n      if (process.env.SENDLY_BASE_URL) {\n        return process.env.SENDLY_BASE_URL as SendlyConfig[K];\n      }\n      break;\n    case \"defaultFormat\":\n      if (process.env.SENDLY_OUTPUT_FORMAT) {\n        const format = process.env.SENDLY_OUTPUT_FORMAT.toLowerCase();\n        if (format === \"json\" || format === \"human\") {\n          return format as SendlyConfig[K];\n        }\n      }\n      break;\n    case \"colorEnabled\":\n      if (isColorDisabled()) {\n        return false as SendlyConfig[K];\n      }\n      break;\n    case \"timeout\":\n      if (process.env.SENDLY_TIMEOUT) {\n        const timeout = parseInt(process.env.SENDLY_TIMEOUT, 10);\n        if (!isNaN(timeout) && timeout > 0) {\n          return timeout as SendlyConfig[K];\n        }\n      }\n      break;\n    case \"maxRetries\":\n      if (process.env.SENDLY_MAX_RETRIES) {\n        const retries = parseInt(process.env.SENDLY_MAX_RETRIES, 10);\n        if (!isNaN(retries) && retries >= 0) {\n          return retries as SendlyConfig[K];\n        }\n      }\n      break;\n  }\n\n  // Fall back to config file value\n  return config.get(key);\n}\n\nexport function getConfig(): SendlyConfig {\n  return config.store;\n}\n\nexport function setConfig<K extends keyof SendlyConfig>(\n  key: K,\n  value: SendlyConfig[K],\n): void {\n  config.set(key, value);\n}\n\nexport function getConfigValue<K extends keyof SendlyConfig>(\n  key: K,\n): SendlyConfig[K] {\n  return config.get(key);\n}\n\nexport function clearConfig(): void {\n  config.clear();\n}\n\nexport function clearAuth(): void {\n  config.delete(\"apiKey\");\n  config.delete(\"accessToken\");\n  config.delete(\"refreshToken\");\n  config.delete(\"tokenExpiresAt\");\n  config.delete(\"userId\");\n  config.delete(\"email\");\n}\n\nexport function isAuthenticated(): boolean {\n  // Check env var first\n  if (process.env.SENDLY_API_KEY) return true;\n\n  const apiKey = config.get(\"apiKey\");\n  const accessToken = config.get(\"accessToken\");\n  return !!(apiKey || accessToken);\n}\n\nexport function getAuthToken(): string | undefined {\n  // Environment variable takes highest precedence\n  if (process.env.SENDLY_API_KEY) {\n    return process.env.SENDLY_API_KEY;\n  }\n\n  // Then stored API key\n  const apiKey = config.get(\"apiKey\");\n  if (apiKey) return apiKey;\n\n  // Finally, access token (if not expired)\n  const accessToken = config.get(\"accessToken\");\n  const expiresAt = config.get(\"tokenExpiresAt\");\n\n  if (accessToken && expiresAt && Date.now() < expiresAt) {\n    return accessToken;\n  }\n\n  return undefined;\n}\n\nexport function setApiKey(apiKey: string): void {\n  // Validate API key format\n  if (!/^sk_(test|live)_v1_[a-zA-Z0-9_-]+$/.test(apiKey)) {\n    throw new Error(\n      \"Invalid API key format. Expected sk_test_v1_xxx or sk_live_v1_xxx\",\n    );\n  }\n\n  config.set(\"apiKey\", apiKey);\n\n  // Set environment based on key type\n  if (apiKey.startsWith(\"sk_test_\")) {\n    config.set(\"environment\", \"test\");\n  } else {\n    config.set(\"environment\", \"live\");\n  }\n}\n\nexport function setAuthTokens(\n  accessToken: string,\n  refreshToken: string,\n  expiresIn: number,\n  userId: string,\n  email: string,\n): void {\n  config.set(\"accessToken\", accessToken);\n  config.set(\"refreshToken\", refreshToken);\n  config.set(\"tokenExpiresAt\", Date.now() + expiresIn * 1000);\n  config.set(\"userId\", userId);\n  config.set(\"email\", email);\n}\n\nexport function getConfigPath(): string {\n  return path.join(CONFIG_DIR, CONFIG_FILE);\n}\n\nexport function getConfigDir(): string {\n  return CONFIG_DIR;\n}\n\nexport { config };\n"]}
@@ -0,0 +1,43 @@
1
+ /**
2
+ * Output Formatting Utilities
3
+ * Handles human-readable and JSON output modes
4
+ */
5
+ import { type Ora } from "ora";
6
+ export type OutputFormat = "human" | "json";
7
+ export declare function setOutputFormat(format: OutputFormat): void;
8
+ export declare function setQuietMode(quiet: boolean): void;
9
+ export declare function getOutputFormat(): OutputFormat;
10
+ export declare function isJsonMode(): boolean;
11
+ export declare function isQuietMode(): boolean;
12
+ export declare const colors: {
13
+ primary: import("chalk").ChalkInstance;
14
+ success: import("chalk").ChalkInstance;
15
+ error: import("chalk").ChalkInstance;
16
+ warning: import("chalk").ChalkInstance;
17
+ info: import("chalk").ChalkInstance;
18
+ dim: import("chalk").ChalkInstance;
19
+ bold: import("chalk").ChalkInstance;
20
+ code: import("chalk").ChalkInstance;
21
+ };
22
+ export declare function success(message: string, data?: Record<string, unknown>): void;
23
+ export declare function error(message: string, details?: Record<string, unknown>): void;
24
+ export declare function warn(message: string): void;
25
+ export declare function info(message: string): void;
26
+ export declare function json(data: unknown): void;
27
+ export interface TableColumn {
28
+ header: string;
29
+ key: string;
30
+ width?: number;
31
+ formatter?: (value: unknown) => string;
32
+ }
33
+ export declare function table(data: Array<Record<string, any>>, columns: TableColumn[]): void;
34
+ export declare function spinner(text: string): Ora;
35
+ export declare function formatStatus(status: string): string;
36
+ export declare function formatDate(date: string | Date | number): string;
37
+ export declare function formatRelativeTime(date: string | Date | number): string;
38
+ export declare function formatCredits(credits: number): string;
39
+ export declare function formatPhone(phone: string): string;
40
+ export declare function header(title: string): void;
41
+ export declare function divider(): void;
42
+ export declare function keyValue(data: Record<string, unknown>): void;
43
+ export declare function codeBlock(code: string, language?: string): void;
@@ -0,0 +1,222 @@
1
+ /**
2
+ * Output Formatting Utilities
3
+ * Handles human-readable and JSON output modes
4
+ */
5
+ import chalk from "chalk";
6
+ import Table from "cli-table3";
7
+ import ora from "ora";
8
+ let currentFormat = "human";
9
+ let quietMode = false;
10
+ export function setOutputFormat(format) {
11
+ currentFormat = format;
12
+ }
13
+ export function setQuietMode(quiet) {
14
+ quietMode = quiet;
15
+ }
16
+ export function getOutputFormat() {
17
+ return currentFormat;
18
+ }
19
+ export function isJsonMode() {
20
+ return currentFormat === "json";
21
+ }
22
+ export function isQuietMode() {
23
+ return quietMode;
24
+ }
25
+ // Colors
26
+ export const colors = {
27
+ primary: chalk.hex("#F59E0B"), // Amber/Orange - Sendly brand
28
+ success: chalk.green,
29
+ error: chalk.red,
30
+ warning: chalk.yellow,
31
+ info: chalk.blue,
32
+ dim: chalk.dim,
33
+ bold: chalk.bold,
34
+ code: chalk.cyan,
35
+ };
36
+ // Success output
37
+ export function success(message, data) {
38
+ if (isJsonMode()) {
39
+ console.log(JSON.stringify({ success: true, message, ...data }, null, 2));
40
+ return;
41
+ }
42
+ if (quietMode) {
43
+ if (data?.id)
44
+ console.log(data.id);
45
+ return;
46
+ }
47
+ console.log(`${colors.success("✓")} ${message}`);
48
+ if (data) {
49
+ Object.entries(data).forEach(([key, value]) => {
50
+ console.log(` ${colors.dim(key + ":")} ${value}`);
51
+ });
52
+ }
53
+ }
54
+ // Error output
55
+ export function error(message, details) {
56
+ if (isJsonMode()) {
57
+ console.error(JSON.stringify({ error: true, message, ...details }, null, 2));
58
+ return;
59
+ }
60
+ console.error(`${colors.error("✗")} ${message}`);
61
+ if (details && !quietMode) {
62
+ Object.entries(details).forEach(([key, value]) => {
63
+ console.error(` ${colors.dim(key + ":")} ${value}`);
64
+ });
65
+ }
66
+ }
67
+ // Warning output
68
+ export function warn(message) {
69
+ if (isJsonMode())
70
+ return;
71
+ if (quietMode)
72
+ return;
73
+ console.log(`${colors.warning("⚠")} ${message}`);
74
+ }
75
+ // Info output
76
+ export function info(message) {
77
+ if (isJsonMode())
78
+ return;
79
+ if (quietMode)
80
+ return;
81
+ console.log(`${colors.info("ℹ")} ${message}`);
82
+ }
83
+ // Print raw JSON
84
+ export function json(data) {
85
+ console.log(JSON.stringify(data, null, 2));
86
+ }
87
+ export function table(data, columns) {
88
+ if (isJsonMode()) {
89
+ console.log(JSON.stringify(data, null, 2));
90
+ return;
91
+ }
92
+ if (data.length === 0) {
93
+ info("No data to display");
94
+ return;
95
+ }
96
+ const t = new Table({
97
+ head: columns.map((c) => colors.bold(c.header)),
98
+ style: {
99
+ head: [],
100
+ border: [],
101
+ },
102
+ colWidths: columns.map((c) => c.width ?? null),
103
+ });
104
+ data.forEach((row) => {
105
+ t.push(columns.map((col) => {
106
+ const value = row[col.key];
107
+ if (col.formatter) {
108
+ return col.formatter(value);
109
+ }
110
+ return String(value ?? "-");
111
+ }));
112
+ });
113
+ console.log(t.toString());
114
+ }
115
+ // Spinner for long operations
116
+ export function spinner(text) {
117
+ if (isJsonMode() || quietMode) {
118
+ return {
119
+ start: () => ({ stop: () => { }, succeed: () => { }, fail: () => { } }),
120
+ stop: () => { },
121
+ succeed: () => { },
122
+ fail: () => { },
123
+ };
124
+ }
125
+ return ora({
126
+ text,
127
+ color: "yellow",
128
+ spinner: "dots",
129
+ });
130
+ }
131
+ // Status formatters
132
+ export function formatStatus(status) {
133
+ switch (status.toLowerCase()) {
134
+ case "delivered":
135
+ case "success":
136
+ case "active":
137
+ case "verified":
138
+ return colors.success(status);
139
+ case "failed":
140
+ case "error":
141
+ case "revoked":
142
+ case "rejected":
143
+ return colors.error(status);
144
+ case "queued":
145
+ case "pending":
146
+ case "processing":
147
+ return colors.warning(status);
148
+ case "sent":
149
+ return colors.info(status);
150
+ default:
151
+ return status;
152
+ }
153
+ }
154
+ // Format date
155
+ export function formatDate(date) {
156
+ const d = new Date(date);
157
+ return d.toLocaleString();
158
+ }
159
+ // Format relative time
160
+ export function formatRelativeTime(date) {
161
+ const d = new Date(date);
162
+ const now = new Date();
163
+ const diff = now.getTime() - d.getTime();
164
+ const seconds = Math.floor(diff / 1000);
165
+ const minutes = Math.floor(seconds / 60);
166
+ const hours = Math.floor(minutes / 60);
167
+ const days = Math.floor(hours / 24);
168
+ if (days > 0)
169
+ return `${days}d ago`;
170
+ if (hours > 0)
171
+ return `${hours}h ago`;
172
+ if (minutes > 0)
173
+ return `${minutes}m ago`;
174
+ return `${seconds}s ago`;
175
+ }
176
+ // Format credits
177
+ export function formatCredits(credits) {
178
+ return `${credits.toLocaleString()} credits`;
179
+ }
180
+ // Format phone number
181
+ export function formatPhone(phone) {
182
+ return colors.code(phone);
183
+ }
184
+ // Header for commands
185
+ export function header(title) {
186
+ if (isJsonMode() || quietMode)
187
+ return;
188
+ console.log();
189
+ console.log(colors.bold(colors.primary(title)));
190
+ console.log(colors.dim("─".repeat(40)));
191
+ }
192
+ // Divider
193
+ export function divider() {
194
+ if (isJsonMode() || quietMode)
195
+ return;
196
+ console.log();
197
+ }
198
+ // Key-value display
199
+ export function keyValue(data) {
200
+ if (isJsonMode()) {
201
+ console.log(JSON.stringify(data, null, 2));
202
+ return;
203
+ }
204
+ const maxKeyLength = Math.max(...Object.keys(data).map((k) => k.length));
205
+ Object.entries(data).forEach(([key, value]) => {
206
+ const paddedKey = key.padEnd(maxKeyLength);
207
+ console.log(` ${colors.dim(paddedKey)} ${value}`);
208
+ });
209
+ }
210
+ // Code block
211
+ export function codeBlock(code, language) {
212
+ if (isJsonMode()) {
213
+ console.log(JSON.stringify({ code, language }, null, 2));
214
+ return;
215
+ }
216
+ console.log();
217
+ console.log(colors.dim("```" + (language || "")));
218
+ console.log(colors.code(code));
219
+ console.log(colors.dim("```"));
220
+ console.log();
221
+ }
222
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"output.js","sourceRoot":"","sources":["../../src/lib/output.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Output Formatting Utilities\n * Handles human-readable and JSON output modes\n */\n\nimport chalk from \"chalk\";\nimport Table from \"cli-table3\";\nimport ora, { type Ora } from \"ora\";\nimport { getConfigValue } from \"./config.js\";\n\nexport type OutputFormat = \"human\" | \"json\";\n\nlet currentFormat: OutputFormat = \"human\";\nlet quietMode = false;\n\nexport function setOutputFormat(format: OutputFormat): void {\n  currentFormat = format;\n}\n\nexport function setQuietMode(quiet: boolean): void {\n  quietMode = quiet;\n}\n\nexport function getOutputFormat(): OutputFormat {\n  return currentFormat;\n}\n\nexport function isJsonMode(): boolean {\n  return currentFormat === \"json\";\n}\n\nexport function isQuietMode(): boolean {\n  return quietMode;\n}\n\n// Colors\nexport const colors = {\n  primary: chalk.hex(\"#F59E0B\"), // Amber/Orange - Sendly brand\n  success: chalk.green,\n  error: chalk.red,\n  warning: chalk.yellow,\n  info: chalk.blue,\n  dim: chalk.dim,\n  bold: chalk.bold,\n  code: chalk.cyan,\n};\n\n// Success output\nexport function success(message: string, data?: Record<string, unknown>): void {\n  if (isJsonMode()) {\n    console.log(JSON.stringify({ success: true, message, ...data }, null, 2));\n    return;\n  }\n\n  if (quietMode) {\n    if (data?.id) console.log(data.id);\n    return;\n  }\n\n  console.log(`${colors.success(\"✓\")} ${message}`);\n  if (data) {\n    Object.entries(data).forEach(([key, value]) => {\n      console.log(`  ${colors.dim(key + \":\")} ${value}`);\n    });\n  }\n}\n\n// Error output\nexport function error(\n  message: string,\n  details?: Record<string, unknown>,\n): void {\n  if (isJsonMode()) {\n    console.error(\n      JSON.stringify({ error: true, message, ...details }, null, 2),\n    );\n    return;\n  }\n\n  console.error(`${colors.error(\"✗\")} ${message}`);\n  if (details && !quietMode) {\n    Object.entries(details).forEach(([key, value]) => {\n      console.error(`  ${colors.dim(key + \":\")} ${value}`);\n    });\n  }\n}\n\n// Warning output\nexport function warn(message: string): void {\n  if (isJsonMode()) return;\n  if (quietMode) return;\n  console.log(`${colors.warning(\"⚠\")} ${message}`);\n}\n\n// Info output\nexport function info(message: string): void {\n  if (isJsonMode()) return;\n  if (quietMode) return;\n  console.log(`${colors.info(\"ℹ\")} ${message}`);\n}\n\n// Print raw JSON\nexport function json(data: unknown): void {\n  console.log(JSON.stringify(data, null, 2));\n}\n\n// Print a table\nexport interface TableColumn {\n  header: string;\n  key: string;\n  width?: number;\n  formatter?: (value: unknown) => string;\n}\n\nexport function table(\n  data: Array<Record<string, any>>,\n  columns: TableColumn[],\n): void {\n  if (isJsonMode()) {\n    console.log(JSON.stringify(data, null, 2));\n    return;\n  }\n\n  if (data.length === 0) {\n    info(\"No data to display\");\n    return;\n  }\n\n  const t = new Table({\n    head: columns.map((c) => colors.bold(c.header)),\n    style: {\n      head: [],\n      border: [],\n    },\n    colWidths: columns.map((c) => c.width ?? null),\n  });\n\n  data.forEach((row) => {\n    t.push(\n      columns.map((col) => {\n        const value = row[col.key];\n        if (col.formatter) {\n          return col.formatter(value);\n        }\n        return String(value ?? \"-\");\n      }),\n    );\n  });\n\n  console.log(t.toString());\n}\n\n// Spinner for long operations\nexport function spinner(text: string): Ora {\n  if (isJsonMode() || quietMode) {\n    return {\n      start: () => ({ stop: () => {}, succeed: () => {}, fail: () => {} }),\n      stop: () => {},\n      succeed: () => {},\n      fail: () => {},\n    } as unknown as Ora;\n  }\n\n  return ora({\n    text,\n    color: \"yellow\",\n    spinner: \"dots\",\n  });\n}\n\n// Status formatters\nexport function formatStatus(status: string): string {\n  switch (status.toLowerCase()) {\n    case \"delivered\":\n    case \"success\":\n    case \"active\":\n    case \"verified\":\n      return colors.success(status);\n    case \"failed\":\n    case \"error\":\n    case \"revoked\":\n    case \"rejected\":\n      return colors.error(status);\n    case \"queued\":\n    case \"pending\":\n    case \"processing\":\n      return colors.warning(status);\n    case \"sent\":\n      return colors.info(status);\n    default:\n      return status;\n  }\n}\n\n// Format date\nexport function formatDate(date: string | Date | number): string {\n  const d = new Date(date);\n  return d.toLocaleString();\n}\n\n// Format relative time\nexport function formatRelativeTime(date: string | Date | number): string {\n  const d = new Date(date);\n  const now = new Date();\n  const diff = now.getTime() - d.getTime();\n\n  const seconds = Math.floor(diff / 1000);\n  const minutes = Math.floor(seconds / 60);\n  const hours = Math.floor(minutes / 60);\n  const days = Math.floor(hours / 24);\n\n  if (days > 0) return `${days}d ago`;\n  if (hours > 0) return `${hours}h ago`;\n  if (minutes > 0) return `${minutes}m ago`;\n  return `${seconds}s ago`;\n}\n\n// Format credits\nexport function formatCredits(credits: number): string {\n  return `${credits.toLocaleString()} credits`;\n}\n\n// Format phone number\nexport function formatPhone(phone: string): string {\n  return colors.code(phone);\n}\n\n// Header for commands\nexport function header(title: string): void {\n  if (isJsonMode() || quietMode) return;\n  console.log();\n  console.log(colors.bold(colors.primary(title)));\n  console.log(colors.dim(\"─\".repeat(40)));\n}\n\n// Divider\nexport function divider(): void {\n  if (isJsonMode() || quietMode) return;\n  console.log();\n}\n\n// Key-value display\nexport function keyValue(data: Record<string, unknown>): void {\n  if (isJsonMode()) {\n    console.log(JSON.stringify(data, null, 2));\n    return;\n  }\n\n  const maxKeyLength = Math.max(...Object.keys(data).map((k) => k.length));\n\n  Object.entries(data).forEach(([key, value]) => {\n    const paddedKey = key.padEnd(maxKeyLength);\n    console.log(`  ${colors.dim(paddedKey)}  ${value}`);\n  });\n}\n\n// Code block\nexport function codeBlock(code: string, language?: string): void {\n  if (isJsonMode()) {\n    console.log(JSON.stringify({ code, language }, null, 2));\n    return;\n  }\n\n  console.log();\n  console.log(colors.dim(\"```\" + (language || \"\")));\n  console.log(colors.code(code));\n  console.log(colors.dim(\"```\"));\n  console.log();\n}\n"]}