// 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 . package store import ( "context" "errors" "fmt" "strings" "time" xnet "github.com/minio/pkg/net" ) 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 string) error } // Store - Used to persist items. type Store[I any] interface { Put(item I) error Get(key string) (I, error) Len() int List() ([]string, error) Del(key string) error Open() error Extension() string } // replayItems - Reads the items from the store and replays. func replayItems[I any](store Store[I], doneCh <-chan struct{}, logger logger, id string) <-chan string { itemKeyCh := make(chan string) go func() { defer close(itemKeyCh) retryTicker := time.NewTicker(retryInterval) defer retryTicker.Stop() for { names, err := store.List() if err != nil { logger(context.Background(), fmt.Errorf("store.List() failed with: %w", err), id) } else { for _, name := range names { select { case itemKeyCh <- strings.TrimSuffix(name, store.Extension()): // Get next key. case <-doneCh: return } } } select { case <-retryTicker.C: case <-doneCh: return } } }() return itemKeyCh } // sendItems - Reads items from the store and re-plays. func sendItems(target Target, itemKeyCh <-chan string, doneCh <-chan struct{}, logger logger) { retryTicker := time.NewTicker(retryInterval) defer retryTicker.Stop() send := func(itemKey string) bool { for { err := target.SendFromStore(itemKey) if err == nil { break } if err != ErrNotConnected && !xnet.IsConnResetErr(err) { logger(context.Background(), fmt.Errorf("target.SendFromStore() failed with '%w'", err), target.Name()) } // Retrying after 3secs back-off select { case <-retryTicker.C: case <-doneCh: return false } } return true } for { select { case itemKey, ok := <-itemKeyCh: if !ok { // closed channel. return } if !send(itemKey) { 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() { // Replays the items from the store. itemKeyCh := replayItems(store, doneCh, logger, target.Name()) // Send items from the store. sendItems(target, itemKeyCh, doneCh, logger) }() }