new-api/relay/relay-mj.go

544 lines
17 KiB
Go
Raw Normal View History

2024-02-29 01:08:18 +08:00
package relay
2023-08-14 22:16:32 +08:00
import (
"bytes"
2023-09-17 21:07:00 +08:00
"context"
2023-08-14 22:16:32 +08:00
"encoding/json"
"fmt"
"io"
"log"
"net/http"
"one-api/common"
"one-api/constant"
2024-02-29 16:21:25 +08:00
"one-api/dto"
2023-08-14 22:16:32 +08:00
"one-api/model"
2024-02-29 16:21:25 +08:00
relayconstant "one-api/relay/constant"
"one-api/service"
2023-09-12 03:17:26 +08:00
"strconv"
2023-08-14 22:16:32 +08:00
"strings"
2023-11-06 02:08:12 +08:00
"time"
2023-08-14 22:16:32 +08:00
"github.com/gin-gonic/gin"
)
func RelayMidjourneyImage(c *gin.Context) {
taskId := c.Param("id")
midjourneyTask := model.GetByOnlyMJId(taskId)
if midjourneyTask == nil {
c.JSON(400, gin.H{
"error": "midjourney_task_not_found",
})
return
}
resp, err := http.Get(midjourneyTask.ImageUrl)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{
"error": "http_get_image_failed",
})
2024-01-14 16:17:45 +08:00
return
}
defer resp.Body.Close()
if resp.StatusCode != http.StatusOK {
responseBody, _ := io.ReadAll(resp.Body)
c.JSON(resp.StatusCode, gin.H{
"error": string(responseBody),
})
return
}
// 从Content-Type头获取MIME类型
contentType := resp.Header.Get("Content-Type")
if contentType == "" {
// 如果无法确定内容类型则默认为jpeg
contentType = "image/jpeg"
}
// 设置响应的内容类型
c.Writer.Header().Set("Content-Type", contentType)
// 将图片流式传输到响应体
_, err = io.Copy(c.Writer, resp.Body)
if err != nil {
log.Println("Failed to stream image:", err)
}
return
}
2024-02-29 16:21:25 +08:00
func RelayMidjourneyNotify(c *gin.Context) *dto.MidjourneyResponse {
var midjRequest dto.MidjourneyDto
2023-08-14 22:16:32 +08:00
err := common.UnmarshalBodyReusable(c, &midjRequest)
if err != nil {
2024-02-29 16:21:25 +08:00
return &dto.MidjourneyResponse{
2023-08-14 22:16:32 +08:00
Code: 4,
Description: "bind_request_body_failed",
Properties: nil,
Result: "",
}
}
midjourneyTask := model.GetByOnlyMJId(midjRequest.MjId)
2023-08-14 22:16:32 +08:00
if midjourneyTask == nil {
2024-02-29 16:21:25 +08:00
return &dto.MidjourneyResponse{
2023-08-14 22:16:32 +08:00
Code: 4,
Description: "midjourney_task_not_found",
Properties: nil,
Result: "",
}
}
midjourneyTask.Progress = midjRequest.Progress
midjourneyTask.PromptEn = midjRequest.PromptEn
midjourneyTask.State = midjRequest.State
midjourneyTask.SubmitTime = midjRequest.SubmitTime
midjourneyTask.StartTime = midjRequest.StartTime
midjourneyTask.FinishTime = midjRequest.FinishTime
midjourneyTask.ImageUrl = midjRequest.ImageUrl
midjourneyTask.Status = midjRequest.Status
midjourneyTask.FailReason = midjRequest.FailReason
err = midjourneyTask.Update()
if err != nil {
2024-02-29 16:21:25 +08:00
return &dto.MidjourneyResponse{
2023-08-14 22:16:32 +08:00
Code: 4,
Description: "update_midjourney_task_failed",
}
}
return nil
}
2024-03-13 16:19:22 +08:00
func coverMidjourneyTaskDto(c *gin.Context, originTask *model.Midjourney) (midjourneyTask dto.MidjourneyDto) {
2023-08-14 22:16:32 +08:00
midjourneyTask.MjId = originTask.MjId
midjourneyTask.Progress = originTask.Progress
midjourneyTask.PromptEn = originTask.PromptEn
midjourneyTask.State = originTask.State
midjourneyTask.SubmitTime = originTask.SubmitTime
midjourneyTask.StartTime = originTask.StartTime
midjourneyTask.FinishTime = originTask.FinishTime
2023-11-06 02:08:12 +08:00
midjourneyTask.ImageUrl = ""
if originTask.ImageUrl != "" {
midjourneyTask.ImageUrl = common.ServerAddress + "/mj/image/" + originTask.MjId
if originTask.Status != "SUCCESS" {
midjourneyTask.ImageUrl += "?rand=" + strconv.FormatInt(time.Now().UnixNano(), 10)
}
}
2023-08-14 22:16:32 +08:00
midjourneyTask.Status = originTask.Status
midjourneyTask.FailReason = originTask.FailReason
midjourneyTask.Action = originTask.Action
midjourneyTask.Description = originTask.Description
midjourneyTask.Prompt = originTask.Prompt
if originTask.Buttons != "" {
var buttons []dto.ActionButton
err := json.Unmarshal([]byte(originTask.Buttons), &buttons)
if err == nil {
midjourneyTask.Buttons = buttons
}
}
2024-03-13 17:46:34 +08:00
if originTask.Properties != "" {
var properties dto.Properties
err := json.Unmarshal([]byte(originTask.Properties), &properties)
if err == nil {
midjourneyTask.Properties = &properties
}
}
return
}
2024-02-29 16:21:25 +08:00
func RelayMidjourneyTask(c *gin.Context, relayMode int) *dto.MidjourneyResponse {
userId := c.GetInt("id")
var err error
var respBody []byte
switch relayMode {
2024-02-29 16:21:25 +08:00
case relayconstant.RelayModeMidjourneyTaskFetch:
taskId := c.Param("id")
originTask := model.GetByMJId(userId, taskId)
if originTask == nil {
2024-02-29 16:21:25 +08:00
return &dto.MidjourneyResponse{
Code: 4,
Description: "task_no_found",
}
}
2024-03-13 16:19:22 +08:00
midjourneyTask := coverMidjourneyTaskDto(c, originTask)
respBody, err = json.Marshal(midjourneyTask)
if err != nil {
2024-02-29 16:21:25 +08:00
return &dto.MidjourneyResponse{
Code: 4,
Description: "unmarshal_response_body_failed",
}
}
2024-02-29 16:21:25 +08:00
case relayconstant.RelayModeMidjourneyTaskFetchByCondition:
var condition = struct {
IDs []string `json:"ids"`
}{}
err = c.BindJSON(&condition)
if err != nil {
2024-02-29 16:21:25 +08:00
return &dto.MidjourneyResponse{
Code: 4,
Description: "do_request_failed",
}
}
var tasks []dto.MidjourneyDto
if len(condition.IDs) != 0 {
originTasks := model.GetByMJIds(userId, condition.IDs)
for _, originTask := range originTasks {
2024-03-13 16:19:22 +08:00
midjourneyTask := coverMidjourneyTaskDto(c, originTask)
tasks = append(tasks, midjourneyTask)
}
}
if tasks == nil {
tasks = make([]dto.MidjourneyDto, 0)
}
respBody, err = json.Marshal(tasks)
if err != nil {
2024-02-29 16:21:25 +08:00
return &dto.MidjourneyResponse{
Code: 4,
Description: "unmarshal_response_body_failed",
}
2023-08-14 22:16:32 +08:00
}
}
c.Writer.Header().Set("Content-Type", "application/json")
_, err = io.Copy(c.Writer, bytes.NewBuffer(respBody))
2023-08-14 22:16:32 +08:00
if err != nil {
2024-02-29 16:21:25 +08:00
return &dto.MidjourneyResponse{
2023-08-14 22:16:32 +08:00
Code: 4,
Description: "copy_response_body_failed",
}
}
return nil
}
2024-02-29 16:21:25 +08:00
func RelayMidjourneySubmit(c *gin.Context, relayMode int) *dto.MidjourneyResponse {
2023-08-14 22:16:32 +08:00
tokenId := c.GetInt("token_id")
//channelType := c.GetInt("channel")
2023-08-14 22:16:32 +08:00
userId := c.GetInt("id")
group := c.GetString("group")
2023-09-17 21:07:00 +08:00
channelId := c.GetInt("channel_id")
consumeQuota := true
2024-02-29 16:21:25 +08:00
var midjRequest dto.MidjourneyRequest
err := common.UnmarshalBodyReusable(c, &midjRequest)
if err != nil {
return service.MidjourneyErrorWrapper(constant.MjRequestError, "bind_request_body_failed")
}
if relayMode == relayconstant.RelayModeMidjourneyAction { // midjourney plus需要从customId中获取任务信息
mjErr := service.CoverPlusActionToNormalAction(&midjRequest)
if mjErr != nil {
return mjErr
2023-08-14 22:16:32 +08:00
}
relayMode = relayconstant.RelayModeMidjourneyChange
2023-08-14 22:16:32 +08:00
}
2024-02-29 16:21:25 +08:00
if relayMode == relayconstant.RelayModeMidjourneyImagine { //绘画任务,此类任务可重复
2023-08-14 22:16:32 +08:00
if midjRequest.Prompt == "" {
return service.MidjourneyErrorWrapper(constant.MjRequestError, "prompt_is_required")
2023-08-14 22:16:32 +08:00
}
2024-03-13 17:46:34 +08:00
midjRequest.Action = constant.MjActionImagine
2024-02-29 16:21:25 +08:00
} else if relayMode == relayconstant.RelayModeMidjourneyDescribe { //按图生文任务,此类任务可重复
2024-03-13 17:46:34 +08:00
midjRequest.Action = constant.MjActionDescribe
} else if relayMode == relayconstant.RelayModeMidjourneyShorten { //缩短任务此类任务可重复plus only
midjRequest.Action = constant.MjActionShorten
2024-02-29 16:21:25 +08:00
} else if relayMode == relayconstant.RelayModeMidjourneyBlend { //绘画任务,此类任务可重复
2024-03-13 18:26:16 +08:00
midjRequest.Action = constant.MjActionBlend
2023-11-06 02:08:12 +08:00
} else if midjRequest.TaskId != "" { //放大、变换任务此类任务如果重复且已有结果远端api会直接返回最终结果
mjId := ""
2024-02-29 16:21:25 +08:00
if relayMode == relayconstant.RelayModeMidjourneyChange {
if midjRequest.TaskId == "" {
return service.MidjourneyErrorWrapper(constant.MjRequestError, "task_id_is_required")
} else if midjRequest.Action == "" {
return service.MidjourneyErrorWrapper(constant.MjRequestError, "action_is_required")
} else if midjRequest.Index == 0 {
return service.MidjourneyErrorWrapper(constant.MjRequestError, "index_is_required")
}
//action = midjRequest.Action
mjId = midjRequest.TaskId
2024-02-29 16:21:25 +08:00
} else if relayMode == relayconstant.RelayModeMidjourneySimpleChange {
if midjRequest.Content == "" {
return service.MidjourneyErrorWrapper(constant.MjRequestError, "content_is_required")
}
params := service.ConvertSimpleChangeParams(midjRequest.Content)
if params == nil {
return service.MidjourneyErrorWrapper(constant.MjRequestError, "content_parse_failed")
}
mjId = params.TaskId
midjRequest.Action = params.Action
} else if relayMode == relayconstant.RelayModeMidjourneyModal {
if midjRequest.MaskBase64 == "" {
return service.MidjourneyErrorWrapper(constant.MjRequestError, "mask_base64_is_required")
}
mjId = midjRequest.TaskId
2024-03-13 17:46:34 +08:00
midjRequest.Action = constant.MjActionInPaint
}
originTask := model.GetByMJId(userId, mjId)
2023-08-14 22:16:32 +08:00
if originTask == nil {
return service.MidjourneyErrorWrapper(constant.MjRequestError, "task_not_found")
} else if originTask.Status != "SUCCESS" && relayMode != relayconstant.RelayModeMidjourneyModal {
return service.MidjourneyErrorWrapper(constant.MjRequestError, "task_status_not_success")
2023-11-06 02:08:12 +08:00
} else { //原任务的Status=SUCCESS则可以做放大UPSCALE、变换VARIATION等动作此时必须使用原来的请求地址才能正确处理
channel, err := model.GetChannelById(originTask.ChannelId, false)
if err != nil {
return service.MidjourneyErrorWrapper(constant.MjRequestError, "get_channel_info_failed")
2023-11-06 02:08:12 +08:00
}
if channel.Status != common.ChannelStatusEnabled {
return service.MidjourneyErrorWrapper(constant.MjRequestError, "该任务所属渠道已被禁用")
}
2023-11-06 02:08:12 +08:00
c.Set("base_url", channel.GetBaseURL())
c.Set("channel_id", originTask.ChannelId)
log.Printf("检测到此操作为放大、变换、重绘获取原channel信息: %s,%s", strconv.Itoa(originTask.ChannelId), channel.GetBaseURL())
2023-08-14 22:16:32 +08:00
}
midjRequest.Prompt = originTask.Prompt
//if channelType == common.ChannelTypeMidjourneyPlus {
// // plus
//} else {
// // 普通版渠道
//
//}
}
if midjRequest.Action == constant.MjActionInPaintPre {
consumeQuota = false
2023-08-14 22:16:32 +08:00
}
// map model name
//modelMapping := c.GetString("model_mapping")
//isModelMapped := false
//if modelMapping != "" {
// modelMap := make(map[string]string)
// err := json.Unmarshal([]byte(modelMapping), &modelMap)
// if err != nil {
// //return errorWrapper(err, "unmarshal_model_mapping_failed", http.StatusInternalServerError)
// return &dto.MidjourneyResponse{
// Code: 4,
// Description: "unmarshal_model_mapping_failed",
// }
// }
// if modelMap[imageModel] != "" {
// imageModel = modelMap[imageModel]
// isModelMapped = true
// }
//}
//baseURL := common.ChannelBaseURLs[channelType]
2023-08-14 22:16:32 +08:00
requestURL := c.Request.URL.String()
baseURL := c.GetString("base_url")
2023-08-14 22:16:32 +08:00
//midjRequest.NotifyHook = "http://127.0.0.1:3000/mj/notify"
fullRequestURL := fmt.Sprintf("%s%s", baseURL, requestURL)
var requestBody io.Reader
//if isModelMapped {
// jsonStr, err := json.Marshal(midjRequest)
// if err != nil {
// return &dto.MidjourneyResponse{
// Code: 4,
// Description: "marshal_text_request_failed",
// }
// }
// requestBody = bytes.NewBuffer(jsonStr)
//} else {
//}
requestBody = c.Request.Body
modelName := service.CoverActionToModelName(midjRequest.Action)
modelPrice := common.GetModelPrice(modelName, true)
// 如果没有配置价格,则使用默认价格
2024-01-05 12:23:26 +08:00
if modelPrice == -1 {
defaultPrice, ok := common.DefaultModelPrice[modelName]
2024-01-05 12:23:26 +08:00
if !ok {
modelPrice = 0.1
} else {
modelPrice = defaultPrice
}
}
2023-08-14 22:16:32 +08:00
groupRatio := common.GetGroupRatio(group)
ratio := modelPrice * groupRatio
2023-08-14 22:16:32 +08:00
userQuota, err := model.CacheGetUserQuota(userId)
if err != nil {
2024-02-29 16:21:25 +08:00
return &dto.MidjourneyResponse{
Code: 4,
Description: err.Error(),
}
2023-09-22 23:34:17 +08:00
}
quota := int(ratio * common.QuotaPerUnit)
2023-08-14 22:16:32 +08:00
if consumeQuota && userQuota-quota < 0 {
2024-02-29 16:21:25 +08:00
return &dto.MidjourneyResponse{
2023-08-14 22:16:32 +08:00
Code: 4,
Description: "quota_not_enough",
}
}
req, err := http.NewRequest(c.Request.Method, fullRequestURL, requestBody)
if err != nil {
2024-02-29 16:21:25 +08:00
return &dto.MidjourneyResponse{
2023-08-14 22:16:32 +08:00
Code: 4,
Description: "create_request_failed",
}
}
2024-02-29 01:08:18 +08:00
//req.Header.Set("ApiKey", c.Request.Header.Get("ApiKey"))
2024-03-13 16:19:22 +08:00
timeout := time.Second * 30
ctx, cancel := context.WithTimeout(context.Background(), timeout)
// 使用带有超时的 context 创建新的请求
req = req.WithContext(ctx)
2023-08-14 22:16:32 +08:00
req.Header.Set("Content-Type", c.Request.Header.Get("Content-Type"))
req.Header.Set("Accept", c.Request.Header.Get("Accept"))
req.Header.Set("mj-api-secret", strings.Split(c.Request.Header.Get("Authorization"), " ")[1])
// print request header
2024-03-13 16:19:22 +08:00
//log.Printf("request header: %s", req.Header)
//log.Printf("request body: %s", midjRequest.Prompt)
2023-08-14 22:16:32 +08:00
2024-03-13 16:19:22 +08:00
defer cancel()
2024-02-29 16:21:25 +08:00
resp, err := service.GetHttpClient().Do(req)
2023-08-14 22:16:32 +08:00
if err != nil {
2024-03-13 17:46:34 +08:00
return service.MidjourneyErrorWrapper(constant.MjErrorUnknown, "do_request_failed")
2023-08-14 22:16:32 +08:00
}
err = req.Body.Close()
if err != nil {
2024-03-13 17:46:34 +08:00
return service.MidjourneyErrorWrapper(constant.MjErrorUnknown, "close_request_body_failed")
2023-08-14 22:16:32 +08:00
}
err = c.Request.Body.Close()
if err != nil {
2024-03-13 17:46:34 +08:00
return service.MidjourneyErrorWrapper(constant.MjErrorUnknown, "close_request_body_failed")
2023-08-14 22:16:32 +08:00
}
2024-02-29 16:21:25 +08:00
var midjResponse dto.MidjourneyResponse
2023-08-14 22:16:32 +08:00
2023-09-17 21:07:00 +08:00
defer func(ctx context.Context) {
2023-08-14 22:16:32 +08:00
if consumeQuota {
2023-11-15 18:27:13 +08:00
err := model.PostConsumeTokenQuota(tokenId, userQuota, quota, 0, true)
2023-08-14 22:16:32 +08:00
if err != nil {
common.SysError("error consuming token remain quota: " + err.Error())
}
err = model.CacheUpdateUserQuota(userId)
if err != nil {
common.SysError("error update user quota cache: " + err.Error())
}
if quota != 0 {
tokenName := c.GetString("token_name")
logContent := fmt.Sprintf("模型固定价格 %.2f,分组倍率 %.2f,操作 %s", modelPrice, groupRatio, midjRequest.Action)
model.RecordConsumeLog(ctx, userId, channelId, 0, 0, modelName, tokenName, quota, logContent, tokenId, userQuota, 0, false)
2023-08-14 22:16:32 +08:00
model.UpdateUserUsedQuotaAndRequestCount(userId, quota)
channelId := c.GetInt("channel_id")
model.UpdateChannelUsedQuota(channelId, quota)
}
}
2023-09-17 21:07:00 +08:00
}(c.Request.Context())
2023-08-14 22:16:32 +08:00
responseBody, err := io.ReadAll(resp.Body)
if err != nil {
2024-03-13 17:46:34 +08:00
return service.MidjourneyErrorWrapper(constant.MjErrorUnknown, "read_response_body_failed")
2023-08-14 22:16:32 +08:00
}
err = resp.Body.Close()
if err != nil {
2024-03-13 17:46:34 +08:00
return service.MidjourneyErrorWrapper(constant.MjErrorUnknown, "close_response_body_failed")
}
if resp.StatusCode != 200 {
return service.MidjourneyErrorWrapper(constant.MjErrorUnknown, "unexpected_response_status")
2023-08-14 22:16:32 +08:00
}
err = json.Unmarshal(responseBody, &midjResponse)
log.Printf("responseBody: %s", string(responseBody))
log.Printf("midjResponse: %v", midjResponse)
if err != nil {
2024-03-13 17:46:34 +08:00
return service.MidjourneyErrorWrapper(constant.MjErrorUnknown, "unmarshal_response_body_failed")
2023-08-14 22:16:32 +08:00
}
2023-11-06 02:08:12 +08:00
// 文档https://github.com/novicezk/midjourney-proxy/blob/main/docs/api.md
//1-提交成功
// 21-任务已存在(处理中或者有结果了) {"code":21,"description":"任务已存在","result":"0741798445574458","properties":{"status":"SUCCESS","imageUrl":"https://xxxx"}}
// 22-排队中 {"code":22,"description":"排队中前面还有1个任务","result":"0741798445574458","properties":{"numberOfQueues":1,"discordInstanceId":"1118138338562560102"}}
// 23-队列已满,请稍后再试 {"code":23,"description":"队列已满,请稍后尝试","result":"14001929738841620","properties":{"discordInstanceId":"1118138338562560102"}}
// 24-prompt包含敏感词 {"code":24,"description":"可能包含敏感词","properties":{"promptEn":"nude body","bannedWord":"nude"}}
// other: 提交错误description为错误描述
2023-08-14 22:16:32 +08:00
midjourneyTask := &model.Midjourney{
UserId: userId,
Code: midjResponse.Code,
Action: midjRequest.Action,
2023-08-14 22:16:32 +08:00
MjId: midjResponse.Result,
Prompt: midjRequest.Prompt,
PromptEn: "",
Description: midjResponse.Description,
State: "",
2023-11-06 02:08:12 +08:00
SubmitTime: time.Now().UnixNano() / int64(time.Millisecond),
2023-08-14 22:16:32 +08:00
StartTime: 0,
FinishTime: 0,
ImageUrl: "",
Status: "",
Progress: "0%",
FailReason: "",
2023-09-09 03:11:42 +08:00
ChannelId: c.GetInt("channel_id"),
2024-01-12 13:45:52 +08:00
Quota: quota,
2023-08-14 22:16:32 +08:00
}
2023-11-06 02:08:12 +08:00
if midjResponse.Code != 1 && midjResponse.Code != 21 && midjResponse.Code != 22 {
//非1-提交成功,21-任务已存在和22-排队中,则记录错误原因
2023-08-14 22:16:32 +08:00
midjourneyTask.FailReason = midjResponse.Description
2023-11-06 02:08:12 +08:00
consumeQuota = false
}
if midjResponse.Code == 21 { //21-任务已存在(处理中或者有结果了)
// 将 properties 转换为一个 map
properties, ok := midjResponse.Properties.(map[string]interface{})
if ok {
imageUrl, ok1 := properties["imageUrl"].(string)
status, ok2 := properties["status"].(string)
if ok1 && ok2 {
midjourneyTask.ImageUrl = imageUrl
midjourneyTask.Status = status
if status == "SUCCESS" {
midjourneyTask.Progress = "100%"
midjourneyTask.StartTime = time.Now().UnixNano() / int64(time.Millisecond)
midjourneyTask.FinishTime = time.Now().UnixNano() / int64(time.Millisecond)
midjResponse.Code = 1
}
}
}
//修改返回值
newBody := strings.Replace(string(responseBody), `"code":21`, `"code":1`, -1)
responseBody = []byte(newBody)
2023-08-14 22:16:32 +08:00
}
2023-11-06 02:08:12 +08:00
2023-08-14 22:16:32 +08:00
err = midjourneyTask.Insert()
if err != nil {
2024-02-29 16:21:25 +08:00
return &dto.MidjourneyResponse{
2023-08-14 22:16:32 +08:00
Code: 4,
Description: "insert_midjourney_task_failed",
}
}
2023-11-06 02:08:12 +08:00
if midjResponse.Code == 22 { //22-排队中,说明任务已存在
//修改返回值
newBody := strings.Replace(string(responseBody), `"code":22`, `"code":1`, -1)
responseBody = []byte(newBody)
}
2023-08-14 22:16:32 +08:00
resp.Body = io.NopCloser(bytes.NewBuffer(responseBody))
for k, v := range resp.Header {
c.Writer.Header().Set(k, v[0])
}
c.Writer.WriteHeader(resp.StatusCode)
_, err = io.Copy(c.Writer, resp.Body)
if err != nil {
2024-02-29 16:21:25 +08:00
return &dto.MidjourneyResponse{
2023-08-14 22:16:32 +08:00
Code: 4,
Description: "copy_response_body_failed",
}
}
err = resp.Body.Close()
if err != nil {
2024-02-29 16:21:25 +08:00
return &dto.MidjourneyResponse{
2023-08-14 22:16:32 +08:00
Code: 4,
Description: "close_response_body_failed",
}
}
return nil
}
type taskChangeParams struct {
ID string
Action string
Index int
}