2023-08-14 22:16:32 +08:00
|
|
|
|
package controller
|
|
|
|
|
|
|
|
|
|
|
|
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/model"
|
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"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
type Midjourney struct {
|
|
|
|
|
|
MjId string `json:"id"`
|
|
|
|
|
|
Action string `json:"action"`
|
|
|
|
|
|
Prompt string `json:"prompt"`
|
|
|
|
|
|
PromptEn string `json:"promptEn"`
|
|
|
|
|
|
Description string `json:"description"`
|
|
|
|
|
|
State string `json:"state"`
|
|
|
|
|
|
SubmitTime int64 `json:"submitTime"`
|
|
|
|
|
|
StartTime int64 `json:"startTime"`
|
|
|
|
|
|
FinishTime int64 `json:"finishTime"`
|
|
|
|
|
|
ImageUrl string `json:"imageUrl"`
|
|
|
|
|
|
Status string `json:"status"`
|
|
|
|
|
|
Progress string `json:"progress"`
|
|
|
|
|
|
FailReason string `json:"failReason"`
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-11-06 02:08:12 +08:00
|
|
|
|
type MidjourneyStatus struct {
|
|
|
|
|
|
Status int `json:"status"`
|
|
|
|
|
|
}
|
|
|
|
|
|
type MidjourneyWithoutStatus struct {
|
|
|
|
|
|
Id int `json:"id"`
|
|
|
|
|
|
Code int `json:"code"`
|
|
|
|
|
|
UserId int `json:"user_id" gorm:"index"`
|
|
|
|
|
|
Action string `json:"action"`
|
|
|
|
|
|
MjId string `json:"mj_id" gorm:"index"`
|
|
|
|
|
|
Prompt string `json:"prompt"`
|
|
|
|
|
|
PromptEn string `json:"prompt_en"`
|
|
|
|
|
|
Description string `json:"description"`
|
|
|
|
|
|
State string `json:"state"`
|
|
|
|
|
|
SubmitTime int64 `json:"submit_time"`
|
|
|
|
|
|
StartTime int64 `json:"start_time"`
|
|
|
|
|
|
FinishTime int64 `json:"finish_time"`
|
|
|
|
|
|
ImageUrl string `json:"image_url"`
|
|
|
|
|
|
Progress string `json:"progress"`
|
|
|
|
|
|
FailReason string `json:"fail_reason"`
|
|
|
|
|
|
ChannelId int `json:"channel_id"`
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-01-05 12:23:26 +08:00
|
|
|
|
var DefaultModelPrice = map[string]float64{
|
|
|
|
|
|
"mj_imagine": 0.1,
|
|
|
|
|
|
"mj_variation": 0.1,
|
|
|
|
|
|
"mj_reroll": 0.1,
|
|
|
|
|
|
"mj_blend": 0.1,
|
|
|
|
|
|
"mj_describe": 0.05,
|
|
|
|
|
|
"mj_upscale": 0.05,
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-09-12 03:46:05 +08:00
|
|
|
|
func RelayMidjourneyImage(c *gin.Context) {
|
|
|
|
|
|
taskId := c.Param("id")
|
2024-01-01 22:46:05 +08:00
|
|
|
|
midjourneyTask := model.GetByOnlyMJId(taskId)
|
2023-09-12 03:46:05 +08:00
|
|
|
|
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
|
2023-09-12 03:46:05 +08:00
|
|
|
|
}
|
|
|
|
|
|
defer resp.Body.Close()
|
2024-01-01 22:46:05 +08:00
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
|
|
responseBody, _ := io.ReadAll(resp.Body)
|
|
|
|
|
|
c.JSON(resp.StatusCode, gin.H{
|
|
|
|
|
|
"error": string(responseBody),
|
|
|
|
|
|
})
|
2023-09-12 03:46:05 +08:00
|
|
|
|
return
|
|
|
|
|
|
}
|
2024-01-01 22:46:05 +08:00
|
|
|
|
// 从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
|
2023-09-12 03:46:05 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2023-08-14 22:16:32 +08:00
|
|
|
|
func relayMidjourneyNotify(c *gin.Context) *MidjourneyResponse {
|
|
|
|
|
|
var midjRequest Midjourney
|
|
|
|
|
|
err := common.UnmarshalBodyReusable(c, &midjRequest)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "bind_request_body_failed",
|
|
|
|
|
|
Properties: nil,
|
|
|
|
|
|
Result: "",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-01-01 22:46:05 +08:00
|
|
|
|
midjourneyTask := model.GetByOnlyMJId(midjRequest.MjId)
|
2023-08-14 22:16:32 +08:00
|
|
|
|
if midjourneyTask == nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
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 {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "update_midjourney_task_failed",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-01-01 22:46:05 +08:00
|
|
|
|
func getMidjourneyTaskModel(c *gin.Context, originTask *model.Midjourney) (midjourneyTask Midjourney) {
|
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
|
2024-01-01 22:46:05 +08:00
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func relayMidjourneyTask(c *gin.Context, relayMode int) *MidjourneyResponse {
|
|
|
|
|
|
userId := c.GetInt("id")
|
|
|
|
|
|
var err error
|
|
|
|
|
|
var respBody []byte
|
|
|
|
|
|
switch relayMode {
|
|
|
|
|
|
case RelayModeMidjourneyTaskFetch:
|
|
|
|
|
|
taskId := c.Param("id")
|
|
|
|
|
|
originTask := model.GetByMJId(userId, taskId)
|
|
|
|
|
|
if originTask == nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "task_no_found",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
midjourneyTask := getMidjourneyTaskModel(c, originTask)
|
|
|
|
|
|
respBody, err = json.Marshal(midjourneyTask)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "unmarshal_response_body_failed",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
case RelayModeMidjourneyTaskFetchByCondition:
|
|
|
|
|
|
var condition = struct {
|
|
|
|
|
|
IDs []string `json:"ids"`
|
|
|
|
|
|
}{}
|
|
|
|
|
|
err = c.BindJSON(&condition)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "do_request_failed",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
var tasks []Midjourney
|
|
|
|
|
|
if len(condition.IDs) != 0 {
|
|
|
|
|
|
originTasks := model.GetByMJIds(userId, condition.IDs)
|
|
|
|
|
|
for _, originTask := range originTasks {
|
|
|
|
|
|
midjourneyTask := getMidjourneyTaskModel(c, originTask)
|
|
|
|
|
|
tasks = append(tasks, midjourneyTask)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
if tasks == nil {
|
|
|
|
|
|
tasks = make([]Midjourney, 0)
|
|
|
|
|
|
}
|
|
|
|
|
|
respBody, err = json.Marshal(tasks)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "unmarshal_response_body_failed",
|
|
|
|
|
|
}
|
2023-08-14 22:16:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-01-01 22:46:05 +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 {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "copy_response_body_failed",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-01-01 22:46:05 +08:00
|
|
|
|
const (
|
|
|
|
|
|
// type 1 根据 mode 价格不同
|
|
|
|
|
|
MJSubmitActionImagine = "IMAGINE"
|
|
|
|
|
|
MJSubmitActionVariation = "VARIATION" //变换
|
|
|
|
|
|
MJSubmitActionBlend = "BLEND" //混图
|
|
|
|
|
|
|
|
|
|
|
|
MJSubmitActionReroll = "REROLL" //重新生成
|
|
|
|
|
|
// type 2 固定价格
|
|
|
|
|
|
MJSubmitActionDescribe = "DESCRIBE"
|
|
|
|
|
|
MJSubmitActionUpscale = "UPSCALE" // 放大
|
|
|
|
|
|
)
|
|
|
|
|
|
|
2023-08-14 22:16:32 +08:00
|
|
|
|
func relayMidjourneySubmit(c *gin.Context, relayMode int) *MidjourneyResponse {
|
|
|
|
|
|
imageModel := "midjourney"
|
|
|
|
|
|
|
|
|
|
|
|
tokenId := c.GetInt("token_id")
|
|
|
|
|
|
channelType := c.GetInt("channel")
|
|
|
|
|
|
userId := c.GetInt("id")
|
|
|
|
|
|
consumeQuota := c.GetBool("consume_quota")
|
|
|
|
|
|
group := c.GetString("group")
|
2023-09-17 21:07:00 +08:00
|
|
|
|
channelId := c.GetInt("channel_id")
|
2023-08-14 22:16:32 +08:00
|
|
|
|
var midjRequest MidjourneyRequest
|
|
|
|
|
|
if consumeQuota {
|
|
|
|
|
|
err := common.UnmarshalBodyReusable(c, &midjRequest)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "bind_request_body_failed",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-01-01 22:46:05 +08:00
|
|
|
|
|
2023-11-06 02:08:12 +08:00
|
|
|
|
if relayMode == RelayModeMidjourneyImagine { //绘画任务,此类任务可重复
|
2023-08-14 22:16:32 +08:00
|
|
|
|
if midjRequest.Prompt == "" {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "prompt_is_required",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
midjRequest.Action = "IMAGINE"
|
2023-11-06 02:08:12 +08:00
|
|
|
|
} else if relayMode == RelayModeMidjourneyDescribe { //按图生文任务,此类任务可重复
|
|
|
|
|
|
midjRequest.Action = "DESCRIBE"
|
|
|
|
|
|
} else if relayMode == RelayModeMidjourneyBlend { //绘画任务,此类任务可重复
|
|
|
|
|
|
midjRequest.Action = "BLEND"
|
|
|
|
|
|
} else if midjRequest.TaskId != "" { //放大、变换任务,此类任务,如果重复且已有结果,远端api会直接返回最终结果
|
2024-01-01 22:46:05 +08:00
|
|
|
|
mjId := ""
|
|
|
|
|
|
if relayMode == RelayModeMidjourneyChange {
|
|
|
|
|
|
if midjRequest.TaskId == "" {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "taskId_is_required",
|
|
|
|
|
|
}
|
|
|
|
|
|
} else if midjRequest.Action == "" {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "action_is_required",
|
|
|
|
|
|
}
|
|
|
|
|
|
} else if midjRequest.Index == 0 {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "index_can_only_be_1_2_3_4",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-01-07 23:40:06 +08:00
|
|
|
|
//action = midjRequest.Action
|
2024-01-01 22:46:05 +08:00
|
|
|
|
mjId = midjRequest.TaskId
|
|
|
|
|
|
} else if relayMode == RelayModeMidjourneySimpleChange {
|
|
|
|
|
|
if midjRequest.Content == "" {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "content_is_required",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
params := convertSimpleChangeParams(midjRequest.Content)
|
|
|
|
|
|
if params == nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "content_parse_failed",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
mjId = params.ID
|
2024-01-07 23:40:06 +08:00
|
|
|
|
midjRequest.Action = params.Action
|
2024-01-01 22:46:05 +08:00
|
|
|
|
}
|
2024-01-07 23:40:06 +08:00
|
|
|
|
|
2024-01-01 22:46:05 +08:00
|
|
|
|
originTask := model.GetByMJId(userId, mjId)
|
2023-08-14 22:16:32 +08:00
|
|
|
|
if originTask == nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "task_no_found",
|
|
|
|
|
|
}
|
|
|
|
|
|
} else if originTask.Action == "UPSCALE" {
|
|
|
|
|
|
//return errorWrapper(errors.New("upscale task can not be change"), "request_params_error", http.StatusBadRequest).
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "upscale_task_can_not_be_change",
|
|
|
|
|
|
}
|
|
|
|
|
|
} else if originTask.Status != "SUCCESS" {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "task_status_is_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 &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "channel_not_found",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
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
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 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 &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "unmarshal_model_mapping_failed",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
if modelMap[imageModel] != "" {
|
|
|
|
|
|
imageModel = modelMap[imageModel]
|
|
|
|
|
|
isModelMapped = true
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
baseURL := common.ChannelBaseURLs[channelType]
|
|
|
|
|
|
requestURL := c.Request.URL.String()
|
|
|
|
|
|
|
|
|
|
|
|
if c.GetString("base_url") != "" {
|
|
|
|
|
|
baseURL = c.GetString("base_url")
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//midjRequest.NotifyHook = "http://127.0.0.1:3000/mj/notify"
|
|
|
|
|
|
|
|
|
|
|
|
fullRequestURL := fmt.Sprintf("%s%s", baseURL, requestURL)
|
2023-11-06 02:08:12 +08:00
|
|
|
|
log.Printf("fullRequestURL: %s", fullRequestURL)
|
2023-08-14 22:16:32 +08:00
|
|
|
|
|
|
|
|
|
|
var requestBody io.Reader
|
|
|
|
|
|
if isModelMapped {
|
|
|
|
|
|
jsonStr, err := json.Marshal(midjRequest)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "marshal_text_request_failed",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
requestBody = bytes.NewBuffer(jsonStr)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
requestBody = c.Request.Body
|
|
|
|
|
|
}
|
2024-01-07 23:40:06 +08:00
|
|
|
|
mjAction := "mj_" + strings.ToLower(midjRequest.Action)
|
2024-01-25 14:56:49 +08:00
|
|
|
|
modelPrice := common.GetModelPrice(mjAction, true)
|
2024-01-07 23:40:06 +08:00
|
|
|
|
// 如果没有配置价格,则使用默认价格
|
2024-01-05 12:23:26 +08:00
|
|
|
|
if modelPrice == -1 {
|
|
|
|
|
|
defaultPrice, ok := DefaultModelPrice[mjAction]
|
|
|
|
|
|
if !ok {
|
|
|
|
|
|
modelPrice = 0.1
|
|
|
|
|
|
} else {
|
|
|
|
|
|
modelPrice = defaultPrice
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2023-08-14 22:16:32 +08:00
|
|
|
|
groupRatio := common.GetGroupRatio(group)
|
2024-01-01 22:46:05 +08:00
|
|
|
|
ratio := modelPrice * groupRatio
|
2023-08-14 22:16:32 +08:00
|
|
|
|
userQuota, err := model.CacheGetUserQuota(userId)
|
2024-01-01 22:46:05 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: err.Error(),
|
|
|
|
|
|
}
|
2023-09-22 23:34:17 +08:00
|
|
|
|
}
|
2024-01-01 22:46:05 +08:00
|
|
|
|
quota := int(ratio * common.QuotaPerUnit)
|
2023-08-14 22:16:32 +08:00
|
|
|
|
|
|
|
|
|
|
if consumeQuota && userQuota-quota < 0 {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "quota_not_enough",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
req, err := http.NewRequest(c.Request.Method, fullRequestURL, requestBody)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "create_request_failed",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2023-11-06 02:08:12 +08:00
|
|
|
|
//req.Header.Set("Authorization", c.Request.Header.Get("Authorization"))
|
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"))
|
2023-08-31 00:44:16 +08:00
|
|
|
|
//mjToken := ""
|
2023-11-06 02:08:12 +08:00
|
|
|
|
//if c.Request.Header.Get("Authorization") != "" {
|
|
|
|
|
|
// mjToken = strings.Split(c.Request.Header.Get("Authorization"), " ")[1]
|
2023-08-31 00:44:16 +08:00
|
|
|
|
//}
|
2023-11-28 23:15:17 +08:00
|
|
|
|
//req.Header.Set("Authorization", "Bearer midjourney-proxy")
|
2023-08-14 22:16:32 +08:00
|
|
|
|
req.Header.Set("mj-api-secret", strings.Split(c.Request.Header.Get("Authorization"), " ")[1])
|
|
|
|
|
|
// print request header
|
|
|
|
|
|
log.Printf("request header: %s", req.Header)
|
|
|
|
|
|
log.Printf("request body: %s", midjRequest.Prompt)
|
|
|
|
|
|
|
|
|
|
|
|
resp, err := httpClient.Do(req)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "do_request_failed",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
err = req.Body.Close()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "close_request_body_failed",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
err = c.Request.Body.Close()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "close_request_body_failed",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
var midjResponse MidjourneyResponse
|
|
|
|
|
|
|
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")
|
2024-01-07 23:40:06 +08:00
|
|
|
|
logContent := fmt.Sprintf("模型固定价格 %.2f,分组倍率 %.2f,操作 %s", modelPrice, groupRatio, midjRequest.Action)
|
2024-01-21 15:01:59 +08:00
|
|
|
|
model.RecordConsumeLog(ctx, userId, channelId, 0, 0, imageModel, 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
|
|
|
|
|
|
|
|
|
|
//if consumeQuota {
|
|
|
|
|
|
//
|
|
|
|
|
|
//}
|
|
|
|
|
|
responseBody, err := io.ReadAll(resp.Body)
|
|
|
|
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "read_response_body_failed",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
err = resp.Body.Close()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "close_response_body_failed",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
err = json.Unmarshal(responseBody, &midjResponse)
|
|
|
|
|
|
log.Printf("responseBody: %s", string(responseBody))
|
|
|
|
|
|
log.Printf("midjResponse: %v", midjResponse)
|
|
|
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
2023-09-12 03:17:26 +08:00
|
|
|
|
Description: "fail_to_fetch_midjourney status_code: " + strconv.Itoa(resp.StatusCode),
|
2023-08-14 22:16:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "unmarshal_response_body_failed",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
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,
|
2024-01-07 23:40:06 +08:00
|
|
|
|
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 {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
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 {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "copy_response_body_failed",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
err = resp.Body.Close()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return &MidjourneyResponse{
|
|
|
|
|
|
Code: 4,
|
|
|
|
|
|
Description: "close_response_body_failed",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
2024-01-01 22:46:05 +08:00
|
|
|
|
|
|
|
|
|
|
type taskChangeParams struct {
|
|
|
|
|
|
ID string
|
|
|
|
|
|
Action string
|
|
|
|
|
|
Index int
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func convertSimpleChangeParams(content string) *taskChangeParams {
|
|
|
|
|
|
split := strings.Split(content, " ")
|
|
|
|
|
|
if len(split) != 2 {
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
action := strings.ToLower(split[1])
|
|
|
|
|
|
changeParams := &taskChangeParams{}
|
|
|
|
|
|
changeParams.ID = split[0]
|
|
|
|
|
|
|
|
|
|
|
|
if action[0] == 'u' {
|
|
|
|
|
|
changeParams.Action = "UPSCALE"
|
|
|
|
|
|
} else if action[0] == 'v' {
|
|
|
|
|
|
changeParams.Action = "VARIATION"
|
|
|
|
|
|
} else if action == "r" {
|
|
|
|
|
|
changeParams.Action = "REROLL"
|
|
|
|
|
|
return changeParams
|
|
|
|
|
|
} else {
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
index, err := strconv.Atoi(action[1:2])
|
|
|
|
|
|
if err != nil || index < 1 || index > 4 {
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
changeParams.Index = index
|
|
|
|
|
|
return changeParams
|
|
|
|
|
|
}
|