2023-04-23 18:24:11 +08:00
|
|
|
package middleware
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"fmt"
|
|
|
|
|
"net/http"
|
|
|
|
|
"one-api/common"
|
|
|
|
|
"one-api/model"
|
|
|
|
|
"strconv"
|
2023-06-11 09:37:36 +08:00
|
|
|
"strings"
|
2023-07-15 12:03:23 +08:00
|
|
|
|
|
|
|
|
"github.com/gin-gonic/gin"
|
2023-04-23 18:24:11 +08:00
|
|
|
)
|
|
|
|
|
|
2023-06-07 23:26:00 +08:00
|
|
|
type ModelRequest struct {
|
|
|
|
|
Model string `json:"model"`
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-23 18:24:11 +08:00
|
|
|
func Distribute() func(c *gin.Context) {
|
|
|
|
|
return func(c *gin.Context) {
|
2023-06-11 11:08:16 +08:00
|
|
|
userId := c.GetInt("id")
|
2023-06-20 19:09:49 +08:00
|
|
|
userGroup, _ := model.CacheGetUserGroup(userId)
|
2023-06-11 11:08:16 +08:00
|
|
|
c.Set("group", userGroup)
|
2023-04-23 18:24:11 +08:00
|
|
|
var channel *model.Channel
|
|
|
|
|
channelId, ok := c.Get("channelId")
|
|
|
|
|
if ok {
|
|
|
|
|
id, err := strconv.Atoi(channelId.(string))
|
|
|
|
|
if err != nil {
|
2023-10-03 14:19:03 +08:00
|
|
|
abortWithMessage(c, http.StatusBadRequest, "无效的渠道 Id")
|
2023-04-23 18:24:11 +08:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
channel, err = model.GetChannelById(id, true)
|
|
|
|
|
if err != nil {
|
2023-10-03 14:19:03 +08:00
|
|
|
abortWithMessage(c, http.StatusBadRequest, "无效的渠道 Id")
|
2023-04-23 18:24:11 +08:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
if channel.Status != common.ChannelStatusEnabled {
|
2023-09-17 15:39:46 +08:00
|
|
|
abortWithMessage(c, http.StatusForbidden, "该渠道已被禁用")
|
2023-04-23 18:24:11 +08:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// Select a channel for the user
|
2023-06-07 23:26:00 +08:00
|
|
|
var modelRequest ModelRequest
|
2023-08-27 15:28:23 +08:00
|
|
|
var err error
|
2023-08-14 22:16:32 +08:00
|
|
|
if strings.HasPrefix(c.Request.URL.Path, "/mj") {
|
2023-09-09 04:29:00 +08:00
|
|
|
// Midjourney
|
2023-08-14 22:16:32 +08:00
|
|
|
if modelRequest.Model == "" {
|
|
|
|
|
modelRequest.Model = "midjourney"
|
|
|
|
|
}
|
2023-11-19 16:35:37 +08:00
|
|
|
} else if !strings.HasPrefix(c.Request.URL.Path, "/v1/audio/transcriptions") {
|
|
|
|
|
err = common.UnmarshalBodyReusable(c, &modelRequest)
|
2023-08-27 15:28:23 +08:00
|
|
|
}
|
2023-06-07 23:26:00 +08:00
|
|
|
if err != nil {
|
2023-09-17 15:39:46 +08:00
|
|
|
abortWithMessage(c, http.StatusBadRequest, "无效的请求")
|
2023-06-07 23:26:00 +08:00
|
|
|
return
|
|
|
|
|
}
|
2023-06-11 09:37:36 +08:00
|
|
|
if strings.HasPrefix(c.Request.URL.Path, "/v1/moderations") {
|
|
|
|
|
if modelRequest.Model == "" {
|
|
|
|
|
modelRequest.Model = "text-moderation-stable"
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-07-15 12:03:23 +08:00
|
|
|
if strings.HasSuffix(c.Request.URL.Path, "embeddings") {
|
|
|
|
|
if modelRequest.Model == "" {
|
|
|
|
|
modelRequest.Model = c.Param("model")
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-07-15 12:30:06 +08:00
|
|
|
if strings.HasPrefix(c.Request.URL.Path, "/v1/images/generations") {
|
|
|
|
|
if modelRequest.Model == "" {
|
|
|
|
|
modelRequest.Model = "dall-e"
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-08-27 15:28:23 +08:00
|
|
|
if strings.HasPrefix(c.Request.URL.Path, "/v1/audio") {
|
|
|
|
|
if modelRequest.Model == "" {
|
2023-11-15 21:05:14 +08:00
|
|
|
if strings.HasPrefix(c.Request.URL.Path, "/v1/audio/speech") {
|
|
|
|
|
modelRequest.Model = "tts-1"
|
|
|
|
|
} else {
|
|
|
|
|
modelRequest.Model = "whisper-1"
|
|
|
|
|
}
|
2023-08-27 15:28:23 +08:00
|
|
|
}
|
|
|
|
|
}
|
2023-09-09 04:29:00 +08:00
|
|
|
channel, err = model.CacheGetRandomSatisfiedChannel(userGroup, modelRequest.Model)
|
2023-04-23 18:24:11 +08:00
|
|
|
if err != nil {
|
2023-07-23 09:49:46 +08:00
|
|
|
message := fmt.Sprintf("当前分组 %s 下对于模型 %s 无可用渠道", userGroup, modelRequest.Model)
|
2023-09-09 03:11:42 +08:00
|
|
|
if channel != nil {
|
|
|
|
|
common.SysError(fmt.Sprintf("渠道不存在:%d", channel.Id))
|
|
|
|
|
message = "数据库一致性已被破坏,请联系管理员"
|
2023-06-29 11:27:34 +08:00
|
|
|
}
|
2023-09-17 15:39:46 +08:00
|
|
|
abortWithMessage(c, http.StatusServiceUnavailable, message)
|
2023-09-09 03:11:42 +08:00
|
|
|
return
|
2023-04-23 18:24:11 +08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
c.Set("channel", channel.Type)
|
2023-05-15 17:34:09 +08:00
|
|
|
c.Set("channel_id", channel.Id)
|
|
|
|
|
c.Set("channel_name", channel.Name)
|
2023-09-25 18:53:59 +08:00
|
|
|
ban := true
|
|
|
|
|
// parse *int to bool
|
|
|
|
|
if channel.AutoBan != nil && *channel.AutoBan == 0 {
|
|
|
|
|
ban = false
|
|
|
|
|
}
|
|
|
|
|
c.Set("auto_ban", ban)
|
2023-09-18 22:07:17 +08:00
|
|
|
c.Set("model_mapping", channel.GetModelMapping())
|
2023-04-23 18:24:11 +08:00
|
|
|
c.Request.Header.Set("Authorization", fmt.Sprintf("Bearer %s", channel.Key))
|
2023-09-18 22:07:17 +08:00
|
|
|
c.Set("base_url", channel.GetBaseURL())
|
2023-09-03 12:51:59 +08:00
|
|
|
switch channel.Type {
|
|
|
|
|
case common.ChannelTypeAzure:
|
2023-06-09 18:30:01 +08:00
|
|
|
c.Set("api_version", channel.Other)
|
2023-09-03 12:51:59 +08:00
|
|
|
case common.ChannelTypeXunfei:
|
2023-06-09 18:30:01 +08:00
|
|
|
c.Set("api_version", channel.Other)
|
2023-09-03 12:51:59 +08:00
|
|
|
case common.ChannelTypeAIProxyLibrary:
|
|
|
|
|
c.Set("library_id", channel.Other)
|
2023-04-23 20:35:49 +08:00
|
|
|
}
|
2023-04-23 18:24:11 +08:00
|
|
|
c.Next()
|
|
|
|
|
}
|
|
|
|
|
}
|