Commit 7a5cac7d authored by Matthias Käppler's avatar Matthias Käppler Committed by Jacob Vosmaer

Set default max image scaler procs to num_cpu / 2

This is a more realistic value for a single node setup and development.

The recommendation is to use num_cores / 2 as a guideline.
parent ddd53598
---
title: Default MaxScalerProcs to num_cores / 2
merge_request: 635
author:
type: other
...@@ -13,5 +13,5 @@ URL = "unix:/home/git/gitlab/redis/redis.socket" ...@@ -13,5 +13,5 @@ URL = "unix:/home/git/gitlab/redis/redis.socket"
azure_storage_access_key = "YOUR ACCOUNT KEY" azure_storage_access_key = "YOUR ACCOUNT KEY"
[image_resizer] [image_resizer]
max_scaler_procs = 100 max_scaler_procs = 4 # Recommendation: CPUs / 2
max_filesize = 250000 max_filesize = 250000
package config package config
import ( import (
"math"
"net/url" "net/url"
"runtime"
"strings" "strings"
"time" "time"
...@@ -105,14 +107,9 @@ type Config struct { ...@@ -105,14 +107,9 @@ type Config struct {
ImageResizerConfig *ImageResizerConfig `toml:"image_resizer"` ImageResizerConfig *ImageResizerConfig `toml:"image_resizer"`
} }
const (
DefaultImageResizerMaxScalerProcs = 100
DefaultImageResizerMaxFilesize = 250 * 1000 // 250kB
)
var DefaultImageResizerConfig = &ImageResizerConfig{ var DefaultImageResizerConfig = &ImageResizerConfig{
MaxScalerProcs: DefaultImageResizerMaxScalerProcs, MaxScalerProcs: uint32(math.Max(2, float64(runtime.NumCPU())/2)),
MaxFilesize: DefaultImageResizerMaxFilesize, MaxFilesize: 250 * 1000, // 250kB,
} }
func LoadConfig(data string) (*Config, error) { func LoadConfig(data string) (*Config, error) {
......
...@@ -21,14 +21,8 @@ func TestLoadEmptyConfig(t *testing.T) { ...@@ -21,14 +21,8 @@ func TestLoadEmptyConfig(t *testing.T) {
cfg, err := LoadConfig(config) cfg, err := LoadConfig(config)
require.NoError(t, err) require.NoError(t, err)
expected := Config{ require.Equal(t, cfg.ImageResizerConfig.MaxFilesize, uint64(250000))
ImageResizerConfig: &ImageResizerConfig{ require.GreaterOrEqual(t, cfg.ImageResizerConfig.MaxScalerProcs, uint32(2))
MaxScalerProcs: DefaultImageResizerMaxScalerProcs,
MaxFilesize: DefaultImageResizerMaxFilesize,
},
}
require.Equal(t, expected, *cfg)
require.Nil(t, cfg.ObjectStorageCredentials) require.Nil(t, cfg.ObjectStorageCredentials)
require.NoError(t, cfg.RegisterGoCloudURLOpeners()) require.NoError(t, cfg.RegisterGoCloudURLOpeners())
......
...@@ -30,6 +30,7 @@ import ( ...@@ -30,6 +30,7 @@ import (
type Resizer struct { type Resizer struct {
config.Config config.Config
senddata.Prefix senddata.Prefix
numScalerProcs processCounter
} }
type resizeParams struct { type resizeParams struct {
...@@ -42,8 +43,6 @@ type processCounter struct { ...@@ -42,8 +43,6 @@ type processCounter struct {
n int32 n int32
} }
var numScalerProcs processCounter
var envInjector = tracing.NewEnvInjector() var envInjector = tracing.NewEnvInjector()
// Images might be located remotely in object storage, in which case we need to stream // Images might be located remotely in object storage, in which case we need to stream
...@@ -122,7 +121,7 @@ func init() { ...@@ -122,7 +121,7 @@ func init() {
} }
func NewResizer(cfg config.Config) *Resizer { func NewResizer(cfg config.Config) *Resizer {
return &Resizer{cfg, "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 // This Injecter forks into a dedicated scaler process to resize an image identified by path or URL
...@@ -158,7 +157,7 @@ func (r *Resizer) Inject(w http.ResponseWriter, req *http.Request, paramsData st ...@@ -158,7 +157,7 @@ func (r *Resizer) Inject(w http.ResponseWriter, req *http.Request, paramsData st
// 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 := tryResizeImage(req, sourceImageReader, logger.Writer(), params, fileSize, r.Config.ImageResizerConfig) imageReader, resizeCmd, err := r.tryResizeImage(req, sourceImageReader, logger.Writer(), params, fileSize, r.Config.ImageResizerConfig)
if err != nil { if err != nil {
// something failed, but we can still write out the original image, do don't return early // something failed, but we can still write out the original image, do don't return early
helper.LogErrorWithFields(req, err, *logFields(0)) helper.LogErrorWithFields(req, err, *logFields(0))
...@@ -228,24 +227,24 @@ func (r *Resizer) unpackParameters(paramsData string) (*resizeParams, error) { ...@@ -228,24 +227,24 @@ 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 tryResizeImage(req *http.Request, r io.Reader, errorWriter io.Writer, params *resizeParams, fileSize int64, cfg *config.ImageResizerConfig) (io.Reader, *exec.Cmd, error) { 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) {
if fileSize > int64(cfg.MaxFilesize) { if fileSize > int64(cfg.MaxFilesize) {
return r, nil, fmt.Errorf("ImageResizer: %db exceeds maximum file size of %db", fileSize, cfg.MaxFilesize) return reader, nil, fmt.Errorf("ImageResizer: %db exceeds maximum file size of %db", fileSize, cfg.MaxFilesize)
} }
if !numScalerProcs.tryIncrement(int32(cfg.MaxScalerProcs)) { if !r.numScalerProcs.tryIncrement(int32(cfg.MaxScalerProcs)) {
return r, nil, fmt.Errorf("ImageResizer: too many running scaler processes") return reader, nil, fmt.Errorf("ImageResizer: too many running scaler processes (%d / %d)", r.numScalerProcs.n, cfg.MaxScalerProcs)
} }
ctx := req.Context() ctx := req.Context()
go func() { go func() {
<-ctx.Done() <-ctx.Done()
numScalerProcs.decrement() r.numScalerProcs.decrement()
}() }()
resizeCmd, resizedImageReader, err := startResizeImageCommand(ctx, r, errorWriter, params) resizeCmd, resizedImageReader, err := startResizeImageCommand(ctx, reader, errorWriter, params)
if err != nil { if err != nil {
return r, nil, fmt.Errorf("ImageResizer: failed forking into scaler process: %w", err) return reader, nil, fmt.Errorf("ImageResizer: failed forking into scaler process: %w", err)
} }
return resizedImageReader, resizeCmd, nil return resizedImageReader, resizeCmd, nil
} }
......
...@@ -17,8 +17,6 @@ import ( ...@@ -17,8 +17,6 @@ import (
"gitlab.com/gitlab-org/gitlab-workhorse/internal/testhelper" "gitlab.com/gitlab-org/gitlab-workhorse/internal/testhelper"
) )
var r = Resizer{}
func TestMain(m *testing.M) { func TestMain(m *testing.M) {
if err := testhelper.BuildExecutables(); err != nil { if err := testhelper.BuildExecutables(); err != nil {
log.WithError(err).Fatal() log.WithError(err).Fatal()
...@@ -28,6 +26,7 @@ func TestMain(m *testing.M) { ...@@ -28,6 +26,7 @@ func TestMain(m *testing.M) {
} }
func TestUnpackParametersReturnsParamsInstanceForValidInput(t *testing.T) { func TestUnpackParametersReturnsParamsInstanceForValidInput(t *testing.T) {
r := Resizer{}
inParams := resizeParams{Location: "/path/to/img", Width: 64, ContentType: "image/png"} inParams := resizeParams{Location: "/path/to/img", Width: 64, ContentType: "image/png"}
outParams, err := r.unpackParameters(encodeParams(t, &inParams)) outParams, err := r.unpackParameters(encodeParams(t, &inParams))
...@@ -37,6 +36,7 @@ func TestUnpackParametersReturnsParamsInstanceForValidInput(t *testing.T) { ...@@ -37,6 +36,7 @@ func TestUnpackParametersReturnsParamsInstanceForValidInput(t *testing.T) {
} }
func TestUnpackParametersReturnsErrorWhenLocationBlank(t *testing.T) { func TestUnpackParametersReturnsErrorWhenLocationBlank(t *testing.T) {
r := Resizer{}
inParams := resizeParams{Location: "", Width: 64, ContentType: "image/jpg"} inParams := resizeParams{Location: "", Width: 64, ContentType: "image/jpg"}
_, err := r.unpackParameters(encodeParams(t, &inParams)) _, err := r.unpackParameters(encodeParams(t, &inParams))
...@@ -45,6 +45,7 @@ func TestUnpackParametersReturnsErrorWhenLocationBlank(t *testing.T) { ...@@ -45,6 +45,7 @@ func TestUnpackParametersReturnsErrorWhenLocationBlank(t *testing.T) {
} }
func TestUnpackParametersReturnsErrorWhenContentTypeBlank(t *testing.T) { func TestUnpackParametersReturnsErrorWhenContentTypeBlank(t *testing.T) {
r := Resizer{}
inParams := resizeParams{Location: "/path/to/img", Width: 64, ContentType: ""} inParams := resizeParams{Location: "/path/to/img", Width: 64, ContentType: ""}
_, err := r.unpackParameters(encodeParams(t, &inParams)) _, err := r.unpackParameters(encodeParams(t, &inParams))
...@@ -53,12 +54,13 @@ func TestUnpackParametersReturnsErrorWhenContentTypeBlank(t *testing.T) { ...@@ -53,12 +54,13 @@ func TestUnpackParametersReturnsErrorWhenContentTypeBlank(t *testing.T) {
} }
func TestTryResizeImageSuccess(t *testing.T) { func TestTryResizeImageSuccess(t *testing.T) {
r := Resizer{}
inParams := resizeParams{Location: "/path/to/img", Width: 64, ContentType: "image/png"} inParams := resizeParams{Location: "/path/to/img", Width: 64, ContentType: "image/png"}
inFile := testImage(t) inFile := testImage(t)
req, err := http.NewRequest("GET", "/foo", nil) req, err := http.NewRequest("GET", "/foo", nil)
require.NoError(t, err) require.NoError(t, err)
reader, cmd, err := tryResizeImage( reader, cmd, err := r.tryResizeImage(
req, req,
inFile, inFile,
os.Stderr, os.Stderr,
...@@ -74,12 +76,13 @@ func TestTryResizeImageSuccess(t *testing.T) { ...@@ -74,12 +76,13 @@ func TestTryResizeImageSuccess(t *testing.T) {
} }
func TestTryResizeImageSkipsResizeWhenSourceImageTooLarge(t *testing.T) { func TestTryResizeImageSkipsResizeWhenSourceImageTooLarge(t *testing.T) {
r := Resizer{}
inParams := resizeParams{Location: "/path/to/img", Width: 64, ContentType: "image/png"} inParams := resizeParams{Location: "/path/to/img", Width: 64, ContentType: "image/png"}
inFile := testImage(t) inFile := testImage(t)
req, err := http.NewRequest("GET", "/foo", nil) req, err := http.NewRequest("GET", "/foo", nil)
require.NoError(t, err) require.NoError(t, err)
reader, cmd, err := tryResizeImage( reader, cmd, err := r.tryResizeImage(
req, req,
inFile, inFile,
os.Stderr, os.Stderr,
...@@ -94,12 +97,13 @@ func TestTryResizeImageSkipsResizeWhenSourceImageTooLarge(t *testing.T) { ...@@ -94,12 +97,13 @@ func TestTryResizeImageSkipsResizeWhenSourceImageTooLarge(t *testing.T) {
} }
func TestTryResizeImageFailsWhenContentTypeNotMatchingFileContents(t *testing.T) { func TestTryResizeImageFailsWhenContentTypeNotMatchingFileContents(t *testing.T) {
r := Resizer{}
inParams := resizeParams{Location: "/path/to/img", Width: 64, ContentType: "image/jpeg"} inParams := resizeParams{Location: "/path/to/img", Width: 64, ContentType: "image/jpeg"}
inFile := testImage(t) // this is a PNG file; the image scaler should fail fast in this case inFile := testImage(t) // this is a PNG file; the image scaler should fail fast in this case
req, err := http.NewRequest("GET", "/foo", nil) req, err := http.NewRequest("GET", "/foo", nil)
require.NoError(t, err) require.NoError(t, err)
_, cmd, err := tryResizeImage( _, cmd, err := r.tryResizeImage(
req, req,
inFile, inFile,
os.Stderr, os.Stderr,
......
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