mirror of
https://github.com/imgproxy/imgproxy.git
synced 2025-01-23 11:14:48 +02:00
98 lines
2.1 KiB
Go
98 lines
2.1 KiB
Go
package imagedata
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"io"
|
|
|
|
"github.com/imgproxy/imgproxy/v3/bufpool"
|
|
"github.com/imgproxy/imgproxy/v3/bufreader"
|
|
"github.com/imgproxy/imgproxy/v3/config"
|
|
"github.com/imgproxy/imgproxy/v3/ierrors"
|
|
"github.com/imgproxy/imgproxy/v3/imagemeta"
|
|
"github.com/imgproxy/imgproxy/v3/security"
|
|
)
|
|
|
|
var (
|
|
ErrSourceFileTooBig = ierrors.New(422, "Source image file is too big", "Invalid source image")
|
|
ErrSourceImageTypeNotSupported = ierrors.New(422, "Source image type not supported", "Invalid source image")
|
|
)
|
|
|
|
var downloadBufPool *bufpool.Pool
|
|
|
|
func initRead() {
|
|
downloadBufPool = bufpool.New("download", config.Concurrency, config.DownloadBufferSize)
|
|
}
|
|
|
|
type hardLimitReader struct {
|
|
r io.Reader
|
|
left int
|
|
}
|
|
|
|
func (lr *hardLimitReader) Read(p []byte) (n int, err error) {
|
|
if lr.left <= 0 {
|
|
return 0, ErrSourceFileTooBig
|
|
}
|
|
if len(p) > lr.left {
|
|
p = p[0:lr.left]
|
|
}
|
|
n, err = lr.r.Read(p)
|
|
lr.left -= n
|
|
return
|
|
}
|
|
|
|
func readAndCheckImage(r io.Reader, contentLength int) (*ImageData, error) {
|
|
if config.MaxSrcFileSize > 0 && contentLength > config.MaxSrcFileSize {
|
|
return nil, ErrSourceFileTooBig
|
|
}
|
|
|
|
buf := downloadBufPool.Get(contentLength, false)
|
|
cancel := func() { downloadBufPool.Put(buf) }
|
|
|
|
if config.MaxSrcFileSize > 0 {
|
|
r = &hardLimitReader{r: r, left: config.MaxSrcFileSize}
|
|
}
|
|
|
|
br := bufreader.New(r, buf)
|
|
|
|
meta, err := imagemeta.DecodeMeta(br)
|
|
if err != nil {
|
|
buf.Reset()
|
|
cancel()
|
|
|
|
if err == imagemeta.ErrFormat {
|
|
return nil, ErrSourceImageTypeNotSupported
|
|
}
|
|
|
|
return nil, checkTimeoutErr(err)
|
|
}
|
|
|
|
if err = security.CheckDimensions(meta.Width(), meta.Height()); err != nil {
|
|
buf.Reset()
|
|
cancel()
|
|
return nil, err
|
|
}
|
|
|
|
if contentLength > buf.Cap() {
|
|
buf.Grow(contentLength - buf.Len())
|
|
}
|
|
|
|
if err = br.Flush(); err != nil {
|
|
cancel()
|
|
return nil, checkTimeoutErr(err)
|
|
}
|
|
|
|
return &ImageData{
|
|
Data: buf.Bytes(),
|
|
Type: meta.Format(),
|
|
cancel: cancel,
|
|
}, nil
|
|
}
|
|
|
|
func BorrowBuffer() (*bytes.Buffer, context.CancelFunc) {
|
|
buf := downloadBufPool.Get(0, false)
|
|
cancel := func() { downloadBufPool.Put(buf) }
|
|
|
|
return buf, cancel
|
|
}
|