@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.
- package/README.md +323 -0
- package/bin/run.js +5 -0
- package/dist/commands/config/get.d.ts +13 -0
- package/dist/commands/config/get.js +38 -0
- package/dist/commands/config/list.d.ts +10 -0
- package/dist/commands/config/list.js +45 -0
- package/dist/commands/config/set.d.ts +14 -0
- package/dist/commands/config/set.js +67 -0
- package/dist/commands/credits/balance.d.ts +10 -0
- package/dist/commands/credits/balance.js +38 -0
- package/dist/commands/credits/history.d.ts +11 -0
- package/dist/commands/credits/history.js +88 -0
- package/dist/commands/doctor.d.ts +23 -0
- package/dist/commands/doctor.js +336 -0
- package/dist/commands/keys/create.d.ts +12 -0
- package/dist/commands/keys/create.js +47 -0
- package/dist/commands/keys/list.d.ts +10 -0
- package/dist/commands/keys/list.js +65 -0
- package/dist/commands/keys/revoke.d.ts +15 -0
- package/dist/commands/keys/revoke.js +68 -0
- package/dist/commands/login.d.ts +12 -0
- package/dist/commands/login.js +114 -0
- package/dist/commands/logout.d.ts +10 -0
- package/dist/commands/logout.js +20 -0
- package/dist/commands/logs/tail.d.ts +17 -0
- package/dist/commands/logs/tail.js +183 -0
- package/dist/commands/sms/batch.d.ts +16 -0
- package/dist/commands/sms/batch.js +163 -0
- package/dist/commands/sms/cancel.d.ts +13 -0
- package/dist/commands/sms/cancel.js +46 -0
- package/dist/commands/sms/get.d.ts +13 -0
- package/dist/commands/sms/get.js +51 -0
- package/dist/commands/sms/list.d.ts +12 -0
- package/dist/commands/sms/list.js +79 -0
- package/dist/commands/sms/schedule.d.ts +14 -0
- package/dist/commands/sms/schedule.js +91 -0
- package/dist/commands/sms/scheduled.d.ts +12 -0
- package/dist/commands/sms/scheduled.js +82 -0
- package/dist/commands/sms/send.d.ts +13 -0
- package/dist/commands/sms/send.js +70 -0
- package/dist/commands/webhooks/list.d.ts +10 -0
- package/dist/commands/webhooks/list.js +80 -0
- package/dist/commands/webhooks/listen.d.ts +20 -0
- package/dist/commands/webhooks/listen.js +202 -0
- package/dist/commands/whoami.d.ts +10 -0
- package/dist/commands/whoami.js +51 -0
- package/dist/index.d.ts +26 -0
- package/dist/index.js +27 -0
- package/dist/lib/api-client.d.ts +52 -0
- package/dist/lib/api-client.js +129 -0
- package/dist/lib/auth.d.ts +52 -0
- package/dist/lib/auth.js +171 -0
- package/dist/lib/base-command.d.ts +17 -0
- package/dist/lib/base-command.js +60 -0
- package/dist/lib/config.d.ts +54 -0
- package/dist/lib/config.js +182 -0
- package/dist/lib/output.d.ts +43 -0
- package/dist/lib/output.js +222 -0
- package/oclif.manifest.json +1147 -0
- package/package.json +98 -0
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Sendly CLI
|
|
3
|
+
*
|
|
4
|
+
* The official command-line interface for Sendly SMS API.
|
|
5
|
+
*
|
|
6
|
+
* @example
|
|
7
|
+
* ```bash
|
|
8
|
+
* # Install
|
|
9
|
+
* npm install -g @sendly/cli
|
|
10
|
+
*
|
|
11
|
+
* # Login
|
|
12
|
+
* sendly login
|
|
13
|
+
*
|
|
14
|
+
* # Send an SMS
|
|
15
|
+
* sendly sms send --to +15551234567 --text "Hello from Sendly!"
|
|
16
|
+
*
|
|
17
|
+
* # List messages
|
|
18
|
+
* sendly sms list
|
|
19
|
+
*
|
|
20
|
+
* # Listen for webhooks locally
|
|
21
|
+
* sendly webhooks listen --forward http://localhost:3000/webhook
|
|
22
|
+
* ```
|
|
23
|
+
*
|
|
24
|
+
* @packageDocumentation
|
|
25
|
+
*/
|
|
26
|
+
export { run } from "@oclif/core";
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Sendly CLI
|
|
3
|
+
*
|
|
4
|
+
* The official command-line interface for Sendly SMS API.
|
|
5
|
+
*
|
|
6
|
+
* @example
|
|
7
|
+
* ```bash
|
|
8
|
+
* # Install
|
|
9
|
+
* npm install -g @sendly/cli
|
|
10
|
+
*
|
|
11
|
+
* # Login
|
|
12
|
+
* sendly login
|
|
13
|
+
*
|
|
14
|
+
* # Send an SMS
|
|
15
|
+
* sendly sms send --to +15551234567 --text "Hello from Sendly!"
|
|
16
|
+
*
|
|
17
|
+
* # List messages
|
|
18
|
+
* sendly sms list
|
|
19
|
+
*
|
|
20
|
+
* # Listen for webhooks locally
|
|
21
|
+
* sendly webhooks listen --forward http://localhost:3000/webhook
|
|
22
|
+
* ```
|
|
23
|
+
*
|
|
24
|
+
* @packageDocumentation
|
|
25
|
+
*/
|
|
26
|
+
export { run } from "@oclif/core";
|
|
27
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXdCRztBQUVILE9BQU8sRUFBRSxHQUFHLEVBQUUsTUFBTSxhQUFhLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFNlbmRseSBDTElcbiAqXG4gKiBUaGUgb2ZmaWNpYWwgY29tbWFuZC1saW5lIGludGVyZmFjZSBmb3IgU2VuZGx5IFNNUyBBUEkuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYGJhc2hcbiAqICMgSW5zdGFsbFxuICogbnBtIGluc3RhbGwgLWcgQHNlbmRseS9jbGlcbiAqXG4gKiAjIExvZ2luXG4gKiBzZW5kbHkgbG9naW5cbiAqXG4gKiAjIFNlbmQgYW4gU01TXG4gKiBzZW5kbHkgc21zIHNlbmQgLS10byArMTU1NTEyMzQ1NjcgLS10ZXh0IFwiSGVsbG8gZnJvbSBTZW5kbHkhXCJcbiAqXG4gKiAjIExpc3QgbWVzc2FnZXNcbiAqIHNlbmRseSBzbXMgbGlzdFxuICpcbiAqICMgTGlzdGVuIGZvciB3ZWJob29rcyBsb2NhbGx5XG4gKiBzZW5kbHkgd2ViaG9va3MgbGlzdGVuIC0tZm9yd2FyZCBodHRwOi8vbG9jYWxob3N0OjMwMDAvd2ViaG9va1xuICogYGBgXG4gKlxuICogQHBhY2thZ2VEb2N1bWVudGF0aW9uXG4gKi9cblxuZXhwb3J0IHsgcnVuIH0gZnJvbSBcIkBvY2xpZi9jb3JlXCI7XG4iXX0=
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* API Client for Sendly CLI
|
|
3
|
+
* Handles all HTTP requests to the Sendly API
|
|
4
|
+
*/
|
|
5
|
+
export interface ApiResponse<T> {
|
|
6
|
+
data?: T;
|
|
7
|
+
error?: {
|
|
8
|
+
code: string;
|
|
9
|
+
message: string;
|
|
10
|
+
details?: Record<string, unknown>;
|
|
11
|
+
};
|
|
12
|
+
}
|
|
13
|
+
export interface RateLimitInfo {
|
|
14
|
+
limit: number;
|
|
15
|
+
remaining: number;
|
|
16
|
+
reset: number;
|
|
17
|
+
}
|
|
18
|
+
export declare class ApiError extends Error {
|
|
19
|
+
code: string;
|
|
20
|
+
statusCode: number;
|
|
21
|
+
details?: Record<string, unknown> | undefined;
|
|
22
|
+
constructor(code: string, message: string, statusCode: number, details?: Record<string, unknown> | undefined);
|
|
23
|
+
}
|
|
24
|
+
export declare class AuthenticationError extends ApiError {
|
|
25
|
+
constructor(message?: string);
|
|
26
|
+
}
|
|
27
|
+
export declare class RateLimitError extends ApiError {
|
|
28
|
+
retryAfter: number;
|
|
29
|
+
constructor(retryAfter: number, message?: string);
|
|
30
|
+
}
|
|
31
|
+
export declare class InsufficientCreditsError extends ApiError {
|
|
32
|
+
constructor(message?: string);
|
|
33
|
+
}
|
|
34
|
+
declare class ApiClient {
|
|
35
|
+
private rateLimitInfo?;
|
|
36
|
+
private getBaseUrl;
|
|
37
|
+
private getHeaders;
|
|
38
|
+
request<T>(method: string, path: string, options?: {
|
|
39
|
+
body?: Record<string, unknown>;
|
|
40
|
+
query?: Record<string, string | number | boolean | undefined>;
|
|
41
|
+
requireAuth?: boolean;
|
|
42
|
+
}): Promise<T>;
|
|
43
|
+
private updateRateLimitInfo;
|
|
44
|
+
private handleError;
|
|
45
|
+
getRateLimitInfo(): RateLimitInfo | undefined;
|
|
46
|
+
get<T>(path: string, query?: Record<string, string | number | boolean | undefined>, requireAuth?: boolean): Promise<T>;
|
|
47
|
+
post<T>(path: string, body?: Record<string, unknown>, requireAuth?: boolean): Promise<T>;
|
|
48
|
+
patch<T>(path: string, body?: Record<string, unknown>, requireAuth?: boolean): Promise<T>;
|
|
49
|
+
delete<T>(path: string, requireAuth?: boolean): Promise<T>;
|
|
50
|
+
}
|
|
51
|
+
export declare const apiClient: ApiClient;
|
|
52
|
+
export {};
|
|
@@ -0,0 +1,129 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* API Client for Sendly CLI
|
|
3
|
+
* Handles all HTTP requests to the Sendly API
|
|
4
|
+
*/
|
|
5
|
+
import { getAuthToken, getConfigValue } from "./config.js";
|
|
6
|
+
export class ApiError extends Error {
|
|
7
|
+
code;
|
|
8
|
+
statusCode;
|
|
9
|
+
details;
|
|
10
|
+
constructor(code, message, statusCode, details) {
|
|
11
|
+
super(message);
|
|
12
|
+
this.code = code;
|
|
13
|
+
this.statusCode = statusCode;
|
|
14
|
+
this.details = details;
|
|
15
|
+
this.name = "ApiError";
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
export class AuthenticationError extends ApiError {
|
|
19
|
+
constructor(message = "Not authenticated. Run 'sendly login' first.") {
|
|
20
|
+
super("authentication_error", message, 401);
|
|
21
|
+
this.name = "AuthenticationError";
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
export class RateLimitError extends ApiError {
|
|
25
|
+
retryAfter;
|
|
26
|
+
constructor(retryAfter, message = "Rate limit exceeded") {
|
|
27
|
+
super("rate_limit_exceeded", message, 429);
|
|
28
|
+
this.retryAfter = retryAfter;
|
|
29
|
+
this.name = "RateLimitError";
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
export class InsufficientCreditsError extends ApiError {
|
|
33
|
+
constructor(message = "Insufficient credits") {
|
|
34
|
+
super("insufficient_credits", message, 402);
|
|
35
|
+
this.name = "InsufficientCreditsError";
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
class ApiClient {
|
|
39
|
+
rateLimitInfo;
|
|
40
|
+
getBaseUrl() {
|
|
41
|
+
return getConfigValue("baseUrl") || "https://sendly.live";
|
|
42
|
+
}
|
|
43
|
+
getHeaders(requireAuth = true) {
|
|
44
|
+
const headers = {
|
|
45
|
+
"Content-Type": "application/json",
|
|
46
|
+
Accept: "application/json",
|
|
47
|
+
"User-Agent": "@sendly/cli/1.0.0",
|
|
48
|
+
};
|
|
49
|
+
if (requireAuth) {
|
|
50
|
+
const token = getAuthToken();
|
|
51
|
+
if (!token) {
|
|
52
|
+
throw new AuthenticationError();
|
|
53
|
+
}
|
|
54
|
+
headers["Authorization"] = `Bearer ${token}`;
|
|
55
|
+
}
|
|
56
|
+
return headers;
|
|
57
|
+
}
|
|
58
|
+
async request(method, path, options = {}) {
|
|
59
|
+
const { body, query, requireAuth = true } = options;
|
|
60
|
+
const url = new URL(`${this.getBaseUrl()}${path}`);
|
|
61
|
+
if (query) {
|
|
62
|
+
Object.entries(query).forEach(([key, value]) => {
|
|
63
|
+
if (value !== undefined) {
|
|
64
|
+
url.searchParams.append(key, String(value));
|
|
65
|
+
}
|
|
66
|
+
});
|
|
67
|
+
}
|
|
68
|
+
const response = await fetch(url.toString(), {
|
|
69
|
+
method,
|
|
70
|
+
headers: this.getHeaders(requireAuth),
|
|
71
|
+
body: body ? JSON.stringify(body) : undefined,
|
|
72
|
+
});
|
|
73
|
+
// Update rate limit info
|
|
74
|
+
this.updateRateLimitInfo(response.headers);
|
|
75
|
+
// Parse response
|
|
76
|
+
const data = await response.json().catch(() => ({}));
|
|
77
|
+
if (!response.ok) {
|
|
78
|
+
this.handleError(response.status, data);
|
|
79
|
+
}
|
|
80
|
+
return data;
|
|
81
|
+
}
|
|
82
|
+
updateRateLimitInfo(headers) {
|
|
83
|
+
const limit = headers.get("X-RateLimit-Limit");
|
|
84
|
+
const remaining = headers.get("X-RateLimit-Remaining");
|
|
85
|
+
const reset = headers.get("X-RateLimit-Reset");
|
|
86
|
+
if (limit && remaining && reset) {
|
|
87
|
+
this.rateLimitInfo = {
|
|
88
|
+
limit: parseInt(limit, 10),
|
|
89
|
+
remaining: parseInt(remaining, 10),
|
|
90
|
+
reset: parseInt(reset, 10),
|
|
91
|
+
};
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
handleError(statusCode, data) {
|
|
95
|
+
const error = data?.error || "unknown_error";
|
|
96
|
+
const message = data?.message || `HTTP ${statusCode}`;
|
|
97
|
+
const details = data?.details;
|
|
98
|
+
switch (statusCode) {
|
|
99
|
+
case 401:
|
|
100
|
+
case 403:
|
|
101
|
+
throw new AuthenticationError(message);
|
|
102
|
+
case 402:
|
|
103
|
+
throw new InsufficientCreditsError(message);
|
|
104
|
+
case 429:
|
|
105
|
+
const retryAfter = data?.retryAfter || 60;
|
|
106
|
+
throw new RateLimitError(retryAfter, message);
|
|
107
|
+
default:
|
|
108
|
+
throw new ApiError(error, message, statusCode, details);
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
getRateLimitInfo() {
|
|
112
|
+
return this.rateLimitInfo;
|
|
113
|
+
}
|
|
114
|
+
// Convenience methods
|
|
115
|
+
async get(path, query, requireAuth = true) {
|
|
116
|
+
return this.request("GET", path, { query, requireAuth });
|
|
117
|
+
}
|
|
118
|
+
async post(path, body, requireAuth = true) {
|
|
119
|
+
return this.request("POST", path, { body, requireAuth });
|
|
120
|
+
}
|
|
121
|
+
async patch(path, body, requireAuth = true) {
|
|
122
|
+
return this.request("PATCH", path, { body, requireAuth });
|
|
123
|
+
}
|
|
124
|
+
async delete(path, requireAuth = true) {
|
|
125
|
+
return this.request("DELETE", path, { requireAuth });
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
export const apiClient = new ApiClient();
|
|
129
|
+
//# 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 { getAuthToken, getConfigValue } from \"./config.js\";\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(message: string = \"Not authenticated. Run 'sendly login' first.\") {\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/1.0.0\",\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\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    const response = await fetch(url.toString(), {\n      method,\n      headers: this.getHeaders(requireAuth),\n      body: body ? JSON.stringify(body) : undefined,\n    });\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  }\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"]}
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Authentication utilities for CLI
|
|
3
|
+
* Handles browser-based login flow and API key authentication
|
|
4
|
+
*/
|
|
5
|
+
export interface DeviceCodeResponse {
|
|
6
|
+
deviceCode: string;
|
|
7
|
+
userCode: string;
|
|
8
|
+
verificationUrl: string;
|
|
9
|
+
expiresIn: number;
|
|
10
|
+
interval: number;
|
|
11
|
+
}
|
|
12
|
+
export interface TokenResponse {
|
|
13
|
+
accessToken: string;
|
|
14
|
+
refreshToken: string;
|
|
15
|
+
expiresIn: number;
|
|
16
|
+
userId: string;
|
|
17
|
+
email: string;
|
|
18
|
+
}
|
|
19
|
+
/**
|
|
20
|
+
* Generate a device code for browser-based authentication
|
|
21
|
+
*/
|
|
22
|
+
export declare function generateDeviceCode(): string;
|
|
23
|
+
/**
|
|
24
|
+
* Generate a secure device code for the auth flow
|
|
25
|
+
*/
|
|
26
|
+
export declare function generateSecureCode(): string;
|
|
27
|
+
/**
|
|
28
|
+
* Start the browser-based login flow
|
|
29
|
+
*/
|
|
30
|
+
export declare function browserLogin(): Promise<TokenResponse>;
|
|
31
|
+
/**
|
|
32
|
+
* Login with an API key directly
|
|
33
|
+
*/
|
|
34
|
+
export declare function apiKeyLogin(apiKey: string): Promise<void>;
|
|
35
|
+
/**
|
|
36
|
+
* Logout - clear all stored credentials
|
|
37
|
+
*/
|
|
38
|
+
export declare function logout(): void;
|
|
39
|
+
/**
|
|
40
|
+
* Check if currently authenticated
|
|
41
|
+
*/
|
|
42
|
+
export declare function checkAuth(): boolean;
|
|
43
|
+
/**
|
|
44
|
+
* Get current auth info for display
|
|
45
|
+
*/
|
|
46
|
+
export declare function getAuthInfo(): Promise<{
|
|
47
|
+
authenticated: boolean;
|
|
48
|
+
email?: string;
|
|
49
|
+
userId?: string;
|
|
50
|
+
environment: string;
|
|
51
|
+
keyType?: string;
|
|
52
|
+
}>;
|
package/dist/lib/auth.js
ADDED
|
@@ -0,0 +1,171 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Authentication utilities for CLI
|
|
3
|
+
* Handles browser-based login flow and API key authentication
|
|
4
|
+
*/
|
|
5
|
+
import open from "open";
|
|
6
|
+
import * as crypto from "node:crypto";
|
|
7
|
+
import { setAuthTokens, setApiKey, clearAuth, getConfigValue, isAuthenticated, getAuthToken, } from "./config.js";
|
|
8
|
+
import { colors, spinner } from "./output.js";
|
|
9
|
+
const DEVICE_CODE_LENGTH = 8;
|
|
10
|
+
const POLL_INTERVAL = 2000; // 2 seconds
|
|
11
|
+
const MAX_POLL_ATTEMPTS = 150; // 5 minutes max
|
|
12
|
+
/**
|
|
13
|
+
* Generate a device code for browser-based authentication
|
|
14
|
+
*/
|
|
15
|
+
export function generateDeviceCode() {
|
|
16
|
+
return crypto.randomBytes(4).toString("hex").toUpperCase();
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* Generate a secure device code for the auth flow
|
|
20
|
+
*/
|
|
21
|
+
export function generateSecureCode() {
|
|
22
|
+
const chars = "ABCDEFGHJKLMNPQRSTUVWXYZ23456789"; // Exclude confusing chars
|
|
23
|
+
let code = "";
|
|
24
|
+
const bytes = crypto.randomBytes(DEVICE_CODE_LENGTH);
|
|
25
|
+
for (let i = 0; i < DEVICE_CODE_LENGTH; i++) {
|
|
26
|
+
code += chars[bytes[i] % chars.length];
|
|
27
|
+
}
|
|
28
|
+
return code;
|
|
29
|
+
}
|
|
30
|
+
/**
|
|
31
|
+
* Start the browser-based login flow
|
|
32
|
+
*/
|
|
33
|
+
export async function browserLogin() {
|
|
34
|
+
const baseUrl = getConfigValue("baseUrl") || "https://sendly.live";
|
|
35
|
+
const deviceCode = generateSecureCode();
|
|
36
|
+
const userCode = `${deviceCode.slice(0, 4)}-${deviceCode.slice(4)}`;
|
|
37
|
+
// Request device code from server
|
|
38
|
+
const response = await fetch(`${baseUrl}/api/cli/auth/device-code`, {
|
|
39
|
+
method: "POST",
|
|
40
|
+
headers: { "Content-Type": "application/json" },
|
|
41
|
+
body: JSON.stringify({ deviceCode }),
|
|
42
|
+
});
|
|
43
|
+
if (!response.ok) {
|
|
44
|
+
const error = (await response.json().catch(() => ({})));
|
|
45
|
+
throw new Error(error.message || "Failed to initiate login");
|
|
46
|
+
}
|
|
47
|
+
const data = (await response.json());
|
|
48
|
+
// Display instructions to user
|
|
49
|
+
console.log();
|
|
50
|
+
console.log(colors.bold("Login to Sendly"));
|
|
51
|
+
console.log();
|
|
52
|
+
console.log(`Open this URL in your browser:`);
|
|
53
|
+
console.log(colors.primary(` ${data.verificationUrl}`));
|
|
54
|
+
console.log();
|
|
55
|
+
console.log(`And enter this code:`);
|
|
56
|
+
console.log(colors.bold(colors.primary(` ${userCode}`)));
|
|
57
|
+
console.log();
|
|
58
|
+
// Try to open browser automatically
|
|
59
|
+
try {
|
|
60
|
+
await open(data.verificationUrl);
|
|
61
|
+
console.log(colors.dim("Browser opened automatically"));
|
|
62
|
+
}
|
|
63
|
+
catch {
|
|
64
|
+
console.log(colors.dim("Please open the URL manually"));
|
|
65
|
+
}
|
|
66
|
+
console.log();
|
|
67
|
+
// Poll for token
|
|
68
|
+
const spin = spinner("Waiting for authorization...");
|
|
69
|
+
spin.start();
|
|
70
|
+
let attempts = 0;
|
|
71
|
+
while (attempts < MAX_POLL_ATTEMPTS) {
|
|
72
|
+
await sleep(data.interval * 1000 || POLL_INTERVAL);
|
|
73
|
+
attempts++;
|
|
74
|
+
try {
|
|
75
|
+
const tokenResponse = await fetch(`${baseUrl}/api/cli/auth/token`, {
|
|
76
|
+
method: "POST",
|
|
77
|
+
headers: { "Content-Type": "application/json" },
|
|
78
|
+
body: JSON.stringify({ deviceCode }),
|
|
79
|
+
});
|
|
80
|
+
if (tokenResponse.ok) {
|
|
81
|
+
const tokens = (await tokenResponse.json());
|
|
82
|
+
spin.succeed("Logged in successfully!");
|
|
83
|
+
// Store tokens
|
|
84
|
+
setAuthTokens(tokens.accessToken, tokens.refreshToken, tokens.expiresIn, tokens.userId, tokens.email);
|
|
85
|
+
return tokens;
|
|
86
|
+
}
|
|
87
|
+
const errorData = (await tokenResponse.json().catch(() => ({})));
|
|
88
|
+
if (errorData.error === "authorization_pending") {
|
|
89
|
+
// Still waiting, continue polling
|
|
90
|
+
continue;
|
|
91
|
+
}
|
|
92
|
+
if (errorData.error === "expired_token") {
|
|
93
|
+
spin.fail("Login request expired. Please try again.");
|
|
94
|
+
throw new Error("Login request expired");
|
|
95
|
+
}
|
|
96
|
+
if (errorData.error === "access_denied") {
|
|
97
|
+
spin.fail("Login was denied");
|
|
98
|
+
throw new Error("Login was denied");
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
catch (error) {
|
|
102
|
+
if (error.message.includes("expired") ||
|
|
103
|
+
error.message.includes("denied")) {
|
|
104
|
+
throw error;
|
|
105
|
+
}
|
|
106
|
+
// Network error, continue polling
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
spin.fail("Login timed out. Please try again.");
|
|
110
|
+
throw new Error("Login timed out");
|
|
111
|
+
}
|
|
112
|
+
/**
|
|
113
|
+
* Login with an API key directly
|
|
114
|
+
*/
|
|
115
|
+
export async function apiKeyLogin(apiKey) {
|
|
116
|
+
const baseUrl = getConfigValue("baseUrl") || "https://sendly.live";
|
|
117
|
+
// Validate the API key with the server
|
|
118
|
+
const response = await fetch(`${baseUrl}/api/cli/auth/verify-key`, {
|
|
119
|
+
method: "POST",
|
|
120
|
+
headers: {
|
|
121
|
+
"Content-Type": "application/json",
|
|
122
|
+
Authorization: `Bearer ${apiKey}`,
|
|
123
|
+
},
|
|
124
|
+
});
|
|
125
|
+
if (!response.ok) {
|
|
126
|
+
const error = (await response.json().catch(() => ({})));
|
|
127
|
+
throw new Error(error.message || "Invalid API key");
|
|
128
|
+
}
|
|
129
|
+
// Store the API key
|
|
130
|
+
setApiKey(apiKey);
|
|
131
|
+
}
|
|
132
|
+
/**
|
|
133
|
+
* Logout - clear all stored credentials
|
|
134
|
+
*/
|
|
135
|
+
export function logout() {
|
|
136
|
+
clearAuth();
|
|
137
|
+
}
|
|
138
|
+
/**
|
|
139
|
+
* Check if currently authenticated
|
|
140
|
+
*/
|
|
141
|
+
export function checkAuth() {
|
|
142
|
+
return isAuthenticated();
|
|
143
|
+
}
|
|
144
|
+
/**
|
|
145
|
+
* Get current auth info for display
|
|
146
|
+
*/
|
|
147
|
+
export async function getAuthInfo() {
|
|
148
|
+
const token = getAuthToken();
|
|
149
|
+
const email = getConfigValue("email");
|
|
150
|
+
const userId = getConfigValue("userId");
|
|
151
|
+
const apiKey = getConfigValue("apiKey");
|
|
152
|
+
const environment = getConfigValue("environment");
|
|
153
|
+
if (!token && !apiKey) {
|
|
154
|
+
return { authenticated: false, environment };
|
|
155
|
+
}
|
|
156
|
+
let keyType;
|
|
157
|
+
if (apiKey) {
|
|
158
|
+
keyType = apiKey.startsWith("sk_test_") ? "test" : "live";
|
|
159
|
+
}
|
|
160
|
+
return {
|
|
161
|
+
authenticated: true,
|
|
162
|
+
email,
|
|
163
|
+
userId,
|
|
164
|
+
environment,
|
|
165
|
+
keyType,
|
|
166
|
+
};
|
|
167
|
+
}
|
|
168
|
+
function sleep(ms) {
|
|
169
|
+
return new Promise((resolve) => setTimeout(resolve, ms));
|
|
170
|
+
}
|
|
171
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth.js","sourceRoot":"","sources":["../../src/lib/auth.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Authentication utilities for CLI\n * Handles browser-based login flow and API key authentication\n */\n\nimport open from \"open\";\nimport * as http from \"node:http\";\nimport * as crypto from \"node:crypto\";\nimport {\n  setAuthTokens,\n  setApiKey,\n  clearAuth,\n  getConfigValue,\n  isAuthenticated,\n  getAuthToken,\n} from \"./config.js\";\nimport { colors, spinner } from \"./output.js\";\n\nconst DEVICE_CODE_LENGTH = 8;\nconst POLL_INTERVAL = 2000; // 2 seconds\nconst MAX_POLL_ATTEMPTS = 150; // 5 minutes max\n\nexport interface DeviceCodeResponse {\n  deviceCode: string;\n  userCode: string;\n  verificationUrl: string;\n  expiresIn: number;\n  interval: number;\n}\n\nexport interface TokenResponse {\n  accessToken: string;\n  refreshToken: string;\n  expiresIn: number;\n  userId: string;\n  email: string;\n}\n\n/**\n * Generate a device code for browser-based authentication\n */\nexport function generateDeviceCode(): string {\n  return crypto.randomBytes(4).toString(\"hex\").toUpperCase();\n}\n\n/**\n * Generate a secure device code for the auth flow\n */\nexport function generateSecureCode(): string {\n  const chars = \"ABCDEFGHJKLMNPQRSTUVWXYZ23456789\"; // Exclude confusing chars\n  let code = \"\";\n  const bytes = crypto.randomBytes(DEVICE_CODE_LENGTH);\n  for (let i = 0; i < DEVICE_CODE_LENGTH; i++) {\n    code += chars[bytes[i] % chars.length];\n  }\n  return code;\n}\n\n/**\n * Start the browser-based login flow\n */\nexport async function browserLogin(): Promise<TokenResponse> {\n  const baseUrl = getConfigValue(\"baseUrl\") || \"https://sendly.live\";\n  const deviceCode = generateSecureCode();\n  const userCode = `${deviceCode.slice(0, 4)}-${deviceCode.slice(4)}`;\n\n  // Request device code from server\n  const response = await fetch(`${baseUrl}/api/cli/auth/device-code`, {\n    method: \"POST\",\n    headers: { \"Content-Type\": \"application/json\" },\n    body: JSON.stringify({ deviceCode }),\n  });\n\n  if (!response.ok) {\n    const error = (await response.json().catch(() => ({}))) as {\n      message?: string;\n    };\n    throw new Error(error.message || \"Failed to initiate login\");\n  }\n\n  const data = (await response.json()) as DeviceCodeResponse;\n\n  // Display instructions to user\n  console.log();\n  console.log(colors.bold(\"Login to Sendly\"));\n  console.log();\n  console.log(`Open this URL in your browser:`);\n  console.log(colors.primary(`  ${data.verificationUrl}`));\n  console.log();\n  console.log(`And enter this code:`);\n  console.log(colors.bold(colors.primary(`  ${userCode}`)));\n  console.log();\n\n  // Try to open browser automatically\n  try {\n    await open(data.verificationUrl);\n    console.log(colors.dim(\"Browser opened automatically\"));\n  } catch {\n    console.log(colors.dim(\"Please open the URL manually\"));\n  }\n\n  console.log();\n\n  // Poll for token\n  const spin = spinner(\"Waiting for authorization...\");\n  spin.start();\n\n  let attempts = 0;\n  while (attempts < MAX_POLL_ATTEMPTS) {\n    await sleep(data.interval * 1000 || POLL_INTERVAL);\n    attempts++;\n\n    try {\n      const tokenResponse = await fetch(`${baseUrl}/api/cli/auth/token`, {\n        method: \"POST\",\n        headers: { \"Content-Type\": \"application/json\" },\n        body: JSON.stringify({ deviceCode }),\n      });\n\n      if (tokenResponse.ok) {\n        const tokens = (await tokenResponse.json()) as TokenResponse;\n        spin.succeed(\"Logged in successfully!\");\n\n        // Store tokens\n        setAuthTokens(\n          tokens.accessToken,\n          tokens.refreshToken,\n          tokens.expiresIn,\n          tokens.userId,\n          tokens.email,\n        );\n\n        return tokens;\n      }\n\n      const errorData = (await tokenResponse.json().catch(() => ({}))) as {\n        error?: string;\n      };\n\n      if (errorData.error === \"authorization_pending\") {\n        // Still waiting, continue polling\n        continue;\n      }\n\n      if (errorData.error === \"expired_token\") {\n        spin.fail(\"Login request expired. Please try again.\");\n        throw new Error(\"Login request expired\");\n      }\n\n      if (errorData.error === \"access_denied\") {\n        spin.fail(\"Login was denied\");\n        throw new Error(\"Login was denied\");\n      }\n    } catch (error) {\n      if (\n        (error as Error).message.includes(\"expired\") ||\n        (error as Error).message.includes(\"denied\")\n      ) {\n        throw error;\n      }\n      // Network error, continue polling\n    }\n  }\n\n  spin.fail(\"Login timed out. Please try again.\");\n  throw new Error(\"Login timed out\");\n}\n\n/**\n * Login with an API key directly\n */\nexport async function apiKeyLogin(apiKey: string): Promise<void> {\n  const baseUrl = getConfigValue(\"baseUrl\") || \"https://sendly.live\";\n\n  // Validate the API key with the server\n  const response = await fetch(`${baseUrl}/api/cli/auth/verify-key`, {\n    method: \"POST\",\n    headers: {\n      \"Content-Type\": \"application/json\",\n      Authorization: `Bearer ${apiKey}`,\n    },\n  });\n\n  if (!response.ok) {\n    const error = (await response.json().catch(() => ({}))) as {\n      message?: string;\n    };\n    throw new Error(error.message || \"Invalid API key\");\n  }\n\n  // Store the API key\n  setApiKey(apiKey);\n}\n\n/**\n * Logout - clear all stored credentials\n */\nexport function logout(): void {\n  clearAuth();\n}\n\n/**\n * Check if currently authenticated\n */\nexport function checkAuth(): boolean {\n  return isAuthenticated();\n}\n\n/**\n * Get current auth info for display\n */\nexport async function getAuthInfo(): Promise<{\n  authenticated: boolean;\n  email?: string;\n  userId?: string;\n  environment: string;\n  keyType?: string;\n}> {\n  const token = getAuthToken();\n  const email = getConfigValue(\"email\");\n  const userId = getConfigValue(\"userId\");\n  const apiKey = getConfigValue(\"apiKey\");\n  const environment = getConfigValue(\"environment\");\n\n  if (!token && !apiKey) {\n    return { authenticated: false, environment };\n  }\n\n  let keyType: string | undefined;\n  if (apiKey) {\n    keyType = apiKey.startsWith(\"sk_test_\") ? \"test\" : \"live\";\n  }\n\n  return {\n    authenticated: true,\n    email,\n    userId,\n    environment,\n    keyType,\n  };\n}\n\nfunction sleep(ms: number): Promise<void> {\n  return new Promise((resolve) => setTimeout(resolve, ms));\n}\n"]}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Base Command class for all Sendly CLI commands
|
|
3
|
+
* Provides common functionality and flags
|
|
4
|
+
*/
|
|
5
|
+
import { Command } from "@oclif/core";
|
|
6
|
+
export declare abstract class BaseCommand extends Command {
|
|
7
|
+
static baseFlags: {
|
|
8
|
+
json: import("@oclif/core/lib/interfaces/parser.js").BooleanFlag<boolean>;
|
|
9
|
+
quiet: import("@oclif/core/lib/interfaces/parser.js").BooleanFlag<boolean>;
|
|
10
|
+
};
|
|
11
|
+
protected init(): Promise<void>;
|
|
12
|
+
protected catch(err: Error): Promise<void>;
|
|
13
|
+
protected requireAuth(): void;
|
|
14
|
+
}
|
|
15
|
+
export declare abstract class AuthenticatedCommand extends BaseCommand {
|
|
16
|
+
protected init(): Promise<void>;
|
|
17
|
+
}
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Base Command class for all Sendly CLI commands
|
|
3
|
+
* Provides common functionality and flags
|
|
4
|
+
*/
|
|
5
|
+
import { Command, Flags } from "@oclif/core";
|
|
6
|
+
import { setOutputFormat, setQuietMode, error } from "./output.js";
|
|
7
|
+
import { isAuthenticated } from "./config.js";
|
|
8
|
+
import { ApiError, AuthenticationError } from "./api-client.js";
|
|
9
|
+
export class BaseCommand extends Command {
|
|
10
|
+
static baseFlags = {
|
|
11
|
+
json: Flags.boolean({
|
|
12
|
+
description: "Output in JSON format",
|
|
13
|
+
default: false,
|
|
14
|
+
}),
|
|
15
|
+
quiet: Flags.boolean({
|
|
16
|
+
char: "q",
|
|
17
|
+
description: "Minimal output",
|
|
18
|
+
default: false,
|
|
19
|
+
}),
|
|
20
|
+
};
|
|
21
|
+
async init() {
|
|
22
|
+
await super.init();
|
|
23
|
+
const { flags } = await this.parse(this.constructor);
|
|
24
|
+
if (flags.json) {
|
|
25
|
+
setOutputFormat("json");
|
|
26
|
+
}
|
|
27
|
+
if (flags.quiet) {
|
|
28
|
+
setQuietMode(true);
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
async catch(err) {
|
|
32
|
+
if (err instanceof AuthenticationError) {
|
|
33
|
+
error("Not authenticated", {
|
|
34
|
+
hint: "Run 'sendly login' to authenticate",
|
|
35
|
+
});
|
|
36
|
+
this.exit(1);
|
|
37
|
+
}
|
|
38
|
+
if (err instanceof ApiError) {
|
|
39
|
+
error(err.message, {
|
|
40
|
+
code: err.code,
|
|
41
|
+
...(err.details || {}),
|
|
42
|
+
});
|
|
43
|
+
this.exit(1);
|
|
44
|
+
}
|
|
45
|
+
error(err.message);
|
|
46
|
+
this.exit(1);
|
|
47
|
+
}
|
|
48
|
+
requireAuth() {
|
|
49
|
+
if (!isAuthenticated()) {
|
|
50
|
+
throw new AuthenticationError();
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
export class AuthenticatedCommand extends BaseCommand {
|
|
55
|
+
async init() {
|
|
56
|
+
await super.init();
|
|
57
|
+
this.requireAuth();
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFzZS1jb21tYW5kLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2xpYi9iYXNlLWNvbW1hbmQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7OztHQUdHO0FBRUgsT0FBTyxFQUFFLE9BQU8sRUFBRSxLQUFLLEVBQUUsTUFBTSxhQUFhLENBQUM7QUFDN0MsT0FBTyxFQUFFLGVBQWUsRUFBRSxZQUFZLEVBQUUsS0FBSyxFQUFFLE1BQU0sYUFBYSxDQUFDO0FBQ25FLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSxhQUFhLENBQUM7QUFDOUMsT0FBTyxFQUFFLFFBQVEsRUFBRSxtQkFBbUIsRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBRWhFLE1BQU0sT0FBZ0IsV0FBWSxTQUFRLE9BQU87SUFDL0MsTUFBTSxDQUFDLFNBQVMsR0FBRztRQUNqQixJQUFJLEVBQUUsS0FBSyxDQUFDLE9BQU8sQ0FBQztZQUNsQixXQUFXLEVBQUUsdUJBQXVCO1lBQ3BDLE9BQU8sRUFBRSxLQUFLO1NBQ2YsQ0FBQztRQUNGLEtBQUssRUFBRSxLQUFLLENBQUMsT0FBTyxDQUFDO1lBQ25CLElBQUksRUFBRSxHQUFHO1lBQ1QsV0FBVyxFQUFFLGdCQUFnQjtZQUM3QixPQUFPLEVBQUUsS0FBSztTQUNmLENBQUM7S0FDSCxDQUFDO0lBRVEsS0FBSyxDQUFDLElBQUk7UUFDbEIsTUFBTSxLQUFLLENBQUMsSUFBSSxFQUFFLENBQUM7UUFDbkIsTUFBTSxFQUFFLEtBQUssRUFBRSxHQUFHLE1BQU0sSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsV0FBaUMsQ0FBQyxDQUFDO1FBRTNFLElBQUksS0FBSyxDQUFDLElBQUksRUFBRSxDQUFDO1lBQ2YsZUFBZSxDQUFDLE1BQU0sQ0FBQyxDQUFDO1FBQzFCLENBQUM7UUFDRCxJQUFJLEtBQUssQ0FBQyxLQUFLLEVBQUUsQ0FBQztZQUNoQixZQUFZLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDckIsQ0FBQztJQUNILENBQUM7SUFFUyxLQUFLLENBQUMsS0FBSyxDQUFDLEdBQVU7UUFDOUIsSUFBSSxHQUFHLFlBQVksbUJBQW1CLEVBQUUsQ0FBQztZQUN2QyxLQUFLLENBQUMsbUJBQW1CLEVBQUU7Z0JBQ3pCLElBQUksRUFBRSxvQ0FBb0M7YUFDM0MsQ0FBQyxDQUFDO1lBQ0gsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNmLENBQUM7UUFFRCxJQUFJLEdBQUcsWUFBWSxRQUFRLEVBQUUsQ0FBQztZQUM1QixLQUFLLENBQUMsR0FBRyxDQUFDLE9BQU8sRUFBRTtnQkFDakIsSUFBSSxFQUFFLEdBQUcsQ0FBQyxJQUFJO2dCQUNkLEdBQUcsQ0FBQyxHQUFHLENBQUMsT0FBTyxJQUFJLEVBQUUsQ0FBQzthQUN2QixDQUFDLENBQUM7WUFDSCxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ2YsQ0FBQztRQUVELEtBQUssQ0FBQyxHQUFHLENBQUMsT0FBTyxDQUFDLENBQUM7UUFDbkIsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUNmLENBQUM7SUFFUyxXQUFXO1FBQ25CLElBQUksQ0FBQyxlQUFlLEVBQUUsRUFBRSxDQUFDO1lBQ3ZCLE1BQU0sSUFBSSxtQkFBbUIsRUFBRSxDQUFDO1FBQ2xDLENBQUM7SUFDSCxDQUFDOztBQUdILE1BQU0sT0FBZ0Isb0JBQXFCLFNBQVEsV0FBVztJQUNsRCxLQUFLLENBQUMsSUFBSTtRQUNsQixNQUFNLEtBQUssQ0FBQyxJQUFJLEVBQUUsQ0FBQztRQUNuQixJQUFJLENBQUMsV0FBVyxFQUFFLENBQUM7SUFDckIsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBCYXNlIENvbW1hbmQgY2xhc3MgZm9yIGFsbCBTZW5kbHkgQ0xJIGNvbW1hbmRzXG4gKiBQcm92aWRlcyBjb21tb24gZnVuY3Rpb25hbGl0eSBhbmQgZmxhZ3NcbiAqL1xuXG5pbXBvcnQgeyBDb21tYW5kLCBGbGFncyB9IGZyb20gXCJAb2NsaWYvY29yZVwiO1xuaW1wb3J0IHsgc2V0T3V0cHV0Rm9ybWF0LCBzZXRRdWlldE1vZGUsIGVycm9yIH0gZnJvbSBcIi4vb3V0cHV0LmpzXCI7XG5pbXBvcnQgeyBpc0F1dGhlbnRpY2F0ZWQgfSBmcm9tIFwiLi9jb25maWcuanNcIjtcbmltcG9ydCB7IEFwaUVycm9yLCBBdXRoZW50aWNhdGlvbkVycm9yIH0gZnJvbSBcIi4vYXBpLWNsaWVudC5qc1wiO1xuXG5leHBvcnQgYWJzdHJhY3QgY2xhc3MgQmFzZUNvbW1hbmQgZXh0ZW5kcyBDb21tYW5kIHtcbiAgc3RhdGljIGJhc2VGbGFncyA9IHtcbiAgICBqc29uOiBGbGFncy5ib29sZWFuKHtcbiAgICAgIGRlc2NyaXB0aW9uOiBcIk91dHB1dCBpbiBKU09OIGZvcm1hdFwiLFxuICAgICAgZGVmYXVsdDogZmFsc2UsXG4gICAgfSksXG4gICAgcXVpZXQ6IEZsYWdzLmJvb2xlYW4oe1xuICAgICAgY2hhcjogXCJxXCIsXG4gICAgICBkZXNjcmlwdGlvbjogXCJNaW5pbWFsIG91dHB1dFwiLFxuICAgICAgZGVmYXVsdDogZmFsc2UsXG4gICAgfSksXG4gIH07XG5cbiAgcHJvdGVjdGVkIGFzeW5jIGluaXQoKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgYXdhaXQgc3VwZXIuaW5pdCgpO1xuICAgIGNvbnN0IHsgZmxhZ3MgfSA9IGF3YWl0IHRoaXMucGFyc2UodGhpcy5jb25zdHJ1Y3RvciBhcyB0eXBlb2YgQmFzZUNvbW1hbmQpO1xuXG4gICAgaWYgKGZsYWdzLmpzb24pIHtcbiAgICAgIHNldE91dHB1dEZvcm1hdChcImpzb25cIik7XG4gICAgfVxuICAgIGlmIChmbGFncy5xdWlldCkge1xuICAgICAgc2V0UXVpZXRNb2RlKHRydWUpO1xuICAgIH1cbiAgfVxuXG4gIHByb3RlY3RlZCBhc3luYyBjYXRjaChlcnI6IEVycm9yKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgaWYgKGVyciBpbnN0YW5jZW9mIEF1dGhlbnRpY2F0aW9uRXJyb3IpIHtcbiAgICAgIGVycm9yKFwiTm90IGF1dGhlbnRpY2F0ZWRcIiwge1xuICAgICAgICBoaW50OiBcIlJ1biAnc2VuZGx5IGxvZ2luJyB0byBhdXRoZW50aWNhdGVcIixcbiAgICAgIH0pO1xuICAgICAgdGhpcy5leGl0KDEpO1xuICAgIH1cblxuICAgIGlmIChlcnIgaW5zdGFuY2VvZiBBcGlFcnJvcikge1xuICAgICAgZXJyb3IoZXJyLm1lc3NhZ2UsIHtcbiAgICAgICAgY29kZTogZXJyLmNvZGUsXG4gICAgICAgIC4uLihlcnIuZGV0YWlscyB8fCB7fSksXG4gICAgICB9KTtcbiAgICAgIHRoaXMuZXhpdCgxKTtcbiAgICB9XG5cbiAgICBlcnJvcihlcnIubWVzc2FnZSk7XG4gICAgdGhpcy5leGl0KDEpO1xuICB9XG5cbiAgcHJvdGVjdGVkIHJlcXVpcmVBdXRoKCk6IHZvaWQge1xuICAgIGlmICghaXNBdXRoZW50aWNhdGVkKCkpIHtcbiAgICAgIHRocm93IG5ldyBBdXRoZW50aWNhdGlvbkVycm9yKCk7XG4gICAgfVxuICB9XG59XG5cbmV4cG9ydCBhYnN0cmFjdCBjbGFzcyBBdXRoZW50aWNhdGVkQ29tbWFuZCBleHRlbmRzIEJhc2VDb21tYW5kIHtcbiAgcHJvdGVjdGVkIGFzeW5jIGluaXQoKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgYXdhaXQgc3VwZXIuaW5pdCgpO1xuICAgIHRoaXMucmVxdWlyZUF1dGgoKTtcbiAgfVxufVxuIl19
|
|
@@ -0,0 +1,54 @@
|
|
|
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
|
+
export interface SendlyConfig {
|
|
16
|
+
apiKey?: string;
|
|
17
|
+
accessToken?: string;
|
|
18
|
+
refreshToken?: string;
|
|
19
|
+
tokenExpiresAt?: number;
|
|
20
|
+
userId?: string;
|
|
21
|
+
email?: string;
|
|
22
|
+
environment: "test" | "live";
|
|
23
|
+
baseUrl: string;
|
|
24
|
+
defaultFormat: "human" | "json";
|
|
25
|
+
colorEnabled: boolean;
|
|
26
|
+
timeout: number;
|
|
27
|
+
maxRetries: number;
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* Check if running in CI environment
|
|
31
|
+
*/
|
|
32
|
+
export declare function isCI(): boolean;
|
|
33
|
+
/**
|
|
34
|
+
* Check if color output is disabled
|
|
35
|
+
*/
|
|
36
|
+
export declare function isColorDisabled(): boolean;
|
|
37
|
+
declare const config: Conf<SendlyConfig>;
|
|
38
|
+
/**
|
|
39
|
+
* Get effective config value with environment variable override
|
|
40
|
+
* Priority: env var > config file > default
|
|
41
|
+
*/
|
|
42
|
+
export declare function getEffectiveValue<K extends keyof SendlyConfig>(key: K): SendlyConfig[K];
|
|
43
|
+
export declare function getConfig(): SendlyConfig;
|
|
44
|
+
export declare function setConfig<K extends keyof SendlyConfig>(key: K, value: SendlyConfig[K]): void;
|
|
45
|
+
export declare function getConfigValue<K extends keyof SendlyConfig>(key: K): SendlyConfig[K];
|
|
46
|
+
export declare function clearConfig(): void;
|
|
47
|
+
export declare function clearAuth(): void;
|
|
48
|
+
export declare function isAuthenticated(): boolean;
|
|
49
|
+
export declare function getAuthToken(): string | undefined;
|
|
50
|
+
export declare function setApiKey(apiKey: string): void;
|
|
51
|
+
export declare function setAuthTokens(accessToken: string, refreshToken: string, expiresIn: number, userId: string, email: string): void;
|
|
52
|
+
export declare function getConfigPath(): string;
|
|
53
|
+
export declare function getConfigDir(): string;
|
|
54
|
+
export { config };
|