image.go 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  1. package service
  2. import (
  3. "bytes"
  4. "encoding/base64"
  5. "errors"
  6. "fmt"
  7. "image"
  8. "io"
  9. "net/http"
  10. "one-api/common"
  11. "one-api/constant"
  12. "strings"
  13. "golang.org/x/image/webp"
  14. )
  15. func DecodeBase64ImageData(base64String string) (image.Config, string, string, error) {
  16. // 去除base64数据的URL前缀(如果有)
  17. if idx := strings.Index(base64String, ","); idx != -1 {
  18. base64String = base64String[idx+1:]
  19. }
  20. // 将base64字符串解码为字节切片
  21. decodedData, err := base64.StdEncoding.DecodeString(base64String)
  22. if err != nil {
  23. fmt.Println("Error: Failed to decode base64 string")
  24. return image.Config{}, "", "", fmt.Errorf("failed to decode base64 string: %s", err.Error())
  25. }
  26. // 创建一个bytes.Buffer用于存储解码后的数据
  27. reader := bytes.NewReader(decodedData)
  28. config, format, err := getImageConfig(reader)
  29. return config, format, base64String, err
  30. }
  31. func DecodeBase64FileData(base64String string) (string, string, error) {
  32. var mimeType string
  33. var idx int
  34. idx = strings.Index(base64String, ",")
  35. if idx == -1 {
  36. _, file_type, base64, err := DecodeBase64ImageData(base64String)
  37. return "image/" + file_type, base64, err
  38. }
  39. mimeType = base64String[:idx]
  40. base64String = base64String[idx+1:]
  41. idx = strings.Index(mimeType, ";")
  42. if idx == -1 {
  43. _, file_type, base64, err := DecodeBase64ImageData(base64String)
  44. return "image/" + file_type, base64, err
  45. }
  46. mimeType = mimeType[:idx]
  47. idx = strings.Index(mimeType, ":")
  48. if idx == -1 {
  49. _, file_type, base64, err := DecodeBase64ImageData(base64String)
  50. return "image/" + file_type, base64, err
  51. }
  52. mimeType = mimeType[idx+1:]
  53. return mimeType, base64String, nil
  54. }
  55. // GetImageFromUrl 获取图片的类型和base64编码的数据
  56. func GetImageFromUrl(url string) (mimeType string, data string, err error) {
  57. resp, err := DoDownloadRequest(url)
  58. if err != nil {
  59. return "", "", fmt.Errorf("failed to download image: %w", err)
  60. }
  61. defer resp.Body.Close()
  62. // Check HTTP status code
  63. if resp.StatusCode != http.StatusOK {
  64. return "", "", fmt.Errorf("failed to download image: HTTP %d", resp.StatusCode)
  65. }
  66. contentType := resp.Header.Get("Content-Type")
  67. if contentType != "application/octet-stream" && !strings.HasPrefix(contentType, "image/") {
  68. return "", "", fmt.Errorf("invalid content type: %s, required image/*", contentType)
  69. }
  70. maxImageSize := int64(constant.MaxFileDownloadMB * 1024 * 1024)
  71. // Check Content-Length if available
  72. if resp.ContentLength > maxImageSize {
  73. return "", "", fmt.Errorf("image size %d exceeds maximum allowed size of %d bytes", resp.ContentLength, maxImageSize)
  74. }
  75. // Use LimitReader to prevent reading oversized images
  76. limitReader := io.LimitReader(resp.Body, maxImageSize)
  77. buffer := &bytes.Buffer{}
  78. written, err := io.Copy(buffer, limitReader)
  79. if err != nil {
  80. return "", "", fmt.Errorf("failed to read image data: %w", err)
  81. }
  82. if written >= maxImageSize {
  83. return "", "", fmt.Errorf("image size exceeds maximum allowed size of %d bytes", maxImageSize)
  84. }
  85. data = base64.StdEncoding.EncodeToString(buffer.Bytes())
  86. mimeType = contentType
  87. // Handle application/octet-stream type
  88. if mimeType == "application/octet-stream" {
  89. _, format, _, err := DecodeBase64ImageData(data)
  90. if err != nil {
  91. return "", "", err
  92. }
  93. mimeType = "image/" + format
  94. }
  95. return mimeType, data, nil
  96. }
  97. func DecodeUrlImageData(imageUrl string) (image.Config, string, error) {
  98. response, err := DoDownloadRequest(imageUrl)
  99. if err != nil {
  100. common.SysLog(fmt.Sprintf("fail to get image from url: %s", err.Error()))
  101. return image.Config{}, "", err
  102. }
  103. defer response.Body.Close()
  104. if response.StatusCode != 200 {
  105. err = errors.New(fmt.Sprintf("fail to get image from url: %s", response.Status))
  106. return image.Config{}, "", err
  107. }
  108. mimeType := response.Header.Get("Content-Type")
  109. if mimeType != "application/octet-stream" && !strings.HasPrefix(mimeType, "image/") {
  110. return image.Config{}, "", fmt.Errorf("invalid content type: %s, required image/*", mimeType)
  111. }
  112. var readData []byte
  113. for _, limit := range []int64{1024 * 8, 1024 * 24, 1024 * 64} {
  114. common.SysLog(fmt.Sprintf("try to decode image config with limit: %d", limit))
  115. // 从response.Body读取更多的数据直到达到当前的限制
  116. additionalData := make([]byte, limit-int64(len(readData)))
  117. n, _ := io.ReadFull(response.Body, additionalData)
  118. readData = append(readData, additionalData[:n]...)
  119. // 使用io.MultiReader组合已经读取的数据和response.Body
  120. limitReader := io.MultiReader(bytes.NewReader(readData), response.Body)
  121. var config image.Config
  122. var format string
  123. config, format, err = getImageConfig(limitReader)
  124. if err == nil {
  125. return config, format, nil
  126. }
  127. }
  128. return image.Config{}, "", err // 返回最后一个错误
  129. }
  130. func getImageConfig(reader io.Reader) (image.Config, string, error) {
  131. // 读取图片的头部信息来获取图片尺寸
  132. config, format, err := image.DecodeConfig(reader)
  133. if err != nil {
  134. err = errors.New(fmt.Sprintf("fail to decode image config(gif, jpg, png): %s", err.Error()))
  135. common.SysLog(err.Error())
  136. config, err = webp.DecodeConfig(reader)
  137. if err != nil {
  138. err = errors.New(fmt.Sprintf("fail to decode image config(webp): %s", err.Error()))
  139. common.SysLog(err.Error())
  140. }
  141. format = "webp"
  142. }
  143. if err != nil {
  144. return image.Config{}, "", err
  145. }
  146. return config, format, nil
  147. }