mirror of
https://github.com/minio/minio.git
synced 2025-01-07 13:03:22 -05:00
1fc4203c19
- old version was unable to retain messages during config reload - old version could not go from memory to disk during reload - new version can batch disk queue entries to single for to reduce I/O load - error logging has been improved, previous version would miss certain errors. - logic for spawning/despawning additional workers has been adjusted to trigger when half capacity is reached, instead of when the log queue becomes full. - old version would json marshall x2 and unmarshal 1x for every log item. Now we only do marshal x1 and then we GetRaw from the store and send it without having to re-marshal.
154 lines
3.5 KiB
Go
154 lines
3.5 KiB
Go
// Copyright (c) 2015-2023 MinIO, Inc.
|
|
//
|
|
// This file is part of MinIO Object Storage stack
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Affero General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
package store
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"strings"
|
|
"time"
|
|
|
|
xioutil "github.com/minio/minio/internal/ioutil"
|
|
)
|
|
|
|
const (
|
|
retryInterval = 3 * time.Second
|
|
)
|
|
|
|
type logger = func(ctx context.Context, err error, id string, errKind ...interface{})
|
|
|
|
// ErrNotConnected - indicates that the target connection is not active.
|
|
var ErrNotConnected = errors.New("not connected to target server/service")
|
|
|
|
// Target - store target interface
|
|
type Target interface {
|
|
Name() string
|
|
SendFromStore(key Key) error
|
|
}
|
|
|
|
// Store - Used to persist items.
|
|
type Store[I any] interface {
|
|
Put(item I) error
|
|
PutMultiple(item []I) error
|
|
Get(key string) (I, error)
|
|
GetRaw(key string) ([]byte, error)
|
|
Len() int
|
|
List() ([]string, error)
|
|
Del(key string) error
|
|
DelList(key []string) error
|
|
Open() error
|
|
Delete() error
|
|
Extension() string
|
|
}
|
|
|
|
// Key denotes the key present in the store.
|
|
type Key struct {
|
|
Name string
|
|
IsLast bool
|
|
}
|
|
|
|
// replayItems - Reads the items from the store and replays.
|
|
func replayItems[I any](store Store[I], doneCh <-chan struct{}, log logger, id string) <-chan Key {
|
|
keyCh := make(chan Key)
|
|
|
|
go func() {
|
|
defer xioutil.SafeClose(keyCh)
|
|
|
|
retryTicker := time.NewTicker(retryInterval)
|
|
defer retryTicker.Stop()
|
|
|
|
for {
|
|
names, err := store.List()
|
|
if err != nil {
|
|
log(context.Background(), fmt.Errorf("store.List() failed with: %w", err), id)
|
|
} else {
|
|
keyCount := len(names)
|
|
for i, name := range names {
|
|
select {
|
|
case keyCh <- Key{strings.TrimSuffix(name, store.Extension()), keyCount == i+1}:
|
|
// Get next key.
|
|
case <-doneCh:
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
select {
|
|
case <-retryTicker.C:
|
|
case <-doneCh:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
|
|
return keyCh
|
|
}
|
|
|
|
// sendItems - Reads items from the store and re-plays.
|
|
func sendItems(target Target, keyCh <-chan Key, doneCh <-chan struct{}, logger logger) {
|
|
retryTicker := time.NewTicker(retryInterval)
|
|
defer retryTicker.Stop()
|
|
|
|
send := func(key Key) bool {
|
|
for {
|
|
err := target.SendFromStore(key)
|
|
if err == nil {
|
|
break
|
|
}
|
|
|
|
logger(
|
|
context.Background(),
|
|
fmt.Errorf("unable to send webhook log entry to '%s' err '%w'", target.Name(), err),
|
|
target.Name(),
|
|
)
|
|
|
|
select {
|
|
// Retrying after 3secs back-off
|
|
case <-retryTicker.C:
|
|
case <-doneCh:
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
for {
|
|
select {
|
|
case key, ok := <-keyCh:
|
|
if !ok {
|
|
return
|
|
}
|
|
|
|
if !send(key) {
|
|
return
|
|
}
|
|
case <-doneCh:
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
// StreamItems reads the keys from the store and replays the corresponding item to the target.
|
|
func StreamItems[I any](store Store[I], target Target, doneCh <-chan struct{}, logger logger) {
|
|
go func() {
|
|
keyCh := replayItems(store, doneCh, logger, target.Name())
|
|
sendItems(target, keyCh, doneCh, logger)
|
|
}()
|
|
}
|