allow configuring scanner cycles dynamically (#11931)

This allows us to speed up or slow down sleeps
between multiple scanner cycles, helps in testing
as well as some deployments might want to run
scanner more frequently.

This change is also dynamic can be applied on
a running cluster, subsequent cycles pickup
the newly set value.
This commit is contained in:
Harshavardhana 2021-03-30 13:59:02 -07:00 committed by GitHub
parent e9fede88b3
commit 014edd3462
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 49 additions and 11 deletions

View File

@ -624,6 +624,8 @@ func applyDynamicConfig(ctx context.Context, objAPI ObjectLayer, s config.Config
globalHealConfig = healCfg globalHealConfig = healCfg
globalHealConfigMu.Unlock() globalHealConfigMu.Unlock()
// update dynamic scanner values.
scannerCycle.Update(scannerCfg.Cycle)
logger.LogIf(ctx, scannerSleeper.Update(scannerCfg.Delay, scannerCfg.MaxWait)) logger.LogIf(ctx, scannerSleeper.Update(scannerCfg.Delay, scannerCfg.MaxWait))
// Update all dynamic config values in memory. // Update all dynamic config values in memory.

View File

@ -1,5 +1,5 @@
/* /*
* MinIO Cloud Storage, (C) 2020 MinIO, Inc. * MinIO Cloud Storage, (C) 2020-2021 MinIO, Inc.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -28,8 +28,10 @@ import (
const ( const (
Delay = "delay" Delay = "delay"
MaxWait = "max_wait" MaxWait = "max_wait"
Cycle = "cycle"
EnvDelay = "MINIO_SCANNER_DELAY" EnvDelay = "MINIO_SCANNER_DELAY"
EnvCycle = "MINIO_SCANNER_CYCLE"
EnvDelayLegacy = "MINIO_CRAWLER_DELAY" EnvDelayLegacy = "MINIO_CRAWLER_DELAY"
EnvMaxWait = "MINIO_SCANNER_MAX_WAIT" EnvMaxWait = "MINIO_SCANNER_MAX_WAIT"
EnvMaxWaitLegacy = "MINIO_CRAWLER_MAX_WAIT" EnvMaxWaitLegacy = "MINIO_CRAWLER_MAX_WAIT"
@ -41,6 +43,8 @@ type Config struct {
Delay float64 `json:"delay"` Delay float64 `json:"delay"`
// MaxWait is maximum wait time between operations // MaxWait is maximum wait time between operations
MaxWait time.Duration MaxWait time.Duration
// Cycle is the time.Duration between each scanner cycles
Cycle time.Duration
} }
var ( var (
@ -54,6 +58,10 @@ var (
Key: MaxWait, Key: MaxWait,
Value: "15s", Value: "15s",
}, },
config.KV{
Key: Cycle,
Value: "1m",
},
} }
// Help provides help for config values // Help provides help for config values
@ -70,6 +78,12 @@ var (
Optional: true, Optional: true,
Type: "duration", Type: "duration",
}, },
config.HelpKV{
Key: Cycle,
Description: `time duration between scanner cycles, defaults to '1m'`,
Optional: true,
Type: "duration",
},
} }
) )
@ -94,5 +108,10 @@ func LookupConfig(kvs config.KVS) (cfg Config, err error) {
if err != nil { if err != nil {
return cfg, err return cfg, err
} }
cfg.Cycle, err = time.ParseDuration(env.Get(EnvCycle, kvs.Get(Cycle)))
if err != nil {
return cfg, err
}
return cfg, nil return cfg, nil
} }

View File

@ -44,7 +44,6 @@ import (
const ( const (
dataScannerSleepPerFolder = time.Millisecond // Time to wait between folders. dataScannerSleepPerFolder = time.Millisecond // Time to wait between folders.
dataScannerStartDelay = 1 * time.Minute // Time to wait on startup and between cycles.
dataUsageUpdateDirCycles = 16 // Visit all folders every n cycles. dataUsageUpdateDirCycles = 16 // Visit all folders every n cycles.
healDeleteDangling = true healDeleteDangling = true
@ -59,6 +58,7 @@ var (
dataScannerLeaderLockTimeout = newDynamicTimeout(30*time.Second, 10*time.Second) dataScannerLeaderLockTimeout = newDynamicTimeout(30*time.Second, 10*time.Second)
// Sleeper values are updated when config is loaded. // Sleeper values are updated when config is loaded.
scannerSleeper = newDynamicSleeper(10, 10*time.Second) scannerSleeper = newDynamicSleeper(10, 10*time.Second)
scannerCycle = &safeDuration{}
) )
// initDataScanner will start the scanner in the background. // initDataScanner will start the scanner in the background.
@ -66,6 +66,23 @@ func initDataScanner(ctx context.Context, objAPI ObjectLayer) {
go runDataScanner(ctx, objAPI) go runDataScanner(ctx, objAPI)
} }
type safeDuration struct {
sync.Mutex
t time.Duration
}
func (s *safeDuration) Update(t time.Duration) {
s.Lock()
defer s.Unlock()
s.t = t
}
func (s *safeDuration) Get() time.Duration {
s.Lock()
defer s.Unlock()
return s.t
}
// runDataScanner will start a data scanner. // runDataScanner will start a data scanner.
// The function will block until the context is canceled. // The function will block until the context is canceled.
// There should only ever be one scanner running per cluster. // There should only ever be one scanner running per cluster.
@ -77,7 +94,7 @@ func runDataScanner(ctx context.Context, objAPI ObjectLayer) {
for { for {
ctx, err = locker.GetLock(ctx, dataScannerLeaderLockTimeout) ctx, err = locker.GetLock(ctx, dataScannerLeaderLockTimeout)
if err != nil { if err != nil {
time.Sleep(time.Duration(r.Float64() * float64(dataScannerStartDelay))) time.Sleep(time.Duration(r.Float64() * float64(scannerCycle.Get())))
continue continue
} }
break break
@ -101,7 +118,7 @@ func runDataScanner(ctx context.Context, objAPI ObjectLayer) {
br.Close() br.Close()
} }
scannerTimer := time.NewTimer(dataScannerStartDelay) scannerTimer := time.NewTimer(scannerCycle.Get())
defer scannerTimer.Stop() defer scannerTimer.Stop()
for { for {
@ -110,7 +127,7 @@ func runDataScanner(ctx context.Context, objAPI ObjectLayer) {
return return
case <-scannerTimer.C: case <-scannerTimer.C:
// Reset the timer for next cycle. // Reset the timer for next cycle.
scannerTimer.Reset(dataScannerStartDelay) scannerTimer.Reset(scannerCycle.Get())
if intDataUpdateTracker.debug { if intDataUpdateTracker.debug {
console.Debugln("starting scanner cycle") console.Debugln("starting scanner cycle")

View File

@ -130,8 +130,8 @@ func (h *Target) startHTTPLogger() {
resp, err := h.client.Do(req) resp, err := h.client.Do(req)
cancel() cancel()
if err != nil { if err != nil {
logger.LogIf(ctx, fmt.Errorf("%s returned '%w', please check your endpoint configuration\n", logger.LogOnceIf(ctx, fmt.Errorf("%s returned '%w', please check your endpoint configuration",
h.endpoint, err)) h.endpoint, err), h.endpoint)
continue continue
} }
@ -141,11 +141,11 @@ func (h *Target) startHTTPLogger() {
if resp.StatusCode != http.StatusOK { if resp.StatusCode != http.StatusOK {
switch resp.StatusCode { switch resp.StatusCode {
case http.StatusForbidden: case http.StatusForbidden:
logger.LogIf(ctx, fmt.Errorf("%s returned '%s', please check if your auth token is correctly set", logger.LogOnceIf(ctx, fmt.Errorf("%s returned '%s', please check if your auth token is correctly set",
h.endpoint, resp.Status)) h.endpoint, resp.Status), h.endpoint)
default: default:
logger.LogIf(ctx, fmt.Errorf("%s returned '%s', please check your endpoint configuration", logger.LogOnceIf(ctx, fmt.Errorf("%s returned '%s', please check your endpoint configuration",
h.endpoint, resp.Status)) h.endpoint, resp.Status), h.endpoint)
} }
} }
} }