| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650 |
- package controller
- import (
- "bytes"
- "context"
- "encoding/json"
- "fmt"
- "io"
- "log"
- "net/http"
- "one-api/common"
- "one-api/model"
- "strconv"
- "strings"
- "time"
- "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"`
- }
- 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"`
- }
- 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,
- }
- 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",
- })
- 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
- }
- 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: "",
- }
- }
- midjourneyTask := model.GetByOnlyMJId(midjRequest.MjId)
- 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
- }
- func getMidjourneyTaskModel(c *gin.Context, originTask *model.Midjourney) (midjourneyTask Midjourney) {
- 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
- 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)
- }
- }
- midjourneyTask.Status = originTask.Status
- midjourneyTask.FailReason = originTask.FailReason
- midjourneyTask.Action = originTask.Action
- midjourneyTask.Description = originTask.Description
- midjourneyTask.Prompt = originTask.Prompt
- 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",
- }
- }
- }
- c.Writer.Header().Set("Content-Type", "application/json")
- _, err = io.Copy(c.Writer, bytes.NewBuffer(respBody))
- if err != nil {
- return &MidjourneyResponse{
- Code: 4,
- Description: "copy_response_body_failed",
- }
- }
- return nil
- }
- const (
- // type 1 根据 mode 价格不同
- MJSubmitActionImagine = "IMAGINE"
- MJSubmitActionVariation = "VARIATION" //变换
- MJSubmitActionBlend = "BLEND" //混图
- MJSubmitActionReroll = "REROLL" //重新生成
- // type 2 固定价格
- MJSubmitActionDescribe = "DESCRIBE"
- MJSubmitActionUpscale = "UPSCALE" // 放大
- )
- 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")
- channelId := c.GetInt("channel_id")
- var midjRequest MidjourneyRequest
- if consumeQuota {
- err := common.UnmarshalBodyReusable(c, &midjRequest)
- if err != nil {
- return &MidjourneyResponse{
- Code: 4,
- Description: "bind_request_body_failed",
- }
- }
- }
- if relayMode == RelayModeMidjourneyImagine { //绘画任务,此类任务可重复
- if midjRequest.Prompt == "" {
- return &MidjourneyResponse{
- Code: 4,
- Description: "prompt_is_required",
- }
- }
- midjRequest.Action = "IMAGINE"
- } else if relayMode == RelayModeMidjourneyDescribe { //按图生文任务,此类任务可重复
- midjRequest.Action = "DESCRIBE"
- } else if relayMode == RelayModeMidjourneyBlend { //绘画任务,此类任务可重复
- midjRequest.Action = "BLEND"
- } else if midjRequest.TaskId != "" { //放大、变换任务,此类任务,如果重复且已有结果,远端api会直接返回最终结果
- 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",
- }
- }
- //action = midjRequest.Action
- 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
- midjRequest.Action = params.Action
- }
- originTask := model.GetByMJId(userId, mjId)
- 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",
- }
- } 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())
- }
- 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)
- log.Printf("fullRequestURL: %s", fullRequestURL)
- 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
- }
- mjAction := "mj_" + strings.ToLower(midjRequest.Action)
- modelPrice := common.GetModelPrice(mjAction)
- // 如果没有配置价格,则使用默认价格
- if modelPrice == -1 {
- defaultPrice, ok := DefaultModelPrice[mjAction]
- if !ok {
- modelPrice = 0.1
- } else {
- modelPrice = defaultPrice
- }
- }
- groupRatio := common.GetGroupRatio(group)
- ratio := modelPrice * groupRatio
- userQuota, err := model.CacheGetUserQuota(userId)
- if err != nil {
- return &MidjourneyResponse{
- Code: 4,
- Description: err.Error(),
- }
- }
- quota := int(ratio * common.QuotaPerUnit)
- 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",
- }
- }
- //req.Header.Set("Authorization", c.Request.Header.Get("Authorization"))
- req.Header.Set("Content-Type", c.Request.Header.Get("Content-Type"))
- req.Header.Set("Accept", c.Request.Header.Get("Accept"))
- //mjToken := ""
- //if c.Request.Header.Get("Authorization") != "" {
- // mjToken = strings.Split(c.Request.Header.Get("Authorization"), " ")[1]
- //}
- //req.Header.Set("Authorization", "Bearer midjourney-proxy")
- 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
- defer func(ctx context.Context) {
- if consumeQuota {
- err := model.PostConsumeTokenQuota(tokenId, userQuota, quota, 0, true)
- 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, imageModel, tokenName, quota, logContent, tokenId, userQuota)
- model.UpdateUserUsedQuotaAndRequestCount(userId, quota)
- channelId := c.GetInt("channel_id")
- model.UpdateChannelUsedQuota(channelId, quota)
- }
- }
- }(c.Request.Context())
- //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,
- Description: "fail_to_fetch_midjourney status_code: " + strconv.Itoa(resp.StatusCode),
- }
- }
- if err != nil {
- return &MidjourneyResponse{
- Code: 4,
- Description: "unmarshal_response_body_failed",
- }
- }
- // 文档: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为错误描述
- midjourneyTask := &model.Midjourney{
- UserId: userId,
- Code: midjResponse.Code,
- Action: midjRequest.Action,
- MjId: midjResponse.Result,
- Prompt: midjRequest.Prompt,
- PromptEn: "",
- Description: midjResponse.Description,
- State: "",
- SubmitTime: time.Now().UnixNano() / int64(time.Millisecond),
- StartTime: 0,
- FinishTime: 0,
- ImageUrl: "",
- Status: "",
- Progress: "0%",
- FailReason: "",
- ChannelId: c.GetInt("channel_id"),
- Quota: quota,
- }
- if midjResponse.Code != 1 && midjResponse.Code != 21 && midjResponse.Code != 22 {
- //非1-提交成功,21-任务已存在和22-排队中,则记录错误原因
- midjourneyTask.FailReason = midjResponse.Description
- 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)
- }
- err = midjourneyTask.Insert()
- if err != nil {
- return &MidjourneyResponse{
- Code: 4,
- Description: "insert_midjourney_task_failed",
- }
- }
- if midjResponse.Code == 22 { //22-排队中,说明任务已存在
- //修改返回值
- newBody := strings.Replace(string(responseBody), `"code":22`, `"code":1`, -1)
- responseBody = []byte(newBody)
- }
- 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
- }
- 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
- }
|