refactor: 重构Gemini适配器以支持多模态输入 fix: 修复React Hooks依赖警告 style: 清理未使用的导入和代码 docs: 更新用户界面文本和提示 perf: 优化图像和视频URL处理性能 test: 添加数据迁移工具和测试 build: 更新依赖项和.gitignore chore: 同步Zenmux模型和价格比例
179 lines
4.8 KiB
Go
179 lines
4.8 KiB
Go
package gemini
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/gin-gonic/gin"
|
|
"github.com/songquanpeng/one-api/common/config"
|
|
"github.com/songquanpeng/one-api/common/helper"
|
|
channelhelper "github.com/songquanpeng/one-api/relay/adaptor"
|
|
"github.com/songquanpeng/one-api/relay/adaptor/openai"
|
|
"github.com/songquanpeng/one-api/relay/meta"
|
|
"github.com/songquanpeng/one-api/relay/model"
|
|
"github.com/songquanpeng/one-api/relay/relaymode"
|
|
)
|
|
|
|
type Adaptor struct {
|
|
}
|
|
|
|
func (a *Adaptor) Init(meta *meta.Meta) {
|
|
}
|
|
|
|
func (a *Adaptor) GetRequestURL(meta *meta.Meta) (string, error) {
|
|
defaultVersion := config.GeminiVersion
|
|
modelLower := strings.ToLower(meta.ActualModelName)
|
|
if strings.Contains(modelLower, "gemini-1.5") ||
|
|
strings.Contains(modelLower, "gemini-2.") ||
|
|
strings.Contains(modelLower, "gemini-3.") {
|
|
defaultVersion = "v1beta"
|
|
}
|
|
|
|
version := helper.AssignOrDefault(meta.Config.APIVersion, defaultVersion)
|
|
|
|
modelName := meta.ActualModelName
|
|
|
|
switch meta.Mode {
|
|
case relaymode.ImagesGenerations:
|
|
return fmt.Sprintf("%s/v1beta/models/%s:predict", meta.BaseURL, modelName), nil
|
|
case relaymode.Embeddings:
|
|
return fmt.Sprintf("%s/%s/models/%s:batchEmbedContents", meta.BaseURL, version, modelName), nil
|
|
}
|
|
|
|
action := "generateContent"
|
|
if meta.IsStream {
|
|
action = "streamGenerateContent?alt=sse"
|
|
}
|
|
return fmt.Sprintf("%s/%s/models/%s:%s", meta.BaseURL, version, modelName, action), nil
|
|
}
|
|
|
|
func (a *Adaptor) SetupRequestHeader(c *gin.Context, req *http.Request, meta *meta.Meta) error {
|
|
channelhelper.SetupCommonRequestHeader(c, req, meta)
|
|
req.Header.Set("x-goog-api-key", meta.APIKey)
|
|
return nil
|
|
}
|
|
|
|
func (a *Adaptor) ConvertRequest(c *gin.Context, relayMode int, request *model.GeneralOpenAIRequest) (any, error) {
|
|
if request == nil {
|
|
return nil, errors.New("request is nil")
|
|
}
|
|
switch relayMode {
|
|
case relaymode.Embeddings:
|
|
geminiEmbeddingRequest := ConvertEmbeddingRequest(*request)
|
|
return geminiEmbeddingRequest, nil
|
|
default:
|
|
geminiRequest := ConvertRequest(*request)
|
|
return geminiRequest, nil
|
|
}
|
|
}
|
|
|
|
func (a *Adaptor) ConvertImageRequest(request *model.ImageRequest) (any, error) {
|
|
if request == nil {
|
|
return nil, errors.New("request is nil")
|
|
}
|
|
n := request.N
|
|
if n <= 0 {
|
|
n = 1
|
|
}
|
|
return ImagenRequest{
|
|
Instances: []ImagenInstance{{Prompt: request.Prompt}},
|
|
Parameters: ImagenParameters{
|
|
SampleCount: n,
|
|
AspectRatio: sizeToAspectRatio(request.Size),
|
|
},
|
|
}, nil
|
|
}
|
|
|
|
// sizeToAspectRatio converts OpenAI size strings to Imagen aspect ratios.
|
|
func sizeToAspectRatio(size string) string {
|
|
switch size {
|
|
case "1792x1024":
|
|
return "16:9"
|
|
case "1024x1792":
|
|
return "9:16"
|
|
case "1024x1024", "":
|
|
return "1:1"
|
|
default:
|
|
return "1:1"
|
|
}
|
|
}
|
|
|
|
func (a *Adaptor) DoRequest(c *gin.Context, meta *meta.Meta, requestBody io.Reader) (*http.Response, error) {
|
|
return channelhelper.DoRequestHelper(a, c, meta, requestBody)
|
|
}
|
|
|
|
func (a *Adaptor) DoResponse(c *gin.Context, resp *http.Response, meta *meta.Meta) (usage *model.Usage, err *model.ErrorWithStatusCode) {
|
|
if meta.IsStream {
|
|
err, usage = StreamHandler(c, resp)
|
|
if usage != nil {
|
|
usage.PromptTokens = meta.PromptTokens
|
|
}
|
|
} else {
|
|
switch meta.Mode {
|
|
case relaymode.Embeddings:
|
|
err, usage = EmbeddingHandler(c, resp)
|
|
case relaymode.ImagesGenerations:
|
|
err = ImagenHandler(c, resp)
|
|
default:
|
|
err, usage = Handler(c, resp, meta.PromptTokens, meta.ActualModelName)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
// ImagenHandler converts a Google Imagen predict response to OpenAI image response format.
|
|
func ImagenHandler(c *gin.Context, resp *http.Response) *model.ErrorWithStatusCode {
|
|
responseBody, readErr := io.ReadAll(resp.Body)
|
|
resp.Body.Close()
|
|
if readErr != nil {
|
|
return openai.ErrorWrapper(readErr, "read_response_body_failed", http.StatusInternalServerError)
|
|
}
|
|
|
|
var imagenResp ImagenResponse
|
|
if err := json.Unmarshal(responseBody, &imagenResp); err != nil {
|
|
return openai.ErrorWrapper(err, "unmarshal_response_body_failed", http.StatusInternalServerError)
|
|
}
|
|
|
|
if imagenResp.Error != nil {
|
|
return openai.ErrorWrapper(
|
|
fmt.Errorf("%s", imagenResp.Error.Message),
|
|
"imagen_error",
|
|
http.StatusInternalServerError,
|
|
)
|
|
}
|
|
|
|
// Convert to OpenAI image response format
|
|
data := make([]openai.ImageData, 0, len(imagenResp.Predictions))
|
|
for _, p := range imagenResp.Predictions {
|
|
data = append(data, openai.ImageData{B64Json: p.BytesBase64Encoded})
|
|
}
|
|
openaiResp := openai.ImageResponse{
|
|
Created: time.Now().Unix(),
|
|
Data: data,
|
|
}
|
|
|
|
jsonBytes, err := json.Marshal(openaiResp)
|
|
if err != nil {
|
|
return openai.ErrorWrapper(err, "marshal_response_failed", http.StatusInternalServerError)
|
|
}
|
|
|
|
c.Writer.Header().Set("Content-Type", "application/json")
|
|
c.Writer.WriteHeader(http.StatusOK)
|
|
c.Writer.Write(jsonBytes)
|
|
return nil
|
|
}
|
|
|
|
func (a *Adaptor) GetModelList() []string {
|
|
return ModelList
|
|
}
|
|
|
|
func (a *Adaptor) GetChannelName() string {
|
|
return "google gemini"
|
|
}
|
|
|