Commit 60980097 authored by Matthias Kaeppler's avatar Matthias Kaeppler

Implement conditional GETs for image resizer

We now set a `Last-Modified` header based on either local
file timestamps or the respective header returned from an
object storage provider.
parent cb2a7bbf
---
title: Add support for conditional GETs for rescaled images
merge_request: 647
author:
type: performance
...@@ -44,6 +44,12 @@ type processCounter struct { ...@@ -44,6 +44,12 @@ type processCounter struct {
type resizeStatus = string type resizeStatus = string
type imageFile struct {
reader io.ReadCloser
contentLength int64
lastModified time.Time
}
const ( const (
statusSuccess = "success" // a rescaled image was served statusSuccess = "success" // a rescaled image was served
statusScalingFailure = "scaling-failed" // scaling failed but the original image was served statusScalingFailure = "scaling-failed" // scaling failed but the original image was served
...@@ -150,7 +156,7 @@ func NewResizer(cfg config.Config) *Resizer { ...@@ -150,7 +156,7 @@ func NewResizer(cfg config.Config) *Resizer {
return &Resizer{Config: cfg, Prefix: "send-scaled-img:"} return &Resizer{Config: cfg, Prefix: "send-scaled-img:"}
} }
// This Injecter forks into a dedicated scaler process to resize an image identified by path or URL // Inject forks into a dedicated scaler process to resize an image identified by path or URL
// and streams the resized image back to the client // and streams the resized image back to the client
func (r *Resizer) Inject(w http.ResponseWriter, req *http.Request, paramsData string) { func (r *Resizer) Inject(w http.ResponseWriter, req *http.Request, paramsData string) {
var status resizeStatus = statusUnknown var status resizeStatus = statusUnknown
...@@ -169,14 +175,14 @@ func (r *Resizer) Inject(w http.ResponseWriter, req *http.Request, paramsData st ...@@ -169,14 +175,14 @@ func (r *Resizer) Inject(w http.ResponseWriter, req *http.Request, paramsData st
return return
} }
sourceImageReader, fileSize, err := openSourceImage(params.Location) imageFile, err := openSourceImage(params.Location)
if err != nil { if err != nil {
// This means we cannot even read the input image; fail fast. // This means we cannot even read the input image; fail fast.
status = statusRequestFailure status = statusRequestFailure
helper.Fail500(w, req, fmt.Errorf("ImageResizer: Failed opening image data stream: %v", err)) helper.Fail500(w, req, fmt.Errorf("ImageResizer: Failed opening image data stream: %v", err))
return return
} }
defer sourceImageReader.Close() defer imageFile.reader.Close()
logFields := func(bytesWritten int64) *log.Fields { logFields := func(bytesWritten int64) *log.Fields {
return &log.Fields{ return &log.Fields{
...@@ -184,13 +190,21 @@ func (r *Resizer) Inject(w http.ResponseWriter, req *http.Request, paramsData st ...@@ -184,13 +190,21 @@ func (r *Resizer) Inject(w http.ResponseWriter, req *http.Request, paramsData st
"duration_s": time.Since(start).Seconds(), "duration_s": time.Since(start).Seconds(),
"target_width": params.Width, "target_width": params.Width,
"content_type": params.ContentType, "content_type": params.ContentType,
"original_filesize": fileSize, "original_filesize": imageFile.contentLength,
} }
} }
setLastModified(w, imageFile.lastModified)
// If the original file has not changed, then any cached resized versions have not changed either.
if checkNotModified(req, imageFile.lastModified) {
logger.WithFields(*logFields(0)).Printf("ImageResizer: Use cached image")
writeNotModified(w)
return
}
// We first attempt to rescale the image; if this should fail for any reason, imageReader // We first attempt to rescale the image; if this should fail for any reason, imageReader
// will point to the original image, i.e. we render it unchanged. // will point to the original image, i.e. we render it unchanged.
imageReader, resizeCmd, err := r.tryResizeImage(req, sourceImageReader, logger.Writer(), params, fileSize, r.Config.ImageResizerConfig) imageReader, resizeCmd, err := r.tryResizeImage(req, imageFile, logger.Writer(), params, r.Config.ImageResizerConfig)
if err != nil { if err != nil {
// Something failed, but we can still write out the original image, so don't return early. // Something failed, but we can still write out the original image, so don't return early.
helper.LogErrorWithFields(req, err, *logFields(0)) helper.LogErrorWithFields(req, err, *logFields(0))
...@@ -260,13 +274,13 @@ func (r *Resizer) unpackParameters(paramsData string) (*resizeParams, error) { ...@@ -260,13 +274,13 @@ func (r *Resizer) unpackParameters(paramsData string) (*resizeParams, error) {
} }
// Attempts to rescale the given image data, or in case of errors, falls back to the original image. // Attempts to rescale the given image data, or in case of errors, falls back to the original image.
func (r *Resizer) tryResizeImage(req *http.Request, reader io.Reader, errorWriter io.Writer, params *resizeParams, fileSize int64, cfg config.ImageResizerConfig) (io.Reader, *exec.Cmd, error) { func (r *Resizer) tryResizeImage(req *http.Request, f *imageFile, errorWriter io.Writer, params *resizeParams, cfg config.ImageResizerConfig) (io.Reader, *exec.Cmd, error) {
if fileSize > int64(cfg.MaxFilesize) { if f.contentLength > int64(cfg.MaxFilesize) {
return reader, nil, fmt.Errorf("ImageResizer: %db exceeds maximum file size of %db", fileSize, cfg.MaxFilesize) return f.reader, nil, fmt.Errorf("ImageResizer: %db exceeds maximum file size of %db", f.contentLength, cfg.MaxFilesize)
} }
if !r.numScalerProcs.tryIncrement(int32(cfg.MaxScalerProcs)) { if !r.numScalerProcs.tryIncrement(int32(cfg.MaxScalerProcs)) {
return reader, nil, fmt.Errorf("ImageResizer: too many running scaler processes (%d / %d)", r.numScalerProcs.n, cfg.MaxScalerProcs) return f.reader, nil, fmt.Errorf("ImageResizer: too many running scaler processes (%d / %d)", r.numScalerProcs.n, cfg.MaxScalerProcs)
} }
ctx := req.Context() ctx := req.Context()
...@@ -276,8 +290,8 @@ func (r *Resizer) tryResizeImage(req *http.Request, reader io.Reader, errorWrite ...@@ -276,8 +290,8 @@ func (r *Resizer) tryResizeImage(req *http.Request, reader io.Reader, errorWrite
}() }()
// Prevents EOF if the file is smaller than 8 bytes // Prevents EOF if the file is smaller than 8 bytes
bufferSize := int(math.Min(maxMagicLen, float64(fileSize))) bufferSize := int(math.Min(maxMagicLen, float64(f.contentLength)))
buffered := bufio.NewReaderSize(reader, bufferSize) buffered := bufio.NewReaderSize(f.reader, bufferSize)
data, err := buffered.Peek(bufferSize) data, err := buffered.Peek(bufferSize)
if err != nil { if err != nil {
...@@ -321,7 +335,7 @@ func isURL(location string) bool { ...@@ -321,7 +335,7 @@ func isURL(location string) bool {
return strings.HasPrefix(location, "http://") || strings.HasPrefix(location, "https://") return strings.HasPrefix(location, "http://") || strings.HasPrefix(location, "https://")
} }
func openSourceImage(location string) (io.ReadCloser, int64, error) { func openSourceImage(location string) (*imageFile, error) {
if isURL(location) { if isURL(location) {
return openFromURL(location) return openFromURL(location)
} }
...@@ -329,35 +343,41 @@ func openSourceImage(location string) (io.ReadCloser, int64, error) { ...@@ -329,35 +343,41 @@ func openSourceImage(location string) (io.ReadCloser, int64, error) {
return openFromFile(location) return openFromFile(location)
} }
func openFromURL(location string) (io.ReadCloser, int64, error) { func openFromURL(location string) (*imageFile, error) {
res, err := httpClient.Get(location) res, err := httpClient.Get(location)
if err != nil { if err != nil {
return nil, 0, err return nil, err
} }
if res.StatusCode != http.StatusOK { switch res.StatusCode {
case http.StatusOK, http.StatusNotModified:
// Extract headers for conditional GETs from response.
lastModified, err := http.ParseTime(res.Header.Get("Last-Modified"))
if err != nil {
// This is unlikely to happen, coming from an object storage provider.
lastModified = time.Now().UTC()
}
return &imageFile{res.Body, res.ContentLength, lastModified}, nil
default:
res.Body.Close() res.Body.Close()
return nil, fmt.Errorf("unexpected upstream response for %q: %d %s",
return nil, 0, fmt.Errorf("ImageResizer: cannot read data from %q: %d %s",
location, res.StatusCode, res.Status) location, res.StatusCode, res.Status)
} }
return res.Body, res.ContentLength, nil
} }
func openFromFile(location string) (io.ReadCloser, int64, error) { func openFromFile(location string) (*imageFile, error) {
file, err := os.Open(location) file, err := os.Open(location)
if err != nil { if err != nil {
return file, 0, err return nil, err
} }
fi, err := file.Stat() fi, err := file.Stat()
if err != nil { if err != nil {
return file, 0, err file.Close()
return nil, err
} }
return file, fi.Size(), nil return &imageFile{file, fi.Size(), fi.ModTime()}, nil
} }
// Only allow more scaling requests if we haven't yet reached the maximum // Only allow more scaling requests if we haven't yet reached the maximum
......
// This file contains code derived from https://github.com/golang/go/blob/master/src/net/http/fs.go
//
// Copyright 2020 GitLab Inc. All rights reserved.
// Copyright 2009 The Go Authors. All rights reserved.
package imageresizer
import (
"net/http"
"time"
)
func checkNotModified(r *http.Request, modtime time.Time) bool {
ims := r.Header.Get("If-Modified-Since")
if ims == "" || isZeroTime(modtime) {
// Treat bogus times as if there was no such header at all
return false
}
t, err := http.ParseTime(ims)
if err != nil {
return false
}
// The Last-Modified header truncates sub-second precision so
// the modtime needs to be truncated too.
return !modtime.Truncate(time.Second).After(t)
}
// isZeroTime reports whether t is obviously unspecified (either zero or Unix epoch time).
func isZeroTime(t time.Time) bool {
return t.IsZero() || t.Equal(time.Unix(0, 0))
}
func setLastModified(w http.ResponseWriter, modtime time.Time) {
if !isZeroTime(modtime) {
w.Header().Set("Last-Modified", modtime.UTC().Format(http.TimeFormat))
}
}
func writeNotModified(w http.ResponseWriter) {
h := w.Header()
h.Del("Content-Type")
h.Del("Content-Length")
w.WriteHeader(http.StatusNotModified)
}
...@@ -10,6 +10,7 @@ import ( ...@@ -10,6 +10,7 @@ import (
"net/http/httptest" "net/http/httptest"
"os" "os"
"testing" "testing"
"time"
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
"gitlab.com/gitlab-org/labkit/log" "gitlab.com/gitlab-org/labkit/log"
...@@ -32,6 +33,9 @@ func TestMain(m *testing.M) { ...@@ -32,6 +33,9 @@ func TestMain(m *testing.M) {
func requestScaledImage(t *testing.T, httpHeaders http.Header, params resizeParams, cfg config.ImageResizerConfig) *http.Response { func requestScaledImage(t *testing.T, httpHeaders http.Header, params resizeParams, cfg config.ImageResizerConfig) *http.Response {
httpRequest := httptest.NewRequest("GET", "/image", nil) httpRequest := httptest.NewRequest("GET", "/image", nil)
if httpHeaders != nil {
httpRequest.Header = httpHeaders
}
responseWriter := httptest.NewRecorder() responseWriter := httptest.NewRecorder()
paramsJSON := encodeParams(t, &params) paramsJSON := encodeParams(t, &params)
...@@ -78,6 +82,46 @@ func TestRequestScaledImageFromPath(t *testing.T) { ...@@ -78,6 +82,46 @@ func TestRequestScaledImageFromPath(t *testing.T) {
} }
} }
func TestRequestScaledImageWithConditionalGetAndImageNotChanged(t *testing.T) {
cfg := config.DefaultImageResizerConfig
params := resizeParams{Location: imagePath, ContentType: "image/png", Width: 64}
clientTime := testImageLastModified(t)
header := http.Header{}
header.Set("If-Modified-Since", httpTimeStr(clientTime))
resp := requestScaledImage(t, header, params, cfg)
require.Equal(t, http.StatusNotModified, resp.StatusCode)
require.Equal(t, httpTimeStr(testImageLastModified(t)), resp.Header.Get("Last-Modified"))
require.Empty(t, resp.Header.Get("Content-Type"))
require.Empty(t, resp.Header.Get("Content-Length"))
}
func TestRequestScaledImageWithConditionalGetAndImageChanged(t *testing.T) {
cfg := config.DefaultImageResizerConfig
params := resizeParams{Location: imagePath, ContentType: "image/png", Width: 64}
clientTime := testImageLastModified(t).Add(-1 * time.Second)
header := http.Header{}
header.Set("If-Modified-Since", httpTimeStr(clientTime))
resp := requestScaledImage(t, header, params, cfg)
require.Equal(t, http.StatusOK, resp.StatusCode)
require.Equal(t, httpTimeStr(testImageLastModified(t)), resp.Header.Get("Last-Modified"))
}
func TestRequestScaledImageWithConditionalGetAndInvalidClientTime(t *testing.T) {
cfg := config.DefaultImageResizerConfig
params := resizeParams{Location: imagePath, ContentType: "image/png", Width: 64}
header := http.Header{}
header.Set("If-Modified-Since", "0")
resp := requestScaledImage(t, header, params, cfg)
require.Equal(t, http.StatusOK, resp.StatusCode)
require.Equal(t, httpTimeStr(testImageLastModified(t)), resp.Header.Get("Last-Modified"))
}
func TestServeOriginalImageWhenSourceImageTooLarge(t *testing.T) { func TestServeOriginalImageWhenSourceImageTooLarge(t *testing.T) {
originalImage := testImage(t) originalImage := testImage(t)
cfg := config.ImageResizerConfig{MaxScalerProcs: 1, MaxFilesize: 1} cfg := config.ImageResizerConfig{MaxScalerProcs: 1, MaxFilesize: 1}
...@@ -174,8 +218,19 @@ func testImage(t *testing.T) image.Image { ...@@ -174,8 +218,19 @@ func testImage(t *testing.T) image.Image {
return image return image
} }
func testImageLastModified(t *testing.T) time.Time {
fi, err := os.Stat(imagePath)
require.NoError(t, err)
return fi.ModTime()
}
func imageFromResponse(t *testing.T, resp *http.Response) image.Image { func imageFromResponse(t *testing.T, resp *http.Response) image.Image {
img, _, err := image.Decode(resp.Body) img, _, err := image.Decode(resp.Body)
require.NoError(t, err, "decode resized image") require.NoError(t, err, "decode resized image")
return img return img
} }
func httpTimeStr(time time.Time) string {
return time.UTC().Format(http.TimeFormat)
}
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment