123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180 |
- package main
- import (
- "bufio"
- "bytes"
- "context"
- "crypto/tls"
- "errors"
- "fmt"
- "image"
- "io"
- "io/ioutil"
- "net/http"
- "sync"
- "time"
- _ "image/gif"
- _ "image/jpeg"
- _ "image/png"
- _ "golang.org/x/image/webp"
- )
- var (
- downloadClient *http.Client
- imageTypeCtxKey = ctxKey("imageType")
- imageDataCtxKey = ctxKey("imageData")
- errSourceDimensionsTooBig = errors.New("Source image dimensions are too big")
- errSourceResolutionTooBig = errors.New("Source image resolution are too big")
- errSourceImageTypeNotSupported = errors.New("Source image type not supported")
- errInvalidImageURL = errors.New("Invalid image url")
- )
- var downloadBufPool = sync.Pool{
- New: func() interface{} {
- return new(bytes.Buffer)
- },
- }
- type netReader struct {
- reader *bufio.Reader
- buf *bytes.Buffer
- }
- func newNetReader(r io.Reader, buf *bytes.Buffer) *netReader {
- return &netReader{
- reader: bufio.NewReader(r),
- buf: buf,
- }
- }
- func (r *netReader) Read(p []byte) (n int, err error) {
- n, err = r.reader.Read(p)
- if err == nil {
- r.buf.Write(p[:n])
- }
- return
- }
- func (r *netReader) Peek(n int) ([]byte, error) {
- return r.reader.Peek(n)
- }
- func (r *netReader) ReadAll() error {
- if _, err := r.buf.ReadFrom(r.reader); err != nil {
- return err
- }
- return nil
- }
- func initDownloading() {
- transport := &http.Transport{
- Proxy: http.ProxyFromEnvironment,
- }
- if conf.IgnoreSslVerification {
- transport.TLSClientConfig = &tls.Config{InsecureSkipVerify: true}
- }
- if conf.LocalFileSystemRoot != "" {
- transport.RegisterProtocol("local", http.NewFileTransport(http.Dir(conf.LocalFileSystemRoot)))
- }
- if conf.S3Enabled {
- transport.RegisterProtocol("s3", newS3Transport())
- }
- if len(conf.GCSKey) > 0 {
- transport.RegisterProtocol("gs", newGCSTransport())
- }
- downloadClient = &http.Client{
- Timeout: time.Duration(conf.DownloadTimeout) * time.Second,
- Transport: transport,
- }
- }
- func checkTypeAndDimensions(r io.Reader) (imageType, error) {
- imgconf, imgtypeStr, err := image.DecodeConfig(r)
- imgtype, imgtypeOk := imageTypes[imgtypeStr]
- if err != nil {
- return imageTypeUnknown, err
- }
- if imgconf.Width > conf.MaxSrcDimension || imgconf.Height > conf.MaxSrcDimension {
- return imageTypeUnknown, errSourceDimensionsTooBig
- }
- if imgconf.Width*imgconf.Height > conf.MaxSrcResolution {
- return imageTypeUnknown, errSourceResolutionTooBig
- }
- if !imgtypeOk || !vipsTypeSupportLoad[imgtype] {
- return imageTypeUnknown, errSourceImageTypeNotSupported
- }
- return imgtype, nil
- }
- func readAndCheckImage(ctx context.Context, res *http.Response) (context.Context, context.CancelFunc, error) {
- buf := downloadBufPool.Get().(*bytes.Buffer)
- cancel := func() {
- buf.Reset()
- downloadBufPool.Put(buf)
- }
- nr := newNetReader(res.Body, buf)
- imgtype, err := checkTypeAndDimensions(nr)
- if err != nil {
- return ctx, cancel, err
- }
- if err = nr.ReadAll(); err == nil {
- ctx = context.WithValue(ctx, imageTypeCtxKey, imgtype)
- ctx = context.WithValue(ctx, imageDataCtxKey, nr.buf)
- }
- return ctx, cancel, err
- }
- func downloadImage(ctx context.Context) (context.Context, context.CancelFunc, error) {
- url := fmt.Sprintf("%s%s", conf.BaseURL, getImageURL(ctx))
- if newRelicEnabled {
- newRelicCancel := startNewRelicSegment(ctx, "Downloading image")
- defer newRelicCancel()
- }
- if prometheusEnabled {
- defer startPrometheusDuration(prometheusDownloadDuration)()
- }
- req, err := http.NewRequest("GET", url, nil)
- if err != nil {
- return ctx, func() {}, err
- }
- req.Header.Set("User-Agent", conf.UserAgent)
- res, err := downloadClient.Do(req)
- if err != nil {
- return ctx, func() {}, err
- }
- defer res.Body.Close()
- if res.StatusCode != 200 {
- body, _ := ioutil.ReadAll(res.Body)
- return ctx, func() {}, fmt.Errorf("Can't download image; Status: %d; %s", res.StatusCode, string(body))
- }
- return readAndCheckImage(ctx, res)
- }
- func getImageType(ctx context.Context) imageType {
- return ctx.Value(imageTypeCtxKey).(imageType)
- }
- func getImageData(ctx context.Context) *bytes.Buffer {
- return ctx.Value(imageDataCtxKey).(*bytes.Buffer)
- }
|