@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,50 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@hasna/connect-assemblyai",
|
|
3
|
+
"version": "0.0.1",
|
|
4
|
+
"description": "AssemblyAI connector - Speech-to-text transcription with multi-profile support",
|
|
5
|
+
"type": "module",
|
|
6
|
+
"bin": {
|
|
7
|
+
"connect-assemblyai": "./bin/index.js"
|
|
8
|
+
},
|
|
9
|
+
"exports": {
|
|
10
|
+
".": {
|
|
11
|
+
"import": "./dist/index.js",
|
|
12
|
+
"types": "./dist/index.d.ts"
|
|
13
|
+
}
|
|
14
|
+
},
|
|
15
|
+
"main": "./dist/index.js",
|
|
16
|
+
"types": "./dist/index.d.ts",
|
|
17
|
+
"scripts": {
|
|
18
|
+
"build": "bun build ./src/index.ts --outdir ./dist --target bun && bun build ./src/cli/index.ts --outdir ./bin --target bun",
|
|
19
|
+
"dev": "bun run ./src/cli/index.ts",
|
|
20
|
+
"typecheck": "tsc --noEmit",
|
|
21
|
+
"prepublishOnly": "bun run build"
|
|
22
|
+
},
|
|
23
|
+
"keywords": [
|
|
24
|
+
"assemblyai",
|
|
25
|
+
"speech-to-text",
|
|
26
|
+
"transcription",
|
|
27
|
+
"audio",
|
|
28
|
+
"connector",
|
|
29
|
+
"cli",
|
|
30
|
+
"typescript",
|
|
31
|
+
"bun"
|
|
32
|
+
],
|
|
33
|
+
"author": "Hasna",
|
|
34
|
+
"license": "Apache-2.0",
|
|
35
|
+
"devDependencies": {
|
|
36
|
+
"@types/bun": "latest",
|
|
37
|
+
"typescript": "^5"
|
|
38
|
+
},
|
|
39
|
+
"dependencies": {
|
|
40
|
+
"commander": "^12.1.0",
|
|
41
|
+
"chalk": "^5.3.0"
|
|
42
|
+
},
|
|
43
|
+
"engines": {
|
|
44
|
+
"bun": ">=1.0.0"
|
|
45
|
+
},
|
|
46
|
+
"repository": {
|
|
47
|
+
"type": "git",
|
|
48
|
+
"url": "git+https://github.com/hasna/open-connectors.git"
|
|
49
|
+
}
|
|
50
|
+
}
|
|
@@ -0,0 +1,192 @@
|
|
|
1
|
+
import type {
|
|
2
|
+
AssemblyAIConfig,
|
|
3
|
+
TranscriptRequest,
|
|
4
|
+
Transcript,
|
|
5
|
+
TranscriptListResponse,
|
|
6
|
+
UploadResponse,
|
|
7
|
+
LemurTaskRequest,
|
|
8
|
+
LemurResponse,
|
|
9
|
+
LemurSummaryRequest,
|
|
10
|
+
LemurQuestionAnswerRequest,
|
|
11
|
+
LemurQuestionAnswerResponse,
|
|
12
|
+
} from '../types';
|
|
13
|
+
import { AssemblyAIApiError } from '../types';
|
|
14
|
+
|
|
15
|
+
const DEFAULT_BASE_URL = 'https://api.assemblyai.com/v2';
|
|
16
|
+
|
|
17
|
+
export interface RequestOptions {
|
|
18
|
+
method?: 'GET' | 'POST' | 'PUT' | 'DELETE' | 'PATCH';
|
|
19
|
+
params?: Record<string, string | number | boolean | undefined>;
|
|
20
|
+
body?: Record<string, unknown> | unknown[] | string;
|
|
21
|
+
headers?: Record<string, string>;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
export class AssemblyAIClient {
|
|
25
|
+
private readonly apiKey: string;
|
|
26
|
+
private readonly baseUrl: string;
|
|
27
|
+
|
|
28
|
+
constructor(config: AssemblyAIConfig) {
|
|
29
|
+
if (!config.apiKey) {
|
|
30
|
+
throw new Error('API key is required');
|
|
31
|
+
}
|
|
32
|
+
this.apiKey = config.apiKey;
|
|
33
|
+
this.baseUrl = config.baseUrl || DEFAULT_BASE_URL;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
private buildUrl(path: string, params?: Record<string, string | number | boolean | undefined>): string {
|
|
37
|
+
const url = new URL(`${this.baseUrl}${path}`);
|
|
38
|
+
|
|
39
|
+
if (params) {
|
|
40
|
+
Object.entries(params).forEach(([key, value]) => {
|
|
41
|
+
if (value !== undefined && value !== null && value !== '') {
|
|
42
|
+
url.searchParams.append(key, String(value));
|
|
43
|
+
}
|
|
44
|
+
});
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
return url.toString();
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
async request<T>(path: string, options: RequestOptions = {}): Promise<T> {
|
|
51
|
+
const { method = 'GET', params, body, headers = {} } = options;
|
|
52
|
+
|
|
53
|
+
const url = this.buildUrl(path, params);
|
|
54
|
+
|
|
55
|
+
// AssemblyAI uses plain API key in Authorization header
|
|
56
|
+
const requestHeaders: Record<string, string> = {
|
|
57
|
+
'Authorization': this.apiKey,
|
|
58
|
+
'Accept': 'application/json',
|
|
59
|
+
...headers,
|
|
60
|
+
};
|
|
61
|
+
|
|
62
|
+
if (body && ['POST', 'PUT', 'PATCH'].includes(method)) {
|
|
63
|
+
requestHeaders['Content-Type'] = 'application/json';
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
const fetchOptions: RequestInit = {
|
|
67
|
+
method,
|
|
68
|
+
headers: requestHeaders,
|
|
69
|
+
};
|
|
70
|
+
|
|
71
|
+
if (body && ['POST', 'PUT', 'PATCH'].includes(method)) {
|
|
72
|
+
fetchOptions.body = typeof body === 'string' ? body : JSON.stringify(body);
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
const response = await fetch(url, fetchOptions);
|
|
76
|
+
|
|
77
|
+
if (response.status === 204) {
|
|
78
|
+
return {} as T;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
let data: unknown;
|
|
82
|
+
const contentType = response.headers.get('content-type') || '';
|
|
83
|
+
|
|
84
|
+
if (contentType.includes('application/json')) {
|
|
85
|
+
const text = await response.text();
|
|
86
|
+
if (text) {
|
|
87
|
+
try {
|
|
88
|
+
data = JSON.parse(text);
|
|
89
|
+
} catch {
|
|
90
|
+
data = text;
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
} else {
|
|
94
|
+
data = await response.text();
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
if (!response.ok) {
|
|
98
|
+
const errorMessage = typeof data === 'object' && data !== null
|
|
99
|
+
? JSON.stringify(data)
|
|
100
|
+
: String(data || response.statusText);
|
|
101
|
+
throw new AssemblyAIApiError(errorMessage, response.status);
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
return data as T;
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
async get<T>(path: string, params?: Record<string, string | number | boolean | undefined>): Promise<T> {
|
|
108
|
+
return this.request<T>(path, { method: 'GET', params });
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
async post<T>(path: string, body?: Record<string, unknown> | unknown[] | string | object): Promise<T> {
|
|
112
|
+
return this.request<T>(path, { method: 'POST', body: body as Record<string, unknown> });
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
async delete<T>(path: string): Promise<T> {
|
|
116
|
+
return this.request<T>(path, { method: 'DELETE' });
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
// ============================================
|
|
120
|
+
// Transcript Methods
|
|
121
|
+
// ============================================
|
|
122
|
+
|
|
123
|
+
async createTranscript(params: TranscriptRequest): Promise<Transcript> {
|
|
124
|
+
return this.post<Transcript>('/transcript', params);
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
async getTranscript(transcriptId: string): Promise<Transcript> {
|
|
128
|
+
return this.get<Transcript>(`/transcript/${transcriptId}`);
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
async listTranscripts(limit?: number, status?: string, created_on?: string): Promise<TranscriptListResponse> {
|
|
132
|
+
return this.get<TranscriptListResponse>('/transcript', { limit, status, created_on });
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
async deleteTranscript(transcriptId: string): Promise<Transcript> {
|
|
136
|
+
return this.delete<Transcript>(`/transcript/${transcriptId}`);
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
async waitForTranscript(transcriptId: string, pollIntervalMs = 3000): Promise<Transcript> {
|
|
140
|
+
while (true) {
|
|
141
|
+
const transcript = await this.getTranscript(transcriptId);
|
|
142
|
+
if (transcript.status === 'completed' || transcript.status === 'error') {
|
|
143
|
+
return transcript;
|
|
144
|
+
}
|
|
145
|
+
await new Promise(resolve => setTimeout(resolve, pollIntervalMs));
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
// ============================================
|
|
150
|
+
// Upload Methods
|
|
151
|
+
// ============================================
|
|
152
|
+
|
|
153
|
+
async upload(audioData: Buffer | Uint8Array): Promise<UploadResponse> {
|
|
154
|
+
const response = await fetch(`${this.baseUrl}/upload`, {
|
|
155
|
+
method: 'POST',
|
|
156
|
+
headers: {
|
|
157
|
+
'Authorization': this.apiKey,
|
|
158
|
+
'Content-Type': 'application/octet-stream',
|
|
159
|
+
},
|
|
160
|
+
body: audioData,
|
|
161
|
+
});
|
|
162
|
+
|
|
163
|
+
if (!response.ok) {
|
|
164
|
+
throw new AssemblyAIApiError(await response.text(), response.status);
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
return response.json();
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
// ============================================
|
|
171
|
+
// LeMUR Methods
|
|
172
|
+
// ============================================
|
|
173
|
+
|
|
174
|
+
async lemurTask(params: LemurTaskRequest): Promise<LemurResponse> {
|
|
175
|
+
return this.post<LemurResponse>('/lemur/v3/generate/task', params);
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
async lemurSummary(params: LemurSummaryRequest): Promise<LemurResponse> {
|
|
179
|
+
return this.post<LemurResponse>('/lemur/v3/generate/summary', params);
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
async lemurQuestionAnswer(params: LemurQuestionAnswerRequest): Promise<LemurQuestionAnswerResponse> {
|
|
183
|
+
return this.post<LemurQuestionAnswerResponse>('/lemur/v3/generate/question-answer', params);
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
getApiKeyPreview(): string {
|
|
187
|
+
if (this.apiKey.length > 10) {
|
|
188
|
+
return `${this.apiKey.substring(0, 6)}...${this.apiKey.substring(this.apiKey.length - 4)}`;
|
|
189
|
+
}
|
|
190
|
+
return '***';
|
|
191
|
+
}
|
|
192
|
+
}
|
|
@@ -0,0 +1,71 @@
|
|
|
1
|
+
import type {
|
|
2
|
+
AssemblyAIConfig,
|
|
3
|
+
TranscriptRequest,
|
|
4
|
+
Transcript,
|
|
5
|
+
TranscriptListResponse,
|
|
6
|
+
UploadResponse,
|
|
7
|
+
LemurTaskRequest,
|
|
8
|
+
LemurResponse,
|
|
9
|
+
LemurSummaryRequest,
|
|
10
|
+
LemurQuestionAnswerRequest,
|
|
11
|
+
LemurQuestionAnswerResponse,
|
|
12
|
+
} from '../types';
|
|
13
|
+
import { AssemblyAIClient } from './client';
|
|
14
|
+
|
|
15
|
+
export class AssemblyAI {
|
|
16
|
+
private readonly client: AssemblyAIClient;
|
|
17
|
+
|
|
18
|
+
constructor(config: AssemblyAIConfig) {
|
|
19
|
+
this.client = new AssemblyAIClient(config);
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
async createTranscript(params: TranscriptRequest): Promise<Transcript> {
|
|
23
|
+
return this.client.createTranscript(params);
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
async getTranscript(transcriptId: string): Promise<Transcript> {
|
|
27
|
+
return this.client.getTranscript(transcriptId);
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
async listTranscripts(limit?: number, status?: string, created_on?: string): Promise<TranscriptListResponse> {
|
|
31
|
+
return this.client.listTranscripts(limit, status, created_on);
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
async deleteTranscript(transcriptId: string): Promise<Transcript> {
|
|
35
|
+
return this.client.deleteTranscript(transcriptId);
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
async waitForTranscript(transcriptId: string, pollIntervalMs?: number): Promise<Transcript> {
|
|
39
|
+
return this.client.waitForTranscript(transcriptId, pollIntervalMs);
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
async upload(audioData: Buffer | Uint8Array): Promise<UploadResponse> {
|
|
43
|
+
return this.client.upload(audioData);
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
async lemurTask(params: LemurTaskRequest): Promise<LemurResponse> {
|
|
47
|
+
return this.client.lemurTask(params);
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
async lemurSummary(params: LemurSummaryRequest): Promise<LemurResponse> {
|
|
51
|
+
return this.client.lemurSummary(params);
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
async lemurQuestionAnswer(params: LemurQuestionAnswerRequest): Promise<LemurQuestionAnswerResponse> {
|
|
55
|
+
return this.client.lemurQuestionAnswer(params);
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
static fromEnv(): AssemblyAI {
|
|
59
|
+
const apiKey = process.env.ASSEMBLYAI_API_KEY;
|
|
60
|
+
if (!apiKey) {
|
|
61
|
+
throw new Error('ASSEMBLYAI_API_KEY environment variable is required');
|
|
62
|
+
}
|
|
63
|
+
return new AssemblyAI({ apiKey });
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
getApiKeyPreview(): string {
|
|
67
|
+
return this.client.getApiKeyPreview();
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
export { AssemblyAIClient } from './client';
|
|
@@ -0,0 +1,384 @@
|
|
|
1
|
+
#!/usr/bin/env bun
|
|
2
|
+
import { Command } from 'commander';
|
|
3
|
+
import chalk from 'chalk';
|
|
4
|
+
import { AssemblyAI } from '../api';
|
|
5
|
+
import {
|
|
6
|
+
getApiKey,
|
|
7
|
+
setApiKey,
|
|
8
|
+
clearConfig,
|
|
9
|
+
getConfigDir,
|
|
10
|
+
setProfileOverride,
|
|
11
|
+
getCurrentProfile,
|
|
12
|
+
setCurrentProfile,
|
|
13
|
+
listProfiles,
|
|
14
|
+
createProfile,
|
|
15
|
+
deleteProfile,
|
|
16
|
+
profileExists,
|
|
17
|
+
loadProfile,
|
|
18
|
+
} from '../utils/config';
|
|
19
|
+
import type { OutputFormat } from '../utils/output';
|
|
20
|
+
import { success, error, info, print } from '../utils/output';
|
|
21
|
+
|
|
22
|
+
const CONNECTOR_NAME = 'connect-assemblyai';
|
|
23
|
+
const VERSION = '0.0.1';
|
|
24
|
+
|
|
25
|
+
const program = new Command();
|
|
26
|
+
|
|
27
|
+
program
|
|
28
|
+
.name(CONNECTOR_NAME)
|
|
29
|
+
.description('AssemblyAI connector CLI - Speech-to-text transcription with multi-profile support')
|
|
30
|
+
.version(VERSION)
|
|
31
|
+
.option('-k, --api-key <key>', 'API key (overrides config)')
|
|
32
|
+
.option('-f, --format <format>', 'Output format (json, pretty)', 'pretty')
|
|
33
|
+
.option('-p, --profile <profile>', 'Use a specific profile')
|
|
34
|
+
.hook('preAction', (thisCommand) => {
|
|
35
|
+
const opts = thisCommand.opts();
|
|
36
|
+
if (opts.profile) {
|
|
37
|
+
if (!profileExists(opts.profile)) {
|
|
38
|
+
error(`Profile "${opts.profile}" does not exist. Create it with "${CONNECTOR_NAME} profile create ${opts.profile}"`);
|
|
39
|
+
process.exit(1);
|
|
40
|
+
}
|
|
41
|
+
setProfileOverride(opts.profile);
|
|
42
|
+
}
|
|
43
|
+
if (opts.apiKey) {
|
|
44
|
+
process.env.ASSEMBLYAI_API_KEY = opts.apiKey;
|
|
45
|
+
}
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
function getFormat(cmd: Command): OutputFormat {
|
|
49
|
+
const parent = cmd.parent;
|
|
50
|
+
return (parent?.opts().format || 'pretty') as OutputFormat;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
function getClient(): AssemblyAI {
|
|
54
|
+
const apiKey = getApiKey();
|
|
55
|
+
if (!apiKey) {
|
|
56
|
+
error(`No API key configured. Run "${CONNECTOR_NAME} config set-key <key>" or set ASSEMBLYAI_API_KEY environment variable.`);
|
|
57
|
+
process.exit(1);
|
|
58
|
+
}
|
|
59
|
+
return new AssemblyAI({ apiKey });
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
// ============================================
|
|
63
|
+
// Profile Commands
|
|
64
|
+
// ============================================
|
|
65
|
+
const profileCmd = program
|
|
66
|
+
.command('profile')
|
|
67
|
+
.description('Manage configuration profiles');
|
|
68
|
+
|
|
69
|
+
profileCmd
|
|
70
|
+
.command('list')
|
|
71
|
+
.description('List all profiles')
|
|
72
|
+
.action(() => {
|
|
73
|
+
const profiles = listProfiles();
|
|
74
|
+
const current = getCurrentProfile();
|
|
75
|
+
if (profiles.length === 0) {
|
|
76
|
+
info('No profiles found. Use "profile create <name>" to create one.');
|
|
77
|
+
return;
|
|
78
|
+
}
|
|
79
|
+
success(`Profiles:`);
|
|
80
|
+
profiles.forEach(p => {
|
|
81
|
+
const isActive = p === current ? chalk.green(' (active)') : '';
|
|
82
|
+
console.log(` ${p}${isActive}`);
|
|
83
|
+
});
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
profileCmd
|
|
87
|
+
.command('use <name>')
|
|
88
|
+
.description('Switch to a profile')
|
|
89
|
+
.action((name: string) => {
|
|
90
|
+
if (!profileExists(name)) {
|
|
91
|
+
error(`Profile "${name}" does not exist. Create it with "profile create ${name}"`);
|
|
92
|
+
process.exit(1);
|
|
93
|
+
}
|
|
94
|
+
setCurrentProfile(name);
|
|
95
|
+
success(`Switched to profile: ${name}`);
|
|
96
|
+
});
|
|
97
|
+
|
|
98
|
+
profileCmd
|
|
99
|
+
.command('create <name>')
|
|
100
|
+
.description('Create a new profile')
|
|
101
|
+
.option('--api-key <key>', 'API key')
|
|
102
|
+
.option('--use', 'Switch to this profile after creation')
|
|
103
|
+
.action((name: string, opts) => {
|
|
104
|
+
if (profileExists(name)) {
|
|
105
|
+
error(`Profile "${name}" already exists`);
|
|
106
|
+
process.exit(1);
|
|
107
|
+
}
|
|
108
|
+
createProfile(name, { apiKey: opts.apiKey });
|
|
109
|
+
success(`Profile "${name}" created`);
|
|
110
|
+
if (opts.use) {
|
|
111
|
+
setCurrentProfile(name);
|
|
112
|
+
info(`Switched to profile: ${name}`);
|
|
113
|
+
}
|
|
114
|
+
});
|
|
115
|
+
|
|
116
|
+
profileCmd
|
|
117
|
+
.command('delete <name>')
|
|
118
|
+
.description('Delete a profile')
|
|
119
|
+
.action((name: string) => {
|
|
120
|
+
if (name === 'default') {
|
|
121
|
+
error('Cannot delete the default profile');
|
|
122
|
+
process.exit(1);
|
|
123
|
+
}
|
|
124
|
+
if (deleteProfile(name)) {
|
|
125
|
+
success(`Profile "${name}" deleted`);
|
|
126
|
+
} else {
|
|
127
|
+
error(`Profile "${name}" not found`);
|
|
128
|
+
process.exit(1);
|
|
129
|
+
}
|
|
130
|
+
});
|
|
131
|
+
|
|
132
|
+
profileCmd
|
|
133
|
+
.command('show [name]')
|
|
134
|
+
.description('Show profile configuration')
|
|
135
|
+
.action((name?: string) => {
|
|
136
|
+
const profileName = name || getCurrentProfile();
|
|
137
|
+
const config = loadProfile(profileName);
|
|
138
|
+
const active = getCurrentProfile();
|
|
139
|
+
console.log(chalk.bold(`Profile: ${profileName}${profileName === active ? chalk.green(' (active)') : ''}`));
|
|
140
|
+
info(`API Key: ${config.apiKey ? `${config.apiKey.substring(0, 8)}...` : chalk.gray('not set')}`);
|
|
141
|
+
});
|
|
142
|
+
|
|
143
|
+
// ============================================
|
|
144
|
+
// Config Commands
|
|
145
|
+
// ============================================
|
|
146
|
+
const configCmd = program
|
|
147
|
+
.command('config')
|
|
148
|
+
.description('Manage CLI configuration (for active profile)');
|
|
149
|
+
|
|
150
|
+
configCmd
|
|
151
|
+
.command('set-key <apiKey>')
|
|
152
|
+
.description('Set API key')
|
|
153
|
+
.action((apiKey: string) => {
|
|
154
|
+
setApiKey(apiKey);
|
|
155
|
+
success(`API key saved to profile: ${getCurrentProfile()}`);
|
|
156
|
+
});
|
|
157
|
+
|
|
158
|
+
configCmd
|
|
159
|
+
.command('show')
|
|
160
|
+
.description('Show current configuration')
|
|
161
|
+
.action(() => {
|
|
162
|
+
const profileName = getCurrentProfile();
|
|
163
|
+
const apiKey = getApiKey();
|
|
164
|
+
console.log(chalk.bold(`Active Profile: ${profileName}`));
|
|
165
|
+
info(`Config directory: ${getConfigDir()}`);
|
|
166
|
+
info(`API Key: ${apiKey ? `${apiKey.substring(0, 8)}...` : chalk.gray('not set')}`);
|
|
167
|
+
});
|
|
168
|
+
|
|
169
|
+
configCmd
|
|
170
|
+
.command('clear')
|
|
171
|
+
.description('Clear configuration for active profile')
|
|
172
|
+
.action(() => {
|
|
173
|
+
clearConfig();
|
|
174
|
+
success(`Configuration cleared for profile: ${getCurrentProfile()}`);
|
|
175
|
+
});
|
|
176
|
+
|
|
177
|
+
// ============================================
|
|
178
|
+
// Transcript Commands
|
|
179
|
+
// ============================================
|
|
180
|
+
const transcriptCmd = program
|
|
181
|
+
.command('transcript')
|
|
182
|
+
.description('Transcription commands');
|
|
183
|
+
|
|
184
|
+
transcriptCmd
|
|
185
|
+
.command('create <audioUrl>')
|
|
186
|
+
.description('Create a new transcript from audio URL')
|
|
187
|
+
.option('-l, --language <code>', 'Language code (e.g., en, es, fr)')
|
|
188
|
+
.option('--speaker-labels', 'Enable speaker diarization')
|
|
189
|
+
.option('--speakers <number>', 'Expected number of speakers')
|
|
190
|
+
.option('--punctuate', 'Enable automatic punctuation')
|
|
191
|
+
.option('--format-text', 'Enable text formatting')
|
|
192
|
+
.option('--summarize', 'Enable summarization')
|
|
193
|
+
.option('--sentiment', 'Enable sentiment analysis')
|
|
194
|
+
.option('--auto-chapters', 'Enable auto chapters')
|
|
195
|
+
.option('--entity-detection', 'Enable entity detection')
|
|
196
|
+
.option('--wait', 'Wait for transcript to complete')
|
|
197
|
+
.action(async (audioUrl: string, opts) => {
|
|
198
|
+
try {
|
|
199
|
+
const client = getClient();
|
|
200
|
+
const result = await client.createTranscript({
|
|
201
|
+
audio_url: audioUrl,
|
|
202
|
+
language_code: opts.language,
|
|
203
|
+
speaker_labels: opts.speakerLabels || undefined,
|
|
204
|
+
speakers_expected: opts.speakers ? parseInt(opts.speakers) : undefined,
|
|
205
|
+
punctuate: opts.punctuate || undefined,
|
|
206
|
+
format_text: opts.formatText || undefined,
|
|
207
|
+
summarization: opts.summarize || undefined,
|
|
208
|
+
sentiment_analysis: opts.sentiment || undefined,
|
|
209
|
+
auto_chapters: opts.autoChapters || undefined,
|
|
210
|
+
entity_detection: opts.entityDetection || undefined,
|
|
211
|
+
});
|
|
212
|
+
|
|
213
|
+
if (opts.wait) {
|
|
214
|
+
info(`Transcript ${result.id} queued. Waiting for completion...`);
|
|
215
|
+
const completed = await client.waitForTranscript(result.id);
|
|
216
|
+
const format = getFormat(transcriptCmd);
|
|
217
|
+
if (format === 'json') {
|
|
218
|
+
print(completed, format);
|
|
219
|
+
} else {
|
|
220
|
+
if (completed.status === 'completed') {
|
|
221
|
+
success(`Transcript completed`);
|
|
222
|
+
info(`ID: ${completed.id}`);
|
|
223
|
+
info(`Duration: ${completed.audio_duration}s`);
|
|
224
|
+
info(`Confidence: ${((completed.confidence || 0) * 100).toFixed(1)}%`);
|
|
225
|
+
if (completed.text) {
|
|
226
|
+
console.log(chalk.bold('\nTranscript:'));
|
|
227
|
+
console.log(completed.text);
|
|
228
|
+
}
|
|
229
|
+
if (completed.summary) {
|
|
230
|
+
console.log(chalk.bold('\nSummary:'));
|
|
231
|
+
console.log(completed.summary);
|
|
232
|
+
}
|
|
233
|
+
} else {
|
|
234
|
+
error(`Transcript failed: ${completed.error}`);
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
} else {
|
|
238
|
+
const format = getFormat(transcriptCmd);
|
|
239
|
+
if (format === 'json') {
|
|
240
|
+
print(result, format);
|
|
241
|
+
} else {
|
|
242
|
+
success(`Transcript created`);
|
|
243
|
+
info(`ID: ${result.id}`);
|
|
244
|
+
info(`Status: ${result.status}`);
|
|
245
|
+
info(`Use "connect-assemblyai transcript get ${result.id}" to check status`);
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
} catch (err) {
|
|
249
|
+
error(String(err));
|
|
250
|
+
process.exit(1);
|
|
251
|
+
}
|
|
252
|
+
});
|
|
253
|
+
|
|
254
|
+
transcriptCmd
|
|
255
|
+
.command('get <transcriptId>')
|
|
256
|
+
.description('Get transcript by ID')
|
|
257
|
+
.action(async (transcriptId: string) => {
|
|
258
|
+
try {
|
|
259
|
+
const client = getClient();
|
|
260
|
+
const result = await client.getTranscript(transcriptId);
|
|
261
|
+
const format = getFormat(transcriptCmd);
|
|
262
|
+
if (format === 'json') {
|
|
263
|
+
print(result, format);
|
|
264
|
+
} else {
|
|
265
|
+
console.log(chalk.bold(`Transcript: ${result.id}`));
|
|
266
|
+
info(`Status: ${result.status}`);
|
|
267
|
+
if (result.status === 'completed') {
|
|
268
|
+
info(`Duration: ${result.audio_duration}s`);
|
|
269
|
+
info(`Confidence: ${((result.confidence || 0) * 100).toFixed(1)}%`);
|
|
270
|
+
if (result.text) {
|
|
271
|
+
console.log(chalk.bold('\nTranscript:'));
|
|
272
|
+
console.log(result.text);
|
|
273
|
+
}
|
|
274
|
+
} else if (result.status === 'error') {
|
|
275
|
+
error(`Error: ${result.error}`);
|
|
276
|
+
}
|
|
277
|
+
}
|
|
278
|
+
} catch (err) {
|
|
279
|
+
error(String(err));
|
|
280
|
+
process.exit(1);
|
|
281
|
+
}
|
|
282
|
+
});
|
|
283
|
+
|
|
284
|
+
transcriptCmd
|
|
285
|
+
.command('list')
|
|
286
|
+
.description('List transcripts')
|
|
287
|
+
.option('-n, --limit <number>', 'Maximum results', '10')
|
|
288
|
+
.option('-s, --status <status>', 'Filter by status (queued, processing, completed, error)')
|
|
289
|
+
.action(async (opts) => {
|
|
290
|
+
try {
|
|
291
|
+
const client = getClient();
|
|
292
|
+
const result = await client.listTranscripts(parseInt(opts.limit), opts.status);
|
|
293
|
+
const format = getFormat(transcriptCmd);
|
|
294
|
+
if (format === 'json') {
|
|
295
|
+
print(result, format);
|
|
296
|
+
} else {
|
|
297
|
+
success(`Transcripts (${result.transcripts.length}):`);
|
|
298
|
+
result.transcripts.forEach(t => {
|
|
299
|
+
console.log(` ${t.id} - ${t.status} - ${t.created}`);
|
|
300
|
+
});
|
|
301
|
+
}
|
|
302
|
+
} catch (err) {
|
|
303
|
+
error(String(err));
|
|
304
|
+
process.exit(1);
|
|
305
|
+
}
|
|
306
|
+
});
|
|
307
|
+
|
|
308
|
+
transcriptCmd
|
|
309
|
+
.command('delete <transcriptId>')
|
|
310
|
+
.description('Delete a transcript')
|
|
311
|
+
.action(async (transcriptId: string) => {
|
|
312
|
+
try {
|
|
313
|
+
const client = getClient();
|
|
314
|
+
await client.deleteTranscript(transcriptId);
|
|
315
|
+
success(`Transcript ${transcriptId} deleted`);
|
|
316
|
+
} catch (err) {
|
|
317
|
+
error(String(err));
|
|
318
|
+
process.exit(1);
|
|
319
|
+
}
|
|
320
|
+
});
|
|
321
|
+
|
|
322
|
+
// ============================================
|
|
323
|
+
// LeMUR Commands
|
|
324
|
+
// ============================================
|
|
325
|
+
const lemurCmd = program
|
|
326
|
+
.command('lemur')
|
|
327
|
+
.description('LeMUR AI commands');
|
|
328
|
+
|
|
329
|
+
lemurCmd
|
|
330
|
+
.command('task <transcriptIds>')
|
|
331
|
+
.description('Run a custom LeMUR task')
|
|
332
|
+
.requiredOption('--prompt <prompt>', 'Task prompt')
|
|
333
|
+
.option('--context <context>', 'Additional context')
|
|
334
|
+
.option('--temperature <number>', 'Temperature (0-1)')
|
|
335
|
+
.action(async (transcriptIds: string, opts) => {
|
|
336
|
+
try {
|
|
337
|
+
const client = getClient();
|
|
338
|
+
const result = await client.lemurTask({
|
|
339
|
+
transcript_ids: transcriptIds.split(','),
|
|
340
|
+
prompt: opts.prompt,
|
|
341
|
+
context: opts.context,
|
|
342
|
+
temperature: opts.temperature ? parseFloat(opts.temperature) : undefined,
|
|
343
|
+
});
|
|
344
|
+
|
|
345
|
+
const format = getFormat(lemurCmd);
|
|
346
|
+
if (format === 'json') {
|
|
347
|
+
print(result, format);
|
|
348
|
+
} else {
|
|
349
|
+
success('LeMUR Response:');
|
|
350
|
+
console.log(result.response);
|
|
351
|
+
info(`\nTokens: ${result.usage.input_tokens} input, ${result.usage.output_tokens} output`);
|
|
352
|
+
}
|
|
353
|
+
} catch (err) {
|
|
354
|
+
error(String(err));
|
|
355
|
+
process.exit(1);
|
|
356
|
+
}
|
|
357
|
+
});
|
|
358
|
+
|
|
359
|
+
lemurCmd
|
|
360
|
+
.command('summary <transcriptIds>')
|
|
361
|
+
.description('Generate a summary of transcripts')
|
|
362
|
+
.option('--context <context>', 'Additional context')
|
|
363
|
+
.action(async (transcriptIds: string, opts) => {
|
|
364
|
+
try {
|
|
365
|
+
const client = getClient();
|
|
366
|
+
const result = await client.lemurSummary({
|
|
367
|
+
transcript_ids: transcriptIds.split(','),
|
|
368
|
+
context: opts.context,
|
|
369
|
+
});
|
|
370
|
+
|
|
371
|
+
const format = getFormat(lemurCmd);
|
|
372
|
+
if (format === 'json') {
|
|
373
|
+
print(result, format);
|
|
374
|
+
} else {
|
|
375
|
+
success('Summary:');
|
|
376
|
+
console.log(result.response);
|
|
377
|
+
}
|
|
378
|
+
} catch (err) {
|
|
379
|
+
error(String(err));
|
|
380
|
+
process.exit(1);
|
|
381
|
+
}
|
|
382
|
+
});
|
|
383
|
+
|
|
384
|
+
program.parse();
|