@hasna/connectors 0.4.1 → 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.
Files changed (205) hide show
  1. package/bin/index.js +126 -2
  2. package/bin/mcp.js +113 -1
  3. package/bin/serve.js +112 -0
  4. package/connectors/connect-assemblyai/.env.example +11 -0
  5. package/connectors/connect-assemblyai/CLAUDE.md +128 -0
  6. package/connectors/connect-assemblyai/README.md +193 -0
  7. package/connectors/connect-assemblyai/package.json +50 -0
  8. package/connectors/connect-assemblyai/src/api/client.ts +192 -0
  9. package/connectors/connect-assemblyai/src/api/index.ts +71 -0
  10. package/connectors/connect-assemblyai/src/cli/index.ts +384 -0
  11. package/connectors/connect-assemblyai/src/index.ts +19 -0
  12. package/connectors/connect-assemblyai/src/types/index.ts +277 -0
  13. package/connectors/connect-assemblyai/src/utils/config.ts +103 -0
  14. package/connectors/connect-assemblyai/src/utils/output.ts +119 -0
  15. package/connectors/connect-assemblyai/tsconfig.json +16 -0
  16. package/connectors/connect-baseten/.env.example +11 -0
  17. package/connectors/connect-baseten/CLAUDE.md +128 -0
  18. package/connectors/connect-baseten/README.md +193 -0
  19. package/connectors/connect-baseten/package.json +51 -0
  20. package/connectors/connect-baseten/src/api/client.ts +71 -0
  21. package/connectors/connect-baseten/src/api/index.ts +40 -0
  22. package/connectors/connect-baseten/src/cli/index.ts +244 -0
  23. package/connectors/connect-baseten/src/index.ts +19 -0
  24. package/connectors/connect-baseten/src/types/index.ts +55 -0
  25. package/connectors/connect-baseten/src/utils/config.ts +103 -0
  26. package/connectors/connect-baseten/src/utils/output.ts +119 -0
  27. package/connectors/connect-baseten/tsconfig.json +16 -0
  28. package/connectors/connect-cerebras/.env.example +11 -0
  29. package/connectors/connect-cerebras/CLAUDE.md +128 -0
  30. package/connectors/connect-cerebras/README.md +193 -0
  31. package/connectors/connect-cerebras/package.json +51 -0
  32. package/connectors/connect-cerebras/src/api/client.ts +64 -0
  33. package/connectors/connect-cerebras/src/api/index.ts +32 -0
  34. package/connectors/connect-cerebras/src/cli/index.ts +244 -0
  35. package/connectors/connect-cerebras/src/index.ts +19 -0
  36. package/connectors/connect-cerebras/src/types/index.ts +65 -0
  37. package/connectors/connect-cerebras/src/utils/config.ts +103 -0
  38. package/connectors/connect-cerebras/src/utils/output.ts +119 -0
  39. package/connectors/connect-cerebras/tsconfig.json +16 -0
  40. package/connectors/connect-cohere/.env.example +11 -0
  41. package/connectors/connect-cohere/CLAUDE.md +128 -0
  42. package/connectors/connect-cohere/README.md +193 -0
  43. package/connectors/connect-cohere/package.json +53 -0
  44. package/connectors/connect-cohere/src/api/client.ts +109 -0
  45. package/connectors/connect-cohere/src/api/index.ts +59 -0
  46. package/connectors/connect-cohere/src/cli/index.ts +255 -0
  47. package/connectors/connect-cohere/src/index.ts +19 -0
  48. package/connectors/connect-cohere/src/types/index.ts +132 -0
  49. package/connectors/connect-cohere/src/utils/config.ts +197 -0
  50. package/connectors/connect-cohere/src/utils/output.ts +119 -0
  51. package/connectors/connect-cohere/tsconfig.json +16 -0
  52. package/connectors/connect-deepgram/.env.example +11 -0
  53. package/connectors/connect-deepgram/CLAUDE.md +128 -0
  54. package/connectors/connect-deepgram/README.md +193 -0
  55. package/connectors/connect-deepgram/package.json +51 -0
  56. package/connectors/connect-deepgram/src/api/client.ts +235 -0
  57. package/connectors/connect-deepgram/src/api/index.ts +57 -0
  58. package/connectors/connect-deepgram/src/cli/index.ts +339 -0
  59. package/connectors/connect-deepgram/src/index.ts +19 -0
  60. package/connectors/connect-deepgram/src/types/index.ts +232 -0
  61. package/connectors/connect-deepgram/src/utils/config.ts +103 -0
  62. package/connectors/connect-deepgram/src/utils/output.ts +119 -0
  63. package/connectors/connect-deepgram/tsconfig.json +16 -0
  64. package/connectors/connect-deepseek/.env.example +11 -0
  65. package/connectors/connect-deepseek/CLAUDE.md +128 -0
  66. package/connectors/connect-deepseek/README.md +193 -0
  67. package/connectors/connect-deepseek/package.json +51 -0
  68. package/connectors/connect-deepseek/src/api/client.ts +108 -0
  69. package/connectors/connect-deepseek/src/api/index.ts +36 -0
  70. package/connectors/connect-deepseek/src/cli/index.ts +167 -0
  71. package/connectors/connect-deepseek/src/index.ts +19 -0
  72. package/connectors/connect-deepseek/src/types/index.ts +72 -0
  73. package/connectors/connect-deepseek/src/utils/config.ts +103 -0
  74. package/connectors/connect-deepseek/src/utils/output.ts +119 -0
  75. package/connectors/connect-deepseek/tsconfig.json +16 -0
  76. package/connectors/connect-fal/.env.example +11 -0
  77. package/connectors/connect-fal/CLAUDE.md +128 -0
  78. package/connectors/connect-fal/README.md +193 -0
  79. package/connectors/connect-fal/package.json +51 -0
  80. package/connectors/connect-fal/src/api/client.ts +172 -0
  81. package/connectors/connect-fal/src/api/index.ts +55 -0
  82. package/connectors/connect-fal/src/cli/index.ts +341 -0
  83. package/connectors/connect-fal/src/index.ts +19 -0
  84. package/connectors/connect-fal/src/types/index.ts +135 -0
  85. package/connectors/connect-fal/src/utils/config.ts +103 -0
  86. package/connectors/connect-fal/src/utils/output.ts +119 -0
  87. package/connectors/connect-fal/tsconfig.json +16 -0
  88. package/connectors/connect-fireworks/.env.example +11 -0
  89. package/connectors/connect-fireworks/CLAUDE.md +128 -0
  90. package/connectors/connect-fireworks/README.md +193 -0
  91. package/connectors/connect-fireworks/package.json +51 -0
  92. package/connectors/connect-fireworks/src/api/client.ts +63 -0
  93. package/connectors/connect-fireworks/src/api/index.ts +36 -0
  94. package/connectors/connect-fireworks/src/cli/index.ts +244 -0
  95. package/connectors/connect-fireworks/src/index.ts +19 -0
  96. package/connectors/connect-fireworks/src/types/index.ts +70 -0
  97. package/connectors/connect-fireworks/src/utils/config.ts +103 -0
  98. package/connectors/connect-fireworks/src/utils/output.ts +119 -0
  99. package/connectors/connect-fireworks/tsconfig.json +16 -0
  100. package/connectors/connect-groq/.env.example +11 -0
  101. package/connectors/connect-groq/CLAUDE.md +128 -0
  102. package/connectors/connect-groq/README.md +193 -0
  103. package/connectors/connect-groq/package.json +52 -0
  104. package/connectors/connect-groq/src/api/client.ts +108 -0
  105. package/connectors/connect-groq/src/api/index.ts +36 -0
  106. package/connectors/connect-groq/src/cli/index.ts +171 -0
  107. package/connectors/connect-groq/src/index.ts +19 -0
  108. package/connectors/connect-groq/src/types/index.ts +69 -0
  109. package/connectors/connect-groq/src/utils/config.ts +103 -0
  110. package/connectors/connect-groq/src/utils/output.ts +119 -0
  111. package/connectors/connect-groq/tsconfig.json +16 -0
  112. package/connectors/connect-luma/.env.example +11 -0
  113. package/connectors/connect-luma/CLAUDE.md +128 -0
  114. package/connectors/connect-luma/README.md +193 -0
  115. package/connectors/connect-luma/package.json +53 -0
  116. package/connectors/connect-luma/src/api/client.ts +85 -0
  117. package/connectors/connect-luma/src/api/index.ts +44 -0
  118. package/connectors/connect-luma/src/cli/index.ts +300 -0
  119. package/connectors/connect-luma/src/index.ts +19 -0
  120. package/connectors/connect-luma/src/types/index.ts +60 -0
  121. package/connectors/connect-luma/src/utils/config.ts +103 -0
  122. package/connectors/connect-luma/src/utils/output.ts +119 -0
  123. package/connectors/connect-luma/tsconfig.json +16 -0
  124. package/connectors/connect-modal/.env.example +11 -0
  125. package/connectors/connect-modal/CLAUDE.md +128 -0
  126. package/connectors/connect-modal/README.md +193 -0
  127. package/connectors/connect-modal/package.json +51 -0
  128. package/connectors/connect-modal/src/api/client.ts +119 -0
  129. package/connectors/connect-modal/src/api/index.ts +69 -0
  130. package/connectors/connect-modal/src/cli/index.ts +224 -0
  131. package/connectors/connect-modal/src/index.ts +21 -0
  132. package/connectors/connect-modal/src/types/index.ts +60 -0
  133. package/connectors/connect-modal/src/utils/config.ts +114 -0
  134. package/connectors/connect-modal/src/utils/output.ts +119 -0
  135. package/connectors/connect-modal/tsconfig.json +16 -0
  136. package/connectors/connect-perplexity/.env.example +4 -0
  137. package/connectors/connect-perplexity/CLAUDE.md +156 -0
  138. package/connectors/connect-perplexity/README.md +184 -0
  139. package/connectors/connect-perplexity/package.json +58 -0
  140. package/connectors/connect-perplexity/scripts/publish.ts +210 -0
  141. package/connectors/connect-perplexity/src/api/client.ts +119 -0
  142. package/connectors/connect-perplexity/src/api/example.ts +118 -0
  143. package/connectors/connect-perplexity/src/api/index.ts +48 -0
  144. package/connectors/connect-perplexity/src/cli/index.ts +421 -0
  145. package/connectors/connect-perplexity/src/index.ts +24 -0
  146. package/connectors/connect-perplexity/src/types/index.ts +140 -0
  147. package/connectors/connect-perplexity/src/utils/config.ts +208 -0
  148. package/connectors/connect-perplexity/src/utils/output.ts +119 -0
  149. package/connectors/connect-perplexity/tsconfig.json +16 -0
  150. package/connectors/connect-replicate/.env.example +11 -0
  151. package/connectors/connect-replicate/CLAUDE.md +128 -0
  152. package/connectors/connect-replicate/README.md +193 -0
  153. package/connectors/connect-replicate/package.json +51 -0
  154. package/connectors/connect-replicate/src/api/client.ts +109 -0
  155. package/connectors/connect-replicate/src/api/index.ts +71 -0
  156. package/connectors/connect-replicate/src/cli/index.ts +250 -0
  157. package/connectors/connect-replicate/src/index.ts +19 -0
  158. package/connectors/connect-replicate/src/types/index.ts +85 -0
  159. package/connectors/connect-replicate/src/utils/config.ts +103 -0
  160. package/connectors/connect-replicate/src/utils/output.ts +119 -0
  161. package/connectors/connect-replicate/tsconfig.json +16 -0
  162. package/connectors/connect-roboflow/.env.example +11 -0
  163. package/connectors/connect-roboflow/CLAUDE.md +272 -0
  164. package/connectors/connect-roboflow/README.md +193 -0
  165. package/connectors/connect-roboflow/package.json +51 -0
  166. package/connectors/connect-roboflow/scripts/release.ts +179 -0
  167. package/connectors/connect-roboflow/src/api/client.ts +213 -0
  168. package/connectors/connect-roboflow/src/api/example.ts +48 -0
  169. package/connectors/connect-roboflow/src/api/index.ts +51 -0
  170. package/connectors/connect-roboflow/src/cli/index.ts +254 -0
  171. package/connectors/connect-roboflow/src/index.ts +103 -0
  172. package/connectors/connect-roboflow/src/types/index.ts +237 -0
  173. package/connectors/connect-roboflow/src/utils/auth.ts +274 -0
  174. package/connectors/connect-roboflow/src/utils/bulk.ts +212 -0
  175. package/connectors/connect-roboflow/src/utils/config.ts +326 -0
  176. package/connectors/connect-roboflow/src/utils/output.ts +175 -0
  177. package/connectors/connect-roboflow/src/utils/settings.ts +114 -0
  178. package/connectors/connect-roboflow/src/utils/storage.ts +198 -0
  179. package/connectors/connect-roboflow/tsconfig.json +16 -0
  180. package/connectors/connect-runway/.env.example +11 -0
  181. package/connectors/connect-runway/CLAUDE.md +128 -0
  182. package/connectors/connect-runway/README.md +193 -0
  183. package/connectors/connect-runway/package.json +52 -0
  184. package/connectors/connect-runway/src/api/client.ts +78 -0
  185. package/connectors/connect-runway/src/api/index.ts +40 -0
  186. package/connectors/connect-runway/src/cli/index.ts +283 -0
  187. package/connectors/connect-runway/src/index.ts +19 -0
  188. package/connectors/connect-runway/src/types/index.ts +52 -0
  189. package/connectors/connect-runway/src/utils/config.ts +103 -0
  190. package/connectors/connect-runway/src/utils/output.ts +119 -0
  191. package/connectors/connect-runway/tsconfig.json +16 -0
  192. package/connectors/connect-together/.env.example +11 -0
  193. package/connectors/connect-together/CLAUDE.md +128 -0
  194. package/connectors/connect-together/README.md +193 -0
  195. package/connectors/connect-together/package.json +52 -0
  196. package/connectors/connect-together/src/api/client.ts +106 -0
  197. package/connectors/connect-together/src/api/index.ts +47 -0
  198. package/connectors/connect-together/src/cli/index.ts +228 -0
  199. package/connectors/connect-together/src/index.ts +19 -0
  200. package/connectors/connect-together/src/types/index.ts +91 -0
  201. package/connectors/connect-together/src/utils/config.ts +142 -0
  202. package/connectors/connect-together/src/utils/output.ts +119 -0
  203. package/connectors/connect-together/tsconfig.json +16 -0
  204. package/dist/index.js +112 -0
  205. package/package.json +1 -1
