From 5ba38dbdaae93cd321bc71b1255796a48c0ef8a5 Mon Sep 17 00:00:00 2001 From: xiaoxin <718949661@qq.com> Date: Mon, 24 Jun 2024 14:19:09 +0800 Subject: [PATCH] =?UTF-8?q?=E3=80=90=E8=A7=A3=E5=86=B3todo=E3=80=91AI?= =?UTF-8?q?=EF=BC=9A=E9=9F=B3=E4=B9=90=E6=8E=A5=E5=85=A5?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../ai/enums/music/AiMusicGenerateEnum.java | 5 +- .../ai/enums/music/AiMusicStatusEnum.java | 15 +-- .../admin/music/AiMusicController.java | 13 +-- .../admin/music/vo/AiSunoGenerateReqVO.java | 39 +++++++ .../controller/admin/music/vo/SunoReqVO.java | 40 ------- .../ai/dal/dataobject/music/AiMusicDO.java | 26 +---- .../iocoder/yudao/module/ai/job/SunoJob.java | 43 +------ .../ai/service/music/AiMusicConvert.java | 42 ------- .../ai/service/music/AiMusicService.java | 20 +++- .../ai/service/music/AiMusicServiceImpl.java | 105 +++++++++++------- 10 files changed, 144 insertions(+), 204 deletions(-) create mode 100644 yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/controller/admin/music/vo/AiSunoGenerateReqVO.java delete mode 100644 yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/controller/admin/music/vo/SunoReqVO.java delete mode 100644 yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/service/music/AiMusicConvert.java diff --git a/yudao-module-ai/yudao-module-ai-api/src/main/java/cn/iocoder/yudao/module/ai/enums/music/AiMusicGenerateEnum.java b/yudao-module-ai/yudao-module-ai-api/src/main/java/cn/iocoder/yudao/module/ai/enums/music/AiMusicGenerateEnum.java index 6f1fc362d..e6cb83b9a 100644 --- a/yudao-module-ai/yudao-module-ai-api/src/main/java/cn/iocoder/yudao/module/ai/enums/music/AiMusicGenerateEnum.java +++ b/yudao-module-ai/yudao-module-ai-api/src/main/java/cn/iocoder/yudao/module/ai/enums/music/AiMusicGenerateEnum.java @@ -12,9 +12,8 @@ import lombok.Getter; @Getter public enum AiMusicGenerateEnum { - // TODO @xin:用数字哈。项目目前枚举都是数字 - LYRIC("lyric", "歌词模式"), - DESCRIPTION("description", "描述模式"); + LYRIC("1", "歌词模式"), + DESCRIPTION("2", "描述模式"); /** * 模式 diff --git a/yudao-module-ai/yudao-module-ai-api/src/main/java/cn/iocoder/yudao/module/ai/enums/music/AiMusicStatusEnum.java b/yudao-module-ai/yudao-module-ai-api/src/main/java/cn/iocoder/yudao/module/ai/enums/music/AiMusicStatusEnum.java index f56bb866b..0922e8ca3 100644 --- a/yudao-module-ai/yudao-module-ai-api/src/main/java/cn/iocoder/yudao/module/ai/enums/music/AiMusicStatusEnum.java +++ b/yudao-module-ai/yudao-module-ai-api/src/main/java/cn/iocoder/yudao/module/ai/enums/music/AiMusicStatusEnum.java @@ -12,27 +12,18 @@ import lombok.Getter; @Getter public enum AiMusicStatusEnum { - // TODO @xin:用数字哈。项目目前枚举都是数字 // @xin 文档中无失败这个返回值 - STREAMING("streaming", "进行中"), - COMPLETE("complete", "完成"); + STREAMING("10", "进行中"), + COMPLETE("20", "完成"); /** * 状态 */ private final String status; + /** * 状态名 */ private final String name; - public static AiMusicStatusEnum valueOfStatus(String status) { - for (AiMusicStatusEnum statusEnum : AiMusicStatusEnum.values()) { - if (statusEnum.getStatus().equals(status)) { - return statusEnum; - } - } - throw new IllegalArgumentException("未知会话状态: " + status); - } - } diff --git a/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/controller/admin/music/AiMusicController.java b/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/controller/admin/music/AiMusicController.java index 188145a62..b9eeed2f2 100644 --- a/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/controller/admin/music/AiMusicController.java +++ b/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/controller/admin/music/AiMusicController.java @@ -1,12 +1,12 @@ package cn.iocoder.yudao.module.ai.controller.admin.music; import cn.iocoder.yudao.framework.common.pojo.CommonResult; -import cn.iocoder.yudao.module.ai.controller.admin.music.vo.SunoReqVO; +import cn.iocoder.yudao.module.ai.controller.admin.music.vo.AiSunoGenerateReqVO; import cn.iocoder.yudao.module.ai.service.music.AiMusicService; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; +import jakarta.annotation.Resource; import jakarta.validation.Valid; -import lombok.RequiredArgsConstructor; import org.springframework.web.bind.annotation.PostMapping; import org.springframework.web.bind.annotation.RequestBody; import org.springframework.web.bind.annotation.RequestMapping; @@ -19,16 +19,15 @@ import static cn.iocoder.yudao.framework.common.pojo.CommonResult.success; @Tag(name = "管理后台 - AI 音乐") @RestController @RequestMapping("/ai/music") -@RequiredArgsConstructor // TODO @xin:通过 @Resource 注入哈 public class AiMusicController { - // TODO @xin:变量不用有 ai 前缀 - private final AiMusicService aiMusicService; + @Resource + private AiMusicService musicService; @PostMapping("/generate") @Operation(summary = "音乐生成") - public CommonResult> generateMusic(@RequestBody @Valid SunoReqVO sunoReqVO) { - return success(aiMusicService.generateMusic(sunoReqVO)); + public CommonResult> generateMusic(@RequestBody @Valid AiSunoGenerateReqVO reqVO) { + return success(musicService.generateMusic(reqVO)); } } diff --git a/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/controller/admin/music/vo/AiSunoGenerateReqVO.java b/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/controller/admin/music/vo/AiSunoGenerateReqVO.java new file mode 100644 index 000000000..fbacfc350 --- /dev/null +++ b/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/controller/admin/music/vo/AiSunoGenerateReqVO.java @@ -0,0 +1,39 @@ +package cn.iocoder.yudao.module.ai.controller.admin.music.vo; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Data; + +import java.util.List; + +/** + * @author xiaoxin + */ +@Schema(description = "管理后台 - 音乐生成 Request VO") +@Data +public class AiSunoGenerateReqVO { + + @Schema(description = "用于生成音乐音频的提示", example = "创作一首带有轻松吉他旋律的流行歌曲,[verse] 描述夏日海滩的宁静,[chorus] 节奏加快,表达对自由的向往。") + private String prompt; + + @Schema(description = "是否纯音乐", example = "true") + private Boolean makeInstrumental; + + @Schema(description = "模型版本, 默认 chirp-v3.5", example = "chirp-v3.5") + private String modelVersion;// 参见 AiModelEnum 枚举 + + @Schema(description = "音乐风格", example = "[\"pop\",\"jazz\",\"punk\"]") + private List tags; + + @Schema(description = "音乐/歌曲名称", example = "夜空中最亮的星") + private String title; + + @Schema(description = "平台", requiredMode = Schema.RequiredMode.REQUIRED, example = "Suno") + @NotBlank(message = "平台不能为空") + private String platform;// 参见 AiPlatformEnum 枚举 + + @Schema(description = "生成模式 1(歌词模式), 2(描述模式)", requiredMode = Schema.RequiredMode.REQUIRED, example = "2") + @NotBlank(message = "生成模式不能为空") + private String generateMode;// 参见 AiMusicGenerateEnum 枚举 + +} \ No newline at end of file diff --git a/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/controller/admin/music/vo/SunoReqVO.java b/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/controller/admin/music/vo/SunoReqVO.java deleted file mode 100644 index a0de21acb..000000000 --- a/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/controller/admin/music/vo/SunoReqVO.java +++ /dev/null @@ -1,40 +0,0 @@ -package cn.iocoder.yudao.module.ai.controller.admin.music.vo; - -import io.swagger.v3.oas.annotations.media.Schema; -import jakarta.validation.constraints.NotBlank; -import lombok.Data; - -import java.util.List; - -// TODO @xin:1)ai 前缀;2)AiSunoGenerateReqVO,要有生成哈。3)swaggger 缺少的属性,也最好加下,类似 example,类上的 swagger 等 -/** - * @author xiaoxin - */ -@Data -public class SunoReqVO { - - @Schema(description = "用于生成音乐音频的提示") - private String prompt; - - @Schema(description = "是否纯音乐") - private Boolean makeInstrumental; - - // TODO @xin:我们自己是不是用 modelVersion?还是什么梗精准,减少非必要的缩写 - @Schema(description = "模型版本") - private String mv; - - @Schema(description = "音乐风格") - private List tags; - - @Schema(description = "音乐/歌曲名称") - private String title; - - @Schema(description = "平台") - @NotBlank(message = "平台不能为空") - private String platform; - - @Schema(description = "生成模式 lyric(歌词模式), description(描述模式)") - @NotBlank(message = "生成模式不能为空") - private String generateMode; - -} \ No newline at end of file diff --git a/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/dal/dataobject/music/AiMusicDO.java b/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/dal/dataobject/music/AiMusicDO.java index 1e7ee17cc..98b3e2282 100644 --- a/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/dal/dataobject/music/AiMusicDO.java +++ b/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/dal/dataobject/music/AiMusicDO.java @@ -1,21 +1,21 @@ package cn.iocoder.yudao.module.ai.dal.dataobject.music; -import cn.iocoder.yudao.framework.common.util.json.JsonUtils; import cn.iocoder.yudao.framework.mybatis.core.dataobject.BaseDO; import cn.iocoder.yudao.module.ai.enums.music.AiMusicStatusEnum; import com.baomidou.mybatisplus.annotation.IdType; import com.baomidou.mybatisplus.annotation.TableField; import com.baomidou.mybatisplus.annotation.TableId; import com.baomidou.mybatisplus.annotation.TableName; -import com.baomidou.mybatisplus.extension.handlers.AbstractJsonTypeHandler; +import com.baomidou.mybatisplus.extension.handlers.JacksonTypeHandler; import lombok.Data; import java.util.List; -// TODO @xin:注释完善下 + /** - * @Author xiaoxin - * @Date 2024/6/5 + * AI 音乐 DO + * + * @author xiaoxin */ @TableName("ai_music") @Data @@ -98,25 +98,11 @@ public class AiMusicDO extends BaseDO { /** * 音乐风格标签 */ - @TableField(typeHandler = AiMusicTagsHandler.class) + @TableField(typeHandler = JacksonTypeHandler.class) private List tags; /** * 任务编号 */ private String taskId; - - // TODO @xin:用 @TableField(typeHandler = JacksonTypeHandler.class) 替代即可 - public static class AiMusicTagsHandler extends AbstractJsonTypeHandler { - - @Override - protected Object parse(String json) { - return JsonUtils.parseArray(json, String.class); - } - - @Override - protected String toJson(Object obj) { - return JsonUtils.toJsonString(obj); - } - } } diff --git a/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/job/SunoJob.java b/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/job/SunoJob.java index 6ae3a856b..711680612 100644 --- a/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/job/SunoJob.java +++ b/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/job/SunoJob.java @@ -1,61 +1,28 @@ package cn.iocoder.yudao.module.ai.job; -import cn.hutool.core.collection.CollUtil; -import cn.iocoder.yudao.framework.ai.core.model.suno.api.SunoApi; import cn.iocoder.yudao.framework.quartz.core.handler.JobHandler; -import cn.iocoder.yudao.module.ai.dal.dataobject.music.AiMusicDO; -import cn.iocoder.yudao.module.ai.service.music.AiMusicConvert; import cn.iocoder.yudao.module.ai.service.music.AiMusicService; import jakarta.annotation.Resource; import lombok.extern.slf4j.Slf4j; import org.springframework.stereotype.Component; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -// TODO @xin:不要直接叫这个名字哈,要有它的目的 /** - * 处理 Suno Job + * 同步 Suno 任务状态以及回写对应的音乐信息 Job + * * @author xiaoxin */ @Component @Slf4j public class SunoJob implements JobHandler { - @Resource - private SunoApi sunoApi; @Resource private AiMusicService musicService; @Override public String execute(String param) { - // TODO @xin:可以考虑,整个逻辑都下沉到 Service 里,有点类似 AccessLogCleanJob - List unCompletedTask = musicService.getUnCompletedTask(); - - if (CollUtil.isEmpty(unCompletedTask)) { - log.info("Suno 无进行中任务需要更新!"); - return "Suno 无进行中任务需要更新!"; - } - - - log.info("Suno 开始同步, 共 [{}] 个任务!", unCompletedTask.size()); - //GET 请求,为避免参数过长,分批次处理 - CollUtil.split(unCompletedTask, 4) - .forEach(chunk -> { - Map taskIdMap = CollUtil.toMap(chunk, new HashMap<>(), AiMusicDO::getTaskId, AiMusicDO::getId); - List musicTaskList = sunoApi.getMusicList(new ArrayList<>(taskIdMap.keySet())); - if (CollUtil.isNotEmpty(musicTaskList)) { - List aiMusicDOS = AiMusicConvert.convertFrom(musicTaskList); - //回填id - aiMusicDOS.forEach(aiMusicDO -> aiMusicDO.setId(taskIdMap.get(aiMusicDO.getTaskId()))); - musicService.updateBatch(aiMusicDOS); - } else { - log.warn("Suno 任务同步失败, 任务ID: [{}]", taskIdMap.keySet()); - } - }); - - return "Suno 同步 - ".concat(String.valueOf(unCompletedTask.size())).concat(" 个任务!"); + Integer count = musicService.syncMusicTask(); + log.info("[execute][Suno 同步任务数量 [{}] 个]", count); + return String.format("Suno 同步 - [%s] 个任务", count); } } diff --git a/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/service/music/AiMusicConvert.java b/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/service/music/AiMusicConvert.java deleted file mode 100644 index 06cbda1c6..000000000 --- a/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/service/music/AiMusicConvert.java +++ /dev/null @@ -1,42 +0,0 @@ -package cn.iocoder.yudao.module.ai.service.music; - -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.module.ai.dal.dataobject.music.AiMusicDO; - -import java.util.List; -import java.util.stream.Collectors; - -/** - * AI 音乐 Convert - * - * @author xiaoxin - */ -public class AiMusicConvert { - - public static AiMusicDO convertFrom(SunoApi.MusicData musicData) { - return new AiMusicDO() - .setTaskId(musicData.id()) - .setPrompt(musicData.prompt()) - .setGptDescriptionPrompt(musicData.gptDescriptionPrompt()) - .setAudioUrl(musicData.audioUrl()) - .setVideoUrl(musicData.videoUrl()) - .setImageUrl(musicData.imageUrl()) - .setLyric(musicData.lyric()) - .setTitle(musicData.title()) - .setStatus(musicData.status()) - .setModel(musicData.modelName()) - .setTags(StrUtil.isNotBlank(musicData.tags()) ? List.of(musicData.tags().split(StrPool.COMMA)) : null); - } - - // TODO @xin:一般情况下,不用 convert,直接逻辑里 convert 就好啦。 - public static List convertFrom(List list) { - // TODO @xin:可以使用 CollectionUtils.convertList 简洁一点 - return list.stream() - .map(AiMusicConvert::convertFrom) - .collect(Collectors.toList()); - } - - -} diff --git a/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/service/music/AiMusicService.java b/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/service/music/AiMusicService.java index f8739d750..a0e564aad 100644 --- a/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/service/music/AiMusicService.java +++ b/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/service/music/AiMusicService.java @@ -1,6 +1,6 @@ package cn.iocoder.yudao.module.ai.service.music; -import cn.iocoder.yudao.module.ai.controller.admin.music.vo.SunoReqVO; +import cn.iocoder.yudao.module.ai.controller.admin.music.vo.AiSunoGenerateReqVO; import cn.iocoder.yudao.module.ai.dal.dataobject.music.AiMusicDO; import java.util.List; @@ -18,7 +18,7 @@ public interface AiMusicService { * @param reqVO 请求参数 * @return 生成的音乐ID */ - List generateMusic(SunoReqVO reqVO); + List generateMusic(AiSunoGenerateReqVO reqVO); /** * 获取未完成状态的任务 @@ -27,6 +27,20 @@ public interface AiMusicService { */ List getUnCompletedTask(); - Boolean updateBatch(List aiMusicDOList); + + /** + * 同步音乐任务 + * + * @return 同步数量 + */ + Integer syncMusicTask(); + + /** + * 批量更新音乐信息 + * + * @param musicDOS 音乐信息 + * @return 是否成功 + */ + Boolean updateBatch(List musicDOS); } diff --git a/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/service/music/AiMusicServiceImpl.java b/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/service/music/AiMusicServiceImpl.java index 727e074ff..ab4a91d0f 100644 --- a/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/service/music/AiMusicServiceImpl.java +++ b/yudao-module-ai/yudao-module-ai-biz/src/main/java/cn/iocoder/yudao/module/ai/service/music/AiMusicServiceImpl.java @@ -2,8 +2,10 @@ package cn.iocoder.yudao.module.ai.service.music; 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.module.ai.controller.admin.music.vo.SunoReqVO; +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; @@ -13,8 +15,7 @@ import jakarta.annotation.Resource; import lombok.extern.slf4j.Slf4j; import org.springframework.stereotype.Service; -import java.util.Collections; -import java.util.List; +import java.util.*; import java.util.stream.Collectors; import static cn.iocoder.yudao.framework.security.core.util.SecurityFrameworkUtils.getLoginUserId; @@ -35,13 +36,22 @@ public class AiMusicServiceImpl implements AiMusicService { private AiMusicMapper musicMapper; @Override - public List generateMusic(SunoReqVO reqVO) { - // TODO @xin:是不是可以 if else 调用对应的 API,然后 insertMusicData 逻辑 - AiMusicGenerateEnum generateEnum = AiMusicGenerateEnum.valueOfMode(reqVO.getGenerateMode()); - return switch (generateEnum) { - case DESCRIPTION -> descriptionMode(reqVO); - case LYRIC -> lyricMode(reqVO); - }; + public List generateMusic(AiSunoGenerateReqVO reqVO) { + List musicDataList; + if (Objects.equals(AiMusicGenerateEnum.LYRIC.getMode(), reqVO.getGenerateMode())) { + //歌词模式 + 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())) { + //描述模式 + SunoApi.MusicGenerateRequest sunoReq = new SunoApi.MusicGenerateRequest(reqVO.getPrompt(), reqVO.getModelVersion(), reqVO.getMakeInstrumental()); + musicDataList = sunoApi.generate(sunoReq); + } else { + log.error("未知的生成模式:{}", reqVO.getGenerateMode()); + throw new IllegalArgumentException("未知的生成模式"); + } + // 2. 插入数据库 + return insertMusicData(musicDataList, reqVO.getGenerateMode(), reqVO.getPlatform()); } @Override @@ -50,36 +60,33 @@ public class AiMusicServiceImpl implements AiMusicService { } @Override - public Boolean updateBatch(List aiMusicDOList) { - return musicMapper.updateBatch(aiMusicDOList); + public Integer syncMusicTask() { + List unCompletedTask = this.getUnCompletedTask(); + if (CollUtil.isEmpty(unCompletedTask)) { + log.info("Suno 无进行中任务需要更新!"); + return 0; + } + log.info("Suno 开始同步, 共 [{}] 个任务!", unCompletedTask.size()); + //GET 请求,为避免参数过长,分批次处理 + CollUtil.split(unCompletedTask, 4) + .forEach(chunk -> { + Map taskIdMap = CollUtil.toMap(chunk, new HashMap<>(), AiMusicDO::getTaskId, AiMusicDO::getId); + List musicTaskList = sunoApi.getMusicList(new ArrayList<>(taskIdMap.keySet())); + if (CollUtil.isNotEmpty(musicTaskList)) { + List aiMusicDOS = buildMusicDOList(musicTaskList); + //回填id + aiMusicDOS.forEach(aiMusicDO -> aiMusicDO.setId(taskIdMap.get(aiMusicDO.getTaskId()))); + this.updateBatch(aiMusicDOS); + } else { + log.warn("Suno 任务同步失败, 任务ID: [{}]", taskIdMap.keySet()); + } + }); + return unCompletedTask.size(); } - /** - * 描述模式生成音乐 - * - * @param reqVO 请求参数 - * @return 生成的音乐ID集合 - */ - public List descriptionMode(SunoReqVO reqVO) { - // 1. 异步生成 - SunoApi.MusicGenerateRequest sunoReq = new SunoApi.MusicGenerateRequest(reqVO.getPrompt(), reqVO.getMv(), reqVO.getMakeInstrumental()); - List musicDataList = sunoApi.generate(sunoReq); - // 2. 插入数据库 - return insertMusicData(musicDataList, reqVO.getGenerateMode(), reqVO.getPlatform()); - } - - /** - * 歌词模式生成音乐 - * - * @param reqVO 请求参数 - * @return 生成的音乐ID集合 - */ - public List lyricMode(SunoReqVO reqVO) { - // 1. 异步生成 - SunoApi.MusicGenerateRequest sunoReq = new SunoApi.MusicGenerateRequest(reqVO.getPrompt(), reqVO.getMv(), CollUtil.join(reqVO.getTags(), StrPool.COMMA), reqVO.getTitle()); - List musicDataList = sunoApi.customGenerate(sunoReq); - // 2. 插入数据库 - return insertMusicData(musicDataList, reqVO.getGenerateMode(), reqVO.getPlatform()); + @Override + public Boolean updateBatch(List musicDOS) { + return musicMapper.updateBatch(musicDOS); } /** @@ -92,7 +99,7 @@ public class AiMusicServiceImpl implements AiMusicService { if (CollUtil.isEmpty(musicDataList)) { return Collections.emptyList(); } - List aiMusicDOList = AiMusicConvert.convertFrom(musicDataList).stream() + List aiMusicDOList = buildMusicDOList(musicDataList).stream() .map(musicDO -> musicDO.setUserId(getLoginUserId()) .setGenerateMode(generateMode) .setPlatform(platform)) @@ -103,4 +110,24 @@ public class AiMusicServiceImpl implements AiMusicService { .collect(Collectors.toList()); } + /** + * 构建 AiMusicDO 集合 + * + * @param musicTaskList suno 音乐任务列表 + * @return AiMusicDO 集合 + */ + private static List buildMusicDOList(List musicTaskList) { + return CollectionUtils.convertList(musicTaskList, musicData -> new AiMusicDO() + .setTaskId(musicData.id()) + .setPrompt(musicData.prompt()) + .setGptDescriptionPrompt(musicData.gptDescriptionPrompt()) + .setAudioUrl(musicData.audioUrl()) + .setVideoUrl(musicData.videoUrl()) + .setImageUrl(musicData.imageUrl()) + .setLyric(musicData.lyric()) + .setTitle(musicData.title()) + .setStatus(Objects.equals("complete", musicData.status()) ? AiMusicStatusEnum.COMPLETE.getStatus() : AiMusicStatusEnum.STREAMING.getStatus()) + .setModel(musicData.modelName()) + .setTags(StrUtil.isNotBlank(musicData.tags()) ? List.of(musicData.tags().split(StrPool.COMMA)) : null)); + } }