| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172 |
- package service
- import (
- "bytes"
- "encoding/base64"
- "errors"
- "fmt"
- "image"
- "io"
- "net/http"
- "one-api/common"
- "one-api/constant"
- "strings"
- "golang.org/x/image/webp"
- )
- func DecodeBase64ImageData(base64String string) (image.Config, string, string, error) {
- // 去除base64数据的URL前缀(如果有)
- if idx := strings.Index(base64String, ","); idx != -1 {
- base64String = base64String[idx+1:]
- }
- // 将base64字符串解码为字节切片
- decodedData, err := base64.StdEncoding.DecodeString(base64String)
- if err != nil {
- fmt.Println("Error: Failed to decode base64 string")
- return image.Config{}, "", "", fmt.Errorf("failed to decode base64 string: %s", err.Error())
- }
- // 创建一个bytes.Buffer用于存储解码后的数据
- reader := bytes.NewReader(decodedData)
- config, format, err := getImageConfig(reader)
- return config, format, base64String, err
- }
- func DecodeBase64FileData(base64String string) (string, string, error) {
- var mimeType string
- var idx int
- idx = strings.Index(base64String, ",")
- if idx == -1 {
- _, file_type, base64, err := DecodeBase64ImageData(base64String)
- return "image/" + file_type, base64, err
- }
- mimeType = base64String[:idx]
- base64String = base64String[idx+1:]
- idx = strings.Index(mimeType, ";")
- if idx == -1 {
- _, file_type, base64, err := DecodeBase64ImageData(base64String)
- return "image/" + file_type, base64, err
- }
- mimeType = mimeType[:idx]
- idx = strings.Index(mimeType, ":")
- if idx == -1 {
- _, file_type, base64, err := DecodeBase64ImageData(base64String)
- return "image/" + file_type, base64, err
- }
- mimeType = mimeType[idx+1:]
- return mimeType, base64String, nil
- }
- // GetImageFromUrl 获取图片的类型和base64编码的数据
- func GetImageFromUrl(url string) (mimeType string, data string, err error) {
- resp, err := DoDownloadRequest(url)
- if err != nil {
- return "", "", fmt.Errorf("failed to download image: %w", err)
- }
- defer resp.Body.Close()
- // Check HTTP status code
- if resp.StatusCode != http.StatusOK {
- return "", "", fmt.Errorf("failed to download image: HTTP %d", resp.StatusCode)
- }
- contentType := resp.Header.Get("Content-Type")
- if contentType != "application/octet-stream" && !strings.HasPrefix(contentType, "image/") {
- return "", "", fmt.Errorf("invalid content type: %s, required image/*", contentType)
- }
- maxImageSize := int64(constant.MaxFileDownloadMB * 1024 * 1024)
- // Check Content-Length if available
- if resp.ContentLength > maxImageSize {
- return "", "", fmt.Errorf("image size %d exceeds maximum allowed size of %d bytes", resp.ContentLength, maxImageSize)
- }
- // Use LimitReader to prevent reading oversized images
- limitReader := io.LimitReader(resp.Body, maxImageSize)
- buffer := &bytes.Buffer{}
- written, err := io.Copy(buffer, limitReader)
- if err != nil {
- return "", "", fmt.Errorf("failed to read image data: %w", err)
- }
- if written >= maxImageSize {
- return "", "", fmt.Errorf("image size exceeds maximum allowed size of %d bytes", maxImageSize)
- }
- data = base64.StdEncoding.EncodeToString(buffer.Bytes())
- mimeType = contentType
- // Handle application/octet-stream type
- if mimeType == "application/octet-stream" {
- _, format, _, err := DecodeBase64ImageData(data)
- if err != nil {
- return "", "", err
- }
- mimeType = "image/" + format
- }
- return mimeType, data, nil
- }
- func DecodeUrlImageData(imageUrl string) (image.Config, string, error) {
- response, err := DoDownloadRequest(imageUrl)
- if err != nil {
- common.SysLog(fmt.Sprintf("fail to get image from url: %s", err.Error()))
- return image.Config{}, "", err
- }
- defer response.Body.Close()
- if response.StatusCode != 200 {
- err = errors.New(fmt.Sprintf("fail to get image from url: %s", response.Status))
- return image.Config{}, "", err
- }
- mimeType := response.Header.Get("Content-Type")
- if mimeType != "application/octet-stream" && !strings.HasPrefix(mimeType, "image/") {
- return image.Config{}, "", fmt.Errorf("invalid content type: %s, required image/*", mimeType)
- }
- var readData []byte
- for _, limit := range []int64{1024 * 8, 1024 * 24, 1024 * 64} {
- common.SysLog(fmt.Sprintf("try to decode image config with limit: %d", limit))
- // 从response.Body读取更多的数据直到达到当前的限制
- additionalData := make([]byte, limit-int64(len(readData)))
- n, _ := io.ReadFull(response.Body, additionalData)
- readData = append(readData, additionalData[:n]...)
- // 使用io.MultiReader组合已经读取的数据和response.Body
- limitReader := io.MultiReader(bytes.NewReader(readData), response.Body)
- var config image.Config
- var format string
- config, format, err = getImageConfig(limitReader)
- if err == nil {
- return config, format, nil
- }
- }
- return image.Config{}, "", err // 返回最后一个错误
- }
- func getImageConfig(reader io.Reader) (image.Config, string, error) {
- // 读取图片的头部信息来获取图片尺寸
- config, format, err := image.DecodeConfig(reader)
- if err != nil {
- err = errors.New(fmt.Sprintf("fail to decode image config(gif, jpg, png): %s", err.Error()))
- common.SysLog(err.Error())
- config, err = webp.DecodeConfig(reader)
- if err != nil {
- err = errors.New(fmt.Sprintf("fail to decode image config(webp): %s", err.Error()))
- common.SysLog(err.Error())
- }
- format = "webp"
- }
- if err != nil {
- return image.Config{}, "", err
- }
- return config, format, nil
- }
|