mirror of
https://github.com/minio/minio.git
synced 2025-04-04 03:40:30 -04:00
Support dynamic reset of minio config (#13626)
If a given MinIO config is dynamic (can be changed without restart), ensure that it can be reset also without restart. Signed-off-by: Shireesh Anjal <shireesh@minio.io>
This commit is contained in:
parent
ea820b30bf
commit
d008e90d50
@ -19,6 +19,7 @@ package cmd
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
|
"context"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"io"
|
"io"
|
||||||
"net/http"
|
"net/http"
|
||||||
@ -78,6 +79,22 @@ func (a adminAPIHandlers) DelConfigKVHandler(w http.ResponseWriter, r *http.Requ
|
|||||||
writeErrorResponseJSON(ctx, w, toAdminAPIErr(ctx, err), r.URL)
|
writeErrorResponseJSON(ctx, w, toAdminAPIErr(ctx, err), r.URL)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
dynamic := config.SubSystemsDynamic.Contains(string(kvBytes))
|
||||||
|
if dynamic {
|
||||||
|
applyDynamic(ctx, objectAPI, cfg, r, w)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func applyDynamic(ctx context.Context, objectAPI ObjectLayer, cfg config.Config, r *http.Request, w http.ResponseWriter) {
|
||||||
|
// Apply dynamic values.
|
||||||
|
if err := applyDynamicConfig(GlobalContext, objectAPI, cfg); err != nil {
|
||||||
|
writeErrorResponseJSON(ctx, w, toAdminAPIErr(ctx, err), r.URL)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
globalNotificationSys.SignalService(serviceReloadDynamic)
|
||||||
|
// Tell the client that dynamic config was applied.
|
||||||
|
w.Header().Set(madmin.ConfigAppliedHeader, madmin.ConfigAppliedTrue)
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetConfigKVHandler - PUT /minio/admin/v3/set-config-kv
|
// SetConfigKVHandler - PUT /minio/admin/v3/set-config-kv
|
||||||
@ -135,14 +152,7 @@ func (a adminAPIHandlers) SetConfigKVHandler(w http.ResponseWriter, r *http.Requ
|
|||||||
}
|
}
|
||||||
|
|
||||||
if dynamic {
|
if dynamic {
|
||||||
// Apply dynamic values.
|
applyDynamic(ctx, objectAPI, cfg, r, w)
|
||||||
if err := applyDynamicConfig(GlobalContext, objectAPI, cfg); err != nil {
|
|
||||||
writeErrorResponseJSON(ctx, w, toAdminAPIErr(ctx, err), r.URL)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
globalNotificationSys.SignalService(serviceReloadDynamic)
|
|
||||||
// If all values were dynamic, tell the client.
|
|
||||||
w.Header().Set(madmin.ConfigAppliedHeader, madmin.ConfigAppliedTrue)
|
|
||||||
}
|
}
|
||||||
writeSuccessResponseHeadersOnly(w)
|
writeSuccessResponseHeadersOnly(w)
|
||||||
}
|
}
|
||||||
|
@ -217,6 +217,20 @@ func (kvs KVS) Empty() bool {
|
|||||||
return len(kvs) == 0
|
return len(kvs) == 0
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Clone - returns a copy of the KVS
|
||||||
|
func (kvs KVS) Clone() KVS {
|
||||||
|
return append(make(KVS, 0, len(kvs)), kvs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetWithDefault - returns default value if key not set
|
||||||
|
func (kvs KVS) GetWithDefault(key string, defaultKVS KVS) string {
|
||||||
|
v := kvs.Get(key)
|
||||||
|
if len(v) == 0 {
|
||||||
|
return defaultKVS.Get(key)
|
||||||
|
}
|
||||||
|
return v
|
||||||
|
}
|
||||||
|
|
||||||
// Keys returns the list of keys for the current KVS
|
// Keys returns the list of keys for the current KVS
|
||||||
func (kvs KVS) Keys() []string {
|
func (kvs KVS) Keys() []string {
|
||||||
var keys = make([]string, len(kvs))
|
var keys = make([]string, len(kvs))
|
||||||
@ -759,10 +773,12 @@ func (c Config) SetKVS(s string, defaultKVS map[string]KVS) (dynamic bool, err e
|
|||||||
kvs.Set(Enable, EnableOn)
|
kvs.Set(Enable, EnableOn)
|
||||||
}
|
}
|
||||||
|
|
||||||
currKVS, ok := c[subSys][tgt]
|
var currKVS KVS
|
||||||
|
ck, ok := c[subSys][tgt]
|
||||||
if !ok {
|
if !ok {
|
||||||
currKVS = defaultKVS[subSys]
|
currKVS = defaultKVS[subSys].Clone()
|
||||||
} else {
|
} else {
|
||||||
|
currKVS = ck.Clone()
|
||||||
for _, kv := range defaultKVS[subSys] {
|
for _, kv := range defaultKVS[subSys] {
|
||||||
if _, ok = currKVS.Lookup(kv.Key); !ok {
|
if _, ok = currKVS.Lookup(kv.Key); !ok {
|
||||||
currKVS.Set(kv.Key, kv.Value)
|
currKVS.Set(kv.Key, kv.Value)
|
||||||
|
@ -150,15 +150,15 @@ func LookupConfig(kvs config.KVS) (cfg Config, err error) {
|
|||||||
if err = config.CheckValidKeys(config.HealSubSys, kvs, DefaultKVS); err != nil {
|
if err = config.CheckValidKeys(config.HealSubSys, kvs, DefaultKVS); err != nil {
|
||||||
return cfg, err
|
return cfg, err
|
||||||
}
|
}
|
||||||
cfg.Bitrot, err = config.ParseBool(env.Get(EnvBitrot, kvs.Get(Bitrot)))
|
cfg.Bitrot, err = config.ParseBool(env.Get(EnvBitrot, kvs.GetWithDefault(Bitrot, DefaultKVS)))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cfg, fmt.Errorf("'heal:bitrotscan' value invalid: %w", err)
|
return cfg, fmt.Errorf("'heal:bitrotscan' value invalid: %w", err)
|
||||||
}
|
}
|
||||||
cfg.Sleep, err = time.ParseDuration(env.Get(EnvSleep, kvs.Get(Sleep)))
|
cfg.Sleep, err = time.ParseDuration(env.Get(EnvSleep, kvs.GetWithDefault(Sleep, DefaultKVS)))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cfg, fmt.Errorf("'heal:max_sleep' value invalid: %w", err)
|
return cfg, fmt.Errorf("'heal:max_sleep' value invalid: %w", err)
|
||||||
}
|
}
|
||||||
cfg.IOCount, err = strconv.Atoi(env.Get(EnvIOCount, kvs.Get(IOCount)))
|
cfg.IOCount, err = strconv.Atoi(env.Get(EnvIOCount, kvs.GetWithDefault(IOCount, DefaultKVS)))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cfg, fmt.Errorf("'heal:max_io' value invalid: %w", err)
|
return cfg, fmt.Errorf("'heal:max_io' value invalid: %w", err)
|
||||||
}
|
}
|
||||||
|
@ -95,7 +95,7 @@ func LookupConfig(kvs config.KVS) (cfg Config, err error) {
|
|||||||
}
|
}
|
||||||
delay := env.Get(EnvDelayLegacy, "")
|
delay := env.Get(EnvDelayLegacy, "")
|
||||||
if delay == "" {
|
if delay == "" {
|
||||||
delay = env.Get(EnvDelay, kvs.Get(Delay))
|
delay = env.Get(EnvDelay, kvs.GetWithDefault(Delay, DefaultKVS))
|
||||||
}
|
}
|
||||||
cfg.Delay, err = strconv.ParseFloat(delay, 64)
|
cfg.Delay, err = strconv.ParseFloat(delay, 64)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -103,14 +103,14 @@ func LookupConfig(kvs config.KVS) (cfg Config, err error) {
|
|||||||
}
|
}
|
||||||
maxWait := env.Get(EnvMaxWaitLegacy, "")
|
maxWait := env.Get(EnvMaxWaitLegacy, "")
|
||||||
if maxWait == "" {
|
if maxWait == "" {
|
||||||
maxWait = env.Get(EnvMaxWait, kvs.Get(MaxWait))
|
maxWait = env.Get(EnvMaxWait, kvs.GetWithDefault(MaxWait, DefaultKVS))
|
||||||
}
|
}
|
||||||
cfg.MaxWait, err = time.ParseDuration(maxWait)
|
cfg.MaxWait, err = time.ParseDuration(maxWait)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cfg, err
|
return cfg, err
|
||||||
}
|
}
|
||||||
|
|
||||||
cfg.Cycle, err = time.ParseDuration(env.Get(EnvCycle, kvs.Get(Cycle)))
|
cfg.Cycle, err = time.ParseDuration(env.Get(EnvCycle, kvs.GetWithDefault(Cycle, DefaultKVS)))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cfg, err
|
return cfg, err
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user