@hasna/connectors 0.4.2 → 0.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/bin/index.js +113 -1
- package/bin/mcp.js +113 -1
- package/bin/serve.js +112 -0
- package/connectors/connect-assemblyai/.env.example +11 -0
- package/connectors/connect-assemblyai/CLAUDE.md +128 -0
- package/connectors/connect-assemblyai/README.md +193 -0
- package/connectors/connect-assemblyai/package.json +50 -0
- package/connectors/connect-assemblyai/src/api/client.ts +192 -0
- package/connectors/connect-assemblyai/src/api/index.ts +71 -0
- package/connectors/connect-assemblyai/src/cli/index.ts +384 -0
- package/connectors/connect-assemblyai/src/index.ts +19 -0
- package/connectors/connect-assemblyai/src/types/index.ts +277 -0
- package/connectors/connect-assemblyai/src/utils/config.ts +103 -0
- package/connectors/connect-assemblyai/src/utils/output.ts +119 -0
- package/connectors/connect-assemblyai/tsconfig.json +16 -0
- package/connectors/connect-baseten/.env.example +11 -0
- package/connectors/connect-baseten/CLAUDE.md +128 -0
- package/connectors/connect-baseten/README.md +193 -0
- package/connectors/connect-baseten/package.json +51 -0
- package/connectors/connect-baseten/src/api/client.ts +71 -0
- package/connectors/connect-baseten/src/api/index.ts +40 -0
- package/connectors/connect-baseten/src/cli/index.ts +244 -0
- package/connectors/connect-baseten/src/index.ts +19 -0
- package/connectors/connect-baseten/src/types/index.ts +55 -0
- package/connectors/connect-baseten/src/utils/config.ts +103 -0
- package/connectors/connect-baseten/src/utils/output.ts +119 -0
- package/connectors/connect-baseten/tsconfig.json +16 -0
- package/connectors/connect-cerebras/.env.example +11 -0
- package/connectors/connect-cerebras/CLAUDE.md +128 -0
- package/connectors/connect-cerebras/README.md +193 -0
- package/connectors/connect-cerebras/package.json +51 -0
- package/connectors/connect-cerebras/src/api/client.ts +64 -0
- package/connectors/connect-cerebras/src/api/index.ts +32 -0
- package/connectors/connect-cerebras/src/cli/index.ts +244 -0
- package/connectors/connect-cerebras/src/index.ts +19 -0
- package/connectors/connect-cerebras/src/types/index.ts +65 -0
- package/connectors/connect-cerebras/src/utils/config.ts +103 -0
- package/connectors/connect-cerebras/src/utils/output.ts +119 -0
- package/connectors/connect-cerebras/tsconfig.json +16 -0
- package/connectors/connect-cohere/.env.example +11 -0
- package/connectors/connect-cohere/CLAUDE.md +128 -0
- package/connectors/connect-cohere/README.md +193 -0
- package/connectors/connect-cohere/package.json +53 -0
- package/connectors/connect-cohere/src/api/client.ts +109 -0
- package/connectors/connect-cohere/src/api/index.ts +59 -0
- package/connectors/connect-cohere/src/cli/index.ts +255 -0
- package/connectors/connect-cohere/src/index.ts +19 -0
- package/connectors/connect-cohere/src/types/index.ts +132 -0
- package/connectors/connect-cohere/src/utils/config.ts +197 -0
- package/connectors/connect-cohere/src/utils/output.ts +119 -0
- package/connectors/connect-cohere/tsconfig.json +16 -0
- package/connectors/connect-deepgram/.env.example +11 -0
- package/connectors/connect-deepgram/CLAUDE.md +128 -0
- package/connectors/connect-deepgram/README.md +193 -0
- package/connectors/connect-deepgram/package.json +51 -0
- package/connectors/connect-deepgram/src/api/client.ts +235 -0
- package/connectors/connect-deepgram/src/api/index.ts +57 -0
- package/connectors/connect-deepgram/src/cli/index.ts +339 -0
- package/connectors/connect-deepgram/src/index.ts +19 -0
- package/connectors/connect-deepgram/src/types/index.ts +232 -0
- package/connectors/connect-deepgram/src/utils/config.ts +103 -0
- package/connectors/connect-deepgram/src/utils/output.ts +119 -0
- package/connectors/connect-deepgram/tsconfig.json +16 -0
- package/connectors/connect-deepseek/.env.example +11 -0
- package/connectors/connect-deepseek/CLAUDE.md +128 -0
- package/connectors/connect-deepseek/README.md +193 -0
- package/connectors/connect-deepseek/package.json +51 -0
- package/connectors/connect-deepseek/src/api/client.ts +108 -0
- package/connectors/connect-deepseek/src/api/index.ts +36 -0
- package/connectors/connect-deepseek/src/cli/index.ts +167 -0
- package/connectors/connect-deepseek/src/index.ts +19 -0
- package/connectors/connect-deepseek/src/types/index.ts +72 -0
- package/connectors/connect-deepseek/src/utils/config.ts +103 -0
- package/connectors/connect-deepseek/src/utils/output.ts +119 -0
- package/connectors/connect-deepseek/tsconfig.json +16 -0
- package/connectors/connect-fal/.env.example +11 -0
- package/connectors/connect-fal/CLAUDE.md +128 -0
- package/connectors/connect-fal/README.md +193 -0
- package/connectors/connect-fal/package.json +51 -0
- package/connectors/connect-fal/src/api/client.ts +172 -0
- package/connectors/connect-fal/src/api/index.ts +55 -0
- package/connectors/connect-fal/src/cli/index.ts +341 -0
- package/connectors/connect-fal/src/index.ts +19 -0
- package/connectors/connect-fal/src/types/index.ts +135 -0
- package/connectors/connect-fal/src/utils/config.ts +103 -0
- package/connectors/connect-fal/src/utils/output.ts +119 -0
- package/connectors/connect-fal/tsconfig.json +16 -0
- package/connectors/connect-fireworks/.env.example +11 -0
- package/connectors/connect-fireworks/CLAUDE.md +128 -0
- package/connectors/connect-fireworks/README.md +193 -0
- package/connectors/connect-fireworks/package.json +51 -0
- package/connectors/connect-fireworks/src/api/client.ts +63 -0
- package/connectors/connect-fireworks/src/api/index.ts +36 -0
- package/connectors/connect-fireworks/src/cli/index.ts +244 -0
- package/connectors/connect-fireworks/src/index.ts +19 -0
- package/connectors/connect-fireworks/src/types/index.ts +70 -0
- package/connectors/connect-fireworks/src/utils/config.ts +103 -0
- package/connectors/connect-fireworks/src/utils/output.ts +119 -0
- package/connectors/connect-fireworks/tsconfig.json +16 -0
- package/connectors/connect-groq/.env.example +11 -0
- package/connectors/connect-groq/CLAUDE.md +128 -0
- package/connectors/connect-groq/README.md +193 -0
- package/connectors/connect-groq/package.json +52 -0
- package/connectors/connect-groq/src/api/client.ts +108 -0
- package/connectors/connect-groq/src/api/index.ts +36 -0
- package/connectors/connect-groq/src/cli/index.ts +171 -0
- package/connectors/connect-groq/src/index.ts +19 -0
- package/connectors/connect-groq/src/types/index.ts +69 -0
- package/connectors/connect-groq/src/utils/config.ts +103 -0
- package/connectors/connect-groq/src/utils/output.ts +119 -0
- package/connectors/connect-groq/tsconfig.json +16 -0
- package/connectors/connect-luma/.env.example +11 -0
- package/connectors/connect-luma/CLAUDE.md +128 -0
- package/connectors/connect-luma/README.md +193 -0
- package/connectors/connect-luma/package.json +53 -0
- package/connectors/connect-luma/src/api/client.ts +85 -0
- package/connectors/connect-luma/src/api/index.ts +44 -0
- package/connectors/connect-luma/src/cli/index.ts +300 -0
- package/connectors/connect-luma/src/index.ts +19 -0
- package/connectors/connect-luma/src/types/index.ts +60 -0
- package/connectors/connect-luma/src/utils/config.ts +103 -0
- package/connectors/connect-luma/src/utils/output.ts +119 -0
- package/connectors/connect-luma/tsconfig.json +16 -0
- package/connectors/connect-modal/.env.example +11 -0
- package/connectors/connect-modal/CLAUDE.md +128 -0
- package/connectors/connect-modal/README.md +193 -0
- package/connectors/connect-modal/package.json +51 -0
- package/connectors/connect-modal/src/api/client.ts +119 -0
- package/connectors/connect-modal/src/api/index.ts +69 -0
- package/connectors/connect-modal/src/cli/index.ts +224 -0
- package/connectors/connect-modal/src/index.ts +21 -0
- package/connectors/connect-modal/src/types/index.ts +60 -0
- package/connectors/connect-modal/src/utils/config.ts +114 -0
- package/connectors/connect-modal/src/utils/output.ts +119 -0
- package/connectors/connect-modal/tsconfig.json +16 -0
- package/connectors/connect-perplexity/.env.example +4 -0
- package/connectors/connect-perplexity/CLAUDE.md +156 -0
- package/connectors/connect-perplexity/README.md +184 -0
- package/connectors/connect-perplexity/package.json +58 -0
- package/connectors/connect-perplexity/scripts/publish.ts +210 -0
- package/connectors/connect-perplexity/src/api/client.ts +119 -0
- package/connectors/connect-perplexity/src/api/example.ts +118 -0
- package/connectors/connect-perplexity/src/api/index.ts +48 -0
- package/connectors/connect-perplexity/src/cli/index.ts +421 -0
- package/connectors/connect-perplexity/src/index.ts +24 -0
- package/connectors/connect-perplexity/src/types/index.ts +140 -0
- package/connectors/connect-perplexity/src/utils/config.ts +208 -0
- package/connectors/connect-perplexity/src/utils/output.ts +119 -0
- package/connectors/connect-perplexity/tsconfig.json +16 -0
- package/connectors/connect-replicate/.env.example +11 -0
- package/connectors/connect-replicate/CLAUDE.md +128 -0
- package/connectors/connect-replicate/README.md +193 -0
- package/connectors/connect-replicate/package.json +51 -0
- package/connectors/connect-replicate/src/api/client.ts +109 -0
- package/connectors/connect-replicate/src/api/index.ts +71 -0
- package/connectors/connect-replicate/src/cli/index.ts +250 -0
- package/connectors/connect-replicate/src/index.ts +19 -0
- package/connectors/connect-replicate/src/types/index.ts +85 -0
- package/connectors/connect-replicate/src/utils/config.ts +103 -0
- package/connectors/connect-replicate/src/utils/output.ts +119 -0
- package/connectors/connect-replicate/tsconfig.json +16 -0
- package/connectors/connect-roboflow/.env.example +11 -0
- package/connectors/connect-roboflow/CLAUDE.md +272 -0
- package/connectors/connect-roboflow/README.md +193 -0
- package/connectors/connect-roboflow/package.json +51 -0
- package/connectors/connect-roboflow/scripts/release.ts +179 -0
- package/connectors/connect-roboflow/src/api/client.ts +213 -0
- package/connectors/connect-roboflow/src/api/example.ts +48 -0
- package/connectors/connect-roboflow/src/api/index.ts +51 -0
- package/connectors/connect-roboflow/src/cli/index.ts +254 -0
- package/connectors/connect-roboflow/src/index.ts +103 -0
- package/connectors/connect-roboflow/src/types/index.ts +237 -0
- package/connectors/connect-roboflow/src/utils/auth.ts +274 -0
- package/connectors/connect-roboflow/src/utils/bulk.ts +212 -0
- package/connectors/connect-roboflow/src/utils/config.ts +326 -0
- package/connectors/connect-roboflow/src/utils/output.ts +175 -0
- package/connectors/connect-roboflow/src/utils/settings.ts +114 -0
- package/connectors/connect-roboflow/src/utils/storage.ts +198 -0
- package/connectors/connect-roboflow/tsconfig.json +16 -0
- package/connectors/connect-runway/.env.example +11 -0
- package/connectors/connect-runway/CLAUDE.md +128 -0
- package/connectors/connect-runway/README.md +193 -0
- package/connectors/connect-runway/package.json +52 -0
- package/connectors/connect-runway/src/api/client.ts +78 -0
- package/connectors/connect-runway/src/api/index.ts +40 -0
- package/connectors/connect-runway/src/cli/index.ts +283 -0
- package/connectors/connect-runway/src/index.ts +19 -0
- package/connectors/connect-runway/src/types/index.ts +52 -0
- package/connectors/connect-runway/src/utils/config.ts +103 -0
- package/connectors/connect-runway/src/utils/output.ts +119 -0
- package/connectors/connect-runway/tsconfig.json +16 -0
- package/connectors/connect-together/.env.example +11 -0
- package/connectors/connect-together/CLAUDE.md +128 -0
- package/connectors/connect-together/README.md +193 -0
- package/connectors/connect-together/package.json +52 -0
- package/connectors/connect-together/src/api/client.ts +106 -0
- package/connectors/connect-together/src/api/index.ts +47 -0
- package/connectors/connect-together/src/cli/index.ts +228 -0
- package/connectors/connect-together/src/index.ts +19 -0
- package/connectors/connect-together/src/types/index.ts +91 -0
- package/connectors/connect-together/src/utils/config.ts +142 -0
- package/connectors/connect-together/src/utils/output.ts +119 -0
- package/connectors/connect-together/tsconfig.json +16 -0
- package/dist/index.js +112 -0
- package/package.json +1 -1
|
@@ -0,0 +1,172 @@
|
|
|
1
|
+
import type {
|
|
2
|
+
FalConfig,
|
|
3
|
+
ImageGenerateRequest,
|
|
4
|
+
ImageGenerateResponse,
|
|
5
|
+
QueueSubmitResponse,
|
|
6
|
+
QueueStatusResponse,
|
|
7
|
+
QueueResultResponse,
|
|
8
|
+
RunOptions,
|
|
9
|
+
} from '../types';
|
|
10
|
+
import { FalApiError, COMMON_MODELS, type CommonModelAlias } from '../types';
|
|
11
|
+
|
|
12
|
+
const DEFAULT_BASE_URL = 'https://fal.run';
|
|
13
|
+
const QUEUE_BASE_URL = 'https://queue.fal.run';
|
|
14
|
+
|
|
15
|
+
export interface RequestOptions {
|
|
16
|
+
method?: 'GET' | 'POST' | 'PUT' | 'DELETE' | 'PATCH';
|
|
17
|
+
params?: Record<string, string | number | boolean | undefined>;
|
|
18
|
+
body?: Record<string, unknown> | unknown[] | string;
|
|
19
|
+
headers?: Record<string, string>;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
export class FalClient {
|
|
23
|
+
private readonly apiKey: string;
|
|
24
|
+
private readonly baseUrl: string;
|
|
25
|
+
private readonly queueUrl: string;
|
|
26
|
+
|
|
27
|
+
constructor(config: FalConfig) {
|
|
28
|
+
if (!config.apiKey) {
|
|
29
|
+
throw new Error('API key is required');
|
|
30
|
+
}
|
|
31
|
+
this.apiKey = config.apiKey;
|
|
32
|
+
this.baseUrl = config.baseUrl || DEFAULT_BASE_URL;
|
|
33
|
+
this.queueUrl = QUEUE_BASE_URL;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
private resolveModel(modelOrAlias: string): string {
|
|
37
|
+
if (modelOrAlias in COMMON_MODELS) {
|
|
38
|
+
return COMMON_MODELS[modelOrAlias as CommonModelAlias];
|
|
39
|
+
}
|
|
40
|
+
return modelOrAlias;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
private buildUrl(baseUrl: string, path: string, params?: Record<string, string | number | boolean | undefined>): string {
|
|
44
|
+
const url = new URL(`${baseUrl}${path}`);
|
|
45
|
+
|
|
46
|
+
if (params) {
|
|
47
|
+
Object.entries(params).forEach(([key, value]) => {
|
|
48
|
+
if (value !== undefined && value !== null && value !== '') {
|
|
49
|
+
url.searchParams.append(key, String(value));
|
|
50
|
+
}
|
|
51
|
+
});
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
return url.toString();
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
async request<T>(baseUrl: string, path: string, options: RequestOptions = {}): Promise<T> {
|
|
58
|
+
const { method = 'GET', params, body, headers = {} } = options;
|
|
59
|
+
|
|
60
|
+
const url = this.buildUrl(baseUrl, path, params);
|
|
61
|
+
|
|
62
|
+
const requestHeaders: Record<string, string> = {
|
|
63
|
+
'Authorization': `Key ${this.apiKey}`,
|
|
64
|
+
'Accept': 'application/json',
|
|
65
|
+
...headers,
|
|
66
|
+
};
|
|
67
|
+
|
|
68
|
+
if (body && ['POST', 'PUT', 'PATCH'].includes(method)) {
|
|
69
|
+
requestHeaders['Content-Type'] = 'application/json';
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
const fetchOptions: RequestInit = {
|
|
73
|
+
method,
|
|
74
|
+
headers: requestHeaders,
|
|
75
|
+
};
|
|
76
|
+
|
|
77
|
+
if (body && ['POST', 'PUT', 'PATCH'].includes(method)) {
|
|
78
|
+
fetchOptions.body = typeof body === 'string' ? body : JSON.stringify(body);
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
const response = await fetch(url, fetchOptions);
|
|
82
|
+
|
|
83
|
+
if (response.status === 204) {
|
|
84
|
+
return {} as T;
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
let data: unknown;
|
|
88
|
+
const contentType = response.headers.get('content-type') || '';
|
|
89
|
+
|
|
90
|
+
if (contentType.includes('application/json')) {
|
|
91
|
+
const text = await response.text();
|
|
92
|
+
if (text) {
|
|
93
|
+
try {
|
|
94
|
+
data = JSON.parse(text);
|
|
95
|
+
} catch {
|
|
96
|
+
data = text;
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
} else {
|
|
100
|
+
data = await response.text();
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
if (!response.ok) {
|
|
104
|
+
const errorMessage = typeof data === 'object' && data !== null
|
|
105
|
+
? JSON.stringify(data)
|
|
106
|
+
: String(data || response.statusText);
|
|
107
|
+
throw new FalApiError(errorMessage, response.status);
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
return data as T;
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
// ============================================
|
|
114
|
+
// Synchronous Run Methods
|
|
115
|
+
// ============================================
|
|
116
|
+
|
|
117
|
+
async run<T = ImageGenerateResponse>(model: string, input: Record<string, unknown>): Promise<T> {
|
|
118
|
+
const resolvedModel = this.resolveModel(model);
|
|
119
|
+
return this.request<T>(this.baseUrl, `/${resolvedModel}`, {
|
|
120
|
+
method: 'POST',
|
|
121
|
+
body: input,
|
|
122
|
+
});
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
async generateImage(model: string, params: ImageGenerateRequest): Promise<ImageGenerateResponse> {
|
|
126
|
+
return this.run<ImageGenerateResponse>(model, params as Record<string, unknown>);
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
// ============================================
|
|
130
|
+
// Queue Methods
|
|
131
|
+
// ============================================
|
|
132
|
+
|
|
133
|
+
async submit(model: string, input: Record<string, unknown>, webhookUrl?: string): Promise<QueueSubmitResponse> {
|
|
134
|
+
const resolvedModel = this.resolveModel(model);
|
|
135
|
+
const body: Record<string, unknown> = { ...input };
|
|
136
|
+
if (webhookUrl) {
|
|
137
|
+
body.webhook_url = webhookUrl;
|
|
138
|
+
}
|
|
139
|
+
return this.request<QueueSubmitResponse>(this.queueUrl, `/${resolvedModel}`, {
|
|
140
|
+
method: 'POST',
|
|
141
|
+
body,
|
|
142
|
+
});
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
async status(model: string, requestId: string): Promise<QueueStatusResponse> {
|
|
146
|
+
const resolvedModel = this.resolveModel(model);
|
|
147
|
+
return this.request<QueueStatusResponse>(this.queueUrl, `/${resolvedModel}/requests/${requestId}/status`, {
|
|
148
|
+
method: 'GET',
|
|
149
|
+
});
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
async result<T = ImageGenerateResponse>(model: string, requestId: string): Promise<QueueResultResponse<T>> {
|
|
153
|
+
const resolvedModel = this.resolveModel(model);
|
|
154
|
+
return this.request<QueueResultResponse<T>>(this.queueUrl, `/${resolvedModel}/requests/${requestId}`, {
|
|
155
|
+
method: 'GET',
|
|
156
|
+
});
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
async cancel(model: string, requestId: string): Promise<void> {
|
|
160
|
+
const resolvedModel = this.resolveModel(model);
|
|
161
|
+
await this.request<void>(this.queueUrl, `/${resolvedModel}/requests/${requestId}/cancel`, {
|
|
162
|
+
method: 'PUT',
|
|
163
|
+
});
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
getApiKeyPreview(): string {
|
|
167
|
+
if (this.apiKey.length > 10) {
|
|
168
|
+
return `${this.apiKey.substring(0, 6)}...${this.apiKey.substring(this.apiKey.length - 4)}`;
|
|
169
|
+
}
|
|
170
|
+
return '***';
|
|
171
|
+
}
|
|
172
|
+
}
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
import type {
|
|
2
|
+
FalConfig,
|
|
3
|
+
ImageGenerateRequest,
|
|
4
|
+
ImageGenerateResponse,
|
|
5
|
+
QueueSubmitResponse,
|
|
6
|
+
QueueStatusResponse,
|
|
7
|
+
QueueResultResponse,
|
|
8
|
+
} from '../types';
|
|
9
|
+
import { FalClient } from './client';
|
|
10
|
+
|
|
11
|
+
export class Fal {
|
|
12
|
+
private readonly client: FalClient;
|
|
13
|
+
|
|
14
|
+
constructor(config: FalConfig) {
|
|
15
|
+
this.client = new FalClient(config);
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
async run<T = ImageGenerateResponse>(model: string, input: Record<string, unknown>): Promise<T> {
|
|
19
|
+
return this.client.run<T>(model, input);
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
async generateImage(model: string, params: ImageGenerateRequest): Promise<ImageGenerateResponse> {
|
|
23
|
+
return this.client.generateImage(model, params);
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
async submit(model: string, input: Record<string, unknown>, webhookUrl?: string): Promise<QueueSubmitResponse> {
|
|
27
|
+
return this.client.submit(model, input, webhookUrl);
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
async status(model: string, requestId: string): Promise<QueueStatusResponse> {
|
|
31
|
+
return this.client.status(model, requestId);
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
async result<T = ImageGenerateResponse>(model: string, requestId: string): Promise<QueueResultResponse<T>> {
|
|
35
|
+
return this.client.result<T>(model, requestId);
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
async cancel(model: string, requestId: string): Promise<void> {
|
|
39
|
+
return this.client.cancel(model, requestId);
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
static fromEnv(): Fal {
|
|
43
|
+
const apiKey = process.env.FAL_API_KEY;
|
|
44
|
+
if (!apiKey) {
|
|
45
|
+
throw new Error('FAL_API_KEY environment variable is required');
|
|
46
|
+
}
|
|
47
|
+
return new Fal({ apiKey });
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
getApiKeyPreview(): string {
|
|
51
|
+
return this.client.getApiKeyPreview();
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
export { FalClient } from './client';
|
|
@@ -0,0 +1,341 @@
|
|
|
1
|
+
#!/usr/bin/env bun
|
|
2
|
+
import { Command } from 'commander';
|
|
3
|
+
import chalk from 'chalk';
|
|
4
|
+
import { Fal } from '../api';
|
|
5
|
+
import { COMMON_MODELS } from '../types';
|
|
6
|
+
import {
|
|
7
|
+
getApiKey,
|
|
8
|
+
setApiKey,
|
|
9
|
+
clearConfig,
|
|
10
|
+
getConfigDir,
|
|
11
|
+
setProfileOverride,
|
|
12
|
+
getCurrentProfile,
|
|
13
|
+
setCurrentProfile,
|
|
14
|
+
listProfiles,
|
|
15
|
+
createProfile,
|
|
16
|
+
deleteProfile,
|
|
17
|
+
profileExists,
|
|
18
|
+
loadProfile,
|
|
19
|
+
} from '../utils/config';
|
|
20
|
+
import type { OutputFormat } from '../utils/output';
|
|
21
|
+
import { success, error, info, print } from '../utils/output';
|
|
22
|
+
|
|
23
|
+
const CONNECTOR_NAME = 'connect-fal';
|
|
24
|
+
const VERSION = '0.0.1';
|
|
25
|
+
|
|
26
|
+
const program = new Command();
|
|
27
|
+
|
|
28
|
+
program
|
|
29
|
+
.name(CONNECTOR_NAME)
|
|
30
|
+
.description('fal.ai connector CLI - Serverless AI inference with multi-profile support')
|
|
31
|
+
.version(VERSION)
|
|
32
|
+
.option('-k, --api-key <key>', 'API key (overrides config)')
|
|
33
|
+
.option('-f, --format <format>', 'Output format (json, pretty)', 'pretty')
|
|
34
|
+
.option('-p, --profile <profile>', 'Use a specific profile')
|
|
35
|
+
.hook('preAction', (thisCommand) => {
|
|
36
|
+
const opts = thisCommand.opts();
|
|
37
|
+
if (opts.profile) {
|
|
38
|
+
if (!profileExists(opts.profile)) {
|
|
39
|
+
error(`Profile "${opts.profile}" does not exist. Create it with "${CONNECTOR_NAME} profile create ${opts.profile}"`);
|
|
40
|
+
process.exit(1);
|
|
41
|
+
}
|
|
42
|
+
setProfileOverride(opts.profile);
|
|
43
|
+
}
|
|
44
|
+
if (opts.apiKey) {
|
|
45
|
+
process.env.FAL_API_KEY = opts.apiKey;
|
|
46
|
+
}
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
function getFormat(cmd: Command): OutputFormat {
|
|
50
|
+
const parent = cmd.parent;
|
|
51
|
+
return (parent?.opts().format || 'pretty') as OutputFormat;
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
function getClient(): Fal {
|
|
55
|
+
const apiKey = getApiKey();
|
|
56
|
+
if (!apiKey) {
|
|
57
|
+
error(`No API key configured. Run "${CONNECTOR_NAME} config set-key <key>" or set FAL_API_KEY environment variable.`);
|
|
58
|
+
process.exit(1);
|
|
59
|
+
}
|
|
60
|
+
return new Fal({ apiKey });
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
// ============================================
|
|
64
|
+
// Profile Commands
|
|
65
|
+
// ============================================
|
|
66
|
+
const profileCmd = program
|
|
67
|
+
.command('profile')
|
|
68
|
+
.description('Manage configuration profiles');
|
|
69
|
+
|
|
70
|
+
profileCmd
|
|
71
|
+
.command('list')
|
|
72
|
+
.description('List all profiles')
|
|
73
|
+
.action(() => {
|
|
74
|
+
const profiles = listProfiles();
|
|
75
|
+
const current = getCurrentProfile();
|
|
76
|
+
if (profiles.length === 0) {
|
|
77
|
+
info('No profiles found. Use "profile create <name>" to create one.');
|
|
78
|
+
return;
|
|
79
|
+
}
|
|
80
|
+
success(`Profiles:`);
|
|
81
|
+
profiles.forEach(p => {
|
|
82
|
+
const isActive = p === current ? chalk.green(' (active)') : '';
|
|
83
|
+
console.log(` ${p}${isActive}`);
|
|
84
|
+
});
|
|
85
|
+
});
|
|
86
|
+
|
|
87
|
+
profileCmd
|
|
88
|
+
.command('use <name>')
|
|
89
|
+
.description('Switch to a profile')
|
|
90
|
+
.action((name: string) => {
|
|
91
|
+
if (!profileExists(name)) {
|
|
92
|
+
error(`Profile "${name}" does not exist. Create it with "profile create ${name}"`);
|
|
93
|
+
process.exit(1);
|
|
94
|
+
}
|
|
95
|
+
setCurrentProfile(name);
|
|
96
|
+
success(`Switched to profile: ${name}`);
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
profileCmd
|
|
100
|
+
.command('create <name>')
|
|
101
|
+
.description('Create a new profile')
|
|
102
|
+
.option('--api-key <key>', 'API key')
|
|
103
|
+
.option('--use', 'Switch to this profile after creation')
|
|
104
|
+
.action((name: string, opts) => {
|
|
105
|
+
if (profileExists(name)) {
|
|
106
|
+
error(`Profile "${name}" already exists`);
|
|
107
|
+
process.exit(1);
|
|
108
|
+
}
|
|
109
|
+
createProfile(name, { apiKey: opts.apiKey });
|
|
110
|
+
success(`Profile "${name}" created`);
|
|
111
|
+
if (opts.use) {
|
|
112
|
+
setCurrentProfile(name);
|
|
113
|
+
info(`Switched to profile: ${name}`);
|
|
114
|
+
}
|
|
115
|
+
});
|
|
116
|
+
|
|
117
|
+
profileCmd
|
|
118
|
+
.command('delete <name>')
|
|
119
|
+
.description('Delete a profile')
|
|
120
|
+
.action((name: string) => {
|
|
121
|
+
if (name === 'default') {
|
|
122
|
+
error('Cannot delete the default profile');
|
|
123
|
+
process.exit(1);
|
|
124
|
+
}
|
|
125
|
+
if (deleteProfile(name)) {
|
|
126
|
+
success(`Profile "${name}" deleted`);
|
|
127
|
+
} else {
|
|
128
|
+
error(`Profile "${name}" not found`);
|
|
129
|
+
process.exit(1);
|
|
130
|
+
}
|
|
131
|
+
});
|
|
132
|
+
|
|
133
|
+
profileCmd
|
|
134
|
+
.command('show [name]')
|
|
135
|
+
.description('Show profile configuration')
|
|
136
|
+
.action((name?: string) => {
|
|
137
|
+
const profileName = name || getCurrentProfile();
|
|
138
|
+
const config = loadProfile(profileName);
|
|
139
|
+
const active = getCurrentProfile();
|
|
140
|
+
console.log(chalk.bold(`Profile: ${profileName}${profileName === active ? chalk.green(' (active)') : ''}`));
|
|
141
|
+
info(`API Key: ${config.apiKey ? `${config.apiKey.substring(0, 8)}...` : chalk.gray('not set')}`);
|
|
142
|
+
});
|
|
143
|
+
|
|
144
|
+
// ============================================
|
|
145
|
+
// Config Commands
|
|
146
|
+
// ============================================
|
|
147
|
+
const configCmd = program
|
|
148
|
+
.command('config')
|
|
149
|
+
.description('Manage CLI configuration (for active profile)');
|
|
150
|
+
|
|
151
|
+
configCmd
|
|
152
|
+
.command('set-key <apiKey>')
|
|
153
|
+
.description('Set API key')
|
|
154
|
+
.action((apiKey: string) => {
|
|
155
|
+
setApiKey(apiKey);
|
|
156
|
+
success(`API key saved to profile: ${getCurrentProfile()}`);
|
|
157
|
+
});
|
|
158
|
+
|
|
159
|
+
configCmd
|
|
160
|
+
.command('show')
|
|
161
|
+
.description('Show current configuration')
|
|
162
|
+
.action(() => {
|
|
163
|
+
const profileName = getCurrentProfile();
|
|
164
|
+
const apiKey = getApiKey();
|
|
165
|
+
console.log(chalk.bold(`Active Profile: ${profileName}`));
|
|
166
|
+
info(`Config directory: ${getConfigDir()}`);
|
|
167
|
+
info(`API Key: ${apiKey ? `${apiKey.substring(0, 8)}...` : chalk.gray('not set')}`);
|
|
168
|
+
});
|
|
169
|
+
|
|
170
|
+
configCmd
|
|
171
|
+
.command('clear')
|
|
172
|
+
.description('Clear configuration for active profile')
|
|
173
|
+
.action(() => {
|
|
174
|
+
clearConfig();
|
|
175
|
+
success(`Configuration cleared for profile: ${getCurrentProfile()}`);
|
|
176
|
+
});
|
|
177
|
+
|
|
178
|
+
// ============================================
|
|
179
|
+
// Run Commands
|
|
180
|
+
// ============================================
|
|
181
|
+
const runCmd = program
|
|
182
|
+
.command('run')
|
|
183
|
+
.description('Run AI models');
|
|
184
|
+
|
|
185
|
+
runCmd
|
|
186
|
+
.command('image <prompt>')
|
|
187
|
+
.description('Generate images using FLUX or other models')
|
|
188
|
+
.option('-m, --model <model>', 'Model to use (flux-schnell, flux-dev, flux-pro, sdxl, etc.)', 'flux-schnell')
|
|
189
|
+
.option('-s, --size <size>', 'Image size (e.g., 1024x1024, landscape_16_9, square)', 'square_hd')
|
|
190
|
+
.option('-n, --num <number>', 'Number of images', '1')
|
|
191
|
+
.option('--steps <number>', 'Number of inference steps')
|
|
192
|
+
.option('--guidance <number>', 'Guidance scale')
|
|
193
|
+
.option('--seed <number>', 'Random seed')
|
|
194
|
+
.option('--no-safety', 'Disable safety checker')
|
|
195
|
+
.action(async (prompt: string, opts) => {
|
|
196
|
+
try {
|
|
197
|
+
const client = getClient();
|
|
198
|
+
const input: Record<string, unknown> = {
|
|
199
|
+
prompt,
|
|
200
|
+
image_size: opts.size,
|
|
201
|
+
num_images: parseInt(opts.num),
|
|
202
|
+
enable_safety_checker: opts.safety !== false,
|
|
203
|
+
};
|
|
204
|
+
|
|
205
|
+
if (opts.steps) input.num_inference_steps = parseInt(opts.steps);
|
|
206
|
+
if (opts.guidance) input.guidance_scale = parseFloat(opts.guidance);
|
|
207
|
+
if (opts.seed) input.seed = parseInt(opts.seed);
|
|
208
|
+
|
|
209
|
+
const result = await client.generateImage(opts.model, input as any);
|
|
210
|
+
|
|
211
|
+
const format = getFormat(runCmd);
|
|
212
|
+
if (format === 'json') {
|
|
213
|
+
print(result, format);
|
|
214
|
+
} else {
|
|
215
|
+
success(`Generated ${result.images.length} image(s)`);
|
|
216
|
+
result.images.forEach((img, i) => {
|
|
217
|
+
console.log(chalk.bold(`\nImage ${i + 1}:`));
|
|
218
|
+
info(`URL: ${img.url}`);
|
|
219
|
+
info(`Size: ${img.width}x${img.height}`);
|
|
220
|
+
});
|
|
221
|
+
if (result.seed !== undefined) {
|
|
222
|
+
info(`\nSeed: ${result.seed}`);
|
|
223
|
+
}
|
|
224
|
+
if (result.timings?.inference) {
|
|
225
|
+
info(`Inference time: ${result.timings.inference.toFixed(2)}s`);
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
} catch (err) {
|
|
229
|
+
error(String(err));
|
|
230
|
+
process.exit(1);
|
|
231
|
+
}
|
|
232
|
+
});
|
|
233
|
+
|
|
234
|
+
// ============================================
|
|
235
|
+
// Queue Commands
|
|
236
|
+
// ============================================
|
|
237
|
+
const queueCmd = program
|
|
238
|
+
.command('queue')
|
|
239
|
+
.description('Queue-based inference (async)');
|
|
240
|
+
|
|
241
|
+
queueCmd
|
|
242
|
+
.command('submit <model>')
|
|
243
|
+
.description('Submit a job to the queue')
|
|
244
|
+
.requiredOption('-i, --input <json>', 'Input JSON')
|
|
245
|
+
.option('-w, --webhook <url>', 'Webhook URL for completion notification')
|
|
246
|
+
.action(async (model: string, opts) => {
|
|
247
|
+
try {
|
|
248
|
+
const client = getClient();
|
|
249
|
+
const input = JSON.parse(opts.input);
|
|
250
|
+
const result = await client.submit(model, input, opts.webhook);
|
|
251
|
+
|
|
252
|
+
const format = getFormat(queueCmd);
|
|
253
|
+
if (format === 'json') {
|
|
254
|
+
print(result, format);
|
|
255
|
+
} else {
|
|
256
|
+
success(`Job submitted`);
|
|
257
|
+
info(`Request ID: ${result.request_id}`);
|
|
258
|
+
info(`Status URL: ${result.status_url}`);
|
|
259
|
+
}
|
|
260
|
+
} catch (err) {
|
|
261
|
+
error(String(err));
|
|
262
|
+
process.exit(1);
|
|
263
|
+
}
|
|
264
|
+
});
|
|
265
|
+
|
|
266
|
+
queueCmd
|
|
267
|
+
.command('status <model> <requestId>')
|
|
268
|
+
.description('Check job status')
|
|
269
|
+
.action(async (model: string, requestId: string) => {
|
|
270
|
+
try {
|
|
271
|
+
const client = getClient();
|
|
272
|
+
const result = await client.status(model, requestId);
|
|
273
|
+
|
|
274
|
+
const format = getFormat(queueCmd);
|
|
275
|
+
if (format === 'json') {
|
|
276
|
+
print(result, format);
|
|
277
|
+
} else {
|
|
278
|
+
console.log(chalk.bold(`Request: ${result.request_id}`));
|
|
279
|
+
info(`Status: ${result.status}`);
|
|
280
|
+
if (result.logs && result.logs.length > 0) {
|
|
281
|
+
console.log(chalk.bold('\nLogs:'));
|
|
282
|
+
result.logs.forEach(log => {
|
|
283
|
+
console.log(` [${log.timestamp}] ${log.message}`);
|
|
284
|
+
});
|
|
285
|
+
}
|
|
286
|
+
}
|
|
287
|
+
} catch (err) {
|
|
288
|
+
error(String(err));
|
|
289
|
+
process.exit(1);
|
|
290
|
+
}
|
|
291
|
+
});
|
|
292
|
+
|
|
293
|
+
queueCmd
|
|
294
|
+
.command('result <model> <requestId>')
|
|
295
|
+
.description('Get job result')
|
|
296
|
+
.action(async (model: string, requestId: string) => {
|
|
297
|
+
try {
|
|
298
|
+
const client = getClient();
|
|
299
|
+
const result = await client.result(model, requestId);
|
|
300
|
+
|
|
301
|
+
const format = getFormat(queueCmd);
|
|
302
|
+
print(result, format);
|
|
303
|
+
} catch (err) {
|
|
304
|
+
error(String(err));
|
|
305
|
+
process.exit(1);
|
|
306
|
+
}
|
|
307
|
+
});
|
|
308
|
+
|
|
309
|
+
queueCmd
|
|
310
|
+
.command('cancel <model> <requestId>')
|
|
311
|
+
.description('Cancel a queued job')
|
|
312
|
+
.action(async (model: string, requestId: string) => {
|
|
313
|
+
try {
|
|
314
|
+
const client = getClient();
|
|
315
|
+
await client.cancel(model, requestId);
|
|
316
|
+
success(`Job cancelled: ${requestId}`);
|
|
317
|
+
} catch (err) {
|
|
318
|
+
error(String(err));
|
|
319
|
+
process.exit(1);
|
|
320
|
+
}
|
|
321
|
+
});
|
|
322
|
+
|
|
323
|
+
// ============================================
|
|
324
|
+
// Model Commands
|
|
325
|
+
// ============================================
|
|
326
|
+
const modelCmd = program
|
|
327
|
+
.command('model')
|
|
328
|
+
.description('Model information');
|
|
329
|
+
|
|
330
|
+
modelCmd
|
|
331
|
+
.command('list')
|
|
332
|
+
.description('List common model aliases')
|
|
333
|
+
.action(() => {
|
|
334
|
+
success('Common model aliases:');
|
|
335
|
+
Object.entries(COMMON_MODELS).forEach(([alias, model]) => {
|
|
336
|
+
console.log(` ${chalk.bold(alias)} → ${model}`);
|
|
337
|
+
});
|
|
338
|
+
info('\nYou can also use full model paths like: fal-ai/flux/dev');
|
|
339
|
+
});
|
|
340
|
+
|
|
341
|
+
program.parse();
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
// fal.ai Connector
|
|
2
|
+
// TypeScript wrapper for fal.ai serverless AI inference
|
|
3
|
+
|
|
4
|
+
export { Fal } from './api';
|
|
5
|
+
export * from './types';
|
|
6
|
+
export { FalClient } from './api';
|
|
7
|
+
|
|
8
|
+
export {
|
|
9
|
+
getApiKey,
|
|
10
|
+
setApiKey,
|
|
11
|
+
getCurrentProfile,
|
|
12
|
+
setCurrentProfile,
|
|
13
|
+
listProfiles,
|
|
14
|
+
createProfile,
|
|
15
|
+
deleteProfile,
|
|
16
|
+
loadProfile,
|
|
17
|
+
saveProfile,
|
|
18
|
+
clearConfig,
|
|
19
|
+
} from './utils/config';
|