|
|
@@ -12,6 +12,7 @@ import (
|
|
|
"one-api/dto"
|
|
|
"one-api/model"
|
|
|
relaycommon "one-api/relay/common"
|
|
|
+ relayconstant "one-api/relay/constant"
|
|
|
"one-api/relay/helper"
|
|
|
"one-api/service"
|
|
|
"one-api/setting"
|
|
|
@@ -20,13 +21,56 @@ import (
|
|
|
|
|
|
func getAndValidImageRequest(c *gin.Context, info *relaycommon.RelayInfo) (*dto.ImageRequest, error) {
|
|
|
imageRequest := &dto.ImageRequest{}
|
|
|
- err := common.UnmarshalBodyReusable(c, imageRequest)
|
|
|
- if err != nil {
|
|
|
- return nil, err
|
|
|
+
|
|
|
+ switch info.RelayMode {
|
|
|
+ case relayconstant.RelayModeImagesEdits:
|
|
|
+ _, err := c.MultipartForm()
|
|
|
+ if err != nil {
|
|
|
+ return nil, err
|
|
|
+ }
|
|
|
+ formData := c.Request.PostForm
|
|
|
+ imageRequest.Prompt = formData.Get("prompt")
|
|
|
+ imageRequest.Model = formData.Get("model")
|
|
|
+ imageRequest.N = common.String2Int(formData.Get("n"))
|
|
|
+ imageRequest.Quality = formData.Get("quality")
|
|
|
+ imageRequest.Size = formData.Get("size")
|
|
|
+
|
|
|
+ if imageRequest.Model == "gpt-image-1" {
|
|
|
+ if imageRequest.Quality == "" {
|
|
|
+ imageRequest.Quality = "standard"
|
|
|
+ }
|
|
|
+ }
|
|
|
+ default:
|
|
|
+ err := common.UnmarshalBodyReusable(c, imageRequest)
|
|
|
+ if err != nil {
|
|
|
+ return nil, err
|
|
|
+ }
|
|
|
+ // Not "256x256", "512x512", or "1024x1024"
|
|
|
+ if imageRequest.Model == "dall-e-2" || imageRequest.Model == "dall-e" {
|
|
|
+ if imageRequest.Size != "" && imageRequest.Size != "256x256" && imageRequest.Size != "512x512" && imageRequest.Size != "1024x1024" {
|
|
|
+ return nil, errors.New("size must be one of 256x256, 512x512, or 1024x1024, dall-e-3 1024x1792 or 1792x1024")
|
|
|
+ }
|
|
|
+ } else if imageRequest.Model == "dall-e-3" {
|
|
|
+ if imageRequest.Size != "" && imageRequest.Size != "1024x1024" && imageRequest.Size != "1024x1792" && imageRequest.Size != "1792x1024" {
|
|
|
+ return nil, errors.New("size must be one of 256x256, 512x512, or 1024x1024, dall-e-3 1024x1792 or 1792x1024")
|
|
|
+ }
|
|
|
+ if imageRequest.Quality == "" {
|
|
|
+ imageRequest.Quality = "standard"
|
|
|
+ }
|
|
|
+ // N should between 1 and 10
|
|
|
+ //if imageRequest.N != 0 && (imageRequest.N < 1 || imageRequest.N > 10) {
|
|
|
+ // return service.OpenAIErrorWrapper(errors.New("n must be between 1 and 10"), "invalid_field_value", http.StatusBadRequest)
|
|
|
+ //}
|
|
|
+ }
|
|
|
}
|
|
|
+
|
|
|
if imageRequest.Prompt == "" {
|
|
|
return nil, errors.New("prompt is required")
|
|
|
}
|
|
|
+
|
|
|
+ if imageRequest.Model == "" {
|
|
|
+ imageRequest.Model = "dall-e-2"
|
|
|
+ }
|
|
|
if strings.Contains(imageRequest.Size, "×") {
|
|
|
return nil, errors.New("size an unexpected error occurred in the parameter, please use 'x' instead of the multiplication sign '×'")
|
|
|
}
|
|
|
@@ -36,30 +80,7 @@ func getAndValidImageRequest(c *gin.Context, info *relaycommon.RelayInfo) (*dto.
|
|
|
if imageRequest.Size == "" {
|
|
|
imageRequest.Size = "1024x1024"
|
|
|
}
|
|
|
- if imageRequest.Model == "" {
|
|
|
- imageRequest.Model = "dall-e-2"
|
|
|
- }
|
|
|
|
|
|
- // Not "256x256", "512x512", or "1024x1024"
|
|
|
- if imageRequest.Model == "dall-e-2" || imageRequest.Model == "dall-e" {
|
|
|
- if imageRequest.Size != "" && imageRequest.Size != "256x256" && imageRequest.Size != "512x512" && imageRequest.Size != "1024x1024" {
|
|
|
- return nil, errors.New("size must be one of 256x256, 512x512, or 1024x1024, dall-e-3 1024x1792 or 1792x1024")
|
|
|
- }
|
|
|
- } else if imageRequest.Model == "dall-e-3" {
|
|
|
- if imageRequest.Size != "" && imageRequest.Size != "1024x1024" && imageRequest.Size != "1024x1792" && imageRequest.Size != "1792x1024" {
|
|
|
- return nil, errors.New("size must be one of 256x256, 512x512, or 1024x1024, dall-e-3 1024x1792 or 1792x1024")
|
|
|
- }
|
|
|
- if imageRequest.Quality == "" {
|
|
|
- imageRequest.Quality = "standard"
|
|
|
- }
|
|
|
- //if imageRequest.N != 1 {
|
|
|
- // return nil, errors.New("n must be 1")
|
|
|
- //}
|
|
|
- }
|
|
|
- // N should between 1 and 10
|
|
|
- //if imageRequest.N != 0 && (imageRequest.N < 1 || imageRequest.N > 10) {
|
|
|
- // return service.OpenAIErrorWrapper(errors.New("n must be between 1 and 10"), "invalid_field_value", http.StatusBadRequest)
|
|
|
- //}
|
|
|
if setting.ShouldCheckPromptSensitive() {
|
|
|
words, err := service.CheckSensitiveInput(imageRequest.Prompt)
|
|
|
if err != nil {
|
|
|
@@ -86,43 +107,59 @@ func ImageHelper(c *gin.Context) *dto.OpenAIErrorWithStatusCode {
|
|
|
|
|
|
imageRequest.Model = relayInfo.UpstreamModelName
|
|
|
|
|
|
- priceData, err := helper.ModelPriceHelper(c, relayInfo, 0, 0)
|
|
|
+ priceData, err := helper.ModelPriceHelper(c, relayInfo, len(imageRequest.Prompt), 0)
|
|
|
if err != nil {
|
|
|
return service.OpenAIErrorWrapperLocal(err, "model_price_error", http.StatusInternalServerError)
|
|
|
}
|
|
|
+ var preConsumedQuota int
|
|
|
+ var quota int
|
|
|
+ var userQuota int
|
|
|
if !priceData.UsePrice {
|
|
|
// modelRatio 16 = modelPrice $0.04
|
|
|
// per 1 modelRatio = $0.04 / 16
|
|
|
- priceData.ModelPrice = 0.0025 * priceData.ModelRatio
|
|
|
- }
|
|
|
-
|
|
|
- userQuota, err := model.GetUserQuota(relayInfo.UserId, false)
|
|
|
-
|
|
|
- sizeRatio := 1.0
|
|
|
- // Size
|
|
|
- if imageRequest.Size == "256x256" {
|
|
|
- sizeRatio = 0.4
|
|
|
- } else if imageRequest.Size == "512x512" {
|
|
|
- sizeRatio = 0.45
|
|
|
- } else if imageRequest.Size == "1024x1024" {
|
|
|
- sizeRatio = 1
|
|
|
- } else if imageRequest.Size == "1024x1792" || imageRequest.Size == "1792x1024" {
|
|
|
- sizeRatio = 2
|
|
|
- }
|
|
|
-
|
|
|
- qualityRatio := 1.0
|
|
|
- if imageRequest.Model == "dall-e-3" && imageRequest.Quality == "hd" {
|
|
|
- qualityRatio = 2.0
|
|
|
- if imageRequest.Size == "1024x1792" || imageRequest.Size == "1792x1024" {
|
|
|
- qualityRatio = 1.5
|
|
|
+ // priceData.ModelPrice = 0.0025 * priceData.ModelRatio
|
|
|
+ var openaiErr *dto.OpenAIErrorWithStatusCode
|
|
|
+ preConsumedQuota, userQuota, openaiErr = preConsumeQuota(c, priceData.ShouldPreConsumedQuota, relayInfo)
|
|
|
+ if openaiErr != nil {
|
|
|
+ return openaiErr
|
|
|
+ }
|
|
|
+ defer func() {
|
|
|
+ if openaiErr != nil {
|
|
|
+ returnPreConsumedQuota(c, relayInfo, userQuota, preConsumedQuota)
|
|
|
+ }
|
|
|
+ }()
|
|
|
+
|
|
|
+ } else {
|
|
|
+ sizeRatio := 1.0
|
|
|
+ // Size
|
|
|
+ if imageRequest.Size == "256x256" {
|
|
|
+ sizeRatio = 0.4
|
|
|
+ } else if imageRequest.Size == "512x512" {
|
|
|
+ sizeRatio = 0.45
|
|
|
+ } else if imageRequest.Size == "1024x1024" {
|
|
|
+ sizeRatio = 1
|
|
|
+ } else if imageRequest.Size == "1024x1792" || imageRequest.Size == "1792x1024" {
|
|
|
+ sizeRatio = 2
|
|
|
}
|
|
|
- }
|
|
|
|
|
|
- priceData.ModelPrice *= sizeRatio * qualityRatio * float64(imageRequest.N)
|
|
|
- quota := int(priceData.ModelPrice * priceData.GroupRatio * common.QuotaPerUnit)
|
|
|
+ qualityRatio := 1.0
|
|
|
+ if imageRequest.Model == "dall-e-3" && imageRequest.Quality == "hd" {
|
|
|
+ qualityRatio = 2.0
|
|
|
+ if imageRequest.Size == "1024x1792" || imageRequest.Size == "1792x1024" {
|
|
|
+ qualityRatio = 1.5
|
|
|
+ }
|
|
|
+ }
|
|
|
|
|
|
- if userQuota-quota < 0 {
|
|
|
- return service.OpenAIErrorWrapperLocal(fmt.Errorf("image pre-consumed quota failed, user quota: %s, need quota: %s", common.FormatQuota(userQuota), common.FormatQuota(quota)), "insufficient_user_quota", http.StatusForbidden)
|
|
|
+ // reset model price
|
|
|
+ priceData.ModelPrice *= sizeRatio * qualityRatio * float64(imageRequest.N)
|
|
|
+ quota = int(priceData.ModelPrice * priceData.GroupRatio * common.QuotaPerUnit)
|
|
|
+ userQuota, err = model.GetUserQuota(relayInfo.UserId, false)
|
|
|
+ if err != nil {
|
|
|
+ return service.OpenAIErrorWrapperLocal(err, "get_user_quota_failed", http.StatusInternalServerError)
|
|
|
+ }
|
|
|
+ if userQuota-quota < 0 {
|
|
|
+ return service.OpenAIErrorWrapperLocal(fmt.Errorf("image pre-consumed quota failed, user quota: %s, need quota: %s", common.FormatQuota(userQuota), common.FormatQuota(quota)), "insufficient_user_quota", http.StatusForbidden)
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
adaptor := GetAdaptor(relayInfo.ApiType)
|
|
|
@@ -137,12 +174,15 @@ func ImageHelper(c *gin.Context) *dto.OpenAIErrorWithStatusCode {
|
|
|
if err != nil {
|
|
|
return service.OpenAIErrorWrapperLocal(err, "convert_request_failed", http.StatusInternalServerError)
|
|
|
}
|
|
|
-
|
|
|
- jsonData, err := json.Marshal(convertedRequest)
|
|
|
- if err != nil {
|
|
|
- return service.OpenAIErrorWrapperLocal(err, "json_marshal_failed", http.StatusInternalServerError)
|
|
|
+ if relayInfo.RelayMode == relayconstant.RelayModeImagesEdits {
|
|
|
+ requestBody = convertedRequest.(io.Reader)
|
|
|
+ } else {
|
|
|
+ jsonData, err := json.Marshal(convertedRequest)
|
|
|
+ if err != nil {
|
|
|
+ return service.OpenAIErrorWrapperLocal(err, "json_marshal_failed", http.StatusInternalServerError)
|
|
|
+ }
|
|
|
+ requestBody = bytes.NewBuffer(jsonData)
|
|
|
}
|
|
|
- requestBody = bytes.NewBuffer(jsonData)
|
|
|
|
|
|
statusCodeMappingStr := c.GetString("status_code_mapping")
|
|
|
|
|
|
@@ -162,24 +202,25 @@ func ImageHelper(c *gin.Context) *dto.OpenAIErrorWithStatusCode {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- _, openaiErr := adaptor.DoResponse(c, httpResp, relayInfo)
|
|
|
+ usage, openaiErr := adaptor.DoResponse(c, httpResp, relayInfo)
|
|
|
if openaiErr != nil {
|
|
|
// reset status code 重置状态码
|
|
|
service.ResetStatusCode(openaiErr, statusCodeMappingStr)
|
|
|
return openaiErr
|
|
|
}
|
|
|
|
|
|
- usage := &dto.Usage{
|
|
|
- PromptTokens: imageRequest.N,
|
|
|
- TotalTokens: imageRequest.N,
|
|
|
+ if usage.(*dto.Usage).TotalTokens == 0 {
|
|
|
+ usage.(*dto.Usage).TotalTokens = imageRequest.N
|
|
|
+ }
|
|
|
+ if usage.(*dto.Usage).PromptTokens == 0 {
|
|
|
+ usage.(*dto.Usage).PromptTokens = imageRequest.N
|
|
|
}
|
|
|
-
|
|
|
quality := "standard"
|
|
|
if imageRequest.Quality == "hd" {
|
|
|
quality = "hd"
|
|
|
}
|
|
|
|
|
|
logContent := fmt.Sprintf("大小 %s, 品质 %s", imageRequest.Size, quality)
|
|
|
- postConsumeQuota(c, relayInfo, usage, 0, userQuota, priceData, logContent)
|
|
|
+ postConsumeQuota(c, relayInfo, usage.(*dto.Usage), preConsumedQuota, userQuota, priceData, logContent)
|
|
|
return nil
|
|
|
}
|