TextToTextJob.php 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. <?php
  2. namespace app\job;
  3. use app\service\AIGatewayService;
  4. use app\service\ImageService;
  5. use think\Db;
  6. use think\queue\Job;
  7. use think\Queue;
  8. /**
  9. * 文生文
  10. */
  11. class TextToTextJob
  12. {
  13. /**
  14. * 队列任务入口
  15. */
  16. public function fire(Job $job, $data)
  17. {
  18. if (isset($data['status_val']) && $data['status_val'] == '文生文') {
  19. $taskId = $data['task_id'];
  20. $redis = getTaskRedis();
  21. $existing = $redis->get("txt_to_txt_task:{$taskId}");
  22. if ($existing) {
  23. $info = json_decode($existing, true);
  24. if (isset($info['status']) && ImageService::isTaskCompleted($info['status'])) {
  25. echo "任务 {$taskId} 已完成,跳过重复执行\n";
  26. $job->delete();
  27. return;
  28. }
  29. }
  30. try {
  31. echo " 开始处理文生文".date('Y-m-d H:i:s')."\n";
  32. $result = $this->get_txt_to_txt($data);
  33. if (is_array($result) && isset($result['code']) && $result['code'] !== 0) {
  34. throw new \Exception($result['msg'] ?? '文生文失败');
  35. }
  36. echo "🎉 任务 {$taskId} 执行完成,文生文生成成功!\n";
  37. echo "结束时间:" . date('Y-m-d H:i:s') . "\n";
  38. $job->delete();
  39. } catch (\Exception $e) {
  40. echo "文生文失败: " . $e->getMessage() . "\n";
  41. $job->delete();
  42. }
  43. $job->delete();
  44. }else{
  45. $logId = $data['log_id'] ?? null;
  46. echo " 开始处理文生文".date('Y-m-d H:i:s')."\n";
  47. try {
  48. if ($logId) {
  49. Db::name('image_task_log')->where('id', $logId)->update([
  50. 'status' => 1,
  51. 'log' => '文生文处理中',
  52. 'update_time' => date('Y-m-d H:i:s')
  53. ]);
  54. }
  55. $fullPath = rtrim($data['sourceDir'], '/') . '/' . ltrim($data['file_name'], '/');
  56. $list = Db::name("text_to_image")
  57. ->where('old_image_url', $fullPath)
  58. ->where('img_name', '<>', '')
  59. ->select();
  60. if (!empty($list)) {
  61. foreach ($list as $index => $row) {
  62. $currentTime = date('Y-m-d H:i:s');
  63. echo "处理时间:{$currentTime}\n";
  64. echo "👉 正在处理第 " . ($index + 1) . " 条,ID: {$row['id']}\n";
  65. $result = $this->get_txt_to_txt($data,$row['id']);
  66. echo $result;
  67. echo "✅ 处理结果:完成\n";
  68. echo "完成时间:" . date('Y-m-d H:i:s') . "\n";
  69. echo "Processed: " . static::class . "\n\n";
  70. }
  71. // 更新日志为成功
  72. if ($logId) {
  73. Db::name('image_task_log')->where('id', $logId)->update([
  74. 'status' => 2,
  75. 'log' => '文生文处理成功',
  76. 'update_time' => date('Y-m-d H:i:s')
  77. ]);
  78. }
  79. echo "处理完成\n";
  80. } else {
  81. echo "⚠ 未找到可处理的数据,跳过执行\n";
  82. if ($logId) {
  83. Db::name('image_task_log')->where('id', $logId)->update([
  84. 'status' => 2,
  85. 'log' => '无数据可处理,已跳过',
  86. 'update_time' => date('Y-m-d H:i:s')
  87. ]);
  88. }
  89. }
  90. // 链式执行:检查是否还有下一个任务
  91. if (!empty($data['chain_next'])) {
  92. $nextType = array_shift($data['chain_next']); // 获取下一个任务类型
  93. $data['type'] = $nextType;
  94. Queue::push('app\job\ImageArrJob', [
  95. 'task_id' => $data['task_id'],
  96. 'data' => [$data] // 继续传一个任务
  97. ], 'arrimage');
  98. }
  99. $job->delete();
  100. } catch (\Exception $e) {
  101. echo "❌ 错误信息: " . $e->getMessage() . "\n";
  102. echo "📄 文件: " . $e->getFile() . ",第 " . $e->getLine() . " 行\n";
  103. if ($logId) {
  104. Db::name('image_task_log')->where('id', $logId)->update([
  105. 'status' => -1,
  106. 'log' => '文生文失败:' . $e->getMessage(),
  107. 'update_time' => date('Y-m-d H:i:s')
  108. ]);
  109. }
  110. }
  111. $job->delete();
  112. }
  113. }
  114. /**
  115. * 文生文核心处理逻辑(调用 GPT 接口)
  116. * @return string
  117. */
  118. public function get_txt_to_txt($data,$id)
  119. {
  120. $ai = new AIGatewayService();
  121. if ($data['status_type'] == 'ProductImageGeneration') {
  122. $content = "你是专业的产品文案优化师,仅返回优化后的文案,无任何多余内容:
  123. 优化规则:
  124. 1. 语气:亲切自然,符合电商产品描述风格;
  125. 2. 结构:突出产品卖点,逻辑清晰;
  126. 3. 禁忌:不添加无关形容词,不修改产品核心信息;
  127. 需要优化的文案:";
  128. $prompt = $content. $data['prompt'];
  129. $gptRes = $ai->buildRequestData($data['model'],$data['status_val'],$prompt);
  130. $gptText = trim($gptRes['choices'][0]['message']['content']);
  131. return $gptText;
  132. }else{
  133. $template = Db::name('template')
  134. ->field('id,english_content,content')
  135. ->where('path', $data['sourceDir'])
  136. ->where('ids', 1)
  137. ->find();
  138. $record = Db::name('text_to_image')
  139. ->field('id,english_description,chinese_description')
  140. ->where('id', $id)
  141. ->order('id desc')
  142. ->find();
  143. if (!$record) {
  144. return '没有找到匹配的图像记录';
  145. }
  146. // 拼接提示词调用文生文接口
  147. $prompt = $template['english_content'] . $record['chinese_description'];
  148. $gptRes = $ai->buildRequestData($data['model'],$data['status_val'],$prompt);
  149. $gptText = trim($gptRes['choices'][0]['message']['content'] ?? '');
  150. // 更新数据库记录
  151. Db::name('text_to_image')->where('id', $record['id'])->update([
  152. 'english_description' => $gptText,
  153. 'status_name' => "文生文"
  154. ]);
  155. return 0;
  156. }
  157. }
  158. }