fix: when logger queue is full exit quickly upon doneCh (#14928)

Additionally only reload requested sub-system not everything
This commit is contained in:
Harshavardhana 2022-05-16 16:10:51 -07:00 committed by GitHub
parent 05685863e3
commit 040ac5cad8
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 59 additions and 24 deletions

View File

@ -104,7 +104,7 @@ func applyDynamic(ctx context.Context, objectAPI ObjectLayer, cfg config.Config,
writeErrorResponseJSON(ctx, w, toAdminAPIErr(ctx, err), r.URL) writeErrorResponseJSON(ctx, w, toAdminAPIErr(ctx, err), r.URL)
return return
} }
globalNotificationSys.SignalService(serviceReloadDynamic) globalNotificationSys.SignalConfigReload(subSys)
// Tell the client that dynamic config was applied. // Tell the client that dynamic config was applied.
w.Header().Set(madmin.ConfigAppliedHeader, madmin.ConfigAppliedTrue) w.Header().Set(madmin.ConfigAppliedHeader, madmin.ConfigAppliedTrue)
} }

View File

@ -414,6 +414,21 @@ func (sys *NotificationSys) ServerUpdate(ctx context.Context, u *url.URL, sha256
return ng.Wait() return ng.Wait()
} }
// SignalConfigReload reloads requested sub-system on a remote peer dynamically.
func (sys *NotificationSys) SignalConfigReload(subSys string) []NotificationPeerErr {
ng := WithNPeers(len(sys.peerClients))
for idx, client := range sys.peerClients {
if client == nil {
continue
}
client := client
ng.Go(GlobalContext, func() error {
return client.SignalService(serviceReloadDynamic, subSys)
}, idx, *client.host)
}
return ng.Wait()
}
// SignalService - calls signal service RPC call on all peers. // SignalService - calls signal service RPC call on all peers.
func (sys *NotificationSys) SignalService(sig serviceSignal) []NotificationPeerErr { func (sys *NotificationSys) SignalService(sig serviceSignal) []NotificationPeerErr {
ng := WithNPeers(len(sys.peerClients)) ng := WithNPeers(len(sys.peerClients))
@ -423,7 +438,7 @@ func (sys *NotificationSys) SignalService(sig serviceSignal) []NotificationPeerE
} }
client := client client := client
ng.Go(GlobalContext, func() error { ng.Go(GlobalContext, func() error {
return client.SignalService(sig) return client.SignalService(sig, "")
}, idx, *client.host) }, idx, *client.host)
} }
return ng.Wait() return ng.Wait()
@ -1541,7 +1556,7 @@ func (sys *NotificationSys) ServiceFreeze(ctx context.Context, freeze bool) []No
} }
client := client client := client
ng.Go(GlobalContext, func() error { ng.Go(GlobalContext, func() error {
return client.SignalService(serviceSig) return client.SignalService(serviceSig, "")
}, idx, *client.host) }, idx, *client.host)
} }
nerrs := ng.Wait() nerrs := ng.Wait()

View File

