mirror of
				https://gitee.com/hhyykk/ipms-sjy.git
				synced 2025-10-31 10:18:42 +08:00 
			
		
		
		
	【新增】AI:音乐接入 API KEY 管理
This commit is contained in:
		| @@ -5,9 +5,9 @@ Authorization: {{token}} | ||||
|  | ||||
| { | ||||
|   "platform": "Suno", | ||||
|   "generateMode": 1, | ||||
|   "prompt": "来一首快乐的歌曲", | ||||
|   "modelVersion": "chirp-v3.5", | ||||
|   "generateMode": 2, | ||||
|   "prompt": "周末啦!", | ||||
|   "model": "chirp-v3.5", | ||||
|   "tags": ["Happy"], | ||||
|   "title": "Happy Song" | ||||
| } | ||||
| @@ -19,8 +19,8 @@ Authorization: {{token}} | ||||
|  | ||||
| { | ||||
|   "platform": "Suno", | ||||
|   "generateMode": 2, | ||||
|   "prompt": "来一首快乐的歌曲", | ||||
|   "makeInstrumental": false, | ||||
|   "title": "Happy Song" | ||||
|   "generateMode": 1, | ||||
|   "model": "chirp-v3.5", | ||||
|   "gptDescriptionPrompt": "今天是星球六,结果是个下雨天,希望心情很美丽", | ||||
|   "makeInstrumental": false | ||||
| } | ||||
| @@ -35,6 +35,16 @@ public class AiMusicController { | ||||
|         return success(BeanUtils.toBean(pageResult, AiMusicRespVO.class)); | ||||
|     } | ||||
|  | ||||
|     @PostMapping("/generate") | ||||
|     @Operation(summary = "音乐生成") | ||||
|     public CommonResult<List<Long>> generateMusic(@RequestBody @Valid AiSunoGenerateReqVO reqVO) { | ||||
|         if (true) { | ||||
|             musicService.syncMusic(); | ||||
|             return null; | ||||
|         } | ||||
|         return success(musicService.generateMusic(getLoginUserId(), reqVO)); | ||||
|     } | ||||
|  | ||||
|     @Operation(summary = "删除【我的】音乐记录") | ||||
|     @DeleteMapping("/delete-my") | ||||
|     @Parameter(name = "id", required = true, description = "音乐编号", example = "1024") | ||||
| @@ -54,6 +64,7 @@ public class AiMusicController { | ||||
|         return success(BeanUtils.toBean(music, AiMusicRespVO.class)); | ||||
|     } | ||||
|  | ||||
|     // TODO @xin:这个搞成 updateMy ,修改【我的】音乐。方便后续支持其它字段;另外,需要校验下,更新的音乐,是不是我的! | ||||
|     @PostMapping("/updateTitle-my") | ||||
|     @Operation(summary = "修改【我的】音乐 目前只支持修改标题") | ||||
|     @Parameter(name = "title", required = true, description = "音乐名称", example = "夜空中最亮的星") | ||||
| @@ -62,12 +73,6 @@ public class AiMusicController { | ||||
|         return success(true); | ||||
|     } | ||||
|  | ||||
|     @PostMapping("/generate") | ||||
|     @Operation(summary = "音乐生成") | ||||
|     public CommonResult<List<Long>> generateMusic(@RequestBody @Valid AiSunoGenerateReqVO reqVO) { | ||||
|         return success(musicService.generateMusic(getLoginUserId(), reqVO)); | ||||
|     } | ||||
|  | ||||
|     // ================ 音乐管理 ================ | ||||
|  | ||||
|     @GetMapping("/page") | ||||
| @@ -87,11 +92,11 @@ public class AiMusicController { | ||||
|         return success(true); | ||||
|     } | ||||
|  | ||||
|     @PutMapping("/update-public-status") | ||||
|     @Operation(summary = "更新音乐发布状态") | ||||
|     @PutMapping("/update") | ||||
|     @Operation(summary = "更新音乐") | ||||
|     @PreAuthorize("@ss.hasPermission('ai:music:update')") | ||||
|     public CommonResult<Boolean> updateMusicPublicStatus(@Valid @RequestBody AiMusicUpdatePublicStatusReqVO updateReqVO) { | ||||
|         musicService.updateMusicPublicStatus(updateReqVO); | ||||
|     public CommonResult<Boolean> updateMusic(@Valid @RequestBody AiMusicUpdateReqVO updateReqVO) { | ||||
|         musicService.updateMusic(updateReqVO); | ||||
|         return success(true); | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -52,6 +52,9 @@ public class AiMusicRespVO { | ||||
|     @Schema(description = "音乐风格标签") | ||||
|     private List<String> tags; | ||||
|  | ||||
|     @Schema(description = "音乐时长", example = "[\"pop\",\"jazz\",\"punk\"]") | ||||
|     private Double duration; | ||||
|  | ||||
|     @Schema(description = "是否发布", requiredMode = Schema.RequiredMode.REQUIRED, example = "true") | ||||
|     private Boolean publicStatus; | ||||
|  | ||||
| @@ -61,9 +64,6 @@ public class AiMusicRespVO { | ||||
|     @Schema(description = "错误信息") | ||||
|     private String errorMessage; | ||||
|  | ||||
|     @Schema(description = "音乐时长") | ||||
|     private Double duration; | ||||
|  | ||||
|     @Schema(description = "创建时间", requiredMode = Schema.RequiredMode.REQUIRED) | ||||
|     private LocalDateTime createTime; | ||||
|  | ||||
|   | ||||
| @@ -4,15 +4,15 @@ import io.swagger.v3.oas.annotations.media.Schema; | ||||
| import jakarta.validation.constraints.NotNull; | ||||
| import lombok.Data; | ||||
| 
 | ||||
| @Schema(description = "管理后台 - AI 音乐修改发布状态 Request VO") | ||||
| @Schema(description = "管理后台 - AI 音乐修改 Request VO") | ||||
| @Data | ||||
| public class AiMusicUpdatePublicStatusReqVO { | ||||
| public class AiMusicUpdateReqVO { | ||||
| 
 | ||||
|     @Schema(description = "编号", requiredMode = Schema.RequiredMode.REQUIRED, example = "15583") | ||||
|     @NotNull(message = "编号不能为空") | ||||
|     private Long id; | ||||
| 
 | ||||
|     @Schema(description = "是否发布", requiredMode = Schema.RequiredMode.REQUIRED, example = "true") | ||||
|     @NotNull(message = "是否发布不能为空") | ||||
|     @Schema(description = "是否发布", example = "true") | ||||
|     private Boolean publicStatus; | ||||
| 
 | ||||
| } | ||||
| @@ -20,11 +20,13 @@ public class AiSunoGenerateReqVO { | ||||
|      * 1. 描述模式:描述词 + 是否纯音乐 + 模型 | ||||
|      * 2. 歌词模式:歌词 + 音乐风格 + 标题 + 模型 | ||||
|      */ | ||||
|     @Schema(description = "生成模式 1.描述模式 2. 歌词模式", requiredMode = Schema.RequiredMode.REQUIRED, example = "2") | ||||
|     @Schema(description = "生成模式", requiredMode = Schema.RequiredMode.REQUIRED, example = "2") | ||||
|     @NotNull(message = "生成模式不能为空") | ||||
|     private Integer generateMode; // 参见 AiMusicGenerateModeEnum 枚举 | ||||
|  | ||||
|     @Schema(description = "歌词模式用:用于生成音乐音频的歌词提示", requiredMode = Schema.RequiredMode.NOT_REQUIRED, | ||||
|     // TODO @xin:方案一:prompt => lyric 歌词;gptDescriptionPrompt => description 描述(db 那字段也改下,避免和 gpt 直接耦合);这样搞完后,会更统一好理解一点 | ||||
|     // TODO @xin:方案二:还是之前的做法,都用 prompt;不过最终 gptDescriptionPrompt 还是存储 description 算描述。可以微信一起讨论下。 | ||||
|     @Schema(description = "用于生成音乐音频的歌词提示", | ||||
|             example = """ | ||||
|                     [Verse] | ||||
|                     阳光下奔跑 多么欢快 | ||||
| @@ -37,23 +39,23 @@ public class AiSunoGenerateReqVO { | ||||
|                     日子太短暂 别再等待 | ||||
|                     马上放假了 梦想起飞 | ||||
|                     """) | ||||
|     private String prompt; | ||||
|     private String prompt; // 歌词模式用 | ||||
|  | ||||
|     @Schema(description = "描述模式用:用于生成音乐音频的描述", requiredMode = Schema.RequiredMode.NOT_REQUIRED, | ||||
|     @Schema(description = "用于生成音乐音频的描述", | ||||
|             example = "创作一首带有轻松吉他旋律的流行歌曲,[verse] 描述夏日海滩的宁静,[chorus] 节奏加快,表达对自由的向往。") | ||||
|     private String gptDescriptionPrompt; | ||||
|     private String gptDescriptionPrompt; // 描述模式用 | ||||
|  | ||||
|     @Schema(description = "是否纯音乐", requiredMode = Schema.RequiredMode.NOT_REQUIRED, example = "true") | ||||
|     @Schema(description = "是否纯音乐", example = "true") | ||||
|     private Boolean makeInstrumental; | ||||
|  | ||||
|     @Schema(description = "模型", requiredMode = Schema.RequiredMode.REQUIRED, example = "chirp-v3.5") | ||||
|     @NotEmpty(message = "模型不能为空") | ||||
|     private String model; // 参见 AiModelEnum 枚举 | ||||
|  | ||||
|     @Schema(description = "音乐风格", requiredMode = Schema.RequiredMode.NOT_REQUIRED, example = "[\"pop\",\"jazz\",\"punk\"]") | ||||
|     @Schema(description = "音乐风格", example = "[\"pop\",\"jazz\",\"punk\"]") | ||||
|     private List<String> tags; | ||||
|  | ||||
|     @Schema(description = "音乐/歌曲名称", requiredMode = Schema.RequiredMode.NOT_REQUIRED, example = "夜空中最亮的星") | ||||
|     @Schema(description = "音乐/歌曲名称", example = "夜空中最亮的星") | ||||
|     private String title; | ||||
|  | ||||
| } | ||||
| @@ -98,6 +98,11 @@ public class AiMusicDO extends BaseDO { | ||||
|     @TableField(typeHandler = JacksonTypeHandler.class) | ||||
|     private List<String> tags; | ||||
|  | ||||
|     /** | ||||
|      * 音乐时长 | ||||
|      */ | ||||
|     private Double duration; | ||||
|  | ||||
|     /** | ||||
|      * 是否公开 | ||||
|      */ | ||||
| @@ -113,10 +118,4 @@ public class AiMusicDO extends BaseDO { | ||||
|      */ | ||||
|     private String errorMessage; | ||||
|  | ||||
|  | ||||
|     /** | ||||
|      * 音乐时长 | ||||
|      */ | ||||
|     private Double duration; | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| package cn.iocoder.yudao.module.ai.service.model; | ||||
|  | ||||
| import cn.iocoder.yudao.framework.ai.core.enums.AiPlatformEnum; | ||||
| import cn.iocoder.yudao.framework.ai.core.model.suno.api.SunoApi; | ||||
| import cn.iocoder.yudao.framework.common.pojo.PageResult; | ||||
| import cn.iocoder.yudao.module.ai.controller.admin.model.vo.apikey.AiApiKeyPageReqVO; | ||||
| import cn.iocoder.yudao.module.ai.controller.admin.model.vo.apikey.AiApiKeySaveReqVO; | ||||
| @@ -91,4 +92,13 @@ public interface AiApiKeyService { | ||||
|      */ | ||||
|     ImageClient getImageClient(AiPlatformEnum platform); | ||||
|  | ||||
|     /** | ||||
|      * 获得 SunoApi 对象 | ||||
|      * | ||||
|      * TODO 可优化点:目前默认获取 Suno 对应的第一个开启的配置用于音乐;后续可以支持配置选择 | ||||
|      * | ||||
|      * @return SunoApi 对象 | ||||
|      */ | ||||
|     SunoApi getSunoApi(); | ||||
|  | ||||
| } | ||||
| @@ -2,6 +2,7 @@ package cn.iocoder.yudao.module.ai.service.model; | ||||
|  | ||||
| import cn.iocoder.yudao.framework.ai.core.enums.AiPlatformEnum; | ||||
| import cn.iocoder.yudao.framework.ai.core.factory.AiClientFactory; | ||||
| import cn.iocoder.yudao.framework.ai.core.model.suno.api.SunoApi; | ||||
| import cn.iocoder.yudao.framework.common.enums.CommonStatusEnum; | ||||
| import cn.iocoder.yudao.framework.common.pojo.PageResult; | ||||
| import cn.iocoder.yudao.framework.common.util.object.BeanUtils; | ||||
| @@ -111,4 +112,14 @@ public class AiApiKeyServiceImpl implements AiApiKeyService { | ||||
|         return clientFactory.getOrCreateImageClient(platform, apiKey.getApiKey(), apiKey.getUrl()); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public SunoApi getSunoApi() { | ||||
|         AiApiKeyDO apiKey = apiKeyMapper.selectFirstByPlatformAndStatus( | ||||
|                 AiPlatformEnum.SUNO.getPlatform(), CommonStatusEnum.ENABLE.getStatus()); | ||||
|         if (apiKey == null) { | ||||
|             return null; | ||||
|         } | ||||
|         return clientFactory.getOrCreateSunoApi(apiKey.getApiKey(), apiKey.getUrl()); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -35,7 +35,7 @@ public interface AiMusicService { | ||||
|      * | ||||
|      * @param updateReqVO 更新信息 | ||||
|      */ | ||||
|     void updateMusicPublicStatus(@Valid AiMusicUpdatePublicStatusReqVO updateReqVO); | ||||
|     void updateMusic(@Valid AiMusicUpdateReqVO updateReqVO); | ||||
|  | ||||
|     /** | ||||
|      * 更新音乐名称 | ||||
| @@ -83,4 +83,5 @@ public interface AiMusicService { | ||||
|      * @return 音乐分页 | ||||
|      */ | ||||
|     PageResult<AiMusicDO> getMusicMyPage(AiMusicPageReqVO pageReqVO, Long userId); | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -12,6 +12,7 @@ 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.AiMusicGenerateModeEnum; | ||||
| import cn.iocoder.yudao.module.ai.enums.music.AiMusicStatusEnum; | ||||
| import cn.iocoder.yudao.module.ai.service.model.AiApiKeyService; | ||||
| import cn.iocoder.yudao.module.infra.api.file.FileApi; | ||||
| import jakarta.annotation.Resource; | ||||
| import lombok.extern.slf4j.Slf4j; | ||||
| @@ -35,7 +36,7 @@ import static cn.iocoder.yudao.module.ai.enums.ErrorCodeConstants.MUSIC_NOT_EXIS | ||||
| public class AiMusicServiceImpl implements AiMusicService { | ||||
|  | ||||
|     @Resource | ||||
|     private SunoApi sunoApi; | ||||
|     private AiApiKeyService apiKeyService; | ||||
|  | ||||
|     @Resource | ||||
|     private AiMusicMapper musicMapper; | ||||
| @@ -46,6 +47,8 @@ public class AiMusicServiceImpl implements AiMusicService { | ||||
|     @Override | ||||
|     public List<Long> generateMusic(Long userId, AiSunoGenerateReqVO reqVO) { | ||||
|         // 1. 调用 Suno 生成音乐 | ||||
|         SunoApi sunoApi = apiKeyService.getSunoApi(); | ||||
|         // TODO @xin:这两个貌似一直没跑成功,你那可以么?用的请求是 AiMusicController.http 的 | ||||
|         List<SunoApi.MusicData> musicDataList; | ||||
|         if (Objects.equals(AiMusicGenerateModeEnum.LYRIC.getMode(), reqVO.getGenerateMode())) { | ||||
|             // 1.1 歌词模式 | ||||
| @@ -80,6 +83,7 @@ public class AiMusicServiceImpl implements AiMusicService { | ||||
|         log.info("[syncMusic][Suno 开始同步, 共 ({}) 个任务]", streamingTask.size()); | ||||
|  | ||||
|         // GET 请求,为避免参数过长,分批次处理 | ||||
|         SunoApi sunoApi = apiKeyService.getSunoApi(); | ||||
|         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())); | ||||
| @@ -96,7 +100,7 @@ public class AiMusicServiceImpl implements AiMusicService { | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void updateMusicPublicStatus(AiMusicUpdatePublicStatusReqVO updateReqVO) { | ||||
|     public void updateMusic(AiMusicUpdateReqVO updateReqVO) { | ||||
|         // 校验存在 | ||||
|         validateMusicExists(updateReqVO.getId()); | ||||
|         // 更新 | ||||
| @@ -152,11 +156,16 @@ public class AiMusicServiceImpl implements AiMusicService { | ||||
|      * @return AiMusicDO 集合 | ||||
|      */ | ||||
|     private List<AiMusicDO> buildMusicDOList(List<SunoApi.MusicData> musicList) { | ||||
|         // TODO @xin:它有 status = error 状态,表示失败噢。 | ||||
|         return convertList(musicList, musicData -> new AiMusicDO() | ||||
|                 .setTaskId(musicData.id()).setModel(musicData.modelName()) | ||||
|                 .setPrompt(musicData.prompt()).setGptDescriptionPrompt(musicData.gptDescriptionPrompt()) | ||||
|                 .setAudioUrl(createFile(musicData.audioUrl())).setVideoUrl(createFile(musicData.videoUrl())).setImageUrl(createFile(musicData.imageUrl())).setDuration(musicData.duration()) | ||||
|                 .setTitle(musicData.title()).setLyric(musicData.lyric()).setTags(StrUtil.split(musicData.tags(), StrPool.COMMA)) | ||||
|                 // TODO @xin:只有在完成的状态,在下载文件 | ||||
|                 .setAudioUrl(downloadFile(musicData.audioUrl())) | ||||
|                 .setVideoUrl(downloadFile(musicData.videoUrl())) | ||||
|                 .setImageUrl(downloadFile(musicData.imageUrl())) | ||||
|                 .setTitle(musicData.title()).setDuration(musicData.duration()) | ||||
|                 .setLyric(musicData.lyric()).setTags(StrUtil.split(musicData.tags(), StrPool.COMMA)) | ||||
|                 .setStatus(Objects.equals("complete", musicData.status()) ? | ||||
|                         AiMusicStatusEnum.SUCCESS.getStatus() : AiMusicStatusEnum.IN_PROGRESS.getStatus())); | ||||
|     } | ||||
| @@ -167,12 +176,17 @@ public class AiMusicServiceImpl implements AiMusicService { | ||||
|      * @param url 音频文件地址 | ||||
|      * @return 内部文件地址 | ||||
|      */ | ||||
|     private String createFile(String url) { | ||||
|     private String downloadFile(String url) { | ||||
|         if (StrUtil.isBlank(url)) { | ||||
|             return null; | ||||
|         } | ||||
|         byte[] bytes = HttpUtil.downloadBytes(url); | ||||
|         return fileApi.createFile(bytes); | ||||
|         try { | ||||
|             byte[] bytes = HttpUtil.downloadBytes(url); | ||||
|             return fileApi.createFile(bytes); | ||||
|         } catch (Exception e) { | ||||
|             log.error("[downloadFile][url({}) 下载失败]", url, e); | ||||
|             return url; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| package cn.iocoder.yudao.framework.ai.core.factory; | ||||
|  | ||||
| import cn.iocoder.yudao.framework.ai.core.enums.AiPlatformEnum; | ||||
| import cn.iocoder.yudao.framework.ai.core.model.suno.api.SunoApi; | ||||
| import org.springframework.ai.chat.StreamingChatClient; | ||||
| import org.springframework.ai.image.ImageClient; | ||||
|  | ||||
| @@ -55,4 +56,15 @@ public interface AiClientFactory { | ||||
|      */ | ||||
|     ImageClient getOrCreateImageClient(AiPlatformEnum platform, String apiKey, String url); | ||||
|  | ||||
|     /** | ||||
|      * 基于指定配置,获得 SunoApi 对象 | ||||
|      * | ||||
|      * 如果不存在,则进行创建 | ||||
|      * | ||||
|      * @param apiKey API KEY | ||||
|      * @param url API URL | ||||
|      * @return SunoApi 对象 | ||||
|      */ | ||||
|     SunoApi getOrCreateSunoApi(String apiKey, String url); | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -9,6 +9,7 @@ import cn.hutool.extra.spring.SpringUtil; | ||||
| import cn.iocoder.yudao.framework.ai.config.YudaoAiAutoConfiguration; | ||||
| import cn.iocoder.yudao.framework.ai.config.YudaoAiProperties; | ||||
| import cn.iocoder.yudao.framework.ai.core.enums.AiPlatformEnum; | ||||
| import cn.iocoder.yudao.framework.ai.core.model.suno.api.SunoApi; | ||||
| import cn.iocoder.yudao.framework.ai.core.model.tongyi.QianWenChatClient; | ||||
| import cn.iocoder.yudao.framework.ai.core.model.tongyi.QianWenChatModal; | ||||
| import cn.iocoder.yudao.framework.ai.core.model.tongyi.api.QianWenApi; | ||||
| @@ -109,6 +110,11 @@ public class AiClientFactoryImpl implements AiClientFactory { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public SunoApi getOrCreateSunoApi(String apiKey, String url) { | ||||
|         return new SunoApi(url); | ||||
|     } | ||||
|  | ||||
|     private static String buildClientCacheKey(Class<?> clazz, Object... params) { | ||||
|         if (ArrayUtil.isEmpty(params)) { | ||||
|             return clazz.getName(); | ||||
|   | ||||
| @@ -201,8 +201,8 @@ yudao.ai: | ||||
|     notify-url: http://java.nat300.top/admin-api/ai/image/midjourney/notify | ||||
|   suno: | ||||
|     enable: true | ||||
|     base-url: https://suno-55ishh05u-status2xxs-projects.vercel.app | ||||
| #    base-url: http://127.0.0.1:3001 | ||||
| #    base-url: https://suno-55ishh05u-status2xxs-projects.vercel.app | ||||
|     base-url: http://127.0.0.1:3001 | ||||
|  | ||||
| --- #################### 芋道相关配置 #################### | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 YunaiV
					YunaiV