mirror of
https://github.com/coaidev/coai.git
synced 2025-05-26 00:10:15 +09:00
fix: fix frontend base64 image matcher, input tokens and b64images content hidden in un-vision models
This commit is contained in:
parent
357d22a940
commit
ccd9f13eaa
@ -2,6 +2,7 @@ package adapter
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"chat/globals"
|
"chat/globals"
|
||||||
|
"chat/utils"
|
||||||
"fmt"
|
"fmt"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
@ -44,3 +45,25 @@ func NewChatRequest(conf globals.ChannelConfig, props *ChatProps, hook globals.H
|
|||||||
|
|
||||||
return conf.ProcessError(err)
|
return conf.ProcessError(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func ClearMessages(model string, messages []globals.Message) []globals.Message {
|
||||||
|
if globals.IsVisionModel(model) {
|
||||||
|
return messages
|
||||||
|
}
|
||||||
|
|
||||||
|
return utils.Each[globals.Message](messages, func(message globals.Message) globals.Message {
|
||||||
|
if message.Role != globals.User {
|
||||||
|
return message
|
||||||
|
}
|
||||||
|
|
||||||
|
images := utils.ExtractBase64Images(message.Content)
|
||||||
|
for _, image := range images {
|
||||||
|
if len(image) <= 46 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
message.Content = strings.Replace(message.Content, image, utils.Extract(image, 46, " ..."), -1)
|
||||||
|
}
|
||||||
|
return message
|
||||||
|
})
|
||||||
|
}
|
||||||
|
@ -24,7 +24,7 @@ export function parseFile(data: string, acceptDownload?: boolean) {
|
|||||||
|
|
||||||
const b64image = useMemo(() => {
|
const b64image = useMemo(() => {
|
||||||
// get base64 image from content (like: )
|
// get base64 image from content (like: )
|
||||||
const match = content.match(/(data:image\/.*;base64,.*=)/);
|
const match = content.match(/data:image\/([^;]+);base64,([a-zA-Z0-9+/=]+)/g);
|
||||||
return match ? match[0] : "";
|
return match ? match[0] : "";
|
||||||
}, [filename, content]);
|
}, [filename, content]);
|
||||||
|
|
||||||
|
@ -50,6 +50,7 @@ func PreflightCache(cache *redis.Client, hash string, buffer *utils.Buffer, hook
|
|||||||
return idx, false, nil
|
return idx, false, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
buffer.SetInputTokens(buf.CountInputToken())
|
||||||
buffer.SetToolCalls(buf.GetToolCalls())
|
buffer.SetToolCalls(buf.GetToolCalls())
|
||||||
buffer.SetFunctionCall(buf.GetFunctionCall())
|
buffer.SetFunctionCall(buf.GetFunctionCall())
|
||||||
return idx, true, hook(data)
|
return idx, true, hook(data)
|
||||||
|
@ -115,6 +115,11 @@ var OpenAIVisionModels = []string{
|
|||||||
GPT4VisionPreview, GPT41106VisionPreview,
|
GPT4VisionPreview, GPT41106VisionPreview,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var VisionModels = []string{
|
||||||
|
GPT4VisionPreview, GPT41106VisionPreview,
|
||||||
|
GeminiProVision,
|
||||||
|
}
|
||||||
|
|
||||||
func in(value string, slice []string) bool {
|
func in(value string, slice []string) bool {
|
||||||
for _, item := range slice {
|
for _, item := range slice {
|
||||||
if item == value || strings.Contains(value, item) {
|
if item == value || strings.Contains(value, item) {
|
||||||
@ -133,3 +138,7 @@ func IsOpenAIVisionModels(model string) bool {
|
|||||||
// enable openai image format for gpt-4-vision-preview models
|
// enable openai image format for gpt-4-vision-preview models
|
||||||
return in(model, OpenAIVisionModels)
|
return in(model, OpenAIVisionModels)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func IsVisionModel(model string) bool {
|
||||||
|
return in(model, VisionModels)
|
||||||
|
}
|
||||||
|
@ -1,38 +0,0 @@
|
|||||||
package manager
|
|
||||||
|
|
||||||
import (
|
|
||||||
"chat/channel"
|
|
||||||
"chat/globals"
|
|
||||||
"chat/utils"
|
|
||||||
"fmt"
|
|
||||||
"github.com/gin-gonic/gin"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
type CacheProps struct {
|
|
||||||
Message []globals.Message `json:"message" required:"true"`
|
|
||||||
Model string `json:"model" required:"true"`
|
|
||||||
Reversible bool `json:"reversible"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type CacheData struct {
|
|
||||||
Message string `json:"message"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func ExtractCacheData(c *gin.Context, props *CacheProps) *CacheData {
|
|
||||||
hash := utils.Md5Encrypt(utils.Marshal(props))
|
|
||||||
data, err := utils.GetCacheFromContext(c).Get(c, fmt.Sprintf(":niodata:%s", hash)).Result()
|
|
||||||
if err == nil && data != "" {
|
|
||||||
return utils.UnmarshalForm[CacheData](data)
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func SaveCacheData(c *gin.Context, props *CacheProps, data *CacheData) {
|
|
||||||
if channel.ChargeInstance.IsBilling(props.Model) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
hash := utils.Md5Encrypt(utils.Marshal(props))
|
|
||||||
utils.GetCacheFromContext(c).Set(c, fmt.Sprintf(":niodata:%s", hash), utils.Marshal(data), time.Hour*12)
|
|
||||||
}
|
|
@ -63,11 +63,12 @@ func ChatHandler(conn *Connection, user *auth.User, instance *conversation.Conve
|
|||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
segment := web.UsingWebSegment(instance)
|
|
||||||
|
|
||||||
model := instance.GetModel()
|
|
||||||
db := conn.GetDB()
|
db := conn.GetDB()
|
||||||
cache := conn.GetCache()
|
cache := conn.GetCache()
|
||||||
|
|
||||||
|
model := instance.GetModel()
|
||||||
|
segment := adapter.ClearMessages(model, web.UsingWebSegment(instance))
|
||||||
|
|
||||||
check, plan := auth.CanEnableModelWithSubscription(db, cache, user, model)
|
check, plan := auth.CanEnableModelWithSubscription(db, cache, user, model)
|
||||||
conn.Send(globals.ChatSegmentResponse{
|
conn.Send(globals.ChatSegmentResponse{
|
||||||
Conversation: instance.GetId(),
|
Conversation: instance.GetId(),
|
||||||
@ -83,15 +84,6 @@ func ChatHandler(conn *Connection, user *auth.User, instance *conversation.Conve
|
|||||||
return message
|
return message
|
||||||
}
|
}
|
||||||
|
|
||||||
if form := ExtractCacheData(conn.GetCtx(), &CacheProps{
|
|
||||||
Message: segment,
|
|
||||||
Model: model,
|
|
||||||
Reversible: plan,
|
|
||||||
}); form != nil {
|
|
||||||
MockStreamSender(conn, form.Message)
|
|
||||||
return form.Message
|
|
||||||
}
|
|
||||||
|
|
||||||
buffer := utils.NewBuffer(model, segment, channel.ChargeInstance.GetCharge(model))
|
buffer := utils.NewBuffer(model, segment, channel.ChargeInstance.GetCharge(model))
|
||||||
hit, err := channel.NewChatRequestWithCache(
|
hit, err := channel.NewChatRequestWithCache(
|
||||||
cache, buffer,
|
cache, buffer,
|
||||||
@ -152,17 +144,5 @@ func ChatHandler(conn *Connection, user *auth.User, instance *conversation.Conve
|
|||||||
Plan: plan,
|
Plan: plan,
|
||||||
})
|
})
|
||||||
|
|
||||||
result := buffer.ReadWithDefault(defaultMessage)
|
return buffer.ReadWithDefault(defaultMessage)
|
||||||
|
|
||||||
if err == nil && result != defaultMessage {
|
|
||||||
SaveCacheData(conn.GetCtx(), &CacheProps{
|
|
||||||
Message: segment,
|
|
||||||
Model: model,
|
|
||||||
Reversible: plan,
|
|
||||||
}, &CacheData{
|
|
||||||
Message: result,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
return result
|
|
||||||
}
|
}
|
||||||
|
@ -121,7 +121,7 @@ func sendTranshipmentResponse(c *gin.Context, form RelayForm, messages []globals
|
|||||||
Index: 0,
|
Index: 0,
|
||||||
Message: globals.Message{
|
Message: globals.Message{
|
||||||
Role: globals.Assistant,
|
Role: globals.Assistant,
|
||||||
Content: buffer.ReadWithDefault(defaultMessage),
|
Content: buffer.Read(),
|
||||||
ToolCalls: buffer.GetToolCalls(),
|
ToolCalls: buffer.GetToolCalls(),
|
||||||
FunctionCall: buffer.GetFunctionCall(),
|
FunctionCall: buffer.GetFunctionCall(),
|
||||||
},
|
},
|
||||||
|
@ -31,14 +31,6 @@ func NativeChatHandler(c *gin.Context, user *auth.User, model string, message []
|
|||||||
return check.Error(), 0
|
return check.Error(), 0
|
||||||
}
|
}
|
||||||
|
|
||||||
if form := ExtractCacheData(c, &CacheProps{
|
|
||||||
Message: segment,
|
|
||||||
Model: model,
|
|
||||||
Reversible: plan,
|
|
||||||
}); form != nil {
|
|
||||||
return form.Message, 0
|
|
||||||
}
|
|
||||||
|
|
||||||
buffer := utils.NewBuffer(model, segment, channel.ChargeInstance.GetCharge(model))
|
buffer := utils.NewBuffer(model, segment, channel.ChargeInstance.GetCharge(model))
|
||||||
hit, err := channel.NewChatRequestWithCache(
|
hit, err := channel.NewChatRequestWithCache(
|
||||||
cache, buffer,
|
cache, buffer,
|
||||||
@ -64,13 +56,5 @@ func NativeChatHandler(c *gin.Context, user *auth.User, model string, message []
|
|||||||
CollectQuota(c, user, buffer, plan, err)
|
CollectQuota(c, user, buffer, plan, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
SaveCacheData(c, &CacheProps{
|
|
||||||
Message: segment,
|
|
||||||
Model: model,
|
|
||||||
Reversible: plan,
|
|
||||||
}, &CacheData{
|
|
||||||
Message: buffer.ReadWithDefault(defaultMessage),
|
|
||||||
})
|
|
||||||
|
|
||||||
return buffer.ReadWithDefault(defaultMessage), buffer.GetQuota()
|
return buffer.ReadWithDefault(defaultMessage), buffer.GetQuota()
|
||||||
}
|
}
|
||||||
|
@ -23,7 +23,7 @@ type Buffer struct {
|
|||||||
Latest string `json:"latest"`
|
Latest string `json:"latest"`
|
||||||
Cursor int `json:"cursor"`
|
Cursor int `json:"cursor"`
|
||||||
Times int `json:"times"`
|
Times int `json:"times"`
|
||||||
History []globals.Message `json:"history"`
|
InputTokens int `json:"input_tokens"`
|
||||||
Images Images `json:"images"`
|
Images Images `json:"images"`
|
||||||
ToolCalls *globals.ToolCalls `json:"tool_calls"`
|
ToolCalls *globals.ToolCalls `json:"tool_calls"`
|
||||||
ToolCallsCursor int `json:"tool_calls_cursor"`
|
ToolCallsCursor int `json:"tool_calls_cursor"`
|
||||||
@ -31,8 +31,8 @@ type Buffer struct {
|
|||||||
Charge Charge `json:"-"`
|
Charge Charge `json:"-"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func initInputToken(charge Charge, model string, history []globals.Message) float32 {
|
func initInputToken(model string, history []globals.Message) int {
|
||||||
if globals.IsOpenAIVisionModels(model) {
|
if globals.IsVisionModel(model) {
|
||||||
for _, message := range history {
|
for _, message := range history {
|
||||||
if message.Role == globals.User {
|
if message.Role == globals.User {
|
||||||
content, _ := ExtractImages(message.Content, true)
|
content, _ := ExtractImages(message.Content, true)
|
||||||
@ -57,15 +57,20 @@ func initInputToken(charge Charge, model string, history []globals.Message) floa
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
return CountInputToken(charge, model, history)
|
return CountTokenPrice(history, model)
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewBuffer(model string, history []globals.Message, charge Charge) *Buffer {
|
func NewBuffer(model string, history []globals.Message, charge Charge) *Buffer {
|
||||||
|
token := initInputToken(model, history)
|
||||||
|
|
||||||
return &Buffer{
|
return &Buffer{
|
||||||
Model: model,
|
Model: model,
|
||||||
Quota: initInputToken(charge, model, history),
|
Quota: CountInputQuota(charge, token),
|
||||||
History: history,
|
InputTokens: token,
|
||||||
Charge: charge,
|
Charge: charge,
|
||||||
|
FunctionCall: nil,
|
||||||
|
ToolCalls: nil,
|
||||||
|
ToolCallsCursor: 0,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -115,7 +120,6 @@ func (b *Buffer) AddToolCalls(toolCalls *globals.ToolCalls) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
b.ToolCalls = toolCalls
|
b.ToolCalls = toolCalls
|
||||||
b.ToolCallsCursor += 1
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Buffer) SetFunctionCall(functionCall *globals.FunctionCall) {
|
func (b *Buffer) SetFunctionCall(functionCall *globals.FunctionCall) {
|
||||||
@ -177,12 +181,12 @@ func (b *Buffer) ReadTimes() int {
|
|||||||
return b.Times
|
return b.Times
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Buffer) ReadHistory() []globals.Message {
|
func (b *Buffer) SetInputTokens(tokens int) {
|
||||||
return b.History
|
b.InputTokens = tokens
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Buffer) CountInputToken() int {
|
func (b *Buffer) CountInputToken() int {
|
||||||
return GetWeightByModel(b.Model) * NumTokensFromMessages(b.History, b.Model)
|
return b.InputTokens
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Buffer) CountOutputToken() int {
|
func (b *Buffer) CountOutputToken() int {
|
||||||
|
@ -69,12 +69,12 @@ func NumTokensFromMessages(messages []globals.Message, model string) (tokens int
|
|||||||
}
|
}
|
||||||
|
|
||||||
func CountTokenPrice(messages []globals.Message, model string) int {
|
func CountTokenPrice(messages []globals.Message, model string) int {
|
||||||
return NumTokensFromMessages(messages, model)
|
return NumTokensFromMessages(messages, model) * GetWeightByModel(model)
|
||||||
}
|
}
|
||||||
|
|
||||||
func CountInputToken(charge Charge, model string, message []globals.Message) float32 {
|
func CountInputQuota(charge Charge, token int) float32 {
|
||||||
if charge.IsBillingType(globals.TokenBilling) {
|
if charge.GetType() == globals.TokenBilling {
|
||||||
return float32(CountTokenPrice(message, model)) / 1000 * charge.GetInput()
|
return float32(token) / 1000 * charge.GetInput()
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0
|
return 0
|
||||||
|
Loading…
Reference in New Issue
Block a user