@ -688,9 +688,10 @@ func (client *peerRESTClient) ServerUpdate(ctx context.Context, u *url.URL, sha2
} }
// SignalService - sends signal to peer nodes. // SignalService - sends signal to peer nodes.
func (client *peerRESTClient) SignalService(sig serviceSignal) error { func (client *peerRESTClient) SignalService(sig serviceSignal, subSys string) error {
values := make(url.Values) values := make(url.Values)
values.Set(peerRESTSignal, strconv.Itoa(int(sig))) values.Set(peerRESTSignal, strconv.Itoa(int(sig)))
values.Set(peerRESTSubSys, subSys)
respBody, err := client.call(peerRESTMethodSignalService, values, nil, -1) respBody, err := client.call(peerRESTMethodSignalService, values, nil, -1)
if err != nil { if err != nil {
return err return err

View File

@ -84,6 +84,7 @@ const (
peerRESTUserOrGroup = "user-or-group" peerRESTUserOrGroup = "user-or-group"
peerRESTIsGroup = "is-group" peerRESTIsGroup = "is-group"
peerRESTSignal = "signal" peerRESTSignal = "signal"
peerRESTSubSys = "sub-sys"
peerRESTProfiler = "profiler" peerRESTProfiler = "profiler"
peerRESTTraceErr = "err" peerRESTTraceErr = "err"
peerRESTTraceInternal = "internal" peerRESTTraceInternal = "internal"

View File

@ -837,7 +837,14 @@ func (s *peerRESTServer) SignalServiceHandler(w http.ResponseWriter, r *http.Req
s.writeErrorResponse(w, err) s.writeErrorResponse(w, err)
return return
} }
if err = applyDynamicConfig(r.Context(), objAPI, srvCfg); err != nil { subSys := r.Form.Get(peerRESTSubSys)
// Apply dynamic values.
if subSys == "" {
err = applyDynamicConfig(r.Context(), objAPI, srvCfg)
} else {
err = applyDynamicConfigForSubSys(r.Context(), objAPI, srvCfg, subSys)
}
if err != nil {
s.writeErrorResponse(w, err) s.writeErrorResponse(w, err)
} }
return return

View File

@ -26,7 +26,6 @@ import (
"net/http" "net/http"
"strings" "strings"
"sync" "sync"
"sync/atomic"
"time" "time"
xhttp "github.com/minio/minio/internal/http" xhttp "github.com/minio/minio/internal/http"
@ -58,8 +57,8 @@ type Config struct {
// buffer is full, new logs are just ignored and an error // buffer is full, new logs are just ignored and an error
// is returned to the caller. // is returned to the caller.
type Target struct { type Target struct {
status int32
wg sync.WaitGroup wg sync.WaitGroup
doneCh chan struct{}
// Channel of log entries // Channel of log entries
logCh chan interface{} logCh chan interface{}
@ -115,7 +114,6 @@ func (h *Target) Init() error {
h.config.Endpoint, resp.Status) h.config.Endpoint, resp.Status)
} }
h.status = 1
go h.startHTTPLogger() go h.startHTTPLogger()
return nil return nil
} }
@ -177,11 +175,15 @@ func (h *Target) logEntry(entry interface{}) {
func (h *Target) startHTTPLogger() { func (h *Target) startHTTPLogger() {
// Create a routine which sends json logs received // Create a routine which sends json logs received
// from an internal channel. // from an internal channel.
h.wg.Add(1)
go func() { go func() {
h.wg.Add(1)
defer h.wg.Done() defer h.wg.Done()
for entry := range h.logCh {
select {
case entry := <-h.logCh:
h.logEntry(entry) h.logEntry(entry)
case <-h.doneCh:
return
} }
}() }()
} }
@ -191,6 +193,7 @@ func (h *Target) startHTTPLogger() {
func New(config Config) *Target { func New(config Config) *Target {
h := &Target{ h := &Target{
logCh: make(chan interface{}, config.QueueSize), logCh: make(chan interface{}, config.QueueSize),
doneCh: make(chan struct{}),
config: config, config: config,
} }
@ -199,12 +202,14 @@ func New(config Config) *Target {
// Send log message 'e' to http target. // Send log message 'e' to http target.
func (h *Target) Send(entry interface{}, errKind string) error { func (h *Target) Send(entry interface{}, errKind string) error {
if atomic.LoadInt32(&h.status) == 0 { select {
// Channel was closed or used before init. case <-h.doneCh:
return nil return nil
default:
} }
select { select {
case <-h.doneCh:
case h.logCh <- entry: case h.logCh <- entry:
default: default:
// log channel is full, do not wait and return // log channel is full, do not wait and return
@ -217,9 +222,8 @@ func (h *Target) Send(entry interface{}, errKind string) error {
// Cancel - cancels the target // Cancel - cancels the target
func (h *Target) Cancel() { func (h *Target) Cancel() {
if atomic.CompareAndSwapInt32(&h.status, 1, 0) { close(h.doneCh)
close(h.logCh) close(h.logCh)
}
h.wg.Wait() h.wg.Wait()
} }

View File

@ -25,7 +25,6 @@ import (
"errors" "errors"
"net" "net"
"sync" "sync"
"sync/atomic"
"github.com/Shopify/sarama" "github.com/Shopify/sarama"
saramatls "github.com/Shopify/sarama/tools/tls" saramatls "github.com/Shopify/sarama/tools/tls"
@ -37,8 +36,8 @@ import (
// Target - Kafka target. // Target - Kafka target.
type Target struct { type Target struct {
status int32
wg sync.WaitGroup wg sync.WaitGroup
doneCh chan struct{}
// Channel of log entries // Channel of log entries
logCh chan interface{} logCh chan interface{}
@ -51,6 +50,13 @@ type Target struct {
// Send log message 'e' to kafka target. // Send log message 'e' to kafka target.
func (h *Target) Send(entry interface{}, errKind string) error { func (h *Target) Send(entry interface{}, errKind string) error {
select { select {
case <-h.doneCh:
return nil
default:
}
select {
case <-h.doneCh:
case h.logCh <- entry: case h.logCh <- entry:
default: default:
// log channel is full, do not wait and return // log channel is full, do not wait and return
@ -86,11 +92,15 @@ func (h *Target) logEntry(entry interface{}) {
func (h *Target) startKakfaLogger() { func (h *Target) startKakfaLogger() {
// Create a routine which sends json logs received // Create a routine which sends json logs received
// from an internal channel. // from an internal channel.
h.wg.Add(1)
go func() { go func() {
h.wg.Add(1)
defer h.wg.Done() defer h.wg.Done()
for entry := range h.logCh {
select {
case entry := <-h.logCh:
h.logEntry(entry) h.logEntry(entry)
case <-h.doneCh:
return
} }
}() }()
} }
@ -204,17 +214,14 @@ func (h *Target) Init() error {
} }
h.producer = producer h.producer = producer
h.status = 1
go h.startKakfaLogger() go h.startKakfaLogger()
return nil return nil
} }
// Cancel - cancels the target // Cancel - cancels the target
func (h *Target) Cancel() { func (h *Target) Cancel() {
if atomic.CompareAndSwapInt32(&h.status, 1, 0) { close(h.doneCh)
close(h.logCh) close(h.logCh)
}
h.wg.Wait() h.wg.Wait()
} }