.agents/skills/cli/references/generate.md
Generate text, images, videos, speech, and transcriptions.
Source: apps/cli/src/commands/generate/
lh generate (alias: gen)
├── text <prompt> # Text generation
├── image <prompt> # Image generation
├── video <prompt> # Video generation
├── tts <text> # Text-to-speech
├── asr <audioFile> # Audio-to-text (speech recognition)
├── download <generationId> <asyncTaskId> # Wait & download generation result
├── status <generationId> <asyncTaskId> # Check async task status
└── list # List generation topics
⚠️ Important:
statusanddownloadrequire anasyncTaskId(UUID format, e.g.7ad0eb13-e9a5-4403-8070-1f7fe95b2f95), not the generation ID (gen_xxx). The asyncTaskId is printed after "→ Task" in thevideo/imagecommand output.
lh generate text <prompt> / lh gen text <prompt>Generate text completion.
Source: apps/cli/src/commands/generate/text.ts
lh gen text "Explain quantum computing" [options]
echo "context" | lh gen text "summarize" --pipe
| Option | Description | Default |
|---|---|---|
-m, --model <model> | Model ID | openai/gpt-4o-mini |
-p, --provider <provider> | Provider name | - |
-s, --system <prompt> | System prompt | - |
--temperature <n> | Temperature (0-2) | - |
--max-tokens <n> | Maximum output tokens | - |
--stream | Enable streaming output | false |
--json | Output full JSON response | false |
--pipe | Read additional context from stdin | false |
When --pipe is used, reads stdin and prepends it to the prompt. Useful for piping file contents:
cat README.md | lh gen text "summarize this" --pipe
lh generate image <prompt> / lh gen image <prompt>Generate images from text prompt. This is an async operation — the command submits the task and returns a generation ID + async task ID for tracking.
Source: apps/cli/src/commands/generate/image.ts
lh gen image "A sunset over mountains" [options]
lh gen image "A cute cat" --model dall-e-3 --provider openai --json
| Option | Description | Default |
|---|---|---|
-m, --model <model> | Model ID | dall-e-3 |
-p, --provider <provider> | Provider name | openai |
-n, --num <n> | Number of images | 1 |
--width <px> | Width in pixels | - |
--height <px> | Height in pixels | - |
--steps <n> | Number of steps | - |
--seed <n> | Random seed | - |
--json | Output raw JSON | false |
Output (non-JSON):
✓ Image generation started
Batch ID: gb_xxx
1 image(s) queued
Generation gen_xxx → Task 7ad0eb13-xxxx-xxxx-xxxx-xxxxxxxxxxxx
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
This is the asyncTaskId — use this for status/download
Use "lh generate status <generationId> <asyncTaskId>" to check progress.
Typical workflow:
# 1. Submit generation — note down BOTH IDs from the output
lh gen image "A cute cat"
# Generation gen_abc123 → Task 7ad0eb13-e9a5-4403-8070-1f7fe95b2f95
# 2. Wait & download using generationId + asyncTaskId (the UUID)
lh gen download gen_abc123 7ad0eb13-e9a5-4403-8070-1f7fe95b2f95 -o cat.png
lh generate video <prompt> / lh gen video <prompt>Generate video from text prompt. This is an async operation.
Source: apps/cli/src/commands/generate/video.ts
lh gen video "A cat playing piano" -m <model> -p <provider> [options]
| Option | Description | Required |
|---|---|---|
-m, --model <model> | Model ID | Yes |
-p, --provider <provider> | Provider name | Yes |
--aspect-ratio <ratio> | Aspect ratio (e.g. 16:9) | No |
--duration <sec> | Duration in seconds | No |
--resolution <res> | Resolution (e.g. 720p) | No |
--seed <n> | Random seed | No |
--json | Output raw JSON | No |
Note: Unlike image, video requires -m and -p (no defaults). Use lh model list <provider> --type video to find available video models.
Output (non-JSON):
✓ Video generation started
Batch ID: gb_xxx
Generation gen_xxx → Task 7ad0eb13-xxxx-xxxx-xxxx-xxxxxxxxxxxx
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
This is the asyncTaskId — use this for status/download
Use "lh generate status <generationId> <asyncTaskId>" to check progress.
Typical workflow:
# 1. Find available video models for a provider
lh model list volcengine --json | grep -i seedance
# 2. Submit generation — note down BOTH IDs from the output
lh gen video "A cat on a runway" -m doubao-seedance-2-0-260128 -p volcengine \
--aspect-ratio 9:16 --duration 5 --resolution 1080p
# Generation gen_abc123 → Task 7ad0eb13-e9a5-4403-8070-1f7fe95b2f95
# 3. Wait & download using generationId + asyncTaskId (the UUID)
lh gen download gen_abc123 7ad0eb13-e9a5-4403-8070-1f7fe95b2f95 -o result.mp4 --timeout 600
lh generate tts <text> / lh gen tts <text>Text-to-speech generation.
Source: apps/cli/src/commands/generate/tts.ts
lh gen tts "Hello, world!" [options]
lh generate asr <audioFile> / lh gen asr <audioFile>Audio-to-text transcription (Automatic Speech Recognition).
Source: apps/cli/src/commands/generate/asr.ts
lh gen asr recording.wav [options]
lh generate download <generationId> <asyncTaskId>Wait for an async generation task to complete and download the result file.
Source: apps/cli/src/commands/generate/index.ts
⚠️
<asyncTaskId>is the UUID printed after "→ Task" in the video/image output. Do not pass the generation ID (gen_xxx) here — that will cause a server error.
lh gen download <generationId> <asyncTaskId> [-o output.png]
lh gen download gen_xxx 7ad0eb13-xxxx-xxxx-xxxx-xxxxxxxxxxxx -o ~/Desktop/result.mp4 --timeout 600
| Option | Description | Default |
|---|---|---|
-o, --output <path> | Output file path (auto-detect extension) | <generationId>.<ext> |
--interval <sec> | Polling interval in seconds | 5 |
--timeout <sec> | Timeout in seconds (0 = no timeout) | 300 |
Behavior:
generation.getGenerationStatus at the specified interval⋯ Status: processing... (42s)lh gen status to check laterlh generate status <generationId> <asyncTaskId>Check the status of an async generation task.
⚠️
<asyncTaskId>is the UUID printed after "→ Task" in the video/image output. Do not pass the generation ID (gen_xxx) here — that will cause a server error.
lh gen status <generationId> <asyncTaskId> [--json]
lh gen status gen_xxx 7ad0eb13-xxxx-xxxx-xxxx-xxxxxxxxxxxx
| Option | Description |
|---|---|
--json | Output raw JSON response |
Displays:
success (green), error (red), processing (yellow), pending (cyan)lh generate listList all generation topics.
lh gen list [--json [fields]]
Table columns: ID, TITLE, TYPE, UPDATED
Image and video generation use an async task pattern:
generationTopic.createTopicimage.createImage / video.createVideo
createAsyncCaller, video via initModelRuntimeFromDB){ data: { batch, generations }, success } with asyncTaskId in each generationgeneration.getGenerationStatus
{ generationId, asyncTaskId } — both are required, and asyncTaskId must be the
UUID from the async_tasks table, not gen_xxx{ status, error, generation } (generation includes asset URLs on success)checkTimeoutTasks which marks tasks as error if they have been
pending or processing for more than ~5 minutes (ASYNC_TASK_TIMEOUT = 298s)Server routes:
src/server/routers/lambda/image/index.ts — image creation (uses authedProcedure + serverDatabase)src/server/routers/lambda/video/index.ts — video creation (uses authedProcedure + serverDatabase)src/server/routers/lambda/generation.ts — status checkingpackages/database/src/models/asyncTask.ts — AsyncTaskModel including checkTimeoutTasksNote: Image/video routes do NOT use the keyVaults middleware — they read API keys from the database via initModelRuntimeFromDB or createAsyncCaller.