【解决todo】 处理 dall 异步调用,采用 @Async

This commit is contained in:
cherishsince 2024-05-30 10:14:34 +08:00
parent 92ee665996
commit 8b1e1c047b

View File

@ -1,56 +1,51 @@
package cn.iocoder.yudao.module.ai.service.image;
import cn.hutool.core.util.IdUtil;
import cn.hutool.core.util.StrUtil;
import cn.hutool.http.HttpUtil;
import cn.iocoder.yudao.framework.ai.core.enums.OpenAiImageModelEnum;
import cn.iocoder.yudao.framework.ai.core.enums.OpenAiImageStyleEnum;
import cn.iocoder.yudao.framework.ai.core.exception.AiException;
import cn.iocoder.yudao.framework.common.exception.util.ServiceExceptionUtil;
import cn.iocoder.yudao.framework.common.pojo.PageResult;
import cn.iocoder.yudao.framework.common.util.json.JsonUtils;
import cn.iocoder.yudao.framework.common.util.object.BeanUtils;
import cn.iocoder.yudao.framework.mybatis.core.query.LambdaQueryWrapperX;
import cn.iocoder.yudao.framework.security.core.util.SecurityFrameworkUtils;
import cn.iocoder.yudao.module.ai.AiCommonConstants;
import cn.iocoder.yudao.module.ai.ErrorCodeConstants;
import cn.iocoder.yudao.module.ai.controller.admin.image.vo.*;
import cn.iocoder.yudao.module.ai.convert.AiImageConvert;
import cn.iocoder.yudao.module.ai.dal.dataobject.image.AiImageDO;
import cn.iocoder.yudao.module.ai.dal.mysql.image.AiImageMapper;
import cn.iocoder.yudao.module.ai.enums.AiImagePublicStatusEnum;
import cn.iocoder.yudao.module.ai.enums.AiImageStatusEnum;
import cn.iocoder.yudao.module.infra.api.file.FileApi;
import com.google.common.collect.ImmutableMap;
import jakarta.annotation.PostConstruct;
import lombok.AllArgsConstructor;
import jakarta.annotation.Resource;
import lombok.extern.slf4j.Slf4j;
import org.springframework.ai.image.ImageGeneration;
import org.springframework.ai.image.ImagePrompt;
import org.springframework.ai.image.ImageResponse;
import org.springframework.ai.models.midjourney.api.MidjourneyInteractionsApi;
import org.springframework.ai.models.midjourney.api.req.ReRollReq;
import org.springframework.ai.models.midjourney.webSocket.MidjourneyWebSocketStarter;
import org.springframework.ai.models.midjourney.webSocket.WssNotify;
import org.springframework.ai.openai.OpenAiImageClient;
import org.springframework.ai.openai.OpenAiImageOptions;
import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import java.util.Collections;
import java.util.List;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
import static cn.iocoder.yudao.framework.common.exception.util.ServiceExceptionUtil.exception;
// TODO @fan注释优化下哈
/**
* ai 作图
* AI 绘画(接入 dall2/dall3midjourney)
*
* @author fansili
* @time 2024/4/25 15:51
* @since 1.0
*/
@AllArgsConstructor
@Service
@Slf4j
public class AiImageServiceImpl implements AiImageService {
@ -58,102 +53,92 @@ public class AiImageServiceImpl implements AiImageService {
// TODO @fan使用 @Resource 注入
// TODO @fanimageMapper
private final AiImageMapper aiImageMapper;
private final FileApi fileApi;
private final OpenAiImageClient openAiImageClient;
private final MidjourneyWebSocketStarter midjourneyWebSocketStarter;
private final MidjourneyInteractionsApi midjourneyInteractionsApi;
private static ThreadPoolExecutor EXECUTOR = new ThreadPoolExecutor(
3, 5, 1, TimeUnit.HOURS, new LinkedBlockingQueue<>(32));
@Resource
private AiImageMapper imageMapper;
@Resource
private FileApi fileApi;
@Resource
private OpenAiImageClient openAiImageClient;
@Resource
private MidjourneyWebSocketStarter midjourneyWebSocketStarter;
@Resource
private MidjourneyInteractionsApi midjourneyInteractionsApi;
// TODO @fan mj proxy
@PostConstruct
public void startMidjourney() {
log.info("midjourney web socket starter...");
midjourneyWebSocketStarter.start(new WssNotify() {
@Override
public void notify(int code, String message) {
log.info("code: {}, message: {}", code, message);
if (message.contains("Authentication failed")) {
// TODO 芋艿这里看怎么处理token无效的时候会认证失败
// 认证失败
log.error("midjourney socket 认证失败检查token是否失效!");
}
}
});
// todo @fan 暂时注释掉
// log.info("midjourney web socket starter...");
// midjourneyWebSocketStarter.start(new WssNotify() {
// @Override
// public void notify(int code, String message) {
// log.info("code: {}, message: {}", code, message);
// if (message.contains("Authentication failed")) {
// // TODO 芋艿这里看怎么处理token无效的时候会认证失败
// // 认证失败
// log.error("midjourney socket 认证失败检查token是否失效!");
// }
// }
// });
}
// TODO @fan1分页然后 loginUser 通过参数传入这样 Service 无状态2另外返回 DOVO 的翻译交给 Controller3还有使用 BeanUtils 替代哈
@Override
public PageResult<AiImageListRespVO> list(AiImageListReqVO req) {
// 获取登录用户
Long loginUserId = SecurityFrameworkUtils.getLoginUserId();
public PageResult<AiImageDO> getImagePageMy(Long loginUserId, AiImageListReqVO req) {
// 查询当前用户下所有的绘画记录
PageResult<AiImageDO> pageResult = aiImageMapper.selectPage(req,
return imageMapper.selectPage(req,
new LambdaQueryWrapperX<AiImageDO>()
.eq(AiImageDO::getUserId, loginUserId)
.orderByDesc(AiImageDO::getId)
);
// 转换 PageResult<AiImageListRespVO> 返回
PageResult<AiImageListRespVO> result = new PageResult<>();
result.setTotal(pageResult.getTotal());
result.setList(AiImageConvert.INSTANCE.convertAiImageListRespVO(pageResult.getList()));
return result;
.orderByDesc(AiImageDO::getId));
}
// TODO @fan1返回 DOVO 的翻译交给 Controller2还有使用 BeanUtils 替代哈
@Override
public AiImageListRespVO getMy(Long id) {
AiImageDO aiImageDO = aiImageMapper.selectById(id);
return AiImageConvert.INSTANCE.convertAiImageListRespVO(aiImageDO);
public AiImageDO getMy(Long id) {
return imageMapper.selectById(id);
}
// TODO @fan1loginUserId 通过 controller 传入
@Override
public AiImageDallRespVO dall(AiImageDallReqVO req) {
Long loginUserId = SecurityFrameworkUtils.getLoginUserId();
public Long dall(Long loginUserId, AiImageDallReqVO req) {
// 保存数据库
// TODO @fan1使用 BeanUtils2使用链式调用哈
AiImageDO aiImageDO = AiImageConvert.INSTANCE.convertAiImageDO(req);
aiImageDO.setStatus(AiImageStatusEnum.IN_PROGRESS.getStatus());
aiImageDO.setUserId(loginUserId);
aiImageMapper.insert(aiImageDO);
AiImageDO aiImageDO = BeanUtils.toBean(req, AiImageDO.class)
.setUserId(loginUserId)
.setWidth(req.getWidth())
.setHeight(req.getHeight())
.setDrawRequest(ImmutableMap.of(AiCommonConstants.DRAW_REQ_KEY_STYLE, req.getStyle()))
.setPublicStatus(AiImagePublicStatusEnum.PRIVATE.getStatus())
.setStatus(AiImageStatusEnum.IN_PROGRESS.getStatus());
imageMapper.insert(aiImageDO);
// 异步执行
// TODO @fan使用 @Async 去调用哈
EXECUTOR.execute(() -> {
try {
// 获取 model
OpenAiImageModelEnum openAiImageModelEnum = OpenAiImageModelEnum.valueOfModel(req.getModel());
OpenAiImageStyleEnum openAiImageStyleEnum = OpenAiImageStyleEnum.valueOfStyle(req.getStyle());
// 转换openai 参数
// TODO @fan需要考虑不同平台参数不同
OpenAiImageOptions openAiImageOptions = new OpenAiImageOptions();
openAiImageOptions.setModel(openAiImageModelEnum.getModel());
openAiImageOptions.setStyle(openAiImageStyleEnum.getStyle());
openAiImageOptions.setSize(req.getSize());
ImageResponse imageResponse = openAiImageClient.call(new ImagePrompt(req.getPrompt(), openAiImageOptions));
// 发送
ImageGeneration imageGeneration = imageResponse.getResult();
// 图片保存到服务器
String filePath = fileApi.createFile(HttpUtil.downloadBytes(imageGeneration.getOutput().getUrl()));
// 更新数据库
aiImageMapper.updateById(new AiImageDO().setId(aiImageDO.getId()).setStatus(AiImageStatusEnum.COMPLETE.getStatus())
.setPicUrl(filePath).setOriginalPicUrl(imageGeneration.getOutput().getUrl()));
} catch (AiException aiException) {
// TODO @fan错误日志也打印下哈因为 aiException.getMessage() 比较精简
aiImageMapper.updateById(new AiImageDO().setId(aiImageDO.getId()).setStatus(AiImageStatusEnum.FAIL.getStatus())
.setErrorMessage(aiException.getMessage()));
}
});
// TODO @fan返回 id 就可以啦
doDall(aiImageDO, req);
// 转换 AiImageDallDrawingRespVO
return AiImageConvert.INSTANCE.convertAiImageDallDrawingRespVO(aiImageDO);
return aiImageDO.getId();
}
@Async
public void doDall(AiImageDO aiImageDO, AiImageDallReqVO req) {
try {
// 获取 model
OpenAiImageModelEnum openAiImageModelEnum = OpenAiImageModelEnum.valueOfModel(req.getModel());
OpenAiImageStyleEnum openAiImageStyleEnum = OpenAiImageStyleEnum.valueOfStyle(req.getStyle());
// 转换openai 参数
// TODO @fan需要考虑不同平台参数不同
OpenAiImageOptions openAiImageOptions = new OpenAiImageOptions();
openAiImageOptions.setModel(openAiImageModelEnum.getModel());
openAiImageOptions.setStyle(openAiImageStyleEnum.getStyle());
openAiImageOptions.setSize(String.format(AiCommonConstants.DALL_SIZE_TEMPLATE, req.getWidth(), req.getHeight()));
ImageResponse imageResponse = openAiImageClient.call(new ImagePrompt(req.getPrompt(), openAiImageOptions));
// 发送
ImageGeneration imageGeneration = imageResponse.getResult();
// 图片保存到服务器
String filePath = fileApi.createFile(HttpUtil.downloadBytes(imageGeneration.getOutput().getUrl()));
// 更新数据库
imageMapper.updateById(new AiImageDO().setId(aiImageDO.getId()).setStatus(AiImageStatusEnum.COMPLETE.getStatus())
.setPicUrl(filePath).setOriginalPicUrl(imageGeneration.getOutput().getUrl()));
} catch (AiException aiException) {
// TODO @fan错误日志也打印下哈因为 aiException.getMessage() 比较精简
imageMapper.updateById(new AiImageDO().setId(aiImageDO.getId()).setStatus(AiImageStatusEnum.FAIL.getStatus())
.setErrorMessage(aiException.getMessage()));
}
}
@Override
@ -198,18 +183,16 @@ public class AiImageServiceImpl implements AiImageService {
// );
}
// TODO @fan1需要校验存在2需要校验属于我
@Override
public void deleteMy(Long id, Long userId) {
// 校验记录是否存在
// TODO @fanaiImageDO 这种命名 image ok 更简洁
// TODO @fan下面这个可以返回图片不存在
AiImageDO aiImageDO = validateExists(id);
if (!aiImageDO.getUserId().equals(userId)) {
throw exception(ErrorCodeConstants.AI_IMAGE_NOT_CREATE_USER);
// 校验是否存在并获取 image
AiImageDO image = validateExists(id);
// 是否属于当前用户
if (!image.getUserId().equals(userId)) {
throw exception(ErrorCodeConstants.AI_IMAGE_NOT_EXISTS);
}
// 删除记录
aiImageMapper.deleteById(id);
imageMapper.deleteById(id);
}
private void validateMessageId(String mjMessageId, String messageId) {
@ -237,7 +220,7 @@ public class AiImageServiceImpl implements AiImageService {
}
private AiImageDO validateExists(Long id) {
AiImageDO aiImageDO = aiImageMapper.selectById(id);
AiImageDO aiImageDO = imageMapper.selectById(id);
if (aiImageDO == null) {
throw ServiceExceptionUtil.exception(ErrorCodeConstants.AI_MIDJOURNEY_IMAGINE_FAIL);
}