video.js
1 /** 2 * Yollomi video generation — POST /api/ai/video 3 * Matches the frontend video-generator.tsx request format exactly. 4 */ 5 import * as path from 'node:path'; 6 import { cli, Strategy } from '@jackwener/opencli/registry'; 7 import { CliError } from '@jackwener/opencli/errors'; 8 import { log } from '@jackwener/opencli/logger'; 9 import { YOLLOMI_DOMAIN, yollomiPost, downloadOutput, fmtBytes } from './utils.js'; 10 cli({ 11 site: 'yollomi', 12 name: 'video', 13 description: 'Generate videos with AI (text-to-video or image-to-video)', 14 domain: YOLLOMI_DOMAIN, 15 strategy: Strategy.COOKIE, 16 args: [ 17 { name: 'prompt', positional: true, required: true, help: 'Text prompt describing the video' }, 18 { name: 'model', default: 'kling-2-1', help: 'Model (kling-2-1, openai-sora-2, google-veo-3-1, wan-2-5-t2v, ...)' }, 19 { name: 'image', help: 'Input image URL for image-to-video' }, 20 { name: 'ratio', default: '16:9', choices: ['1:1', '16:9', '9:16', '4:3', '3:4'], help: 'Aspect ratio' }, 21 { name: 'output', default: './yollomi-output', help: 'Output directory' }, 22 { name: 'no-download', type: 'boolean', default: false, help: 'Only show URL, skip download' }, 23 ], 24 columns: ['status', 'file', 'size', 'credits', 'url'], 25 func: async (page, kwargs) => { 26 const prompt = kwargs.prompt; 27 const modelId = kwargs.model; 28 const inputs = { 29 aspect_ratio: kwargs.ratio, 30 }; 31 if (kwargs.image) 32 inputs.image = kwargs.image; 33 const body = { modelId, prompt, inputs }; 34 log.status(`Generating video with ${modelId} (may take a while)...`); 35 const data = await yollomiPost(page, '/api/ai/video', body); 36 const videoUrl = data.video || ''; 37 if (!videoUrl) 38 throw new CliError('EMPTY_RESPONSE', 'No video returned', 'Try a different prompt or model'); 39 const credits = data.remainingCredits; 40 const noDownload = kwargs['no-download']; 41 const outputDir = kwargs.output; 42 if (noDownload) { 43 return [{ status: 'generated', file: '-', size: '-', credits: credits ?? '-', url: videoUrl }]; 44 } 45 try { 46 const filename = `yollomi_${modelId}_${Date.now()}.mp4`; 47 const { path: fp, size } = await downloadOutput(videoUrl, outputDir, filename); 48 if (credits !== undefined) 49 log.status(`Credits remaining: ${credits}`); 50 return [{ status: 'saved', file: path.relative('.', fp), size: fmtBytes(size), credits: credits ?? '-', url: videoUrl }]; 51 } 52 catch { 53 return [{ status: 'download-failed', file: '-', size: '-', credits: credits ?? '-', url: videoUrl }]; 54 } 55 }, 56 });