|
@@ -8,15 +8,13 @@ import cn.iocoder.yudao.framework.common.util.collection.CollectionUtils;
|
|
|
import cn.iocoder.yudao.module.ai.controller.admin.music.vo.AiSunoGenerateReqVO;
|
|
|
import cn.iocoder.yudao.module.ai.dal.dataobject.music.AiMusicDO;
|
|
|
import cn.iocoder.yudao.module.ai.dal.mysql.music.AiMusicMapper;
|
|
|
-import cn.iocoder.yudao.module.ai.enums.music.AiMusicGenerateEnum;
|
|
|
+import cn.iocoder.yudao.module.ai.enums.music.AiMusicGenerateModeEnum;
|
|
|
import cn.iocoder.yudao.module.ai.enums.music.AiMusicStatusEnum;
|
|
|
-import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
|
|
|
import jakarta.annotation.Resource;
|
|
|
import lombok.extern.slf4j.Slf4j;
|
|
|
import org.springframework.stereotype.Service;
|
|
|
|
|
|
import java.util.*;
|
|
|
-import java.util.stream.Collectors;
|
|
|
|
|
|
import static cn.iocoder.yudao.framework.security.core.util.SecurityFrameworkUtils.getLoginUserId;
|
|
|
|
|
@@ -38,87 +36,54 @@ public class AiMusicServiceImpl implements AiMusicService {
|
|
|
@Override
|
|
|
public List<Long> generateMusic(AiSunoGenerateReqVO reqVO) {
|
|
|
List<SunoApi.MusicData> musicDataList;
|
|
|
- if (Objects.equals(AiMusicGenerateEnum.LYRIC.getMode(), reqVO.getGenerateMode())) {
|
|
|
+ if (Objects.equals(AiMusicGenerateModeEnum.LYRIC.getMode(), reqVO.getGenerateMode())) {
|
|
|
// 1.1 歌词模式
|
|
|
SunoApi.MusicGenerateRequest sunoReq = new SunoApi.MusicGenerateRequest(
|
|
|
reqVO.getPrompt(), reqVO.getModelVersion(), CollUtil.join(reqVO.getTags(), StrPool.COMMA), reqVO.getTitle());
|
|
|
musicDataList = sunoApi.customGenerate(sunoReq);
|
|
|
- } else if (Objects.equals(AiMusicGenerateEnum.DESCRIPTION.getMode(), reqVO.getGenerateMode())) {
|
|
|
+ } else if (Objects.equals(AiMusicGenerateModeEnum.DESCRIPTION.getMode(), reqVO.getGenerateMode())) {
|
|
|
// 1.2 描述模式
|
|
|
SunoApi.MusicGenerateRequest sunoReq = new SunoApi.MusicGenerateRequest(
|
|
|
reqVO.getPrompt(), reqVO.getModelVersion(), reqVO.getMakeInstrumental());
|
|
|
musicDataList = sunoApi.generate(sunoReq);
|
|
|
} else {
|
|
|
- // TODO @xin:不用 log error,直接抛异常,吧 reqVO 呆进去,有全局处理的哈
|
|
|
- log.error("未知的生成模式:{}", reqVO.getGenerateMode());
|
|
|
- throw new IllegalArgumentException("未知的生成模式");
|
|
|
+ throw new IllegalArgumentException(StrUtil.format("未知生成模式({})", reqVO));
|
|
|
}
|
|
|
-
|
|
|
// 2. 插入数据库
|
|
|
- // TODO @xin:因为 insertMusicData 复用的比较少,所以不用愁单独的方法,直接写在这里就好啦
|
|
|
- return insertMusicData(musicDataList, reqVO.getGenerateMode(), reqVO.getPlatform());
|
|
|
- }
|
|
|
+ if (CollUtil.isEmpty(musicDataList)) {
|
|
|
|
|
|
- // TODO @xin:1)service 里面,不要直接查询 db;2)不要用 ne,用 STREAMING 哈
|
|
|
- @Override
|
|
|
- public List<AiMusicDO> getUnCompletedTask() {
|
|
|
- return musicMapper.selectList(new LambdaQueryWrapper<AiMusicDO>().ne(AiMusicDO::getStatus, AiMusicStatusEnum.COMPLETE.getStatus()));
|
|
|
+ return Collections.emptyList();
|
|
|
+ }
|
|
|
+ List<AiMusicDO> aiMusicDOList = CollectionUtils.convertList(buildMusicDOList(musicDataList), musicDO ->
|
|
|
+ musicDO.setUserId(getLoginUserId())
|
|
|
+ .setGenerateMode(reqVO.getGenerateMode())
|
|
|
+ .setPlatform(reqVO.getPlatform()
|
|
|
+ ));
|
|
|
+ musicMapper.insertBatch(aiMusicDOList);
|
|
|
+ return CollectionUtils.convertList(aiMusicDOList, AiMusicDO::getId);
|
|
|
}
|
|
|
|
|
|
@Override
|
|
|
public Integer syncMusic() {
|
|
|
- List<AiMusicDO> unCompletedTask = this.getUnCompletedTask();
|
|
|
- if (CollUtil.isEmpty(unCompletedTask)) {
|
|
|
- // TODO @xin:这里不用打,反正 Job 也打了
|
|
|
- log.info("Suno 无进行中任务需要更新!");
|
|
|
+ List<AiMusicDO> streamingTask = musicMapper.selectListByStatus(AiMusicStatusEnum.STREAMING.getStatus());
|
|
|
+ if (CollUtil.isEmpty(streamingTask)) {
|
|
|
return 0;
|
|
|
}
|
|
|
- log.info("[syncMusic][Suno 开始同步, 共 ({}) 个任务]", unCompletedTask.size());
|
|
|
+ log.info("[syncMusic][Suno 开始同步, 共 ({}) 个任务]", streamingTask.size());
|
|
|
// GET 请求,为避免参数过长,分批次处理
|
|
|
- // TODO @xin:建议批量更大一些。
|
|
|
- CollUtil.split(unCompletedTask, 4).forEach(chunk -> {
|
|
|
- // TODO @xin:可以使用 CollectionUtils 里的 map 转换
|
|
|
- Map<String, Long> taskIdMap = CollUtil.toMap(chunk, new HashMap<>(), AiMusicDO::getTaskId, AiMusicDO::getId);
|
|
|
+ CollUtil.split(streamingTask, 36).forEach(chunk -> {
|
|
|
+ Map<String, Long> taskIdMap = CollectionUtils.convertMap(chunk, AiMusicDO::getTaskId, AiMusicDO::getId);
|
|
|
List<SunoApi.MusicData> musicTaskList = sunoApi.getMusicList(new ArrayList<>(taskIdMap.keySet()));
|
|
|
- // TODO @xin:查询不到,直接 return;这样真正逻辑的 85 - 87 就不用多一层括号
|
|
|
- if (CollUtil.isNotEmpty(musicTaskList)) {
|
|
|
- List<AiMusicDO> aiMusicDOS = buildMusicDOList(musicTaskList);
|
|
|
- //回填id
|
|
|
- aiMusicDOS.forEach(aiMusicDO -> aiMusicDO.setId(taskIdMap.get(aiMusicDO.getTaskId())));
|
|
|
- this.updateBatch(aiMusicDOS);
|
|
|
- } else {
|
|
|
+ if (CollUtil.isEmpty(musicTaskList)) {
|
|
|
log.warn("Suno 任务同步失败, 任务ID: [{}]", taskIdMap.keySet());
|
|
|
+ return;
|
|
|
}
|
|
|
+ List<AiMusicDO> aiMusicDOS = buildMusicDOList(musicTaskList);
|
|
|
+ //回填id
|
|
|
+ aiMusicDOS.forEach(aiMusicDO -> aiMusicDO.setId(taskIdMap.get(aiMusicDO.getTaskId())));
|
|
|
+ musicMapper.updateBatch(aiMusicDOS);
|
|
|
});
|
|
|
- return unCompletedTask.size();
|
|
|
- }
|
|
|
-
|
|
|
- // TODO @xin:这个方法,看着不用啦
|
|
|
- @Override
|
|
|
- public Boolean updateBatch(List<AiMusicDO> musicDOS) {
|
|
|
- return musicMapper.updateBatch(musicDOS);
|
|
|
- }
|
|
|
-
|
|
|
- /**
|
|
|
- * 新增音乐数据并提交 suno任务
|
|
|
- *
|
|
|
- * @param musicDataList 音乐数据列表
|
|
|
- * @return 音乐id集合
|
|
|
- */
|
|
|
- private List<Long> insertMusicData(List<SunoApi.MusicData> musicDataList, String generateMode, String platform) {
|
|
|
- if (CollUtil.isEmpty(musicDataList)) {
|
|
|
- return Collections.emptyList();
|
|
|
- }
|
|
|
- List<AiMusicDO> aiMusicDOList = buildMusicDOList(musicDataList).stream()
|
|
|
- .map(musicDO -> musicDO.setUserId(getLoginUserId())
|
|
|
- .setGenerateMode(generateMode)
|
|
|
- .setPlatform(platform))
|
|
|
- .toList();
|
|
|
- musicMapper.insertBatch(aiMusicDOList);
|
|
|
- // TODO @xin:用 CollectionUtils 简化操作
|
|
|
- return aiMusicDOList.stream()
|
|
|
- .map(AiMusicDO::getId)
|
|
|
- .collect(Collectors.toList());
|
|
|
+ return streamingTask.size();
|
|
|
}
|
|
|
|
|
|
/**
|
|
@@ -140,8 +105,7 @@ public class AiMusicServiceImpl implements AiMusicService {
|
|
|
.setTitle(musicData.title())
|
|
|
.setStatus(Objects.equals("complete", musicData.status()) ? AiMusicStatusEnum.COMPLETE.getStatus() : AiMusicStatusEnum.STREAMING.getStatus())
|
|
|
.setModel(musicData.modelName())
|
|
|
- // TODO @xin:可以用 hutool 的 StrUtil 的 split 之类的
|
|
|
- .setTags(StrUtil.isNotBlank(musicData.tags()) ? List.of(musicData.tags().split(StrPool.COMMA)) : null));
|
|
|
+ .setTags(StrUtil.split(musicData.tags(), StrPool.COMMA)));
|
|
|
}
|
|
|
|
|
|
}
|