| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354 |
- <?php
- namespace app\api\controller;
- use app\common\controller\Api;
- use app\job\ImageJob;
- use app\service\AIGatewayService;
- use app\service\ImageService;
- use think\App;
- use think\Db;
- use think\Exception;
- use think\Log;
- use think\Queue;
- use think\queue\job\Redis;
- use think\Request;
- class WorkOrder extends Api{
- protected $noNeedLogin = ['*'];
- protected $noNeedRight = ['*'];
- public function index(){echo '访问成功';}
- /**
- * AI队列入口处理 出图接口
- * 此方法处理图像转换为文本的请求,将图像信息存入队列以供后续处理。
- */
- public function imageToText()
- {
- $params = $this->request->param();
- $service = new ImageService();
- $service->handleImage($params);
- $this->success('任务成功提交至队列');
- }
- /**
- * 将远程图片保存到本地指定路径
- * @param string $url 远程图片URL
- * @return string|null 本地保存的相对路径,如果保存失败则返回null
- */
- private function saveImageFromUrl($url) {
- if (!function_exists('curl_init')) {
- return null;
- }
- // 设置保存路径:/uploads/operate/ai/dall-e/default/年月日/
- $date = date('Ymd');
- $basePath = '/uploads/operate/ai/dall-e/default/' . $date . '/';
- $savePath = ROOT_PATH . 'public' . $basePath;
- // 创建目录结构
- if (!is_dir($savePath)) {
- if (!mkdir($savePath, 0777, true)) {
- return null;
- }
- }
- // 生成唯一的文件名
- $extension = pathinfo($url, PATHINFO_EXTENSION);
- if (empty($extension)) {
- $extension = 'png';
- }
- $filename = uniqid() . '.' . $extension;
- $fullPath = $savePath . $filename;
- // 使用curl下载图片
- $ch = curl_init();
- curl_setopt($ch, CURLOPT_URL, $url);
- curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
- curl_setopt($ch, CURLOPT_FOLLOWLOCATION, true);
- curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
- curl_setopt($ch, CURLOPT_TIMEOUT, 30);
- curl_setopt($ch, CURLOPT_USERAGENT, 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.212 Safari/537.36');
- $imageData = curl_exec($ch);
- curl_close($ch);
- // 检查下载是否成功
- if (empty($imageData)) {
- return null;
- }
- // 保存图片到本地
- if (file_put_contents($fullPath, $imageData)) {
- // 返回相对路径
- return $basePath . $filename;
- } else {
- return null;
- }
- }
- /**
- * 图片下载接口
- * 该方法用于从远程URL下载图片并返回给客户端
- */
- public function download_image() {
- $imageUrl = $this->request->post('image_url');
-
- if (!$imageUrl) {
- return json(['code' => 1, 'message' => '图片URL不能为空']);
- }
-
- // 使用curl获取图片数据
- if (!function_exists('curl_init')) {
- return json(['code' => 1, 'message' => '服务器不支持curl']);
- }
-
- $ch = curl_init();
- curl_setopt($ch, CURLOPT_URL, $imageUrl);
- curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
- curl_setopt($ch, CURLOPT_FOLLOWLOCATION, true);
- curl_setopt($ch, CURLOPT_TIMEOUT, 30);
- curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
- curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);
-
- $imageData = curl_exec($ch);
- $httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
- $contentType = curl_getinfo($ch, CURLINFO_CONTENT_TYPE);
- curl_close($ch);
-
- if ($httpCode != 200 || empty($imageData)) {
- return json(['code' => 1, 'message' => '图片下载失败']);
- }
- $extension = 'png';
- if (strpos($contentType, 'image/jpeg') !== false) {
- $extension = 'jpg';
- } elseif (strpos($contentType, 'image/png') !== false) {
- $extension = 'png';
- } elseif (strpos($contentType, 'image/gif') !== false) {
- $extension = 'gif';
- }
-
- // 设置响应头
- header('Content-Type: application/octet-stream');
- header('Content-Disposition: attachment; filename="generated-image-' . date('YmdHis') . '.' . $extension . '"');
- header('Content-Length: ' . strlen($imageData));
-
- // 输出图片数据
- echo $imageData;
- exit;
- }
-
- //扩写文本内容提示词
- public function GetTxtToTxt(){
- $params = $this->request->param();
- if($params['status_val'] == '文生图'){
- $service = new ImageService();
- $result = $service->handleTextToImg($params);
- // 处理文生图结果
- if ($result['success']) {
- $remoteUrl = $result['url'];
- // // 保存图片到本地指定路径
- // $localFilePath = $this->saveImageFromUrl($remoteUrl);
- //
- // // 保存图片信息到product表
- // $productData = [
- // 'chinese_description' => $params['prompt'],
- // 'txt_image_url' => $localFilePath,
- // 'create_time' => date('Y-m-d H:i:s')
- // ];
- // $productId = Db::name('product')->insertGetId($productData);
- $this->success('图片生成成功', [
- // 'local_url' => $localFilePath,
- // 'product_id' => $productId,
- 'url' => $remoteUrl
- ]);
- } else {
- $this->error('文生图请求失败', ['message' => $result['message']]);
- }
- }elseif($params['status_val'] == '文生文'){
- $fullPrompt = $params['prompt'];
- // 优化提示词指令:放在所有内容之后,明确要求生成连续的一段话
- $fullPrompt .= "
- 请根据上述内容生成一段完整的话术,要求:
- 1. 内容必须是连贯的一段话,不要使用列表、分隔线或其他结构化格式
- 2. 不要包含非文本元素的描述
- 3. 不要添加任何额外的引导语、解释或开场白
- 4. 语言流畅自然";
- $textToTextParams = [
- 'prompt' => $fullPrompt,
- 'model' => $params['model'] ?? 'gemini-2.0-flash' // 默认使用文生文gtp-4模型
- ];
- $service = new ImageService();
- $result = $service->handleTextToText($textToTextParams);
- if ($result['success']) {
- $this->success('文生文生成成功', [
- 'content' => $result['data']
- ]);
- } else {
- $this->error('文生文请求失败', $result['message']);
- }
- }else{
- $this->error('请求失败');
- }
- }
- //获取服务器URL IP地址:端口
- public function GetHttpUrl(){
- $data = Db::name('http_url')->find();
- $fullUrl = "http://" . $data['baseUrl'] . ":" . $data['port'];
- $res = [
- 'code' => 0,
- 'msg' => '成功',
- 'data' => [
- 'full_url' => $fullUrl,
- 'id' => $data['id'],
- 'baseUrl' => $data['baseUrl'],
- 'port' => $data['port']
- ]
- ];
- return json($res);
- }
- //获取视频列表
- public function Getvideolist(){
- if (!$this->request->isGet()) {
- $this->error('请求方式错误');
- }
- $params = $this->request->param();
- $search = input('search', '');
- $page = isset($params['page']) ? (int)$params['page'] : 1;
- $limit = isset($params['limit']) ? (int)$params['limit'] : 50;
- $where = [];
- if (!empty($search)) {
- $where['prompt'] = ['like', '%' . $search . '%'];
- }
- $list = Db::name('video')->where('mod_rq', null)
- ->where($where)
- ->order('id desc')
- ->limit(($page - 1) * $limit, $limit)
- ->select();
- $total = Db::name('video')->where('mod_rq', null)
- ->where($where)
- ->count();
- $res['code'] = 0;
- $res['msg'] = '成功';
- $res['count'] = $total;
- $res['data'] = $list;
- return json($res);
- }
- //文生视频
- public function video(){
- $params = $this->request->param();
- $apiUrl = 'https://chatapi.onechats.ai/v1/videos';
- $apiKey = 'sk-sWW1GFlnjbrDRb1DkMEzePIxgdvLK6cZt0Qg93yDMVP2z1yN';
- $postData = [
- 'prompt' => $params['prompt'],
- 'model' => $params['model'],
- 'seconds' => $params['seconds'],
- 'size' => $params['size'],
- ];
- // 初始化CURL
- $ch = curl_init();
- curl_setopt($ch, CURLOPT_URL, $apiUrl);
- curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
- curl_setopt($ch, CURLOPT_POST, true);
- curl_setopt($ch, CURLOPT_POSTFIELDS, $postData);
- curl_setopt($ch, CURLOPT_HTTPHEADER, [
- 'Authorization: Bearer ' . $apiKey
- ]);
- curl_setopt($ch, CURLOPT_TIMEOUT, 300);
- curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
- curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, 0);
- curl_setopt($ch, CURLOPT_HEADER, true); // 获取响应头
- curl_setopt($ch, CURLOPT_VERBOSE, true); // 启用详细输出以进行调试
- // 创建临时文件来捕获详细的cURL输出
- $verbose = fopen('php://temp', 'w+');
- curl_setopt($ch, CURLOPT_STDERR, $verbose);
- // 执行请求
- $response = curl_exec($ch);
- //HTTP状态码
- $httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
- // 获取详细的cURL调试信息
- rewind($verbose);
- //CURL调试信息
- $verboseLog = stream_get_contents($verbose);
- fclose($verbose);
- // 分离头部和主体
- $header_size = curl_getinfo($ch, CURLINFO_HEADER_SIZE);
- //响应头部
- $header = substr($response, 0, $header_size);
- //响应主体
- $body = substr($response, $header_size);
- // 检查CURL错误
- $curlError = curl_error($ch);
- curl_close($ch);
- $responseData = json_decode($body, true);
- // 检查API是否返回了错误信息
- if (isset($responseData['error'])) {
- $errorMessage = isset($responseData['error']['message']) ? $responseData['error']['message'] : 'API请求失败';
- return json([
- 'code' => 1,
- 'msg' => '视频生成请求失败',
- 'data' => [
- 'error_type' => isset($responseData['error']['type']) ? $responseData['error']['type'] : 'unknown',
- 'error_code' => isset($responseData['error']['code']) ? $responseData['error']['code'] : 'unknown',
- 'error_message' => $errorMessage
- ]
- ]);
- }
- // 检查是否有自定义错误格式
- if (isset($responseData['code']) && $responseData['code'] === 'fail_to_fetch_task' && isset($responseData['message'])) {
- return json([
- 'code' => 1,
- 'msg' => '视频生成请求失败',
- 'data' => [
- 'error_code' => $responseData['code'],
- 'error_message' => $responseData['message']
- ]
- ]);
- }
- // 检查是否存在id字段
- if (!isset($responseData['id'])) {
- return json([
- 'code' => 1,
- 'msg' => '无法获取视频ID',
- 'data' => [
- 'response_data' => $responseData,
- 'http_code' => $httpCode
- ]
- ]);
- }
- $videoData = [
- 'video_id' => $responseData['id'],
- 'prompt' => $postData['prompt'],
- 'model' => $postData['model'],
- 'seconds' => $postData['seconds'],
- 'size' => $postData['size'],
- 'sys_rq' => date("Y-m-d H:i:s")
- ];
- // 尝试插入数据
- try {
- $res = Db::name('video')->insert($videoData);
- return json([
- 'code' => 0,
- 'msg' => '视频正在生成中',
- 'data ' => [
- 'video_id' => $responseData['id'],
- 'insert_result' => $res
- ]
- ]);
- } catch (Exception $e) {
- return json([
- 'code' => 1,
- 'msg' => '数据库操作失败',
- 'data' => [
- 'error_message' => $e->getMessage()
- ]
- ]);
- }
- }
- /**
- * 获取视频内容
- * 下载已完成的视频内容
- */
- public function videoContent(){
- // 从请求参数获取video_id,如果没有则使用默认值
- $video_id = input('get.video_id');
- $apiKey = 'sk-sWW1GFlnjbrDRb1DkMEzePIxgdvLK6cZt0Qg93yDMVP2z1yN';
- // 1. 先检查视频状态
- $statusUrl = 'https://chatapi.onechats.ai/v1/videos/' . $video_id;
- $statusData = $this->fetchVideoStatus($statusUrl, $apiKey);
- // 检查视频状态
- if ($statusData['status'] !== 'completed') {
- return json([
- 'code' => 202,
- 'msg' => '视频尚未生成完成',
- 'data' => [
- 'video_id' => $video_id,
- 'status' => $statusData['status'],
- 'progress' => $statusData['progress'],
- 'created_at' => $statusData['created_at'],
- 'message' => '请稍后再试,视频仍在' . ($statusData['status'] === 'queued' ? '排队中' : '处理中')
- ]
- ]);
- }
- // 2. 视频生成完成,准备下载
- $apiUrl = 'https://chatapi.onechats.ai/v1/videos/' . $video_id . '/content';
- // 获取可选的variant参数
- $variant = input('get.variant', '');
- if (!empty($variant)) {
- $apiUrl .= '?variant=' . urlencode($variant);
- }
- // 创建保存目录
- $saveDir = ROOT_PATH . 'public' . DS . 'uploads' . DS . 'videos' . DS . date('Ymd');
- if (!is_dir($saveDir)) {
- mkdir($saveDir, 0755, true);
- }
- // 生成唯一文件名
- $filename = $video_id . '.mp4';
- $localPath = DS . 'uploads' . DS . 'videos' . DS . date('Ymd') . DS . $filename;
- $fullPath = $saveDir . DS . $filename;
- // 3. 下载视频
- $videoData = $this->downloadVideo($apiUrl, $apiKey);
- // 4. 保存视频文件
- if (file_put_contents($fullPath, $videoData) === false) {
- throw new Exception('视频保存失败');
- }
- // 确保路径使用正斜杠,并只保存相对路径部分
- $localPath = str_replace('\\', '/', $localPath);
- // 移除开头的斜杠,确保路径格式为uploads/videos/...
- $savePath = ltrim($localPath, '/');
- // 将正确格式的文件路径存入数据库
- Db::name('video')->where('video_id', $video_id)->update([
- 'web_url' => $savePath
- ]);
- // 返回成功响应
- return json([
- 'code' => 0,
- 'msg' => '视频下载成功',
- 'data' => [
- 'video_id' => $video_id,
- 'local_path' => $localPath,
- 'web_url' => $savePath,
- 'file_size' => filesize($fullPath)
- ]
- ]);
- }
- /**
- * 获取视频状态
- */
- private function fetchVideoStatus($url, $apiKey) {
- $ch = curl_init();
- curl_setopt($ch, CURLOPT_URL, $url);
- curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
- curl_setopt($ch, CURLOPT_HTTPGET, true);
- curl_setopt($ch, CURLOPT_HTTPHEADER, [
- 'Authorization: Bearer ' . $apiKey,
- 'Accept: application/json'
- ]);
- curl_setopt($ch, CURLOPT_TIMEOUT, 30);
- curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
- curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, 0);
- $response = curl_exec($ch);
- $httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
- $error = curl_error($ch);
- curl_close($ch);
- if ($error) {
- throw new Exception('获取视频状态失败: ' . $error);
- }
- if ($httpCode < 200 || $httpCode >= 300) {
- throw new Exception('获取视频状态失败,HTTP状态码: ' . $httpCode);
- }
- $data = json_decode($response, true);
- if (!is_array($data)) {
- throw new Exception('视频状态数据格式错误');
- }
- return $data;
- }
- /**
- * 下载视频文件
- */
- private function downloadVideo($url, $apiKey) {
- $ch = curl_init();
- curl_setopt($ch, CURLOPT_URL, $url);
- curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
- curl_setopt($ch, CURLOPT_HTTPGET, true);
- curl_setopt($ch, CURLOPT_HTTPHEADER, [
- 'Authorization: Bearer ' . $apiKey
- ]);
- curl_setopt($ch, CURLOPT_TIMEOUT, 300);
- curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
- curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, 0);
- $response = curl_exec($ch);
- $httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
- $error = curl_error($ch);
- curl_close($ch);
- if ($error) {
- throw new Exception('视频下载失败: ' . $error);
- }
- if ($httpCode < 200 || $httpCode >= 300) {
- throw new Exception('视频下载失败,HTTP状态码: ' . $httpCode);
- }
- return $response;
- }
- public function textToImage()
- {
- $outputDirRaw = 'uploads/operate/ai/Preview/undefined_BRdP4/3';
- $img_name = '42131321';
- $rootPath = str_replace('\\', '/', ROOT_PATH);
- $outputDir = rtrim($rootPath . 'public/' . $outputDirRaw, '/') . '/';
- $dateDir = date('Y-m-d') . '/';
- $fullBaseDir = $outputDir . $dateDir;
- // 确保目录存在
- if (!is_dir($fullBaseDir . '2048x2048/')) {
- mkdir($fullBaseDir . '2048x2048/', 0755, true);
- }
- // // API配置
- // $config = [
- // 'api_url' => 'https://chatapi.onechats.ai/mj/submit/imagine',
- // 'fetch_url' => 'https://chatapi.onechats.ai/mj/task/',
- // 'api_key' => 'sk-iURfrAgzAjhZ4PpPLwzmWIAhM7zKfrkwDvyxk4RVBQ4ouJNK',
- // 'default_prompt' => '恐怖的树状怪物,拥有尖牙利爪和血红眼睛,在蓝色背景下显得阴森可怕,风格为黑暗奇幻。'
- // ];
- //
- // // 1. 准备请求数据
- // $prompt = $config['default_prompt'];
- // $postData = [
- // 'botType' => 'MID_JOURNEY',
- // 'prompt' => $prompt,
- // 'base64Array' => [],
- // 'accountFilter' => [
- // 'channelId' => "",
- // 'instanceId' => "",
- // 'modes' => [],
- // 'remark' => "",
- // 'remix' => true,
- // 'remixAutoConsidered' => true
- // ],
- // 'notifyHook' => "",
- // 'state' => ""
- // ];
- // // 2. 提交生成请求
- // $generateResponse = $this->sendApiRequest($config['api_url'], $postData, $config['api_key']);
- // echo "<pre>";
- // print_r($generateResponse);
- // echo "<pre>";
- // $generateData = json_decode($generateResponse, true);
- // echo "<pre>";
- // print_r($generateData);
- // echo "<pre>";
- // if (empty($generateData['result'])) {
- // throw new Exception('生成失败: '.($generateData['message'] ?? '未知错误'));
- // }
- //
- // die;
- // $taskId = "1755229064353588";
- $taskId = "1755234114253556";
- // $taskId = "1755224966357751";
- // 3. 等待图片生成完成
- sleep(3);
- $imageUrl = $this->getImageSeed($taskId);
- echo "<pre>";
- print_r($imageUrl);
- echo "<pre>";die;
- $img_name = mb_substr(preg_replace('/[^\x{4e00}-\x{9fa5}A-Za-z0-9_\- ]/u', '', $img_name), 0, 30);
- $filename = $img_name . '.png';
- $path512 = $fullBaseDir . '2048x2048/' . $filename;
- // 下载并保存图片
- // file_get_contents($imageUrl['data']['imageUrl']);
- // 下载并保存图片到本地文件夹中
- $imageData = file_get_contents($imageUrl['data']['imageUrl']);
- $result = file_put_contents($path512, $imageData);
- if ($result === false) {
- die('保存图片失败,请检查目录权限');
- }
- // 数据库更新
- Db::name('text_to_image')->where('id', '10694')->update([
- 'new_image_url' => str_replace($rootPath . 'public/', '', $path512),
- 'img_name' => $img_name,
- 'model' => 'MID_JOURNEY',
- 'status' => trim($img_name) === '' ? 0 : 1,
- 'status_name' => "文生图",
- 'size' => "2048",
- 'quality' => 'standard',
- 'style' => 'vivid',
- 'error_msg' => '',
- 'update_time' => date('Y-m-d H:i:s')
- ]);
- return "成功";
- }
- public function getImageSeed($taskId)
- {
- // 配置参数
- $apiUrl = 'https://chatapi.onechats.ai/mj/task/' . $taskId . '/fetch';
- $apiKey = 'sk-iURfrAgzAjhZ4PpPLwzmWIAhM7zKfrkwDvyxk4RVBQ4ouJNK';
- try {
- // 初始化cURL
- $ch = curl_init();
- // 设置cURL选项
- curl_setopt_array($ch, [
- CURLOPT_URL => $apiUrl,
- CURLOPT_RETURNTRANSFER => true,
- CURLOPT_CUSTOMREQUEST => 'GET', // 明确指定GET方法
- CURLOPT_HTTPHEADER => [
- 'Authorization: Bearer ' . $apiKey,
- 'Accept: application/json',
- 'Content-Type: application/json'
- ],
- CURLOPT_SSL_VERIFYPEER => false,
- CURLOPT_SSL_VERIFYHOST => false,
- CURLOPT_TIMEOUT => 60,
- CURLOPT_FAILONERROR => true // 添加失败时返回错误
- ]);
- // 执行请求
- $response = curl_exec($ch);
- $httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
- // 错误处理
- if (curl_errno($ch)) {
- throw new Exception('cURL请求失败: ' . curl_error($ch));
- }
- // 关闭连接
- curl_close($ch);
- // 验证HTTP状态码
- if ($httpCode < 200 || $httpCode >= 300) {
- throw new Exception('API返回错误状态码: ' . $httpCode);
- }
- // 解析JSON响应
- $responseData = json_decode($response, true);
- if (json_last_error() !== JSON_ERROR_NONE) {
- throw new Exception('JSON解析失败: ' . json_last_error_msg());
- }
- // 返回结构化数据
- return [
- 'success' => true,
- 'http_code' => $httpCode,
- 'data' => $responseData
- ];
- } catch (Exception $e) {
- // 确保关闭cURL连接
- if (isset($ch) && is_resource($ch)) {
- curl_close($ch);
- }
- return [
- 'success' => false,
- 'error' => $e->getMessage(),
- 'http_code' => $httpCode ?? 0
- ];
- }
- }
- private function sendPostRequest($url, $data, $apiKey)
- {
- $ch = curl_init();
- curl_setopt($ch, CURLOPT_URL, $url);
- curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
- curl_setopt($ch, CURLOPT_POST, true);
- curl_setopt($ch, CURLOPT_HTTPHEADER, [
- 'Authorization: Bearer ' . $apiKey,
- 'Accept: application/json',
- 'Content-Type: application/json'
- ]);
- curl_setopt($ch, CURLOPT_POSTFIELDS, json_encode($data));
- curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
- curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);
- curl_setopt($ch, CURLOPT_TIMEOUT, 60); // 延长超时时间
- $response = curl_exec($ch);
- $httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
- $error = curl_error($ch);
- curl_close($ch);
- return [
- 'response' => $response,
- 'http_code' => $httpCode,
- 'error' => $error
- ];
- }
- /**
- * 文本生成图片并保存第一张结果
- * @param array $params 请求参数
- * @return array 返回结果
- */
- public function txttowimg()
- {
- // API配置
- $config = [
- 'api_url' => 'https://chatapi.onechats.ai/mj/submit/imagine',
- 'fetch_url' => 'https://chatapi.onechats.ai/mj/task/',
- 'api_key' => 'sk-iURfrAgzAjhZ4PpPLwzmWIAhM7zKfrkwDvyxk4RVBQ4ouJNK',
- 'default_prompt' => '一个猫',
- 'wait_time' => 3 // 等待生成完成的秒数
- ];
- try {
- // 1. 准备请求数据
- $prompt = $config['default_prompt'];
- $postData = [
- 'botType' => 'MID_JOURNEY',
- 'prompt' => $prompt,
- 'base64Array' => [],
- 'accountFilter' => [
- 'channelId' => "",
- 'instanceId' => "",
- 'modes' => [],
- 'remark' => "",
- 'remix' => true,
- 'remixAutoConsidered' => true
- ],
- 'notifyHook' => "",
- 'state' => ""
- ];
- // 2. 提交生成请求
- $generateResponse = $this->sendApiRequest($config['api_url'], $postData, $config['api_key']);
- $generateData = json_decode($generateResponse, true);
- if (empty($generateData['result'])) {
- throw new Exception('生成失败: '.($generateData['message'] ?? '未知错误'));
- }
- $taskId = $generateData['result'];
- // 3. 等待图片生成完成
- sleep($config['wait_time']);
- // 4. 获取生成结果
- $fetchUrl = $config['fetch_url'].$taskId.'/fetch';
- $fetchResponse = $this->sendApiRequest($fetchUrl, [], $config['api_key'], 'GET');
- $fetchData = json_decode($fetchResponse, true);
- if (empty($fetchData['imageUrl'])) {
- throw new Exception('获取图片失败: '.($fetchData['message'] ?? '未知错误'));
- }
- // 5. 处理返回的图片数组(取第一张)
- $imageUrls = is_array($fetchData['imageUrl']) ? $fetchData['imageUrl'] : [$fetchData['imageUrl']];
- $firstImageUrl = $imageUrls[0];
- // 6. 保存图片到本地
- $savePath = $this->saveImage($firstImageUrl);
- // 7. 返回结果
- return [
- 'code' => 200,
- 'msg' => '图片生成并保存成功',
- 'data' => [
- 'local_path' => $savePath,
- 'web_url' => request()->domain().$savePath,
- 'task_id' => $taskId
- ]
- ];
- } catch (Exception $e) {
- // 错误处理
- return [
- 'code' => 500,
- 'msg' => '处理失败: '.$e->getMessage(),
- 'data' => null
- ];
- }
- }
- /**
- * 发送API请求
- * @param string $url 请求地址
- * @param array $data 请求数据
- * @param string $apiKey API密钥
- * @param string $method 请求方法
- * @return string 响应内容
- * @throws Exception
- */
- private function sendApiRequest($url, $data, $apiKey, $method = 'POST')
- {
- $ch = curl_init();
- curl_setopt_array($ch, [
- CURLOPT_URL => $url,
- CURLOPT_RETURNTRANSFER => true,
- CURLOPT_CUSTOMREQUEST => $method,
- CURLOPT_HTTPHEADER => [
- 'Authorization: Bearer '.$apiKey,
- 'Accept: application/json',
- 'Content-Type: application/json'
- ],
- CURLOPT_POSTFIELDS => $method === 'POST' ? json_encode($data) : null,
- CURLOPT_SSL_VERIFYPEER => false,
- CURLOPT_SSL_VERIFYHOST => false,
- CURLOPT_TIMEOUT => 60,
- CURLOPT_FAILONERROR => true
- ]);
- $response = curl_exec($ch);
- $error = curl_error($ch);
- $httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
- curl_close($ch);
- if ($error) {
- throw new Exception('API请求失败: '.$error);
- }
- if ($httpCode < 200 || $httpCode >= 300) {
- throw new Exception('API返回错误状态码: '.$httpCode);
- }
- return $response;
- }
- /**
- * 保存图片到本地
- * @param string $imageUrl 图片URL
- * @return string 本地保存路径
- * @throws Exception
- */
- private function saveImage($imageUrl)
- {
- // 1. 创建存储目录
- $saveDir = ROOT_PATH.'public'.DS.'uploads'.DS.'midjourney'.DS.date('Ymd');
- if (!is_dir($saveDir)) {
- mkdir($saveDir, 0755, true);
- }
- // 2. 生成唯一文件名
- $filename = uniqid().'.png';
- $localPath = DS.'uploads'.DS.'midjourney'.DS.date('Ymd').DS.$filename;
- $fullPath = $saveDir.DS.$filename;
- // 3. 下载图片
- $ch = curl_init($imageUrl);
- curl_setopt_array($ch, [
- CURLOPT_RETURNTRANSFER => true,
- CURLOPT_FOLLOWLOCATION => true,
- CURLOPT_SSL_VERIFYPEER => false,
- CURLOPT_CONNECTTIMEOUT => 15
- ]);
- $imageData = curl_exec($ch);
- $error = curl_error($ch);
- curl_close($ch);
- if (!$imageData) {
- throw new Exception('图片下载失败: '.$error);
- }
- // 4. 验证图片类型
- $imageInfo = getimagesizefromstring($imageData);
- if (!in_array($imageInfo['mime'] ?? '', ['image/png', 'image/jpeg'])) {
- throw new Exception('下载内容不是有效图片');
- }
- // 5. 保存文件
- if (!file_put_contents($fullPath, $imageData)) {
- throw new Exception('图片保存失败');
- }
- return $localPath;
- }
- /**
- * 图生图本地测试
- */
- public function imgtowimg()
- {
- $prompt = $this->request->param('prompt', '');
- $imgRelPath = 'uploads/operate/ai/Preview/arr/0835006071623.png';
- $imgPath = ROOT_PATH . 'public/' . $imgRelPath;
- if (!file_exists($imgPath)) {
- return json(['code' => 1, 'msg' => '原图不存在:' . $imgRelPath]);
- }
- $imgData = file_get_contents($imgPath);
- $base64Img = 'data:image/png;base64,' . base64_encode($imgData);
- $params = [
- 'prompt' => $prompt,
- 'negative_prompt' => '(deformed, distorted, disfigured:1.3), poorly drawn, bad anatomy',
- 'steps' => 20,
- 'sampler_name' => 'DPM++ 2M SDE',
- 'cfg_scale' => 7,
- 'seed' => -1,
- 'width' => 1024,
- 'height' => 1303,
- 'override_settings' => [
- 'sd_model_checkpoint' => 'realisticVisionV51_v51VAE-inpainting',
- 'sd_vae' => 'vae-ft-mse-840000-ema-pruned',
- 'CLIP_stop_at_last_layers' => 2
- ],
- 'clip_skip' => 2,
- 'alwayson_scripts' => [
- 'controlnet' => [
- 'args' => [[
- 'enabled' => true,
- 'input_image' => $base64Img,
- 'module' => 'inpaint_only+lama',
- 'model' => 'control_v11p_sd15_inpaint_fp16 [be8bc0ed]',
- 'weight' => 1,
- 'resize_mode' => 'Resize and Fill',
- 'pixel_perfect' => false,
- 'control_mode' => 'ControlNet is more important',
- 'starting_control_step' => 0,
- 'ending_control_step' => 1
- ]]
- ]
- ]
- ];
- $apiUrl = "http://20.0.17.188:45001/sdapi/v1/txt2img";
- $headers = ['Content-Type: application/json'];
- $ch = curl_init();
- curl_setopt($ch, CURLOPT_URL, $apiUrl);
- curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
- curl_setopt($ch, CURLOPT_POST, true);
- curl_setopt($ch, CURLOPT_HTTPHEADER, $headers);
- curl_setopt($ch, CURLOPT_POSTFIELDS, json_encode($params, JSON_UNESCAPED_UNICODE));
- curl_setopt($ch, CURLOPT_TIMEOUT, 180);
- $response = curl_exec($ch);
- $error = curl_error($ch);
- curl_close($ch);
- if ($error) {
- return json(['code' => 1, 'msg' => '请求失败:' . $error]);
- }
- $data = json_decode($response, true);
- if (!isset($data['images'][0])) {
- return json(['code' => 1, 'msg' => '接口未返回图像数据']);
- }
- $resultImg = base64_decode($data['images'][0]);
- $saveDir = ROOT_PATH . 'public/uploads/img2img/';
- if (!is_dir($saveDir)) {
- mkdir($saveDir, 0755, true);
- }
- $originalBaseName = pathinfo($imgRelPath, PATHINFO_FILENAME);
- $fileName = $originalBaseName . '-' . time() . '-1024x1248.png';
- $savePath = $saveDir . $fileName;
- file_put_contents($savePath, $resultImg);
- return json([
- 'code' => 0,
- 'msg' => '图像生成成功',
- 'data' => [
- 'origin_url' => '/uploads/img2img/' . $fileName
- ]
- ]);
- }
- /**
- * 后期图像处理-单张图片高清放大处理
- */
- public function extra_image()
- {
- // 配置参数
- $config = [
- 'input_dir' => 'uploads/img2img/',
- 'output_dir' => 'uploads/extra_image/',
- 'api_url' => 'http://20.0.17.188:45001/sdapi/v1/extra-single-image',
- 'timeout' => 120, // 增加超时时间,高清处理可能耗时较长
- 'upscale_params' => [
- 'resize_mode' => 0,
- 'show_extras_results' => true,
- 'gfpgan_visibility' => 0, // 人脸修复关闭
- 'codeformer_visibility' => 0, // 人脸修复关闭
- 'codeformer_weight' => 0,
- 'upscaling_resize' => 2.45, // 放大倍数
- 'upscaling_crop' => true,
- 'upscaler_1' => 'R-ESRGAN 4x+ Anime6B', // 主放大模型
- 'upscaler_2' => 'None', // 不使用第二放大器
- 'extras_upscaler_2_visibility' => 0,
- 'upscale_first' => false,
- ]
- ];
- // 输入文件处理
- $imgRelPath = '0835006071623-1757406184-1024x1248.png';
- $imgPath = ROOT_PATH . 'public/' . $config['input_dir'] . $imgRelPath;
- if (!file_exists($imgPath)) {
- return json(['code' => 1, 'msg' => '原图不存在:' . $imgRelPath]);
- }
- // 读取并编码图片
- try {
- $imgData = file_get_contents($imgPath);
- if ($imgData === false) {
- throw new Exception('无法读取图片文件');
- }
- $base64Img = base64_encode($imgData);
- } catch (Exception $e) {
- return json(['code' => 1, 'msg' => '图片处理失败:' . $e->getMessage()]);
- }
- // 准备API请求数据
- $postData = array_merge($config['upscale_params'], ['image' => $base64Img]);
- $jsonData = json_encode($postData);
- if ($jsonData === false) {
- return json(['code' => 1, 'msg' => 'JSON编码失败']);
- }
- // 调用API进行高清放大
- $ch = curl_init();
- curl_setopt_array($ch, [
- CURLOPT_URL => $config['api_url'],
- CURLOPT_RETURNTRANSFER => true,
- CURLOPT_POST => true,
- CURLOPT_HTTPHEADER => ['Content-Type: application/json'],
- CURLOPT_POSTFIELDS => $jsonData,
- CURLOPT_TIMEOUT => $config['timeout'],
- CURLOPT_CONNECTTIMEOUT => 30,
- ]);
- $response = curl_exec($ch);
- $error = curl_error($ch);
- $httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
- curl_close($ch);
- if ($error) {
- return json(['code' => 1, 'msg' => 'API请求失败:' . $error]);
- }
- if ($httpCode !== 200) {
- return json(['code' => 1, 'msg' => 'API返回错误状态码:' . $httpCode]);
- }
- $data = json_decode($response, true);
- if (json_last_error() !== JSON_ERROR_NONE) {
- return json(['code' => 1, 'msg' => 'API返回数据解析失败']);
- }
- if (!isset($data['image']) || empty($data['image'])) {
- return json(['code' => 1, 'msg' => '接口未返回有效的图像数据']);
- }
- // 保存处理后的图片
- try {
- $resultImg = base64_decode($data['image']);
- if ($resultImg === false) {
- throw new Exception('Base64解码失败');
- }
- $saveDir = ROOT_PATH . 'public/' . $config['output_dir'];
- if (!is_dir($saveDir) && !mkdir($saveDir, 0755, true)) {
- throw new Exception('无法创建输出目录');
- }
- $originalBaseName = pathinfo($imgRelPath, PATHINFO_FILENAME);
- $fileName = $originalBaseName . '-hd.png'; // 使用-hd后缀更明确
- $savePath = $saveDir . $fileName;
- if (file_put_contents($savePath, $resultImg) === false) {
- throw new Exception('无法保存处理后的图片');
- }
- // 返回成功响应
- return json([
- 'code' => 0,
- 'msg' => '图像高清放大处理成功',
- 'data' => [
- 'url' => '/' . $config['output_dir'] . $fileName,
- 'original_size' => filesize($imgPath),
- 'processed_size' => filesize($savePath),
- 'resolution' => getimagesize($savePath), // 返回新图片的分辨率
- ]
- ]);
- } catch (Exception $e) {
- return json(['code' => 1, 'msg' => '保存结果失败:' . $e->getMessage()]);
- }
- }
- /**
- * 查询队列列表
- * 统计文件对应的队列情况
- */
- public function get_queue_logs()
- {
- $params = $this->request->param('old_image_file', '');
- $queue_logs = Db::name('queue_logs')
- ->where('old_image_file', $params)
- ->order('id desc')
- ->select();
- $result = []; //初始化变量,避免未定义错误
- foreach ($queue_logs as &$log) {
- $taskId = $log['id'];
- $statusCount = Db::name('image_task_log')
- ->field('status, COUNT(*) as count')
- ->where('task_id', $taskId)
- ->where('mod_rq', null)
- ->group('status')
- ->select();
- $log['已完成数量'] = 0;
- $log['处理中数量'] = 0;
- $log['排队中的数量'] = 0;
- $log['失败数量'] = 0;
- foreach ($statusCount as $item) {
- switch ($item['status']) {
- case 0:
- $log['排队中的数量'] = $item['count'];
- break;
- case 1:
- $log['处理中数量'] = $item['count'];
- break;
- case 2:
- $log['已完成数量'] = $item['count'];
- break;
- case -1:
- $log['失败数量'] = $item['count'];
- break;
- }
- }
- // if ($log['排队中的数量'] >$log['已完成数量']) {
- // $result[] = $log;
- // }
- if ($log['排队中的数量']) {
- $result[] = $log;
- }
- // if ($log['处理中数量'] >= 0) {
- // $result[] = $log;
- // }
- }
- return json([
- 'code' => 0,
- 'msg' => '查询成功',
- 'data' => $result,
- 'count' => count($result)
- ]);
- }
- /**
- * 查询总队列状态(统计当前处理的数据量)
- */
- public function queueStats()
- {
- $statusList = Db::name('image_task_log')
- ->field('status, COUNT(*) as total')
- ->where('mod_rq', null)
- ->where('create_time', '>=', date('Y-m-d 00:00:00'))
- ->group('status')
- ->select();
- $statusCount = [];
- foreach ($statusList as $item) {
- $statusCount[$item['status']] = $item['total'];
- }
- // 总数为所有状态和
- $total = array_sum($statusCount);
- //获取队列当前状态
- $statusText = Db::name('queue_logs')->order('id desc')->value('status');
- return json([
- 'code' => 0,
- 'msg' => '获取成功',
- 'data' => [
- '总任务数' => $total,
- '待处理' => $statusCount[0] ?? 0,
- '处理中' => $statusCount[1] ?? 0,
- '成功' => $statusCount[2] ?? 0,
- '失败' => $statusCount[-1] ?? 0,
- '当前状态' => $statusText
- ]
- ]);
- }
- /**
- * 显示当前运行中的队列监听进程
- */
- public function viewQueueStatus()
- {
- $redis = new \Redis();
- $redis->connect('127.0.0.1', 6379);
- $redis->auth('123456');
- $redis->select(15);
- $key = 'queues:imgtotxt';
- // 判断 key 是否存在,避免报错
- if (!$redis->exists($key)) {
- return json([
- 'code' => 0,
- 'msg' => '查询成功,队列为空',
- 'count' => 0,
- 'tasks_preview' => []
- ]);
- }
- $count = $redis->lLen($key);
- $list = $redis->lRange($key, 0, 9);
- // 解码 JSON 内容,确保每一项都有效
- $parsed = array_filter(array_map(function ($item) {
- return json_decode($item, true);
- }, $list), function ($item) {
- return !is_null($item);
- });
- return json([
- 'code' => 0,
- 'msg' => '查询成功',
- 'count' => $count,
- 'tasks_preview' => $parsed
- ]);
- }
- /**
- * 清空队列并删除队列日志记录
- */
- public function stopQueueProcesses()
- {
- Db::name('image_task_log')
- ->where('log', '队列中')
- ->whereOr('status', 1)
- ->where('create_time', '>=', date('Y-m-d 00:00:00'))
- ->update([
- 'status' => "-1",
- 'log' => '清空取消队列',
- 'mod_rq' => date('Y-m-d H:i:s')
- ]);
- Db::name('image_task_log')
- ->whereLike('log', '%处理中%')
- ->where('create_time', '>=', date('Y-m-d 00:00:00'))
- ->update([
- 'status' => "-1",
- 'log' => '清空取消队列',
- 'mod_rq' => date('Y-m-d H:i:s')
- ]);
- $redis = new \Redis();
- $redis->connect('127.0.0.1', 6379);
- $redis->auth('123456');
- $redis->select(15);
- $key_txttoimg = 'queues:txttoimg:reserved';
- $key_txttotxt = 'queues:txttotxt:reserved';
- $key_imgtotxt = 'queues:imgtotxt:reserved';
- $key_imgtoimg = 'queues:imgtoimg:reserved';
- // 清空 Redis 队列
- $redis->del($key_txttoimg);
- $redis->del($key_txttotxt);
- $redis->del($key_imgtotxt);
- $redis->del($key_imgtoimg);
- $count = $redis->lLen($key_txttoimg) + $redis->lLen($key_txttotxt) + $redis->lLen($key_imgtotxt) + $redis->lLen($key_imgtoimg);
- // if ($count === 0) {
- // return json([
- // 'code' => 1,
- // 'msg' => '暂无队列需要停止'
- // ]);
- // }
- return json([
- 'code' => 0,
- 'msg' => '成功停止队列任务'
- ]);
- }
- /**
- * 开启队列任务
- * 暂时用不到、服务器已开启自动开启队列模式
- */
- // public function kaiStats()
- // {
- // // 判断是否已有监听进程在运行
- // $check = shell_exec("ps aux | grep 'queue:listen' | grep -v grep");
- // if ($check) {
- // return json([
- // 'code' => 1,
- // 'msg' => '监听进程已存在,请勿重复启动'
- // ]);
- // }
- // // 启动监听
- // $command = 'nohup php think queue:listen --queue --timeout=300 --sleep=3 --memory=256 > /var/log/job_queue.log 2>&1 &';
- // exec($command, $output, $status);
- // if ($status === 0) {
- // return json([
- // 'code' => 0,
- // 'msg' => '队列监听已启动'
- // ]);
- // } else {
- // return json([
- // 'code' => 1,
- // 'msg' => '队列启动失败',
- // 'output' => $output
- // ]);
- // }
- // }
- /**
- * 通过店铺ID-查询对应店铺表数据
- *
- */
- public function PatternApi()
- {
- $params = $this->request->param('pattern_id', '');
- $tableName = 'pattern-' . $params;
- // 连接 MongoDB
- $mongo = Db::connect('mongodb');
- // 查询指定 skc 的数据
- $data = $mongo->table($tableName)
- ->field('
- name,
- skc,
- file
- ')
- ->where("skc", '0853004152036')
- ->select();
- $data = json_decode(json_encode($data), true); // 数组
- return json([
- 'code' => 0,
- 'msg' => '获取成功',
- 'data' => $data
- ]);
- }
- }
|