@vibeframe/cli 0.27.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 (420) hide show
  1. package/.turbo/turbo-build.log +4 -0
  2. package/.turbo/turbo-lint.log +21 -0
  3. package/.turbo/turbo-test.log +689 -0
  4. package/dist/agent/adapters/claude.d.ts +15 -0
  5. package/dist/agent/adapters/claude.d.ts.map +1 -0
  6. package/dist/agent/adapters/claude.js +119 -0
  7. package/dist/agent/adapters/claude.js.map +1 -0
  8. package/dist/agent/adapters/gemini.d.ts +15 -0
  9. package/dist/agent/adapters/gemini.d.ts.map +1 -0
  10. package/dist/agent/adapters/gemini.js +132 -0
  11. package/dist/agent/adapters/gemini.js.map +1 -0
  12. package/dist/agent/adapters/index.d.ts +27 -0
  13. package/dist/agent/adapters/index.d.ts.map +1 -0
  14. package/dist/agent/adapters/index.js +38 -0
  15. package/dist/agent/adapters/index.js.map +1 -0
  16. package/dist/agent/adapters/ollama.d.ts +20 -0
  17. package/dist/agent/adapters/ollama.d.ts.map +1 -0
  18. package/dist/agent/adapters/ollama.js +186 -0
  19. package/dist/agent/adapters/ollama.js.map +1 -0
  20. package/dist/agent/adapters/openai.d.ts +15 -0
  21. package/dist/agent/adapters/openai.d.ts.map +1 -0
  22. package/dist/agent/adapters/openai.js +92 -0
  23. package/dist/agent/adapters/openai.js.map +1 -0
  24. package/dist/agent/adapters/xai.d.ts +15 -0
  25. package/dist/agent/adapters/xai.d.ts.map +1 -0
  26. package/dist/agent/adapters/xai.js +95 -0
  27. package/dist/agent/adapters/xai.js.map +1 -0
  28. package/dist/agent/index.d.ts +69 -0
  29. package/dist/agent/index.d.ts.map +1 -0
  30. package/dist/agent/index.js +180 -0
  31. package/dist/agent/index.js.map +1 -0
  32. package/dist/agent/memory/index.d.ts +70 -0
  33. package/dist/agent/memory/index.d.ts.map +1 -0
  34. package/dist/agent/memory/index.js +132 -0
  35. package/dist/agent/memory/index.js.map +1 -0
  36. package/dist/agent/prompts/system.d.ts +6 -0
  37. package/dist/agent/prompts/system.d.ts.map +1 -0
  38. package/dist/agent/prompts/system.js +103 -0
  39. package/dist/agent/prompts/system.js.map +1 -0
  40. package/dist/agent/tools/ai-editing.d.ts +15 -0
  41. package/dist/agent/tools/ai-editing.d.ts.map +1 -0
  42. package/dist/agent/tools/ai-editing.js +763 -0
  43. package/dist/agent/tools/ai-editing.js.map +1 -0
  44. package/dist/agent/tools/ai-generation.d.ts +13 -0
  45. package/dist/agent/tools/ai-generation.d.ts.map +1 -0
  46. package/dist/agent/tools/ai-generation.js +973 -0
  47. package/dist/agent/tools/ai-generation.js.map +1 -0
  48. package/dist/agent/tools/ai-pipeline.d.ts +14 -0
  49. package/dist/agent/tools/ai-pipeline.d.ts.map +1 -0
  50. package/dist/agent/tools/ai-pipeline.js +961 -0
  51. package/dist/agent/tools/ai-pipeline.js.map +1 -0
  52. package/dist/agent/tools/ai.d.ts +13 -0
  53. package/dist/agent/tools/ai.d.ts.map +1 -0
  54. package/dist/agent/tools/ai.js +19 -0
  55. package/dist/agent/tools/ai.js.map +1 -0
  56. package/dist/agent/tools/batch.d.ts +6 -0
  57. package/dist/agent/tools/batch.d.ts.map +1 -0
  58. package/dist/agent/tools/batch.js +383 -0
  59. package/dist/agent/tools/batch.js.map +1 -0
  60. package/dist/agent/tools/e2e.test.d.ts +26 -0
  61. package/dist/agent/tools/e2e.test.d.ts.map +1 -0
  62. package/dist/agent/tools/e2e.test.js +397 -0
  63. package/dist/agent/tools/e2e.test.js.map +1 -0
  64. package/dist/agent/tools/export.d.ts +6 -0
  65. package/dist/agent/tools/export.d.ts.map +1 -0
  66. package/dist/agent/tools/export.js +171 -0
  67. package/dist/agent/tools/export.js.map +1 -0
  68. package/dist/agent/tools/filesystem.d.ts +6 -0
  69. package/dist/agent/tools/filesystem.d.ts.map +1 -0
  70. package/dist/agent/tools/filesystem.js +212 -0
  71. package/dist/agent/tools/filesystem.js.map +1 -0
  72. package/dist/agent/tools/index.d.ts +65 -0
  73. package/dist/agent/tools/index.d.ts.map +1 -0
  74. package/dist/agent/tools/index.js +120 -0
  75. package/dist/agent/tools/index.js.map +1 -0
  76. package/dist/agent/tools/integration.test.d.ts +11 -0
  77. package/dist/agent/tools/integration.test.d.ts.map +1 -0
  78. package/dist/agent/tools/integration.test.js +659 -0
  79. package/dist/agent/tools/integration.test.js.map +1 -0
  80. package/dist/agent/tools/media.d.ts +6 -0
  81. package/dist/agent/tools/media.d.ts.map +1 -0
  82. package/dist/agent/tools/media.js +616 -0
  83. package/dist/agent/tools/media.js.map +1 -0
  84. package/dist/agent/tools/project.d.ts +6 -0
  85. package/dist/agent/tools/project.d.ts.map +1 -0
  86. package/dist/agent/tools/project.js +284 -0
  87. package/dist/agent/tools/project.js.map +1 -0
  88. package/dist/agent/tools/timeline.d.ts +6 -0
  89. package/dist/agent/tools/timeline.d.ts.map +1 -0
  90. package/dist/agent/tools/timeline.js +873 -0
  91. package/dist/agent/tools/timeline.js.map +1 -0
  92. package/dist/agent/types.d.ts +59 -0
  93. package/dist/agent/types.d.ts.map +1 -0
  94. package/dist/agent/types.js +5 -0
  95. package/dist/agent/types.js.map +1 -0
  96. package/dist/commands/agent.d.ts +21 -0
  97. package/dist/commands/agent.d.ts.map +1 -0
  98. package/dist/commands/agent.js +290 -0
  99. package/dist/commands/agent.js.map +1 -0
  100. package/dist/commands/ai-analyze.d.ts +106 -0
  101. package/dist/commands/ai-analyze.d.ts.map +1 -0
  102. package/dist/commands/ai-analyze.js +327 -0
  103. package/dist/commands/ai-analyze.js.map +1 -0
  104. package/dist/commands/ai-animated-caption.d.ts +64 -0
  105. package/dist/commands/ai-animated-caption.d.ts.map +1 -0
  106. package/dist/commands/ai-animated-caption.js +272 -0
  107. package/dist/commands/ai-animated-caption.js.map +1 -0
  108. package/dist/commands/ai-audio.d.ts +20 -0
  109. package/dist/commands/ai-audio.d.ts.map +1 -0
  110. package/dist/commands/ai-audio.js +808 -0
  111. package/dist/commands/ai-audio.js.map +1 -0
  112. package/dist/commands/ai-broll.d.ts +15 -0
  113. package/dist/commands/ai-broll.d.ts.map +1 -0
  114. package/dist/commands/ai-broll.js +406 -0
  115. package/dist/commands/ai-broll.js.map +1 -0
  116. package/dist/commands/ai-edit-cli.d.ts +14 -0
  117. package/dist/commands/ai-edit-cli.d.ts.map +1 -0
  118. package/dist/commands/ai-edit-cli.js +579 -0
  119. package/dist/commands/ai-edit-cli.js.map +1 -0
  120. package/dist/commands/ai-edit.d.ts +398 -0
  121. package/dist/commands/ai-edit.d.ts.map +1 -0
  122. package/dist/commands/ai-edit.js +1019 -0
  123. package/dist/commands/ai-edit.js.map +1 -0
  124. package/dist/commands/ai-fill-gaps.d.ts +14 -0
  125. package/dist/commands/ai-fill-gaps.d.ts.map +1 -0
  126. package/dist/commands/ai-fill-gaps.js +451 -0
  127. package/dist/commands/ai-fill-gaps.js.map +1 -0
  128. package/dist/commands/ai-helpers.d.ts +20 -0
  129. package/dist/commands/ai-helpers.d.ts.map +1 -0
  130. package/dist/commands/ai-helpers.js +59 -0
  131. package/dist/commands/ai-helpers.js.map +1 -0
  132. package/dist/commands/ai-highlights.d.ts +127 -0
  133. package/dist/commands/ai-highlights.d.ts.map +1 -0
  134. package/dist/commands/ai-highlights.js +1026 -0
  135. package/dist/commands/ai-highlights.js.map +1 -0
  136. package/dist/commands/ai-image.d.ts +34 -0
  137. package/dist/commands/ai-image.d.ts.map +1 -0
  138. package/dist/commands/ai-image.js +653 -0
  139. package/dist/commands/ai-image.js.map +1 -0
  140. package/dist/commands/ai-motion.d.ts +50 -0
  141. package/dist/commands/ai-motion.d.ts.map +1 -0
  142. package/dist/commands/ai-motion.js +271 -0
  143. package/dist/commands/ai-motion.js.map +1 -0
  144. package/dist/commands/ai-narrate.d.ts +66 -0
  145. package/dist/commands/ai-narrate.d.ts.map +1 -0
  146. package/dist/commands/ai-narrate.js +329 -0
  147. package/dist/commands/ai-narrate.js.map +1 -0
  148. package/dist/commands/ai-review.d.ts +57 -0
  149. package/dist/commands/ai-review.d.ts.map +1 -0
  150. package/dist/commands/ai-review.js +251 -0
  151. package/dist/commands/ai-review.js.map +1 -0
  152. package/dist/commands/ai-script-pipeline-cli.d.ts +9 -0
  153. package/dist/commands/ai-script-pipeline-cli.d.ts.map +1 -0
  154. package/dist/commands/ai-script-pipeline-cli.js +1494 -0
  155. package/dist/commands/ai-script-pipeline-cli.js.map +1 -0
  156. package/dist/commands/ai-script-pipeline.d.ts +259 -0
  157. package/dist/commands/ai-script-pipeline.d.ts.map +1 -0
  158. package/dist/commands/ai-script-pipeline.js +1027 -0
  159. package/dist/commands/ai-script-pipeline.js.map +1 -0
  160. package/dist/commands/ai-suggest-edit.d.ts +14 -0
  161. package/dist/commands/ai-suggest-edit.d.ts.map +1 -0
  162. package/dist/commands/ai-suggest-edit.js +220 -0
  163. package/dist/commands/ai-suggest-edit.js.map +1 -0
  164. package/dist/commands/ai-video-fx.d.ts +14 -0
  165. package/dist/commands/ai-video-fx.d.ts.map +1 -0
  166. package/dist/commands/ai-video-fx.js +395 -0
  167. package/dist/commands/ai-video-fx.js.map +1 -0
  168. package/dist/commands/ai-video.d.ts +15 -0
  169. package/dist/commands/ai-video.d.ts.map +1 -0
  170. package/dist/commands/ai-video.js +785 -0
  171. package/dist/commands/ai-video.js.map +1 -0
  172. package/dist/commands/ai-viral.d.ts +15 -0
  173. package/dist/commands/ai-viral.d.ts.map +1 -0
  174. package/dist/commands/ai-viral.js +519 -0
  175. package/dist/commands/ai-viral.js.map +1 -0
  176. package/dist/commands/ai-visual-fx.d.ts +14 -0
  177. package/dist/commands/ai-visual-fx.d.ts.map +1 -0
  178. package/dist/commands/ai-visual-fx.js +505 -0
  179. package/dist/commands/ai-visual-fx.js.map +1 -0
  180. package/dist/commands/ai.d.ts +38 -0
  181. package/dist/commands/ai.d.ts.map +1 -0
  182. package/dist/commands/ai.js +225 -0
  183. package/dist/commands/ai.js.map +1 -0
  184. package/dist/commands/ai.test.d.ts +2 -0
  185. package/dist/commands/ai.test.d.ts.map +1 -0
  186. package/dist/commands/ai.test.js +554 -0
  187. package/dist/commands/ai.test.js.map +1 -0
  188. package/dist/commands/analyze.d.ts +16 -0
  189. package/dist/commands/analyze.d.ts.map +1 -0
  190. package/dist/commands/analyze.js +247 -0
  191. package/dist/commands/analyze.js.map +1 -0
  192. package/dist/commands/audio.d.ts +18 -0
  193. package/dist/commands/audio.d.ts.map +1 -0
  194. package/dist/commands/audio.js +539 -0
  195. package/dist/commands/audio.js.map +1 -0
  196. package/dist/commands/batch.d.ts +3 -0
  197. package/dist/commands/batch.d.ts.map +1 -0
  198. package/dist/commands/batch.js +366 -0
  199. package/dist/commands/batch.js.map +1 -0
  200. package/dist/commands/batch.test.d.ts +2 -0
  201. package/dist/commands/batch.test.d.ts.map +1 -0
  202. package/dist/commands/batch.test.js +203 -0
  203. package/dist/commands/batch.test.js.map +1 -0
  204. package/dist/commands/detect.d.ts +3 -0
  205. package/dist/commands/detect.d.ts.map +1 -0
  206. package/dist/commands/detect.js +273 -0
  207. package/dist/commands/detect.js.map +1 -0
  208. package/dist/commands/doctor.d.ts +6 -0
  209. package/dist/commands/doctor.d.ts.map +1 -0
  210. package/dist/commands/doctor.js +191 -0
  211. package/dist/commands/doctor.js.map +1 -0
  212. package/dist/commands/edit-cmd.d.ts +26 -0
  213. package/dist/commands/edit-cmd.d.ts.map +1 -0
  214. package/dist/commands/edit-cmd.js +870 -0
  215. package/dist/commands/edit-cmd.js.map +1 -0
  216. package/dist/commands/export.d.ts +39 -0
  217. package/dist/commands/export.d.ts.map +1 -0
  218. package/dist/commands/export.js +730 -0
  219. package/dist/commands/export.js.map +1 -0
  220. package/dist/commands/generate.d.ts +25 -0
  221. package/dist/commands/generate.d.ts.map +1 -0
  222. package/dist/commands/generate.js +1885 -0
  223. package/dist/commands/generate.js.map +1 -0
  224. package/dist/commands/media.d.ts +3 -0
  225. package/dist/commands/media.d.ts.map +1 -0
  226. package/dist/commands/media.js +165 -0
  227. package/dist/commands/media.js.map +1 -0
  228. package/dist/commands/output.d.ts +45 -0
  229. package/dist/commands/output.d.ts.map +1 -0
  230. package/dist/commands/output.js +122 -0
  231. package/dist/commands/output.js.map +1 -0
  232. package/dist/commands/pipeline.d.ts +19 -0
  233. package/dist/commands/pipeline.d.ts.map +1 -0
  234. package/dist/commands/pipeline.js +345 -0
  235. package/dist/commands/pipeline.js.map +1 -0
  236. package/dist/commands/project.d.ts +3 -0
  237. package/dist/commands/project.d.ts.map +1 -0
  238. package/dist/commands/project.js +139 -0
  239. package/dist/commands/project.js.map +1 -0
  240. package/dist/commands/project.test.d.ts +2 -0
  241. package/dist/commands/project.test.d.ts.map +1 -0
  242. package/dist/commands/project.test.js +105 -0
  243. package/dist/commands/project.test.js.map +1 -0
  244. package/dist/commands/sanitize.d.ts +21 -0
  245. package/dist/commands/sanitize.d.ts.map +1 -0
  246. package/dist/commands/sanitize.js +56 -0
  247. package/dist/commands/sanitize.js.map +1 -0
  248. package/dist/commands/schema.d.ts +11 -0
  249. package/dist/commands/schema.d.ts.map +1 -0
  250. package/dist/commands/schema.js +101 -0
  251. package/dist/commands/schema.js.map +1 -0
  252. package/dist/commands/setup.d.ts +6 -0
  253. package/dist/commands/setup.d.ts.map +1 -0
  254. package/dist/commands/setup.js +440 -0
  255. package/dist/commands/setup.js.map +1 -0
  256. package/dist/commands/timeline.d.ts +3 -0
  257. package/dist/commands/timeline.d.ts.map +1 -0
  258. package/dist/commands/timeline.js +469 -0
  259. package/dist/commands/timeline.js.map +1 -0
  260. package/dist/commands/timeline.test.d.ts +2 -0
  261. package/dist/commands/timeline.test.d.ts.map +1 -0
  262. package/dist/commands/timeline.test.js +320 -0
  263. package/dist/commands/timeline.test.js.map +1 -0
  264. package/dist/commands/validate.d.ts +32 -0
  265. package/dist/commands/validate.d.ts.map +1 -0
  266. package/dist/commands/validate.js +63 -0
  267. package/dist/commands/validate.js.map +1 -0
  268. package/dist/config/config.test.d.ts +2 -0
  269. package/dist/config/config.test.d.ts.map +1 -0
  270. package/dist/config/config.test.js +164 -0
  271. package/dist/config/config.test.js.map +1 -0
  272. package/dist/config/index.d.ts +35 -0
  273. package/dist/config/index.d.ts.map +1 -0
  274. package/dist/config/index.js +101 -0
  275. package/dist/config/index.js.map +1 -0
  276. package/dist/config/schema.d.ts +43 -0
  277. package/dist/config/schema.d.ts.map +1 -0
  278. package/dist/config/schema.js +42 -0
  279. package/dist/config/schema.js.map +1 -0
  280. package/dist/engine/index.d.ts +3 -0
  281. package/dist/engine/index.d.ts.map +1 -0
  282. package/dist/engine/index.js +2 -0
  283. package/dist/engine/index.js.map +1 -0
  284. package/dist/engine/project.d.ts +84 -0
  285. package/dist/engine/project.d.ts.map +1 -0
  286. package/dist/engine/project.js +355 -0
  287. package/dist/engine/project.js.map +1 -0
  288. package/dist/engine/project.test.d.ts +2 -0
  289. package/dist/engine/project.test.d.ts.map +1 -0
  290. package/dist/engine/project.test.js +599 -0
  291. package/dist/engine/project.test.js.map +1 -0
  292. package/dist/index.d.ts +7 -0
  293. package/dist/index.d.ts.map +1 -0
  294. package/dist/index.js +131 -0
  295. package/dist/index.js.map +1 -0
  296. package/dist/utils/api-key.d.ts +36 -0
  297. package/dist/utils/api-key.d.ts.map +1 -0
  298. package/dist/utils/api-key.js +211 -0
  299. package/dist/utils/api-key.js.map +1 -0
  300. package/dist/utils/api-key.test.d.ts +2 -0
  301. package/dist/utils/api-key.test.d.ts.map +1 -0
  302. package/dist/utils/api-key.test.js +35 -0
  303. package/dist/utils/api-key.test.js.map +1 -0
  304. package/dist/utils/audio.d.ts +23 -0
  305. package/dist/utils/audio.d.ts.map +1 -0
  306. package/dist/utils/audio.js +79 -0
  307. package/dist/utils/audio.js.map +1 -0
  308. package/dist/utils/exec-safe.d.ts +22 -0
  309. package/dist/utils/exec-safe.d.ts.map +1 -0
  310. package/dist/utils/exec-safe.js +62 -0
  311. package/dist/utils/exec-safe.js.map +1 -0
  312. package/dist/utils/first-run.d.ts +13 -0
  313. package/dist/utils/first-run.d.ts.map +1 -0
  314. package/dist/utils/first-run.js +48 -0
  315. package/dist/utils/first-run.js.map +1 -0
  316. package/dist/utils/provider-resolver.d.ts +15 -0
  317. package/dist/utils/provider-resolver.d.ts.map +1 -0
  318. package/dist/utils/provider-resolver.js +42 -0
  319. package/dist/utils/provider-resolver.js.map +1 -0
  320. package/dist/utils/remotion.d.ts +210 -0
  321. package/dist/utils/remotion.d.ts.map +1 -0
  322. package/dist/utils/remotion.js +731 -0
  323. package/dist/utils/remotion.js.map +1 -0
  324. package/dist/utils/subtitle.d.ts +65 -0
  325. package/dist/utils/subtitle.d.ts.map +1 -0
  326. package/dist/utils/subtitle.js +135 -0
  327. package/dist/utils/subtitle.js.map +1 -0
  328. package/dist/utils/subtitle.test.d.ts +2 -0
  329. package/dist/utils/subtitle.test.d.ts.map +1 -0
  330. package/dist/utils/subtitle.test.js +175 -0
  331. package/dist/utils/subtitle.test.js.map +1 -0
  332. package/dist/utils/tty.d.ts +45 -0
  333. package/dist/utils/tty.d.ts.map +1 -0
  334. package/dist/utils/tty.js +172 -0
  335. package/dist/utils/tty.js.map +1 -0
  336. package/package.json +102 -0
  337. package/src/agent/adapters/claude.ts +143 -0
  338. package/src/agent/adapters/gemini.ts +159 -0
  339. package/src/agent/adapters/index.ts +61 -0
  340. package/src/agent/adapters/ollama.ts +231 -0
  341. package/src/agent/adapters/openai.ts +116 -0
  342. package/src/agent/adapters/xai.ts +119 -0
  343. package/src/agent/index.ts +251 -0
  344. package/src/agent/memory/index.ts +151 -0
  345. package/src/agent/prompts/system.ts +106 -0
  346. package/src/agent/tools/ai-editing.ts +845 -0
  347. package/src/agent/tools/ai-generation.ts +1073 -0
  348. package/src/agent/tools/ai-pipeline.ts +1055 -0
  349. package/src/agent/tools/ai.ts +21 -0
  350. package/src/agent/tools/batch.ts +429 -0
  351. package/src/agent/tools/e2e.test.ts +545 -0
  352. package/src/agent/tools/export.ts +184 -0
  353. package/src/agent/tools/filesystem.ts +237 -0
  354. package/src/agent/tools/index.ts +150 -0
  355. package/src/agent/tools/integration.test.ts +775 -0
  356. package/src/agent/tools/media.ts +697 -0
  357. package/src/agent/tools/project.ts +313 -0
  358. package/src/agent/tools/timeline.ts +951 -0
  359. package/src/agent/types.ts +68 -0
  360. package/src/commands/agent.ts +340 -0
  361. package/src/commands/ai-analyze.ts +429 -0
  362. package/src/commands/ai-animated-caption.ts +390 -0
  363. package/src/commands/ai-audio.ts +941 -0
  364. package/src/commands/ai-broll.ts +490 -0
  365. package/src/commands/ai-edit-cli.ts +658 -0
  366. package/src/commands/ai-edit.ts +1542 -0
  367. package/src/commands/ai-fill-gaps.ts +566 -0
  368. package/src/commands/ai-helpers.ts +65 -0
  369. package/src/commands/ai-highlights.ts +1303 -0
  370. package/src/commands/ai-image.ts +761 -0
  371. package/src/commands/ai-motion.ts +347 -0
  372. package/src/commands/ai-narrate.ts +451 -0
  373. package/src/commands/ai-review.ts +309 -0
  374. package/src/commands/ai-script-pipeline-cli.ts +1710 -0
  375. package/src/commands/ai-script-pipeline.ts +1365 -0
  376. package/src/commands/ai-suggest-edit.ts +264 -0
  377. package/src/commands/ai-video-fx.ts +445 -0
  378. package/src/commands/ai-video.ts +915 -0
  379. package/src/commands/ai-viral.ts +595 -0
  380. package/src/commands/ai-visual-fx.ts +601 -0
  381. package/src/commands/ai.test.ts +627 -0
  382. package/src/commands/ai.ts +307 -0
  383. package/src/commands/analyze.ts +282 -0
  384. package/src/commands/audio.ts +644 -0
  385. package/src/commands/batch.test.ts +279 -0
  386. package/src/commands/batch.ts +440 -0
  387. package/src/commands/detect.ts +329 -0
  388. package/src/commands/doctor.ts +237 -0
  389. package/src/commands/edit-cmd.ts +1014 -0
  390. package/src/commands/export.ts +918 -0
  391. package/src/commands/generate.ts +2146 -0
  392. package/src/commands/media.ts +177 -0
  393. package/src/commands/output.ts +142 -0
  394. package/src/commands/pipeline.ts +398 -0
  395. package/src/commands/project.test.ts +127 -0
  396. package/src/commands/project.ts +149 -0
  397. package/src/commands/sanitize.ts +60 -0
  398. package/src/commands/schema.ts +130 -0
  399. package/src/commands/setup.ts +509 -0
  400. package/src/commands/timeline.test.ts +499 -0
  401. package/src/commands/timeline.ts +529 -0
  402. package/src/commands/validate.ts +77 -0
  403. package/src/config/config.test.ts +197 -0
  404. package/src/config/index.ts +125 -0
  405. package/src/config/schema.ts +82 -0
  406. package/src/engine/index.ts +2 -0
  407. package/src/engine/project.test.ts +702 -0
  408. package/src/engine/project.ts +439 -0
  409. package/src/index.ts +146 -0
  410. package/src/utils/api-key.test.ts +41 -0
  411. package/src/utils/api-key.ts +247 -0
  412. package/src/utils/audio.ts +83 -0
  413. package/src/utils/exec-safe.ts +75 -0
  414. package/src/utils/first-run.ts +52 -0
  415. package/src/utils/provider-resolver.ts +56 -0
  416. package/src/utils/remotion.ts +951 -0
  417. package/src/utils/subtitle.test.ts +227 -0
  418. package/src/utils/subtitle.ts +169 -0
  419. package/src/utils/tty.ts +196 -0
  420. package/tsconfig.json +20 -0
