自荐一个支持OpenAi所有可用API的chatgpt-spring-boot-starter
概览
- 支持设置多个 API Key,并且支持对其设置权重,以及是否启用
- 支持设置请求代理
- 支持自定义请求 API(如果对 OpenAi 的 API 做了中转/代理)
- 支持OpenAi所有可以使用 API Key 访问的 API
- 支持流式响应,即所谓的”打字机”模式
- 请求参数自动校验
- 支持token计算
支持的功能
✅ 模型查询(Model)
✅ 流式、非流式对话聊天(Stream Chat/completion)
✅ 根据提示生成文本(Edit)
✅ 自然语言转换为向量表示
✅ 音频、视频语音转文本(Create transcription)
✅ 文本翻译(Create translation)
✅ 文件的查询、上传、删除(File - List/Upload/Delete/Retrieve)
✅ 预训练模型的微调、查询、放弃、过程(事件)(Fine-tunes - Create/List/Retrieve/Cancel/Events)
✅ 内容审核(Moderation)
✅ 用户余额、使用量查询(Billing/Usage)
✅ 用户信息查询(User)
✅ 根据提示创建、编辑图像、根据图像生成多版本图像(Image - Create/Create edit/Create variation)
项目地址
- Github:https://github.com/lzhpo/chatgpt-spring-boot-starter
- Gitee:https://gitee.com/lzhpo/chatgpt-spring-boot-starter
导入依赖
<dependency>
<groupId>com.lzhpo</groupId>
<artifactId>chatgpt-spring-boot-starter</artifactId>
<version>${version}</version>
</dependency>
配置示例
1. 支持配置多个 API Key(权重、是否启用)
可以对当前 api key 设置权重,以及是否需要启用此 api key
openai:
keys:
- key: "sk-xxx1"
weight: 1.0
enabled: true
- key: "sk-xxx2"
weight: 2.0
enabled: false
- key: "sk-xxx3"
weight: 3.0
enabled: false
2. 支持配置代理
openai:
proxy:
host: "127.0.0.1"
port: 7890
type: http
header-name: "Proxy-Authorization"
username: admin
password: 123456
3. 支持配置超时时间
openai:
connect-timeout: 1m
read-timeout: 1m
write-timeout: 1m
4. 支持自定义请求API
如果没有配置代理,也没有定制完整请求地址的需求,那么无需配置
openai.domain
以及openai.urls
,会自动使用默认的。
4.1 方式1 - 只配置了代理
如果只是配置了国内中转代理,那么只需要配置openai.domain
为代理地址即可,默认值为https://api.openai.com
openai:
domain: "https://api.openai.com"
4.2 方式2 - 定制完整的请求地址
如果有定制完整请求地址的需求,可以按照如下配置,优先级比openai.domain
更高,但需要的是完整的请求地址。
openai:
urls:
moderations: "https://api.openai.com/v1/moderations"
completions: "https://api.openai.com/v1/completions"
edits: "https://api.openai.com/v1/edits"
chat-completions: "https://api.openai.com/v1/chat/completions"
list-models: "https://api.openai.com/v1/models"
retrieve-model: "https://api.openai.com/v1/models/{model}"
embeddings: "https://api.openai.com/v1/embeddings"
list-files: "https://api.openai.com/v1/files"
upload-file: "https://api.openai.com/v1/files"
delete-file: "https://api.openai.com/v1/files/{file_id}"
retrieve-file: "https://api.openai.com/v1/files/{file_id}"
retrieve-file-content: "https://api.openai.com/v1/files/{file_id}/content"
create_fine_tune: "https://api.openai.com/v1/fine-tunes"
list_fine_tune: "https://api.openai.com/v1/fine-tunes"
retrieve_fine_tune: "https://api.openai.com/v1/fine-tunes/{fine_tune_id}"
cancel_fine_tune: "https://api.openai.com/v1/fine-tunes/{fine_tune_id}/cancel"
list_fine_tune_events: "https://api.openai.com/v1/fine-tunes/{fine_tune_id}/events"
delete_fine_tune_events: "https://api.openai.com/v1/models/{model}"
create-transcription: "https://api.openai.com/v1/audio/transcriptions"
create-translation: "https://api.openai.com/v1/audio/translations"
create_image: "https://api.openai.com/v1/images/generations"
create_image_edit: "https://api.openai.com/v1/images/edits"
create_image_variation: "https://api.openai.com/v1/images/variations"
billing-credit-grants: "https://api.openai.com/dashboard/billing/credit_grants"
users: "https://api.openai.com/v1/organizations/{organizationId}/users"
billing-subscription: "https://api.openai.com/v1/dashboard/billing/subscription"
billing-usage: "https://api.openai.com/v1/dashboard/billing/usage?start_date={start_date}&end_date={end_date}"
代码示例
1. 流式输出(“打字机”)
1.1 SSE方式示例
这里快速了解一下SSE(Server-Sent Events)
SSE和WebSocket都是用于实现服务器和浏览器之间实时通信的技术。
WebSocket是全双工通信协议,适用于双向通信的实时场景,而SSE是单向通信协议,适用于服务器向客户端推送消息的实时场景。
简单网页效果示例(“打字机”效果):
后端代码简单示例:
@RestController
@RequestMapping("/")
@RequiredArgsConstructor
public class ChatController {
private final OpenAiClient openAiClient;
@GetMapping("/chat/sse")
public SseEmitter sseStreamChat(@RequestParam String message) {
SseEmitter sseEmitter = new SseEmitter();
ChatCompletionRequest request = ChatCompletionRequest.create(message);
openAiClient.streamChatCompletions(request, new SseEventSourceListener(sseEmitter));
return sseEmitter;
}
}
SseEventSourceListener
是基于okhttp3.sse.EventSourceListener
实现的,可以接收text/event-stream
类型的流式数据。
前端代码简单示例:
// message为需要发送的消息
const eventSource = new EventSource(`http://127.0.0.1:6060/chat/sse?message=${message}`);
// 收到消息处理
eventSource.onmessage = function(event) {
// 略...
}
由于SSE协议只支持GET方法,不支持POST方法。
如果要支持POST方法可以参考:https://github.com/Azure/fetch-event-source
详细代码见仓库目录下:
templates/chat.html
templates/sse-stream-chat.html
com.lzhpo.chatgpt.OpenAiTestController
1.2 WebSocket方式示例
效果和SSE方式一样,即“打字机”效果。
声明WebSocket端点:
@Slf4j
@Component
@ServerEndpoint("/chat/websocket")
public class OpenAiWebSocketTest {
@OnOpen
public void onOpen(Session session) {
log.info("sessionId={} joined.", session.getId());
}
@OnMessage
public void onMessage(Session session, String message) {
log.info("Received sessionId={} message={}", session.getId(), message);
ChatCompletionRequest request = ChatCompletionRequest.create(message);
WebSocketEventSourceListener listener = new WebSocketEventSourceListener(session);
SpringUtil.getBean(OpenAiClient.class).streamChatCompletions(request, listener);
}
@OnClose
public void onClose(Session session) {
log.info("Closed sessionId={} connection.", session.getId());
}
@OnError
public void onError(Session session, Throwable e) {
log.error("sessionId={} error: {}", session.getId(), e.getMessage(), e);
}
}
开启WebSocket端点:
@Bean
public ServerEndpointExporter serverEndpointExporter() {
return new ServerEndpointExporter();
}
前端代码主要逻辑如下:
const websocket = new WebSocket("ws://127.0.0.1:6060/chat/websocket");
// 发送消息(message为需要发送的消息)
websocket.send(message);
// 收到消息
websocket.onmessage = function(event) {
// 略...
}
详细代码见仓库目录下的templates/websocket-stream-chat.html
2. 自定义请求拦截器
实现okhttp3.Interceptor
接口,并将其声明为bean即可。
例如:
@Slf4j
public class OpenAiLoggingInterceptor implements Interceptor {
@NotNull
@Override
public Response intercept(@NotNull Chain chain) throws IOException {
Request request = chain.request();
log.info("Request url: {} {}", request.method(), request.url());
log.info("Request header: {}", request.headers());
Response response = chain.proceed(request);
log.info("Response code: {}", response.code());
log.info("Response body: {}", response.body().string());
return response;
}
}
3. 全部API测试示例
见:com.lzhpo.chatgpt.OpenAiClientTest
@SpringBootTest
@TestMethodOrder(MethodOrderer.OrderAnnotation.class)
class OpenAiClientTest {
@Autowired
private OpenAiClient openAiService;
@Test
@Order(1)
void moderations() {
ModerationRequest request = new ModerationRequest();
request.setInput(ListUtil.of("I want to kill them."));
ModerationResponse response = openAiService.moderations(request);
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(2)
void completions() {
CompletionRequest request = new CompletionRequest();
request.setModel("text-davinci-003");
request.setPrompt("Say this is a test");
request.setMaxTokens(7);
request.setTemperature(0);
CompletionResponse response = openAiService.completions(request);
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(3)
void streamCompletions() throws InterruptedException {
CompletionRequest request = new CompletionRequest();
request.setStream(true);
request.setModel("text-davinci-003");
request.setPrompt("Say this is a test");
request.setMaxTokens(7);
request.setTemperature(0);
final CountDownLatch countDownLatch = new CountDownLatch(1);
CountDownLatchEventSourceListener eventSourceListener = new CountDownLatchEventSourceListener(countDownLatch);
assertDoesNotThrow(() -> openAiService.streamCompletions(request, eventSourceListener));
countDownLatch.await();
}
@Test
@Order(4)
void edits() {
EditRequest request = new EditRequest();
request.setModel("text-davinci-edit-001");
request.setInput("What day of the wek is it?");
request.setInstruction("Fix the spelling mistakes");
EditResponse response = openAiService.edits(request);
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(5)
void chatCompletions() {
List<ChatCompletionMessage> messages = new ArrayList<>();
ChatCompletionMessage message = new ChatCompletionMessage();
message.setRole("user");
message.setContent("Hello");
messages.add(message);
ChatCompletionRequest request = new ChatCompletionRequest();
request.setModel("gpt-3.5-turbo");
request.setMessages(messages);
ChatCompletionResponse response = openAiService.chatCompletions(request);
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(6)
void streamChatCompletions() throws InterruptedException {
List<ChatCompletionMessage> messages = new ArrayList<>();
ChatCompletionMessage message = new ChatCompletionMessage();
message.setRole("user");
message.setContent("Hello");
messages.add(message);
ChatCompletionRequest request = new ChatCompletionRequest();
request.setStream(true);
request.setModel("gpt-3.5-turbo");
request.setMessages(messages);
final CountDownLatch countDownLatch = new CountDownLatch(1);
CountDownLatchEventSourceListener eventSourceListener = new CountDownLatchEventSourceListener(countDownLatch);
assertDoesNotThrow(() -> openAiService.streamChatCompletions(request, eventSourceListener));
countDownLatch.await();
}
@Test
@Order(7)
void models() {
ListModelsResponse response = openAiService.models();
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(8)
void retrieveModel() {
RetrieveModelResponse response = openAiService.retrieveModel("babbage");
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(9)
void embeddings() {
EmbeddingRequest request = new EmbeddingRequest();
request.setModel("text-embedding-ada-002");
request.setInput(ListUtil.of("The food was delicious and the waiter..."));
EmbeddingResponse response = openAiService.embeddings(request);
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(10)
void uploadFile() {
final String path = "C:\\Users\\lzhpo\\Desktop\\xxx.txt";
FileSystemResource fileResource = new FileSystemResource(path);
UploadFileResponse response = openAiService.uploadFile(fileResource, "fine-tune");
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(11)
void listFiles() {
ListFileResponse response = openAiService.listFiles();
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(12)
void retrieveFile() {
final String fileId = "file-xxx";
RetrieveFileResponse response = openAiService.retrieveFile(fileId);
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(13)
void deleteFile() {
final String fileId = "file-xxx";
DeleteFileResponse response = openAiService.deleteFile(fileId);
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(14)
void createFineTune() {
CreateFineTuneRequest request = new CreateFineTuneRequest();
request.setTrainingFile("file-xxx");
CreateFineTuneResponse response = openAiService.createFineTune(request);
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(15)
void listFineTunes() {
ListFineTuneResponse response = openAiService.listFineTunes();
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(16)
void retrieveFineTunes() {
final String fineTuneId = "ft-xxx";
RetrieveFineTuneResponse response = openAiService.retrieveFineTunes(fineTuneId);
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(17)
void cancelFineTune() {
final String fineTuneId = "ft-xxx";
CancelFineTuneResponse response = openAiService.cancelFineTune(fineTuneId);
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(18)
void listFineTuneEvents() {
final String fineTuneId = "ft-xxx";
ListFineTuneEventResponse response = openAiService.listFineTuneEvents(fineTuneId);
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(19)
void deleteFineTuneModel() {
final String modelId = "curie:ft-xxx";
DeleteFineTuneModelResponse response = openAiService.deleteFineTuneModel(modelId);
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(20)
void createTranscription() {
final String path = "C:\\Users\\lzhpo\\Downloads\\xxx.mp3";
FileSystemResource fileResource = new FileSystemResource(path);
CreateAudioRequest request = new CreateAudioRequest();
request.setModel("whisper-1");
CreateAudioResponse response = openAiService.createTranscription(fileResource, request);
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(21)
void createTranslation() {
final String path = "C:\\Users\\lzhpo\\Downloads\\xxx.mp3";
FileSystemResource fileResource = new FileSystemResource(path);
CreateAudioRequest request = new CreateAudioRequest();
request.setModel("whisper-1");
CreateAudioResponse response = openAiService.createTranslation(fileResource, request);
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(22)
void createImage() {
CreateImageRequest request = new CreateImageRequest();
request.setPrompt("A cute baby sea otter.");
request.setN(2);
request.setSize(CreateImageSize.X_512_512);
request.setResponseFormat(CreateImageResponseFormat.URL);
CreateImageResponse response = openAiService.createImage(request);
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(23)
void createImageEdit() {
final String imagePath = "C:\\Users\\lzhpo\\Downloads\\img-xxx.png";
FileSystemResource imageResource = new FileSystemResource(imagePath);
final String markPath = "C:\\Users\\lzhpo\\Downloads\\img-xxx.png";
FileSystemResource markResource = new FileSystemResource(markPath);
CreateImageRequest request = new CreateImageRequest();
request.setPrompt("A cute baby sea otter.");
request.setN(2);
request.setSize(CreateImageSize.X_512_512);
request.setResponseFormat(CreateImageResponseFormat.URL);
CreateImageResponse response = openAiService.createImageEdit(imageResource, markResource, request);
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(24)
void createImageVariation() {
final String imagePath = "C:\\Users\\lzhpo\\Downloads\\img-xxx.png";
FileSystemResource imageResource = new FileSystemResource(imagePath);
CreateImageVariationRequest request = new CreateImageVariationRequest();
request.setN(2);
request.setSize(CreateImageSize.X_512_512);
request.setResponseFormat(CreateImageResponseFormat.URL);
CreateImageResponse response = openAiService.createImageVariation(imageResource, request);
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(25)
void billingCreditGrants() {
CreditGrantsResponse response = openAiService.billingCreditGrants();
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(26)
void users() {
UserResponse response = openAiService.users("org-xxx");
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(27)
void billingSubscription() {
SubscriptionResponse response = openAiService.billingSubscription();
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
@Test
@Order(28)
void billingUsage() {
Date nowDate = new Date();
String startDate = DateUtil.format(DateUtil.offsetDay(nowDate, -100), DatePattern.NORM_DATE_PATTERN);
String endDate = DateUtil.format(nowDate, DatePattern.NORM_DATE_PATTERN);
UsageResponse response = openAiService.billingUsage(startDate, endDate);
assertNotNull(response);
Console.log(JsonUtils.toJsonPrettyString(response));
}
}
- 本文标签: ChatGPT SpringBoot
- 本文链接: http://www.lzhpo.com/article/1647174093341523970
- 版权声明: 本文由lzhpo原创发布,转载请遵循《署名-非商业性使用-相同方式共享 4.0 国际 (CC BY-NC-SA 4.0)》许可协议授权