mark2web/filter/image_process.go

241 lines
6.0 KiB
Go
Raw Normal View History

2019-03-18 13:34:52 +01:00
package filter
2019-02-27 15:58:10 +01:00
import (
2019-03-04 16:05:29 +01:00
"crypto/md5"
2019-03-03 13:25:00 +01:00
"errors"
"fmt"
2019-03-04 16:05:29 +01:00
"image"
"net/http"
"net/url"
"os"
"path"
2019-03-03 13:25:00 +01:00
"strconv"
"strings"
2019-03-04 16:05:29 +01:00
"gitbase.de/apairon/mark2web/config"
2019-03-18 13:34:52 +01:00
"gitbase.de/apairon/mark2web/context"
"gitbase.de/apairon/mark2web/helper"
2019-03-03 13:25:00 +01:00
"github.com/disintegration/imaging"
2019-02-27 15:58:10 +01:00
"github.com/flosch/pongo2"
)
2019-03-04 16:05:29 +01:00
func parseImageParams(str string) (*config.ImagingConfig, error) {
p := config.ImagingConfig{}
2019-03-03 13:25:00 +01:00
if str == "" {
2019-03-18 13:34:52 +01:00
helper.Merge(&p, context.CurrentPathConfig.Imaging)
2019-03-04 16:05:29 +01:00
// Filename and Format are only valid for current image
p.Filename = ""
p.Format = ""
return &p, nil
2019-03-03 13:25:00 +01:00
}
for _, s := range strings.Split(str, ",") {
e := strings.Split(s, "=")
if len(e) < 2 {
return nil, fmt.Errorf("invalid image parameter: %s", s)
}
var err error
switch e[0] {
case "w":
2019-03-04 16:05:29 +01:00
p.Width, err = strconv.Atoi(e[1])
2019-03-03 13:25:00 +01:00
case "h":
2019-03-04 16:05:29 +01:00
p.Height, err = strconv.Atoi(e[1])
2019-03-03 13:25:00 +01:00
case "f":
2019-03-04 16:05:29 +01:00
p.Filename = e[1]
case "t":
p.TargetDir = e[1]
2019-03-04 16:05:29 +01:00
case "p":
p.Process = e[1]
case "a":
p.Anchor = e[1]
case "q":
p.Quality, err = strconv.Atoi(e[1])
if p.Quality < 0 || p.Quality > 100 {
err = errors.New("q= must be between 1 and 100")
}
2019-03-03 13:25:00 +01:00
default:
return nil, fmt.Errorf("invalid image parameter: %s", s)
}
if err != nil {
return nil, fmt.Errorf("could not convert image parameter to correct value type for '%s': %s", s, err)
}
}
return &p, nil
}
2019-03-04 16:05:29 +01:00
func getImageFromURL(url string) (image.Image, string, error) {
resp, err := http.Get(url)
if err != nil {
return nil, "", fmt.Errorf("could not get url '%s': %s", url, err)
}
img, format, err := image.Decode(resp.Body)
if err != nil {
return nil, "", fmt.Errorf("could read body from url '%s': %s", url, err)
}
return img, format, nil
}
// ImageProcessFilter read the image url and process parameters and saves the resized/processed image
2019-03-03 13:25:00 +01:00
// param: w=WITDH,h=HEIGHT
2019-03-04 16:05:29 +01:00
func ImageProcessFilter(in *pongo2.Value, param *pongo2.Value) (*pongo2.Value, *pongo2.Error) {
2019-03-03 13:25:00 +01:00
imgSource := in.String()
p, err := parseImageParams(param.String())
if err != nil {
return nil, &pongo2.Error{
Sender: "filter:image_resize",
OrigError: err,
}
}
2019-03-04 16:05:29 +01:00
if p == nil {
return nil, &pongo2.Error{
Sender: "filter:image_resize",
OrigError: errors.New("no imaging config defined"),
}
}
2019-03-03 13:25:00 +01:00
2019-03-04 16:05:29 +01:00
var img image.Image
if p.Process == "" {
p.Process = "resize"
2019-03-03 13:25:00 +01:00
}
2019-03-04 16:05:29 +01:00
filePrefix := fmt.Sprintf(
"%s_%dx%d_q%03d",
p.Process,
p.Width,
p.Height,
p.Quality,
)
if strings.HasPrefix(imgSource, "http://") || strings.HasPrefix(imgSource, "https://") {
// remote file
img, p.Format, err = getImageFromURL(imgSource)
if err != nil {
return nil, &pongo2.Error{
Sender: "filter:image_resize",
OrigError: fmt.Errorf("could not open image '%s': %s", imgSource, err),
}
}
2019-03-04 16:05:29 +01:00
// build filename
if p.Filename == "" {
var fBase string
if u, _ := url.Parse(imgSource); u != nil {
fBase = strings.Split(path.Base(u.Path), ".")[0]
}
2019-03-03 13:25:00 +01:00
2019-03-04 16:05:29 +01:00
p.Filename = fmt.Sprintf(
"%s_%x_%s.%s",
filePrefix,
md5.Sum([]byte(imgSource)),
fBase,
p.Format,
)
}
} else {
// local file
2019-03-18 13:34:52 +01:00
imgSource = context.ResolveInputPath(imgSource)
2019-03-04 16:05:29 +01:00
if p.Filename == "" {
p.Filename = fmt.Sprintf(
"%s_%s",
filePrefix,
path.Base(imgSource),
)
}
}
2019-03-03 13:25:00 +01:00
var imgTarget string
if p.TargetDir != "" {
2019-03-18 13:34:52 +01:00
imgTarget = context.ResolveOutputPath(
path.Clean(p.TargetDir) + "/" +
p.Filename,
)
pt := path.Dir(imgTarget)
if _, err := os.Stat(pt); os.IsNotExist(err) {
2019-03-18 13:34:52 +01:00
helper.Log.Infof("create image target dir: %s", pt)
if err := os.MkdirAll(pt, 0755); err != nil {
return nil, &pongo2.Error{
Sender: "filter:image_resize",
OrigError: fmt.Errorf("could not create image target dir '%s': %s", pt, err),
}
}
}
2019-03-18 13:34:52 +01:00
p.Filename = context.ResolveNavPath(p.TargetDir + "/" + p.Filename)
} else {
2019-03-18 13:34:52 +01:00
imgTarget = context.ResolveOutputPath(p.Filename)
}
2019-03-04 16:05:29 +01:00
if f, err := os.Stat(imgTarget); err == nil && !f.IsDir() {
2019-03-18 13:34:52 +01:00
helper.Log.Noticef("skipped processing image from %s to %s, file already exists", imgSource, imgTarget)
2019-03-04 16:05:29 +01:00
} else {
2019-03-18 13:34:52 +01:00
helper.Log.Noticef("processing image from %s to %s", imgSource, imgTarget)
if strings.HasPrefix(imgSource, "http://") || strings.HasPrefix(imgSource, "https://") {
// webrequest before finding target filename, because of file format in filename
} else {
img, err = imaging.Open(imgSource, imaging.AutoOrientation(true))
if err != nil {
return nil, &pongo2.Error{
Sender: "filter:image_resize",
OrigError: fmt.Errorf("could not open image '%s': %s", imgSource, err),
}
}
}
2019-03-04 16:05:29 +01:00
switch p.Process {
case "resize":
img = imaging.Resize(img, p.Width, p.Height, imaging.Lanczos)
case "fit":
img = imaging.Fit(img, p.Width, p.Height, imaging.Lanczos)
case "fill":
var anchor imaging.Anchor
switch strings.ToLower(p.Anchor) {
case "":
fallthrough
case "center":
anchor = imaging.Center
case "topleft":
anchor = imaging.TopLeft
case "top":
anchor = imaging.Top
case "topright":
anchor = imaging.TopRight
case "left":
anchor = imaging.Left
case "right":
anchor = imaging.Right
case "bottomleft":
anchor = imaging.BottomLeft
case "bottom":
anchor = imaging.Bottom
case "bottomright":
anchor = imaging.BottomRight
default:
return nil, &pongo2.Error{
Sender: "filter:image_resize",
OrigError: fmt.Errorf("unknown anchor a=%s definition", p.Anchor),
}
}
img = imaging.Fill(img, p.Width, p.Height, anchor, imaging.Lanczos)
default:
return nil, &pongo2.Error{
Sender: "filter:image_resize",
OrigError: fmt.Errorf("invalid p parameter '%s'", p.Process),
}
}
var encodeOptions = make([]imaging.EncodeOption, 0)
if p.Quality > 0 {
encodeOptions = append(encodeOptions, imaging.JPEGQuality(p.Quality))
2019-03-03 13:25:00 +01:00
}
2019-03-04 16:05:29 +01:00
err = imaging.Save(img, imgTarget, encodeOptions...)
if err != nil {
return nil, &pongo2.Error{
Sender: "filter:image_resize",
OrigError: fmt.Errorf("could save image '%s': %s", imgTarget, err),
}
}
}
2019-03-18 13:34:52 +01:00
return pongo2.AsValue(context.ResolveNavPath(p.Filename)), nil
}