@@ -0,0 +1,653 @@
1
+ /**
2
+ * @module ai-image
3
+ * @description Image generation and editing commands for the VibeFrame CLI.
4
+ *
5
+ * ## Commands: vibe ai image, vibe ai thumbnail, vibe ai background,
6
+ * vibe ai gemini, vibe ai gemini-edit
7
+ * ## Dependencies: OpenAI, Gemini, FFmpeg
8
+ *
9
+ * Extracted from ai.ts as part of modularisation.
10
+ * ai.ts calls registerImageCommands(aiCommand).
11
+ * @see MODELS.md for AI model configuration
12
+ */
13
+ import { resolve, dirname, basename, extname } from 'node:path';
14
+ import { fileURLToPath } from 'node:url';
15
+ import { readFile, writeFile, mkdir } from 'node:fs/promises';
16
+ import { existsSync } from 'node:fs';
17
+ import chalk from 'chalk';
18
+ import ora from 'ora';
19
+ import { GeminiProvider, OpenAIImageProvider, } from '@vibeframe/ai-providers';
20
+ import { getApiKey } from '../utils/api-key.js';
21
+ import { execSafe, commandExists } from '../utils/exec-safe.js';
22
+ function _registerImageCommands(aiCommand) {
23
+ aiCommand
24
+ .command("image")
25
+ .description("Generate image using AI (Gemini or DALL-E)")
26
+ .argument("<prompt>", "Image description prompt")
27
+ .option("-p, --provider <provider>", "Provider: gemini, openai, runway (dalle is deprecated)", "gemini")
28
+ .option("-k, --api-key <key>", "API key (or set env: OPENAI_API_KEY, GOOGLE_API_KEY)")
29
+ .option("-o, --output <path>", "Output file path (downloads image)")
30
+ .option("-s, --size <size>", "Image size (openai: 1024x1024, 1536x1024, 1024x1536)", "1024x1024")
31
+ .option("-r, --ratio <ratio>", "Aspect ratio (gemini: 1:1, 1:4, 1:8, 4:1, 8:1, 16:9, 9:16, 3:4, 4:3, etc.)", "1:1")
32
+ .option("-q, --quality <quality>", "Quality: standard, hd (openai only)", "standard")
33
+ .option("--style <style>", "Style: vivid, natural (openai only)", "vivid")
34
+ .option("-n, --count <n>", "Number of images to generate", "1")
35
+ .option("-m, --model <model>", "Gemini model: flash, 3.1-flash, latest (Nano Banana 2), pro (4K)")
36
+ .action(async (prompt, options) => {
37
+ try {
38
+ const provider = options.provider.toLowerCase();
39
+ const validProviders = ["openai", "dalle", "gemini", "runway"];
40
+ if (!validProviders.includes(provider)) {
41
+ console.error(chalk.red(`Invalid provider: ${provider}`));
42
+ console.error(chalk.dim(`Available providers: openai, gemini, runway`));
43
+ process.exit(1);
44
+ }
45
+ // Show deprecation warning for "dalle"
46
+ if (provider === "dalle") {
47
+ console.log(chalk.yellow('Warning: "dalle" is deprecated. Use "openai" instead.'));
48
+ }
49
+ // Get API key based on provider
50
+ const envKeyMap = {
51
+ openai: "OPENAI_API_KEY",
52
+ dalle: "OPENAI_API_KEY", // backward compatibility
53
+ gemini: "GOOGLE_API_KEY",
54
+ runway: "RUNWAY_API_SECRET",
55
+ };
56
+ const providerNameMap = {
57
+ openai: "OpenAI",
58
+ dalle: "OpenAI", // backward compatibility
59
+ gemini: "Google",
60
+ runway: "Runway",
61
+ };
62
+ const envKey = envKeyMap[provider];
63
+ const providerName = providerNameMap[provider];
64
+ const apiKey = await getApiKey(envKey, providerName, options.apiKey);
65
+ if (!apiKey) {
66
+ console.error(chalk.red(`${providerName} API key required. Set ${envKey} in .env or run: vibe setup`));
67
+ console.error(chalk.dim(`Use --api-key or set ${envKey} environment variable`));
68
+ process.exit(1);
69
+ }
70
+ const spinner = ora(`Generating image with ${providerName}...`).start();
71
+ if (provider === "dalle" || provider === "openai") {
72
+ const openaiImage = new OpenAIImageProvider();
73
+ await openaiImage.initialize({ apiKey });
74
+ const result = await openaiImage.generateImage(prompt, {
75
+ size: options.size,
76
+ quality: options.quality,
77
+ style: options.style,
78
+ n: parseInt(options.count),
79
+ });
80
+ if (!result.success || !result.images) {
81
+ spinner.fail(chalk.red(result.error || "Image generation failed"));
82
+ process.exit(1);
83
+ }
84
+ spinner.succeed(chalk.green(`Generated ${result.images.length} image(s) with OpenAI GPT Image 1.5`));
85
+ console.log();
86
+ console.log(chalk.bold.cyan("Generated Images"));
87
+ console.log(chalk.dim("─".repeat(60)));
88
+ for (let i = 0; i < result.images.length; i++) {
89
+ const img = result.images[i];
90
+ console.log();
91
+ if (img.url) {
92
+ console.log(`${chalk.yellow(`[${i + 1}]`)} ${img.url}`);
93
+ }
94
+ else if (img.base64) {
95
+ console.log(`${chalk.yellow(`[${i + 1}]`)} (base64 image data)`);
96
+ }
97
+ if (img.revisedPrompt) {
98
+ console.log(chalk.dim(` Revised: ${img.revisedPrompt.slice(0, 100)}...`));
99
+ }
100
+ }
101
+ console.log();
102
+ // Save if output specified
103
+ if (options.output && result.images.length > 0) {
104
+ const img = result.images[0];
105
+ const saveSpinner = ora("Saving image...").start();
106
+ try {
107
+ let buffer;
108
+ if (img.url) {
109
+ // Download from URL
110
+ const response = await fetch(img.url);
111
+ buffer = Buffer.from(await response.arrayBuffer());
112
+ }
113
+ else if (img.base64) {
114
+ // Decode base64
115
+ buffer = Buffer.from(img.base64, "base64");
116
+ }
117
+ else {
118
+ throw new Error("No image data available");
119
+ }
120
+ const outputPath = resolve(process.cwd(), options.output);
121
+ await mkdir(dirname(outputPath), { recursive: true });
122
+ await writeFile(outputPath, buffer);
123
+ saveSpinner.succeed(chalk.green(`Saved to: ${outputPath}`));
124
+ }
125
+ catch (err) {
126
+ saveSpinner.fail(chalk.red(`Failed to save image: ${err instanceof Error ? err.message : err}`));
127
+ }
128
+ }
129
+ }
130
+ else if (provider === "gemini") {
131
+ // Validate model name
132
+ const validGeminiModels = ["flash", "3.1-flash", "latest", "pro"];
133
+ if (options.model && !validGeminiModels.includes(options.model)) {
134
+ console.warn(chalk.yellow(`Unknown model "${options.model}", using flash. Valid: ${validGeminiModels.join(", ")}`));
135
+ options.model = "flash";
136
+ }
137
+ // Validate aspect ratio
138
+ const validRatios = ["1:1", "1:4", "1:8", "2:3", "3:2", "3:4", "4:1", "4:3", "4:5", "5:4", "8:1", "9:16", "16:9", "21:9"];
139
+ if (options.ratio && !validRatios.includes(options.ratio)) {
140
+ console.error(chalk.red(`Invalid ratio "${options.ratio}". Valid: ${validRatios.join(", ")}`));
141
+ process.exit(1);
142
+ }
143
+ const gemini = new GeminiProvider();
144
+ await gemini.initialize({ apiKey });
145
+ const geminiModelNames = {
146
+ flash: "Nano Banana",
147
+ "3.1-flash": "Nano Banana 2",
148
+ latest: "Nano Banana 2",
149
+ pro: "Nano Banana Pro",
150
+ };
151
+ const modelLabel = geminiModelNames[options.model] || "Nano Banana";
152
+ let result = await gemini.generateImage(prompt, {
153
+ model: options.model,
154
+ aspectRatio: options.ratio,
155
+ });
156
+ // Auto-fallback: if latest/3.1-flash fails, retry with flash
157
+ let usedLabel = modelLabel;
158
+ const fallbackModels = ["latest", "3.1-flash"];
159
+ if (!result.success && options.model && fallbackModels.includes(options.model)) {
160
+ spinner.text = `${chalk.dim(result.error || "Failed")} — retrying with Nano Banana (flash)...`;
161
+ result = await gemini.generateImage(prompt, {
162
+ model: "flash",
163
+ aspectRatio: options.ratio,
164
+ });
165
+ usedLabel = "Nano Banana (fallback)";
166
+ }
167
+ if (!result.success || !result.images) {
168
+ spinner.fail(chalk.red(result.error || "Image generation failed"));
169
+ process.exit(1);
170
+ }
171
+ spinner.succeed(chalk.green(`Generated ${result.images.length} image(s) with Gemini (${usedLabel})`));
172
+ console.log();
173
+ console.log(chalk.bold.cyan("Generated Images"));
174
+ console.log(chalk.dim("─".repeat(60)));
175
+ // Gemini returns base64, we need to save or display
176
+ for (let i = 0; i < result.images.length; i++) {
177
+ const img = result.images[i];
178
+ console.log();
179
+ console.log(`${chalk.yellow(`[${i + 1}]`)} (base64 image, ${img.mimeType})`);
180
+ }
181
+ console.log();
182
+ // Save if output specified
183
+ if (options.output && result.images.length > 0) {
184
+ const saveSpinner = ora("Saving image...").start();
185
+ try {
186
+ const img = result.images[0];
187
+ const buffer = Buffer.from(img.base64, "base64");
188
+ const outputPath = resolve(process.cwd(), options.output);
189
+ await mkdir(dirname(outputPath), { recursive: true });
190
+ await writeFile(outputPath, buffer);
191
+ saveSpinner.succeed(chalk.green(`Saved to: ${outputPath}`));
192
+ }
193
+ catch (err) {
194
+ saveSpinner.fail(chalk.red(`Failed to save image: ${err instanceof Error ? err.message : err}`));
195
+ }
196
+ }
197
+ else {
198
+ console.log(chalk.yellow("Use -o to save the generated image to a file"));
199
+ }
200
+ }
201
+ else if (provider === "runway") {
202
+ // Use Runway's Gemini model for text-to-image (no reference needed)
203
+ const { spawn } = await import("child_process");
204
+ const __filename = fileURLToPath(import.meta.url);
205
+ const __dirname = dirname(__filename);
206
+ const scriptPath = resolve(__dirname, "../../../../.claude/skills/runway-video/scripts/image.py");
207
+ if (!options.output) {
208
+ spinner.fail(chalk.red("Output path required for Runway. Use -o option."));
209
+ process.exit(1);
210
+ }
211
+ const outputPath = resolve(process.cwd(), options.output);
212
+ const args = [scriptPath, prompt, "-o", outputPath, "-r", options.ratio || "16:9"];
213
+ spinner.text = "Generating image with Runway (gemini_2.5_flash)...";
214
+ await new Promise((resolvePromise, reject) => {
215
+ const proc = spawn("python3", args, {
216
+ env: { ...process.env, RUNWAY_API_SECRET: apiKey },
217
+ stdio: ["ignore", "pipe", "pipe"],
218
+ });
219
+ let stdout = "";
220
+ let stderr = "";
221
+ proc.stdout.on("data", (data) => {
222
+ stdout += data.toString();
223
+ });
224
+ proc.stderr.on("data", (data) => {
225
+ stderr += data.toString();
226
+ });
227
+ proc.on("close", (code) => {
228
+ if (code === 0) {
229
+ spinner.succeed(chalk.green("Generated image with Runway"));
230
+ console.log(chalk.dim(stdout.trim()));
231
+ resolvePromise();
232
+ }
233
+ else {
234
+ spinner.fail(chalk.red("Runway image generation failed"));
235
+ console.error(chalk.red(stderr || stdout));
236
+ reject(new Error("Runway generation failed"));
237
+ }
238
+ });
239
+ proc.on("error", (err) => {
240
+ spinner.fail(chalk.red("Failed to run Runway script"));
241
+ reject(err);
242
+ });
243
+ });
244
+ }
245
+ }
246
+ catch (error) {
247
+ console.error(chalk.red("Image generation failed"));
248
+ console.error(error);
249
+ process.exit(1);
250
+ }
251
+ });
252
+ aiCommand
253
+ .command("thumbnail")
254
+ .description("Generate video thumbnail (DALL-E) or extract best frame from video (Gemini)")
255
+ .argument("[description]", "Thumbnail description (for DALL-E generation)")
256
+ .option("-k, --api-key <key>", "API key (OpenAI for generation, Google for best-frame)")
257
+ .option("-o, --output <path>", "Output file path")
258
+ .option("-s, --style <style>", "Platform style: youtube, instagram, tiktok, twitter")
259
+ .option("--best-frame <video>", "Extract best thumbnail frame from video using Gemini AI")
260
+ .option("--prompt <prompt>", "Custom prompt for best-frame analysis")
261
+ .option("--model <model>", "Gemini model: flash, latest, pro (default: flash)", "flash")
262
+ .action(async (description, options) => {
263
+ try {
264
+ // Best-frame mode: analyze video with Gemini and extract frame
265
+ if (options.bestFrame) {
266
+ const absVideoPath = resolve(process.cwd(), options.bestFrame);
267
+ if (!existsSync(absVideoPath)) {
268
+ console.error(chalk.red(`Video not found: ${absVideoPath}`));
269
+ process.exit(1);
270
+ }
271
+ if (!commandExists("ffmpeg")) {
272
+ console.error(chalk.red("FFmpeg not found. Please install FFmpeg."));
273
+ process.exit(1);
274
+ }
275
+ const apiKey = await getApiKey("GOOGLE_API_KEY", "Google", options.apiKey);
276
+ if (!apiKey) {
277
+ console.error(chalk.red("Google API key required for Gemini video analysis. Set GOOGLE_API_KEY in .env or run: vibe setup"));
278
+ console.error(chalk.dim("Use --api-key or set GOOGLE_API_KEY"));
279
+ process.exit(1);
280
+ }
281
+ const name = basename(options.bestFrame, extname(options.bestFrame));
282
+ const outputPath = options.output || `${name}-thumbnail.png`;
283
+ const spinner = ora("Analyzing video for best frame...").start();
284
+ const result = await executeThumbnailBestFrame({
285
+ videoPath: absVideoPath,
286
+ outputPath: resolve(process.cwd(), outputPath),
287
+ prompt: options.prompt,
288
+ model: options.model,
289
+ apiKey,
290
+ });
291
+ if (!result.success) {
292
+ spinner.fail(chalk.red(result.error || "Best frame extraction failed"));
293
+ process.exit(1);
294
+ }
295
+ spinner.succeed(chalk.green("Best frame extracted"));
296
+ console.log();
297
+ console.log(chalk.bold.cyan("Best Frame Result"));
298
+ console.log(chalk.dim("─".repeat(60)));
299
+ console.log(`Timestamp: ${chalk.bold(result.timestamp.toFixed(2))}s`);
300
+ if (result.reason)
301
+ console.log(`Reason: ${chalk.dim(result.reason)}`);
302
+ console.log(`Output: ${chalk.green(result.outputPath)}`);
303
+ console.log();
304
+ return;
305
+ }
306
+ // Generation mode: create thumbnail with DALL-E
307
+ if (!description) {
308
+ console.error(chalk.red("Description required for thumbnail generation."));
309
+ console.error(chalk.dim("Usage: vibe ai thumbnail <description> or vibe ai thumbnail --best-frame <video>"));
310
+ process.exit(1);
311
+ }
312
+ const apiKey = await getApiKey("OPENAI_API_KEY", "OpenAI", options.apiKey);
313
+ if (!apiKey) {
314
+ console.error(chalk.red("OpenAI API key required. Set OPENAI_API_KEY in .env or run: vibe setup"));
315
+ process.exit(1);
316
+ }
317
+ const spinner = ora("Generating thumbnail...").start();
318
+ const openaiImage = new OpenAIImageProvider();
319
+ await openaiImage.initialize({ apiKey });
320
+ const result = await openaiImage.generateThumbnail(description, options.style);
321
+ if (!result.success || !result.images) {
322
+ spinner.fail(chalk.red(result.error || "Thumbnail generation failed"));
323
+ process.exit(1);
324
+ }
325
+ spinner.succeed(chalk.green("Thumbnail generated"));
326
+ const img = result.images[0];
327
+ console.log();
328
+ console.log(chalk.bold.cyan("Generated Thumbnail"));
329
+ console.log(chalk.dim("─".repeat(60)));
330
+ console.log(`URL: ${img.url}`);
331
+ if (img.revisedPrompt) {
332
+ console.log(chalk.dim(`Prompt: ${img.revisedPrompt.slice(0, 100)}...`));
333
+ }
334
+ console.log();
335
+ // Save if output specified
336
+ if (options.output) {
337
+ const saveSpinner = ora("Saving thumbnail...").start();
338
+ try {
339
+ let buffer;
340
+ if (img.url) {
341
+ const response = await fetch(img.url);
342
+ buffer = Buffer.from(await response.arrayBuffer());
343
+ }
344
+ else if (img.base64) {
345
+ buffer = Buffer.from(img.base64, "base64");
346
+ }
347
+ else {
348
+ throw new Error("No image data available");
349
+ }
350
+ const outputPath = resolve(process.cwd(), options.output);
351
+ await mkdir(dirname(outputPath), { recursive: true });
352
+ await writeFile(outputPath, buffer);
353
+ saveSpinner.succeed(chalk.green(`Saved to: ${outputPath}`));
354
+ }
355
+ catch (err) {
356
+ saveSpinner.fail(chalk.red("Failed to save thumbnail"));
357
+ }
358
+ }
359
+ }
360
+ catch (error) {
361
+ console.error(chalk.red("Thumbnail generation failed"));
362
+ console.error(error);
363
+ process.exit(1);
364
+ }
365
+ });
366
+ aiCommand
367
+ .command("background")
368
+ .description("Generate video background using DALL-E")
369
+ .argument("<description>", "Background description")
370
+ .option("-k, --api-key <key>", "OpenAI API key (or set OPENAI_API_KEY env)")
371
+ .option("-o, --output <path>", "Output file path (downloads image)")
372
+ .option("-a, --aspect <ratio>", "Aspect ratio: 16:9, 9:16, 1:1", "16:9")
373
+ .action(async (description, options) => {
374
+ try {
375
+ const apiKey = await getApiKey("OPENAI_API_KEY", "OpenAI", options.apiKey);
376
+ if (!apiKey) {
377
+ console.error(chalk.red("OpenAI API key required. Set OPENAI_API_KEY in .env or run: vibe setup"));
378
+ process.exit(1);
379
+ }
380
+ const spinner = ora("Generating background...").start();
381
+ const openaiImage = new OpenAIImageProvider();
382
+ await openaiImage.initialize({ apiKey });
383
+ const result = await openaiImage.generateBackground(description, options.aspect);
384
+ if (!result.success || !result.images) {
385
+ spinner.fail(chalk.red(result.error || "Background generation failed"));
386
+ process.exit(1);
387
+ }
388
+ spinner.succeed(chalk.green("Background generated"));
389
+ const img = result.images[0];
390
+ console.log();
391
+ console.log(chalk.bold.cyan("Generated Background"));
392
+ console.log(chalk.dim("─".repeat(60)));
393
+ console.log(`Image: ${img.url || "(base64 data)"}`);
394
+ if (img.revisedPrompt) {
395
+ console.log(chalk.dim(`Prompt: ${img.revisedPrompt.slice(0, 100)}...`));
396
+ }
397
+ console.log();
398
+ // Save if output specified
399
+ if (options.output) {
400
+ const saveSpinner = ora("Saving background...").start();
401
+ try {
402
+ let buffer;
403
+ if (img.url) {
404
+ const response = await fetch(img.url);
405
+ buffer = Buffer.from(await response.arrayBuffer());
406
+ }
407
+ else if (img.base64) {
408
+ buffer = Buffer.from(img.base64, "base64");
409
+ }
410
+ else {
411
+ throw new Error("No image data available");
412
+ }
413
+ const outputPath = resolve(process.cwd(), options.output);
414
+ await mkdir(dirname(outputPath), { recursive: true });
415
+ await writeFile(outputPath, buffer);
416
+ saveSpinner.succeed(chalk.green(`Saved to: ${outputPath}`));
417
+ }
418
+ catch (err) {
419
+ saveSpinner.fail(chalk.red("Failed to save background"));
420
+ }
421
+ }
422
+ }
423
+ catch (error) {
424
+ console.error(chalk.red("Background generation failed"));
425
+ console.error(error);
426
+ process.exit(1);
427
+ }
428
+ });
429
+ // Gemini (Nano Banana) commands
430
+ aiCommand
431
+ .command("gemini")
432
+ .description("Generate image using Gemini (Nano Banana)")
433
+ .argument("<prompt>", "Text prompt describing the image")
434
+ .option("-k, --api-key <key>", "Google API key (or set GOOGLE_API_KEY env)")
435
+ .option("-o, --output <path>", "Output file path", "output.png")
436
+ .option("-m, --model <model>", "Model: flash (fast), 3.1-flash / latest (Nano Banana 2), pro (professional, 4K)", "flash")
437
+ .option("-r, --ratio <ratio>", "Aspect ratio: 1:1, 1:4, 1:8, 4:1, 8:1, 16:9, 9:16, 4:3, 3:4, 21:9, etc.", "1:1")
438
+ .option("-s, --size <resolution>", "Resolution: 512px, 1K, 2K, 4K")
439
+ .option("--grounding", "Enable Google Search grounding (Pro only)")
440
+ .option("--thinking <level>", "Enable thinking mode: minimal or high")
441
+ .option("--image-search", "Enable Image Search grounding (3.1 Flash only)")
442
+ .action(async (prompt, options) => {
443
+ try {
444
+ const apiKey = await getApiKey("GOOGLE_API_KEY", "Google", options.apiKey);
445
+ if (!apiKey) {
446
+ console.error(chalk.red("Google API key required. Set GOOGLE_API_KEY in .env or run: vibe setup"));
447
+ console.error(chalk.dim("Use --api-key or set GOOGLE_API_KEY environment variable"));
448
+ process.exit(1);
449
+ }
450
+ const modelNames = {
451
+ flash: "gemini-2.5-flash-image",
452
+ "3.1-flash": "gemini-3.1-flash-image-preview",
453
+ latest: "gemini-3.1-flash-image-preview",
454
+ pro: "gemini-3-pro-image-preview",
455
+ };
456
+ const modelName = modelNames[options.model] || modelNames.flash;
457
+ const spinner = ora(`Generating image with ${modelName}...`).start();
458
+ const gemini = new GeminiProvider();
459
+ await gemini.initialize({ apiKey });
460
+ let result = await gemini.generateImage(prompt, {
461
+ model: options.model,
462
+ aspectRatio: options.ratio,
463
+ resolution: options.size,
464
+ grounding: options.grounding,
465
+ thinkingConfig: options.thinking ? { thinkingLevel: options.thinking } : undefined,
466
+ imageSearchGrounding: options.imageSearch,
467
+ });
468
+ // Auto-fallback: if latest/3.1-flash fails, retry with flash
469
+ const fallbackModels = ["latest", "3.1-flash"];
470
+ if (!result.success && fallbackModels.includes(options.model)) {
471
+ spinner.text = `${chalk.dim(result.error || `${modelName} failed`)} — retrying with flash...`;
472
+ result = await gemini.generateImage(prompt, {
473
+ model: "flash",
474
+ aspectRatio: options.ratio,
475
+ resolution: options.size,
476
+ });
477
+ }
478
+ if (!result.success || !result.images || result.images.length === 0) {
479
+ spinner.fail(chalk.red(result.error || "Image generation failed"));
480
+ process.exit(1);
481
+ }
482
+ spinner.succeed(chalk.green("Image generated"));
483
+ if (result.model) {
484
+ console.log(chalk.dim(`Model: ${result.model}`));
485
+ }
486
+ const img = result.images[0];
487
+ if (img.base64) {
488
+ const outputPath = resolve(process.cwd(), options.output);
489
+ await mkdir(dirname(outputPath), { recursive: true });
490
+ const buffer = Buffer.from(img.base64, "base64");
491
+ await writeFile(outputPath, buffer);
492
+ console.log(chalk.green(`Saved to: ${outputPath}`));
493
+ }
494
+ }
495
+ catch (error) {
496
+ console.error(chalk.red("Image generation failed"));
497
+ console.error(error);
498
+ process.exit(1);
499
+ }
500
+ });
501
+ aiCommand
502
+ .command("gemini-edit")
503
+ .description("Edit image(s) using Gemini (Nano Banana)")
504
+ .argument("<images...>", "Input image file(s) followed by edit prompt")
505
+ .option("-k, --api-key <key>", "Google API key (or set GOOGLE_API_KEY env)")
506
+ .option("-o, --output <path>", "Output file path", "edited.png")
507
+ .option("-m, --model <model>", "Model: flash (max 3 images), 3.1-flash / latest (max 3 images), pro (max 14 images)", "flash")
508
+ .option("-r, --ratio <ratio>", "Output aspect ratio")
509
+ .option("-s, --size <resolution>", "Resolution: 1K, 2K, 4K (Pro model only)")
510
+ .action(async (args, options) => {
511
+ try {
512
+ // Last argument is the prompt, rest are image paths
513
+ if (args.length < 2) {
514
+ console.error(chalk.red("Need at least one image and a prompt"));
515
+ process.exit(1);
516
+ }
517
+ const prompt = args[args.length - 1];
518
+ const imagePaths = args.slice(0, -1);
519
+ const apiKey = await getApiKey("GOOGLE_API_KEY", "Google", options.apiKey);
520
+ if (!apiKey) {
521
+ console.error(chalk.red("Google API key required. Set GOOGLE_API_KEY in .env or run: vibe setup"));
522
+ process.exit(1);
523
+ }
524
+ const spinner = ora(`Reading ${imagePaths.length} image(s)...`).start();
525
+ // Load all images
526
+ const imageBuffers = [];
527
+ for (const imagePath of imagePaths) {
528
+ const absPath = resolve(process.cwd(), imagePath);
529
+ const buffer = await readFile(absPath);
530
+ imageBuffers.push(buffer);
531
+ }
532
+ const editModelNames = {
533
+ flash: "gemini-2.5-flash-image",
534
+ "3.1-flash": "gemini-3.1-flash-image-preview",
535
+ latest: "gemini-3.1-flash-image-preview",
536
+ pro: "gemini-3-pro-image-preview",
537
+ };
538
+ const editModelName = editModelNames[options.model] || editModelNames.flash;
539
+ spinner.text = `Editing with ${editModelName}...`;
540
+ const gemini = new GeminiProvider();
541
+ await gemini.initialize({ apiKey });
542
+ let result = await gemini.editImage(imageBuffers, prompt, {
543
+ model: options.model,
544
+ aspectRatio: options.ratio,
545
+ resolution: options.size,
546
+ });
547
+ // Auto-fallback: if latest/3.1-flash fails, retry with flash
548
+ const fallbackModels = ["latest", "3.1-flash"];
549
+ if (!result.success && fallbackModels.includes(options.model)) {
550
+ spinner.text = `${chalk.dim(result.error || `${editModelName} failed`)} — retrying with flash...`;
551
+ result = await gemini.editImage(imageBuffers, prompt, {
552
+ model: "flash",
553
+ aspectRatio: options.ratio,
554
+ resolution: options.size,
555
+ });
556
+ }
557
+ if (!result.success || !result.images || result.images.length === 0) {
558
+ spinner.fail(chalk.red(result.error || "Image editing failed"));
559
+ process.exit(1);
560
+ }
561
+ spinner.succeed(chalk.green("Image edited"));
562
+ if (result.model) {
563
+ console.log(chalk.dim(`Model: ${result.model}`));
564
+ }
565
+ const img = result.images[0];
566
+ if (img.base64) {
567
+ const outputPath = resolve(process.cwd(), options.output);
568
+ await mkdir(dirname(outputPath), { recursive: true });
569
+ const buffer = Buffer.from(img.base64, "base64");
570
+ await writeFile(outputPath, buffer);
571
+ console.log(chalk.green(`Saved to: ${outputPath}`));
572
+ }
573
+ }
574
+ catch (error) {
575
+ console.error(chalk.red("Image editing failed"));
576
+ console.error(error);
577
+ process.exit(1);
578
+ }
579
+ });
580
+ }
581
+ export async function executeThumbnailBestFrame(options) {
582
+ const { videoPath, outputPath, prompt, model = "flash", apiKey, } = options;
583
+ if (!existsSync(videoPath)) {
584
+ return { success: false, error: `Video not found: ${videoPath}` };
585
+ }
586
+ if (!commandExists("ffmpeg")) {
587
+ return { success: false, error: "FFmpeg not found. Please install FFmpeg." };
588
+ }
589
+ const googleKey = apiKey || process.env.GOOGLE_API_KEY;
590
+ if (!googleKey) {
591
+ return { success: false, error: "Google API key required for Gemini video analysis. Run 'vibe setup' or set GOOGLE_API_KEY in .env" };
592
+ }
593
+ try {
594
+ const gemini = new GeminiProvider();
595
+ await gemini.initialize({ apiKey: googleKey });
596
+ const videoData = await readFile(videoPath);
597
+ const analysisPrompt = prompt ||
598
+ "Analyze this video and find the single best frame for a thumbnail. " +
599
+ "Look for frames that are visually striking, well-composed, emotionally engaging, " +
600
+ "and representative of the video content. Avoid blurry frames, transitions, or dark scenes. " +
601
+ "Return ONLY a JSON object: {\"timestamp\": <seconds as number>, \"reason\": \"<brief explanation>\"}";
602
+ const modelMap = {
603
+ flash: "gemini-3-flash-preview",
604
+ latest: "gemini-2.5-flash",
605
+ "flash-2.5": "gemini-2.5-flash", // backward compat
606
+ pro: "gemini-2.5-pro",
607
+ };
608
+ const modelId = modelMap[model] || "gemini-3-flash-preview";
609
+ const result = await gemini.analyzeVideo(videoData, analysisPrompt, {
610
+ model: modelId,
611
+ fps: 1,
612
+ });
613
+ if (!result.success || !result.response) {
614
+ return { success: false, error: result.error || "Gemini analysis failed" };
615
+ }
616
+ // Parse timestamp from response
617
+ const jsonMatch = result.response.match(/\{[\s\S]*?"timestamp"\s*:\s*([\d.]+)[\s\S]*?\}/);
618
+ if (!jsonMatch) {
619
+ return { success: false, error: `Could not parse timestamp from Gemini response: ${result.response.slice(0, 200)}` };
620
+ }
621
+ const timestamp = parseFloat(jsonMatch[1]);
622
+ let reason;
623
+ const reasonMatch = result.response.match(/"reason"\s*:\s*"([^"]+)"/);
624
+ if (reasonMatch) {
625
+ reason = reasonMatch[1];
626
+ }
627
+ // Extract frame with FFmpeg
628
+ await execSafe("ffmpeg", ["-ss", String(timestamp), "-i", videoPath, "-frames:v", "1", "-q:v", "2", outputPath, "-y"], { timeout: 60000, maxBuffer: 50 * 1024 * 1024 });
629
+ if (!existsSync(outputPath)) {
630
+ return { success: false, error: "FFmpeg failed to extract frame" };
631
+ }
632
+ return {
633
+ success: true,
634
+ outputPath,
635
+ timestamp,
636
+ reason,
637
+ };
638
+ }
639
+ catch (error) {
640
+ return {
641
+ success: false,
642
+ error: `Best frame extraction failed: ${error instanceof Error ? error.message : String(error)}`,
643
+ };
644
+ }
645
+ }
646
+ /**
647
+ * Register all image sub-commands on the given parent command.
648
+ * Called from ai.ts: registerImageCommands(aiCommand)
649
+ */
650
+ export function registerImageCommands(aiCommand) {
651
+ _registerImageCommands(aiCommand);
652
+ }
653
+ //# sourceMappingURL=ai-image.js.map