/ clis / yollomi / video.js
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  });