image.go 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107
  1. package service
  2. import (
  3. "bytes"
  4. "encoding/base64"
  5. "errors"
  6. "fmt"
  7. "golang.org/x/image/webp"
  8. "image"
  9. "io"
  10. "one-api/common"
  11. "strings"
  12. )
  13. func DecodeBase64ImageData(base64String string) (image.Config, string, string, error) {
  14. // 去除base64数据的URL前缀(如果有)
  15. if idx := strings.Index(base64String, ","); idx != -1 {
  16. base64String = base64String[idx+1:]
  17. }
  18. // 将base64字符串解码为字节切片
  19. decodedData, err := base64.StdEncoding.DecodeString(base64String)
  20. if err != nil {
  21. fmt.Println("Error: Failed to decode base64 string")
  22. return image.Config{}, "", "", err
  23. }
  24. // 创建一个bytes.Buffer用于存储解码后的数据
  25. reader := bytes.NewReader(decodedData)
  26. config, format, err := getImageConfig(reader)
  27. return config, format, base64String, err
  28. }
  29. // GetImageFromUrl 获取图片的类型和base64编码的数据
  30. func GetImageFromUrl(url string) (mimeType string, data string, err error) {
  31. resp, err := DoImageRequest(url)
  32. if err != nil {
  33. return
  34. }
  35. if !strings.HasPrefix(resp.Header.Get("Content-Type"), "image/") {
  36. return
  37. }
  38. defer resp.Body.Close()
  39. buffer := bytes.NewBuffer(nil)
  40. _, err = buffer.ReadFrom(resp.Body)
  41. if err != nil {
  42. return
  43. }
  44. mimeType = resp.Header.Get("Content-Type")
  45. data = base64.StdEncoding.EncodeToString(buffer.Bytes())
  46. return
  47. }
  48. func DecodeUrlImageData(imageUrl string) (image.Config, string, error) {
  49. response, err := DoImageRequest(imageUrl)
  50. if err != nil {
  51. common.SysLog(fmt.Sprintf("fail to get image from url: %s", err.Error()))
  52. return image.Config{}, "", err
  53. }
  54. defer response.Body.Close()
  55. if response.StatusCode != 200 {
  56. err = errors.New(fmt.Sprintf("fail to get image from url: %s", response.Status))
  57. return image.Config{}, "", err
  58. }
  59. var readData []byte
  60. for _, limit := range []int64{1024 * 8, 1024 * 24, 1024 * 64} {
  61. common.SysLog(fmt.Sprintf("try to decode image config with limit: %d", limit))
  62. // 从response.Body读取更多的数据直到达到当前的限制
  63. additionalData := make([]byte, limit-int64(len(readData)))
  64. n, _ := io.ReadFull(response.Body, additionalData)
  65. readData = append(readData, additionalData[:n]...)
  66. // 使用io.MultiReader组合已经读取的数据和response.Body
  67. limitReader := io.MultiReader(bytes.NewReader(readData), response.Body)
  68. var config image.Config
  69. var format string
  70. config, format, err = getImageConfig(limitReader)
  71. if err == nil {
  72. return config, format, nil
  73. }
  74. }
  75. return image.Config{}, "", err // 返回最后一个错误
  76. }
  77. func getImageConfig(reader io.Reader) (image.Config, string, error) {
  78. // 读取图片的头部信息来获取图片尺寸
  79. config, format, err := image.DecodeConfig(reader)
  80. if err != nil {
  81. err = errors.New(fmt.Sprintf("fail to decode image config(gif, jpg, png): %s", err.Error()))
  82. common.SysLog(err.Error())
  83. config, err = webp.DecodeConfig(reader)
  84. if err != nil {
  85. err = errors.New(fmt.Sprintf("fail to decode image config(webp): %s", err.Error()))
  86. common.SysLog(err.Error())
  87. }
  88. format = "webp"
  89. }
  90. if err != nil {
  91. return image.Config{}, "", err
  92. }
  93. return config, format, nil
  94. }