mirror of
https://github.com/coaidev/coai.git
synced 2025-05-19 21:10:18 +09:00
This commit is contained in:
parent
ccd9f13eaa
commit
efc76c3642
@ -99,33 +99,40 @@ func (c *ChatInstance) CreateStreamChatRequest(props *ChatProps, callback global
|
||||
if url, err := c.CreateImage(props); err != nil {
|
||||
return err
|
||||
} else {
|
||||
return callback(url)
|
||||
return callback(&globals.Chunk{Content: url})
|
||||
}
|
||||
}
|
||||
|
||||
isCompletionType := props.Model == globals.GPT3TurboInstruct
|
||||
|
||||
ticks := 0
|
||||
err := utils.EventScanner(&utils.EventScannerProps{
|
||||
Method: "POST",
|
||||
Uri: c.GetChatEndpoint(props),
|
||||
Headers: c.GetHeader(),
|
||||
Body: c.GetChatBody(props, true),
|
||||
Callback: func(data string) error {
|
||||
partial, err := c.ProcessLine(props.Buffer, data, isCompletionType)
|
||||
ticks += 1
|
||||
|
||||
partial, err := c.ProcessLine(data, isCompletionType)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return callback(partial)
|
||||
},
|
||||
})
|
||||
|
||||
if err != nil {
|
||||
if form := processChatErrorResponse(err.Body); form != nil {
|
||||
return errors.New(fmt.Sprintf("%s (type: %s)", form.Error.Message, form.Error.Type))
|
||||
msg := fmt.Sprintf("%s (type: %s)", form.Error.Message, form.Error.Type)
|
||||
return errors.New(msg)
|
||||
}
|
||||
return err.Error
|
||||
}
|
||||
|
||||
if ticks == 0 {
|
||||
return errors.New("no response")
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
@ -74,16 +74,18 @@ func processChatErrorResponse(data string) *ChatStreamErrorResponse {
|
||||
return utils.UnmarshalForm[ChatStreamErrorResponse](data)
|
||||
}
|
||||
|
||||
func getChoices(buffer utils.Buffer, form *ChatStreamResponse) string {
|
||||
func getChoices(form *ChatStreamResponse) *globals.Chunk {
|
||||
if len(form.Choices) == 0 {
|
||||
return ""
|
||||
return &globals.Chunk{Content: ""}
|
||||
}
|
||||
|
||||
choice := form.Choices[0].Delta
|
||||
|
||||
buffer.AddToolCalls(choice.ToolCalls)
|
||||
buffer.SetFunctionCall(choice.FunctionCall)
|
||||
return choice.Content
|
||||
return &globals.Chunk{
|
||||
Content: choice.Content,
|
||||
ToolCall: choice.ToolCalls,
|
||||
FunctionCall: choice.FunctionCall,
|
||||
}
|
||||
}
|
||||
|
||||
func getCompletionChoices(form *CompletionResponse) string {
|
||||
@ -109,25 +111,27 @@ func getRobustnessResult(chunk string) string {
|
||||
}
|
||||
}
|
||||
|
||||
func (c *ChatInstance) ProcessLine(obj utils.Buffer, data string, isCompletionType bool) (string, error) {
|
||||
func (c *ChatInstance) ProcessLine(data string, isCompletionType bool) (*globals.Chunk, error) {
|
||||
if isCompletionType {
|
||||
// legacy support
|
||||
// openai legacy support
|
||||
if completion := processCompletionResponse(data); completion != nil {
|
||||
return getCompletionChoices(completion), nil
|
||||
return &globals.Chunk{
|
||||
Content: getCompletionChoices(completion),
|
||||
}, nil
|
||||
}
|
||||
|
||||
globals.Warn(fmt.Sprintf("chatgpt error: cannot parse completion response: %s", data))
|
||||
return "", errors.New("parser error: cannot parse completion response")
|
||||
return &globals.Chunk{Content: ""}, errors.New("parser error: cannot parse completion response")
|
||||
}
|
||||
|
||||
if form := processChatResponse(data); form != nil {
|
||||
return getChoices(obj, form), nil
|
||||
return getChoices(form), nil
|
||||
}
|
||||
|
||||
if form := processChatErrorResponse(data); form != nil {
|
||||
return "", errors.New(fmt.Sprintf("chatgpt error: %s (type: %s)", form.Error.Message, form.Error.Type))
|
||||
return &globals.Chunk{Content: ""}, errors.New(fmt.Sprintf("chatgpt error: %s (type: %s)", form.Error.Message, form.Error.Type))
|
||||
}
|
||||
|
||||
globals.Warn(fmt.Sprintf("chatgpt error: cannot parse chat completion response: %s", data))
|
||||
return "", errors.New("parser error: cannot parse chat completion response")
|
||||
return &globals.Chunk{Content: ""}, errors.New("parser error: cannot parse chat completion response")
|
||||
}
|
||||
|
@ -3,8 +3,8 @@ package baichuan
|
||||
import (
|
||||
"chat/globals"
|
||||
"chat/utils"
|
||||
"errors"
|
||||
"fmt"
|
||||
"strings"
|
||||
)
|
||||
|
||||
type ChatProps struct {
|
||||
@ -72,44 +72,26 @@ func (c *ChatInstance) CreateChatRequest(props *ChatProps) (string, error) {
|
||||
|
||||
// CreateStreamChatRequest is the stream response body for baichuan
|
||||
func (c *ChatInstance) CreateStreamChatRequest(props *ChatProps, callback globals.Hook) error {
|
||||
buf := ""
|
||||
cursor := 0
|
||||
chunk := ""
|
||||
|
||||
err := utils.EventSource(
|
||||
"POST",
|
||||
c.GetChatEndpoint(),
|
||||
c.GetHeader(),
|
||||
c.GetChatBody(props, true),
|
||||
func(data string) error {
|
||||
data, err := c.ProcessLine(buf, data)
|
||||
chunk += data
|
||||
|
||||
err := utils.EventScanner(&utils.EventScannerProps{
|
||||
Method: "POST",
|
||||
Uri: c.GetChatEndpoint(),
|
||||
Headers: c.GetHeader(),
|
||||
Body: c.GetChatBody(props, true),
|
||||
Callback: func(data string) error {
|
||||
partial, err := c.ProcessLine(data)
|
||||
if err != nil {
|
||||
if strings.HasPrefix(err.Error(), "baichuan error") {
|
||||
return err
|
||||
}
|
||||
|
||||
// error when break line
|
||||
buf = buf + data
|
||||
return nil
|
||||
return err
|
||||
}
|
||||
|
||||
buf = ""
|
||||
if data != "" {
|
||||
cursor += 1
|
||||
if err := callback(data); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
return nil
|
||||
return callback(partial)
|
||||
},
|
||||
)
|
||||
})
|
||||
|
||||
if err != nil {
|
||||
return err
|
||||
} else if len(chunk) == 0 {
|
||||
return fmt.Errorf("empty response")
|
||||
if form := processChatErrorResponse(err.Body); form != nil {
|
||||
msg := fmt.Sprintf("%s (type: %s)", form.Error.Message, form.Error.Type)
|
||||
return errors.New(msg)
|
||||
}
|
||||
return err.Error
|
||||
}
|
||||
|
||||
return nil
|
||||
|
@ -5,95 +5,37 @@ import (
|
||||
"chat/utils"
|
||||
"errors"
|
||||
"fmt"
|
||||
"strings"
|
||||
)
|
||||
|
||||
func processFormat(data string) string {
|
||||
rep := strings.NewReplacer(
|
||||
"data: {",
|
||||
"\"data\": {",
|
||||
)
|
||||
item := rep.Replace(data)
|
||||
if !strings.HasPrefix(item, "{") {
|
||||
item = "{" + item
|
||||
}
|
||||
if !strings.HasSuffix(item, "}}") {
|
||||
item = item + "}"
|
||||
}
|
||||
|
||||
return item
|
||||
}
|
||||
|
||||
func processChatResponse(data string) *ChatStreamResponse {
|
||||
if strings.HasPrefix(data, "{") {
|
||||
var form *ChatStreamResponse
|
||||
if form = utils.UnmarshalForm[ChatStreamResponse](data); form != nil {
|
||||
return form
|
||||
}
|
||||
|
||||
if form = utils.UnmarshalForm[ChatStreamResponse](data[:len(data)-1]); form != nil {
|
||||
return form
|
||||
}
|
||||
|
||||
if form = utils.UnmarshalForm[ChatStreamResponse](data + "}"); form != nil {
|
||||
return form
|
||||
}
|
||||
}
|
||||
|
||||
return nil
|
||||
return utils.UnmarshalForm[ChatStreamResponse](data)
|
||||
}
|
||||
|
||||
func processChatErrorResponse(data string) *ChatStreamErrorResponse {
|
||||
if strings.HasPrefix(data, "{") {
|
||||
var form *ChatStreamErrorResponse
|
||||
if form = utils.UnmarshalForm[ChatStreamErrorResponse](data); form != nil {
|
||||
return form
|
||||
}
|
||||
if form = utils.UnmarshalForm[ChatStreamErrorResponse](data + "}"); form != nil {
|
||||
return form
|
||||
}
|
||||
}
|
||||
|
||||
return nil
|
||||
return utils.UnmarshalForm[ChatStreamErrorResponse](data)
|
||||
}
|
||||
|
||||
func isDone(data string) bool {
|
||||
return utils.Contains[string](data, []string{
|
||||
"{data: [DONE]}", "{data: [DONE]}}", "null}}", "{null}",
|
||||
"{[DONE]}", "{data:}", "{data:}}", "data: [DONE]}}",
|
||||
})
|
||||
}
|
||||
|
||||
func getChoices(form *ChatStreamResponse) string {
|
||||
if len(form.Data.Choices) == 0 {
|
||||
if len(form.Choices) > 0 {
|
||||
return form.Choices[0].Delta.Content
|
||||
}
|
||||
func getChoices(form *ChatStreamResponse) *globals.Chunk {
|
||||
if len(form.Choices) == 0 {
|
||||
return &globals.Chunk{Content: ""}
|
||||
}
|
||||
|
||||
return form.Data.Choices[0].Delta.Content
|
||||
choice := form.Choices[0].Delta
|
||||
|
||||
return &globals.Chunk{
|
||||
Content: choice.Content,
|
||||
}
|
||||
}
|
||||
|
||||
func (c *ChatInstance) ProcessLine(buf, data string) (string, error) {
|
||||
item := processFormat(buf + data)
|
||||
if isDone(item) {
|
||||
return "", nil
|
||||
}
|
||||
|
||||
if form := processChatResponse(item); form == nil {
|
||||
// recursive call
|
||||
if len(buf) > 0 {
|
||||
return c.ProcessLine("", buf+item)
|
||||
}
|
||||
|
||||
if err := processChatErrorResponse(item); err == nil || err.Data.Error.Message == "" {
|
||||
globals.Warn(fmt.Sprintf("baichuan error: cannot parse response: %s", item))
|
||||
return data, errors.New("parser error: cannot parse response")
|
||||
} else {
|
||||
return "", fmt.Errorf("baichuan error: %s (type: %s)", err.Data.Error.Message, err.Data.Error.Type)
|
||||
}
|
||||
|
||||
} else {
|
||||
func (c *ChatInstance) ProcessLine(data string) (*globals.Chunk, error) {
|
||||
if form := processChatResponse(data); form != nil {
|
||||
return getChoices(form), nil
|
||||
}
|
||||
|
||||
if form := processChatErrorResponse(data); form != nil {
|
||||
return &globals.Chunk{Content: ""}, errors.New(fmt.Sprintf("baichuan error: %s (type: %s)", form.Error.Message, form.Error.Type))
|
||||
}
|
||||
|
||||
globals.Warn(fmt.Sprintf("baichuan error: cannot parse chat completion response: %s", data))
|
||||
return &globals.Chunk{Content: ""}, errors.New("parser error: cannot parse chat completion response")
|
||||
}
|
||||
|
@ -32,19 +32,6 @@ type ChatResponse struct {
|
||||
|
||||
// ChatStreamResponse is the stream response body for baichuan
|
||||
type ChatStreamResponse struct {
|
||||
Data struct {
|
||||
ID string `json:"id"`
|
||||
Object string `json:"object"`
|
||||
Created int64 `json:"created"`
|
||||
Model string `json:"model"`
|
||||
Choices []struct {
|
||||
Delta struct {
|
||||
Content string `json:"content"`
|
||||
}
|
||||
Index int `json:"index"`
|
||||
} `json:"choices"`
|
||||
} `json:"data"`
|
||||
|
||||
ID string `json:"id"`
|
||||
Object string `json:"object"`
|
||||
Created int64 `json:"created"`
|
||||
@ -58,10 +45,8 @@ type ChatStreamResponse struct {
|
||||
}
|
||||
|
||||
type ChatStreamErrorResponse struct {
|
||||
Data struct {
|
||||
Error struct {
|
||||
Message string `json:"message"`
|
||||
Type string `json:"type"`
|
||||
} `json:"error"`
|
||||
} `json:"data"`
|
||||
Error struct {
|
||||
Message string `json:"message"`
|
||||
Type string `json:"type"`
|
||||
} `json:"error"`
|
||||
}
|
||||
|
@ -39,7 +39,9 @@ func (c *ChatInstance) CreateStreamChatRequest(props *ChatProps, hook globals.Ho
|
||||
return nil
|
||||
}
|
||||
|
||||
if err := hook(form.Response); err != nil {
|
||||
if err := hook(&globals.Chunk{
|
||||
Content: form.Response,
|
||||
}); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
@ -109,7 +109,9 @@ func (c *ChatInstance) CreateStreamChatRequest(props *ChatProps, callback global
|
||||
if url, err := c.CreateImage(props); err != nil {
|
||||
return err
|
||||
} else {
|
||||
return callback(url)
|
||||
return callback(&globals.Chunk{
|
||||
Content: url,
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
@ -124,7 +126,7 @@ func (c *ChatInstance) CreateStreamChatRequest(props *ChatProps, callback global
|
||||
Callback: func(data string) error {
|
||||
ticks += 1
|
||||
|
||||
partial, err := c.ProcessLine(props.Buffer, data, isCompletionType)
|
||||
partial, err := c.ProcessLine(data, isCompletionType)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
@ -72,16 +72,18 @@ func processChatErrorResponse(data string) *ChatStreamErrorResponse {
|
||||
return utils.UnmarshalForm[ChatStreamErrorResponse](data)
|
||||
}
|
||||
|
||||
func getChoices(buffer utils.Buffer, form *ChatStreamResponse) string {
|
||||
func getChoices(form *ChatStreamResponse) *globals.Chunk {
|
||||
if len(form.Choices) == 0 {
|
||||
return ""
|
||||
return &globals.Chunk{Content: ""}
|
||||
}
|
||||
|
||||
choice := form.Choices[0].Delta
|
||||
|
||||
buffer.AddToolCalls(choice.ToolCalls)
|
||||
buffer.SetFunctionCall(choice.FunctionCall)
|
||||
return choice.Content
|
||||
return &globals.Chunk{
|
||||
Content: choice.Content,
|
||||
ToolCall: choice.ToolCalls,
|
||||
FunctionCall: choice.FunctionCall,
|
||||
}
|
||||
}
|
||||
|
||||
func getCompletionChoices(form *CompletionResponse) string {
|
||||
@ -107,25 +109,27 @@ func getRobustnessResult(chunk string) string {
|
||||
}
|
||||
}
|
||||
|
||||
func (c *ChatInstance) ProcessLine(obj utils.Buffer, data string, isCompletionType bool) (string, error) {
|
||||
func (c *ChatInstance) ProcessLine(data string, isCompletionType bool) (*globals.Chunk, error) {
|
||||
if isCompletionType {
|
||||
// legacy support
|
||||
// openai legacy support
|
||||
if completion := processCompletionResponse(data); completion != nil {
|
||||
return getCompletionChoices(completion), nil
|
||||
return &globals.Chunk{
|
||||
Content: getCompletionChoices(completion),
|
||||
}, nil
|
||||
}
|
||||
|
||||
globals.Warn(fmt.Sprintf("chatgpt error: cannot parse completion response: %s", data))
|
||||
return "", errors.New("parser error: cannot parse completion response")
|
||||
return &globals.Chunk{Content: ""}, errors.New("parser error: cannot parse completion response")
|
||||
}
|
||||
|
||||
if form := processChatResponse(data); form != nil {
|
||||
return getChoices(obj, form), nil
|
||||
return getChoices(form), nil
|
||||
}
|
||||
|
||||
if form := processChatErrorResponse(data); form != nil {
|
||||
return "", errors.New(fmt.Sprintf("chatgpt error: %s (type: %s)", form.Error.Message, form.Error.Type))
|
||||
return &globals.Chunk{Content: ""}, errors.New(fmt.Sprintf("chatgpt error: %s (type: %s)", form.Error.Message, form.Error.Type))
|
||||
}
|
||||
|
||||
globals.Warn(fmt.Sprintf("chatgpt error: cannot parse chat completion response: %s", data))
|
||||
return "", errors.New("parser error: cannot parse chat completion response")
|
||||
return &globals.Chunk{Content: ""}, errors.New("parser error: cannot parse chat completion response")
|
||||
}
|
||||
|
@ -127,7 +127,7 @@ func (c *ChatInstance) CreateStreamChatRequest(props *ChatProps, hook globals.Ho
|
||||
|
||||
if resp, err := c.ProcessLine(buf, data); err == nil && len(resp) > 0 {
|
||||
buf = ""
|
||||
if err := hook(resp); err != nil {
|
||||
if err := hook(&globals.Chunk{Content: resp}); err != nil {
|
||||
return err
|
||||
}
|
||||
} else {
|
||||
|
@ -126,7 +126,7 @@ func (c *ChatInstance) CreateStreamChatRequest(props *ChatProps, callback global
|
||||
return fmt.Errorf("dashscope error: %s", form.Message)
|
||||
}
|
||||
|
||||
if err := callback(form.Output.Text); err != nil {
|
||||
if err := callback(&globals.Chunk{Content: form.Output.Text}); err != nil {
|
||||
return err
|
||||
}
|
||||
return nil
|
||||
|
@ -50,7 +50,12 @@ func (c *ChatInstance) CreateStreamChatRequest(props *ChatProps, callback global
|
||||
break
|
||||
}
|
||||
|
||||
if err := callback(chunk.Choices[0].Delta.Content); err != nil {
|
||||
if len(chunk.Choices) == 0 {
|
||||
continue
|
||||
}
|
||||
|
||||
choice := chunk.Choices[0].Delta
|
||||
if err := callback(&globals.Chunk{Content: choice.Content}); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
@ -79,21 +79,21 @@ func (c *ChatInstance) CreateStreamChatRequest(props *ChatProps, callback global
|
||||
form, err := c.CreateStreamTask(action, prompt, func(form *StorageForm, progress int) error {
|
||||
if progress == 0 {
|
||||
begin = true
|
||||
if err := callback("```progress\n"); err != nil {
|
||||
if err := callback(&globals.Chunk{Content: "```progress\n"}); err != nil {
|
||||
return err
|
||||
}
|
||||
} else if progress == 100 && !begin {
|
||||
if err := callback("```progress\n"); err != nil {
|
||||
if err := callback(&globals.Chunk{Content: "```progress\n"}); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
if err := callback(fmt.Sprintf("%d\n", progress)); err != nil {
|
||||
if err := callback(&globals.Chunk{Content: fmt.Sprintf("%d\n", progress)}); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if progress == 100 {
|
||||
if err := callback("```\n"); err != nil {
|
||||
if err := callback(&globals.Chunk{Content: "```\n"}); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
@ -105,7 +105,7 @@ func (c *ChatInstance) CreateStreamChatRequest(props *ChatProps, callback global
|
||||
return fmt.Errorf("error from midjourney: %s", err.Error())
|
||||
}
|
||||
|
||||
if err := callback(utils.GetImageMarkdown(form.Url)); err != nil {
|
||||
if err := callback(&globals.Chunk{Content: utils.GetImageMarkdown(form.Url)}); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
@ -133,5 +133,7 @@ func (c *ChatInstance) CallbackActions(form *StorageForm, callback globals.Hook)
|
||||
|
||||
reroll := fmt.Sprintf("[REROLL](%s)", toVirtualMessage(fmt.Sprintf("/REROLL %s", form.Task)))
|
||||
|
||||
return callback(fmt.Sprintf("\n\n%s\n\n%s\n\n%s\n", upscale, variation, reroll))
|
||||
return callback(&globals.Chunk{
|
||||
Content: fmt.Sprintf("\n\n%s\n\n%s\n\n%s\n", upscale, variation, reroll),
|
||||
})
|
||||
}
|
||||
|
@ -129,7 +129,7 @@ func (c *ChatInstance) CreateStreamChatRequest(props *ChatProps, callback global
|
||||
}
|
||||
|
||||
for _, item := range utils.SplitItem(response, " ") {
|
||||
if err := callback(item); err != nil {
|
||||
if err := callback(&globals.Chunk{Content: item}); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
@ -66,14 +66,17 @@ func (c *ChatInstance) CreateRequest(props *ChatProps) *api.ChatReq {
|
||||
}
|
||||
}
|
||||
|
||||
func getChoice(choice *api.ChatResp, buffer utils.Buffer) string {
|
||||
func getChoice(choice *api.ChatResp) *globals.Chunk {
|
||||
if choice == nil {
|
||||
return ""
|
||||
return &globals.Chunk{Content: ""}
|
||||
}
|
||||
|
||||
calls := choice.Choice.Message.FunctionCall
|
||||
if calls != nil {
|
||||
buffer.AddToolCalls(&globals.ToolCalls{
|
||||
message := choice.Choice.Message
|
||||
|
||||
calls := message.FunctionCall
|
||||
return &globals.Chunk{
|
||||
Content: message.Content,
|
||||
ToolCall: utils.Multi(calls != nil, &globals.ToolCalls{
|
||||
globals.ToolCall{
|
||||
Type: "function",
|
||||
Id: globals.ToolCallId(fmt.Sprintf("%s-%s", calls.Name, choice.ReqId)),
|
||||
@ -82,9 +85,8 @@ func getChoice(choice *api.ChatResp, buffer utils.Buffer) string {
|
||||
Arguments: calls.Arguments,
|
||||
},
|
||||
},
|
||||
})
|
||||
}, nil),
|
||||
}
|
||||
return choice.Choice.Message.Content
|
||||
}
|
||||
|
||||
func (c *ChatInstance) CreateStreamChatRequest(props *ChatProps, callback globals.Hook) error {
|
||||
@ -99,7 +101,7 @@ func (c *ChatInstance) CreateStreamChatRequest(props *ChatProps, callback global
|
||||
return partial.Error
|
||||
}
|
||||
|
||||
if err := callback(getChoice(partial, props.Buffer)); err != nil {
|
||||
if err := callback(getChoice(partial)); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
@ -77,7 +77,7 @@ func (c *ChatInstance) ProcessPartialResponse(res chan types.PartialResponse, ho
|
||||
if data.Error != nil {
|
||||
return data.Error
|
||||
} else if data.Text != "" {
|
||||
if err := hook(data.Text); err != nil {
|
||||
if err := hook(&globals.Chunk{Content: data.Text}); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
@ -67,26 +67,26 @@ func (c *ChatInstance) GetFunctionCalling(props *ChatProps) *FunctionsPayload {
|
||||
}
|
||||
}
|
||||
|
||||
func getChoice(form *ChatResponse, buffer utils.Buffer) string {
|
||||
resp := form.Payload.Choices.Text
|
||||
if len(resp) == 0 {
|
||||
return ""
|
||||
func getChoice(form *ChatResponse) *globals.Chunk {
|
||||
if len(form.Payload.Choices.Text) == 0 {
|
||||
return &globals.Chunk{Content: ""}
|
||||
}
|
||||
|
||||
if resp[0].FunctionCall != nil {
|
||||
buffer.AddToolCalls(&globals.ToolCalls{
|
||||
choice := form.Payload.Choices.Text[0]
|
||||
|
||||
return &globals.Chunk{
|
||||
Content: choice.Content,
|
||||
ToolCall: utils.Multi(choice.FunctionCall != nil, &globals.ToolCalls{
|
||||
globals.ToolCall{
|
||||
Type: "function",
|
||||
Id: globals.ToolCallId(fmt.Sprintf("%s-%s", resp[0].FunctionCall.Name, resp[0].FunctionCall.Arguments)),
|
||||
Id: globals.ToolCallId(fmt.Sprintf("%s-%s", choice.FunctionCall.Name, choice.FunctionCall.Arguments)),
|
||||
Function: globals.ToolCallFunction{
|
||||
Name: resp[0].FunctionCall.Name,
|
||||
Arguments: resp[0].FunctionCall.Arguments,
|
||||
Name: choice.FunctionCall.Name,
|
||||
Arguments: choice.FunctionCall.Arguments,
|
||||
},
|
||||
},
|
||||
})
|
||||
}, nil),
|
||||
}
|
||||
|
||||
return resp[0].Content
|
||||
}
|
||||
|
||||
func (c *ChatInstance) CreateStreamChatRequest(props *ChatProps, hook globals.Hook) error {
|
||||
@ -130,7 +130,7 @@ func (c *ChatInstance) CreateStreamChatRequest(props *ChatProps, hook globals.Ho
|
||||
return fmt.Errorf("sparkdesk error: %s (sid: %s)", form.Header.Message, form.Header.Sid)
|
||||
}
|
||||
|
||||
if err := hook(getChoice(form, props.Buffer)); err != nil {
|
||||
if err := hook(getChoice(form)); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
@ -103,7 +103,7 @@ func (c *ChatInstance) CreateStreamChatRequest(props *ChatProps, callback global
|
||||
buf = ""
|
||||
if data != "" {
|
||||
cursor += 1
|
||||
if err := callback(data); err != nil {
|
||||
if err := callback(&globals.Chunk{Content: data}); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
@ -73,7 +73,7 @@ func (c *ChatInstance) CreateStreamChatRequest(props *ChatProps, hook globals.Ho
|
||||
}
|
||||
|
||||
data = strings.TrimPrefix(data, "data:")
|
||||
return hook(data)
|
||||
return hook(&globals.Chunk{Content: data})
|
||||
},
|
||||
)
|
||||
}
|
||||
|
@ -21,9 +21,9 @@ func CreateGeneration(group, model, prompt, path string, hook func(buffer *utils
|
||||
Model: model,
|
||||
Message: message,
|
||||
Buffer: *buffer,
|
||||
}, func(data string) error {
|
||||
buffer.Write(data)
|
||||
hook(buffer, data)
|
||||
}, func(data *globals.Chunk) error {
|
||||
buffer.WriteChunk(data)
|
||||
hook(buffer, data.Content)
|
||||
return nil
|
||||
})
|
||||
|
||||
|
@ -24,7 +24,9 @@ export function parseFile(data: string, acceptDownload?: boolean) {
|
||||
|
||||
const b64image = useMemo(() => {
|
||||
// get base64 image from content (like: )
|
||||
const match = content.match(/data:image\/([^;]+);base64,([a-zA-Z0-9+/=]+)/g);
|
||||
const match = content.match(
|
||||
/data:image\/([^;]+);base64,([a-zA-Z0-9+/=]+)/g,
|
||||
);
|
||||
return match ? match[0] : "";
|
||||
}, [filename, content]);
|
||||
|
||||
|
@ -356,7 +356,7 @@
|
||||
"align": "聊天框居中",
|
||||
"memory": "内存占用",
|
||||
"max-tokens": "最大回复 Token 数",
|
||||
"max-tokens-tip": "最大回复 Token 数,超过此数值将会被截断",
|
||||
"max-tokens-tip": "最大回复 Token 数,超过此数值将会被截断(过高的数值可能会导致超过模型的最大 Token 导致请求失败)",
|
||||
"temperature": "温度",
|
||||
"temperature-tip": "随机采样的比例,高温度会产生更多的随机性,低温度会产生较集中和确定性的文本",
|
||||
"top-p": "核采样概率阈值",
|
||||
|
@ -306,7 +306,7 @@
|
||||
"temperature": "temperature",
|
||||
"temperature-tip": "Random sampling ratio, high temperature produces more randomness, low temperature produces more concentrated and deterministic text",
|
||||
"max-tokens": "Maximum number of response tokens",
|
||||
"max-tokens-tip": "Maximum number of reply tokens, exceeding this value will be truncated",
|
||||
"max-tokens-tip": "Maximum number of reply tokens, exceeding this value will be truncated (too high value may cause the request to fail due to exceeding the model's maximum token)",
|
||||
"top-p": "Kernel Sampling Probability Threshold",
|
||||
"top-p-tip": "(TopP) The higher the probability value, the higher the randomness generated; the lower the value, the higher the certainty generated",
|
||||
"top-k": "Sample Candidate Set Size",
|
||||
|
@ -306,7 +306,7 @@
|
||||
"temperature": "温度",
|
||||
"temperature-tip": "ランダムサンプリング比、高温はよりランダム性を生み、低温はより集中的で決定論的なテキストを生成します",
|
||||
"max-tokens": "レスポンストークンの最大数",
|
||||
"max-tokens-tip": "この値を超える返信トークンの最大数は切り捨てられます",
|
||||
"max-tokens-tip": "この値を超えるリプライトークンの最大数は切り捨てられます(値が高すぎると、モデルの最大トークンを超えるために要求が失敗する可能性があります)",
|
||||
"top-p": "カーネルサンプリング確率閾値",
|
||||
"top-p-tip": "( TopP )確率値が高いほど生成されるランダム性が高く、値が低いほど生成される確実性が高くなります",
|
||||
"top-k": "サンプル候補セットサイズ",
|
||||
|
@ -306,7 +306,7 @@
|
||||
"temperature": "Температура",
|
||||
"temperature-tip": "Коэффициент случайной выборки, высокая температура создает больше случайности, низкая температура создает более концентрированный и детерминированный текст",
|
||||
"max-tokens": "Максимальное количество маркеров ответа",
|
||||
"max-tokens-tip": "Максимальное количество маркеров ответа, превышающее это значение, будет усечено",
|
||||
"max-tokens-tip": "Максимальное количество маркеров ответа, превышающее это значение, будет усечено (слишком высокое значение может привести к сбою запроса из-за превышения максимального маркера модели)",
|
||||
"top-p": "Порог вероятности отбора проб ядра",
|
||||
"top-p-tip": "(TopP) Чем выше значение вероятности, тем выше генерируемая случайность; чем ниже значение, тем выше генерируемая определенность",
|
||||
"top-k": "Размер набора образцов-кандидатов",
|
||||
|
@ -46,5 +46,7 @@ export function doTranslate(
|
||||
from = getFormattedLanguage(from);
|
||||
to = getFormattedLanguage(to);
|
||||
|
||||
if (content.startsWith("!!")) content = content.substring(2);
|
||||
|
||||
return translate(content, from, to);
|
||||
}
|
||||
|
@ -23,6 +23,7 @@ export function getMigration(
|
||||
switch (typeof template) {
|
||||
case "string":
|
||||
if (typeof translation !== "string") return val;
|
||||
else if (template.startsWith("!!")) return val;
|
||||
break;
|
||||
case "object":
|
||||
return getMigration(template, translation, val[0]);
|
||||
|
@ -53,7 +53,12 @@ func PreflightCache(cache *redis.Client, hash string, buffer *utils.Buffer, hook
|
||||
buffer.SetInputTokens(buf.CountInputToken())
|
||||
buffer.SetToolCalls(buf.GetToolCalls())
|
||||
buffer.SetFunctionCall(buf.GetFunctionCall())
|
||||
return idx, true, hook(data)
|
||||
|
||||
return idx, true, hook(&globals.Chunk{
|
||||
Content: data,
|
||||
FunctionCall: buf.GetFunctionCall(),
|
||||
ToolCall: buf.GetToolCalls(),
|
||||
})
|
||||
}
|
||||
|
||||
func StoreCache(cache *redis.Client, hash string, index int64, buffer *utils.Buffer) {
|
||||
|
@ -1,6 +1,7 @@
|
||||
package globals
|
||||
|
||||
type Hook func(data string) error
|
||||
type Hook func(data *Chunk) error
|
||||
|
||||
type Message struct {
|
||||
Role string `json:"role"`
|
||||
Content string `json:"content"`
|
||||
@ -10,6 +11,12 @@ type Message struct {
|
||||
ToolCalls *ToolCalls `json:"tool_calls,omitempty"` // only `assistant` role
|
||||
}
|
||||
|
||||
type Chunk struct {
|
||||
Content string `json:"content"`
|
||||
ToolCall *ToolCalls `json:"tool_call,omitempty"`
|
||||
FunctionCall *FunctionCall `json:"function_call,omitempty"`
|
||||
}
|
||||
|
||||
type ChatSegmentResponse struct {
|
||||
Conversation int64 `json:"conversation"`
|
||||
Quota float32 `json:"quota"`
|
||||
|
@ -100,13 +100,13 @@ func ChatHandler(conn *Connection, user *auth.User, instance *conversation.Conve
|
||||
FrequencyPenalty: instance.GetFrequencyPenalty(),
|
||||
RepetitionPenalty: instance.GetRepetitionPenalty(),
|
||||
},
|
||||
func(data string) error {
|
||||
func(data *globals.Chunk) error {
|
||||
if signal := conn.PeekWithType(StopType); signal != nil {
|
||||
// stop signal from client
|
||||
return fmt.Errorf("signal")
|
||||
}
|
||||
return conn.SendClient(globals.ChatSegmentResponse{
|
||||
Message: buffer.Write(data),
|
||||
Message: buffer.WriteChunk(data),
|
||||
Quota: buffer.GetQuota(),
|
||||
End: false,
|
||||
Plan: plan,
|
||||
|
@ -93,8 +93,8 @@ func sendTranshipmentResponse(c *gin.Context, form RelayForm, messages []globals
|
||||
cache := utils.GetCacheFromContext(c)
|
||||
|
||||
buffer := utils.NewBuffer(form.Model, messages, channel.ChargeInstance.GetCharge(form.Model))
|
||||
hit, err := channel.NewChatRequestWithCache(cache, buffer, auth.GetGroup(db, user), getChatProps(form, messages, buffer, plan), func(data string) error {
|
||||
buffer.Write(data)
|
||||
hit, err := channel.NewChatRequestWithCache(cache, buffer, auth.GetGroup(db, user), getChatProps(form, messages, buffer, plan), func(data *globals.Chunk) error {
|
||||
buffer.WriteChunk(data)
|
||||
return nil
|
||||
})
|
||||
|
||||
@ -137,14 +137,7 @@ func sendTranshipmentResponse(c *gin.Context, form RelayForm, messages []globals
|
||||
})
|
||||
}
|
||||
|
||||
func getStreamTranshipmentForm(id string, created int64, form RelayForm, data string, buffer *utils.Buffer, end bool, err error) RelayStreamResponse {
|
||||
toolCalling := buffer.GetToolCalls()
|
||||
|
||||
var functionCalling *globals.FunctionCall
|
||||
if end {
|
||||
functionCalling = buffer.GetFunctionCall()
|
||||
}
|
||||
|
||||
func getStreamTranshipmentForm(id string, created int64, form RelayForm, data *globals.Chunk, buffer *utils.Buffer, end bool, err error) RelayStreamResponse {
|
||||
return RelayStreamResponse{
|
||||
Id: fmt.Sprintf("chatcmpl-%s", id),
|
||||
Object: "chat.completion.chunk",
|
||||
@ -155,9 +148,9 @@ func getStreamTranshipmentForm(id string, created int64, form RelayForm, data st
|
||||
Index: 0,
|
||||
Delta: globals.Message{
|
||||
Role: globals.Assistant,
|
||||
Content: data,
|
||||
ToolCalls: toolCalling,
|
||||
FunctionCall: functionCalling,
|
||||
Content: data.Content,
|
||||
ToolCalls: data.ToolCall,
|
||||
FunctionCall: data.FunctionCall,
|
||||
},
|
||||
FinishReason: utils.Multi[interface{}](end, "stop", nil),
|
||||
},
|
||||
@ -177,23 +170,30 @@ func sendStreamTranshipmentResponse(c *gin.Context, form RelayForm, messages []g
|
||||
db := utils.GetDBFromContext(c)
|
||||
cache := utils.GetCacheFromContext(c)
|
||||
|
||||
group := auth.GetGroup(db, user)
|
||||
charge := channel.ChargeInstance.GetCharge(form.Model)
|
||||
|
||||
go func() {
|
||||
buffer := utils.NewBuffer(form.Model, messages, channel.ChargeInstance.GetCharge(form.Model))
|
||||
hit, err := channel.NewChatRequestWithCache(cache, buffer, auth.GetGroup(db, user), getChatProps(form, messages, buffer, plan), func(data string) error {
|
||||
partial <- getStreamTranshipmentForm(id, created, form, buffer.Write(data), buffer, false, nil)
|
||||
return nil
|
||||
})
|
||||
buffer := utils.NewBuffer(form.Model, messages, charge)
|
||||
hit, err := channel.NewChatRequestWithCache(
|
||||
cache, buffer, group, getChatProps(form, messages, buffer, plan),
|
||||
func(data *globals.Chunk) error {
|
||||
buffer.WriteChunk(data)
|
||||
partial <- getStreamTranshipmentForm(id, created, form, data, buffer, false, nil)
|
||||
return nil
|
||||
},
|
||||
)
|
||||
|
||||
admin.AnalysisRequest(form.Model, buffer, err)
|
||||
if err != nil {
|
||||
auth.RevertSubscriptionUsage(db, cache, user, form.Model)
|
||||
globals.Warn(fmt.Sprintf("error from chat request api: %s (instance: %s, client: %s)", err.Error(), form.Model, c.ClientIP()))
|
||||
partial <- getStreamTranshipmentForm(id, created, form, err.Error(), buffer, true, err)
|
||||
partial <- getStreamTranshipmentForm(id, created, form, &globals.Chunk{Content: err.Error()}, buffer, true, err)
|
||||
close(partial)
|
||||
return
|
||||
}
|
||||
|
||||
partial <- getStreamTranshipmentForm(id, created, form, "", buffer, true, nil)
|
||||
partial <- getStreamTranshipmentForm(id, created, form, &globals.Chunk{Content: ""}, buffer, true, nil)
|
||||
|
||||
if !hit {
|
||||
CollectQuota(c, user, buffer, plan, err)
|
||||
|
@ -40,8 +40,8 @@ func NativeChatHandler(c *gin.Context, user *auth.User, model string, message []
|
||||
Message: segment,
|
||||
Buffer: *buffer,
|
||||
},
|
||||
func(resp string) error {
|
||||
buffer.Write(resp)
|
||||
func(resp *globals.Chunk) error {
|
||||
buffer.WriteChunk(resp)
|
||||
return nil
|
||||
},
|
||||
)
|
||||
|
@ -89,8 +89,8 @@ func createRelayImageObject(c *gin.Context, form RelayImageForm, prompt string,
|
||||
}
|
||||
|
||||
buffer := utils.NewBuffer(form.Model, messages, channel.ChargeInstance.GetCharge(form.Model))
|
||||
hit, err := channel.NewChatRequestWithCache(cache, buffer, auth.GetGroup(db, user), getImageProps(form, messages, buffer), func(data string) error {
|
||||
buffer.Write(data)
|
||||
hit, err := channel.NewChatRequestWithCache(cache, buffer, auth.GetGroup(db, user), getImageProps(form, messages, buffer), func(data *globals.Chunk) error {
|
||||
buffer.WriteChunk(data)
|
||||
return nil
|
||||
})
|
||||
|
||||
|
@ -81,6 +81,15 @@ func InsertSlice[T any](arr []T, index int, value []T) []T {
|
||||
return arr
|
||||
}
|
||||
|
||||
func Collect[T any](arr ...[]T) []T {
|
||||
res := make([]T, 0)
|
||||
|
||||
for _, v := range arr {
|
||||
res = append(res, v...)
|
||||
}
|
||||
return res
|
||||
}
|
||||
|
||||
func Append[T any](arr []T, value T) []T {
|
||||
return append(arr, value)
|
||||
}
|
||||
|
@ -90,6 +90,18 @@ func (b *Buffer) Write(data string) string {
|
||||
return data
|
||||
}
|
||||
|
||||
func (b *Buffer) WriteChunk(data *globals.Chunk) string {
|
||||
if data == nil {
|
||||
return ""
|
||||
}
|
||||
|
||||
b.Write(data.Content)
|
||||
b.AddToolCalls(data.ToolCall)
|
||||
b.SetFunctionCall(data.FunctionCall)
|
||||
|
||||
return data.Content
|
||||
}
|
||||
|
||||
func (b *Buffer) GetChunk() string {
|
||||
return b.Latest
|
||||
}
|
||||
@ -114,12 +126,52 @@ func (b *Buffer) SetToolCalls(toolCalls *globals.ToolCalls) {
|
||||
b.ToolCalls = toolCalls
|
||||
}
|
||||
|
||||
func hitTool(tool globals.ToolCall, tools globals.ToolCalls) (int, *globals.ToolCall) {
|
||||
for i, t := range tools {
|
||||
if t.Id == tool.Id {
|
||||
return i, &t
|
||||
}
|
||||
}
|
||||
|
||||
if len(tool.Type) == 0 && len(tool.Id) == 0 {
|
||||
length := len(tools)
|
||||
|
||||
if length > 0 {
|
||||
// if the tool is empty, return the last tool as the hit
|
||||
return length - 1, &tools[length-1]
|
||||
}
|
||||
}
|
||||
|
||||
return 0, nil
|
||||
}
|
||||
|
||||
func mixTools(source *globals.ToolCalls, target *globals.ToolCalls) *globals.ToolCalls {
|
||||
if source == nil {
|
||||
return target
|
||||
}
|
||||
|
||||
tools := make(globals.ToolCalls, 0)
|
||||
arr := Collect[globals.ToolCall](*source, *target)
|
||||
|
||||
for _, tool := range arr {
|
||||
idx, hit := hitTool(tool, tools)
|
||||
|
||||
if hit != nil {
|
||||
tools[idx].Function.Arguments += tool.Function.Arguments
|
||||
} else {
|
||||
tools = append(tools, tool)
|
||||
}
|
||||
}
|
||||
|
||||
return &tools
|
||||
}
|
||||
|
||||
func (b *Buffer) AddToolCalls(toolCalls *globals.ToolCalls) {
|
||||
if toolCalls == nil {
|
||||
return
|
||||
}
|
||||
|
||||
b.ToolCalls = toolCalls
|
||||
b.ToolCalls = mixTools(b.ToolCalls, toolCalls)
|
||||
}
|
||||
|
||||
func (b *Buffer) SetFunctionCall(functionCall *globals.FunctionCall) {
|
||||
|
Loading…
Reference in New Issue
Block a user