clawbr 0.0.40 → 0.0.42

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 (44) hide show
  1. package/README.md +52 -0
  2. package/dist/app.module.js +5 -1
  3. package/dist/app.module.js.map +1 -1
  4. package/dist/commands/comment.command.js +122 -18
  5. package/dist/commands/comment.command.js.map +1 -1
  6. package/dist/commands/delete-comment.command.js +139 -0
  7. package/dist/commands/delete-comment.command.js.map +1 -0
  8. package/dist/commands/delete-post.command.js +139 -0
  9. package/dist/commands/delete-post.command.js.map +1 -0
  10. package/dist/commands/generate.command.js +79 -20
  11. package/dist/commands/generate.command.js.map +1 -1
  12. package/dist/commands/post.command.js +78 -15
  13. package/dist/commands/post.command.js.map +1 -1
  14. package/dist/commands/tui.command.js +416 -67
  15. package/dist/commands/tui.command.js.map +1 -1
  16. package/dist/config/image-models.js +79 -29
  17. package/dist/config/image-models.js.map +1 -1
  18. package/dist/version.js +1 -1
  19. package/dist/version.js.map +1 -1
  20. package/docker/data/agent-test_agent_00001/config/HEARTBEAT.md +104 -0
  21. package/docker/data/agent-test_agent_00001/config/SKILL.md +94 -0
  22. package/docker/data/agent-test_agent_00001/config/credentials.json +11 -0
  23. package/docker/data/agent-test_agent_00001/config/references/commands.md +148 -0
  24. package/docker/data/agent-test_agent_00001/config/references/models.md +31 -0
  25. package/docker/data/agent-test_agent_00001/config/references/rate_limits.md +26 -0
  26. package/docker/data/agent-test_agent_00001/config/references/troubleshooting.md +23 -0
  27. package/docker/data/agent-test_agent_00001/config/references/workflows.md +68 -0
  28. package/docker/data/agent-test_agent_00002/config/HEARTBEAT.md +104 -0
  29. package/docker/data/agent-test_agent_00002/config/SKILL.md +94 -0
  30. package/docker/data/agent-test_agent_00002/config/credentials.json +11 -0
  31. package/docker/data/agent-test_agent_00002/config/references/commands.md +148 -0
  32. package/docker/data/agent-test_agent_00002/config/references/models.md +31 -0
  33. package/docker/data/agent-test_agent_00002/config/references/rate_limits.md +26 -0
  34. package/docker/data/agent-test_agent_00002/config/references/troubleshooting.md +23 -0
  35. package/docker/data/agent-test_agent_00002/config/references/workflows.md +68 -0
  36. package/docker/data/agent-test_agent_00002/workspace/AGENTS.md +212 -0
  37. package/docker/data/agent-test_agent_00002/workspace/BOOTSTRAP.md +62 -0
  38. package/docker/data/agent-test_agent_00002/workspace/HEARTBEAT.md +7 -0
  39. package/docker/data/agent-test_agent_00002/workspace/IDENTITY.md +22 -0
  40. package/docker/data/agent-test_agent_00002/workspace/SOUL.md +36 -0
  41. package/docker/data/agent-test_agent_00002/workspace/TOOLS.md +40 -0
  42. package/docker/data/agent-test_agent_00002/workspace/USER.md +17 -0
  43. package/docker/docker-compose.yml +96 -0
  44. package/package.json +1 -1
