@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.
Files changed (205) hide show
  1. package/bin/index.js +113 -1
  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,193 @@
1
+ # scaffold-connector
2
+
3
+ A TypeScript scaffold for building API connector CLIs with multi-profile support.
4
+
5
+ ## Features
6
+
7
+ - Multi-profile configuration (switch between different API keys/accounts)
8
+ - Bearer token authentication (easily customizable)
9
+ - Clean CLI structure with Commander.js
10
+ - Pretty and JSON output formats
11
+ - TypeScript with strict mode
12
+
13
+ ## Quick Start
14
+
15
+ ### 1. Clone and Rename
16
+
17
+ ```bash
18
+ # Clone for your connector
19
+ git clone https://github.com/hasna/scaffold-connector.git connect-yourapi
20
+ cd connect-yourapi
21
+
22
+ # Update package.json name
23
+ # Change "@hasna/scaffold-connector" to "@hasna/connect-yourapi"
24
+ ```
25
+
26
+ ### 2. Update Configuration
27
+
28
+ Search for `TODO` comments throughout the codebase and update:
29
+
30
+ - `src/cli/index.ts` - Update `CONNECTOR_NAME` and description
31
+ - `src/utils/config.ts` - Update `CONNECTOR_NAME` and env var names
32
+ - `src/api/client.ts` - Update `DEFAULT_BASE_URL` and authentication method
33
+ - `src/api/index.ts` - Rename `Connector` class to your API name
34
+ - `src/types/index.ts` - Add your API's type definitions
35
+ - `package.json` - Update name, description, bin command
36
+ - `.env.example` - Update environment variable names
37
+
38
+ ### 3. Install and Test
39
+
40
+ ```bash
41
+ # Install dependencies
42
+ bun install
43
+
44
+ # Run CLI
45
+ bun run dev
46
+
47
+ # Or run specific commands
48
+ bun run dev profile list
49
+ bun run dev config show
50
+ ```
51
+
52
+ ## CLI Structure
53
+
54
+ ```bash
55
+ connector [options] [command]
56
+
57
+ Options:
58
+ -k, --api-key <key> API key (overrides config)
59
+ -f, --format <format> Output format (json, pretty)
60
+ -p, --profile <profile> Use a specific profile
61
+
62
+ Commands:
63
+ profile list List all profiles
64
+ profile use <name> Switch to a profile
65
+ profile create <name> Create a new profile
66
+ profile delete <name> Delete a profile
67
+ profile show [name] Show profile configuration
68
+
69
+ config set-key <key> Set API key for active profile
70
+ config show Show current configuration
71
+ config clear Clear configuration
72
+
73
+ example list Example API command (replace)
74
+ example get <id> Example API command (replace)
75
+ example create Example API command (replace)
76
+ ```
77
+
78
+ ## Project Structure
79
+
80
+ ```
81
+ src/
82
+ ├── api/
83
+ │ ├── client.ts # HTTP client with authentication
84
+ │ ├── example.ts # Example API module (replace with your API)
85
+ │ └── index.ts # Main connector class
86
+ ├── cli/
87
+ │ └── index.ts # CLI commands
88
+ ├── types/
89
+ │ └── index.ts # Type definitions
90
+ ├── utils/
91
+ │ ├── config.ts # Multi-profile configuration
92
+ │ └── output.ts # CLI output formatting
93
+ └── index.ts # Library exports
94
+ ```
95
+
96
+ ## Multi-Profile Configuration
97
+
98
+ Profiles are stored in `~/.connect/{connector-name}/profiles/`:
99
+
100
+ ```
101
+ ~/.connect/connector/
102
+ ├── current_profile # Name of active profile
103
+ └── profiles/
104
+ ├── default.json # Default profile
105
+ ├── work.json # Named profile
106
+ └── personal.json # Named profile
107
+ ```
108
+
109
+ ### Profile Commands
110
+
111
+ ```bash
112
+ # Create profiles
113
+ connector profile create work --api-key sk-xxx --use
114
+ connector profile create personal --api-key sk-yyy
115
+
116
+ # Switch profiles
117
+ connector profile use work
118
+
119
+ # Use profile for single command
120
+ connector -p personal example list
121
+
122
+ # List profiles
123
+ connector profile list
124
+ ```
125
+
126
+ ## Customizing Authentication
127
+
128
+ Edit `src/api/client.ts` to change authentication:
129
+
130
+ ```typescript
131
+ // Bearer token (default)
132
+ 'Authorization': `Bearer ${this.apiKey}`,
133
+
134
+ // API Key header
135
+ 'X-API-Key': this.apiKey,
136
+
137
+ // Basic auth
138
+ 'Authorization': `Basic ${Buffer.from(`${this.apiKey}:${this.apiSecret}`).toString('base64')}`,
139
+ ```
140
+
141
+ ## Adding API Endpoints
142
+
143
+ 1. Create a new file in `src/api/` (e.g., `users.ts`)
144
+ 2. Export it from `src/api/index.ts`
145
+ 3. Add types in `src/types/index.ts`
146
+ 4. Add CLI commands in `src/cli/index.ts`
147
+
148
+ Example API module:
149
+
150
+ ```typescript
151
+ // src/api/users.ts
152
+ import type { ConnectorClient } from './client';
153
+
154
+ export class UsersApi {
155
+ constructor(private readonly client: ConnectorClient) {}
156
+
157
+ async list(options?: { limit?: number }) {
158
+ return this.client.get('/users', { limit: options?.limit });
159
+ }
160
+
161
+ async get(id: string) {
162
+ return this.client.get(`/users/${id}`);
163
+ }
164
+ }
165
+ ```
166
+
167
+ ## Environment Variables
168
+
169
+ | Variable | Description |
170
+ |----------|-------------|
171
+ | `CONNECTOR_API_KEY` | API key (overrides profile config) |
172
+ | `CONNECTOR_API_SECRET` | API secret (optional) |
173
+ | `CONNECTOR_BASE_URL` | Override base URL (optional) |
174
+
175
+ ## Development
176
+
177
+ ```bash
178
+ # Install dependencies
179
+ bun install
180
+
181
+ # Run CLI in development
182
+ bun run dev
183
+
184
+ # Build for distribution
185
+ bun run build
186
+
187
+ # Type check
188
+ bun run typecheck
189
+ ```
190
+
191
+ ## License
192
+
193
+ MIT
@@ -0,0 +1,51 @@
1
+ {
2
+ "name": "@hasna/connect-baseten",
3
+ "version": "0.0.1",
4
+ "description": "Baseten connector CLI - ML model deployment platform with multi-profile support",
5
+ "type": "module",
6
+ "bin": {
7
+ "connect-baseten": "./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
+ "baseten",
25
+ "ai",
26
+ "api",
27
+ "connector",
28
+ "cli",
29
+ "typescript",
30
+ "bun",
31
+ "ml",
32
+ "deployment"
33
+ ],
34
+ "author": "Hasna",
35
+ "license": "Apache-2.0",
36
+ "devDependencies": {
37
+ "@types/bun": "latest",
38
+ "typescript": "^5"
39
+ },
40
+ "dependencies": {
41
+ "commander": "^12.1.0",
42
+ "chalk": "^5.3.0"
43
+ },
44
+ "engines": {
45
+ "bun": ">=1.0.0"
46
+ },
47
+ "repository": {
48
+ "type": "git",
49
+ "url": "git+https://github.com/hasna/open-connectors.git"
50
+ }
51
+ }
@@ -0,0 +1,71 @@
1
+ import { BasetenApiError, type BasetenConfig, type ModelsResponse, type PredictionRequest, type PredictionResponse } from '../types';
2
+
3
+ const DEFAULT_BASE_URL = 'https://api.baseten.co/v1';
4
+
5
+ export class BasetenClient {
6
+ private readonly apiKey: string;
7
+ private readonly baseUrl: string;
8
+
9
+ constructor(config: BasetenConfig) {
10
+ if (!config.apiKey) {
11
+ throw new Error('API key is required');
12
+ }
13
+ this.apiKey = config.apiKey;
14
+ this.baseUrl = config.baseUrl || DEFAULT_BASE_URL;
15
+ }
16
+
17
+ private async request<T>(
18
+ endpoint: string,
19
+ options: RequestInit = {}
20
+ ): Promise<T> {
21
+ const url = `${this.baseUrl}${endpoint}`;
22
+
23
+ const response = await fetch(url, {
24
+ ...options,
25
+ headers: {
26
+ 'Content-Type': 'application/json',
27
+ 'Authorization': `Api-Key ${this.apiKey}`,
28
+ ...options.headers,
29
+ },
30
+ });
31
+
32
+ const data = await response.json().catch(() => null);
33
+
34
+ if (!response.ok) {
35
+ let errorMessage = String(data || response.statusText);
36
+ if (typeof data === 'object' && data !== null) {
37
+ const errorObj = data as { error?: { message?: string }; message?: string; detail?: string };
38
+ errorMessage = errorObj.error?.message || errorObj.message || errorObj.detail || JSON.stringify(data);
39
+ }
40
+ throw new BasetenApiError(errorMessage, response.status);
41
+ }
42
+
43
+ return data as T;
44
+ }
45
+
46
+ async listModels(): Promise<ModelsResponse> {
47
+ return this.request<ModelsResponse>('/models');
48
+ }
49
+
50
+ async getModel(modelId: string): Promise<{ model: unknown }> {
51
+ return this.request<{ model: unknown }>(`/models/${modelId}`);
52
+ }
53
+
54
+ async predict(request: PredictionRequest): Promise<PredictionResponse> {
55
+ const { model_id, version_id, input } = request;
56
+ const endpoint = version_id
57
+ ? `/models/${model_id}/versions/${version_id}/predict`
58
+ : `/models/${model_id}/predict`;
59
+ return this.request<PredictionResponse>(endpoint, {
60
+ method: 'POST',
61
+ body: JSON.stringify(input),
62
+ });
63
+ }
64
+
65
+ getApiKeyPreview(): string {
66
+ if (this.apiKey.length > 10) {
67
+ return `${this.apiKey.substring(0, 6)}...${this.apiKey.substring(this.apiKey.length - 4)}`;
68
+ }
69
+ return '***';
70
+ }
71
+ }
@@ -0,0 +1,40 @@
1
+ import type { BasetenConfig, ModelsResponse, PredictionRequest, PredictionResponse } from '../types';
2
+ import { BasetenClient } from './client';
3
+
4
+ export class Baseten {
5
+ private readonly client: BasetenClient;
6
+
7
+ constructor(config: BasetenConfig) {
8
+ this.client = new BasetenClient(config);
9
+ }
10
+
11
+ static fromEnv(): Baseten {
12
+ const apiKey = process.env.BASETEN_API_KEY;
13
+ if (!apiKey) {
14
+ throw new Error('BASETEN_API_KEY environment variable is required');
15
+ }
16
+ return new Baseten({ apiKey });
17
+ }
18
+
19
+ async listModels(): Promise<ModelsResponse> {
20
+ return this.client.listModels();
21
+ }
22
+
23
+ async getModel(modelId: string): Promise<{ model: unknown }> {
24
+ return this.client.getModel(modelId);
25
+ }
26
+
27
+ async predict(request: PredictionRequest): Promise<PredictionResponse> {
28
+ return this.client.predict(request);
29
+ }
30
+
31
+ getApiKeyPreview(): string {
32
+ return this.client.getApiKeyPreview();
33
+ }
34
+
35
+ getClient(): BasetenClient {
36
+ return this.client;
37
+ }
38
+ }
39
+
40
+ export { BasetenClient } from './client';
@@ -0,0 +1,244 @@
1
+ #!/usr/bin/env bun
2
+ import { Command } from 'commander';
3
+ import chalk from 'chalk';
4
+ import { Baseten } 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-baseten';
23
+ const VERSION = '0.0.1';
24
+
25
+ const program = new Command();
26
+
27
+ program
28
+ .name(CONNECTOR_NAME)
29
+ .description('Baseten connector CLI - ML model deployment platform 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.BASETEN_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(): Baseten {
54
+ const apiKey = getApiKey();
55
+ if (!apiKey) {
56
+ error(`No API key configured. Run "${CONNECTOR_NAME} config set-key <key>" or set BASETEN_API_KEY environment variable.`);
57
+ process.exit(1);
58
+ }
59
+ return new Baseten({ 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
+ // Models Commands
179
+ // ============================================
180
+ const modelsCmd = program
181
+ .command('models')
182
+ .description('Manage Baseten models');
183
+
184
+ modelsCmd
185
+ .command('list')
186
+ .description('List deployed models')
187
+ .action(async () => {
188
+ try {
189
+ const client = getClient();
190
+ const result = await client.listModels();
191
+ print(result, getFormat(modelsCmd));
192
+ } catch (err) {
193
+ error(String(err));
194
+ process.exit(1);
195
+ }
196
+ });
197
+
198
+ modelsCmd
199
+ .command('get <modelId>')
200
+ .description('Get model details')
201
+ .action(async (modelId: string) => {
202
+ try {
203
+ const client = getClient();
204
+ const result = await client.getModel(modelId);
205
+ print(result, getFormat(modelsCmd));
206
+ } catch (err) {
207
+ error(String(err));
208
+ process.exit(1);
209
+ }
210
+ });
211
+
212
+ // ============================================
213
+ // Predict Commands
214
+ // ============================================
215
+ const predictCmd = program
216
+ .command('predict')
217
+ .description('Run predictions on models');
218
+
219
+ predictCmd
220
+ .command('run <modelId>')
221
+ .description('Run a prediction')
222
+ .requiredOption('-i, --input <json>', 'Input data as JSON')
223
+ .option('-v, --version <versionId>', 'Specific model version')
224
+ .action(async (modelId: string, opts) => {
225
+ try {
226
+ const client = getClient();
227
+ const input = JSON.parse(opts.input);
228
+ const result = await client.predict({
229
+ model_id: modelId,
230
+ input,
231
+ version_id: opts.version,
232
+ });
233
+ print(result, getFormat(predictCmd));
234
+ } catch (err) {
235
+ if (err instanceof SyntaxError) {
236
+ error('Invalid JSON input');
237
+ } else {
238
+ error(String(err));
239
+ }
240
+ process.exit(1);
241
+ }
242
+ });
243
+
244
+ program.parse();
@@ -0,0 +1,19 @@
1
+ // Baseten Connector
2
+ // TypeScript wrapper for Baseten API
3
+
4
+ export { Baseten } from './api';
5
+ export * from './types';
6
+ export { BasetenClient } 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,55 @@
1
+ // Baseten API Types
2
+
3
+ export interface BasetenConfig {
4
+ apiKey: string;
5
+ baseUrl?: string;
6
+ }
7
+
8
+ // Model Types
9
+ export interface BasetenModel {
10
+ id: string;
11
+ name: string;
12
+ created_at?: string;
13
+ updated_at?: string;
14
+ status?: string;
15
+ }
16
+
17
+ export interface ModelsResponse {
18
+ models: BasetenModel[];
19
+ }
20
+
21
+ // Prediction Types
22
+ export interface PredictionRequest {
23
+ model_id: string;
24
+ input: Record<string, unknown>;
25
+ version_id?: string;
26
+ }
27
+
28
+ export interface PredictionResponse {
29
+ id: string;
30
+ model_id: string;
31
+ status: string;
32
+ output?: unknown;
33
+ created_at?: string;
34
+ completed_at?: string;
35
+ }
36
+
37
+ // Deployment Types
38
+ export interface DeploymentStatus {
39
+ id: string;
40
+ model_id: string;
41
+ status: string;
42
+ created_at?: string;
43
+ }
44
+
45
+ // Error Types
46
+ export class BasetenApiError extends Error {
47
+ constructor(
48
+ message: string,
49
+ public readonly status: number,
50
+ public readonly code?: string
51
+ ) {
52
+ super(message);
53
+ this.name = 'BasetenApiError';
54
+ }
55
+ }