245 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			245 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| //go:generate go-enum --sql --marshal --file $GOFILE
 | |
| package img
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"context"
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"image"
 | |
| 	"io"
 | |
| 
 | |
| 	"github.com/disintegration/imaging"
 | |
| 	"github.com/dsoprea/go-exif/v3"
 | |
| 	"github.com/marusama/semaphore/v2"
 | |
| 
 | |
| 	exifcommon "github.com/dsoprea/go-exif/v3/common"
 | |
| )
 | |
| 
 | |
| // ErrUnsupportedFormat means the given image format is not supported.
 | |
| var ErrUnsupportedFormat = errors.New("unsupported image format")
 | |
| 
 | |
| // Service
 | |
| type Service struct {
 | |
| 	sem semaphore.Semaphore
 | |
| }
 | |
| 
 | |
| func New(workers int) *Service {
 | |
| 	return &Service{
 | |
| 		sem: semaphore.New(workers),
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // Format is an image file format.
 | |
| /*
 | |
| ENUM(
 | |
| jpeg
 | |
| png
 | |
| gif
 | |
| tiff
 | |
| bmp
 | |
| )
 | |
| */
 | |
| type Format int
 | |
| 
 | |
| func (x Format) toImaging() imaging.Format {
 | |
| 	switch x {
 | |
| 	case FormatJpeg:
 | |
| 		return imaging.JPEG
 | |
| 	case FormatPng:
 | |
| 		return imaging.PNG
 | |
| 	case FormatGif:
 | |
| 		return imaging.GIF
 | |
| 	case FormatTiff:
 | |
| 		return imaging.TIFF
 | |
| 	case FormatBmp:
 | |
| 		return imaging.BMP
 | |
| 	default:
 | |
| 		return imaging.JPEG
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /*
 | |
| ENUM(
 | |
| high
 | |
| medium
 | |
| low
 | |
| )
 | |
| */
 | |
| type Quality int
 | |
| 
 | |
| func (x Quality) resampleFilter() imaging.ResampleFilter {
 | |
| 	switch x {
 | |
| 	case QualityHigh:
 | |
| 		return imaging.Lanczos
 | |
| 	case QualityMedium:
 | |
| 		return imaging.Box
 | |
| 	case QualityLow:
 | |
| 		return imaging.NearestNeighbor
 | |
| 	default:
 | |
| 		return imaging.Box
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /*
 | |
| ENUM(
 | |
| fit
 | |
| fill
 | |
| )
 | |
| */
 | |
| type ResizeMode int
 | |
| 
 | |
| func (s *Service) FormatFromExtension(ext string) (Format, error) {
 | |
| 	format, err := imaging.FormatFromExtension(ext)
 | |
| 	if err != nil {
 | |
| 		return -1, ErrUnsupportedFormat
 | |
| 	}
 | |
| 	switch format {
 | |
| 	case imaging.JPEG:
 | |
| 		return FormatJpeg, nil
 | |
| 	case imaging.PNG:
 | |
| 		return FormatPng, nil
 | |
| 	case imaging.GIF:
 | |
| 		return FormatGif, nil
 | |
| 	case imaging.TIFF:
 | |
| 		return FormatTiff, nil
 | |
| 	case imaging.BMP:
 | |
| 		return FormatBmp, nil
 | |
| 	}
 | |
| 	return -1, ErrUnsupportedFormat
 | |
| }
 | |
| 
 | |
| type resizeConfig struct {
 | |
| 	format     Format
 | |
| 	resizeMode ResizeMode
 | |
| 	quality    Quality
 | |
| }
 | |
| 
 | |
| type Option func(*resizeConfig)
 | |
| 
 | |
| func WithFormat(format Format) Option {
 | |
| 	return func(config *resizeConfig) {
 | |
| 		config.format = format
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func WithMode(mode ResizeMode) Option {
 | |
| 	return func(config *resizeConfig) {
 | |
| 		config.resizeMode = mode
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func WithQuality(quality Quality) Option {
 | |
| 	return func(config *resizeConfig) {
 | |
| 		config.quality = quality
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (s *Service) Resize(ctx context.Context, in io.Reader, width, height int, out io.Writer, options ...Option) error {
 | |
| 	if err := s.sem.Acquire(ctx, 1); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	defer s.sem.Release(1)
 | |
| 
 | |
| 	format, wrappedReader, err := s.detectFormat(in)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	config := resizeConfig{
 | |
| 		format:     format,
 | |
| 		resizeMode: ResizeModeFit,
 | |
| 		quality:    QualityMedium,
 | |
| 	}
 | |
| 	for _, option := range options {
 | |
| 		option(&config)
 | |
| 	}
 | |
| 
 | |
| 	if config.quality == QualityLow && format == FormatJpeg {
 | |
| 		thm, newWrappedReader, errThm := getEmbeddedThumbnail(wrappedReader)
 | |
| 		wrappedReader = newWrappedReader
 | |
| 		if errThm == nil {
 | |
| 			_, err = out.Write(thm)
 | |
| 			if err == nil {
 | |
| 				return nil
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	img, err := imaging.Decode(wrappedReader, imaging.AutoOrientation(true))
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	switch config.resizeMode {
 | |
| 	case ResizeModeFill:
 | |
| 		img = imaging.Fill(img, width, height, imaging.Center, config.quality.resampleFilter())
 | |
| 	case ResizeModeFit:
 | |
| 		fallthrough //nolint:gocritic
 | |
| 	default:
 | |
| 		img = imaging.Fit(img, width, height, config.quality.resampleFilter())
 | |
| 	}
 | |
| 
 | |
| 	return imaging.Encode(out, img, config.format.toImaging())
 | |
| }
 | |
| 
 | |
| func (s *Service) detectFormat(in io.Reader) (Format, io.Reader, error) {
 | |
| 	buf := &bytes.Buffer{}
 | |
| 	r := io.TeeReader(in, buf)
 | |
| 
 | |
| 	_, imgFormat, err := image.DecodeConfig(r)
 | |
| 	if err != nil {
 | |
| 		return 0, nil, fmt.Errorf("%s: %w", err.Error(), ErrUnsupportedFormat)
 | |
| 	}
 | |
| 
 | |
| 	format, err := ParseFormat(imgFormat)
 | |
| 	if err != nil {
 | |
| 		return 0, nil, ErrUnsupportedFormat
 | |
| 	}
 | |
| 
 | |
| 	return format, io.MultiReader(buf, in), nil
 | |
| }
 | |
| 
 | |
| func getEmbeddedThumbnail(in io.Reader) ([]byte, io.Reader, error) {
 | |
| 	buf := &bytes.Buffer{}
 | |
| 	r := io.TeeReader(in, buf)
 | |
| 	wrappedReader := io.MultiReader(buf, in)
 | |
| 
 | |
| 	offset := 0
 | |
| 	offsets := []int{12, 30}
 | |
| 	head := make([]byte, 0xffff) //nolint:gomnd
 | |
| 
 | |
| 	_, err := r.Read(head)
 | |
| 	if err != nil {
 | |
| 		return nil, wrappedReader, err
 | |
| 	}
 | |
| 
 | |
| 	for _, offset = range offsets {
 | |
| 		if _, err = exif.ParseExifHeader(head[offset:]); err == nil {
 | |
| 			break
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if err != nil {
 | |
| 		return nil, wrappedReader, err
 | |
| 	}
 | |
| 
 | |
| 	im, err := exifcommon.NewIfdMappingWithStandard()
 | |
| 	if err != nil {
 | |
| 		return nil, wrappedReader, err
 | |
| 	}
 | |
| 
 | |
| 	_, index, err := exif.Collect(im, exif.NewTagIndex(), head[offset:])
 | |
| 	if err != nil {
 | |
| 		return nil, wrappedReader, err
 | |
| 	}
 | |
| 
 | |
| 	ifd := index.RootIfd.NextIfd()
 | |
| 	if ifd == nil {
 | |
| 		return nil, wrappedReader, exif.ErrNoThumbnail
 | |
| 	}
 | |
| 
 | |
| 	thm, err := ifd.Thumbnail()
 | |
| 	return thm, wrappedReader, err
 | |
| }
 |