ImageJob.php 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  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. */
  11. class ImageJob{
  12. /**
  13. * 队列主入口
  14. * @param Job $job 当前队列任务对象
  15. * @param array $data 包含图片路径、提示词、任务ID、链式信息等
  16. */
  17. public function fire(Job $job, $data)
  18. {
  19. $logId = $data['log_id'];
  20. try {
  21. echo date('Y-m-d H:i:s')."图生文开始\n";
  22. // 更新日志状态:处理中
  23. if ($logId) {
  24. Db::name('image_task_log')->where('id', $logId)->update([
  25. 'status' => 1,
  26. 'log' => '图生文处理中',
  27. 'update_time' => date('Y-m-d H:i:s')
  28. ]);
  29. }
  30. //执行逻辑
  31. $result = $this->processImage($data);
  32. echo $result;
  33. // 更新日志状态:成功
  34. if ($logId) {
  35. Db::name('image_task_log')->where('id', $logId)->update([
  36. 'status' => 2,
  37. 'log' => '图生文处理成功',
  38. 'update_time' => date('Y-m-d H:i:s')
  39. ]);
  40. }
  41. echo date('Y-m-d H:i:s')."图生文结束\n";
  42. //链式任务:图生文成功后继续推送文生文
  43. if (!empty($data['chain_next'])) {
  44. // 获取下一个任务类型
  45. $nextType = array_shift($data['chain_next']);
  46. $data['type'] = $nextType;
  47. // 保留剩余链,继续传下去
  48. Queue::push('app\job\ImageArrJob', [
  49. 'task_id' => $data['task_id'],
  50. 'data' => [ $data ]
  51. ], 'arrimage');
  52. }
  53. // 删除当前任务
  54. $job->delete();
  55. } catch (\Exception $e) {
  56. //异常处理,记录失败日志
  57. echo "错误信息: " . $e->getMessage() . "\n";
  58. echo "文件: " . $e->getFile() . "\n";
  59. echo "行号: " . $e->getLine() . "\n";
  60. if ($logId) {
  61. Db::name('image_task_log')->where('id', $logId)->update([
  62. 'status' => -1,
  63. 'log' => '图生文失败:' . $e->getMessage(),
  64. 'update_time' => date('Y-m-d H:i:s')
  65. ]);
  66. }
  67. // 删除当前任务
  68. $job->delete();
  69. }
  70. }
  71. /**
  72. * 失败回调(可用于后续通知或重试机制)
  73. */
  74. public function failed($data)
  75. {
  76. echo "ImageJob failed: " . json_encode($data);
  77. }
  78. public function processImage($data)
  79. {
  80. // 根据传入的数据处理图片
  81. $res = $this->imageToText($data["sourceDir"],$data["file_name"],$data["prompt"],$data["sys_id"],$data["imgtotxt_selectedOption"],$data);
  82. echo $res;
  83. }
  84. /**
  85. * 执行图生文逻辑(图像转文本)
  86. */
  87. public function imageToText($sourceDirRaw, $fileName, $prompt,$sys_id,$imgtotxt_selectedOption,$call_data)
  88. {
  89. // 自动拆分文件名
  90. if (!$fileName && preg_match('/([^\/]+\.(jpg|jpeg|png))$/i', $sourceDirRaw, $matches)) {
  91. $fileName = $matches[1];
  92. $sourceDirRaw = preg_replace('/\/' . preg_quote($fileName, '/') . '$/', '', $sourceDirRaw);
  93. }
  94. if ($sourceDirRaw === '' || $fileName === '') {
  95. return '参数错误:原图路径 或 图片名称 不能为空';
  96. }
  97. // 构建路径
  98. $rootPath = str_replace('\\', '/', ROOT_PATH);
  99. $sourceDir = rtrim($rootPath . 'public/' . $sourceDirRaw, '/') . '/';
  100. $filePath = $sourceDir . $fileName;
  101. $relativePath = $sourceDirRaw . '/' . $fileName;
  102. // 文件夹是否存在(绝对路径检查)
  103. if (!is_dir($sourceDir)) {
  104. return '源目录不存在:' . $sourceDir;
  105. }
  106. // 文件是否存在
  107. if (!is_file($filePath)) {
  108. return '文件不存在:' . $filePath;
  109. }
  110. // 获取图片信息
  111. $ext = strtolower(pathinfo($filePath, PATHINFO_EXTENSION));
  112. $mime = ($ext === 'jpg' || $ext === 'jpeg') ? 'jpeg' : $ext;
  113. list($width, $height) = getimagesize($filePath);
  114. $imageData = base64_encode(file_get_contents($filePath));
  115. if (!$imageData || strlen($imageData) < 1000) {
  116. throw new \Exception('图片内容读取失败');
  117. }
  118. $imageUrl = "data:image/{$mime};base64,{$imageData}";
  119. // 记录提示词日志
  120. $logDir = $rootPath . 'runtime/logs/';
  121. if (!is_dir($logDir)) mkdir($logDir, 0755, true);
  122. // 调用 图生文 接口
  123. $ai = new AIGatewayService();
  124. $gptRes = $ai->callGptApi($imageUrl, $prompt,$imgtotxt_selectedOption);
  125. $gptText = trim($gptRes['choices'][0]['message']['content'] ?? '');
  126. //图生文返回内容日志runtime/logs/img_to_txt.txt
  127. // file_put_contents(
  128. // $logDir . 'img_to_txt.txt',
  129. // "\n======== " . date('Y-m-d H:i:s') . " ========\n" .
  130. // $gptText. "\n\n",
  131. // FILE_APPEND
  132. // );die;
  133. // 验证 GPT 返回格式
  134. if (strpos($gptText, '---json json---') === false) {
  135. return 'GPT 返回格式不正确,缺少分隔符';
  136. }
  137. // 以 ---json json--- 分割
  138. $parts = array_map('trim', explode('---json json---', $gptText));
  139. // 清理“第一段”、“第二段”等标签前缀
  140. $cleanPrefix = function ($text) {
  141. return preg_replace('/^第[一二三四五六七八九十]+段[::]?\s*/u', '', $text);
  142. };
  143. // 防止越界,逐个安全提取
  144. $chineseDesc = isset($parts[0]) ? $cleanPrefix(trim($parts[0])) : '';
  145. $englishDesc = isset($parts[1]) ? $cleanPrefix(trim($parts[1])) : '';
  146. $part2 = isset($parts[2]) ? $cleanPrefix(trim($parts[2])) : '';
  147. // 只保留中英文、数字、下划线、短横线、空格
  148. $img_name = preg_replace('/[^\x{4e00}-\x{9fa5}A-Za-z0-9_\- ]/u', '', $part2);
  149. //图生文返回日志拆分后内容runtime/logs/img_to_txt.txt
  150. // file_put_contents(
  151. // $logDir . 'img_to_txt.txt',
  152. // "\n======== " . date('Y-m-d H:i:s') . " ========\n" .
  153. // $englishDesc . "\n--- 中文 ---\n" .
  154. // $chineseDesc . "\n---英文 ---\n" .
  155. // $img_name . "\n--- 命名 ---\n" ,
  156. // FILE_APPEND
  157. // );
  158. $now = date('Y-m-d H:i:s');
  159. $record = [
  160. 'chinese_description' => $chineseDesc,
  161. 'english_description' => $englishDesc,
  162. 'img_name' => $img_name,
  163. 'sys_id' => $sys_id,
  164. 'status' => 0,
  165. 'status_name' => "图生文",
  166. 'model' => "",
  167. 'size' => "",
  168. 'error_msg' => '',
  169. 'update_time' => $now
  170. ];
  171. // 查询是否存在相同的原图记录
  172. $exists = Db::name('text_to_image')->where('old_image_url', $relativePath)->find();
  173. if ($exists) {
  174. // 如果已存在,执行更新
  175. Db::name('text_to_image')->where('id', $exists['id'])->update($record);
  176. } else {
  177. // 不存在则添加必要字段并插入
  178. $record['old_image_url'] = $relativePath;
  179. $record['new_image_url'] = '';
  180. $record['custom_image_url'] = '';
  181. $record['created_time'] = $now;
  182. Db::name('text_to_image')->insert($record);
  183. }
  184. return ;
  185. }
  186. }