|
|
package service |
|
|
|
|
|
import ( |
|
|
"bytes" |
|
|
"encoding/base64" |
|
|
"errors" |
|
|
"fmt" |
|
|
"image" |
|
|
"io" |
|
|
"net/http" |
|
|
"strings" |
|
|
|
|
|
"github.com/QuantumNous/new-api/common" |
|
|
"github.com/QuantumNous/new-api/constant" |
|
|
|
|
|
"golang.org/x/image/webp" |
|
|
) |
|
|
|
|
|
|
|
|
func DecodeBase64ImageData(base64String string) (image.Config, string, string, error) { |
|
|
|
|
|
if idx := strings.Index(base64String, ","); idx != -1 { |
|
|
base64String = base64String[idx+1:] |
|
|
} |
|
|
|
|
|
if len(base64String) == 0 { |
|
|
return image.Config{}, "", "", errors.New("base64 string is empty") |
|
|
} |
|
|
|
|
|
|
|
|
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()) |
|
|
} |
|
|
|
|
|
|
|
|
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 |
|
|
} |
|
|
|
|
|
|
|
|
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() |
|
|
|
|
|
|
|
|
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) |
|
|
|
|
|
|
|
|
if resp.ContentLength > maxImageSize { |
|
|
return "", "", fmt.Errorf("image size %d exceeds maximum allowed size of %d bytes", resp.ContentLength, maxImageSize) |
|
|
} |
|
|
|
|
|
|
|
|
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 |
|
|
|
|
|
|
|
|
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)) |
|
|
|
|
|
|
|
|
additionalData := make([]byte, limit-int64(len(readData))) |
|
|
n, _ := io.ReadFull(response.Body, additionalData) |
|
|
readData = append(readData, additionalData[:n]...) |
|
|
|
|
|
|
|
|
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 |
|
|
} |
|
|
|