@optima-chat/gen-cli 1.0.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 (153) hide show
  1. package/dist/backends/bfl.d.ts +18 -0
  2. package/dist/backends/bfl.d.ts.map +1 -0
  3. package/dist/backends/bfl.js +130 -0
  4. package/dist/backends/bfl.js.map +1 -0
  5. package/dist/backends/comfyui.d.ts +10 -0
  6. package/dist/backends/comfyui.d.ts.map +1 -0
  7. package/dist/backends/comfyui.js +121 -0
  8. package/dist/backends/comfyui.js.map +1 -0
  9. package/dist/backends/index.d.ts +24 -0
  10. package/dist/backends/index.d.ts.map +1 -0
  11. package/dist/backends/index.js +59 -0
  12. package/dist/backends/index.js.map +1 -0
  13. package/dist/backends/types.d.ts +99 -0
  14. package/dist/backends/types.d.ts.map +1 -0
  15. package/dist/backends/types.js +2 -0
  16. package/dist/backends/types.js.map +1 -0
  17. package/dist/backends/video-dashscope.d.ts +10 -0
  18. package/dist/backends/video-dashscope.d.ts.map +1 -0
  19. package/dist/backends/video-dashscope.js +108 -0
  20. package/dist/backends/video-dashscope.js.map +1 -0
  21. package/dist/backends/video-index.d.ts +23 -0
  22. package/dist/backends/video-index.d.ts.map +1 -0
  23. package/dist/backends/video-index.js +57 -0
  24. package/dist/backends/video-index.js.map +1 -0
  25. package/dist/commands/asr.d.ts +3 -0
  26. package/dist/commands/asr.d.ts.map +1 -0
  27. package/dist/commands/asr.js +133 -0
  28. package/dist/commands/asr.js.map +1 -0
  29. package/dist/commands/config.d.ts +3 -0
  30. package/dist/commands/config.d.ts.map +1 -0
  31. package/dist/commands/config.js +105 -0
  32. package/dist/commands/config.js.map +1 -0
  33. package/dist/commands/download.d.ts +3 -0
  34. package/dist/commands/download.d.ts.map +1 -0
  35. package/dist/commands/download.js +169 -0
  36. package/dist/commands/download.js.map +1 -0
  37. package/dist/commands/edit.d.ts +3 -0
  38. package/dist/commands/edit.d.ts.map +1 -0
  39. package/dist/commands/edit.js +155 -0
  40. package/dist/commands/edit.js.map +1 -0
  41. package/dist/commands/generate.d.ts +3 -0
  42. package/dist/commands/generate.d.ts.map +1 -0
  43. package/dist/commands/generate.js +135 -0
  44. package/dist/commands/generate.js.map +1 -0
  45. package/dist/commands/image.d.ts +3 -0
  46. package/dist/commands/image.d.ts.map +1 -0
  47. package/dist/commands/image.js +139 -0
  48. package/dist/commands/image.js.map +1 -0
  49. package/dist/commands/init.d.ts +3 -0
  50. package/dist/commands/init.d.ts.map +1 -0
  51. package/dist/commands/init.js +144 -0
  52. package/dist/commands/init.js.map +1 -0
  53. package/dist/commands/interrupt.d.ts +3 -0
  54. package/dist/commands/interrupt.d.ts.map +1 -0
  55. package/dist/commands/interrupt.js +54 -0
  56. package/dist/commands/interrupt.js.map +1 -0
  57. package/dist/commands/model.d.ts +3 -0
  58. package/dist/commands/model.d.ts.map +1 -0
  59. package/dist/commands/model.js +82 -0
  60. package/dist/commands/model.js.map +1 -0
  61. package/dist/commands/node.d.ts +3 -0
  62. package/dist/commands/node.d.ts.map +1 -0
  63. package/dist/commands/node.js +148 -0
  64. package/dist/commands/node.js.map +1 -0
  65. package/dist/commands/queue.d.ts +3 -0
  66. package/dist/commands/queue.d.ts.map +1 -0
  67. package/dist/commands/queue.js +199 -0
  68. package/dist/commands/queue.js.map +1 -0
  69. package/dist/commands/system.d.ts +3 -0
  70. package/dist/commands/system.d.ts.map +1 -0
  71. package/dist/commands/system.js +85 -0
  72. package/dist/commands/system.js.map +1 -0
  73. package/dist/commands/task.d.ts +3 -0
  74. package/dist/commands/task.d.ts.map +1 -0
  75. package/dist/commands/task.js +176 -0
  76. package/dist/commands/task.js.map +1 -0
  77. package/dist/commands/tryon.d.ts +3 -0
  78. package/dist/commands/tryon.d.ts.map +1 -0
  79. package/dist/commands/tryon.js +178 -0
  80. package/dist/commands/tryon.js.map +1 -0
  81. package/dist/commands/tts.d.ts +3 -0
  82. package/dist/commands/tts.d.ts.map +1 -0
  83. package/dist/commands/tts.js +157 -0
  84. package/dist/commands/tts.js.map +1 -0
  85. package/dist/commands/video.d.ts +3 -0
  86. package/dist/commands/video.d.ts.map +1 -0
  87. package/dist/commands/video.js +137 -0
  88. package/dist/commands/video.js.map +1 -0
  89. package/dist/commands/workflow.d.ts +3 -0
  90. package/dist/commands/workflow.d.ts.map +1 -0
  91. package/dist/commands/workflow.js +244 -0
  92. package/dist/commands/workflow.js.map +1 -0
  93. package/dist/index.d.ts +3 -0
  94. package/dist/index.d.ts.map +1 -0
  95. package/dist/index.js +58 -0
  96. package/dist/index.js.map +1 -0
  97. package/dist/services/api.d.ts +23 -0
  98. package/dist/services/api.d.ts.map +1 -0
  99. package/dist/services/api.js +108 -0
  100. package/dist/services/api.js.map +1 -0
  101. package/dist/services/auth.d.ts +10 -0
  102. package/dist/services/auth.d.ts.map +1 -0
  103. package/dist/services/auth.js +34 -0
  104. package/dist/services/auth.js.map +1 -0
  105. package/dist/services/bfl-api.d.ts +67 -0
  106. package/dist/services/bfl-api.d.ts.map +1 -0
  107. package/dist/services/bfl-api.js +108 -0
  108. package/dist/services/bfl-api.js.map +1 -0
  109. package/dist/services/config.d.ts +12 -0
  110. package/dist/services/config.d.ts.map +1 -0
  111. package/dist/services/config.js +30 -0
  112. package/dist/services/config.js.map +1 -0
  113. package/dist/services/dashscope-api.d.ts +69 -0
  114. package/dist/services/dashscope-api.d.ts.map +1 -0
  115. package/dist/services/dashscope-api.js +134 -0
  116. package/dist/services/dashscope-api.js.map +1 -0
  117. package/dist/services/dashscope-tts-api.d.ts +23 -0
  118. package/dist/services/dashscope-tts-api.d.ts.map +1 -0
  119. package/dist/services/dashscope-tts-api.js +56 -0
  120. package/dist/services/dashscope-tts-api.js.map +1 -0
  121. package/dist/services/fashn-api.d.ts +57 -0
  122. package/dist/services/fashn-api.d.ts.map +1 -0
  123. package/dist/services/fashn-api.js +84 -0
  124. package/dist/services/fashn-api.js.map +1 -0
  125. package/dist/services/generation-api.d.ts +139 -0
  126. package/dist/services/generation-api.d.ts.map +1 -0
  127. package/dist/services/generation-api.js +236 -0
  128. package/dist/services/generation-api.js.map +1 -0
  129. package/dist/services/groq-api.d.ts +18 -0
  130. package/dist/services/groq-api.d.ts.map +1 -0
  131. package/dist/services/groq-api.js +30 -0
  132. package/dist/services/groq-api.js.map +1 -0
  133. package/dist/services/websocket.d.ts +33 -0
  134. package/dist/services/websocket.d.ts.map +1 -0
  135. package/dist/services/websocket.js +217 -0
  136. package/dist/services/websocket.js.map +1 -0
  137. package/dist/utils/helpText.d.ts +15 -0
  138. package/dist/utils/helpText.d.ts.map +1 -0
  139. package/dist/utils/helpText.js +37 -0
  140. package/dist/utils/helpText.js.map +1 -0
  141. package/dist/utils/logger.d.ts +5 -0
  142. package/dist/utils/logger.d.ts.map +1 -0
  143. package/dist/utils/logger.js +14 -0
  144. package/dist/utils/logger.js.map +1 -0
  145. package/dist/utils/output.d.ts +54 -0
  146. package/dist/utils/output.d.ts.map +1 -0
  147. package/dist/utils/output.js +76 -0
  148. package/dist/utils/output.js.map +1 -0
  149. package/dist/utils/workflow.d.ts +17 -0
  150. package/dist/utils/workflow.d.ts.map +1 -0
  151. package/dist/utils/workflow.js +81 -0
  152. package/dist/utils/workflow.js.map +1 -0
  153. package/package.json +61 -0