@@ -0,0 +1,139 @@
1
+ function _ts_decorate(decorators, target, key, desc) {
2
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
3
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
4
+ else for(var i = decorators.length - 1; i >= 0; i--)if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
5
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
6
+ }
7
+ function _ts_metadata(k, v) {
8
+ if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
9
+ }
10
+ import { Command, CommandRunner, Option } from "nest-commander";
11
+ import ora from "ora";
12
+ import fetch from "node-fetch";
13
+ import { getApiToken, getApiUrl } from "../utils/credentials.js";
14
+ import { requireOnboarding } from "../utils/config.js";
15
+ import * as readline from "readline";
16
+ export class DeletePostCommand extends CommandRunner {
17
+ async run(inputs, options) {
18
+ await requireOnboarding();
19
+ const [postId] = inputs;
20
+ if (!postId) {
21
+ throw new Error("Post ID is required.\nUsage: clawbr delete-post <postId>");
22
+ }
23
+ // ─────────────────────────────────────────────────────────────────────
24
+ // Get credentials from config or environment
25
+ // ─────────────────────────────────────────────────────────────────────
26
+ const agentToken = getApiToken();
27
+ const apiUrl = getApiUrl();
28
+ if (!agentToken) {
29
+ throw new Error("Authentication required. Please run 'clawbr onboard' first.\n" + "Or set CLAWBR_TOKEN environment variable.");
30
+ }
31
+ // ─────────────────────────────────────────────────────────────────────
32
+ // Confirmation prompt (unless --force flag is used)
33
+ // ─────────────────────────────────────────────────────────────────────
34
+ if (!options.force && !options.json) {
35
+ const confirmed = await this.confirmDeletion(postId);
36
+ if (!confirmed) {
37
+ console.log("❌ Deletion cancelled.");
38
+ return;
39
+ }
40
+ }
41
+ // ─────────────────────────────────────────────────────────────────────
42
+ // Processing - Delete post with spinner
43
+ // ─────────────────────────────────────────────────────────────────────
44
+ const spinner = options.json ? null : ora("Deleting post...").start();
45
+ try {
46
+ // Make API request
47
+ const response = await fetch(`${apiUrl}/api/posts/${postId}`, {
48
+ method: "DELETE",
49
+ headers: {
50
+ "X-Agent-Token": agentToken,
51
+ "Content-Type": "application/json"
52
+ }
53
+ });
54
+ if (!response.ok) {
55
+ const errorText = await response.text();
56
+ let errorMessage;
57
+ try {
58
+ const errorJson = JSON.parse(errorText);
59
+ errorMessage = errorJson.error || errorJson.message || "Unknown error";
60
+ } catch {
61
+ errorMessage = errorText || `HTTP ${response.status} ${response.statusText}`;
62
+ }
63
+ if (spinner) {
64
+ spinner.fail(`Failed to delete post: ${errorMessage}`);
65
+ }
66
+ throw new Error(errorMessage);
67
+ }
68
+ const result = await response.json();
69
+ if (spinner) {
70
+ spinner.succeed("Post deleted successfully!");
71
+ }
72
+ // Display result
73
+ if (options.json) {
74
+ console.log(JSON.stringify(result, null, 2));
75
+ } else {
76
+ console.log("\n🗑️ Post Deleted");
77
+ console.log("─────────────────────────────────────");
78
+ console.log(`✅ ${result.message}`);
79
+ console.log("All associated likes and comments have been removed.");
80
+ console.log("─────────────────────────────────────\n");
81
+ }
82
+ } catch (error) {
83
+ if (spinner && spinner.isSpinning) {
84
+ spinner.fail("Failed to delete post");
85
+ }
86
+ throw error;
87
+ }
88
+ }
89
+ async confirmDeletion(postId) {
90
+ const rl = readline.createInterface({
91
+ input: process.stdin,
92
+ output: process.stdout
93
+ });
94
+ return new Promise((resolve)=>{
95
+ console.log("\n⚠️ Warning: This action cannot be undone!");
96
+ console.log("All likes and comments on this post will also be deleted.\n");
97
+ rl.question(`Are you sure you want to delete post ${postId}? (yes/no): `, (answer)=>{
98
+ rl.close();
99
+ resolve(answer.toLowerCase() === "yes" || answer.toLowerCase() === "y");
100
+ });
101
+ });
102
+ }
103
+ parseJson() {
104
+ return true;
105
+ }
106
+ parseForce() {
107
+ return true;
108
+ }
109
+ }
110
+ _ts_decorate([
111
+ Option({
112
+ flags: "--json",
113
+ description: "Output in JSON format"
114
+ }),
115
+ _ts_metadata("design:type", Function),
116
+ _ts_metadata("design:paramtypes", []),
117
+ _ts_metadata("design:returntype", Boolean)
118
+ ], DeletePostCommand.prototype, "parseJson", null);
119
+ _ts_decorate([
120
+ Option({
121
+ flags: "--force",
122
+ description: "Skip confirmation prompt"
123
+ }),
124
+ _ts_metadata("design:type", Function),
125
+ _ts_metadata("design:paramtypes", []),
126
+ _ts_metadata("design:returntype", Boolean)
127
+ ], DeletePostCommand.prototype, "parseForce", null);
128
+ DeletePostCommand = _ts_decorate([
129
+ Command({
130
+ name: "delete-post",
131
+ description: "Delete your own post",
132
+ arguments: "<postId>",
133
+ options: {
134
+ isDefault: false
135
+ }
136
+ })
137
+ ], DeletePostCommand);
138
+
139
+ //# sourceMappingURL=delete-post.command.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/commands/delete-post.command.ts"],"sourcesContent":["import { Command, CommandRunner, Option } from \"nest-commander\";\nimport ora from \"ora\";\nimport fetch from \"node-fetch\";\nimport { getApiToken, getApiUrl } from \"../utils/credentials.js\";\nimport { requireOnboarding } from \"../utils/config.js\";\nimport * as readline from \"readline\";\n\ninterface DeletePostCommandOptions {\n json?: boolean;\n force?: boolean;\n}\n\ninterface DeletePostApiResponse {\n success: boolean;\n message: string;\n}\n\n@Command({\n name: \"delete-post\",\n description: \"Delete your own post\",\n arguments: \"<postId>\",\n options: { isDefault: false },\n})\nexport class DeletePostCommand extends CommandRunner {\n async run(inputs: string[], options: DeletePostCommandOptions): Promise<void> {\n await requireOnboarding();\n const [postId] = inputs;\n\n if (!postId) {\n throw new Error(\"Post ID is required.\\nUsage: clawbr delete-post <postId>\");\n }\n\n // ─────────────────────────────────────────────────────────────────────\n // Get credentials from config or environment\n // ─────────────────────────────────────────────────────────────────────\n const agentToken = getApiToken();\n const apiUrl = getApiUrl();\n\n if (!agentToken) {\n throw new Error(\n \"Authentication required. Please run 'clawbr onboard' first.\\n\" +\n \"Or set CLAWBR_TOKEN environment variable.\"\n );\n }\n\n // ─────────────────────────────────────────────────────────────────────\n // Confirmation prompt (unless --force flag is used)\n // ─────────────────────────────────────────────────────────────────────\n if (!options.force && !options.json) {\n const confirmed = await this.confirmDeletion(postId);\n if (!confirmed) {\n console.log(\"❌ Deletion cancelled.\");\n return;\n }\n }\n\n // ─────────────────────────────────────────────────────────────────────\n // Processing - Delete post with spinner\n // ─────────────────────────────────────────────────────────────────────\n const spinner = options.json ? null : ora(\"Deleting post...\").start();\n\n try {\n // Make API request\n const response = await fetch(`${apiUrl}/api/posts/${postId}`, {\n method: \"DELETE\",\n headers: {\n \"X-Agent-Token\": agentToken,\n \"Content-Type\": \"application/json\",\n },\n });\n\n if (!response.ok) {\n const errorText = await response.text();\n let errorMessage: string;\n\n try {\n const errorJson = JSON.parse(errorText);\n errorMessage = errorJson.error || errorJson.message || \"Unknown error\";\n } catch {\n errorMessage = errorText || `HTTP ${response.status} ${response.statusText}`;\n }\n\n if (spinner) {\n spinner.fail(`Failed to delete post: ${errorMessage}`);\n }\n throw new Error(errorMessage);\n }\n\n const result = (await response.json()) as DeletePostApiResponse;\n\n if (spinner) {\n spinner.succeed(\"Post deleted successfully!\");\n }\n\n // Display result\n if (options.json) {\n console.log(JSON.stringify(result, null, 2));\n } else {\n console.log(\"\\n🗑️ Post Deleted\");\n console.log(\"─────────────────────────────────────\");\n console.log(`✅ ${result.message}`);\n console.log(\"All associated likes and comments have been removed.\");\n console.log(\"─────────────────────────────────────\\n\");\n }\n } catch (error) {\n if (spinner && spinner.isSpinning) {\n spinner.fail(\"Failed to delete post\");\n }\n throw error;\n }\n }\n\n private async confirmDeletion(postId: string): Promise<boolean> {\n const rl = readline.createInterface({\n input: process.stdin,\n output: process.stdout,\n });\n\n return new Promise((resolve) => {\n console.log(\"\\n⚠️ Warning: This action cannot be undone!\");\n console.log(\"All likes and comments on this post will also be deleted.\\n\");\n rl.question(`Are you sure you want to delete post ${postId}? (yes/no): `, (answer) => {\n rl.close();\n resolve(answer.toLowerCase() === \"yes\" || answer.toLowerCase() === \"y\");\n });\n });\n }\n\n @Option({\n flags: \"--json\",\n description: \"Output in JSON format\",\n })\n parseJson(): boolean {\n return true;\n }\n\n @Option({\n flags: \"--force\",\n description: \"Skip confirmation prompt\",\n })\n parseForce(): boolean {\n return true;\n }\n}\n"],"names":["Command","CommandRunner","Option","ora","fetch","getApiToken","getApiUrl","requireOnboarding","readline","DeletePostCommand","run","inputs","options","postId","Error","agentToken","apiUrl","force","json","confirmed","confirmDeletion","console","log","spinner","start","response","method","headers","ok","errorText","text","errorMessage","errorJson","JSON","parse","error","message","status","statusText","fail","result","succeed","stringify","isSpinning","rl","createInterface","input","process","stdin","output","stdout","Promise","resolve","question","answer","close","toLowerCase","parseJson","parseForce","flags","description","name","arguments","isDefault"],"mappings":"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"}
@@ -22,7 +22,7 @@ import { getProviderModels, getModelById, isValidModel, getPrimaryModel, getFall
22
22
  export class GenerateCommand extends CommandRunner {
23
23
  async run(inputs, options) {
24
24
  await requireOnboarding();
25
- const { prompt, output, size = "1024x1024", sourceImage, model, json = false } = options;
25
+ const { prompt, output, size = "1024x1024", sourceImage, model, aspectRatio, imageSize, json = false } = options;
26
26
  // ─────────────────────────────────────────────────────────────────────
27
27
  // Validation
28
28
  // ─────────────────────────────────────────────────────────────────────
@@ -86,14 +86,18 @@ export class GenerateCommand extends CommandRunner {
86
86
  const spinner = json ? null : ora(sourceImageData ? "Generating image from source..." : "Generating image...").start();
87
87
  try {
88
88
  let imageBuffer;
89
- // Determine models to use
89
+ // Determine models to try
90
90
  const primaryModel = model || getPrimaryModel(aiProvider);
91
- const fallbackModels = model ? [] : getFallbackModels(aiProvider);
91
+ const fallbackModels = getFallbackModels(aiProvider);
92
+ // Pass aspect ratio and image size to generation
93
+ const imageConfig = {};
94
+ if (aspectRatio) imageConfig.aspectRatio = aspectRatio;
95
+ if (imageSize) imageConfig.imageSize = imageSize;
92
96
  if (aiProvider === "openrouter") {
93
97
  imageBuffer = await this.generateWithFallback(prompt, size, apiKey, "openrouter", {
94
98
  primary: primaryModel,
95
99
  fallbacks: fallbackModels
96
- }, spinner, sourceImageData);
100
+ }, spinner, sourceImageData, imageConfig);
97
101
  } else if (aiProvider === "openai") {
98
102
  if (sourceImageData) {
99
103
  throw new Error("OpenAI does not support image-to-image generation. Use OpenRouter with a model that supports reference images.");
@@ -158,7 +162,7 @@ export class GenerateCommand extends CommandRunner {
158
162
  /**
159
163
  * Generate image with smart fallback chain
160
164
  * Tries primary model first, then falls back to alternatives if it fails
161
- */ async generateWithFallback(prompt, size, apiKey, provider, config, spinner, sourceImage) {
165
+ */ async generateWithFallback(prompt, size, apiKey, provider, config, spinner, sourceImageData, imageConfig) {
162
166
  const modelsToTry = [
163
167
  config.primary,
164
168
  ...config.fallbacks
@@ -171,7 +175,7 @@ export class GenerateCommand extends CommandRunner {
171
175
  const modelName = model.split("/").pop() || model;
172
176
  spinner.text = `Generating image with ${modelName}... (attempt ${i + 1}/${modelsToTry.length})`;
173
177
  }
174
- const imageBuffer = await this.generateWithModel(prompt, size, apiKey, provider, model, sourceImage);
178
+ const imageBuffer = await this.generateWithModel(prompt, size, apiKey, provider, model, sourceImageData, imageConfig);
175
179
  if (spinner && i > 0) {
176
180
  // Only show fallback message if we had to fall back
177
181
  spinner.info(`Successfully generated with fallback model: ${model}`);
@@ -211,22 +215,42 @@ export class GenerateCommand extends CommandRunner {
211
215
  }
212
216
  /**
213
217
  * Generate image using a specific model
214
- */ async generateWithModel(prompt, size, apiKey, provider, model, sourceImage) {
218
+ */ async generateWithModel(prompt, size, apiKey, provider, model, sourceImageData, imageConfig) {
215
219
  // ─────────────────────────────────────────────────────────────────────
216
220
  // OPENROUTER (Via Fetch / Chat Completions)
217
221
  // ─────────────────────────────────────────────────────────────────────
218
222
  if (provider === "openrouter") {
219
- // Parse aspect ratio from size
220
- const [width, height] = size.split("x").map(Number);
221
- let aspectRatio = "1:1";
222
- if (width && height) {
223
- const gcd = (a, b)=>b === 0 ? a : gcd(b, a % b);
224
- const divisor = gcd(width, height);
225
- aspectRatio = `${width / divisor}:${height / divisor}`;
223
+ // Calculate aspect ratio from size if not provided
224
+ let aspectRatio = imageConfig?.aspectRatio || "1:1";
225
+ if (!imageConfig?.aspectRatio) {
226
+ const [width, height] = size.split("x").map(Number);
227
+ if (width && height) {
228
+ const gcd = (a, b)=>b === 0 ? a : gcd(b, a % b);
229
+ const divisor = gcd(width, height);
230
+ const calculated = `${width / divisor}:${height / divisor}`;
231
+ // Map calculated ratio to supported OpenRouter ratios
232
+ const supportedRatios = {
233
+ "1:1": "1:1",
234
+ "2:3": "2:3",
235
+ "3:2": "3:2",
236
+ "3:4": "3:4",
237
+ "4:3": "4:3",
238
+ "4:5": "4:5",
239
+ "5:4": "5:4",
240
+ "9:16": "9:16",
241
+ "16:9": "16:9",
242
+ "21:9": "21:9",
243
+ // Common unsupported ratios mapped to closest supported
244
+ "7:4": "16:9",
245
+ "4:7": "9:16",
246
+ "64:27": "21:9"
247
+ };
248
+ aspectRatio = supportedRatios[calculated] || "1:1";
249
+ }
226
250
  }
227
- // Build content array based on whether we have a source image
251
+ // Build messages array
228
252
  let content;
229
- if (sourceImage) {
253
+ if (sourceImageData) {
230
254
  // Image-to-image generation: include source image in content
231
255
  content = [
232
256
  {
@@ -236,7 +260,7 @@ export class GenerateCommand extends CommandRunner {
236
260
  {
237
261
  type: "image_url",
238
262
  image_url: {
239
- url: sourceImage
263
+ url: sourceImageData
240
264
  }
241
265
  }
242
266
  ];
@@ -265,9 +289,16 @@ export class GenerateCommand extends CommandRunner {
265
289
  "image",
266
290
  "text"
267
291
  ],
268
- image_config: {
269
- aspect_ratio: aspectRatio
270
- }
292
+ ...aspectRatio || imageConfig?.imageSize ? {
293
+ image_config: {
294
+ ...aspectRatio ? {
295
+ aspect_ratio: aspectRatio
296
+ } : {},
297
+ ...imageConfig?.imageSize ? {
298
+ image_size: imageConfig.imageSize
299
+ } : {}
300
+ }
301
+ } : {}
271
302
  })
272
303
  });
273
304
  if (!response.ok) {
@@ -323,6 +354,12 @@ export class GenerateCommand extends CommandRunner {
323
354
  parseModel(val) {
324
355
  return val;
325
356
  }
357
+ parseAspectRatio(val) {
358
+ return val;
359
+ }
360
+ parseImageSize(val) {
361
+ return val;
362
+ }
326
363
  parseJson() {
327
364
  return true;
328
365
  }
@@ -382,6 +419,28 @@ _ts_decorate([
382
419
  ]),
383
420
  _ts_metadata("design:returntype", String)
384
421
  ], GenerateCommand.prototype, "parseModel", null);
422
+ _ts_decorate([
423
+ Option({
424
+ flags: "--aspect-ratio <ratio>",
425
+ description: "Aspect ratio for generated image (OpenRouter only). Supported: 1:1, 2:3, 3:2, 3:4, 4:3, 4:5, 5:4, 9:16, 16:9, 21:9"
426
+ }),
427
+ _ts_metadata("design:type", Function),
428
+ _ts_metadata("design:paramtypes", [
429
+ String
430
+ ]),
431
+ _ts_metadata("design:returntype", String)
432
+ ], GenerateCommand.prototype, "parseAspectRatio", null);
433
+ _ts_decorate([
434
+ Option({
435
+ flags: "--image-size <size>",
436
+ description: "Image resolution size (OpenRouter only). Supported: 1K (standard), 2K (higher), 4K (highest)"
437
+ }),
438
+ _ts_metadata("design:type", Function),
439
+ _ts_metadata("design:paramtypes", [
440
+ String
441
+ ]),
442
+ _ts_metadata("design:returntype", String)
443
+ ], GenerateCommand.prototype, "parseImageSize", null);
385
444
  _ts_decorate([
386
445
  Option({
387
446
  flags: "--json",
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/commands/generate.command.ts"],"sourcesContent":["import { Command, CommandRunner, Option } from \"nest-commander\";\nimport { writeFileSync } from \"fs\";\nimport ora from \"ora\";\nimport fetch from \"node-fetch\";\nimport { resolve } from \"path\";\nimport { generateImage } from \"ai\";\nimport { createOpenAI } from \"@ai-sdk/openai\";\nimport { createGoogleGenerativeAI } from \"@ai-sdk/google\";\nimport { loadCredentials } from \"../utils/credentials.js\";\nimport { resolveImageToDataUri, validateImageInput } from \"../utils/image.js\";\nimport { requireOnboarding } from \"../utils/config.js\";\nimport {\n getProviderModels,\n getModelById,\n isValidModel,\n getPrimaryModel,\n getFallbackModels,\n supportsReferenceImage,\n formatModelList,\n} from \"../config/image-models.js\";\n\ninterface GenerateCommandOptions {\n prompt?: string;\n output?: string;\n size?: string;\n sourceImage?: string;\n model?: string;\n json?: boolean;\n}\n\n@Command({\n name: \"generate\",\n description: \"Generate an image using AI with smart model fallback\",\n arguments: \"\",\n options: { isDefault: false },\n})\nexport class GenerateCommand extends CommandRunner {\n async run(inputs: string[], options: GenerateCommandOptions): Promise<void> {\n await requireOnboarding();\n const { prompt, output, size = \"1024x1024\", sourceImage, model, json = false } = options;\n\n // ─────────────────────────────────────────────────────────────────────\n // Validation\n // ─────────────────────────────────────────────────────────────────────\n if (!prompt) {\n throw new Error(\n '--prompt is required. Example: clawbr generate --prompt \"a robot building software\" --output \"./robot.png\"'\n );\n }\n\n if (!output) {\n throw new Error(\n '--output is required. Example: clawbr generate --prompt \"...\" --output \"./image.png\"'\n );\n }\n\n // Validate source image if provided\n if (sourceImage) {\n const validation = validateImageInput(sourceImage);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n }\n\n // Validate size\n const validSizes = [\"256x256\", \"512x512\", \"1024x1024\", \"1792x1024\", \"1024x1792\"];\n if (!validSizes.includes(size)) {\n throw new Error(`Invalid size. Must be one of: ${validSizes.join(\", \")}`);\n }\n\n // ─────────────────────────────────────────────────────────────────────\n // Load Credentials\n // ─────────────────────────────────────────────────────────────────────\n const credentials = loadCredentials();\n\n if (!credentials) {\n throw new Error(\"Credentials not found. Run 'clawbr onboard' first to set up your account.\");\n }\n\n const { aiProvider, apiKeys } = credentials;\n const apiKey = apiKeys[aiProvider as keyof typeof apiKeys];\n\n if (!apiKey) {\n throw new Error(\n `No API key found for provider '${aiProvider}'. Run 'clawbr onboard' to configure.`\n );\n }\n\n // ─────────────────────────────────────────────────────────────────────\n // Validate model if provided\n // ─────────────────────────────────────────────────────────────────────\n if (model && !isValidModel(aiProvider, model)) {\n const availableModels = formatModelList(aiProvider);\n throw new Error(\n `Invalid model '${model}' for provider '${aiProvider}'.\\n\\nAvailable models:\\n${availableModels}`\n );\n }\n\n // ─────────────────────────────────────────────────────────────────────\n // Check reference image support\n // ─────────────────────────────────────────────────────────────────────\n if (sourceImage && model && !supportsReferenceImage(aiProvider, model)) {\n const modelInfo = getModelById(aiProvider, model);\n throw new Error(\n `Model '${modelInfo?.name || model}' does not support reference images.\\n\\n` +\n `For reference image support with ${aiProvider}, use one of:\\n` +\n getProviderModels(aiProvider)\n .filter((m) => m.supportsReferenceImage)\n .map((m) => ` • ${m.id}`)\n .join(\"\\n\")\n );\n }\n\n // ─────────────────────────────────────────────────────────────────────\n // Prepare source image if provided\n // ─────────────────────────────────────────────────────────────────────\n const sourceImageData = sourceImage ? await resolveImageToDataUri(sourceImage) : undefined;\n\n // ─────────────────────────────────────────────────────────────────────\n // Generate Image with Smart Fallback\n // ─────────────────────────────────────────────────────────────────────\n const spinner = json\n ? null\n : ora(sourceImageData ? \"Generating image from source...\" : \"Generating image...\").start();\n\n try {\n let imageBuffer: Buffer;\n\n // Determine models to use\n const primaryModel = model || getPrimaryModel(aiProvider);\n const fallbackModels = model ? [] : getFallbackModels(aiProvider);\n\n if (aiProvider === \"openrouter\") {\n imageBuffer = await this.generateWithFallback(\n prompt,\n size,\n apiKey,\n \"openrouter\",\n { primary: primaryModel, fallbacks: fallbackModels },\n spinner,\n sourceImageData\n );\n } else if (aiProvider === \"openai\") {\n if (sourceImageData) {\n throw new Error(\n \"OpenAI does not support image-to-image generation. Use OpenRouter with a model that supports reference images.\"\n );\n }\n imageBuffer = await this.generateWithFallback(\n prompt,\n size,\n apiKey,\n \"openai\",\n { primary: primaryModel, fallbacks: fallbackModels },\n spinner\n );\n } else if (aiProvider === \"google\") {\n if (sourceImageData) {\n throw new Error(\n \"Google Imagen does not support image-to-image generation. Use OpenRouter with a model that supports reference images.\"\n );\n }\n imageBuffer = await this.generateWithFallback(\n prompt,\n size,\n apiKey,\n \"google\",\n { primary: primaryModel, fallbacks: fallbackModels },\n spinner\n );\n } else {\n if (spinner) spinner.fail();\n throw new Error(`Unsupported AI provider: ${aiProvider}`);\n }\n\n // ─────────────────────────────────────────────────────────────────────\n // Save Image\n // ─────────────────────────────────────────────────────────────────────\n const outputPath = resolve(output);\n writeFileSync(outputPath, imageBuffer);\n\n if (spinner) {\n spinner.succeed(`Image generated and saved to: ${outputPath}`);\n }\n\n // ─────────────────────────────────────────────────────────────────────\n // Output\n // ─────────────────────────────────────────────────────────────────────\n if (json) {\n console.log(\n JSON.stringify(\n {\n success: true,\n prompt,\n output: outputPath,\n size,\n provider: aiProvider,\n },\n null,\n 2\n )\n );\n } else {\n console.log(\"\\n🎨 Image Generation Complete!\");\n console.log(\"─────────────────────────────────────\");\n console.log(`Prompt: ${prompt}`);\n console.log(`Size: ${size}`);\n if (sourceImageData) {\n console.log(`Source Image: ${sourceImage}`);\n }\n console.log(`Output: ${outputPath}`);\n console.log(`Provider: ${aiProvider}`);\n if (model) {\n console.log(`Model: ${model}`);\n }\n console.log(\"─────────────────────────────────────\\n\");\n }\n } catch (error) {\n if (spinner && spinner.isSpinning) {\n spinner.fail(\"Image generation failed\");\n }\n throw error;\n }\n }\n\n /**\n * Generate image with smart fallback chain\n * Tries primary model first, then falls back to alternatives if it fails\n */\n private async generateWithFallback(\n prompt: string,\n size: string,\n apiKey: string,\n provider: \"openrouter\" | \"openai\" | \"google\",\n config: { primary: string | null; fallbacks: string[] },\n spinner: {\n text: string;\n info: (msg: string) => void;\n warn: (msg: string) => void;\n isSpinning?: boolean;\n } | null,\n sourceImage?: string\n ): Promise<Buffer> {\n const modelsToTry = [config.primary, ...config.fallbacks].filter(\n (model): model is string => model !== null\n );\n\n let lastError: Error | null = null;\n\n for (let i = 0; i < modelsToTry.length; i++) {\n const model = modelsToTry[i];\n\n try {\n if (spinner) {\n const modelName = model.split(\"/\").pop() || model;\n spinner.text = `Generating image with ${modelName}... (attempt ${i + 1}/${modelsToTry.length})`;\n }\n\n const imageBuffer = await this.generateWithModel(\n prompt,\n size,\n apiKey,\n provider,\n model,\n sourceImage\n );\n\n if (spinner && i > 0) {\n // Only show fallback message if we had to fall back\n spinner.info(`Successfully generated with fallback model: ${model}`);\n }\n\n return imageBuffer;\n } catch (error) {\n lastError = error as Error;\n\n // If this wasn't the last model, log the failure and try the next one\n if (i < modelsToTry.length - 1) {\n if (spinner) {\n spinner.warn(`Model ${model} failed, trying fallback...`);\n } else {\n console.warn(`Model ${model} failed: ${lastError.message}`);\n }\n continue;\n }\n }\n }\n\n // If we get here, all models failed\n throw new Error(\n `All models failed to generate image. Last error: ${lastError?.message || \"Unknown error\"}`\n );\n }\n\n /**\n * get the model configuration for the AI SDK\n */\n private getImageModel(provider: string, apiKey: string, model: string) {\n if (provider === \"openai\") {\n const openai = createOpenAI({ apiKey });\n return openai.image(model);\n } else if (provider === \"google\") {\n const google = createGoogleGenerativeAI({ apiKey });\n return google.image(model);\n }\n throw new Error(`Provider ${provider} not supported via AI SDK`);\n }\n\n /**\n * Generate image using a specific model\n */\n private async generateWithModel(\n prompt: string,\n size: string,\n apiKey: string,\n provider: \"openrouter\" | \"openai\" | \"google\",\n model: string,\n sourceImage?: string\n ): Promise<Buffer> {\n // ─────────────────────────────────────────────────────────────────────\n // OPENROUTER (Via Fetch / Chat Completions)\n // ─────────────────────────────────────────────────────────────────────\n if (provider === \"openrouter\") {\n // Parse aspect ratio from size\n const [width, height] = size.split(\"x\").map(Number);\n let aspectRatio = \"1:1\";\n if (width && height) {\n const gcd = (a: number, b: number): number => (b === 0 ? a : gcd(b, a % b));\n const divisor = gcd(width, height);\n aspectRatio = `${width / divisor}:${height / divisor}`;\n }\n\n // Build content array based on whether we have a source image\n let content: any;\n if (sourceImage) {\n // Image-to-image generation: include source image in content\n content = [\n {\n type: \"text\",\n text: prompt,\n },\n {\n type: \"image_url\",\n image_url: {\n url: sourceImage,\n },\n },\n ];\n } else {\n // Text-to-image generation: just the prompt\n content = prompt;\n }\n\n const response = await fetch(\"https://openrouter.ai/api/v1/chat/completions\", {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${apiKey}`,\n \"Content-Type\": \"application/json\",\n \"HTTP-Referer\": \"https://clawbr.bricks-studio.ai\",\n \"X-Title\": \"clawbr CLI\",\n },\n body: JSON.stringify({\n model: model,\n messages: [\n {\n role: \"user\",\n content: content,\n },\n ],\n // Specific to Gemini/OpenRouter multimodal\n modalities: [\"image\", \"text\"],\n image_config: {\n aspect_ratio: aspectRatio,\n // image_size: \"4K\", // Optional based on snippet, but maybe risky for all models\n },\n }),\n });\n\n if (!response.ok) {\n const text = await response.text();\n throw new Error(`OpenRouter API error: ${text}`);\n }\n\n const result = (await response.json()) as any;\n\n if (result.choices?.[0]?.message?.images?.[0]?.image_url?.url) {\n const imageUrl = result.choices[0].message.images[0].image_url.url;\n\n // If it's a URL, fetch it\n if (imageUrl.startsWith(\"http\")) {\n const imgRes = await fetch(imageUrl);\n const arrayBuffer = await imgRes.arrayBuffer();\n return Buffer.from(arrayBuffer);\n }\n\n // If it's base64 data URI\n if (imageUrl.startsWith(\"data:image\")) {\n const base64Data = imageUrl.split(\",\")[1];\n return Buffer.from(base64Data, \"base64\");\n }\n\n throw new Error(\"Unknown image URL format\");\n }\n\n throw new Error(\"No image generated from OpenRouter response\");\n }\n\n // ─────────────────────────────────────────────────────────────────────\n // OPENAI / GOOGLE (Via AI SDK)\n // ─────────────────────────────────────────────────────────────────────\n const imageModel = this.getImageModel(provider, apiKey, model);\n\n // Pass size as string directly as per SDK requirements.\n // We cast to 'any' to avoid strict template literal validation errors\n // since we know validSizes allows specifically \"1024x1024\" etc.\n const { image } = await generateImage({\n model: imageModel,\n prompt,\n n: 1,\n size: size as any,\n });\n\n // The image object from 'ai' SDK contains the base64 string\n return Buffer.from(image.base64, \"base64\");\n }\n\n @Option({\n flags: \"-p, --prompt <text>\",\n description: \"Text description of the image to generate\",\n })\n parsePrompt(val: string): string {\n return val;\n }\n\n @Option({\n flags: \"-o, --output <path>\",\n description: \"Path where the generated image will be saved\",\n })\n parseOutput(val: string): string {\n return val;\n }\n\n @Option({\n flags: \"-s, --size <size>\",\n description: \"Image size (256x256, 512x512, 1024x1024, 1792x1024, 1024x1792)\",\n })\n parseSize(val: string): string {\n return val;\n }\n\n @Option({\n flags: \"--source-image <path>\",\n description: \"Path to source image or URL (for image-to-image generation, OpenRouter only)\",\n })\n parseSourceImage(val: string): string {\n return val;\n }\n\n @Option({\n flags: \"-m, --model <modelId>\",\n description:\n \"Specific model to use (provider-dependent). Use model ID from your provider's list. Note: Not all models support reference images (--source-image).\",\n })\n parseModel(val: string): string {\n return val;\n }\n\n @Option({\n flags: \"--json\",\n description: \"Output result in JSON format\",\n })\n parseJson(): boolean {\n return true;\n }\n}\n"],"names":["Command","CommandRunner","Option","writeFileSync","ora","fetch","resolve","generateImage","createOpenAI","createGoogleGenerativeAI","loadCredentials","resolveImageToDataUri","validateImageInput","requireOnboarding","getProviderModels","getModelById","isValidModel","getPrimaryModel","getFallbackModels","supportsReferenceImage","formatModelList","GenerateCommand","run","inputs","options","prompt","output","size","sourceImage","model","json","Error","validation","valid","error","validSizes","includes","join","credentials","aiProvider","apiKeys","apiKey","availableModels","modelInfo","name","filter","m","map","id","sourceImageData","undefined","spinner","start","imageBuffer","primaryModel","fallbackModels","generateWithFallback","primary","fallbacks","fail","outputPath","succeed","console","log","JSON","stringify","success","provider","isSpinning","config","modelsToTry","lastError","i","length","modelName","split","pop","text","generateWithModel","info","warn","message","getImageModel","openai","image","google","width","height","Number","aspectRatio","gcd","a","b","divisor","content","type","image_url","url","response","method","headers","Authorization","body","messages","role","modalities","image_config","aspect_ratio","ok","result","choices","images","imageUrl","startsWith","imgRes","arrayBuffer","Buffer","from","base64Data","imageModel","n","base64","parsePrompt","val","parseOutput","parseSize","parseSourceImage","parseModel","parseJson","flags","description","arguments","isDefault"],"mappings":"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"}
1
+ {"version":3,"sources":["../../src/commands/generate.command.ts"],"sourcesContent":["import { Command, CommandRunner, Option } from \"nest-commander\";\nimport { writeFileSync } from \"fs\";\nimport ora from \"ora\";\nimport fetch from \"node-fetch\";\nimport { resolve } from \"path\";\nimport { generateImage } from \"ai\";\nimport { createOpenAI } from \"@ai-sdk/openai\";\nimport { createGoogleGenerativeAI } from \"@ai-sdk/google\";\nimport { loadCredentials } from \"../utils/credentials.js\";\nimport { resolveImageToDataUri, validateImageInput } from \"../utils/image.js\";\nimport { requireOnboarding } from \"../utils/config.js\";\nimport {\n getProviderModels,\n getModelById,\n isValidModel,\n getPrimaryModel,\n getFallbackModels,\n supportsReferenceImage,\n formatModelList,\n} from \"../config/image-models.js\";\n\ninterface GenerateCommandOptions {\n prompt?: string;\n output?: string;\n size?: string;\n sourceImage?: string;\n model?: string;\n aspectRatio?: string;\n imageSize?: string;\n json?: boolean;\n}\n\n@Command({\n name: \"generate\",\n description: \"Generate an image using AI with smart model fallback\",\n arguments: \"\",\n options: { isDefault: false },\n})\nexport class GenerateCommand extends CommandRunner {\n async run(inputs: string[], options: GenerateCommandOptions): Promise<void> {\n await requireOnboarding();\n const {\n prompt,\n output,\n size = \"1024x1024\",\n sourceImage,\n model,\n aspectRatio,\n imageSize,\n json = false,\n } = options;\n\n // ─────────────────────────────────────────────────────────────────────\n // Validation\n // ─────────────────────────────────────────────────────────────────────\n if (!prompt) {\n throw new Error(\n '--prompt is required. Example: clawbr generate --prompt \"a robot building software\" --output \"./robot.png\"'\n );\n }\n\n if (!output) {\n throw new Error(\n '--output is required. Example: clawbr generate --prompt \"...\" --output \"./image.png\"'\n );\n }\n\n // Validate source image if provided\n if (sourceImage) {\n const validation = validateImageInput(sourceImage);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n }\n\n // Validate size\n const validSizes = [\"256x256\", \"512x512\", \"1024x1024\", \"1792x1024\", \"1024x1792\"];\n if (!validSizes.includes(size)) {\n throw new Error(`Invalid size. Must be one of: ${validSizes.join(\", \")}`);\n }\n\n // ─────────────────────────────────────────────────────────────────────\n // Load Credentials\n // ─────────────────────────────────────────────────────────────────────\n const credentials = loadCredentials();\n\n if (!credentials) {\n throw new Error(\"Credentials not found. Run 'clawbr onboard' first to set up your account.\");\n }\n\n const { aiProvider, apiKeys } = credentials;\n const apiKey = apiKeys[aiProvider as keyof typeof apiKeys];\n\n if (!apiKey) {\n throw new Error(\n `No API key found for provider '${aiProvider}'. Run 'clawbr onboard' to configure.`\n );\n }\n\n // ─────────────────────────────────────────────────────────────────────\n // Validate model if provided\n // ─────────────────────────────────────────────────────────────────────\n if (model && !isValidModel(aiProvider, model)) {\n const availableModels = formatModelList(aiProvider);\n throw new Error(\n `Invalid model '${model}' for provider '${aiProvider}'.\\n\\nAvailable models:\\n${availableModels}`\n );\n }\n\n // ─────────────────────────────────────────────────────────────────────\n // Check reference image support\n // ─────────────────────────────────────────────────────────────────────\n if (sourceImage && model && !supportsReferenceImage(aiProvider, model)) {\n const modelInfo = getModelById(aiProvider, model);\n throw new Error(\n `Model '${modelInfo?.name || model}' does not support reference images.\\n\\n` +\n `For reference image support with ${aiProvider}, use one of:\\n` +\n getProviderModels(aiProvider)\n .filter((m) => m.supportsReferenceImage)\n .map((m) => ` • ${m.id}`)\n .join(\"\\n\")\n );\n }\n\n // ─────────────────────────────────────────────────────────────────────\n // Prepare source image if provided\n // ─────────────────────────────────────────────────────────────────────\n const sourceImageData = sourceImage ? await resolveImageToDataUri(sourceImage) : undefined;\n\n // ─────────────────────────────────────────────────────────────────────\n // Generate Image with Smart Fallback\n // ─────────────────────────────────────────────────────────────────────\n const spinner = json\n ? null\n : ora(sourceImageData ? \"Generating image from source...\" : \"Generating image...\").start();\n\n try {\n let imageBuffer: Buffer;\n\n // Determine models to try\n const primaryModel = model || getPrimaryModel(aiProvider);\n const fallbackModels = getFallbackModels(aiProvider);\n\n // Pass aspect ratio and image size to generation\n const imageConfig: { aspectRatio?: string; imageSize?: string } = {};\n if (aspectRatio) imageConfig.aspectRatio = aspectRatio;\n if (imageSize) imageConfig.imageSize = imageSize;\n\n if (aiProvider === \"openrouter\") {\n imageBuffer = await this.generateWithFallback(\n prompt,\n size,\n apiKey,\n \"openrouter\",\n { primary: primaryModel, fallbacks: fallbackModels },\n spinner,\n sourceImageData,\n imageConfig\n );\n } else if (aiProvider === \"openai\") {\n if (sourceImageData) {\n throw new Error(\n \"OpenAI does not support image-to-image generation. Use OpenRouter with a model that supports reference images.\"\n );\n }\n imageBuffer = await this.generateWithFallback(\n prompt,\n size,\n apiKey,\n \"openai\",\n { primary: primaryModel, fallbacks: fallbackModels },\n spinner\n );\n } else if (aiProvider === \"google\") {\n if (sourceImageData) {\n throw new Error(\n \"Google Imagen does not support image-to-image generation. Use OpenRouter with a model that supports reference images.\"\n );\n }\n imageBuffer = await this.generateWithFallback(\n prompt,\n size,\n apiKey,\n \"google\",\n { primary: primaryModel, fallbacks: fallbackModels },\n spinner\n );\n } else {\n if (spinner) spinner.fail();\n throw new Error(`Unsupported AI provider: ${aiProvider}`);\n }\n\n // ─────────────────────────────────────────────────────────────────────\n // Save Image\n // ─────────────────────────────────────────────────────────────────────\n const outputPath = resolve(output);\n writeFileSync(outputPath, imageBuffer);\n\n if (spinner) {\n spinner.succeed(`Image generated and saved to: ${outputPath}`);\n }\n\n // ─────────────────────────────────────────────────────────────────────\n // Output\n // ─────────────────────────────────────────────────────────────────────\n if (json) {\n console.log(\n JSON.stringify(\n {\n success: true,\n prompt,\n output: outputPath,\n size,\n provider: aiProvider,\n },\n null,\n 2\n )\n );\n } else {\n console.log(\"\\n🎨 Image Generation Complete!\");\n console.log(\"─────────────────────────────────────\");\n console.log(`Prompt: ${prompt}`);\n console.log(`Size: ${size}`);\n if (sourceImageData) {\n console.log(`Source Image: ${sourceImage}`);\n }\n console.log(`Output: ${outputPath}`);\n console.log(`Provider: ${aiProvider}`);\n if (model) {\n console.log(`Model: ${model}`);\n }\n console.log(\"─────────────────────────────────────\\n\");\n }\n } catch (error) {\n if (spinner && spinner.isSpinning) {\n spinner.fail(\"Image generation failed\");\n }\n throw error;\n }\n }\n\n /**\n * Generate image with smart fallback chain\n * Tries primary model first, then falls back to alternatives if it fails\n */\n private async generateWithFallback(\n prompt: string,\n size: string,\n apiKey: string,\n provider: \"openrouter\" | \"openai\" | \"google\",\n config: { primary: string | null; fallbacks: string[] },\n spinner: {\n text: string;\n info: (msg: string) => void;\n warn: (msg: string) => void;\n isSpinning?: boolean;\n } | null,\n sourceImageData?: string,\n imageConfig?: { aspectRatio?: string; imageSize?: string }\n ): Promise<Buffer> {\n const modelsToTry = [config.primary, ...config.fallbacks].filter(\n (model): model is string => model !== null\n );\n\n let lastError: Error | null = null;\n\n for (let i = 0; i < modelsToTry.length; i++) {\n const model = modelsToTry[i];\n\n try {\n if (spinner) {\n const modelName = model.split(\"/\").pop() || model;\n spinner.text = `Generating image with ${modelName}... (attempt ${i + 1}/${modelsToTry.length})`;\n }\n\n const imageBuffer = await this.generateWithModel(\n prompt,\n size,\n apiKey,\n provider,\n model,\n sourceImageData,\n imageConfig\n );\n\n if (spinner && i > 0) {\n // Only show fallback message if we had to fall back\n spinner.info(`Successfully generated with fallback model: ${model}`);\n }\n\n return imageBuffer;\n } catch (error) {\n lastError = error as Error;\n\n // If this wasn't the last model, log the failure and try the next one\n if (i < modelsToTry.length - 1) {\n if (spinner) {\n spinner.warn(`Model ${model} failed, trying fallback...`);\n } else {\n console.warn(`Model ${model} failed: ${lastError.message}`);\n }\n continue;\n }\n }\n }\n\n // If we get here, all models failed\n throw new Error(\n `All models failed to generate image. Last error: ${lastError?.message || \"Unknown error\"}`\n );\n }\n\n /**\n * get the model configuration for the AI SDK\n */\n private getImageModel(provider: string, apiKey: string, model: string) {\n if (provider === \"openai\") {\n const openai = createOpenAI({ apiKey });\n return openai.image(model);\n } else if (provider === \"google\") {\n const google = createGoogleGenerativeAI({ apiKey });\n return google.image(model);\n }\n throw new Error(`Provider ${provider} not supported via AI SDK`);\n }\n\n /**\n * Generate image using a specific model\n */\n private async generateWithModel(\n prompt: string,\n size: string,\n apiKey: string,\n provider: \"openrouter\" | \"openai\" | \"google\",\n model: string,\n sourceImageData?: string,\n imageConfig?: { aspectRatio?: string; imageSize?: string }\n ): Promise<Buffer> {\n // ─────────────────────────────────────────────────────────────────────\n // OPENROUTER (Via Fetch / Chat Completions)\n // ─────────────────────────────────────────────────────────────────────\n if (provider === \"openrouter\") {\n // Calculate aspect ratio from size if not provided\n let aspectRatio = imageConfig?.aspectRatio || \"1:1\";\n if (!imageConfig?.aspectRatio) {\n const [width, height] = size.split(\"x\").map(Number);\n if (width && height) {\n const gcd = (a: number, b: number): number => (b === 0 ? a : gcd(b, a % b));\n const divisor = gcd(width, height);\n const calculated = `${width / divisor}:${height / divisor}`;\n\n // Map calculated ratio to supported OpenRouter ratios\n const supportedRatios: Record<string, string> = {\n \"1:1\": \"1:1\",\n \"2:3\": \"2:3\",\n \"3:2\": \"3:2\",\n \"3:4\": \"3:4\",\n \"4:3\": \"4:3\",\n \"4:5\": \"4:5\",\n \"5:4\": \"5:4\",\n \"9:16\": \"9:16\",\n \"16:9\": \"16:9\",\n \"21:9\": \"21:9\",\n // Common unsupported ratios mapped to closest supported\n \"7:4\": \"16:9\", // 1792x1024\n \"4:7\": \"9:16\", // 1024x1792\n \"64:27\": \"21:9\", // ultrawide variants\n };\n\n aspectRatio = supportedRatios[calculated] || \"1:1\";\n }\n }\n\n // Build messages array\n let content: Array<{ type: string; text?: string; image_url?: { url: string } }> | string;\n if (sourceImageData) {\n // Image-to-image generation: include source image in content\n content = [\n {\n type: \"text\",\n text: prompt,\n },\n {\n type: \"image_url\",\n image_url: {\n url: sourceImageData,\n },\n },\n ];\n } else {\n // Text-to-image generation: just the prompt\n content = prompt;\n }\n\n const response = await fetch(\"https://openrouter.ai/api/v1/chat/completions\", {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${apiKey}`,\n \"Content-Type\": \"application/json\",\n \"HTTP-Referer\": \"https://clawbr.bricks-studio.ai\",\n \"X-Title\": \"clawbr CLI\",\n },\n body: JSON.stringify({\n model: model,\n messages: [\n {\n role: \"user\",\n content: content,\n },\n ],\n // Specific to Gemini/OpenRouter multimodal\n modalities: [\"image\", \"text\"],\n ...(aspectRatio || imageConfig?.imageSize\n ? {\n image_config: {\n ...(aspectRatio ? { aspect_ratio: aspectRatio } : {}),\n ...(imageConfig?.imageSize ? { image_size: imageConfig.imageSize } : {}),\n },\n }\n : {}),\n }),\n });\n\n if (!response.ok) {\n const text = await response.text();\n throw new Error(`OpenRouter API error: ${text}`);\n }\n\n const result = (await response.json()) as any;\n\n if (result.choices?.[0]?.message?.images?.[0]?.image_url?.url) {\n const imageUrl = result.choices[0].message.images[0].image_url.url;\n\n // If it's a URL, fetch it\n if (imageUrl.startsWith(\"http\")) {\n const imgRes = await fetch(imageUrl);\n const arrayBuffer = await imgRes.arrayBuffer();\n return Buffer.from(arrayBuffer);\n }\n\n // If it's base64 data URI\n if (imageUrl.startsWith(\"data:image\")) {\n const base64Data = imageUrl.split(\",\")[1];\n return Buffer.from(base64Data, \"base64\");\n }\n\n throw new Error(\"Unknown image URL format\");\n }\n\n throw new Error(\"No image generated from OpenRouter response\");\n }\n\n // ─────────────────────────────────────────────────────────────────────\n // OPENAI / GOOGLE (Via AI SDK)\n // ─────────────────────────────────────────────────────────────────────\n const imageModel = this.getImageModel(provider, apiKey, model);\n\n // Pass size as string directly as per SDK requirements.\n // We cast to 'any' to avoid strict template literal validation errors\n // since we know validSizes allows specifically \"1024x1024\" etc.\n const { image } = await generateImage({\n model: imageModel,\n prompt,\n n: 1,\n size: size as any,\n });\n\n // The image object from 'ai' SDK contains the base64 string\n return Buffer.from(image.base64, \"base64\");\n }\n\n @Option({\n flags: \"-p, --prompt <text>\",\n description: \"Text description of the image to generate\",\n })\n parsePrompt(val: string): string {\n return val;\n }\n\n @Option({\n flags: \"-o, --output <path>\",\n description: \"Path where the generated image will be saved\",\n })\n parseOutput(val: string): string {\n return val;\n }\n\n @Option({\n flags: \"-s, --size <size>\",\n description: \"Image size (256x256, 512x512, 1024x1024, 1792x1024, 1024x1792)\",\n })\n parseSize(val: string): string {\n return val;\n }\n\n @Option({\n flags: \"--source-image <path>\",\n description: \"Path to source image or URL (for image-to-image generation, OpenRouter only)\",\n })\n parseSourceImage(val: string): string {\n return val;\n }\n\n @Option({\n flags: \"-m, --model <modelId>\",\n description:\n \"Specific model to use (provider-dependent). Use model ID from your provider's list. Note: Not all models support reference images (--source-image).\",\n })\n parseModel(val: string): string {\n return val;\n }\n\n @Option({\n flags: \"--aspect-ratio <ratio>\",\n description:\n \"Aspect ratio for generated image (OpenRouter only). Supported: 1:1, 2:3, 3:2, 3:4, 4:3, 4:5, 5:4, 9:16, 16:9, 21:9\",\n })\n parseAspectRatio(val: string): string {\n return val;\n }\n\n @Option({\n flags: \"--image-size <size>\",\n description:\n \"Image resolution size (OpenRouter only). Supported: 1K (standard), 2K (higher), 4K (highest)\",\n })\n parseImageSize(val: string): string {\n return val;\n }\n\n @Option({\n flags: \"--json\",\n description: \"Output result in JSON format\",\n })\n parseJson(): boolean {\n return true;\n }\n}\n"],"names":["Command","CommandRunner","Option","writeFileSync","ora","fetch","resolve","generateImage","createOpenAI","createGoogleGenerativeAI","loadCredentials","resolveImageToDataUri","validateImageInput","requireOnboarding","getProviderModels","getModelById","isValidModel","getPrimaryModel","getFallbackModels","supportsReferenceImage","formatModelList","GenerateCommand","run","inputs","options","prompt","output","size","sourceImage","model","aspectRatio","imageSize","json","Error","validation","valid","error","validSizes","includes","join","credentials","aiProvider","apiKeys","apiKey","availableModels","modelInfo","name","filter","m","map","id","sourceImageData","undefined","spinner","start","imageBuffer","primaryModel","fallbackModels","imageConfig","generateWithFallback","primary","fallbacks","fail","outputPath","succeed","console","log","JSON","stringify","success","provider","isSpinning","config","modelsToTry","lastError","i","length","modelName","split","pop","text","generateWithModel","info","warn","message","getImageModel","openai","image","google","width","height","Number","gcd","a","b","divisor","calculated","supportedRatios","content","type","image_url","url","response","method","headers","Authorization","body","messages","role","modalities","image_config","aspect_ratio","image_size","ok","result","choices","images","imageUrl","startsWith","imgRes","arrayBuffer","Buffer","from","base64Data","imageModel","n","base64","parsePrompt","val","parseOutput","parseSize","parseSourceImage","parseModel","parseAspectRatio","parseImageSize","parseJson","flags","description","arguments","isDefault"],"mappings":"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"}