|
@@ -4,7 +4,6 @@ import cn.hutool.core.collection.CollUtil;
|
|
|
import cn.hutool.core.text.StrPool;
|
|
|
import cn.hutool.core.util.StrUtil;
|
|
|
import cn.iocoder.yudao.framework.ai.core.model.suno.api.SunoApi;
|
|
|
-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;
|
|
@@ -16,7 +15,8 @@ import org.springframework.stereotype.Service;
|
|
|
|
|
|
import java.util.*;
|
|
|
|
|
|
-import static cn.iocoder.yudao.framework.security.core.util.SecurityFrameworkUtils.getLoginUserId;
|
|
|
+import static cn.iocoder.yudao.framework.common.util.collection.CollectionUtils.convertList;
|
|
|
+import static cn.iocoder.yudao.framework.common.util.collection.CollectionUtils.convertMap;
|
|
|
|
|
|
/**
|
|
|
* AI 音乐 Service 实现类
|
|
@@ -34,54 +34,53 @@ public class AiMusicServiceImpl implements AiMusicService {
|
|
|
private AiMusicMapper musicMapper;
|
|
|
|
|
|
@Override
|
|
|
- public List<Long> generateMusic(AiSunoGenerateReqVO reqVO) {
|
|
|
+ public List<Long> generateMusic(Long userId, AiSunoGenerateReqVO reqVO) {
|
|
|
+ // 1. 调用 Suno 生成音乐
|
|
|
List<SunoApi.MusicData> musicDataList;
|
|
|
if (Objects.equals(AiMusicGenerateModeEnum.LYRIC.getMode(), reqVO.getGenerateMode())) {
|
|
|
// 1.1 歌词模式
|
|
|
- SunoApi.MusicGenerateRequest sunoReq = new SunoApi.MusicGenerateRequest(
|
|
|
+ SunoApi.MusicGenerateRequest generateRequest = new SunoApi.MusicGenerateRequest(
|
|
|
reqVO.getPrompt(), reqVO.getModelVersion(), CollUtil.join(reqVO.getTags(), StrPool.COMMA), reqVO.getTitle());
|
|
|
- musicDataList = sunoApi.customGenerate(sunoReq);
|
|
|
+ musicDataList = sunoApi.customGenerate(generateRequest);
|
|
|
} else if (Objects.equals(AiMusicGenerateModeEnum.DESCRIPTION.getMode(), reqVO.getGenerateMode())) {
|
|
|
// 1.2 描述模式
|
|
|
- SunoApi.MusicGenerateRequest sunoReq = new SunoApi.MusicGenerateRequest(
|
|
|
+ SunoApi.MusicGenerateRequest generateRequest = new SunoApi.MusicGenerateRequest(
|
|
|
reqVO.getPrompt(), reqVO.getModelVersion(), reqVO.getMakeInstrumental());
|
|
|
- musicDataList = sunoApi.generate(sunoReq);
|
|
|
+ musicDataList = sunoApi.generate(generateRequest);
|
|
|
} else {
|
|
|
throw new IllegalArgumentException(StrUtil.format("未知生成模式({})", reqVO));
|
|
|
}
|
|
|
+
|
|
|
// 2. 插入数据库
|
|
|
if (CollUtil.isEmpty(musicDataList)) {
|
|
|
-
|
|
|
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);
|
|
|
+ List<AiMusicDO> musicList = buildMusicDOList(musicDataList);
|
|
|
+ musicList.forEach(music -> music.setUserId(userId).setPlatform(music.getPlatform()).setGenerateMode(reqVO.getGenerateMode()));
|
|
|
+ musicMapper.insertBatch(musicList);
|
|
|
+ return convertList(musicList, AiMusicDO::getId);
|
|
|
}
|
|
|
|
|
|
@Override
|
|
|
public Integer syncMusic() {
|
|
|
- List<AiMusicDO> streamingTask = musicMapper.selectListByStatus(AiMusicStatusEnum.STREAMING.getStatus());
|
|
|
+ List<AiMusicDO> streamingTask = musicMapper.selectListByStatus(AiMusicStatusEnum.IN_PROGRESS.getStatus());
|
|
|
if (CollUtil.isEmpty(streamingTask)) {
|
|
|
return 0;
|
|
|
}
|
|
|
log.info("[syncMusic][Suno 开始同步, 共 ({}) 个任务]", streamingTask.size());
|
|
|
+
|
|
|
// GET 请求,为避免参数过长,分批次处理
|
|
|
- CollUtil.split(streamingTask, 36).forEach(chunk -> {
|
|
|
- Map<String, Long> taskIdMap = CollectionUtils.convertMap(chunk, AiMusicDO::getTaskId, AiMusicDO::getId);
|
|
|
+ CollUtil.split(streamingTask, 36).forEach(chunkList -> {
|
|
|
+ Map<String, Long> taskIdMap = convertMap(chunkList, AiMusicDO::getTaskId, AiMusicDO::getId);
|
|
|
List<SunoApi.MusicData> musicTaskList = sunoApi.getMusicList(new ArrayList<>(taskIdMap.keySet()));
|
|
|
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);
|
|
|
+ // 更新进度
|
|
|
+ List<AiMusicDO> updateMusicList = buildMusicDOList(musicTaskList);
|
|
|
+ updateMusicList.forEach(music -> music.setId(taskIdMap.get(music.getTaskId())));
|
|
|
+ musicMapper.updateBatch(updateMusicList);
|
|
|
});
|
|
|
return streamingTask.size();
|
|
|
}
|
|
@@ -89,16 +88,16 @@ public class AiMusicServiceImpl implements AiMusicService {
|
|
|
/**
|
|
|
* 构建 AiMusicDO 集合
|
|
|
*
|
|
|
- * @param musicTaskList suno 音乐任务列表
|
|
|
+ * @param musicList suno 音乐任务列表
|
|
|
* @return AiMusicDO 集合
|
|
|
*/
|
|
|
- private static List<AiMusicDO> buildMusicDOList(List<SunoApi.MusicData> musicTaskList) {
|
|
|
- return CollectionUtils.convertList(musicTaskList, musicData -> new AiMusicDO()
|
|
|
- .setTaskId(musicData.id())
|
|
|
+ private static List<AiMusicDO> buildMusicDOList(List<SunoApi.MusicData> musicList) {
|
|
|
+ return convertList(musicList, musicData -> new AiMusicDO()
|
|
|
+ .setTaskId(musicData.id()).setModel(musicData.modelName())
|
|
|
.setPrompt(musicData.prompt()).setGptDescriptionPrompt(musicData.gptDescriptionPrompt())
|
|
|
.setAudioUrl(musicData.audioUrl()).setVideoUrl(musicData.videoUrl()).setImageUrl(musicData.imageUrl())
|
|
|
.setTitle(musicData.title()).setLyric(musicData.lyric()).setTags(StrUtil.split(musicData.tags(), StrPool.COMMA))
|
|
|
- .setModel(musicData.modelName()).setStatus(Objects.equals("complete", musicData.status()) ? AiMusicStatusEnum.COMPLETE.getStatus() : AiMusicStatusEnum.STREAMING.getStatus()));
|
|
|
+ .setStatus(Objects.equals("complete", musicData.status()) ? AiMusicStatusEnum.SUCCESS.getStatus() : AiMusicStatusEnum.IN_PROGRESS.getStatus()));
|
|
|
|
|
|
}
|
|
|
}
|