ai-gateway/providers/baidu/chat.go

204 lines
5.5 KiB
Go
Raw Permalink Normal View History

package baidu
2023-11-28 10:32:26 +00:00
import (
"bufio"
"encoding/json"
"io"
"net/http"
"one-api/common"
"one-api/providers/base"
2023-11-28 10:32:26 +00:00
"one-api/types"
"strings"
)
func (baiduResponse *BaiduChatResponse) ResponseHandler(resp *http.Response) (OpenAIResponse any, errWithCode *types.OpenAIErrorWithStatusCode) {
2023-11-28 10:32:26 +00:00
if baiduResponse.ErrorMsg != "" {
return nil, &types.OpenAIErrorWithStatusCode{
OpenAIError: types.OpenAIError{
Message: baiduResponse.ErrorMsg,
Type: "baidu_error",
Param: "",
Code: baiduResponse.ErrorCode,
},
StatusCode: resp.StatusCode,
}
}
choice := types.ChatCompletionChoice{
Index: 0,
Message: types.ChatCompletionMessage{
Role: "assistant",
Content: baiduResponse.Result,
},
FinishReason: "stop",
}
OpenAIResponse = types.ChatCompletionResponse{
2023-11-28 10:32:26 +00:00
ID: baiduResponse.Id,
Object: "chat.completion",
Created: baiduResponse.Created,
Choices: []types.ChatCompletionChoice{choice},
Usage: baiduResponse.Usage,
}
return
2023-11-28 10:32:26 +00:00
}
func (p *BaiduProvider) getChatRequestBody(request *types.ChatCompletionRequest) *BaiduChatRequest {
messages := make([]BaiduMessage, 0, len(request.Messages))
for _, message := range request.Messages {
if message.Role == "system" {
messages = append(messages, BaiduMessage{
Role: "user",
Content: message.StringContent(),
})
messages = append(messages, BaiduMessage{
Role: "assistant",
Content: "Okay",
})
} else {
messages = append(messages, BaiduMessage{
Role: message.Role,
Content: message.StringContent(),
})
}
}
return &BaiduChatRequest{
Messages: messages,
Stream: request.Stream,
}
}
func (p *BaiduProvider) ChatAction(request *types.ChatCompletionRequest, isModelMapped bool, promptTokens int) (usage *types.Usage, errWithCode *types.OpenAIErrorWithStatusCode) {
2023-11-28 10:32:26 +00:00
requestBody := p.getChatRequestBody(request)
fullRequestURL := p.GetFullRequestURL(p.ChatCompletions, request.Model)
if fullRequestURL == "" {
2023-12-01 19:28:18 +00:00
return nil, common.ErrorWrapper(nil, "invalid_baidu_config", http.StatusInternalServerError)
2023-11-28 10:32:26 +00:00
}
headers := p.GetRequestHeaders()
if request.Stream {
headers["Accept"] = "text/event-stream"
}
client := common.NewClient()
req, err := client.NewRequest(p.Context.Request.Method, fullRequestURL, common.WithBody(requestBody), common.WithHeader(headers))
if err != nil {
2023-12-01 19:28:18 +00:00
return nil, common.ErrorWrapper(err, "new_request_failed", http.StatusInternalServerError)
2023-11-28 10:32:26 +00:00
}
if request.Stream {
usage, errWithCode = p.sendStreamRequest(req, request.Model)
if errWithCode != nil {
2023-11-28 10:32:26 +00:00
return
}
} else {
baiduChatRequest := &BaiduChatResponse{
Model: request.Model,
}
errWithCode = p.SendRequest(req, baiduChatRequest, false)
if errWithCode != nil {
2023-11-28 10:32:26 +00:00
return
}
usage = baiduChatRequest.Usage
}
return
}
func (p *BaiduProvider) streamResponseBaidu2OpenAI(baiduResponse *BaiduChatStreamResponse) *types.ChatCompletionStreamResponse {
var choice types.ChatCompletionStreamChoice
choice.Delta.Content = baiduResponse.Result
if baiduResponse.IsEnd {
choice.FinishReason = &base.StopFinishReason
2023-11-28 10:32:26 +00:00
}
response := types.ChatCompletionStreamResponse{
ID: baiduResponse.Id,
Object: "chat.completion.chunk",
Created: baiduResponse.Created,
Model: baiduResponse.Model,
2023-11-28 10:32:26 +00:00
Choices: []types.ChatCompletionStreamChoice{choice},
}
return &response
}
func (p *BaiduProvider) sendStreamRequest(req *http.Request, model string) (usage *types.Usage, errWithCode *types.OpenAIErrorWithStatusCode) {
2023-12-01 02:54:07 +00:00
defer req.Body.Close()
2023-11-28 10:32:26 +00:00
usage = &types.Usage{}
// 发送请求
2023-12-26 10:42:39 +00:00
client := common.GetHttpClient(p.Channel.Proxy)
resp, err := client.Do(req)
2023-11-28 10:32:26 +00:00
if err != nil {
2023-12-01 19:28:18 +00:00
return nil, common.ErrorWrapper(err, "http_request_failed", http.StatusInternalServerError)
2023-11-28 10:32:26 +00:00
}
2023-12-26 10:42:39 +00:00
common.PutHttpClient(client)
2023-11-28 10:32:26 +00:00
if common.IsFailureStatusCode(resp) {
return nil, common.HandleErrorResp(resp)
2023-11-28 10:32:26 +00:00
}
defer resp.Body.Close()
scanner := bufio.NewScanner(resp.Body)
scanner.Split(func(data []byte, atEOF bool) (advance int, token []byte, err error) {
if atEOF && len(data) == 0 {
return 0, nil, nil
}
if i := strings.Index(string(data), "\n"); i >= 0 {
return i + 1, data[0:i], nil
}
if atEOF {
return len(data), data, nil
}
return 0, nil, nil
})
dataChan := make(chan string)
stopChan := make(chan bool)
go func() {
for scanner.Scan() {
data := scanner.Text()
if len(data) < 6 { // ignore blank line or wrong format
continue
}
data = data[6:]
dataChan <- data
}
stopChan <- true
}()
common.SetEventStreamHeaders(p.Context)
2023-11-28 10:32:26 +00:00
p.Context.Stream(func(w io.Writer) bool {
select {
case data := <-dataChan:
var baiduResponse BaiduChatStreamResponse
err := json.Unmarshal([]byte(data), &baiduResponse)
if err != nil {
common.SysError("error unmarshalling stream response: " + err.Error())
return true
}
if baiduResponse.Usage.TotalTokens != 0 {
usage.TotalTokens = baiduResponse.Usage.TotalTokens
usage.PromptTokens = baiduResponse.Usage.PromptTokens
usage.CompletionTokens = baiduResponse.Usage.TotalTokens - baiduResponse.Usage.PromptTokens
}
baiduResponse.Model = model
2023-11-28 10:32:26 +00:00
response := p.streamResponseBaidu2OpenAI(&baiduResponse)
jsonResponse, err := json.Marshal(response)
if err != nil {
common.SysError("error marshalling stream response: " + err.Error())
return true
}
p.Context.Render(-1, common.CustomEvent{Data: "data: " + string(jsonResponse)})
return true
case <-stopChan:
p.Context.Render(-1, common.CustomEvent{Data: "data: [DONE]"})
return false
}
})
return usage, nil
2023-11-28 10:32:26 +00:00
}