@hasna/connectors 0.5.0 → 0.5.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/bin/index.js +147 -6
- package/bin/mcp.js +92 -1
- package/bin/serve.js +91 -0
- package/connectors/connect-ably/.env.example +11 -0
- package/connectors/connect-ably/CLAUDE.md +111 -0
- package/connectors/connect-ably/README.md +193 -0
- package/connectors/connect-ably/package.json +54 -0
- package/connectors/connect-ably/scripts/release.ts +179 -0
- package/connectors/connect-ably/src/api/channels.ts +33 -0
- package/connectors/connect-ably/src/api/client.ts +203 -0
- package/connectors/connect-ably/src/api/index.ts +59 -0
- package/connectors/connect-ably/src/api/messages.ts +48 -0
- package/connectors/connect-ably/src/api/presence.ts +39 -0
- package/connectors/connect-ably/src/api/stats.ts +29 -0
- package/connectors/connect-ably/src/cli/index.ts +397 -0
- package/connectors/connect-ably/src/index.ts +102 -0
- package/connectors/connect-ably/src/types/index.ts +294 -0
- package/connectors/connect-ably/src/utils/auth.ts +274 -0
- package/connectors/connect-ably/src/utils/bulk.ts +212 -0
- package/connectors/connect-ably/src/utils/config.ts +323 -0
- package/connectors/connect-ably/src/utils/output.ts +175 -0
- package/connectors/connect-ably/src/utils/settings.ts +114 -0
- package/connectors/connect-ably/src/utils/storage.ts +198 -0
- package/connectors/connect-ably/tsconfig.json +16 -0
- package/connectors/connect-box/.env.example +11 -0
- package/connectors/connect-box/CLAUDE.md +272 -0
- package/connectors/connect-box/README.md +193 -0
- package/connectors/connect-box/package.json +51 -0
- package/connectors/connect-box/scripts/release.ts +179 -0
- package/connectors/connect-box/src/api/client.ts +213 -0
- package/connectors/connect-box/src/api/example.ts +48 -0
- package/connectors/connect-box/src/api/index.ts +51 -0
- package/connectors/connect-box/src/cli/index.ts +254 -0
- package/connectors/connect-box/src/index.ts +103 -0
- package/connectors/connect-box/src/types/index.ts +237 -0
- package/connectors/connect-box/src/utils/auth.ts +274 -0
- package/connectors/connect-box/src/utils/bulk.ts +212 -0
- package/connectors/connect-box/src/utils/config.ts +326 -0
- package/connectors/connect-box/src/utils/output.ts +175 -0
- package/connectors/connect-box/src/utils/settings.ts +114 -0
- package/connectors/connect-box/src/utils/storage.ts +198 -0
- package/connectors/connect-box/tsconfig.json +16 -0
- package/connectors/connect-clearbit/.env.example +11 -0
- package/connectors/connect-clearbit/CLAUDE.md +272 -0
- package/connectors/connect-clearbit/README.md +193 -0
- package/connectors/connect-clearbit/package.json +51 -0
- package/connectors/connect-clearbit/scripts/release.ts +179 -0
- package/connectors/connect-clearbit/src/api/client.ts +213 -0
- package/connectors/connect-clearbit/src/api/example.ts +48 -0
- package/connectors/connect-clearbit/src/api/index.ts +51 -0
- package/connectors/connect-clearbit/src/cli/index.ts +254 -0
- package/connectors/connect-clearbit/src/index.ts +103 -0
- package/connectors/connect-clearbit/src/types/index.ts +237 -0
- package/connectors/connect-clearbit/src/utils/auth.ts +274 -0
- package/connectors/connect-clearbit/src/utils/bulk.ts +212 -0
- package/connectors/connect-clearbit/src/utils/config.ts +326 -0
- package/connectors/connect-clearbit/src/utils/output.ts +175 -0
- package/connectors/connect-clearbit/src/utils/settings.ts +114 -0
- package/connectors/connect-clearbit/src/utils/storage.ts +198 -0
- package/connectors/connect-clearbit/tsconfig.json +16 -0
- package/connectors/connect-coda/.env.example +11 -0
- package/connectors/connect-coda/CLAUDE.md +272 -0
- package/connectors/connect-coda/README.md +193 -0
- package/connectors/connect-coda/package.json +51 -0
- package/connectors/connect-coda/scripts/release.ts +179 -0
- package/connectors/connect-coda/src/api/client.ts +213 -0
- package/connectors/connect-coda/src/api/example.ts +48 -0
- package/connectors/connect-coda/src/api/index.ts +51 -0
- package/connectors/connect-coda/src/cli/index.ts +254 -0
- package/connectors/connect-coda/src/index.ts +103 -0
- package/connectors/connect-coda/src/types/index.ts +237 -0
- package/connectors/connect-coda/src/utils/auth.ts +274 -0
- package/connectors/connect-coda/src/utils/bulk.ts +212 -0
- package/connectors/connect-coda/src/utils/config.ts +326 -0
- package/connectors/connect-coda/src/utils/output.ts +175 -0
- package/connectors/connect-coda/src/utils/settings.ts +114 -0
- package/connectors/connect-coda/src/utils/storage.ts +198 -0
- package/connectors/connect-coda/tsconfig.json +16 -0
- package/connectors/connect-dropbox/.env.example +11 -0
- package/connectors/connect-dropbox/CLAUDE.md +119 -0
- package/connectors/connect-dropbox/README.md +193 -0
- package/connectors/connect-dropbox/package.json +51 -0
- package/connectors/connect-dropbox/src/api/client.ts +222 -0
- package/connectors/connect-dropbox/src/api/index.ts +395 -0
- package/connectors/connect-dropbox/src/cli/index.ts +627 -0
- package/connectors/connect-dropbox/src/index.ts +20 -0
- package/connectors/connect-dropbox/src/types/index.ts +516 -0
- package/connectors/connect-dropbox/src/utils/config.ts +197 -0
- package/connectors/connect-dropbox/tsconfig.json +16 -0
- package/connectors/connect-linode/.env.example +11 -0
- package/connectors/connect-linode/CLAUDE.md +272 -0
- package/connectors/connect-linode/README.md +193 -0
- package/connectors/connect-linode/package.json +51 -0
- package/connectors/connect-linode/scripts/release.ts +179 -0
- package/connectors/connect-linode/src/api/client.ts +213 -0
- package/connectors/connect-linode/src/api/example.ts +48 -0
- package/connectors/connect-linode/src/api/index.ts +51 -0
- package/connectors/connect-linode/src/cli/index.ts +254 -0
- package/connectors/connect-linode/src/index.ts +103 -0
- package/connectors/connect-linode/src/types/index.ts +237 -0
- package/connectors/connect-linode/src/utils/auth.ts +274 -0
- package/connectors/connect-linode/src/utils/bulk.ts +212 -0
- package/connectors/connect-linode/src/utils/config.ts +326 -0
- package/connectors/connect-linode/src/utils/output.ts +175 -0
- package/connectors/connect-linode/src/utils/settings.ts +114 -0
- package/connectors/connect-linode/src/utils/storage.ts +198 -0
- package/connectors/connect-linode/tsconfig.json +16 -0
- package/connectors/connect-mailgun/.env.example +11 -0
- package/connectors/connect-mailgun/CLAUDE.md +272 -0
- package/connectors/connect-mailgun/README.md +193 -0
- package/connectors/connect-mailgun/package.json +51 -0
- package/connectors/connect-mailgun/scripts/release.ts +179 -0
- package/connectors/connect-mailgun/src/api/client.ts +213 -0
- package/connectors/connect-mailgun/src/api/example.ts +48 -0
- package/connectors/connect-mailgun/src/api/index.ts +51 -0
- package/connectors/connect-mailgun/src/cli/index.ts +254 -0
- package/connectors/connect-mailgun/src/index.ts +103 -0
- package/connectors/connect-mailgun/src/types/index.ts +237 -0
- package/connectors/connect-mailgun/src/utils/auth.ts +274 -0
- package/connectors/connect-mailgun/src/utils/bulk.ts +212 -0
- package/connectors/connect-mailgun/src/utils/config.ts +326 -0
- package/connectors/connect-mailgun/src/utils/output.ts +175 -0
- package/connectors/connect-mailgun/src/utils/settings.ts +114 -0
- package/connectors/connect-mailgun/src/utils/storage.ts +198 -0
- package/connectors/connect-mailgun/tsconfig.json +16 -0
- package/connectors/connect-messagebird/.env.example +11 -0
- package/connectors/connect-messagebird/CLAUDE.md +272 -0
- package/connectors/connect-messagebird/README.md +193 -0
- package/connectors/connect-messagebird/package.json +51 -0
- package/connectors/connect-messagebird/scripts/release.ts +179 -0
- package/connectors/connect-messagebird/src/api/client.ts +213 -0
- package/connectors/connect-messagebird/src/api/example.ts +48 -0
- package/connectors/connect-messagebird/src/api/index.ts +51 -0
- package/connectors/connect-messagebird/src/cli/index.ts +254 -0
- package/connectors/connect-messagebird/src/index.ts +103 -0
- package/connectors/connect-messagebird/src/types/index.ts +237 -0
- package/connectors/connect-messagebird/src/utils/auth.ts +274 -0
- package/connectors/connect-messagebird/src/utils/bulk.ts +212 -0
- package/connectors/connect-messagebird/src/utils/config.ts +326 -0
- package/connectors/connect-messagebird/src/utils/output.ts +175 -0
- package/connectors/connect-messagebird/src/utils/settings.ts +114 -0
- package/connectors/connect-messagebird/src/utils/storage.ts +198 -0
- package/connectors/connect-messagebird/tsconfig.json +16 -0
- package/connectors/connect-miro/.env.example +11 -0
- package/connectors/connect-miro/CLAUDE.md +272 -0
- package/connectors/connect-miro/README.md +193 -0
- package/connectors/connect-miro/package.json +51 -0
- package/connectors/connect-miro/scripts/release.ts +179 -0
- package/connectors/connect-miro/src/api/client.ts +213 -0
- package/connectors/connect-miro/src/api/example.ts +48 -0
- package/connectors/connect-miro/src/api/index.ts +51 -0
- package/connectors/connect-miro/src/cli/index.ts +254 -0
- package/connectors/connect-miro/src/index.ts +103 -0
- package/connectors/connect-miro/src/types/index.ts +237 -0
- package/connectors/connect-miro/src/utils/auth.ts +274 -0
- package/connectors/connect-miro/src/utils/bulk.ts +212 -0
- package/connectors/connect-miro/src/utils/config.ts +326 -0
- package/connectors/connect-miro/src/utils/output.ts +175 -0
- package/connectors/connect-miro/src/utils/settings.ts +114 -0
- package/connectors/connect-miro/src/utils/storage.ts +198 -0
- package/connectors/connect-miro/tsconfig.json +16 -0
- package/connectors/connect-monday/.env.example +11 -0
- package/connectors/connect-monday/CLAUDE.md +128 -0
- package/connectors/connect-monday/README.md +193 -0
- package/connectors/connect-monday/package.json +52 -0
- package/connectors/connect-monday/src/api/client.ts +59 -0
- package/connectors/connect-monday/src/api/index.ts +539 -0
- package/connectors/connect-monday/src/cli/index.ts +479 -0
- package/connectors/connect-monday/src/index.ts +19 -0
- package/connectors/connect-monday/src/types/index.ts +274 -0
- package/connectors/connect-monday/src/utils/config.ts +197 -0
- package/connectors/connect-monday/src/utils/output.ts +119 -0
- package/connectors/connect-monday/tsconfig.json +16 -0
- package/connectors/connect-pipedrive/.env.example +11 -0
- package/connectors/connect-pipedrive/CLAUDE.md +128 -0
- package/connectors/connect-pipedrive/README.md +193 -0
- package/connectors/connect-pipedrive/package.json +52 -0
- package/connectors/connect-pipedrive/src/api/client.ts +121 -0
- package/connectors/connect-pipedrive/src/api/index.ts +306 -0
- package/connectors/connect-pipedrive/src/cli/index.ts +824 -0
- package/connectors/connect-pipedrive/src/index.ts +19 -0
- package/connectors/connect-pipedrive/src/types/index.ts +335 -0
- package/connectors/connect-pipedrive/src/utils/config.ts +171 -0
- package/connectors/connect-pipedrive/src/utils/output.ts +119 -0
- package/connectors/connect-pipedrive/tsconfig.json +16 -0
- package/connectors/connect-pusher/.env.example +11 -0
- package/connectors/connect-pusher/CLAUDE.md +272 -0
- package/connectors/connect-pusher/README.md +193 -0
- package/connectors/connect-pusher/package.json +51 -0
- package/connectors/connect-pusher/scripts/release.ts +179 -0
- package/connectors/connect-pusher/src/api/client.ts +213 -0
- package/connectors/connect-pusher/src/api/example.ts +48 -0
- package/connectors/connect-pusher/src/api/index.ts +51 -0
- package/connectors/connect-pusher/src/cli/index.ts +254 -0
- package/connectors/connect-pusher/src/index.ts +103 -0
- package/connectors/connect-pusher/src/types/index.ts +237 -0
- package/connectors/connect-pusher/src/utils/auth.ts +274 -0
- package/connectors/connect-pusher/src/utils/bulk.ts +212 -0
- package/connectors/connect-pusher/src/utils/config.ts +326 -0
- package/connectors/connect-pusher/src/utils/output.ts +175 -0
- package/connectors/connect-pusher/src/utils/settings.ts +114 -0
- package/connectors/connect-pusher/src/utils/storage.ts +198 -0
- package/connectors/connect-pusher/tsconfig.json +16 -0
- package/connectors/connect-vonage/.env.example +11 -0
- package/connectors/connect-vonage/CLAUDE.md +272 -0
- package/connectors/connect-vonage/README.md +193 -0
- package/connectors/connect-vonage/package.json +51 -0
- package/connectors/connect-vonage/scripts/release.ts +179 -0
- package/connectors/connect-vonage/src/api/client.ts +213 -0
- package/connectors/connect-vonage/src/api/example.ts +48 -0
- package/connectors/connect-vonage/src/api/index.ts +51 -0
- package/connectors/connect-vonage/src/cli/index.ts +254 -0
- package/connectors/connect-vonage/src/index.ts +103 -0
- package/connectors/connect-vonage/src/types/index.ts +237 -0
- package/connectors/connect-vonage/src/utils/auth.ts +274 -0
- package/connectors/connect-vonage/src/utils/bulk.ts +212 -0
- package/connectors/connect-vonage/src/utils/config.ts +326 -0
- package/connectors/connect-vonage/src/utils/output.ts +175 -0
- package/connectors/connect-vonage/src/utils/settings.ts +114 -0
- package/connectors/connect-vonage/src/utils/storage.ts +198 -0
- package/connectors/connect-vonage/tsconfig.json +16 -0
- package/dist/index.js +91 -0
- package/package.json +1 -1
|
@@ -0,0 +1,212 @@
|
|
|
1
|
+
// ============================================
|
|
2
|
+
// Bulk Operations Utility
|
|
3
|
+
// ============================================
|
|
4
|
+
|
|
5
|
+
/**
|
|
6
|
+
* Options for bulk operations
|
|
7
|
+
*/
|
|
8
|
+
export interface BulkOperationOptions<T> {
|
|
9
|
+
/** Items to process */
|
|
10
|
+
items: T[];
|
|
11
|
+
/** Maximum concurrent operations (default: 10) */
|
|
12
|
+
concurrency?: number;
|
|
13
|
+
/** Dry run - preview without making changes */
|
|
14
|
+
dryRun?: boolean;
|
|
15
|
+
/** Progress callback */
|
|
16
|
+
onProgress?: (current: number, total: number, item: T) => void;
|
|
17
|
+
/** Error callback (return true to continue, false to stop) */
|
|
18
|
+
onError?: (error: Error, item: T) => boolean | void;
|
|
19
|
+
/** Delay between batches in ms (useful for rate limiting) */
|
|
20
|
+
batchDelay?: number;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* Result of a bulk operation
|
|
25
|
+
*/
|
|
26
|
+
export interface BulkOperationResult<T, R = void> {
|
|
27
|
+
/** Total items processed */
|
|
28
|
+
total: number;
|
|
29
|
+
/** Successfully processed count */
|
|
30
|
+
success: number;
|
|
31
|
+
/** Failed count */
|
|
32
|
+
failed: number;
|
|
33
|
+
/** Skipped count (dry run) */
|
|
34
|
+
skipped: number;
|
|
35
|
+
/** List of errors */
|
|
36
|
+
errors: Array<{ item: T; error: string }>;
|
|
37
|
+
/** Successfully processed items with results */
|
|
38
|
+
results: Array<{ item: T; result: R }>;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
/**
|
|
42
|
+
* Split an array into chunks of a given size
|
|
43
|
+
*/
|
|
44
|
+
export function chunkArray<T>(array: T[], size: number): T[][] {
|
|
45
|
+
const chunks: T[][] = [];
|
|
46
|
+
for (let i = 0; i < array.length; i += size) {
|
|
47
|
+
chunks.push(array.slice(i, i + size));
|
|
48
|
+
}
|
|
49
|
+
return chunks;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Sleep for a given number of milliseconds
|
|
54
|
+
*/
|
|
55
|
+
export function sleep(ms: number): Promise<void> {
|
|
56
|
+
return new Promise(resolve => setTimeout(resolve, ms));
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
/**
|
|
60
|
+
* Execute a bulk operation with concurrency control
|
|
61
|
+
*
|
|
62
|
+
* @param options - Bulk operation options
|
|
63
|
+
* @param operation - Async function to execute for each item
|
|
64
|
+
* @returns Result summary
|
|
65
|
+
*
|
|
66
|
+
* @example
|
|
67
|
+
* ```typescript
|
|
68
|
+
* const result = await executeBulk(
|
|
69
|
+
* {
|
|
70
|
+
* items: users,
|
|
71
|
+
* concurrency: 5,
|
|
72
|
+
* onProgress: (current, total, user) => {
|
|
73
|
+
* console.log(`Processing ${current}/${total}: ${user.email}`);
|
|
74
|
+
* },
|
|
75
|
+
* },
|
|
76
|
+
* async (user) => {
|
|
77
|
+
* await api.updateUser(user.id, { status: 'active' });
|
|
78
|
+
* }
|
|
79
|
+
* );
|
|
80
|
+
* console.log(`Success: ${result.success}, Failed: ${result.failed}`);
|
|
81
|
+
* ```
|
|
82
|
+
*/
|
|
83
|
+
export async function executeBulk<T, R = void>(
|
|
84
|
+
options: BulkOperationOptions<T>,
|
|
85
|
+
operation: (item: T) => Promise<R>
|
|
86
|
+
): Promise<BulkOperationResult<T, R>> {
|
|
87
|
+
const {
|
|
88
|
+
items,
|
|
89
|
+
concurrency = 10,
|
|
90
|
+
dryRun = false,
|
|
91
|
+
onProgress,
|
|
92
|
+
onError,
|
|
93
|
+
batchDelay = 0,
|
|
94
|
+
} = options;
|
|
95
|
+
|
|
96
|
+
const result: BulkOperationResult<T, R> = {
|
|
97
|
+
total: items.length,
|
|
98
|
+
success: 0,
|
|
99
|
+
failed: 0,
|
|
100
|
+
skipped: 0,
|
|
101
|
+
errors: [],
|
|
102
|
+
results: [],
|
|
103
|
+
};
|
|
104
|
+
|
|
105
|
+
if (items.length === 0) {
|
|
106
|
+
return result;
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
// Process in batches with concurrency control
|
|
110
|
+
const chunks = chunkArray(items, concurrency);
|
|
111
|
+
let shouldStop = false;
|
|
112
|
+
|
|
113
|
+
for (let chunkIndex = 0; chunkIndex < chunks.length && !shouldStop; chunkIndex++) {
|
|
114
|
+
const chunk = chunks[chunkIndex];
|
|
115
|
+
|
|
116
|
+
await Promise.all(
|
|
117
|
+
chunk.map(async (item) => {
|
|
118
|
+
if (shouldStop) return;
|
|
119
|
+
|
|
120
|
+
try {
|
|
121
|
+
if (dryRun) {
|
|
122
|
+
result.skipped++;
|
|
123
|
+
result.results.push({ item, result: undefined as R });
|
|
124
|
+
} else {
|
|
125
|
+
const opResult = await operation(item);
|
|
126
|
+
result.success++;
|
|
127
|
+
result.results.push({ item, result: opResult });
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
if (onProgress) {
|
|
131
|
+
onProgress(result.success + result.failed + result.skipped, result.total, item);
|
|
132
|
+
}
|
|
133
|
+
} catch (err) {
|
|
134
|
+
result.failed++;
|
|
135
|
+
const errorMessage = err instanceof Error ? err.message : String(err);
|
|
136
|
+
result.errors.push({ item, error: errorMessage });
|
|
137
|
+
|
|
138
|
+
if (onError) {
|
|
139
|
+
const shouldContinue = onError(err instanceof Error ? err : new Error(errorMessage), item);
|
|
140
|
+
if (shouldContinue === false) {
|
|
141
|
+
shouldStop = true;
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
})
|
|
146
|
+
);
|
|
147
|
+
|
|
148
|
+
// Add delay between batches if specified (for rate limiting)
|
|
149
|
+
if (batchDelay > 0 && chunkIndex < chunks.length - 1) {
|
|
150
|
+
await sleep(batchDelay);
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
return result;
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
/**
|
|
158
|
+
* Execute operations sequentially (one at a time)
|
|
159
|
+
* Useful when operations must be processed in order
|
|
160
|
+
*/
|
|
161
|
+
export async function executeSequential<T, R = void>(
|
|
162
|
+
options: Omit<BulkOperationOptions<T>, 'concurrency'>,
|
|
163
|
+
operation: (item: T) => Promise<R>
|
|
164
|
+
): Promise<BulkOperationResult<T, R>> {
|
|
165
|
+
return executeBulk({ ...options, concurrency: 1 }, operation);
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
/**
|
|
169
|
+
* Create a progress reporter for bulk operations
|
|
170
|
+
*/
|
|
171
|
+
export function createProgressReporter(prefix: string = 'Processing') {
|
|
172
|
+
let lastPercent = -1;
|
|
173
|
+
|
|
174
|
+
return (current: number, total: number, _item: unknown): void => {
|
|
175
|
+
const percent = Math.floor((current / total) * 100);
|
|
176
|
+
if (percent !== lastPercent) {
|
|
177
|
+
lastPercent = percent;
|
|
178
|
+
process.stdout.write(`\r${prefix}: ${current}/${total} (${percent}%)`);
|
|
179
|
+
if (current === total) {
|
|
180
|
+
process.stdout.write('\n');
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
};
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
/**
|
|
187
|
+
* Format bulk operation result for display
|
|
188
|
+
*/
|
|
189
|
+
export function formatBulkResult<T>(result: BulkOperationResult<T, unknown>): string {
|
|
190
|
+
const lines: string[] = [
|
|
191
|
+
`Total: ${result.total}`,
|
|
192
|
+
`Success: ${result.success}`,
|
|
193
|
+
`Failed: ${result.failed}`,
|
|
194
|
+
];
|
|
195
|
+
|
|
196
|
+
if (result.skipped > 0) {
|
|
197
|
+
lines.push(`Skipped (dry run): ${result.skipped}`);
|
|
198
|
+
}
|
|
199
|
+
|
|
200
|
+
if (result.errors.length > 0) {
|
|
201
|
+
lines.push('');
|
|
202
|
+
lines.push('Errors:');
|
|
203
|
+
for (const { error } of result.errors.slice(0, 10)) {
|
|
204
|
+
lines.push(` - ${error}`);
|
|
205
|
+
}
|
|
206
|
+
if (result.errors.length > 10) {
|
|
207
|
+
lines.push(` ... and ${result.errors.length - 10} more errors`);
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
return lines.join('\n');
|
|
212
|
+
}
|
|
@@ -0,0 +1,323 @@
|
|
|
1
|
+
import { existsSync, readFileSync, writeFileSync, mkdirSync, readdirSync, rmSync } from 'fs';
|
|
2
|
+
import { homedir } from 'os';
|
|
3
|
+
import { join } from 'path';
|
|
4
|
+
import type { OAuth2Config, OAuth2Tokens } from '../types';
|
|
5
|
+
|
|
6
|
+
const CONNECTOR_NAME = 'connect-ably';
|
|
7
|
+
const DEFAULT_PROFILE = 'default';
|
|
8
|
+
|
|
9
|
+
export interface ProfileConfig {
|
|
10
|
+
// API Key authentication
|
|
11
|
+
apiKey?: string;
|
|
12
|
+
token?: string; // Alias for apiKey
|
|
13
|
+
apiSecret?: string;
|
|
14
|
+
|
|
15
|
+
// OAuth2 authentication
|
|
16
|
+
accessToken?: string;
|
|
17
|
+
refreshToken?: string;
|
|
18
|
+
expiresAt?: number;
|
|
19
|
+
tokenType?: string;
|
|
20
|
+
scope?: string;
|
|
21
|
+
|
|
22
|
+
// OAuth2 client credentials (stored separately for security)
|
|
23
|
+
clientId?: string;
|
|
24
|
+
clientSecret?: string;
|
|
25
|
+
|
|
26
|
+
// Add more config fields as needed for your API
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
// Store for --profile flag override (set by CLI before commands run)
|
|
30
|
+
let profileOverride: string | undefined;
|
|
31
|
+
|
|
32
|
+
// Config directory: ~/.connect/{connector-name}/
|
|
33
|
+
const CONFIG_DIR = join(homedir(), '.connect', CONNECTOR_NAME);
|
|
34
|
+
const PROFILES_DIR = join(CONFIG_DIR, 'profiles');
|
|
35
|
+
const CURRENT_PROFILE_FILE = join(CONFIG_DIR, 'current_profile');
|
|
36
|
+
|
|
37
|
+
// ============================================
|
|
38
|
+
// Profile Management
|
|
39
|
+
// ============================================
|
|
40
|
+
|
|
41
|
+
export function setProfileOverride(profile: string | undefined): void {
|
|
42
|
+
profileOverride = profile;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
export function ensureConfigDir(): void {
|
|
46
|
+
if (!existsSync(CONFIG_DIR)) {
|
|
47
|
+
mkdirSync(CONFIG_DIR, { recursive: true });
|
|
48
|
+
}
|
|
49
|
+
if (!existsSync(PROFILES_DIR)) {
|
|
50
|
+
mkdirSync(PROFILES_DIR, { recursive: true });
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
function getProfilePath(profile: string): string {
|
|
55
|
+
return join(PROFILES_DIR, `${profile}.json`);
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
/**
|
|
59
|
+
* Get the current active profile name
|
|
60
|
+
*/
|
|
61
|
+
export function getCurrentProfile(): string {
|
|
62
|
+
if (profileOverride) {
|
|
63
|
+
return profileOverride;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
ensureConfigDir();
|
|
67
|
+
|
|
68
|
+
if (existsSync(CURRENT_PROFILE_FILE)) {
|
|
69
|
+
try {
|
|
70
|
+
const profile = readFileSync(CURRENT_PROFILE_FILE, 'utf-8').trim();
|
|
71
|
+
if (profile && profileExists(profile)) {
|
|
72
|
+
return profile;
|
|
73
|
+
}
|
|
74
|
+
} catch {
|
|
75
|
+
// Fall through to default
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
return DEFAULT_PROFILE;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
/**
|
|
83
|
+
* Set the current active profile
|
|
84
|
+
*/
|
|
85
|
+
export function setCurrentProfile(profile: string): void {
|
|
86
|
+
ensureConfigDir();
|
|
87
|
+
|
|
88
|
+
if (!profileExists(profile) && profile !== DEFAULT_PROFILE) {
|
|
89
|
+
throw new Error(`Profile "${profile}" does not exist`);
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
writeFileSync(CURRENT_PROFILE_FILE, profile);
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
/**
|
|
96
|
+
* Check if a profile exists
|
|
97
|
+
*/
|
|
98
|
+
export function profileExists(profile: string): boolean {
|
|
99
|
+
return existsSync(getProfilePath(profile));
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
/**
|
|
103
|
+
* List all available profiles
|
|
104
|
+
*/
|
|
105
|
+
export function listProfiles(): string[] {
|
|
106
|
+
ensureConfigDir();
|
|
107
|
+
|
|
108
|
+
if (!existsSync(PROFILES_DIR)) {
|
|
109
|
+
return [];
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
return readdirSync(PROFILES_DIR)
|
|
113
|
+
.filter(f => f.endsWith('.json'))
|
|
114
|
+
.map(f => f.replace('.json', ''))
|
|
115
|
+
.sort();
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
/**
|
|
119
|
+
* Create a new profile
|
|
120
|
+
*/
|
|
121
|
+
export function createProfile(profile: string, config: ProfileConfig = {}): boolean {
|
|
122
|
+
ensureConfigDir();
|
|
123
|
+
|
|
124
|
+
if (profileExists(profile)) {
|
|
125
|
+
return false;
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
// Validate profile name
|
|
129
|
+
if (!/^[a-zA-Z0-9_-]+$/.test(profile)) {
|
|
130
|
+
throw new Error('Profile name can only contain letters, numbers, hyphens, and underscores');
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
writeFileSync(getProfilePath(profile), JSON.stringify(config, null, 2));
|
|
134
|
+
return true;
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
/**
|
|
138
|
+
* Delete a profile
|
|
139
|
+
*/
|
|
140
|
+
export function deleteProfile(profile: string): boolean {
|
|
141
|
+
if (profile === DEFAULT_PROFILE) {
|
|
142
|
+
return false;
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
if (!profileExists(profile)) {
|
|
146
|
+
return false;
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
// Switch to default if deleting current profile
|
|
150
|
+
if (getCurrentProfile() === profile) {
|
|
151
|
+
setCurrentProfile(DEFAULT_PROFILE);
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
rmSync(getProfilePath(profile));
|
|
155
|
+
return true;
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
/**
|
|
159
|
+
* Load profile config
|
|
160
|
+
*/
|
|
161
|
+
export function loadProfile(profile?: string): ProfileConfig {
|
|
162
|
+
ensureConfigDir();
|
|
163
|
+
const profileName = profile || getCurrentProfile();
|
|
164
|
+
const profilePath = getProfilePath(profileName);
|
|
165
|
+
|
|
166
|
+
if (!existsSync(profilePath)) {
|
|
167
|
+
return {};
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
try {
|
|
171
|
+
return JSON.parse(readFileSync(profilePath, 'utf-8'));
|
|
172
|
+
} catch {
|
|
173
|
+
return {};
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
/**
|
|
178
|
+
* Save profile config
|
|
179
|
+
*/
|
|
180
|
+
export function saveProfile(config: ProfileConfig, profile?: string): void {
|
|
181
|
+
ensureConfigDir();
|
|
182
|
+
const profileName = profile || getCurrentProfile();
|
|
183
|
+
writeFileSync(getProfilePath(profileName), JSON.stringify(config, null, 2));
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
// ============================================
|
|
187
|
+
// API Key Management
|
|
188
|
+
// ============================================
|
|
189
|
+
|
|
190
|
+
export function getApiKey(): string | undefined {
|
|
191
|
+
return process.env.ABLY_API_KEY || loadProfile().apiKey;
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
export function setApiKey(apiKey: string): void {
|
|
195
|
+
const config = loadProfile();
|
|
196
|
+
config.apiKey = apiKey;
|
|
197
|
+
saveProfile(config);
|
|
198
|
+
}
|
|
199
|
+
|
|
200
|
+
export function getApiSecret(): string | undefined {
|
|
201
|
+
return process.env.ABLY_API_SECRET || loadProfile().apiSecret;
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
export function setApiSecret(apiSecret: string): void {
|
|
205
|
+
const config = loadProfile();
|
|
206
|
+
config.apiSecret = apiSecret;
|
|
207
|
+
saveProfile(config);
|
|
208
|
+
}
|
|
209
|
+
|
|
210
|
+
// ============================================
|
|
211
|
+
// Utility Functions
|
|
212
|
+
// ============================================
|
|
213
|
+
|
|
214
|
+
export function clearConfig(): void {
|
|
215
|
+
saveProfile({});
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
export function getConfigDir(): string {
|
|
219
|
+
return CONFIG_DIR;
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
export function getActiveProfileName(): string {
|
|
223
|
+
return getCurrentProfile();
|
|
224
|
+
}
|
|
225
|
+
|
|
226
|
+
// ============================================
|
|
227
|
+
// Token/API Key Alias Functions
|
|
228
|
+
// ============================================
|
|
229
|
+
|
|
230
|
+
/**
|
|
231
|
+
* Get token (alias for getApiKey)
|
|
232
|
+
*/
|
|
233
|
+
export function getToken(): string | undefined {
|
|
234
|
+
return process.env.ABLY_TOKEN || process.env.ABLY_API_KEY || loadProfile().token || loadProfile().apiKey;
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
/**
|
|
238
|
+
* Set token (alias for setApiKey)
|
|
239
|
+
*/
|
|
240
|
+
export function setToken(token: string): void {
|
|
241
|
+
const config = loadProfile();
|
|
242
|
+
config.token = token;
|
|
243
|
+
config.apiKey = token; // Keep both in sync
|
|
244
|
+
saveProfile(config);
|
|
245
|
+
}
|
|
246
|
+
|
|
247
|
+
// ============================================
|
|
248
|
+
// OAuth2 Configuration Functions
|
|
249
|
+
// ============================================
|
|
250
|
+
|
|
251
|
+
/**
|
|
252
|
+
* Get OAuth2 client configuration
|
|
253
|
+
*/
|
|
254
|
+
export function getOAuthConfig(): OAuth2Config | null {
|
|
255
|
+
const profile = loadProfile();
|
|
256
|
+
if (profile.clientId && profile.clientSecret) {
|
|
257
|
+
return {
|
|
258
|
+
clientId: profile.clientId,
|
|
259
|
+
clientSecret: profile.clientSecret,
|
|
260
|
+
};
|
|
261
|
+
}
|
|
262
|
+
return null;
|
|
263
|
+
}
|
|
264
|
+
|
|
265
|
+
/**
|
|
266
|
+
* Set OAuth2 client credentials
|
|
267
|
+
*/
|
|
268
|
+
export function setOAuthConfig(config: OAuth2Config): void {
|
|
269
|
+
const profile = loadProfile();
|
|
270
|
+
profile.clientId = config.clientId;
|
|
271
|
+
profile.clientSecret = config.clientSecret;
|
|
272
|
+
saveProfile(profile);
|
|
273
|
+
}
|
|
274
|
+
|
|
275
|
+
/**
|
|
276
|
+
* Load OAuth2 tokens
|
|
277
|
+
*/
|
|
278
|
+
export function loadOAuthTokens(): OAuth2Tokens | null {
|
|
279
|
+
const profile = loadProfile();
|
|
280
|
+
if (profile.accessToken) {
|
|
281
|
+
return {
|
|
282
|
+
accessToken: profile.accessToken,
|
|
283
|
+
refreshToken: profile.refreshToken,
|
|
284
|
+
expiresAt: profile.expiresAt || 0,
|
|
285
|
+
tokenType: profile.tokenType,
|
|
286
|
+
scope: profile.scope,
|
|
287
|
+
};
|
|
288
|
+
}
|
|
289
|
+
return null;
|
|
290
|
+
}
|
|
291
|
+
|
|
292
|
+
/**
|
|
293
|
+
* Save OAuth2 tokens
|
|
294
|
+
*/
|
|
295
|
+
export function saveOAuthTokens(tokens: OAuth2Tokens): void {
|
|
296
|
+
const profile = loadProfile();
|
|
297
|
+
profile.accessToken = tokens.accessToken;
|
|
298
|
+
profile.refreshToken = tokens.refreshToken;
|
|
299
|
+
profile.expiresAt = tokens.expiresAt;
|
|
300
|
+
profile.tokenType = tokens.tokenType;
|
|
301
|
+
profile.scope = tokens.scope;
|
|
302
|
+
saveProfile(profile);
|
|
303
|
+
}
|
|
304
|
+
|
|
305
|
+
/**
|
|
306
|
+
* Clear OAuth2 tokens (logout)
|
|
307
|
+
*/
|
|
308
|
+
export function clearOAuthTokens(): void {
|
|
309
|
+
const profile = loadProfile();
|
|
310
|
+
delete profile.accessToken;
|
|
311
|
+
delete profile.refreshToken;
|
|
312
|
+
delete profile.expiresAt;
|
|
313
|
+
delete profile.tokenType;
|
|
314
|
+
delete profile.scope;
|
|
315
|
+
saveProfile(profile);
|
|
316
|
+
}
|
|
317
|
+
|
|
318
|
+
/**
|
|
319
|
+
* Get the access token (for OAuth2 authentication)
|
|
320
|
+
*/
|
|
321
|
+
export function getAccessToken(): string | undefined {
|
|
322
|
+
return loadProfile().accessToken;
|
|
323
|
+
}
|
|
@@ -0,0 +1,175 @@
|
|
|
1
|
+
import chalk from 'chalk';
|
|
2
|
+
|
|
3
|
+
export type OutputFormat = 'json' | 'table' | 'pretty';
|
|
4
|
+
|
|
5
|
+
// Global verbose mode flag
|
|
6
|
+
let verboseMode = false;
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Enable or disable verbose mode
|
|
10
|
+
*/
|
|
11
|
+
export function setVerboseMode(enabled: boolean): void {
|
|
12
|
+
verboseMode = enabled;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* Check if verbose mode is enabled
|
|
17
|
+
*/
|
|
18
|
+
export function isVerboseMode(): boolean {
|
|
19
|
+
return verboseMode;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
/**
|
|
23
|
+
* Log a debug message (only shown in verbose mode)
|
|
24
|
+
*/
|
|
25
|
+
export function debug(message: string, data?: unknown): void {
|
|
26
|
+
if (!verboseMode) return;
|
|
27
|
+
const timestamp = new Date().toISOString().split('T')[1].slice(0, 12);
|
|
28
|
+
console.log(chalk.gray(`[${timestamp}]`), chalk.dim(message));
|
|
29
|
+
if (data !== undefined) {
|
|
30
|
+
if (typeof data === 'object') {
|
|
31
|
+
console.log(chalk.gray(JSON.stringify(data, null, 2)));
|
|
32
|
+
} else {
|
|
33
|
+
console.log(chalk.gray(String(data)));
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* Log a request (only shown in verbose mode)
|
|
40
|
+
*/
|
|
41
|
+
export function debugRequest(method: string, url: string, body?: unknown): void {
|
|
42
|
+
if (!verboseMode) return;
|
|
43
|
+
debug(`→ ${method} ${url}`);
|
|
44
|
+
if (body) {
|
|
45
|
+
debug(' Body:', body);
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
/**
|
|
50
|
+
* Log a response (only shown in verbose mode)
|
|
51
|
+
*/
|
|
52
|
+
export function debugResponse(status: number, duration: number, body?: unknown): void {
|
|
53
|
+
if (!verboseMode) return;
|
|
54
|
+
const statusColor = status >= 400 ? chalk.red : status >= 300 ? chalk.yellow : chalk.green;
|
|
55
|
+
debug(`← ${statusColor(status)} (${duration}ms)`);
|
|
56
|
+
if (body && verboseMode) {
|
|
57
|
+
debug(' Response:', body);
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
export function formatOutput(data: unknown, format: OutputFormat = 'pretty'): string {
|
|
62
|
+
switch (format) {
|
|
63
|
+
case 'json':
|
|
64
|
+
return JSON.stringify(data, null, 2);
|
|
65
|
+
case 'table':
|
|
66
|
+
return formatAsTable(data);
|
|
67
|
+
case 'pretty':
|
|
68
|
+
default:
|
|
69
|
+
return formatPretty(data);
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
function formatAsTable(data: unknown): string {
|
|
74
|
+
if (!Array.isArray(data)) {
|
|
75
|
+
data = [data];
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
const items = data as Record<string, unknown>[];
|
|
79
|
+
if (items.length === 0) {
|
|
80
|
+
return 'No data';
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
const firstItem = items[0];
|
|
84
|
+
if (!firstItem || typeof firstItem !== 'object') {
|
|
85
|
+
return 'No data';
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
const keys = Object.keys(firstItem);
|
|
89
|
+
const colWidths = keys.map(key => {
|
|
90
|
+
const maxValue = Math.max(
|
|
91
|
+
key.length,
|
|
92
|
+
...items.map(item => String(item[key] ?? '').length)
|
|
93
|
+
);
|
|
94
|
+
return Math.min(maxValue, 40);
|
|
95
|
+
});
|
|
96
|
+
|
|
97
|
+
const header = keys.map((key, i) => key.padEnd(colWidths[i] ?? 10)).join(' | ');
|
|
98
|
+
const separator = colWidths.map(w => '-'.repeat(w)).join('-+-');
|
|
99
|
+
|
|
100
|
+
const rows = items.map(item =>
|
|
101
|
+
keys.map((key, i) => {
|
|
102
|
+
const value = String(item[key] ?? '');
|
|
103
|
+
const width = colWidths[i] ?? 10;
|
|
104
|
+
return value.length > width
|
|
105
|
+
? value.substring(0, width - 3) + '...'
|
|
106
|
+
: value.padEnd(width);
|
|
107
|
+
}).join(' | ')
|
|
108
|
+
);
|
|
109
|
+
|
|
110
|
+
return [header, separator, ...rows].join('\n');
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
function formatPretty(data: unknown): string {
|
|
114
|
+
if (Array.isArray(data)) {
|
|
115
|
+
return data.map((item, i) => `${chalk.cyan(`[${i + 1}]`)} ${formatPrettyItem(item)}`).join('\n\n');
|
|
116
|
+
}
|
|
117
|
+
return formatPrettyItem(data);
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
function formatPrettyItem(item: unknown, indent = 0): string {
|
|
121
|
+
if (item === null || item === undefined) {
|
|
122
|
+
return chalk.gray('null');
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
if (typeof item !== 'object') {
|
|
126
|
+
return String(item);
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
const spaces = ' '.repeat(indent);
|
|
130
|
+
const entries = Object.entries(item as Record<string, unknown>);
|
|
131
|
+
|
|
132
|
+
return entries
|
|
133
|
+
.map(([key, value]) => {
|
|
134
|
+
if (Array.isArray(value)) {
|
|
135
|
+
if (value.length === 0) {
|
|
136
|
+
return `${spaces}${chalk.blue(key)}: ${chalk.gray('[]')}`;
|
|
137
|
+
}
|
|
138
|
+
if (typeof value[0] === 'object') {
|
|
139
|
+
return `${spaces}${chalk.blue(key)}:\n${value.map(v => formatPrettyItem(v, indent + 1)).join('\n')}`;
|
|
140
|
+
}
|
|
141
|
+
return `${spaces}${chalk.blue(key)}: ${value.join(', ')}`;
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
if (typeof value === 'object' && value !== null) {
|
|
145
|
+
return `${spaces}${chalk.blue(key)}:\n${formatPrettyItem(value, indent + 1)}`;
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
return `${spaces}${chalk.blue(key)}: ${chalk.white(String(value))}`;
|
|
149
|
+
})
|
|
150
|
+
.join('\n');
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
export function success(message: string): void {
|
|
154
|
+
console.log(chalk.green('✓'), message);
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
export function error(message: string): void {
|
|
158
|
+
console.error(chalk.red('✗'), message);
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
export function warn(message: string): void {
|
|
162
|
+
console.warn(chalk.yellow('⚠'), message);
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
export function info(message: string): void {
|
|
166
|
+
console.log(chalk.blue('ℹ'), message);
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
export function heading(message: string): void {
|
|
170
|
+
console.log(chalk.bold.cyan(`\n${message}\n`));
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
export function print(data: unknown, format: OutputFormat = 'pretty'): void {
|
|
174
|
+
console.log(formatOutput(data, format));
|
|
175
|
+
}
|