|
@@ -2,22 +2,36 @@ package main
|
|
|
|
|
|
import (
|
|
import (
|
|
"bytes"
|
|
"bytes"
|
|
|
|
+ "context"
|
|
"encoding/base64"
|
|
"encoding/base64"
|
|
"fmt"
|
|
"fmt"
|
|
"os"
|
|
"os"
|
|
)
|
|
)
|
|
|
|
|
|
|
|
+type imageData struct {
|
|
|
|
+ Data []byte
|
|
|
|
+ Type imageType
|
|
|
|
+
|
|
|
|
+ cancel context.CancelFunc
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+func (d *imageData) Close() {
|
|
|
|
+ if d.cancel != nil {
|
|
|
|
+ d.cancel()
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
func getWatermarkData() (*imageData, error) {
|
|
func getWatermarkData() (*imageData, error) {
|
|
if len(conf.WatermarkData) > 0 {
|
|
if len(conf.WatermarkData) > 0 {
|
|
- return base64ImageData(conf.WatermarkData)
|
|
|
|
|
|
+ return base64ImageData(conf.WatermarkData, "watermark")
|
|
}
|
|
}
|
|
|
|
|
|
if len(conf.WatermarkPath) > 0 {
|
|
if len(conf.WatermarkPath) > 0 {
|
|
- return fileImageData(conf.WatermarkPath)
|
|
|
|
|
|
+ return fileImageData(conf.WatermarkPath, "watermark")
|
|
}
|
|
}
|
|
|
|
|
|
if len(conf.WatermarkURL) > 0 {
|
|
if len(conf.WatermarkURL) > 0 {
|
|
- return remoteImageData(conf.WatermarkURL)
|
|
|
|
|
|
+ return remoteImageData(conf.WatermarkURL, "watermark")
|
|
}
|
|
}
|
|
|
|
|
|
return nil, nil
|
|
return nil, nil
|
|
@@ -25,65 +39,65 @@ func getWatermarkData() (*imageData, error) {
|
|
|
|
|
|
func getFallbackImageData() (*imageData, error) {
|
|
func getFallbackImageData() (*imageData, error) {
|
|
if len(conf.FallbackImageData) > 0 {
|
|
if len(conf.FallbackImageData) > 0 {
|
|
- return base64ImageData(conf.FallbackImageData)
|
|
|
|
|
|
+ return base64ImageData(conf.FallbackImageData, "fallback image")
|
|
}
|
|
}
|
|
|
|
|
|
if len(conf.FallbackImagePath) > 0 {
|
|
if len(conf.FallbackImagePath) > 0 {
|
|
- return fileImageData(conf.FallbackImagePath)
|
|
|
|
|
|
+ return fileImageData(conf.FallbackImagePath, "fallback image")
|
|
}
|
|
}
|
|
|
|
|
|
if len(conf.FallbackImageURL) > 0 {
|
|
if len(conf.FallbackImageURL) > 0 {
|
|
- return remoteImageData(conf.FallbackImageURL)
|
|
|
|
|
|
+ return remoteImageData(conf.FallbackImageURL, "fallback image")
|
|
}
|
|
}
|
|
|
|
|
|
return nil, nil
|
|
return nil, nil
|
|
}
|
|
}
|
|
|
|
|
|
-func base64ImageData(encoded string) (*imageData, error) {
|
|
|
|
|
|
+func base64ImageData(encoded, desc string) (*imageData, error) {
|
|
data, err := base64.StdEncoding.DecodeString(encoded)
|
|
data, err := base64.StdEncoding.DecodeString(encoded)
|
|
if err != nil {
|
|
if err != nil {
|
|
- return nil, fmt.Errorf("Can't decode image data: %s", err)
|
|
|
|
|
|
+ return nil, fmt.Errorf("Can't decode %s data: %s", desc, err)
|
|
}
|
|
}
|
|
|
|
|
|
imgtype, err := checkTypeAndDimensions(bytes.NewReader(data))
|
|
imgtype, err := checkTypeAndDimensions(bytes.NewReader(data))
|
|
if err != nil {
|
|
if err != nil {
|
|
- return nil, fmt.Errorf("Can't decode image: %s", err)
|
|
|
|
|
|
+ return nil, fmt.Errorf("Can't decode %s: %s", desc, err)
|
|
}
|
|
}
|
|
|
|
|
|
return &imageData{Data: data, Type: imgtype}, nil
|
|
return &imageData{Data: data, Type: imgtype}, nil
|
|
}
|
|
}
|
|
|
|
|
|
-func fileImageData(path string) (*imageData, error) {
|
|
|
|
|
|
+func fileImageData(path, desc string) (*imageData, error) {
|
|
f, err := os.Open(path)
|
|
f, err := os.Open(path)
|
|
if err != nil {
|
|
if err != nil {
|
|
- return nil, fmt.Errorf("Can't read watermark: %s", err)
|
|
|
|
|
|
+ return nil, fmt.Errorf("Can't read %s: %s", desc, err)
|
|
}
|
|
}
|
|
|
|
|
|
fi, err := f.Stat()
|
|
fi, err := f.Stat()
|
|
if err != nil {
|
|
if err != nil {
|
|
- return nil, fmt.Errorf("Can't read watermark: %s", err)
|
|
|
|
|
|
+ return nil, fmt.Errorf("Can't read %s: %s", desc, err)
|
|
}
|
|
}
|
|
|
|
|
|
imgdata, err := readAndCheckImage(f, int(fi.Size()))
|
|
imgdata, err := readAndCheckImage(f, int(fi.Size()))
|
|
if err != nil {
|
|
if err != nil {
|
|
- return nil, fmt.Errorf("Can't read watermark: %s", err)
|
|
|
|
|
|
+ return nil, fmt.Errorf("Can't read %s: %s", desc, err)
|
|
}
|
|
}
|
|
|
|
|
|
return imgdata, err
|
|
return imgdata, err
|
|
}
|
|
}
|
|
|
|
|
|
-func remoteImageData(imageURL string) (*imageData, error) {
|
|
|
|
|
|
+func remoteImageData(imageURL, desc string) (*imageData, error) {
|
|
res, err := requestImage(imageURL)
|
|
res, err := requestImage(imageURL)
|
|
if res != nil {
|
|
if res != nil {
|
|
defer res.Body.Close()
|
|
defer res.Body.Close()
|
|
}
|
|
}
|
|
if err != nil {
|
|
if err != nil {
|
|
- return nil, fmt.Errorf("Can't download image: %s", err)
|
|
|
|
|
|
+ return nil, fmt.Errorf("Can't download %s: %s", desc, err)
|
|
}
|
|
}
|
|
|
|
|
|
imgdata, err := readAndCheckImage(res.Body, int(res.ContentLength))
|
|
imgdata, err := readAndCheckImage(res.Body, int(res.ContentLength))
|
|
if err != nil {
|
|
if err != nil {
|
|
- return nil, fmt.Errorf("Can't download image: %s", err)
|
|
|
|
|
|
+ return nil, fmt.Errorf("Can't download %s: %s", desc, err)
|
|
}
|
|
}
|
|
|
|
|
|
return imgdata, err
|
|
return imgdata, err
|