relay_utils.go 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. package common
  2. import (
  3. "fmt"
  4. "net/http"
  5. "one-api/common"
  6. "one-api/constant"
  7. "one-api/dto"
  8. "strconv"
  9. "strings"
  10. "github.com/gin-gonic/gin"
  11. "github.com/samber/lo"
  12. )
  13. type HasPrompt interface {
  14. GetPrompt() string
  15. }
  16. type HasImage interface {
  17. HasImage() bool
  18. }
  19. func GetFullRequestURL(baseURL string, requestURL string, channelType int) string {
  20. fullRequestURL := fmt.Sprintf("%s%s", baseURL, requestURL)
  21. if strings.HasPrefix(baseURL, "https://gateway.ai.cloudflare.com") {
  22. switch channelType {
  23. case constant.ChannelTypeOpenAI:
  24. fullRequestURL = fmt.Sprintf("%s%s", baseURL, strings.TrimPrefix(requestURL, "/v1"))
  25. case constant.ChannelTypeAzure:
  26. fullRequestURL = fmt.Sprintf("%s%s", baseURL, strings.TrimPrefix(requestURL, "/openai/deployments"))
  27. }
  28. }
  29. return fullRequestURL
  30. }
  31. func GetAPIVersion(c *gin.Context) string {
  32. query := c.Request.URL.Query()
  33. apiVersion := query.Get("api-version")
  34. if apiVersion == "" {
  35. apiVersion = c.GetString("api_version")
  36. }
  37. return apiVersion
  38. }
  39. func createTaskError(err error, code string, statusCode int, localError bool) *dto.TaskError {
  40. return &dto.TaskError{
  41. Code: code,
  42. Message: err.Error(),
  43. StatusCode: statusCode,
  44. LocalError: localError,
  45. Error: err,
  46. }
  47. }
  48. func storeTaskRequest(c *gin.Context, info *RelayInfo, action string, requestObj TaskSubmitReq) {
  49. info.Action = action
  50. c.Set("task_request", requestObj)
  51. }
  52. func validatePrompt(prompt string) *dto.TaskError {
  53. if strings.TrimSpace(prompt) == "" {
  54. return createTaskError(fmt.Errorf("prompt is required"), "invalid_request", http.StatusBadRequest, true)
  55. }
  56. return nil
  57. }
  58. func validateMultipartTaskRequest(c *gin.Context, info *RelayInfo, action string) (TaskSubmitReq, error) {
  59. var req TaskSubmitReq
  60. if _, err := c.MultipartForm(); err != nil {
  61. return req, err
  62. }
  63. formData := c.Request.PostForm
  64. req = TaskSubmitReq{
  65. Prompt: formData.Get("prompt"),
  66. Model: formData.Get("model"),
  67. Mode: formData.Get("mode"),
  68. Image: formData.Get("image"),
  69. Size: formData.Get("size"),
  70. Metadata: make(map[string]interface{}),
  71. }
  72. if durationStr := formData.Get("seconds"); durationStr != "" {
  73. if duration, err := strconv.Atoi(durationStr); err == nil {
  74. req.Duration = duration
  75. }
  76. }
  77. if images := formData["images"]; len(images) > 0 {
  78. req.Images = images
  79. }
  80. for key, values := range formData {
  81. if len(values) > 0 && !isKnownTaskField(key) {
  82. if intVal, err := strconv.Atoi(values[0]); err == nil {
  83. req.Metadata[key] = intVal
  84. } else if floatVal, err := strconv.ParseFloat(values[0], 64); err == nil {
  85. req.Metadata[key] = floatVal
  86. } else {
  87. req.Metadata[key] = values[0]
  88. }
  89. }
  90. }
  91. return req, nil
  92. }
  93. func ValidateMultipartDirect(c *gin.Context, info *RelayInfo) *dto.TaskError {
  94. contentType := c.GetHeader("Content-Type")
  95. var prompt string
  96. var hasInputReference bool
  97. if strings.HasPrefix(contentType, "multipart/form-data") {
  98. form, err := common.ParseMultipartFormReusable(c)
  99. if err != nil {
  100. return createTaskError(err, "invalid_multipart_form", http.StatusBadRequest, true)
  101. }
  102. defer form.RemoveAll()
  103. prompts, ok := form.Value["prompt"]
  104. if !ok || len(prompts) == 0 {
  105. return createTaskError(fmt.Errorf("prompt field is required"), "missing_prompt", http.StatusBadRequest, true)
  106. }
  107. prompt = prompts[0]
  108. if _, ok := form.Value["model"]; !ok {
  109. return createTaskError(fmt.Errorf("model field is required"), "missing_model", http.StatusBadRequest, true)
  110. }
  111. if _, ok := form.File["input_reference"]; ok {
  112. hasInputReference = true
  113. }
  114. } else {
  115. var req TaskSubmitReq
  116. if err := common.UnmarshalBodyReusable(c, &req); err != nil {
  117. return createTaskError(err, "invalid_json", http.StatusBadRequest, true)
  118. }
  119. prompt = req.Prompt
  120. if strings.TrimSpace(req.Model) == "" {
  121. return createTaskError(fmt.Errorf("model field is required"), "missing_model", http.StatusBadRequest, true)
  122. }
  123. if req.HasImage() {
  124. hasInputReference = true
  125. }
  126. }
  127. if taskErr := validatePrompt(prompt); taskErr != nil {
  128. return taskErr
  129. }
  130. action := constant.TaskActionTextGenerate
  131. if hasInputReference {
  132. action = constant.TaskActionGenerate
  133. }
  134. info.Action = action
  135. model := form.Value["model"][0]
  136. if strings.HasPrefix(model, "sora-2") {
  137. seconds := 4
  138. size := "720x1280"
  139. if ss, ok := form.Value["seconds"]; ok {
  140. sInt := common.String2Int(ss[0])
  141. if sInt > seconds {
  142. seconds = common.String2Int(ss[0])
  143. }
  144. }
  145. if s, ok := form.Value["size"]; ok {
  146. size = s[0]
  147. }
  148. if model == "sora-2" && !lo.Contains([]string{"720x1280", "1280x720"}, size) {
  149. return createTaskError(fmt.Errorf("sora-2 size is invalid"), "invalid_size", http.StatusBadRequest, true)
  150. }
  151. if model == "sora-2-pro" && !lo.Contains([]string{"720x1280", "1280x720", "1792x1024", "1024x1792"}, size) {
  152. return createTaskError(fmt.Errorf("sora-2 size is invalid"), "invalid_size", http.StatusBadRequest, true)
  153. }
  154. info.PriceData.OtherRatios = map[string]float64{
  155. "seconds": float64(seconds),
  156. "size": 1,
  157. }
  158. if lo.Contains([]string{"1792x1024", "1024x1792"}, size) {
  159. info.PriceData.OtherRatios["size"] = 1.666667
  160. }
  161. }
  162. return nil
  163. }
  164. func isKnownTaskField(field string) bool {
  165. knownFields := map[string]bool{
  166. "prompt": true,
  167. "model": true,
  168. "mode": true,
  169. "image": true,
  170. "images": true,
  171. "size": true,
  172. "duration": true,
  173. "input_reference": true, // Sora 特有字段
  174. }
  175. return knownFields[field]
  176. }
  177. func ValidateBasicTaskRequest(c *gin.Context, info *RelayInfo, action string) *dto.TaskError {
  178. var err error
  179. contentType := c.GetHeader("Content-Type")
  180. var req TaskSubmitReq
  181. if strings.HasPrefix(contentType, "multipart/form-data") {
  182. req, err = validateMultipartTaskRequest(c, info, action)
  183. if err != nil {
  184. return createTaskError(err, "invalid_multipart_form", http.StatusBadRequest, true)
  185. }
  186. } else if err := common.UnmarshalBodyReusable(c, &req); err != nil {
  187. return createTaskError(err, "invalid_request", http.StatusBadRequest, true)
  188. }
  189. if taskErr := validatePrompt(req.Prompt); taskErr != nil {
  190. return taskErr
  191. }
  192. if len(req.Images) == 0 && strings.TrimSpace(req.Image) != "" {
  193. // 兼容单图上传
  194. req.Images = []string{req.Image}
  195. }
  196. if req.HasImage() {
  197. action = constant.TaskActionGenerate
  198. if info.ChannelType == constant.ChannelTypeVidu {
  199. // vidu 增加 首尾帧生视频和参考图生视频
  200. if len(req.Images) == 2 {
  201. action = constant.TaskActionFirstTailGenerate
  202. } else if len(req.Images) > 2 {
  203. action = constant.TaskActionReferenceGenerate
  204. }
  205. }
  206. }
  207. storeTaskRequest(c, info, action, req)
  208. return nil
  209. }