@@ -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();
@@ -0,0 +1,19 @@
1
+ // AssemblyAI Connector
2
+ // TypeScript wrapper for AssemblyAI speech-to-text API
3
+
4
+ export { AssemblyAI } from './api';
5
+ export * from './types';
6
+ export { AssemblyAIClient } 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';
@@ -0,0 +1,277 @@
1
+ // AssemblyAI Connector Types
2
+
3
+ // ============================================
4
+ // Configuration
5
+ // ============================================
6
+
7
+ export interface AssemblyAIConfig {
8
+ apiKey: string;
9
+ baseUrl?: string;
10
+ }
11
+
12
+ // ============================================
13
+ // Transcript Types
14
+ // ============================================
15
+
16
+ export type TranscriptStatus = 'queued' | 'processing' | 'completed' | 'error';
17
+
18
+ export interface TranscriptRequest {
19
+ audio_url: string;
20
+ language_code?: string;
21
+ punctuate?: boolean;
22
+ format_text?: boolean;
23
+ dual_channel?: boolean;
24
+ webhook_url?: string;
25
+ webhook_auth_header_name?: string;
26
+ webhook_auth_header_value?: string;
27
+ auto_highlights?: boolean;
28
+ audio_start_from?: number;
29
+ audio_end_at?: number;
30
+ word_boost?: string[];
31
+ boost_param?: 'low' | 'default' | 'high';
32
+ filter_profanity?: boolean;
33
+ redact_pii?: boolean;
34
+ redact_pii_audio?: boolean;
35
+ redact_pii_policies?: string[];
36
+ redact_pii_sub?: 'entity_name' | 'hash';
37
+ speaker_labels?: boolean;
38
+ speakers_expected?: number;
39
+ content_safety?: boolean;
40
+ iab_categories?: boolean;
41
+ custom_spelling?: Array<{ from: string[]; to: string }>;
42
+ disfluencies?: boolean;
43
+ sentiment_analysis?: boolean;
44
+ auto_chapters?: boolean;
45
+ entity_detection?: boolean;
46
+ speech_threshold?: number;
47
+ summarization?: boolean;
48
+ summary_model?: 'informative' | 'conversational' | 'catchy';
49
+ summary_type?: 'bullets' | 'bullets_verbose' | 'gist' | 'headline' | 'paragraph';
50
+ language_detection?: boolean;
51
+ }
52
+
53
+ export interface Word {
54
+ text: string;
55
+ start: number;
56
+ end: number;
57
+ confidence: number;
58
+ speaker?: string;
59
+ }
60
+
61
+ export interface Utterance {
62
+ confidence: number;
63
+ end: number;
64
+ speaker: string;
65
+ start: number;
66
+ text: string;
67
+ words: Word[];
68
+ }
69
+
70
+ export interface Chapter {
71
+ gist: string;
72
+ headline: string;
73
+ start: number;
74
+ end: number;
75
+ summary: string;
76
+ }
77
+
78
+ export interface Entity {
79
+ entity_type: string;
80
+ text: string;
81
+ start: number;
82
+ end: number;
83
+ }
84
+
85
+ export interface SentimentResult {
86
+ text: string;
87
+ start: number;
88
+ end: number;
89
+ sentiment: 'POSITIVE' | 'NEGATIVE' | 'NEUTRAL';
90
+ confidence: number;
91
+ speaker?: string;
92
+ }
93
+
94
+ export interface Transcript {
95
+ id: string;
96
+ status: TranscriptStatus;
97
+ audio_url: string;
98
+ text?: string;
99
+ words?: Word[];
100
+ utterances?: Utterance[];
101
+ confidence?: number;
102
+ audio_duration?: number;
103
+ punctuate?: boolean;
104
+ format_text?: boolean;
105
+ dual_channel?: boolean;
106
+ webhook_url?: string;
107
+ webhook_status_code?: number;
108
+ webhook_auth?: boolean;
109
+ auto_highlights?: boolean;
110
+ auto_highlights_result?: {
111
+ status: string;
112
+ results: Array<{
113
+ count: number;
114
+ rank: number;
115
+ text: string;
116
+ timestamps: Array<{ start: number; end: number }>;
117
+ }>;
118
+ };
119
+ audio_start_from?: number;
120
+ audio_end_at?: number;
121
+ word_boost?: string[];
122
+ boost_param?: string;
123
+ filter_profanity?: boolean;
124
+ redact_pii?: boolean;
125
+ redact_pii_audio?: boolean;
126
+ redact_pii_audio_quality?: string;
127
+ redact_pii_policies?: string[];
128
+ redact_pii_sub?: string;
129
+ speaker_labels?: boolean;
130
+ speakers_expected?: number;
131
+ content_safety?: boolean;
132
+ content_safety_labels?: {
133
+ status: string;
134
+ results: Array<{
135
+ text: string;
136
+ labels: Array<{
137
+ label: string;
138
+ confidence: number;
139
+ severity: number;
140
+ }>;
141
+ timestamp: { start: number; end: number };
142
+ }>;
143
+ summary: Record<string, number>;
144
+ severity_score_summary: Record<string, { low: number; medium: number; high: number }>;
145
+ };
146
+ iab_categories?: boolean;
147
+ iab_categories_result?: {
148
+ status: string;
149
+ results: Array<{
150
+ text: string;
151
+ labels: Array<{ relevance: number; label: string }>;
152
+ timestamp: { start: number; end: number };
153
+ }>;
154
+ summary: Record<string, number>;
155
+ };
156
+ language_code?: string;
157
+ custom_spelling?: Array<{ from: string[]; to: string }>;
158
+ disfluencies?: boolean;
159
+ sentiment_analysis?: boolean;
160
+ sentiment_analysis_results?: SentimentResult[];
161
+ auto_chapters?: boolean;
162
+ chapters?: Chapter[];
163
+ entity_detection?: boolean;
164
+ entities?: Entity[];
165
+ speech_threshold?: number;
166
+ summarization?: boolean;
167
+ summary?: string;
168
+ summary_type?: string;
169
+ summary_model?: string;
170
+ language_detection?: boolean;
171
+ language_confidence_threshold?: number;
172
+ language_confidence?: number;
173
+ error?: string;
174
+ }
175
+
176
+ export interface TranscriptListResponse {
177
+ page_details: {
178
+ limit: number;
179
+ result_count: number;
180
+ current_url: string;
181
+ prev_url?: string;
182
+ next_url?: string;
183
+ };
184
+ transcripts: Array<{
185
+ id: string;
186
+ resource_url: string;
187
+ status: TranscriptStatus;
188
+ created: string;
189
+ completed?: string;
190
+ audio_url: string;
191
+ error?: string;
192
+ }>;
193
+ }
194
+
195
+ // ============================================
196
+ // Upload Types
197
+ // ============================================
198
+
199
+ export interface UploadResponse {
200
+ upload_url: string;
201
+ }
202
+
203
+ // ============================================
204
+ // LeMUR Types
205
+ // ============================================
206
+
207
+ export interface LemurTaskRequest {
208
+ transcript_ids: string[];
209
+ prompt: string;
210
+ context?: string;
211
+ final_model?: 'default' | 'basic' | 'assemblyai/mistral-7b';
212
+ max_output_size?: number;
213
+ temperature?: number;
214
+ }
215
+
216
+ export interface LemurResponse {
217
+ request_id: string;
218
+ response: string;
219
+ usage: {
220
+ input_tokens: number;
221
+ output_tokens: number;
222
+ };
223
+ }
224
+
225
+ export interface LemurSummaryRequest {
226
+ transcript_ids: string[];
227
+ context?: string;
228
+ final_model?: string;
229
+ max_output_size?: number;
230
+ answer_format?: string;
231
+ }
232
+
233
+ export interface LemurQuestionAnswerRequest {
234
+ transcript_ids: string[];
235
+ questions: Array<{
236
+ question: string;
237
+ context?: string;
238
+ answer_format?: string;
239
+ answer_options?: string[];
240
+ }>;
241
+ context?: string;
242
+ final_model?: string;
243
+ }
244
+
245
+ export interface LemurQuestionAnswerResponse {
246
+ request_id: string;
247
+ response: Array<{
248
+ question: string;
249
+ answer: string;
250
+ }>;
251
+ usage: {
252
+ input_tokens: number;
253
+ output_tokens: number;
254
+ };
255
+ }
256
+
257
+ // ============================================
258
+ // API Error Types
259
+ // ============================================
260
+
261
+ export interface ApiErrorDetail {
262
+ code: string;
263
+ message: string;
264
+ field?: string;
265
+ }
266
+
267
+ export class AssemblyAIApiError extends Error {
268
+ public readonly statusCode: number;
269
+ public readonly errors?: ApiErrorDetail[];
270
+
271
+ constructor(message: string, statusCode: number, errors?: ApiErrorDetail[]) {
272
+ super(message);
273
+ this.name = 'AssemblyAIApiError';
274
+ this.statusCode = statusCode;
275
+ this.errors = errors;
276
+ }
277
+ }