AnimeController.php 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880
  1. <?php
  2. namespace App\Http\Controllers\Anime;
  3. use App\Transformer\Anime\AnimeTransformer;
  4. use App\Facade\Site;
  5. use App\Consts\ErrorConst;
  6. use App\Exceptions\ApiException;
  7. use App\Libs\ApiResponse;
  8. use App\Libs\Utils;
  9. use App\Models\MpGenerateVideoTask;
  10. use App\Services\AIGeneration\AIImageGenerationService;
  11. use App\Services\AIGeneration\AIVideoGenerationService;
  12. use App\Services\Anime\AnimeService;
  13. use Illuminate\Http\Request;
  14. use Illuminate\Routing\Controller as BaseController;
  15. use Illuminate\Support\Facades\DB;
  16. use Illuminate\Support\Facades\Redis;
  17. use Illuminate\Support\Facades\Validator;
  18. class AnimeController extends BaseController
  19. {
  20. use ApiResponse;
  21. protected $AnimeService;
  22. protected $AIImageGenerationService;
  23. protected $AIVideoGenerationService;
  24. public function __construct(
  25. AnimeService $AnimeService,
  26. AIImageGenerationService $AIImageGenerationService,
  27. AIVideoGenerationService $AIVideoGenerationService
  28. ) {
  29. $this->AnimeService = $AnimeService;
  30. $this->AIImageGenerationService = $AIImageGenerationService;
  31. $this->AIVideoGenerationService = $AIVideoGenerationService;
  32. }
  33. public function textModel() {
  34. $models = DB::table('mp_text_models')->where('is_enabled', 1)->select('model', 'name')->get()->map(function ($value) {
  35. return (array)$value;
  36. })->toArray();
  37. return $this->success($models);
  38. }
  39. public function artStyleList()
  40. {
  41. $result = [
  42. [
  43. 'art_style' => '日系动漫风格',
  44. 'pic_url' => "https://cdn-zwai.ycsd.cn/mp_audio/art_styles/日系动漫风格.jpg"
  45. ],
  46. [
  47. 'art_style' => '国漫风格',
  48. 'pic_url' => "https://cdn-zwai.ycsd.cn/mp_audio/art_styles/国漫风格.jpg"
  49. ],
  50. [
  51. 'art_style' => 'Q版卡通风格',
  52. 'pic_url' => "https://cdn-zwai.ycsd.cn/mp_audio/art_styles/Q版卡通风格.jpg"
  53. ],
  54. [
  55. 'art_style' => '简约扁平风格',
  56. 'pic_url' => "https://cdn-zwai.ycsd.cn/mp_audio/art_styles/简约扁平风格.jpg"
  57. ],
  58. [
  59. 'art_style' => '古风仙侠风格',
  60. 'pic_url' => "https://cdn-zwai.ycsd.cn/mp_audio/art_styles/古风仙侠风格.jpg"
  61. ],
  62. [
  63. 'art_style' => '武侠风格',
  64. 'pic_url' => "https://cdn-zwai.ycsd.cn/mp_audio/art_styles/武侠风格.jpg"
  65. ],
  66. [
  67. 'art_style' => '新中式水墨风格',
  68. 'pic_url' => "https://cdn-zwai.ycsd.cn/mp_audio/art_styles/新中式水墨风格.jpg"
  69. ],
  70. [
  71. 'art_style' => '写实插画风格',
  72. 'pic_url' => "https://cdn-zwai.ycsd.cn/mp_audio/art_styles/写实插画风格.jpg"
  73. ],
  74. [
  75. 'art_style' => '3D卡通风格',
  76. 'pic_url' => "https://cdn-zwai.ycsd.cn/mp_audio/art_styles/3D卡通风格.jpg"
  77. ],
  78. [
  79. 'art_style' => '条漫风格',
  80. 'pic_url' => "https://cdn-zwai.ycsd.cn/mp_audio/art_styles/条漫风格.jpg"
  81. ],
  82. [
  83. 'art_style' => '赛博朋克风格',
  84. 'pic_url' => "https://cdn-zwai.ycsd.cn/mp_audio/art_styles/赛博朋克风格.jpg"
  85. ],
  86. [
  87. 'art_style' => '暗黑悬疑风格',
  88. 'pic_url' => "https://cdn-zwai.ycsd.cn/mp_audio/art_styles/暗黑悬疑风格.jpg"
  89. ],
  90. [
  91. 'art_style' => '治愈清新风格',
  92. 'pic_url' => "https://cdn-zwai.ycsd.cn/mp_audio/art_styles/治愈清新风格.jpg"
  93. ],
  94. ];
  95. return $this->success($result);
  96. }
  97. // 批量生成主体图片
  98. public function batchSetRoleImg(Request $request) {
  99. // 忽略所有超时限制
  100. set_time_limit(0);
  101. ini_set('max_execution_time', '0');
  102. $data = $request->all();
  103. $result = $this->AnimeService->batchSetRoleImg($data);
  104. return $this->success($result);
  105. }
  106. // 批量生成场景图片
  107. public function batchSetSceneImg(Request $request) {
  108. // 忽略所有超时限制
  109. set_time_limit(0);
  110. ini_set('max_execution_time', '0');
  111. $data = $request->all();
  112. $result = $this->AnimeService->batchSetSceneImg($data);
  113. return $this->success($result);
  114. }
  115. // 编辑动漫
  116. public function editAnime(Request $request) {
  117. $data = $request->all();
  118. $result = $this->AnimeService->editAnime($data);
  119. return $this->success(['success'=>$result ? 1 : 0]);
  120. }
  121. // 修改分集主体列表
  122. public function changeEpisodeRoles(Request $request) {
  123. $data = $request->all();
  124. $result = $this->AnimeService->changeEpisodeRoles($data);
  125. return $this->success(['success'=>$result ? 1 : 0]);
  126. }
  127. // 修改分集场景列表
  128. public function changeEpisodeScenes(Request $request) {
  129. $data = $request->all();
  130. $result = $this->AnimeService->changeEpisodeScenes($data);
  131. return $this->success(['success'=>$result ? 1 : 0]);
  132. }
  133. // 编辑分镜剧本
  134. public function editSegment(Request $request) {
  135. $data = $request->all();
  136. $result = $this->AnimeService->editSegment($data);
  137. return $this->success(['success'=>$result ? 1 : 0]);
  138. }
  139. // 动漫对话列表
  140. public function chatList(Request $request) {
  141. $data = $request->all();
  142. $result = $this->AnimeService->chatList($data);
  143. return $this->success($result, [new AnimeTransformer(), 'newBuildChatList']);
  144. }
  145. // 动漫对话历史记录
  146. public function chatHistory(Request $request) {
  147. $data = $request->all();
  148. $result = $this->AnimeService->chatHistory($data);
  149. return $this->success($result);
  150. }
  151. // 动漫大纲
  152. public function animeDetail(Request $request) {
  153. $data = $request->all();
  154. $result = $this->AnimeService->animeDetail($data);
  155. return $this->success($result);
  156. }
  157. // 动漫剧集
  158. public function episodeInfo(Request $request) {
  159. $data = $request->all();
  160. $result = $this->AnimeService->episodeInfo($data);
  161. return $this->success($result);
  162. }
  163. // 分镜信息
  164. public function segmentInfo(Request $request) {
  165. $data = $request->all();
  166. $result = $this->AnimeService->segmentInfo($data);
  167. return $this->success($result);
  168. }
  169. // 复制剧集副本
  170. public function copyEpisodeVersion(Request $request) {
  171. $data = $request->all();
  172. $result = $this->AnimeService->copyEpisodeVersion($data);
  173. return $this->success(['success' => $result ? 1 : 0]);
  174. }
  175. // 剧集副本列表
  176. public function episodeVersions(Request $request) {
  177. $data = $request->all();
  178. $result = $this->AnimeService->episodeVersions($data);
  179. return $this->success($result);
  180. }
  181. // 绑定剧集副本
  182. public function bindEpisodeVersion(Request $request) {
  183. $data = $request->all();
  184. $result = $this->AnimeService->bindEpisodeVersion($data);
  185. return $this->success(['success' => $result ? 1 : 0]);
  186. }
  187. // 对话改图
  188. public function chatChangeImg(Request $request) {
  189. $data = $request->all();
  190. $result = $this->AnimeService->chatChangeImg($data);
  191. return $this->success(['img_url' => $result]);
  192. }
  193. // 一键生成分镜
  194. public function batchSetSegmentPics(Request $request) {
  195. // 忽略所有超时限制
  196. set_time_limit(0);
  197. ini_set('max_execution_time', '0');
  198. $data = $request->all();
  199. $result = $this->AnimeService->batchSetSegmentPics($data);
  200. return $this->success($result);
  201. }
  202. // 重新生成分镜
  203. public function reGenerateSegment(Request $request) {
  204. // 忽略所有超时限制
  205. set_time_limit(0);
  206. ini_set('max_execution_time', '0');
  207. $data = $request->all();
  208. $result = $this->AnimeService->reGenerateSegment($data);
  209. return $this->success(['img_url'=>$result]);
  210. }
  211. // 添加分镜
  212. public function addSegment(Request $request) {
  213. // 忽略所有超时限制
  214. set_time_limit(0);
  215. ini_set('max_execution_time', '0');
  216. $data = $request->all();
  217. $result = $this->AnimeService->addSegment($data);
  218. return $this->success($result);
  219. }
  220. // 复制分镜
  221. public function copySegment(Request $request) {
  222. $data = $request->all();
  223. $result = $this->AnimeService->copySegment($data);
  224. return $this->success($result);
  225. }
  226. // 移动分镜
  227. public function moveSegment(Request $request) {
  228. $data = $request->all();
  229. $result = $this->AnimeService->moveSegment($data);
  230. return $this->success(['success'=>$result ? 1: 0]);
  231. }
  232. // 删除分镜
  233. public function delSegment(Request $request) {
  234. $data = $request->all();
  235. $result = $this->AnimeService->delSegment($data);
  236. return $this->success(['success'=>$result ? 1: 0]);
  237. }
  238. // 分镜历史图片|视频
  239. public function segmentHistory(Request $request) {
  240. $data = $request->all();
  241. $result = $this->AnimeService->segmentHistory($data);
  242. return $this->success($result);
  243. }
  244. public function createSegmentVideoTask(Request $request) {
  245. // 忽略所有超时限制
  246. set_time_limit(0);
  247. ini_set('max_execution_time', '0');
  248. $data = $request->all();
  249. // 验证参数
  250. $validator = Validator::make($data, [
  251. 'segment_id' => 'required|string',
  252. 'tail_frame' => 'nullable|string|max:500',
  253. ], [
  254. 'segment_id.required' => '分镜ID不能为空',
  255. 'tail_frame.max' => '尾帧描述不能超过500个字符',
  256. ]);
  257. if ($validator->fails()) {
  258. Utils::throwError('1002:' . $validator->errors()->first());
  259. }
  260. // 创建视频生成任务
  261. $result = $this->AnimeService->createSegmentVideoTask($data);
  262. $taskId = $result['task_id'];
  263. // 设置 SSE 响应头
  264. return response()->stream(function () use ($taskId, $result) {
  265. // 设置 SSE 响应头
  266. echo "data: " . json_encode([
  267. 'type' => 'task_created',
  268. 'data' => $result
  269. ]) . "\n\n";
  270. ob_flush();
  271. flush();
  272. $startTime = time();
  273. $maxDuration = 300; // 5分钟超时
  274. $checkInterval = 5; // 每5秒检查一次
  275. while (time() - $startTime < $maxDuration) {
  276. try {
  277. // 查询任务状态
  278. $task = \App\Models\MpGenerateVideoTask::find($taskId);
  279. if (!$task) {
  280. echo "data: " . json_encode([
  281. 'type' => 'error',
  282. 'message' => '任务不存在'
  283. ]) . "\n\n";
  284. ob_flush();
  285. flush();
  286. break;
  287. }
  288. // 如果任务还在处理中,查询最新状态
  289. if ($task->status === 'processing') {
  290. $statusResult = $this->AIVideoGenerationService->querySeedanceTaskStatus($task);
  291. if (isset($statusResult['status'])) {
  292. // 更新任务状态
  293. $task->update([
  294. 'status' => $statusResult['status'],
  295. 'result_url' => $statusResult['result_url'] ?? null,
  296. 'last_frame_url' => $statusResult['last_frame_url'] ?? '',
  297. 'error_message' => $statusResult['error_message'] ?? null,
  298. 'completed_at' => in_array($statusResult['status'], [
  299. 'success',
  300. 'failed'
  301. ]) ? now() : null
  302. ]);
  303. // 如果任务成功,更新分镜表
  304. if ($statusResult['status'] === 'success' && isset($statusResult['result_url'])) {
  305. DB::table('mp_episode_segments')
  306. ->where('video_task_id', $taskId)
  307. ->update([
  308. 'video_url' => $statusResult['result_url'],
  309. 'video_task_status' => '已完成',
  310. 'last_frame_url' => $statusResult['last_frame_url'] ?? '',
  311. 'updated_at' => date('Y-m-d H:i:s')
  312. ]);
  313. } elseif ($statusResult['status'] === 'failed') {
  314. DB::table('mp_episode_segments')
  315. ->where('video_task_id', $taskId)
  316. ->update([
  317. 'video_task_status' => '失败',
  318. 'updated_at' => date('Y-m-d H:i:s')
  319. ]);
  320. }
  321. }
  322. }
  323. // 发送当前状态
  324. echo "data: " . json_encode([
  325. 'type' => 'status_update',
  326. 'data' => [
  327. 'task_id' => $task->id,
  328. 'status' => $task->status,
  329. 'result_url' => $task->result_url,
  330. 'error_message' => $task->error_message,
  331. 'progress' => $this->getTaskProgress($task->status),
  332. 'elapsed_time' => time() - $startTime
  333. ]
  334. ]) . "\n\n";
  335. ob_flush();
  336. flush();
  337. // 如果任务完成(成功或失败),结束连接
  338. if (in_array($task->status, [
  339. 'success',
  340. 'failed'
  341. ])) {
  342. echo "data: " . json_encode([
  343. 'type' => 'completed',
  344. 'data' => [
  345. 'task_id' => $task->id,
  346. 'status' => $task->status,
  347. 'video_url' => $task->result_url,
  348. 'last_frame_url' => $task->last_frame_url,
  349. 'error_message' => $task->error_message
  350. ]
  351. ]) . "\n\n";
  352. ob_flush();
  353. flush();
  354. break;
  355. }
  356. sleep($checkInterval);
  357. } catch (\Exception $e) {
  358. echo "data: " . json_encode([
  359. 'type' => 'error',
  360. 'message' => '查询任务状态失败: ' . $e->getMessage()
  361. ]) . "\n\n";
  362. ob_flush();
  363. flush();
  364. sleep($checkInterval);
  365. }
  366. }
  367. // 超时处理
  368. if (time() - $startTime >= $maxDuration) {
  369. echo "data: " . json_encode([
  370. 'type' => 'timeout',
  371. 'message' => '任务执行超时,请稍后查询任务状态',
  372. 'data' => [
  373. 'task_id' => $taskId
  374. ]
  375. ]) . "\n\n";
  376. ob_flush();
  377. flush();
  378. }
  379. }, 200, [
  380. 'Content-Type' => 'text/event-stream',
  381. 'Cache-Control' => 'no-cache',
  382. 'Connection' => 'keep-alive',
  383. 'X-Accel-Buffering' => 'no', // 禁用 Nginx 缓冲
  384. ]);
  385. }
  386. /**
  387. * 获取任务进度百分比
  388. */
  389. private function getTaskProgress($status) {
  390. switch ($status) {
  391. case \App\Models\MpGenerateVideoTask::STATUS_PENDING:
  392. return 10;
  393. case \App\Models\MpGenerateVideoTask::STATUS_PROCESSING:
  394. return 50;
  395. case \App\Models\MpGenerateVideoTask::STATUS_SUCCESS:
  396. return 100;
  397. case \App\Models\MpGenerateVideoTask::STATUS_FAILED:
  398. return 0;
  399. default:
  400. return 0;
  401. }
  402. }
  403. /**
  404. * 批量生成分镜视频
  405. */
  406. public function batchSetSegmentVideos(Request $request) {
  407. // 忽略所有超时限制
  408. set_time_limit(0);
  409. ini_set('max_execution_time', '0');
  410. $data = $request->all();
  411. // 验证参数
  412. $validator = Validator::make($data, [
  413. 'anime_id' => 'required|string',
  414. 'episode_number' => 'required|integer',
  415. ], [
  416. 'anime_id.required' => '动漫对话ID不能为空',
  417. 'episode_number.required' => '剧集序号不能为空',
  418. 'episode_number.integer' => '剧集序号必须是整数',
  419. ]);
  420. if ($validator->fails()) {
  421. Utils::throwError('1002:' . $validator->errors()->first());
  422. }
  423. $animeId = $data['anime_id'];
  424. $episodeId = $data['episode_id'];
  425. $episodeNumber = $data['episode_number'];
  426. // 获取所有分镜信息
  427. $segments = DB::table('mp_episode_segments')
  428. ->where('anime_id', $animeId)
  429. ->where('episode_id', $episodeId)
  430. ->where('video_url', '')
  431. ->orderBy('segment_number')
  432. ->limit(2)
  433. ->get();
  434. if ($segments->isEmpty()) {
  435. Utils::throwError('20003:未找到分镜数据');
  436. }
  437. // 批量创建视频任务
  438. $taskIds = [];
  439. $segmentTasks = [];
  440. foreach ($segments as $segment) {
  441. try {
  442. // 检查是否已有视频或正在生成中
  443. if (!empty($segment->video_url) || $segment->video_task_status === '生成中') {
  444. continue;
  445. }
  446. // 获取分镜内容
  447. $segmentContent = $segment->segment_content ?: '';
  448. $tailFrame = $segment->tail_frame ?: '';
  449. // 构建完整的提示词
  450. $fullPrompt = $segmentContent;
  451. if ($tailFrame) {
  452. $fullPrompt .= "\n尾帧描述:$tailFrame";
  453. }
  454. // 智能选择视频时长
  455. // $videoDuration = $this->AnimeService->calculateOptimalVideoDuration($segmentContent, $tailFrame);
  456. $videoDuration = -1;
  457. // 构建视频生成参数
  458. $videoParams = [
  459. 'model' => 'doubao-seedance-1-0-pro-250528',
  460. 'alias_segment_id' => $segment->segment_id,
  461. 'prompt' => $fullPrompt,
  462. 'video_duration' => $videoDuration,
  463. 'video_resolution' => '720P',
  464. 'seed' => -1,
  465. 'ratio' => '16:9',
  466. 'generate_audio' => false,
  467. 'draft' => false,
  468. 'watermark' => false,
  469. 'camera_fixed' => false,
  470. // 'callback_url' => 'http://mpaudio.yqsd.cn/api/video/seedanceCallback'
  471. ];
  472. // 如果分镜有图片,作为首帧
  473. if (!empty($segment->img_url)) {
  474. $videoParams['first_frame_url'] = $segment->img_url;
  475. }
  476. // 构建content数组
  477. $videoParams['content'] = [
  478. [
  479. 'type' => 'text',
  480. 'text' => $videoParams['prompt'],
  481. ]
  482. ];
  483. // 如果有首帧图片,添加到content中
  484. if (isset($videoParams['first_frame_url'])) {
  485. $videoParams['content'][] = [
  486. 'type' => 'image_url',
  487. 'image_url' => [
  488. 'url' => $videoParams['first_frame_url'],
  489. ],
  490. 'role' => 'first_frame',
  491. ];
  492. }
  493. // 创建视频任务
  494. $task = $this->AIVideoGenerationService->createSeedanceTask($videoParams);
  495. $taskIds[] = $task->id;
  496. // 更新分镜表的视频任务信息
  497. DB::table('mp_episode_segments')
  498. ->where('segment_id', $segment->segment_id)
  499. ->update([
  500. 'video_task_id' => $task->id,
  501. 'video_task_status' => '生成中',
  502. 'updated_at' => date('Y-m-d H:i:s')
  503. ]);
  504. $segmentTasks[] = [
  505. 'segment_id' => $segment->segment_id,
  506. 'task_id' => $task->id,
  507. 'segment_number' => $segment->segment_number,
  508. ];
  509. } catch (\Exception $e) {
  510. dLog('anime')->error('创建分镜视频任务失败: ' . $e->getMessage(), [
  511. 'segment_id' => $segment->segment_id,
  512. 'anime_id' => $animeId,
  513. 'episode_number' => $episodeNumber
  514. ]);
  515. continue;
  516. }
  517. }
  518. if (empty($taskIds)) {
  519. Utils::throwError('20003:没有需要生成视频的分镜');
  520. }
  521. // 记录开始时间(包含创建任务的时间)
  522. $startTime = time();
  523. $maxDuration = 1800; // 30分钟超时
  524. // 设置 SSE 响应头并开始长连接
  525. return response()->stream(function () use ($taskIds, $segmentTasks, $animeId, $episodeNumber, $episodeId, $startTime, $maxDuration) {
  526. // 发送初始任务创建信息
  527. echo "data: " . json_encode([
  528. 'type' => 'tasks_created',
  529. 'data' => [
  530. 'anime_id' => $animeId,
  531. 'episode_id' => $episodeId,
  532. 'episode_number' => $episodeNumber,
  533. 'total_tasks' => count($taskIds),
  534. 'task_ids' => $taskIds,
  535. 'segments' => $segmentTasks,
  536. 'start_time' => $startTime,
  537. 'max_duration' => $maxDuration
  538. ]
  539. ]) . "\n\n";
  540. ob_flush();
  541. flush();
  542. $checkInterval = 10; // 每10秒检查一次
  543. $completedTasks = [];
  544. $failedTasks = [];
  545. // 检查是否已经超时(包含创建任务的时间)
  546. while (time() - $startTime < $maxDuration) {
  547. try {
  548. $allCompleted = true;
  549. $currentStatus = [];
  550. foreach ($segmentTasks as $segmentTask) {
  551. $taskId = $segmentTask['task_id'];
  552. $segmentId = $segmentTask['segment_id'];
  553. // 跳过已完成或失败的任务
  554. if (in_array($taskId, $completedTasks) || in_array($taskId, $failedTasks)) {
  555. continue;
  556. }
  557. // 查询任务状态
  558. $task = \App\Models\MpGenerateVideoTask::find($taskId);
  559. if (!$task) {
  560. $failedTasks[] = $taskId;
  561. continue;
  562. }
  563. // 如果任务还在处理中,查询最新状态
  564. if ($task->status === 'processing') {
  565. $statusResult = $this->AIVideoGenerationService->querySeedanceTaskStatus($task);
  566. if (isset($statusResult['status'])) {
  567. // 更新任务状态
  568. $task->update([
  569. 'status' => $statusResult['status'],
  570. 'result_json' => $statusResult['result_json'] ?? [],
  571. 'result_url' => $statusResult['result_url'] ?? null,
  572. 'last_frame_url' => $statusResult['last_frame_url'] ?? '',
  573. 'error_message' => $statusResult['error_message'] ?? null,
  574. 'completed_at' => in_array($statusResult['status'], [
  575. 'success',
  576. 'failed'
  577. ]) ? now() : null
  578. ]);
  579. // 如果任务成功,更新分镜表
  580. if ($statusResult['status'] === 'success' && isset($statusResult['result_url'])) {
  581. DB::table('mp_episode_segments')
  582. ->where('segment_id', $segmentId)
  583. ->update([
  584. 'video_url' => $statusResult['result_url'],
  585. 'video_task_status' => '已完成',
  586. 'last_frame_url' => $statusResult['last_frame_url'] ?? '',
  587. 'updated_at' => date('Y-m-d H:i:s')
  588. ]);
  589. $completedTasks[] = $taskId;
  590. // 发送单个任务完成通知
  591. echo "data: " . json_encode([
  592. 'type' => 'task_completed',
  593. 'data' => [
  594. 'segment_id' => $segmentId,
  595. 'task_id' => $taskId,
  596. 'segment_number' => $segmentTask['segment_number'],
  597. 'status' => 'success',
  598. 'video_url' => $statusResult['result_url'],
  599. 'last_frame_url' => $statusResult['last_frame_url'] ?? '',
  600. 'completed_count' => count($completedTasks),
  601. 'total_count' => count($segmentTasks)
  602. ]
  603. ]) . "\n\n";
  604. ob_flush();
  605. flush();
  606. } elseif ($statusResult['status'] === 'failed') {
  607. DB::table('mp_episode_segments')
  608. ->where('segment_id', $segmentId)
  609. ->update([
  610. 'video_task_status' => '失败',
  611. 'updated_at' => date('Y-m-d H:i:s')
  612. ]);
  613. $failedTasks[] = $taskId;
  614. // 发送任务失败通知
  615. echo "data: " . json_encode([
  616. 'type' => 'task_failed',
  617. 'data' => [
  618. 'segment_id' => $segmentId,
  619. 'task_id' => $taskId,
  620. 'segment_number' => $segmentTask['segment_number'],
  621. 'status' => 'failed',
  622. 'error_message' => $statusResult['error_message'] ?? '视频生成失败',
  623. 'failed_count' => count($failedTasks),
  624. 'total_count' => count($segmentTasks)
  625. ]
  626. ]) . "\n\n";
  627. ob_flush();
  628. flush();
  629. }
  630. }
  631. }
  632. // 收集当前状态
  633. $currentStatus[] = [
  634. 'segment_id' => $segmentId,
  635. 'task_id' => $taskId,
  636. 'segment_number' => $segmentTask['segment_number'],
  637. 'status' => $task->status,
  638. 'progress' => $this->getTaskProgress($task->status),
  639. 'video_url' => $task->result_url,
  640. 'error_message' => $task->error_message
  641. ];
  642. // 检查是否还有未完成的任务
  643. if (!in_array($task->status, ['success', 'failed'])) {
  644. $allCompleted = false;
  645. }
  646. }
  647. // 发送整体进度更新
  648. echo "data: " . json_encode([
  649. 'type' => 'progress_update',
  650. 'data' => [
  651. 'anime_id' => $animeId,
  652. 'episode_id' => $episodeId,
  653. 'episode_number' => $episodeNumber,
  654. 'completed_count' => count($completedTasks),
  655. 'failed_count' => count($failedTasks),
  656. 'total_count' => count($segmentTasks),
  657. 'elapsed_time' => time() - $startTime,
  658. 'remaining_time' => max(0, $maxDuration - (time() - $startTime)),
  659. 'segments_status' => $currentStatus
  660. ]
  661. ]) . "\n\n";
  662. ob_flush();
  663. flush();
  664. // 如果所有任务都完成了,结束连接
  665. if ($allCompleted) {
  666. // 获取最终结果
  667. $finalSegments = DB::table('mp_episode_segments')
  668. ->where('anime_id', $animeId)
  669. ->where('episode_id', $episodeId)
  670. ->orderBy('segment_number')
  671. ->select('segment_id', 'segment_number', 'video_url', 'video_task_status', 'last_frame_url')
  672. ->get();
  673. echo "data: " . json_encode([
  674. 'type' => 'all_completed',
  675. 'data' => [
  676. 'anime_id' => $animeId,
  677. 'episode_id' => $episodeId,
  678. 'episode_number' => $episodeNumber,
  679. 'completed_count' => count($completedTasks),
  680. 'failed_count' => count($failedTasks),
  681. 'total_count' => count($segmentTasks),
  682. 'total_elapsed_time' => time() - $startTime,
  683. 'segments' => $finalSegments->map(function($segment) {
  684. return [
  685. 'segment_id' => $segment->segment_id,
  686. 'segment_number' => $segment->segment_number,
  687. 'video_url' => $segment->video_url,
  688. 'status' => $segment->video_task_status,
  689. 'last_frame_url' => $segment->last_frame_url
  690. ];
  691. })->toArray()
  692. ]
  693. ]) . "\n\n";
  694. ob_flush();
  695. flush();
  696. break;
  697. }
  698. // 在睡眠前再次检查是否超时,避免不必要的等待
  699. if (time() - $startTime >= $maxDuration) {
  700. break;
  701. }
  702. sleep($checkInterval);
  703. } catch (\Exception $e) {
  704. echo "data: " . json_encode([
  705. 'type' => 'error',
  706. 'message' => '查询任务状态失败: ' . $e->getMessage(),
  707. 'elapsed_time' => time() - $startTime,
  708. 'remaining_time' => max(0, $maxDuration - (time() - $startTime))
  709. ]) . "\n\n";
  710. ob_flush();
  711. flush();
  712. // 检查是否超时,如果超时则停止循环
  713. if (time() - $startTime >= $maxDuration) {
  714. break;
  715. }
  716. sleep($checkInterval);
  717. }
  718. }
  719. // 超时处理 - 只有在真正超时时才执行
  720. if (time() - $startTime >= $maxDuration) {
  721. // 获取当前所有任务状态
  722. $timeoutSegments = DB::table('mp_episode_segments')
  723. ->where('anime_id', $animeId)
  724. ->where('episode_id', $episodeId)
  725. ->orderBy('segment_number')
  726. ->select('segment_id', 'segment_number', 'video_task_id', 'video_task_status', 'video_url')
  727. ->get();
  728. echo "data: " . json_encode([
  729. 'type' => 'timeout',
  730. 'message' => '批量视频生成已超时30分钟,停止状态查询',
  731. 'data' => [
  732. 'anime_id' => $animeId,
  733. 'episode_id' => $episodeId,
  734. 'episode_number' => $episodeNumber,
  735. 'completed_count' => count($completedTasks),
  736. 'failed_count' => count($failedTasks),
  737. 'total_count' => count($segmentTasks),
  738. 'total_elapsed_time' => time() - $startTime,
  739. 'timeout_duration' => $maxDuration,
  740. 'segments' => $timeoutSegments->map(function($segment) {
  741. return [
  742. 'segment_id' => $segment->segment_id,
  743. 'segment_number' => $segment->segment_number,
  744. 'task_id' => $segment->video_task_id,
  745. 'status' => $segment->video_task_status,
  746. 'video_url' => $segment->video_url
  747. ];
  748. })->toArray()
  749. ]
  750. ]) . "\n\n";
  751. ob_flush();
  752. flush();
  753. }
  754. }, 200, [
  755. 'Content-Type' => 'text/event-stream',
  756. 'Cache-Control' => 'no-cache',
  757. 'Connection' => 'keep-alive',
  758. 'X-Accel-Buffering' => 'no', // 禁用 Nginx 缓冲
  759. ]);
  760. }
  761. // 应用有声制作参数
  762. public function applyAudioData(Request $request) {
  763. $data = $request->all();
  764. $result = $this->AnimeService->applyAudioData($data);
  765. return $this->success($result);
  766. }
  767. }