ImageJob.php 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316
  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. if (empty($data['status_val']) || $data['status_val'] == '图生文') {
  20. // 获取任务ID
  21. $taskId = $data['id'] ?? null;
  22. if (empty($taskId)) {
  23. $job->delete();
  24. return;
  25. }
  26. // 创建Redis连接
  27. $redis = new \Redis();
  28. $redis->connect('127.0.0.1', 6379);
  29. $redis->auth('123456');
  30. $redis->select(15);
  31. // 设置锁键名
  32. $lockKey = "image_to_text_lock:{$taskId}";
  33. $lockExpire = 300; // 锁的过期时间,单位:秒(5分钟)
  34. // 尝试获取锁(使用SET命令的NX和EX选项进行原子操作)
  35. // NX: 只在键不存在时设置
  36. // EX: 设置过期时间
  37. $lockAcquired = $redis->set($lockKey, time(), ['NX', 'EX' => $lockExpire]);
  38. if (!$lockAcquired) {
  39. // 无法获取锁,任务正在执行
  40. echo "❌ 检测到相同ID({$taskId})的任务正在执行,当前任务跳过\n";
  41. $job->delete();
  42. return;
  43. } else {
  44. echo "🔒 成功获取任务锁,ID: {$taskId}\n";
  45. }
  46. echo "━━━━━━━━━━ ▶ 图生文任务开始处理━━━━━━━━━━\n";
  47. $result = $this->get_img_to_txt($data);
  48. // 标准化结果文本
  49. if ($result === true || $result === 1 || $result === '成功') {
  50. $resultText = '成功';
  51. } else {
  52. $resultText = (string) $result ?: '失败或无返回';
  53. }
  54. echo "✅ 处理结果:{$resultText}\n";
  55. echo "完成时间:" . date('Y-m-d H:i:s') . "\n";
  56. echo "图生文已处理完成\n";
  57. // 释放锁 - 使用del()替代被弃用的delete()方法
  58. $redis->del($lockKey);
  59. echo "🔓 释放任务锁,ID: {$taskId}\n";
  60. $job->delete();
  61. die;
  62. } else {
  63. $logId = $data['log_id'];
  64. try {
  65. echo date('Y-m-d H:i:s')."图生文开始\n";
  66. // 更新日志状态:处理中
  67. if ($logId) {
  68. Db::name('image_task_log')->where('id', $logId)->update([
  69. 'status' => 1,
  70. 'log' => '图生文处理中',
  71. 'update_time' => date('Y-m-d H:i:s')
  72. ]);
  73. }
  74. //执行逻辑
  75. $result = $this->processImage($data);
  76. echo $result;
  77. // 更新日志状态:成功
  78. if ($logId) {
  79. Db::name('image_task_log')->where('id', $logId)->update([
  80. 'status' => 2,
  81. 'log' => '图生文处理成功',
  82. 'update_time' => date('Y-m-d H:i:s')
  83. ]);
  84. }
  85. echo date('Y-m-d H:i:s')."图生文结束\n";
  86. //链式任务:图生文成功后继续推送文生文
  87. if (!empty($data['chain_next'])) {
  88. // 获取下一个任务类型
  89. $nextType = array_shift($data['chain_next']);
  90. $data['type'] = $nextType;
  91. // 保留剩余链,继续传下去
  92. Queue::push('app\job\ImageArrJob', [
  93. 'task_id' => $data['task_id'],
  94. 'data' => [ $data ]
  95. ], 'arrimage');
  96. }
  97. // 删除当前任务
  98. $job->delete();
  99. } catch (\Exception $e) {
  100. //异常处理,记录失败日志
  101. echo "错误信息: " . $e->getMessage() . "\n";
  102. echo "文件: " . $e->getFile() . "\n";
  103. echo "行号: " . $e->getLine() . "\n";
  104. if ($logId) {
  105. Db::name('image_task_log')->where('id', $logId)->update([
  106. 'status' => -1,
  107. 'log' => '图生文失败:' . $e->getMessage(),
  108. 'update_time' => date('Y-m-d H:i:s')
  109. ]);
  110. }
  111. // 删除当前任务
  112. $job->delete();
  113. }
  114. }
  115. $job->delete();
  116. }
  117. /**
  118. * 失败回调(可用于后续通知或重试机制)
  119. */
  120. public function failed($data)
  121. {
  122. echo "ImageJob failed: " . json_encode($data);
  123. }
  124. public function processImage($data)
  125. {
  126. // 根据传入的数据处理图片
  127. $res = $this->imageToText($data["sourceDir"],$data["file_name"],$data["prompt"],$data["sys_id"],$data["imgtotxt_selectedOption"],$data);
  128. echo $res;
  129. }
  130. public function get_img_to_txt($params){
  131. $prompt = $params['prompt'];//提示词
  132. $old_path = $params['path'] ?? '';//原图路径
  133. $model = $params['model'];//模型
  134. if(empty($model)) {
  135. return json(['code' => 1, 'msg' => '模型请求失败']);
  136. }
  137. $aiGateway = new AIGatewayService();
  138. // 获取图片的base64数据和MIME类型
  139. $imageData = AIGatewayService::file_get_contents($old_path);
  140. $base64Data = $imageData['base64Data'];
  141. $mimeType = $imageData['mimeType'];
  142. $formattedPrompt = "1. 输出语言:所有内容必须为纯简体中文,禁止出现任何英文、拼音、数字、注释、解释性文字、引导语、示例、标点外的特殊符号;
  143. 2. 第一步(提取原图产品):
  144. - 用1句完整中文描述原图产品,字数控制在50字以内;
  145. - 必须包含「主体细节、产品名称、商标、类型、颜色、风格」核心;
  146. 3. 第二步(生成新图提示词):
  147. - 仅替换【模板提示词】中「产品主体」相关内容为第一步的产品描述;
  148. - 严格保留模板中「设计风格、光影、背景、比例、排版」等非产品相关信息;
  149. - 完全排除模板中的标题类信息,仅保留提示词核心内容;
  150. - 替换后必须完全保留原图产品的核心特征,禁止修改模板非产品核心信息;
  151. 4. 输出格式:不允许添加任何解释、引导、说明、示例、备注,仅返回「产品描述 + 替换后提示词」,直接输出纯文本;
  152. 【模板提示词】{$prompt}";
  153. $result = $aiGateway->callGptApi($model, $formattedPrompt, $mimeType, $base64Data);
  154. // Gemini模型响应格式处理
  155. $imgToTxtContent = $result['candidates'][0]['content']['parts'][0]['text'];
  156. // // 图生文成功后,调用文生文功能(gemini-2.0-flash)
  157. $txtToTxtPrompt = "转换成中文格式,去掉其他特殊符号,不允许添加任何解释、引导、说明、示例等文字:\n\n{$imgToTxtContent}";
  158. $txtToTxtResult = $aiGateway->txtGptApi($txtToTxtPrompt, 'gemini-2.0-flash');
  159. $finalContent = $txtToTxtResult['candidates'][0]['content']['parts'][0]['text'];
  160. // 处理调试输出内容
  161. $debugContent = json_encode($finalContent, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT);
  162. // 去掉特殊符号(只保留字母、数字、中文、常用标点和换行)
  163. $debugContent = preg_replace('/[^\p{Han}\w\s\n\r\.,,。!!??\-\::;;]/u', '', $debugContent);
  164. // 去掉第一个冒号前的文字
  165. $debugContent = preg_replace('/^[^:]+:/', '', $debugContent);
  166. Db::name('product')->where('id', $params['id'])->update(['content' => $debugContent]);
  167. return '成功';
  168. }
  169. /**
  170. * 执行图生文逻辑(图像转文本)
  171. */
  172. public function imageToText($sourceDirRaw, $fileName, $prompt,$sys_id,$imgtotxt_selectedOption,$call_data)
  173. {
  174. // 自动拆分文件名
  175. if (!$fileName && preg_match('/([^\/]+\.(jpg|jpeg|png))$/i', $sourceDirRaw, $matches)) {
  176. $fileName = $matches[1];
  177. $sourceDirRaw = preg_replace('/\/' . preg_quote($fileName, '/') . '$/', '', $sourceDirRaw);
  178. }
  179. if ($sourceDirRaw === '' || $fileName === '') {
  180. return '参数错误:原图路径 或 图片名称 不能为空';
  181. }
  182. // 构建路径
  183. $rootPath = str_replace('\\', '/', ROOT_PATH);
  184. $sourceDir = rtrim($rootPath . 'public/' . $sourceDirRaw, '/') . '/';
  185. $filePath = $sourceDir . $fileName;
  186. $relativePath = $sourceDirRaw . '/' . $fileName;
  187. // 文件夹是否存在(绝对路径检查)
  188. if (!is_dir($sourceDir)) {
  189. return '源目录不存在:' . $sourceDir;
  190. }
  191. // 文件是否存在
  192. if (!is_file($filePath)) {
  193. return '文件不存在:' . $filePath;
  194. }
  195. // 获取图片信息
  196. $ext = strtolower(pathinfo($filePath, PATHINFO_EXTENSION));
  197. $mime = ($ext === 'jpg' || $ext === 'jpeg') ? 'jpeg' : $ext;
  198. list($width, $height) = getimagesize($filePath);
  199. $imageData = base64_encode(file_get_contents($filePath));
  200. if (!$imageData || strlen($imageData) < 1000) {
  201. throw new \Exception('图片内容读取失败');
  202. }
  203. $imageUrl = "data:image/{$mime};base64,{$imageData}";
  204. // 记录提示词日志
  205. $logDir = $rootPath . 'runtime/logs/';
  206. if (!is_dir($logDir)) mkdir($logDir, 0755, true);
  207. // 调用 图生文 接口
  208. $ai = new AIGatewayService();
  209. $gptRes = $ai->callGptApi($imgtotxt_selectedOption,$prompt,$imageUrl,'');
  210. $gptText = trim($gptRes['choices'][0]['message']['content'] ?? '');
  211. echo "<pre>";
  212. print_r($gptText);
  213. echo "<pre>";die;
  214. //图生文返回内容日志runtime/logs/img_to_txt.txt
  215. // file_put_contents(
  216. // $logDir . 'img_to_txt.txt',
  217. // "\n======== " . date('Y-m-d H:i:s') . " ========\n" .
  218. // $gptText. "\n\n",
  219. // FILE_APPEND
  220. // );die;
  221. // 验证 GPT 返回格式
  222. if (strpos($gptText, '---json json---') === false) {
  223. return 'GPT 返回格式不正确,缺少分隔符';
  224. }
  225. // 以 ---json json--- 分割
  226. $parts = array_map('trim', explode('---json json---', $gptText));
  227. // 清理“第一段”、“第二段”等标签前缀
  228. $cleanPrefix = function ($text) {
  229. return preg_replace('/^第[一二三四五六七八九十]+段[::]?\s*/u', '', $text);
  230. };
  231. // 防止越界,逐个安全提取
  232. $chineseDesc = isset($parts[0]) ? $cleanPrefix(trim($parts[0])) : '';
  233. $englishDesc = isset($parts[1]) ? $cleanPrefix(trim($parts[1])) : '';
  234. $part2 = isset($parts[2]) ? $cleanPrefix(trim($parts[2])) : '';
  235. // 只保留中英文、数字、下划线、短横线、空格
  236. $img_name = preg_replace('/[^\x{4e00}-\x{9fa5}A-Za-z0-9_\- ]/u', '', $part2);
  237. //图生文返回日志拆分后内容runtime/logs/img_to_txt.txt
  238. // file_put_contents(
  239. // $logDir . 'img_to_txt.txt',
  240. // "\n======== " . date('Y-m-d H:i:s') . " ========\n" .
  241. // $englishDesc . "\n--- 中文 ---\n" .
  242. // $chineseDesc . "\n---英文 ---\n" .
  243. // $img_name . "\n--- 命名 ---\n" ,
  244. // FILE_APPEND
  245. // );
  246. $now = date('Y-m-d H:i:s');
  247. $record = [
  248. 'chinese_description' => $chineseDesc,
  249. 'english_description' => $englishDesc,
  250. 'img_name' => $img_name,
  251. 'sys_id' => $sys_id,
  252. 'status' => 0,
  253. 'status_name' => "图生文",
  254. 'model' => "",
  255. 'size' => "",
  256. 'error_msg' => '',
  257. 'update_time' => $now
  258. ];
  259. // 查询是否存在相同的原图记录
  260. $exists = Db::name('text_to_image')->where('old_image_url', $relativePath)->find();
  261. if ($exists) {
  262. // 如果已存在,执行更新
  263. Db::name('text_to_image')->where('id', $exists['id'])->update($record);
  264. } else {
  265. // 不存在则添加必要字段并插入
  266. $record['old_image_url'] = $relativePath;
  267. $record['new_image_url'] = '';
  268. $record['custom_image_url'] = '';
  269. $record['created_time'] = $now;
  270. Db::name('text_to_image')->insert($record);
  271. }
  272. return ;
  273. }
  274. }