@@ -0,0 +1,176 @@
1
+ import { success, info } from '../utils/logger.js';
2
+ import { outputSuccess, outputError, isPrettyMode } from '../utils/output.js';
3
+ import { addEnhancedHelp } from '../utils/helpText.js';
4
+ import { GenerationApiClient, GenerationApiError, } from '../services/generation-api.js';
5
+ export function registerTaskCommand(program) {
6
+ const task = program
7
+ .command('task')
8
+ .description('任务管理 - 查看、取消、重试生成任务');
9
+ // gen task get <id>
10
+ const getCmd = task
11
+ .command('get')
12
+ .description('查看任务状态')
13
+ .argument('<task_id>', '任务 ID')
14
+ .action(async (taskId, options) => {
15
+ const pretty = isPrettyMode(options);
16
+ try {
17
+ const client = new GenerationApiClient();
18
+ const result = await client.getTask(taskId);
19
+ if (pretty) {
20
+ success(`任务 ${result.task_id}`);
21
+ info(`类型: ${result.type || '-'}`);
22
+ info(`状态: ${result.status}`);
23
+ if (result.result_url)
24
+ info(`结果: ${result.result_url}`);
25
+ if (result.error_message)
26
+ info(`错误: ${result.error_message}`);
27
+ if (result.created_at)
28
+ info(`创建: ${result.created_at}`);
29
+ if (result.completed_at)
30
+ info(`完成: ${result.completed_at}`);
31
+ }
32
+ else {
33
+ outputSuccess(result);
34
+ }
35
+ }
36
+ catch (err) {
37
+ handleError(err, 'TASK_GET_FAILED', options);
38
+ }
39
+ });
40
+ // gen task list
41
+ const listCmd = task
42
+ .command('list')
43
+ .description('列出我的任务')
44
+ .option('--status <status>', '按状态筛选: pending | completed | failed | cancelled')
45
+ .option('--type <type>', '按类型筛选: image | video | tts | asr | tryon')
46
+ .option('--limit <n>', '返回数量', '20')
47
+ .action(async (options) => {
48
+ const pretty = isPrettyMode(options);
49
+ try {
50
+ const client = new GenerationApiClient();
51
+ const result = await client.listTasks({
52
+ status: options.status,
53
+ type: options.type,
54
+ limit: parseInt(options.limit),
55
+ });
56
+ if (pretty) {
57
+ success(`共 ${result.total} 个任务`);
58
+ for (const task of result.tasks) {
59
+ const statusIcon = task.status === 'completed' ? '✓' :
60
+ task.status === 'failed' ? '✗' :
61
+ task.status === 'pending' ? '○' :
62
+ task.status === 'processing' ? '◐' : '•';
63
+ info(` ${statusIcon} ${task.task_id} ${task.type || '-'} ${task.status} ${task.created_at || ''}`);
64
+ }
65
+ }
66
+ else {
67
+ outputSuccess(result);
68
+ }
69
+ }
70
+ catch (err) {
71
+ handleError(err, 'TASK_LIST_FAILED', options);
72
+ }
73
+ });
74
+ // gen task cancel <id>
75
+ const cancelCmd = task
76
+ .command('cancel')
77
+ .description('取消任务')
78
+ .argument('<task_id>', '任务 ID')
79
+ .action(async (taskId, options) => {
80
+ const pretty = isPrettyMode(options);
81
+ try {
82
+ const client = new GenerationApiClient();
83
+ const result = await client.cancelTask(taskId);
84
+ if (pretty) {
85
+ if (result.status === 'cancelled') {
86
+ success(`任务 ${taskId} 已取消`);
87
+ }
88
+ else {
89
+ info(`取消结果: ${result.status}`);
90
+ if (result.error_message)
91
+ info(result.error_message);
92
+ }
93
+ }
94
+ else {
95
+ outputSuccess(result);
96
+ }
97
+ }
98
+ catch (err) {
99
+ handleError(err, 'TASK_CANCEL_FAILED', options);
100
+ }
101
+ });
102
+ // gen task retry <id>
103
+ const retryCmd = task
104
+ .command('retry')
105
+ .description('重试失败的任务(用原始参数重新提交)')
106
+ .argument('<task_id>', '任务 ID')
107
+ .action(async (taskId, options) => {
108
+ const pretty = isPrettyMode(options);
109
+ try {
110
+ const client = new GenerationApiClient();
111
+ const result = await client.retryTask(taskId);
112
+ if (pretty) {
113
+ success(`已重新提交,新任务 ID: ${result.task_id}`);
114
+ info(`状态: ${result.status}`);
115
+ }
116
+ else {
117
+ outputSuccess(result);
118
+ }
119
+ }
120
+ catch (err) {
121
+ handleError(err, 'TASK_RETRY_FAILED', options);
122
+ }
123
+ });
124
+ // Enhanced help
125
+ addEnhancedHelp(task, {
126
+ examples: [
127
+ '# 查看任务状态',
128
+ '$ gen task get abc-123',
129
+ '',
130
+ '# 列出所有任务',
131
+ '$ gen task list',
132
+ '',
133
+ '# 列出失败的图片任务',
134
+ '$ gen task list --status failed --type image',
135
+ '',
136
+ '# 取消正在进行的任务',
137
+ '$ gen task cancel abc-123',
138
+ '',
139
+ '# 重试失败的任务',
140
+ '$ gen task retry abc-123',
141
+ ],
142
+ outputJson: `{
143
+ "success": true,
144
+ "data": {
145
+ "task_id": "abc-123",
146
+ "type": "image",
147
+ "status": "completed",
148
+ "result_url": "https://...",
149
+ "created_at": "2026-03-21T10:00:00Z",
150
+ "completed_at": "2026-03-21T10:00:15Z"
151
+ }
152
+ }`,
153
+ relatedCommands: [
154
+ { command: 'image <prompt>', description: '生成图像' },
155
+ { command: 'video <image>', description: '生成视频' },
156
+ { command: 'tts <text>', description: '文本转语音' },
157
+ { command: 'asr <file>', description: '语音识别' },
158
+ { command: 'tryon <model> <garment>', description: '虚拟试穿' },
159
+ ],
160
+ notes: [
161
+ '所有生成命令会自动轮询等待结果',
162
+ '如果轮询超时,可以用 gen task get <id> 手动查看',
163
+ '取消正在执行的任务时,如果官方 API 不支持取消,任务会继续执行,结果仍可查看',
164
+ ],
165
+ });
166
+ }
167
+ function handleError(err, code, options) {
168
+ if (err instanceof GenerationApiError) {
169
+ outputError(err.code, err.message, options);
170
+ }
171
+ else {
172
+ const message = err instanceof Error ? err.message : String(err);
173
+ outputError(code, message, options);
174
+ }
175
+ }
176
+ //# sourceMappingURL=task.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"task.js","sourceRoot":"","sources":["../../src/commands/task.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import { Command } from 'commander';
2
+ export declare function registerTryonCommand(program: Command): void;
3
+ //# sourceMappingURL=tryon.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tryon.d.ts","sourceRoot":"","sources":["../../src/commands/tryon.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAepC,wBAAgB,oBAAoB,CAAC,OAAO,EAAE,OAAO,QAiLpD"}
@@ -0,0 +1,178 @@
1
+ import * as path from 'path';
2
+ import ora from 'ora';
3
+ import { success, info } from '../utils/logger.js';
4
+ import { outputSuccess, outputError, isPrettyMode } from '../utils/output.js';
5
+ import { addEnhancedHelp } from '../utils/helpText.js';
6
+ import { getConfigValue } from '../services/config.js';
7
+ import { GenerationApiClient, GenerationApiError, imageToBase64DataUrl, downloadResult, } from '../services/generation-api.js';
8
+ export function registerTryonCommand(program) {
9
+ const cmd = program
10
+ .command('tryon')
11
+ .description('虚拟试穿 (Virtual Try-On) - 使用 FASHN AI tryon-v1.6')
12
+ .argument('<model_image>', '人物照片路径或 URL')
13
+ .argument('<garment_image>', '服装照片路径或 URL')
14
+ .option('-o, --output <path>', '输出文件路径', '')
15
+ .option('-c, --category <type>', '服装类别: auto, tops, bottoms, one-pieces', 'auto')
16
+ .option('-m, --mode <mode>', '生成模式: performance (~7s), balanced (~9s), quality (~13s)', 'quality')
17
+ .option('--garment-type <type>', '服装照片类型: auto, flat-lay, model', 'auto')
18
+ .option('-n, --num-samples <n>', '生成数量 (1-4)', '1')
19
+ .option('--format <fmt>', '输出格式: png, jpeg', 'png')
20
+ .option('--seed <n>', '随机种子(用于复现)')
21
+ .action(async (modelImage, garmentImage, options) => {
22
+ const pretty = isPrettyMode(options);
23
+ try {
24
+ // 处理图像输入:本地文件转 base64,URL 直接使用
25
+ const resolveImage = async (input) => {
26
+ if (input.startsWith('http://') || input.startsWith('https://')) {
27
+ return input;
28
+ }
29
+ return imageToBase64DataUrl(input);
30
+ };
31
+ const [modelImg, garmentImg] = await Promise.all([
32
+ resolveImage(modelImage),
33
+ resolveImage(garmentImage),
34
+ ]);
35
+ const numSamples = parseInt(options.numSamples, 10);
36
+ const outputDir = getConfigValue('outputDir') || './gen-output';
37
+ const params = {
38
+ model_image: modelImg,
39
+ garment_image: garmentImg,
40
+ category: options.category,
41
+ mode: options.mode,
42
+ garment_photo_type: options.garmentType,
43
+ num_samples: numSamples,
44
+ output_format: options.format,
45
+ seed: options.seed ? parseInt(options.seed, 10) : undefined,
46
+ };
47
+ const client = new GenerationApiClient();
48
+ const startTime = Date.now();
49
+ if (pretty) {
50
+ const spinner = ora(`正在提交虚拟试穿任务 (${options.mode} 模式)...`).start();
51
+ try {
52
+ const result = await client.submitAndPoll('tryon', params, {
53
+ onProgress: (status) => {
54
+ spinner.text = `虚拟试穿中 (${status})...`;
55
+ },
56
+ });
57
+ if (result.status === 'completed' && result.result_url) {
58
+ spinner.text = '正在下载结果图像...';
59
+ // result_url 可能是单个 URL 或 result_meta 里有多个
60
+ const outputs = [];
61
+ const urls = result.result_meta?.output_urls || [result.result_url];
62
+ const ts = Date.now();
63
+ for (let i = 0; i < urls.length; i++) {
64
+ const outputPath = options.output && urls.length === 1
65
+ ? path.resolve(options.output)
66
+ : path.resolve(outputDir, `tryon_${ts}_${i}.${options.format}`);
67
+ const dir = path.dirname(outputPath);
68
+ const filename = path.basename(outputPath);
69
+ const saved = await downloadResult(urls[i], dir, filename);
70
+ outputs.push(saved);
71
+ }
72
+ const durationMs = Date.now() - startTime;
73
+ spinner.succeed('虚拟试穿完成!');
74
+ for (const out of outputs) {
75
+ success(`输出文件: ${out}`);
76
+ }
77
+ info(`模式: ${options.mode}, 类别: ${options.category}, 耗时: ${(durationMs / 1000).toFixed(1)}s`);
78
+ }
79
+ else {
80
+ spinner.info(`任务状态: ${result.status}`);
81
+ if (result.error_message)
82
+ info(`错误: ${result.error_message}`);
83
+ info(`使用 'gen task get ${result.task_id}' 查看最新状态`);
84
+ }
85
+ }
86
+ catch (err) {
87
+ spinner.fail(`试穿失败: ${err?.message || err}`);
88
+ throw err;
89
+ }
90
+ }
91
+ else {
92
+ const result = await client.submitAndPoll('tryon', params);
93
+ if (result.status === 'completed' && result.result_url) {
94
+ const outputs = [];
95
+ const urls = result.result_meta?.output_urls || [result.result_url];
96
+ const ts = Date.now();
97
+ for (let i = 0; i < urls.length; i++) {
98
+ const outputPath = options.output && urls.length === 1
99
+ ? path.resolve(options.output)
100
+ : path.resolve(outputDir, `tryon_${ts}_${i}.${options.format}`);
101
+ const dir = path.dirname(outputPath);
102
+ const filename = path.basename(outputPath);
103
+ const saved = await downloadResult(urls[i], dir, filename);
104
+ outputs.push(saved);
105
+ }
106
+ const durationMs = Date.now() - startTime;
107
+ outputSuccess({
108
+ task_id: result.task_id,
109
+ outputs,
110
+ mode: options.mode,
111
+ category: options.category,
112
+ numSamples,
113
+ durationMs,
114
+ });
115
+ }
116
+ else {
117
+ outputSuccess({
118
+ task_id: result.task_id,
119
+ status: result.status,
120
+ error_message: result.error_message,
121
+ });
122
+ }
123
+ }
124
+ }
125
+ catch (err) {
126
+ if (err instanceof GenerationApiError) {
127
+ outputError(err.code, err.message, options);
128
+ }
129
+ else {
130
+ outputError('TRYON_FAILED', `虚拟试穿失败: ${err}`, options);
131
+ }
132
+ }
133
+ });
134
+ addEnhancedHelp(cmd, {
135
+ examples: [
136
+ '# 基本用法 - 本地图片试穿',
137
+ '$ gen tryon person.jpg garment.jpg',
138
+ '',
139
+ '# 使用 URL 图片',
140
+ '$ gen tryon https://example.com/model.jpg https://example.com/dress.jpg',
141
+ '',
142
+ '# 指定服装类别和生成模式',
143
+ '$ gen tryon person.jpg top.jpg -c tops -m performance',
144
+ '',
145
+ '# 生成多张结果',
146
+ '$ gen tryon person.jpg dress.jpg -n 4 -c one-pieces',
147
+ '',
148
+ '# 指定输出路径',
149
+ '$ gen tryon person.jpg garment.jpg -o result.png',
150
+ '',
151
+ '# 使用种子复现结果',
152
+ '$ gen tryon person.jpg garment.jpg --seed 42',
153
+ ],
154
+ outputJson: `{
155
+ "success": true,
156
+ "data": {
157
+ "task_id": "abc123",
158
+ "outputs": ["./gen-output/tryon_1234567890_0.png"],
159
+ "mode": "quality",
160
+ "category": "auto",
161
+ "numSamples": 1,
162
+ "durationMs": 13000
163
+ }
164
+ }`,
165
+ relatedCommands: [
166
+ { command: 'image <prompt>', description: '文本生成图像' },
167
+ { command: 'task get <id>', description: '查看任务状态' },
168
+ ],
169
+ notes: [
170
+ '通过 optima-generation 调用 FASHN AI tryon-v1.6 模型',
171
+ '需要 Optima 账号(OPTIMA_TOKEN 或 ~/.optima/token.json)',
172
+ '支持本地图片路径和 URL',
173
+ '服装类别: auto (自动检测), tops (上装), bottoms (下装), one-pieces (连体)',
174
+ '生成模式: performance (~7s), balanced (~9s), quality (~13s)',
175
+ ],
176
+ });
177
+ }
178
+ //# sourceMappingURL=tryon.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tryon.js","sourceRoot":"","sources":["../../src/commands/tryon.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import { Command } from 'commander';
2
+ export declare function registerTtsCommand(program: Command): void;
3
+ //# sourceMappingURL=tts.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tts.d.ts","sourceRoot":"","sources":["../../src/commands/tts.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAqBpC,wBAAgB,kBAAkB,CAAC,OAAO,EAAE,OAAO,QAoJlD"}
@@ -0,0 +1,157 @@
1
+ import * as path from 'path';
2
+ import ora from 'ora';
3
+ import { success, info } from '../utils/logger.js';
4
+ import { outputSuccess, outputError, isPrettyMode } from '../utils/output.js';
5
+ import { addEnhancedHelp } from '../utils/helpText.js';
6
+ import { getConfigValue } from '../services/config.js';
7
+ import { GenerationApiClient, GenerationApiError, validateTTSParams, downloadResult, } from '../services/generation-api.js';
8
+ const TTS_VOICES = [
9
+ 'Cherry', 'Serena', 'Chelsie', 'Aria', 'Bella', 'Vivian', 'Jennifer', 'Katerina',
10
+ 'Breeze', 'Maple', 'Sakura',
11
+ 'Ethan', 'Aiden', 'Brandon', 'Ryan', 'Luke', 'Leo',
12
+ ];
13
+ export function registerTtsCommand(program) {
14
+ const cmd = program
15
+ .command('tts')
16
+ .description('文本转语音 (Text-to-Speech) - 使用 DashScope qwen3-tts-flash')
17
+ .argument('<text>', '要转换的文本内容')
18
+ .option('-o, --output <path>', '输出文件路径', '')
19
+ .option('--voice <name>', '声音名称 (默认: Cherry)', 'Cherry')
20
+ .option('--voices', '列出可用声音')
21
+ .option('--play', '生成后自动播放音频')
22
+ .action(async (text, options) => {
23
+ const pretty = isPrettyMode(options);
24
+ // 列出可用声音
25
+ if (options.voices) {
26
+ if (pretty) {
27
+ info('可用声音列表 (qwen3-tts-flash):');
28
+ console.log(` 女声: Cherry, Serena, Chelsie, Aria, Bella, Vivian, Jennifer, Katerina, Breeze, Maple, Sakura`);
29
+ console.log(` 男声: Ethan, Aiden, Brandon, Ryan, Luke, Leo`);
30
+ }
31
+ else {
32
+ outputSuccess({ voices: TTS_VOICES });
33
+ }
34
+ return;
35
+ }
36
+ try {
37
+ // CLI 层快速校验
38
+ validateTTSParams({ text });
39
+ const outputDir = getConfigValue('outputDir') || './gen-output';
40
+ const outputPath = options.output
41
+ ? path.resolve(options.output)
42
+ : path.resolve(outputDir, `tts_${Date.now()}.wav`);
43
+ const params = {
44
+ text,
45
+ voice: options.voice,
46
+ };
47
+ const client = new GenerationApiClient();
48
+ const startTime = Date.now();
49
+ if (pretty) {
50
+ const spinner = ora(`正在合成语音 (${options.voice})...`).start();
51
+ try {
52
+ const result = await client.submitAndPoll('tts', params, {
53
+ onProgress: (status) => {
54
+ spinner.text = `语音合成中 (${status})...`;
55
+ },
56
+ });
57
+ if (result.status === 'completed' && result.result_url) {
58
+ spinner.text = '正在下载音频...';
59
+ const filename = path.basename(outputPath);
60
+ const dir = path.dirname(outputPath);
61
+ await downloadResult(result.result_url, dir, filename);
62
+ const durationMs = Date.now() - startTime;
63
+ spinner.succeed('语音合成完成!');
64
+ success(`输出文件: ${outputPath}`);
65
+ info(`声音: ${options.voice}, 耗时: ${(durationMs / 1000).toFixed(1)}s`);
66
+ // 自动播放
67
+ if (options.play) {
68
+ const { execSync } = await import('child_process');
69
+ info('正在播放...');
70
+ execSync(`afplay "${outputPath}"`, { stdio: 'inherit' });
71
+ }
72
+ }
73
+ else {
74
+ spinner.info(`任务状态: ${result.status}`);
75
+ if (result.error_message)
76
+ info(`错误: ${result.error_message}`);
77
+ info(`使用 'gen task get ${result.task_id}' 查看最新状态`);
78
+ }
79
+ }
80
+ catch (err) {
81
+ spinner.fail(`合成失败: ${err?.message || err}`);
82
+ throw err;
83
+ }
84
+ }
85
+ else {
86
+ const result = await client.submitAndPoll('tts', params);
87
+ if (result.status === 'completed' && result.result_url) {
88
+ const filename = path.basename(outputPath);
89
+ const dir = path.dirname(outputPath);
90
+ await downloadResult(result.result_url, dir, filename);
91
+ const durationMs = Date.now() - startTime;
92
+ outputSuccess({
93
+ task_id: result.task_id,
94
+ output: outputPath,
95
+ voice: options.voice,
96
+ text,
97
+ durationMs,
98
+ });
99
+ }
100
+ else {
101
+ outputSuccess({
102
+ task_id: result.task_id,
103
+ status: result.status,
104
+ error_message: result.error_message,
105
+ });
106
+ }
107
+ }
108
+ }
109
+ catch (err) {
110
+ if (err instanceof GenerationApiError) {
111
+ outputError(err.code, err.message, options);
112
+ }
113
+ else {
114
+ outputError('TTS_FAILED', `语音合成失败: ${err}`, options);
115
+ }
116
+ }
117
+ });
118
+ addEnhancedHelp(cmd, {
119
+ examples: [
120
+ '# 基本用法 - 中文语音合成',
121
+ '$ gen tts "你好,欢迎使用语音合成"',
122
+ '',
123
+ '# 指定输出文件和声音',
124
+ '$ gen tts "Hello world" -o hello.wav --voice Ethan',
125
+ '',
126
+ '# 使用男声',
127
+ '$ gen tts "今天天气真好" --voice Ryan',
128
+ '',
129
+ '# 生成后自动播放',
130
+ '$ gen tts "测试语音" --play',
131
+ '',
132
+ '# 列出可用声音',
133
+ '$ gen tts "" --voices',
134
+ ],
135
+ outputJson: `{
136
+ "success": true,
137
+ "data": {
138
+ "task_id": "abc-123",
139
+ "output": "./gen-output/tts_1234567890.wav",
140
+ "voice": "Cherry",
141
+ "text": "你好",
142
+ "durationMs": 1200
143
+ }
144
+ }`,
145
+ relatedCommands: [
146
+ { command: 'asr <file>', description: '语音识别(语音转文字)' },
147
+ { command: 'task get <id>', description: '查看任务状态' },
148
+ ],
149
+ notes: [
150
+ '通过 optima-generation 调用 DashScope qwen3-tts-flash 模型',
151
+ '需要 Optima 账号(OPTIMA_TOKEN 或 ~/.optima/token.json)',
152
+ '支持 50+ 种声音,支持中、英、日、韩等 10 种语言',
153
+ '输出格式为 WAV',
154
+ ],
155
+ });
156
+ }
157
+ //# sourceMappingURL=tts.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tts.js","sourceRoot":"","sources":["../../src/commands/tts.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import { Command } from 'commander';
2
+ export declare function registerVideoCommand(program: Command): void;
3
+ //# sourceMappingURL=video.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"video.d.ts","sourceRoot":"","sources":["../../src/commands/video.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAiBpC,wBAAgB,oBAAoB,CAAC,OAAO,EAAE,OAAO,QAkIpD"}