TextToTextJob.php 7.0 KB

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