【代码评审】AI:文心一言的接入调整
This commit is contained in:
parent
276ef98ff1
commit
04021ce068
|
@ -5,7 +5,7 @@ import org.springframework.ai.chat.ChatClient;
|
||||||
import org.springframework.ai.chat.StreamingChatClient;
|
import org.springframework.ai.chat.StreamingChatClient;
|
||||||
import org.springframework.ai.models.tongyi.QianWenChatClient;
|
import org.springframework.ai.models.tongyi.QianWenChatClient;
|
||||||
import org.springframework.ai.models.xinghuo.XingHuoChatClient;
|
import org.springframework.ai.models.xinghuo.XingHuoChatClient;
|
||||||
import org.springframework.ai.models.yiyan.YiYanChatClient;
|
import cn.iocoder.yudao.framework.ai.core.model.yiyan.YiYanChatClient;
|
||||||
import org.springframework.ai.ollama.OllamaChatClient;
|
import org.springframework.ai.ollama.OllamaChatClient;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
import org.springframework.context.ApplicationContext;
|
import org.springframework.context.ApplicationContext;
|
||||||
|
@ -27,7 +27,7 @@ public class AiChatClientFactory {
|
||||||
public ChatClient getChatClient(AiPlatformEnum platformEnum) {
|
public ChatClient getChatClient(AiPlatformEnum platformEnum) {
|
||||||
if (AiPlatformEnum.QIAN_WEN == platformEnum) {
|
if (AiPlatformEnum.QIAN_WEN == platformEnum) {
|
||||||
return applicationContext.getBean(QianWenChatClient.class);
|
return applicationContext.getBean(QianWenChatClient.class);
|
||||||
} else if (AiPlatformEnum.YI_YAN == platformEnum) {
|
} else if (AiPlatformEnum.YIYAN == platformEnum) {
|
||||||
return applicationContext.getBean(YiYanChatClient.class);
|
return applicationContext.getBean(YiYanChatClient.class);
|
||||||
} else if (AiPlatformEnum.XING_HUO == platformEnum) {
|
} else if (AiPlatformEnum.XING_HUO == platformEnum) {
|
||||||
return applicationContext.getBean(XingHuoChatClient.class);
|
return applicationContext.getBean(XingHuoChatClient.class);
|
||||||
|
@ -42,7 +42,7 @@ public class AiChatClientFactory {
|
||||||
// }
|
// }
|
||||||
if (AiPlatformEnum.QIAN_WEN == platformEnum) {
|
if (AiPlatformEnum.QIAN_WEN == platformEnum) {
|
||||||
return applicationContext.getBean(QianWenChatClient.class);
|
return applicationContext.getBean(QianWenChatClient.class);
|
||||||
} else if (AiPlatformEnum.YI_YAN == platformEnum) {
|
} else if (AiPlatformEnum.YIYAN == platformEnum) {
|
||||||
return applicationContext.getBean(YiYanChatClient.class);
|
return applicationContext.getBean(YiYanChatClient.class);
|
||||||
} else if (AiPlatformEnum.XING_HUO == platformEnum) {
|
} else if (AiPlatformEnum.XING_HUO == platformEnum) {
|
||||||
return applicationContext.getBean(XingHuoChatClient.class);
|
return applicationContext.getBean(XingHuoChatClient.class);
|
||||||
|
|
|
@ -1,20 +1,14 @@
|
||||||
package cn.iocoder.yudao.module.ai.controller.admin.chat.vo.conversation;
|
package cn.iocoder.yudao.module.ai.controller.admin.chat.vo.conversation;
|
||||||
|
|
||||||
import cn.iocoder.yudao.module.ai.controller.admin.model.vo.chatModel.AiChatModelRespVO;
|
|
||||||
import cn.iocoder.yudao.module.ai.dal.dataobject.model.AiChatModelDO;
|
import cn.iocoder.yudao.module.ai.dal.dataobject.model.AiChatModelDO;
|
||||||
import cn.iocoder.yudao.module.ai.dal.dataobject.model.AiChatRoleDO;
|
import cn.iocoder.yudao.module.ai.dal.dataobject.model.AiChatRoleDO;
|
||||||
import com.baomidou.mybatisplus.annotation.FieldFill;
|
|
||||||
import com.baomidou.mybatisplus.annotation.TableField;
|
|
||||||
import com.fhs.core.trans.anno.Trans;
|
import com.fhs.core.trans.anno.Trans;
|
||||||
import com.fhs.core.trans.constant.TransType;
|
import com.fhs.core.trans.constant.TransType;
|
||||||
import com.fhs.core.trans.vo.VO;
|
import com.fhs.core.trans.vo.VO;
|
||||||
import io.swagger.v3.oas.annotations.media.Schema;
|
import io.swagger.v3.oas.annotations.media.Schema;
|
||||||
import jakarta.validation.constraints.NotNull;
|
|
||||||
import lombok.Data;
|
import lombok.Data;
|
||||||
import lombok.experimental.Accessors;
|
|
||||||
|
|
||||||
import java.time.LocalDateTime;
|
import java.time.LocalDateTime;
|
||||||
import java.time.LocalTime;
|
|
||||||
|
|
||||||
@Schema(description = "管理后台 - AI 聊天会话 Response VO")
|
@Schema(description = "管理后台 - AI 聊天会话 Response VO")
|
||||||
@Data
|
@Data
|
||||||
|
@ -58,7 +52,7 @@ public class AiChatConversationRespVO implements VO {
|
||||||
@Schema(description = "上下文的最大 Message 数量", requiredMode = Schema.RequiredMode.REQUIRED, example = "10")
|
@Schema(description = "上下文的最大 Message 数量", requiredMode = Schema.RequiredMode.REQUIRED, example = "10")
|
||||||
private Integer maxContexts;
|
private Integer maxContexts;
|
||||||
|
|
||||||
@Schema(description = "最后更新时间", requiredMode = Schema.RequiredMode.REQUIRED, example = "2024-05-16")
|
@Schema(description = "最后更新时间", requiredMode = Schema.RequiredMode.REQUIRED)
|
||||||
private LocalDateTime updateTime;
|
private LocalDateTime updateTime;
|
||||||
|
|
||||||
// ========== 关联 role 信息 ==========
|
// ========== 关联 role 信息 ==========
|
||||||
|
|
|
@ -4,7 +4,7 @@ import cn.iocoder.yudao.framework.ai.core.enums.AiPlatformEnum;
|
||||||
import lombok.Data;
|
import lombok.Data;
|
||||||
import lombok.experimental.Accessors;
|
import lombok.experimental.Accessors;
|
||||||
import org.springframework.ai.models.xinghuo.XingHuoChatModel;
|
import org.springframework.ai.models.xinghuo.XingHuoChatModel;
|
||||||
import org.springframework.ai.models.yiyan.YiYanChatModel;
|
import cn.iocoder.yudao.framework.ai.core.model.yiyan.api.YiYanChatModel;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* modal config
|
* modal config
|
||||||
|
|
|
@ -150,9 +150,9 @@ public class AiChatServiceImpl implements AiChatService {
|
||||||
// 1.3 user message 新发送消息
|
// 1.3 user message 新发送消息
|
||||||
chatMessages.add(new UserMessage(sendReqVO.getContent()));
|
chatMessages.add(new UserMessage(sendReqVO.getContent()));
|
||||||
|
|
||||||
// 2. 构建 ChatOptions 对象
|
// 2. 构建 ChatOptions 对象 TODO 芋艿:临时注释掉;等文心一言兼容了;
|
||||||
ChatOptions chatOptions = ChatOptionsBuilder.builder().withTemperature(conversation.getTemperature().floatValue()).build();
|
// ChatOptions chatOptions = ChatOptionsBuilder.builder().withTemperature(conversation.getTemperature().floatValue()).build();
|
||||||
return new Prompt(chatMessages, chatOptions);
|
return new Prompt(chatMessages, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
private AiChatMessageDO createChatMessage(Long conversationId, AiChatModelDO model,
|
private AiChatMessageDO createChatMessage(Long conversationId, AiChatModelDO model,
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
package cn.iocoder.yudao.module.ai.service.model;
|
package cn.iocoder.yudao.module.ai.service.model;
|
||||||
|
|
||||||
import cn.hutool.core.util.ObjectUtil;
|
import cn.hutool.core.util.ObjectUtil;
|
||||||
|
import cn.hutool.core.util.StrUtil;
|
||||||
import cn.iocoder.yudao.framework.common.enums.CommonStatusEnum;
|
import cn.iocoder.yudao.framework.common.enums.CommonStatusEnum;
|
||||||
import cn.iocoder.yudao.framework.common.pojo.PageResult;
|
import cn.iocoder.yudao.framework.common.pojo.PageResult;
|
||||||
import cn.iocoder.yudao.framework.common.util.object.BeanUtils;
|
import cn.iocoder.yudao.framework.common.util.object.BeanUtils;
|
||||||
|
@ -15,6 +16,8 @@ import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
|
import java.util.function.Function;
|
||||||
|
import java.util.function.Predicate;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
import static cn.iocoder.yudao.framework.common.exception.util.ServiceExceptionUtil.exception;
|
import static cn.iocoder.yudao.framework.common.exception.util.ServiceExceptionUtil.exception;
|
||||||
|
@ -134,7 +137,7 @@ public class AiChatRoleServiceImpl implements AiChatRoleService {
|
||||||
@Override
|
@Override
|
||||||
public List<String> getChatRoleCategoryList() {
|
public List<String> getChatRoleCategoryList() {
|
||||||
List<AiChatRoleDO> list = chatRoleMapper.selectListGroupByCategory(CommonStatusEnum.ENABLE.getStatus());
|
List<AiChatRoleDO> list = chatRoleMapper.selectListGroupByCategory(CommonStatusEnum.ENABLE.getStatus());
|
||||||
return convertList(list.stream().filter(Objects::nonNull).collect(Collectors.toList()), AiChatRoleDO::getCategory);
|
return convertList(list, AiChatRoleDO::getCategory, role -> StrUtil.isNotBlank(role.getCategory()));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,25 +10,18 @@
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
<artifactId>yudao-spring-boot-starter-ai</artifactId>
|
<artifactId>yudao-spring-boot-starter-ai</artifactId>
|
||||||
<!-- TODO 芋艿:这里需要进一步减少 -->
|
|
||||||
|
|
||||||
<dependencies>
|
<dependencies>
|
||||||
<dependency>
|
|
||||||
<groupId>io.springboot.ai</groupId>
|
|
||||||
<artifactId>spring-ai-core</artifactId>
|
|
||||||
<version>1.0.3</version>
|
|
||||||
</dependency>
|
|
||||||
<dependency>
|
|
||||||
<groupId>io.springboot.ai</groupId>
|
|
||||||
<artifactId>spring-ai-openai</artifactId>
|
|
||||||
<version>1.0.3</version>
|
|
||||||
</dependency>
|
|
||||||
|
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>io.springboot.ai</groupId>
|
<groupId>io.springboot.ai</groupId>
|
||||||
<artifactId>spring-ai-ollama-spring-boot-starter</artifactId>
|
<artifactId>spring-ai-ollama-spring-boot-starter</artifactId>
|
||||||
<version>1.0.3</version>
|
<version>1.0.3</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.springboot.ai</groupId>
|
||||||
|
<artifactId>spring-ai-openai-spring-boot-starter</artifactId>
|
||||||
|
<version>1.0.3</version>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>cn.iocoder.boot</groupId>
|
<groupId>cn.iocoder.boot</groupId>
|
||||||
|
|
|
@ -8,9 +8,9 @@ import org.springframework.ai.models.tongyi.api.QianWenApi;
|
||||||
import org.springframework.ai.models.xinghuo.XingHuoChatClient;
|
import org.springframework.ai.models.xinghuo.XingHuoChatClient;
|
||||||
import org.springframework.ai.models.xinghuo.XingHuoOptions;
|
import org.springframework.ai.models.xinghuo.XingHuoOptions;
|
||||||
import org.springframework.ai.models.xinghuo.api.XingHuoApi;
|
import org.springframework.ai.models.xinghuo.api.XingHuoApi;
|
||||||
import org.springframework.ai.models.yiyan.YiYanChatClient;
|
import cn.iocoder.yudao.framework.ai.core.model.yiyan.YiYanChatClient;
|
||||||
import org.springframework.ai.models.yiyan.YiYanOptions;
|
import cn.iocoder.yudao.framework.ai.core.model.yiyan.YiYanChatOptions;
|
||||||
import org.springframework.ai.models.yiyan.api.YiYanApi;
|
import cn.iocoder.yudao.framework.ai.core.model.yiyan.api.YiYanApi;
|
||||||
import org.springframework.ai.models.midjourney.MidjourneyConfig;
|
import org.springframework.ai.models.midjourney.MidjourneyConfig;
|
||||||
import org.springframework.ai.models.midjourney.MidjourneyMessage;
|
import org.springframework.ai.models.midjourney.MidjourneyMessage;
|
||||||
import org.springframework.ai.models.midjourney.api.MidjourneyInteractionsApi;
|
import org.springframework.ai.models.midjourney.api.MidjourneyInteractionsApi;
|
||||||
|
@ -91,7 +91,7 @@ public class YudaoAiAutoConfiguration {
|
||||||
public YiYanChatClient yiYanChatClient(YudaoAiProperties yudaoAiProperties) {
|
public YiYanChatClient yiYanChatClient(YudaoAiProperties yudaoAiProperties) {
|
||||||
YudaoAiProperties.YiYanProperties yiYanProperties = yudaoAiProperties.getYiyan();
|
YudaoAiProperties.YiYanProperties yiYanProperties = yudaoAiProperties.getYiyan();
|
||||||
// 转换配置
|
// 转换配置
|
||||||
YiYanOptions yiYanOptions = new YiYanOptions();
|
YiYanChatOptions yiYanOptions = new YiYanChatOptions();
|
||||||
// yiYanOptions.setTopK(yiYanProperties.getTopK()); TODO 芋艿:后续弄
|
// yiYanOptions.setTopK(yiYanProperties.getTopK()); TODO 芋艿:后续弄
|
||||||
yiYanOptions.setTopP(yiYanProperties.getTopP());
|
yiYanOptions.setTopP(yiYanProperties.getTopP());
|
||||||
yiYanOptions.setTemperature(yiYanProperties.getTemperature());
|
yiYanOptions.setTemperature(yiYanProperties.getTemperature());
|
||||||
|
|
|
@ -3,7 +3,7 @@ package cn.iocoder.yudao.framework.ai.config;
|
||||||
import cn.iocoder.yudao.framework.ai.core.enums.AiPlatformEnum;
|
import cn.iocoder.yudao.framework.ai.core.enums.AiPlatformEnum;
|
||||||
import org.springframework.ai.models.xinghuo.XingHuoChatModel;
|
import org.springframework.ai.models.xinghuo.XingHuoChatModel;
|
||||||
import org.springframework.ai.models.xinghuo.XingHuoOptions;
|
import org.springframework.ai.models.xinghuo.XingHuoOptions;
|
||||||
import org.springframework.ai.models.yiyan.YiYanChatModel;
|
import cn.iocoder.yudao.framework.ai.core.model.yiyan.api.YiYanChatModel;
|
||||||
import lombok.Data;
|
import lombok.Data;
|
||||||
import lombok.experimental.Accessors;
|
import lombok.experimental.Accessors;
|
||||||
|
|
||||||
|
|
|
@ -2,7 +2,7 @@ package cn.iocoder.yudao.framework.ai.config;
|
||||||
|
|
||||||
import cn.iocoder.yudao.framework.ai.core.enums.AiPlatformEnum;
|
import cn.iocoder.yudao.framework.ai.core.enums.AiPlatformEnum;
|
||||||
import org.springframework.ai.models.xinghuo.XingHuoChatModel;
|
import org.springframework.ai.models.xinghuo.XingHuoChatModel;
|
||||||
import org.springframework.ai.models.yiyan.YiYanChatModel;
|
import cn.iocoder.yudao.framework.ai.core.model.yiyan.api.YiYanChatModel;
|
||||||
import cn.iocoder.yudao.framework.ai.core.enums.OpenAiImageModelEnum;
|
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.enums.OpenAiImageStyleEnum;
|
||||||
import lombok.Data;
|
import lombok.Data;
|
||||||
|
|
|
@ -16,8 +16,8 @@ public enum AiPlatformEnum {
|
||||||
|
|
||||||
OPENAI("OpenAI", "OpenAI"),
|
OPENAI("OpenAI", "OpenAI"),
|
||||||
OLLAMA("Ollama", "Ollama"),
|
OLLAMA("Ollama", "Ollama"),
|
||||||
|
YIYAN("YiYan", "文心一言"),
|
||||||
|
|
||||||
YI_YAN("yiyan", "一言"),
|
|
||||||
QIAN_WEN("qianwen", "千问"),
|
QIAN_WEN("qianwen", "千问"),
|
||||||
XING_HUO("xinghuo", "星火"),
|
XING_HUO("xinghuo", "星火"),
|
||||||
OPEN_AI_DALL("dall", "dall"),
|
OPEN_AI_DALL("dall", "dall"),
|
||||||
|
|
|
@ -0,0 +1,12 @@
|
||||||
|
/**
|
||||||
|
* model 包,接入各种大模型,对标 https://github.com/spring-projects/spring-ai/tree/main/models
|
||||||
|
*
|
||||||
|
* 1. yiyan 包:【百度】文心一言
|
||||||
|
* 2. TODO 芋艿:
|
||||||
|
* tongyi 包:【阿里】通义千问,对标 spring-cloud-alibaba 提供的 ai 包
|
||||||
|
* 2.2
|
||||||
|
* 2.3 xinghuo 包:【讯飞】星火,自己实现
|
||||||
|
* 2.4 openai 包:【OpenAI】ChatGPT,拷贝 spring-ai 提供的 models/openai 包
|
||||||
|
* 2.5 midjourney 包:Midjourney,参考 https://github.com/novicezk/midjourney-proxy 实现
|
||||||
|
*/
|
||||||
|
package cn.iocoder.yudao.framework.ai.core.model;
|
|
@ -0,0 +1,154 @@
|
||||||
|
package cn.iocoder.yudao.framework.ai.core.model.yiyan;
|
||||||
|
|
||||||
|
import cn.hutool.core.bean.BeanUtil;
|
||||||
|
import cn.iocoder.yudao.framework.ai.core.exception.ChatException;
|
||||||
|
import cn.iocoder.yudao.framework.ai.core.model.yiyan.api.YiYanApi;
|
||||||
|
import cn.iocoder.yudao.framework.ai.core.model.yiyan.api.YiYanChatCompletionResponse;
|
||||||
|
import cn.iocoder.yudao.framework.ai.core.model.yiyan.api.YiYanChatCompletionRequest;
|
||||||
|
import cn.iocoder.yudao.framework.ai.core.model.yiyan.exception.YiYanApiException;
|
||||||
|
import lombok.extern.slf4j.Slf4j;
|
||||||
|
import org.springframework.ai.chat.ChatClient;
|
||||||
|
import org.springframework.ai.chat.ChatResponse;
|
||||||
|
import org.springframework.ai.chat.Generation;
|
||||||
|
import org.springframework.ai.chat.StreamingChatClient;
|
||||||
|
import org.springframework.ai.chat.messages.Message;
|
||||||
|
import org.springframework.ai.chat.messages.MessageType;
|
||||||
|
import org.springframework.ai.chat.prompt.ChatOptions;
|
||||||
|
import org.springframework.ai.chat.prompt.Prompt;
|
||||||
|
import org.springframework.http.ResponseEntity;
|
||||||
|
import org.springframework.retry.RetryCallback;
|
||||||
|
import org.springframework.retry.RetryContext;
|
||||||
|
import org.springframework.retry.RetryListener;
|
||||||
|
import org.springframework.retry.support.RetryTemplate;
|
||||||
|
import org.springframework.util.Assert;
|
||||||
|
import reactor.core.publisher.Flux;
|
||||||
|
|
||||||
|
import java.time.Duration;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 文心一言的 {@link ChatClient} 实现类
|
||||||
|
*
|
||||||
|
* @author fansili
|
||||||
|
*/
|
||||||
|
@Slf4j
|
||||||
|
public class YiYanChatClient implements ChatClient, StreamingChatClient {
|
||||||
|
|
||||||
|
private final YiYanApi yiYanApi;
|
||||||
|
|
||||||
|
private YiYanChatOptions defaultOptions;
|
||||||
|
|
||||||
|
// TODO @fan:参考 OpenAiChatClient 调整下 retryTemplate;使用 RetryUtils.DEFAULT_RETRY_TEMPLATE;加允许传入?
|
||||||
|
|
||||||
|
public YiYanChatClient(YiYanApi yiYanApi) {
|
||||||
|
this.yiYanApi = yiYanApi;
|
||||||
|
// TODO @fan:这个情况,是不是搞个 defaultOptions;OpenAiChatOptions.builder().withModel(OpenAiApi.DEFAULT_CHAT_MODEL).withTemperature(0.7f).build()
|
||||||
|
}
|
||||||
|
|
||||||
|
public YiYanChatClient(YiYanApi yiYanApi, YiYanChatOptions defaultOptions) {
|
||||||
|
Assert.notNull(yiYanApi, "OllamaApi must not be null");
|
||||||
|
Assert.notNull(defaultOptions, "DefaultOptions must not be null");
|
||||||
|
this.yiYanApi = yiYanApi;
|
||||||
|
this.defaultOptions = defaultOptions;
|
||||||
|
}
|
||||||
|
|
||||||
|
public final RetryTemplate retryTemplate = RetryTemplate.builder()
|
||||||
|
.maxAttempts(10)
|
||||||
|
.retryOn(YiYanApiException.class)
|
||||||
|
.exponentialBackoff(Duration.ofMillis(3000), 2, Duration.ofMillis(3 * 60000))
|
||||||
|
.withListener(new RetryListener() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public <T, E extends Throwable> void onError(RetryContext context,
|
||||||
|
RetryCallback<T, E> callback, Throwable throwable) {
|
||||||
|
log.warn("重试异常:" + context.getRetryCount(), throwable);
|
||||||
|
}
|
||||||
|
|
||||||
|
})
|
||||||
|
.build();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ChatResponse call(Prompt prompt) {
|
||||||
|
YiYanChatCompletionRequest request = createRequest(prompt, false);
|
||||||
|
return this.retryTemplate.execute(ctx -> {
|
||||||
|
// 发送请求
|
||||||
|
ResponseEntity<YiYanChatCompletionResponse> response = yiYanApi.chatCompletionEntity(request);
|
||||||
|
// 获取结果封装 ChatResponse
|
||||||
|
YiYanChatCompletionResponse chatCompletion = response.getBody();
|
||||||
|
// TODO @fan:为空时,参考 OpenAiChatClient 的封装;
|
||||||
|
// TODO @fan:chatResponseMetadata,参考 OpenAiChatResponseMetadata.from(completionEntity.getBody())
|
||||||
|
return new ChatResponse(List.of(new Generation(chatCompletion.getResult())));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Flux<ChatResponse> stream(Prompt prompt) {
|
||||||
|
YiYanChatCompletionRequest request = this.createRequest(prompt, true);
|
||||||
|
// TODO @fan:return this.retryTemplate.execute(ctx -> {
|
||||||
|
// 调用 callWithFunctionSupport 发送请求
|
||||||
|
Flux<YiYanChatCompletionResponse> response = this.yiYanApi.chatCompletionStream(request);
|
||||||
|
// TODO @fan:下面的 doOnComplete 是不是可以删除哈?
|
||||||
|
response.doOnComplete(new Runnable() {
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
String a = ";";
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return response.map(chunk -> {
|
||||||
|
// TODO @fan:ChatResponseMetadata chatResponseMetadata
|
||||||
|
return new ChatResponse(List.of(new Generation(chunk.getResult())));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private YiYanChatCompletionRequest createRequest(Prompt prompt, boolean stream) {
|
||||||
|
// 参考 https://cloud.baidu.com/doc/WENXINWORKSHOP/s/clntwmv7t 文档,system 是独立字段
|
||||||
|
// 1.1 获取 user 和 assistant
|
||||||
|
List<YiYanChatCompletionRequest.Message> messageList = prompt.getInstructions().stream()
|
||||||
|
// 过滤 system
|
||||||
|
.filter(msg -> MessageType.SYSTEM != msg.getMessageType())
|
||||||
|
.map(message -> new YiYanChatCompletionRequest.Message()
|
||||||
|
.setRole(message.getMessageType().getValue()).setContent(message.getContent())
|
||||||
|
).toList();
|
||||||
|
// 1.2 获取 system
|
||||||
|
String systemPrompt = prompt.getInstructions().stream()
|
||||||
|
.filter(message -> MessageType.SYSTEM == message.getMessageType())
|
||||||
|
.map(Message::getContent)
|
||||||
|
.collect(Collectors.joining());
|
||||||
|
|
||||||
|
// 3. 创建 request
|
||||||
|
YiYanChatCompletionRequest request = new YiYanChatCompletionRequest(messageList);
|
||||||
|
// 复制 YiYanOptions 属性,到 request 中(这里 options 属性和 request 基本保持一致)
|
||||||
|
YiYanChatOptions useOptions = getYiYanOptions(prompt);
|
||||||
|
BeanUtil.copyProperties(useOptions, request);
|
||||||
|
request.setTop_p(useOptions.getTopP())
|
||||||
|
.setMax_output_tokens(useOptions.getMaxOutputTokens())
|
||||||
|
.setTemperature(useOptions.getTemperature())
|
||||||
|
.setSystem(systemPrompt)
|
||||||
|
.setStream(stream);
|
||||||
|
return request;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO @fan:Options 的处理,参考下 OpenAiChatClient 的 createRequest
|
||||||
|
private YiYanChatOptions getYiYanOptions(Prompt prompt) {
|
||||||
|
// 两个都为null 则没有配置文件
|
||||||
|
if (defaultOptions == null && prompt.getOptions() == null) {
|
||||||
|
// TODO @fan:IllegalArgumentException 参数更好哈
|
||||||
|
throw new ChatException("ChatOptions 未配置参数!");
|
||||||
|
}
|
||||||
|
// 优先使用 Prompt 里面的 ChatOptions
|
||||||
|
ChatOptions options = defaultOptions;
|
||||||
|
if (prompt.getOptions() != null) {
|
||||||
|
options = (ChatOptions) prompt.getOptions();
|
||||||
|
}
|
||||||
|
// Prompt 里面是一个 ChatOptions,用户可以随意传入,这里做一下判断
|
||||||
|
if (!(options instanceof YiYanChatOptions)) {
|
||||||
|
// TODO @fan:IllegalArgumentException 参数更好哈
|
||||||
|
// TODO @fan:需要兼容 ChatOptionsBuilder 创建出来的
|
||||||
|
throw new ChatException("Prompt 传入的不是 YiYanOptions!");
|
||||||
|
}
|
||||||
|
// 转换 YiYanOptions
|
||||||
|
return (YiYanChatOptions) options;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -1,23 +1,22 @@
|
||||||
package org.springframework.ai.models.yiyan;
|
package cn.iocoder.yudao.framework.ai.core.model.yiyan;
|
||||||
|
|
||||||
import org.springframework.ai.chat.prompt.ChatOptions;
|
import cn.iocoder.yudao.framework.ai.core.model.yiyan.api.YiYanChatCompletionRequest;
|
||||||
import org.springframework.ai.models.yiyan.api.YiYanChatCompletionRequest;
|
|
||||||
import lombok.Data;
|
import lombok.Data;
|
||||||
import lombok.experimental.Accessors;
|
import org.springframework.ai.chat.prompt.ChatOptions;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
|
// TODO @fan:字段命名,penalty_score 类似的,建议改成驼峰原则
|
||||||
|
// TODO @fan:字段的注释,可以都删除掉,让用户 https://cloud.baidu.com/doc/WENXINWORKSHOP/s/clntwmv7t 即可
|
||||||
/**
|
/**
|
||||||
* 百度 问心一言
|
* 文心一言的 {@link ChatOptions} 实现类
|
||||||
*
|
*
|
||||||
* 文档地址:https://cloud.baidu.com/doc/WENXINWORKSHOP/s/clntwmv7t
|
* 字段说明:参考 <a href="https://cloud.baidu.com/doc/WENXINWORKSHOP/s/clntwmv7t">ERNIE-4.0-8K</a>
|
||||||
*
|
*
|
||||||
* author: fansili
|
* @author fansili
|
||||||
* time: 2024/3/16 19:33
|
|
||||||
*/
|
*/
|
||||||
@Data
|
@Data
|
||||||
@Accessors(chain = true)
|
public class YiYanChatOptions implements ChatOptions {
|
||||||
public class YiYanOptions implements ChatOptions {
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 一个可触发函数的描述列表,说明:
|
* 一个可触发函数的描述列表,说明:
|
||||||
|
@ -106,37 +105,24 @@ public class YiYanOptions implements ChatOptions {
|
||||||
*/
|
*/
|
||||||
private String tool_choice;
|
private String tool_choice;
|
||||||
|
|
||||||
//
|
|
||||||
// 以下兼容 spring-ai ChatOptions 暂时没有其他地方用到
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Float getTemperature() {
|
public Float getTemperature() {
|
||||||
return this.temperature;
|
return this.temperature;
|
||||||
}
|
}
|
||||||
|
|
||||||
// @Override
|
|
||||||
// public void setTemperature(Float temperature) {
|
|
||||||
// this.temperature = temperature;
|
|
||||||
// }
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Float getTopP() {
|
public Float getTopP() {
|
||||||
return topP;
|
return topP;
|
||||||
}
|
}
|
||||||
|
|
||||||
// @Override
|
/**
|
||||||
// public void setTopP(Float topP) {
|
* 百度么有 topK
|
||||||
// this.topP = topP;
|
*
|
||||||
// }
|
* @return null
|
||||||
|
*/
|
||||||
// 百度么有 topK
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Integer getTopK() {
|
public Integer getTopK() {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
// @Override
|
|
||||||
// public void setTopK(Integer topK) {
|
|
||||||
// }
|
|
||||||
}
|
}
|
|
@ -1,8 +1,6 @@
|
||||||
package org.springframework.ai.models.yiyan.api;
|
package cn.iocoder.yudao.framework.ai.core.model.yiyan.api;
|
||||||
|
|
||||||
import org.springframework.ai.models.yiyan.YiYanChatModel;
|
import cn.iocoder.yudao.framework.ai.core.model.yiyan.exception.YiYanApiException;
|
||||||
import org.springframework.ai.models.yiyan.exception.YiYanApiException;
|
|
||||||
import lombok.Data;
|
|
||||||
import org.springframework.http.HttpStatusCode;
|
import org.springframework.http.HttpStatusCode;
|
||||||
import org.springframework.http.ResponseEntity;
|
import org.springframework.http.ResponseEntity;
|
||||||
import org.springframework.web.reactive.function.client.WebClient;
|
import org.springframework.web.reactive.function.client.WebClient;
|
||||||
|
@ -10,47 +8,55 @@ import reactor.core.publisher.Flux;
|
||||||
import reactor.core.publisher.Mono;
|
import reactor.core.publisher.Mono;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 文心一言
|
* 文心一言 API
|
||||||
* <p>
|
*
|
||||||
* author: fansili
|
* @author fansili
|
||||||
* time: 2024/3/8 21:47
|
|
||||||
*/
|
*/
|
||||||
@Data
|
|
||||||
public class YiYanApi {
|
public class YiYanApi {
|
||||||
|
|
||||||
private static final String DEFAULT_BASE_URL = "https://aip.baidubce.com";
|
private static final String DEFAULT_BASE_URL = "https://aip.baidubce.com";
|
||||||
|
|
||||||
private static final String AUTH_2_TOKEN_URI = "/oauth/2.0/token";
|
private static final String AUTH_2_TOKEN_URI = "/oauth/2.0/token";
|
||||||
|
|
||||||
public static final String DEFAULT_CHAT_MODEL = "ERNIE 4.0";
|
public static final String DEFAULT_CHAT_MODEL = YiYanChatModel.ERNIE4_0.getModel();
|
||||||
|
|
||||||
// 获取access_token流程 https://cloud.baidu.com/doc/WENXINWORKSHOP/s/Ilkkrb0i5
|
private final String appKey;
|
||||||
private String appKey;
|
private final String secretKey;
|
||||||
private String secretKey;
|
/**
|
||||||
private String token;
|
* TODO fan:这个是不是要有个刷新机制哈;如果目前不需要,可以删除掉 refreshTokenSecondTime;整体更简洁;
|
||||||
// token刷新时间(秒)
|
*/
|
||||||
|
private final String token;
|
||||||
|
/**
|
||||||
|
* token 刷新时间(秒)
|
||||||
|
*/
|
||||||
private int refreshTokenSecondTime;
|
private int refreshTokenSecondTime;
|
||||||
// 发送请求 webClient
|
/**
|
||||||
|
* 发送请求 webClient
|
||||||
|
*/
|
||||||
private final WebClient webClient;
|
private final WebClient webClient;
|
||||||
// 使用的模型
|
/**
|
||||||
private YiYanChatModel useChatModel;
|
* 使用的模型
|
||||||
|
*/
|
||||||
|
private final YiYanChatModel useChatModel;
|
||||||
|
|
||||||
public YiYanApi(String appKey, String secretKey, YiYanChatModel useChatModel, int refreshTokenSecondTime) {
|
public YiYanApi(String appKey, String secretKey, YiYanChatModel useChatModel, int refreshTokenSecondTime) {
|
||||||
this.appKey = appKey;
|
this.appKey = appKey;
|
||||||
this.secretKey = secretKey;
|
this.secretKey = secretKey;
|
||||||
this.useChatModel = useChatModel;
|
this.useChatModel = useChatModel;
|
||||||
this.refreshTokenSecondTime = refreshTokenSecondTime;
|
this.refreshTokenSecondTime = refreshTokenSecondTime;
|
||||||
|
this.webClient = WebClient.builder().baseUrl(DEFAULT_BASE_URL).build();
|
||||||
this.webClient = WebClient.builder()
|
// 获取访问令牌
|
||||||
.baseUrl(DEFAULT_BASE_URL)
|
|
||||||
.build();
|
|
||||||
|
|
||||||
token = getToken();
|
token = getToken();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获得访问令牌
|
||||||
|
*
|
||||||
|
* @see <a href="https://cloud.baidu.com/doc/WENXINWORKSHOP/s/Ilkkrb0i5">文档地址</>
|
||||||
|
* @return 访问令牌
|
||||||
|
*/
|
||||||
private String getToken() {
|
private String getToken() {
|
||||||
// 文档地址: https://cloud.baidu.com/doc/WENXINWORKSHOP/s/Ilkkrb0i5
|
ResponseEntity<YiYanAuthResponse> response = this.webClient.post()
|
||||||
ResponseEntity<YiYanAuthRes> response = this.webClient.post()
|
|
||||||
.uri(uriBuilder -> uriBuilder.path(AUTH_2_TOKEN_URI)
|
.uri(uriBuilder -> uriBuilder.path(AUTH_2_TOKEN_URI)
|
||||||
.queryParam("grant_type", "client_credentials")
|
.queryParam("grant_type", "client_credentials")
|
||||||
.queryParam("client_id", appKey)
|
.queryParam("client_id", appKey)
|
||||||
|
@ -58,17 +64,19 @@ public class YiYanApi {
|
||||||
.build()
|
.build()
|
||||||
)
|
)
|
||||||
.retrieve()
|
.retrieve()
|
||||||
.toEntity(YiYanAuthRes.class)
|
.toEntity(YiYanAuthResponse.class)
|
||||||
.block();
|
.block();
|
||||||
// 检查请求状态
|
// 检查请求状态
|
||||||
if (HttpStatusCode.valueOf(200) != response.getStatusCode()) {
|
// TODO @fan:可以使用 response.getStatusCode().is2xxSuccessful()
|
||||||
|
if (HttpStatusCode.valueOf(200) != response.getStatusCode()
|
||||||
|
|| response.getBody() == null) {
|
||||||
|
// TODO @fan:可以使用 IllegalStateException 替代;另外,最好打印下返回;方便排错;
|
||||||
throw new YiYanApiException("一言认证失败! api:https://aip.baidubce.com/oauth/2.0/token 请检查 client_id、client_secret 是否正确!");
|
throw new YiYanApiException("一言认证失败! api:https://aip.baidubce.com/oauth/2.0/token 请检查 client_id、client_secret 是否正确!");
|
||||||
}
|
}
|
||||||
YiYanAuthRes body = response.getBody();
|
return response.getBody().getAccess_token();
|
||||||
return body.getAccess_token();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public ResponseEntity<YiYanChatCompletion> chatCompletionEntity(YiYanChatCompletionRequest request) {
|
public ResponseEntity<YiYanChatCompletionResponse> chatCompletionEntity(YiYanChatCompletionRequest request) {
|
||||||
// TODO: 2024/3/10 小范 这里错误信息返回的结构不一样
|
// TODO: 2024/3/10 小范 这里错误信息返回的结构不一样
|
||||||
// {"error_code":17,"error_msg":"Open api daily request limit reached"}
|
// {"error_code":17,"error_msg":"Open api daily request limit reached"}
|
||||||
return this.webClient.post()
|
return this.webClient.post()
|
||||||
|
@ -78,11 +86,11 @@ public class YiYanApi {
|
||||||
.build())
|
.build())
|
||||||
.body(Mono.just(request), YiYanChatCompletionRequest.class)
|
.body(Mono.just(request), YiYanChatCompletionRequest.class)
|
||||||
.retrieve()
|
.retrieve()
|
||||||
.toEntity(YiYanChatCompletion.class)
|
.toEntity(YiYanChatCompletionResponse.class)
|
||||||
.block();
|
.block();
|
||||||
}
|
}
|
||||||
|
|
||||||
public Flux<YiYanChatCompletion> chatCompletionStream(YiYanChatCompletionRequest request) {
|
public Flux<YiYanChatCompletionResponse> chatCompletionStream(YiYanChatCompletionRequest request) {
|
||||||
return this.webClient.post()
|
return this.webClient.post()
|
||||||
.uri(uriBuilder
|
.uri(uriBuilder
|
||||||
-> uriBuilder.path(useChatModel.getUri())
|
-> uriBuilder.path(useChatModel.getUri())
|
||||||
|
@ -90,6 +98,7 @@ public class YiYanApi {
|
||||||
.build())
|
.build())
|
||||||
.body(Mono.just(request), YiYanChatCompletionRequest.class)
|
.body(Mono.just(request), YiYanChatCompletionRequest.class)
|
||||||
.retrieve()
|
.retrieve()
|
||||||
.bodyToFlux(YiYanChatCompletion.class);
|
.bodyToFlux(YiYanChatCompletionResponse.class);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -1,15 +1,15 @@
|
||||||
package org.springframework.ai.models.yiyan.api;
|
package cn.iocoder.yudao.framework.ai.core.model.yiyan.api;
|
||||||
|
|
||||||
import lombok.Data;
|
import lombok.Data;
|
||||||
|
|
||||||
|
// TODO @fan:字段驼峰;字段注释都可以删除,贴个链接就好;
|
||||||
/**
|
/**
|
||||||
* 一言 获取access_token
|
* 获取文心一言的 access_token 的 Response
|
||||||
*
|
*
|
||||||
* author: fansili
|
* @author fansili
|
||||||
* time: 2024/3/10 08:51
|
|
||||||
*/
|
*/
|
||||||
@Data
|
@Data
|
||||||
public class YiYanAuthRes {
|
public class YiYanAuthResponse {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 访问凭证
|
* 访问凭证
|
|
@ -1,16 +1,16 @@
|
||||||
package org.springframework.ai.models.yiyan.api;
|
package cn.iocoder.yudao.framework.ai.core.model.yiyan.api;
|
||||||
|
|
||||||
import lombok.Data;
|
import lombok.Data;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
|
// TODO @fan:字段驼峰;字段注释都可以删除,贴个链接就好;
|
||||||
/**
|
/**
|
||||||
* 一言 Completion req
|
* 文心一言 Completion Request
|
||||||
*
|
*
|
||||||
* 百度千帆文档:https://cloud.baidu.com/doc/WENXINWORKSHOP/s/jlil56u11
|
* 百度千帆文档:https://cloud.baidu.com/doc/WENXINWORKSHOP/s/jlil56u11
|
||||||
*
|
*
|
||||||
* author: fansili
|
* @author fansili
|
||||||
* time: 2024/3/9 10:34
|
|
||||||
*/
|
*/
|
||||||
@Data
|
@Data
|
||||||
public class YiYanChatCompletionRequest {
|
public class YiYanChatCompletionRequest {
|
||||||
|
@ -114,9 +114,11 @@ public class YiYanChatCompletionRequest {
|
||||||
|
|
||||||
@Data
|
@Data
|
||||||
public static class Message {
|
public static class Message {
|
||||||
|
|
||||||
private String role;
|
private String role;
|
||||||
|
|
||||||
private String content;
|
private String content;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Data
|
@Data
|
|
@ -1,16 +1,16 @@
|
||||||
package org.springframework.ai.models.yiyan.api;
|
package cn.iocoder.yudao.framework.ai.core.model.yiyan.api;
|
||||||
|
|
||||||
import lombok.Data;
|
import lombok.Data;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 聊天返回
|
* 文心一言 Completion Response
|
||||||
|
*
|
||||||
* 百度链接: https://cloud.baidu.com/doc/WENXINWORKSHOP/s/clntwmv7t
|
* 百度链接: https://cloud.baidu.com/doc/WENXINWORKSHOP/s/clntwmv7t
|
||||||
*
|
*
|
||||||
* author: fansili
|
* @author fansili
|
||||||
* time: 2024/3/9 10:34
|
|
||||||
*/
|
*/
|
||||||
@Data
|
@Data
|
||||||
public class YiYanChatCompletion {
|
public class YiYanChatCompletionResponse {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 本轮对话的id
|
* 本轮对话的id
|
||||||
|
@ -88,4 +88,5 @@ public class YiYanChatCompletion {
|
||||||
*/
|
*/
|
||||||
private int total_tokens;
|
private int total_tokens;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -1,15 +1,14 @@
|
||||||
package org.springframework.ai.models.yiyan;
|
package cn.iocoder.yudao.framework.ai.core.model.yiyan.api;
|
||||||
|
|
||||||
import lombok.AllArgsConstructor;
|
import lombok.AllArgsConstructor;
|
||||||
import lombok.Getter;
|
import lombok.Getter;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 一言模型
|
* 文心一言的模型枚举
|
||||||
*
|
*
|
||||||
* 可参考百度文档:https://cloud.baidu.com/doc/WENXINWORKSHOP/s/clntwmv7t
|
* 可参考 <a href="https://cloud.baidu.com/doc/WENXINWORKSHOP/s/clntwmv7t">百度文档</>
|
||||||
*
|
*
|
||||||
* author: fansili
|
* @author fansili
|
||||||
* time: 2024/3/9 12:01
|
|
||||||
*/
|
*/
|
||||||
@Getter
|
@Getter
|
||||||
@AllArgsConstructor
|
@AllArgsConstructor
|
||||||
|
@ -18,21 +17,24 @@ public enum YiYanChatModel {
|
||||||
ERNIE4_0("ERNIE 4.0", "/rpc/2.0/ai_custom/v1/wenxinworkshop/chat/completions_pro"),
|
ERNIE4_0("ERNIE 4.0", "/rpc/2.0/ai_custom/v1/wenxinworkshop/chat/completions_pro"),
|
||||||
ERNIE4_3_5_8K("ERNIE-3.5-8K", "/rpc/2.0/ai_custom/v1/wenxinworkshop/chat/completions"),
|
ERNIE4_3_5_8K("ERNIE-3.5-8K", "/rpc/2.0/ai_custom/v1/wenxinworkshop/chat/completions"),
|
||||||
ERNIE4_3_5_8K_0205("ERNIE-3.5-8K-0205", "/rpc/2.0/ai_custom/v1/wenxinworkshop/chat/ernie-3.5-8k-0205"),
|
ERNIE4_3_5_8K_0205("ERNIE-3.5-8K-0205", "/rpc/2.0/ai_custom/v1/wenxinworkshop/chat/ernie-3.5-8k-0205"),
|
||||||
|
|
||||||
ERNIE4_3_5_8K_1222("ERNIE-3.5-8K-1222", "/rpc/2.0/ai_custom/v1/wenxinworkshop/chat/ernie-3.5-8k-1222"),
|
ERNIE4_3_5_8K_1222("ERNIE-3.5-8K-1222", "/rpc/2.0/ai_custom/v1/wenxinworkshop/chat/ernie-3.5-8k-1222"),
|
||||||
ERNIE4_BOT_8K("ERNIE-Bot-8K", "/rpc/2.0/ai_custom/v1/wenxinworkshop/chat/ernie_bot_8k"),
|
ERNIE4_BOT_8K("ERNIE-Bot-8K", "/rpc/2.0/ai_custom/v1/wenxinworkshop/chat/ernie_bot_8k"),
|
||||||
ERNIE4_3_5_4K_0205("ERNIE-3.5-4K-0205", "/rpc/2.0/ai_custom/v1/wenxinworkshop/chat/ernie-3.5-4k-0205"),
|
ERNIE4_3_5_4K_0205("ERNIE-3.5-4K-0205", "/rpc/2.0/ai_custom/v1/wenxinworkshop/chat/ernie-3.5-4k-0205"),
|
||||||
|
|
||||||
;
|
;
|
||||||
|
|
||||||
private String model;
|
/**
|
||||||
|
* 模型名
|
||||||
private String uri;
|
*/
|
||||||
|
private final String model;
|
||||||
|
/**
|
||||||
|
* API URL
|
||||||
|
*/
|
||||||
|
private final String uri;
|
||||||
|
|
||||||
public static YiYanChatModel valueOfModel(String model) {
|
public static YiYanChatModel valueOfModel(String model) {
|
||||||
for (YiYanChatModel itemEnum : YiYanChatModel.values()) {
|
for (YiYanChatModel modelEnum : YiYanChatModel.values()) {
|
||||||
if (itemEnum.getModel().equals(model)) {
|
if (modelEnum.getModel().equals(model)) {
|
||||||
return itemEnum;
|
return modelEnum;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
throw new IllegalArgumentException("Invalid MessageType value: " + model);
|
throw new IllegalArgumentException("Invalid MessageType value: " + model);
|
|
@ -1,4 +1,4 @@
|
||||||
package org.springframework.ai.models.yiyan.exception;
|
package cn.iocoder.yudao.framework.ai.core.model.yiyan.exception;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 一言 api 调用异常
|
* 一言 api 调用异常
|
|
@ -5,7 +5,7 @@ import org.springframework.ai.chat.*;
|
||||||
import org.springframework.ai.chat.prompt.ChatOptions;
|
import org.springframework.ai.chat.prompt.ChatOptions;
|
||||||
import org.springframework.ai.chat.prompt.Prompt;
|
import org.springframework.ai.chat.prompt.Prompt;
|
||||||
import org.springframework.ai.models.tongyi.api.QianWenApi;
|
import org.springframework.ai.models.tongyi.api.QianWenApi;
|
||||||
import org.springframework.ai.models.yiyan.exception.YiYanApiException;
|
import cn.iocoder.yudao.framework.ai.core.model.yiyan.exception.YiYanApiException;
|
||||||
import com.alibaba.dashscope.aigc.generation.GenerationResult;
|
import com.alibaba.dashscope.aigc.generation.GenerationResult;
|
||||||
import com.alibaba.dashscope.aigc.generation.models.QwenParam;
|
import com.alibaba.dashscope.aigc.generation.models.QwenParam;
|
||||||
import com.alibaba.dashscope.common.Message;
|
import com.alibaba.dashscope.common.Message;
|
||||||
|
|
|
@ -1,158 +0,0 @@
|
||||||
package org.springframework.ai.models.yiyan;
|
|
||||||
|
|
||||||
import cn.hutool.core.bean.BeanUtil;
|
|
||||||
import cn.iocoder.yudao.framework.ai.core.exception.ChatException;
|
|
||||||
import org.springframework.ai.chat.*;
|
|
||||||
import org.springframework.ai.chat.messages.Message;
|
|
||||||
import org.springframework.ai.chat.messages.MessageType;
|
|
||||||
import org.springframework.ai.chat.prompt.ChatOptions;
|
|
||||||
import org.springframework.ai.chat.prompt.Prompt;
|
|
||||||
import org.springframework.ai.models.yiyan.api.YiYanApi;
|
|
||||||
import org.springframework.ai.models.yiyan.api.YiYanChatCompletion;
|
|
||||||
import org.springframework.ai.models.yiyan.api.YiYanChatCompletionRequest;
|
|
||||||
import org.springframework.ai.models.yiyan.exception.YiYanApiException;
|
|
||||||
import lombok.extern.slf4j.Slf4j;
|
|
||||||
import org.jetbrains.annotations.NotNull;
|
|
||||||
import org.springframework.http.ResponseEntity;
|
|
||||||
import org.springframework.retry.RetryCallback;
|
|
||||||
import org.springframework.retry.RetryContext;
|
|
||||||
import org.springframework.retry.RetryListener;
|
|
||||||
import org.springframework.retry.support.RetryTemplate;
|
|
||||||
import reactor.core.publisher.Flux;
|
|
||||||
|
|
||||||
import java.time.Duration;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.stream.Collectors;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 文心一言
|
|
||||||
* <p>
|
|
||||||
* author: fansili
|
|
||||||
* time: 2024/3/8 19:11
|
|
||||||
*/
|
|
||||||
@Slf4j
|
|
||||||
public class YiYanChatClient implements ChatClient, StreamingChatClient {
|
|
||||||
|
|
||||||
private YiYanApi yiYanApi;
|
|
||||||
|
|
||||||
private YiYanOptions yiYanOptions;
|
|
||||||
|
|
||||||
public YiYanChatClient(YiYanApi yiYanApi) {
|
|
||||||
this.yiYanApi = yiYanApi;
|
|
||||||
}
|
|
||||||
|
|
||||||
public YiYanChatClient(YiYanApi yiYanApi, YiYanOptions yiYanOptions) {
|
|
||||||
this.yiYanApi = yiYanApi;
|
|
||||||
this.yiYanOptions = yiYanOptions;
|
|
||||||
}
|
|
||||||
|
|
||||||
public final RetryTemplate retryTemplate = RetryTemplate.builder()
|
|
||||||
// 最大重试次数 10
|
|
||||||
.maxAttempts(10)
|
|
||||||
.retryOn(YiYanApiException.class)
|
|
||||||
// 最大重试5次,第一次间隔3000ms,第二次3000ms * 2,第三次3000ms * 3,以此类推,最大间隔3 * 60000ms
|
|
||||||
.exponentialBackoff(Duration.ofMillis(3000), 2, Duration.ofMillis(3 * 60000))
|
|
||||||
.withListener(new RetryListener() {
|
|
||||||
@Override
|
|
||||||
public <T extends Object, E extends Throwable> void onError(RetryContext context,
|
|
||||||
RetryCallback<T, E> callback, Throwable throwable) {
|
|
||||||
log.warn("重试异常:" + context.getRetryCount(), throwable);
|
|
||||||
}
|
|
||||||
|
|
||||||
;
|
|
||||||
})
|
|
||||||
.build();
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String call(String message) {
|
|
||||||
return ChatClient.super.call(message);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public ChatResponse call(Prompt prompt) {
|
|
||||||
return this.retryTemplate.execute(ctx -> {
|
|
||||||
// ctx 会有重试的信息
|
|
||||||
// 创建 request 请求,stream模式需要供应商支持
|
|
||||||
YiYanChatCompletionRequest request = this.createRequest(prompt, false);
|
|
||||||
// 调用 callWithFunctionSupport 发送请求
|
|
||||||
ResponseEntity<YiYanChatCompletion> response = yiYanApi.chatCompletionEntity(request);
|
|
||||||
// 获取结果封装 ChatResponse
|
|
||||||
YiYanChatCompletion chatCompletion = response.getBody();
|
|
||||||
return new ChatResponse(List.of(new Generation(chatCompletion.getResult())));
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Flux<ChatResponse> stream(Prompt prompt) {
|
|
||||||
// ctx 会有重试的信息
|
|
||||||
// 创建 request 请求,stream模式需要供应商支持
|
|
||||||
YiYanChatCompletionRequest request = this.createRequest(prompt, true);
|
|
||||||
// 调用 callWithFunctionSupport 发送请求
|
|
||||||
Flux<YiYanChatCompletion> response = this.yiYanApi.chatCompletionStream(request);
|
|
||||||
response.doOnComplete(new Runnable() {
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
String a = ";";
|
|
||||||
}
|
|
||||||
});
|
|
||||||
return response.map(res -> {
|
|
||||||
// TODO @fan:这里缺少了 usage 的封装
|
|
||||||
return new ChatResponse(List.of(new Generation(res.getResult())));
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
private YiYanChatCompletionRequest createRequest(Prompt prompt, boolean stream) {
|
|
||||||
// 获取配置
|
|
||||||
YiYanOptions useOptions = getYiYanOptions(prompt);
|
|
||||||
// 创建 request
|
|
||||||
|
|
||||||
// tip: 百度的 system 不在 message 里面
|
|
||||||
// tip:百度的 message 只有 user 和 assistant
|
|
||||||
// https://cloud.baidu.com/doc/WENXINWORKSHOP/s/clntwmv7t
|
|
||||||
|
|
||||||
// 获取 user 和 assistant
|
|
||||||
List<YiYanChatCompletionRequest.Message> messageList = prompt.getInstructions().stream()
|
|
||||||
// 过滤 system
|
|
||||||
.filter(msg -> MessageType.SYSTEM != msg.getMessageType())
|
|
||||||
.map(msg -> new YiYanChatCompletionRequest.Message()
|
|
||||||
.setRole(msg.getMessageType().getValue())
|
|
||||||
.setContent(msg.getContent())
|
|
||||||
).toList();
|
|
||||||
// 获取 system
|
|
||||||
String systemPrompt = prompt.getInstructions().stream()
|
|
||||||
.filter(msg -> MessageType.SYSTEM == msg.getMessageType())
|
|
||||||
.map(Message::getContent)
|
|
||||||
.collect(Collectors.joining());
|
|
||||||
|
|
||||||
YiYanChatCompletionRequest request = new YiYanChatCompletionRequest(messageList);
|
|
||||||
// 复制 qianWenOptions 属性取 request(这里 options 属性和 request 基本保持一致)
|
|
||||||
// top: 由于遵循 spring-ai规范,支持在构建client的时候传入默认的 chatOptions
|
|
||||||
BeanUtil.copyProperties(useOptions, request);
|
|
||||||
request.setTop_p(useOptions.getTopP());
|
|
||||||
request.setMax_output_tokens(useOptions.getMaxOutputTokens());
|
|
||||||
request.setTemperature(useOptions.getTemperature());
|
|
||||||
request.setSystem(systemPrompt);
|
|
||||||
// 设置 stream
|
|
||||||
request.setStream(stream);
|
|
||||||
return request;
|
|
||||||
}
|
|
||||||
|
|
||||||
private @NotNull YiYanOptions getYiYanOptions(Prompt prompt) {
|
|
||||||
// 两个都为null 则没有配置文件
|
|
||||||
if (yiYanOptions == null && prompt.getOptions() == null) {
|
|
||||||
throw new ChatException("ChatOptions 未配置参数!");
|
|
||||||
}
|
|
||||||
// 优先使用 Prompt 里面的 ChatOptions
|
|
||||||
ChatOptions options = yiYanOptions;
|
|
||||||
if (prompt.getOptions() != null) {
|
|
||||||
options = (ChatOptions) prompt.getOptions();
|
|
||||||
}
|
|
||||||
// Prompt 里面是一个 ChatOptions,用户可以随意传入,这里做一下判断
|
|
||||||
if (!(options instanceof YiYanOptions)) {
|
|
||||||
throw new ChatException("Prompt 传入的不是 YiYanOptions!");
|
|
||||||
}
|
|
||||||
// 转换 YiYanOptions
|
|
||||||
YiYanOptions useOptions = (YiYanOptions) options;
|
|
||||||
return useOptions;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,8 +0,0 @@
|
||||||
package org.springframework.ai.models.yiyan.api;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* author: fansili
|
|
||||||
* time: 2024/3/9 10:37
|
|
||||||
*/
|
|
||||||
public class YiYanChatCompletionMessage {
|
|
||||||
}
|
|
|
@ -5,10 +5,10 @@ import org.springframework.ai.chat.messages.Message;
|
||||||
import org.springframework.ai.chat.messages.SystemMessage;
|
import org.springframework.ai.chat.messages.SystemMessage;
|
||||||
import org.springframework.ai.chat.messages.UserMessage;
|
import org.springframework.ai.chat.messages.UserMessage;
|
||||||
import org.springframework.ai.chat.prompt.Prompt;
|
import org.springframework.ai.chat.prompt.Prompt;
|
||||||
import org.springframework.ai.models.yiyan.YiYanChatClient;
|
import cn.iocoder.yudao.framework.ai.core.model.yiyan.YiYanChatClient;
|
||||||
import org.springframework.ai.models.yiyan.YiYanChatModel;
|
import cn.iocoder.yudao.framework.ai.core.model.yiyan.api.YiYanChatModel;
|
||||||
import org.springframework.ai.models.yiyan.YiYanOptions;
|
import cn.iocoder.yudao.framework.ai.core.model.yiyan.YiYanChatOptions;
|
||||||
import org.springframework.ai.models.yiyan.api.YiYanApi;
|
import cn.iocoder.yudao.framework.ai.core.model.yiyan.api.YiYanApi;
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
import reactor.core.publisher.Flux;
|
import reactor.core.publisher.Flux;
|
||||||
|
@ -35,7 +35,7 @@ public class YiYanChatTests {
|
||||||
YiYanChatModel.ERNIE4_3_5_8K,
|
YiYanChatModel.ERNIE4_3_5_8K,
|
||||||
86400
|
86400
|
||||||
);
|
);
|
||||||
YiYanOptions yiYanOptions = new YiYanOptions();
|
YiYanChatOptions yiYanOptions = new YiYanChatOptions();
|
||||||
yiYanOptions.setMaxOutputTokens(2048);
|
yiYanOptions.setMaxOutputTokens(2048);
|
||||||
yiYanOptions.setTopP(0.6f);
|
yiYanOptions.setTopP(0.6f);
|
||||||
yiYanOptions.setTemperature(0.85f);
|
yiYanOptions.setTemperature(0.85f);
|
||||||
|
|
|
@ -230,17 +230,6 @@ yudao:
|
||||||
appKey: cb6415c19d6162cda07b47316fcb0416
|
appKey: cb6415c19d6162cda07b47316fcb0416
|
||||||
secretKey: Y2JiYTIxZjA3MDMxMjNjZjQzYzVmNzdh
|
secretKey: Y2JiYTIxZjA3MDMxMjNjZjQzYzVmNzdh
|
||||||
model: XING_HUO_3_5
|
model: XING_HUO_3_5
|
||||||
yiyan:
|
|
||||||
enable: true
|
|
||||||
aiPlatform: YI_YAN
|
|
||||||
max-tokens: 1500
|
|
||||||
temperature: 0.85
|
|
||||||
topP: 0.8
|
|
||||||
topK: 0
|
|
||||||
appKey: x0cuLZ7XsaTCU08vuJWO87Lg
|
|
||||||
secretKey: R9mYF9dl9KASgi5RUq0FQt3wRisSnOcK
|
|
||||||
refreshTokenSecondTime: 86400
|
|
||||||
model: ERNIE4_3_5_8K
|
|
||||||
openAiImage:
|
openAiImage:
|
||||||
enable: true
|
enable: true
|
||||||
api-key: ${OPEN_AI_KEY}
|
api-key: ${OPEN_AI_KEY}
|
||||||
|
|
|
@ -150,6 +150,19 @@ spring.ai:
|
||||||
chat:
|
chat:
|
||||||
model: llama3
|
model: llama3
|
||||||
|
|
||||||
|
yudao.ai:
|
||||||
|
yiyan:
|
||||||
|
enable: true
|
||||||
|
aiPlatform: YIYAN # TODO @fan:建议每个都独立配置属性类
|
||||||
|
max-tokens: 1500
|
||||||
|
temperature: 0.85
|
||||||
|
topP: 0.8
|
||||||
|
topK: 0
|
||||||
|
appKey: x0cuLZ7XsaTCU08vuJWO87Lg
|
||||||
|
secretKey: R9mYF9dl9KASgi5RUq0FQt3wRisSnOcK
|
||||||
|
refreshTokenSecondTime: 86400
|
||||||
|
model: ERNIE4_3_5_8K
|
||||||
|
|
||||||
--- #################### 芋道相关配置 ####################
|
--- #################### 芋道相关配置 ####################
|
||||||
|
|
||||||
yudao:
|
yudao:
|
||||||
|
|
Loading…
Reference in New Issue