mirror of
https://github.com/minio/minio.git
synced 2025-04-04 11:50:36 -04:00
fix: peer replication bandwidth monitoring in distributed setup (#10652)
This commit is contained in:
parent
c2f16ee846
commit
8ceb2a93fd
@ -41,13 +41,10 @@ import (
|
|||||||
"github.com/minio/minio/cmd/logger"
|
"github.com/minio/minio/cmd/logger"
|
||||||
"github.com/minio/minio/cmd/logger/message/log"
|
"github.com/minio/minio/cmd/logger/message/log"
|
||||||
"github.com/minio/minio/pkg/auth"
|
"github.com/minio/minio/pkg/auth"
|
||||||
bandwidth "github.com/minio/minio/pkg/bandwidth"
|
|
||||||
bucketBandwidth "github.com/minio/minio/pkg/bucket/bandwidth"
|
|
||||||
"github.com/minio/minio/pkg/handlers"
|
"github.com/minio/minio/pkg/handlers"
|
||||||
iampolicy "github.com/minio/minio/pkg/iam/policy"
|
iampolicy "github.com/minio/minio/pkg/iam/policy"
|
||||||
"github.com/minio/minio/pkg/madmin"
|
"github.com/minio/minio/pkg/madmin"
|
||||||
xnet "github.com/minio/minio/pkg/net"
|
xnet "github.com/minio/minio/pkg/net"
|
||||||
"github.com/minio/minio/pkg/sync/errgroup"
|
|
||||||
trace "github.com/minio/minio/pkg/trace"
|
trace "github.com/minio/minio/pkg/trace"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -1441,44 +1438,9 @@ func (a adminAPIHandlers) BandwidthMonitorHandler(w http.ResponseWriter, r *http
|
|||||||
}
|
}
|
||||||
|
|
||||||
setEventStreamHeaders(w)
|
setEventStreamHeaders(w)
|
||||||
peers := newPeerRestClients(globalEndpoints)
|
|
||||||
bucketsRequestedString := r.URL.Query().Get("buckets")
|
bucketsRequestedString := r.URL.Query().Get("buckets")
|
||||||
var bucketsRequested []string
|
bucketsRequested := strings.Split(bucketsRequestedString, ",")
|
||||||
reports := make([]*bandwidth.Report, len(peers))
|
consolidatedReport := globalNotificationSys.GetBandwidthReports(ctx, bucketsRequested...)
|
||||||
selectBuckets := bucketBandwidth.SelectAllBuckets()
|
|
||||||
if bucketsRequestedString != "" {
|
|
||||||
bucketsRequested = strings.Split(bucketsRequestedString, ",")
|
|
||||||
selectBuckets = bucketBandwidth.SelectBuckets(bucketsRequested...)
|
|
||||||
}
|
|
||||||
reports = append(reports, globalBucketMonitor.GetReport(selectBuckets))
|
|
||||||
g := errgroup.WithNErrs(len(peers))
|
|
||||||
for index, peer := range peers {
|
|
||||||
if peer == nil {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
index := index
|
|
||||||
g.Go(func() error {
|
|
||||||
var err error
|
|
||||||
reports[index], err = peer.MonitorBandwidth(ctx, bucketsRequested)
|
|
||||||
return err
|
|
||||||
}, index)
|
|
||||||
}
|
|
||||||
consolidatedReport := bandwidth.Report{
|
|
||||||
BucketStats: make(map[string]bandwidth.Details),
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, report := range reports {
|
|
||||||
for bucket := range report.BucketStats {
|
|
||||||
d, ok := consolidatedReport.BucketStats[bucket]
|
|
||||||
if !ok {
|
|
||||||
consolidatedReport.BucketStats[bucket] = bandwidth.Details{}
|
|
||||||
d = consolidatedReport.BucketStats[bucket]
|
|
||||||
d.LimitInBytesPerSecond = report.BucketStats[bucket].LimitInBytesPerSecond
|
|
||||||
}
|
|
||||||
d.CurrentBandwidthInBytesPerSecond += report.BucketStats[bucket].CurrentBandwidthInBytesPerSecond
|
|
||||||
consolidatedReport.BucketStats[bucket] = d
|
|
||||||
}
|
|
||||||
}
|
|
||||||
enc := json.NewEncoder(w)
|
enc := json.NewEncoder(w)
|
||||||
err := enc.Encode(consolidatedReport)
|
err := enc.Encode(consolidatedReport)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -235,8 +235,8 @@ func replicateObject(ctx context.Context, objInfo ObjectInfo, objectAPI ObjectLa
|
|||||||
replicationStatus := replication.Complete
|
replicationStatus := replication.Complete
|
||||||
|
|
||||||
// Setup bandwidth throttling
|
// Setup bandwidth throttling
|
||||||
peerCount := len(globalEndpoints)
|
totalNodesCount := len(GetRemotePeers(globalEndpoints)) + 1
|
||||||
b := target.BandwidthLimit / int64(peerCount)
|
b := target.BandwidthLimit / int64(totalNodesCount)
|
||||||
var headerSize int
|
var headerSize int
|
||||||
for k, v := range putOpts.Header() {
|
for k, v := range putOpts.Header() {
|
||||||
headerSize += len(k) + len(v)
|
headerSize += len(k) + len(v)
|
||||||
|
@ -33,6 +33,8 @@ import (
|
|||||||
"github.com/minio/minio-go/v7/pkg/set"
|
"github.com/minio/minio-go/v7/pkg/set"
|
||||||
"github.com/minio/minio/cmd/crypto"
|
"github.com/minio/minio/cmd/crypto"
|
||||||
"github.com/minio/minio/cmd/logger"
|
"github.com/minio/minio/cmd/logger"
|
||||||
|
bandwidth "github.com/minio/minio/pkg/bandwidth"
|
||||||
|
bucketBandwidth "github.com/minio/minio/pkg/bucket/bandwidth"
|
||||||
"github.com/minio/minio/pkg/bucket/policy"
|
"github.com/minio/minio/pkg/bucket/policy"
|
||||||
"github.com/minio/minio/pkg/event"
|
"github.com/minio/minio/pkg/event"
|
||||||
"github.com/minio/minio/pkg/madmin"
|
"github.com/minio/minio/pkg/madmin"
|
||||||
@ -1283,3 +1285,47 @@ func sendEvent(args eventArgs) {
|
|||||||
|
|
||||||
globalNotificationSys.Send(args)
|
globalNotificationSys.Send(args)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// GetBandwidthReports - gets the bandwidth report from all nodes including self.
|
||||||
|
func (sys *NotificationSys) GetBandwidthReports(ctx context.Context, buckets ...string) bandwidth.Report {
|
||||||
|
reports := make([]*bandwidth.Report, len(sys.peerClients))
|
||||||
|
g := errgroup.WithNErrs(len(sys.peerClients))
|
||||||
|
for index, peer := range sys.peerClients {
|
||||||
|
if peer == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
index := index
|
||||||
|
g.Go(func() error {
|
||||||
|
var err error
|
||||||
|
reports[index], err = peer.MonitorBandwidth(ctx, buckets)
|
||||||
|
return err
|
||||||
|
}, index)
|
||||||
|
}
|
||||||
|
|
||||||
|
for index, err := range g.Wait() {
|
||||||
|
reqInfo := (&logger.ReqInfo{}).AppendTags("peerAddress",
|
||||||
|
sys.peerClients[index].host.String())
|
||||||
|
ctx := logger.SetReqInfo(ctx, reqInfo)
|
||||||
|
logger.LogOnceIf(ctx, err, sys.peerClients[index].host.String())
|
||||||
|
}
|
||||||
|
reports = append(reports, globalBucketMonitor.GetReport(bucketBandwidth.SelectBuckets(buckets...)))
|
||||||
|
consolidatedReport := bandwidth.Report{
|
||||||
|
BucketStats: make(map[string]bandwidth.Details),
|
||||||
|
}
|
||||||
|
for _, report := range reports {
|
||||||
|
if report == nil || report.BucketStats == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
for bucket := range report.BucketStats {
|
||||||
|
d, ok := consolidatedReport.BucketStats[bucket]
|
||||||
|
if !ok {
|
||||||
|
consolidatedReport.BucketStats[bucket] = bandwidth.Details{}
|
||||||
|
d = consolidatedReport.BucketStats[bucket]
|
||||||
|
d.LimitInBytesPerSecond = report.BucketStats[bucket].LimitInBytesPerSecond
|
||||||
|
}
|
||||||
|
d.CurrentBandwidthInBytesPerSecond += report.BucketStats[bucket].CurrentBandwidthInBytesPerSecond
|
||||||
|
consolidatedReport.BucketStats[bucket] = d
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return consolidatedReport
|
||||||
|
}
|
||||||
|
@ -891,7 +891,6 @@ func newPeerRESTClient(peer *xnet.Host) *peerRESTClient {
|
|||||||
func (client *peerRESTClient) MonitorBandwidth(ctx context.Context, buckets []string) (*bandwidth.Report, error) {
|
func (client *peerRESTClient) MonitorBandwidth(ctx context.Context, buckets []string) (*bandwidth.Report, error) {
|
||||||
values := make(url.Values)
|
values := make(url.Values)
|
||||||
values.Set(peerRESTBuckets, strings.Join(buckets, ","))
|
values.Set(peerRESTBuckets, strings.Join(buckets, ","))
|
||||||
|
|
||||||
respBody, err := client.callWithContext(ctx, peerRESTMethodGetBandwidth, values, nil, -1)
|
respBody, err := client.callWithContext(ctx, peerRESTMethodGetBandwidth, values, nil, -1)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
@ -30,7 +30,6 @@ import (
|
|||||||
|
|
||||||
"github.com/gorilla/mux"
|
"github.com/gorilla/mux"
|
||||||
"github.com/minio/minio/cmd/logger"
|
"github.com/minio/minio/cmd/logger"
|
||||||
"github.com/minio/minio/pkg/bandwidth"
|
|
||||||
b "github.com/minio/minio/pkg/bucket/bandwidth"
|
b "github.com/minio/minio/pkg/bucket/bandwidth"
|
||||||
"github.com/minio/minio/pkg/event"
|
"github.com/minio/minio/pkg/event"
|
||||||
"github.com/minio/minio/pkg/madmin"
|
"github.com/minio/minio/pkg/madmin"
|
||||||
@ -1062,12 +1061,8 @@ func (s *peerRESTServer) GetBandwidth(w http.ResponseWriter, r *http.Request) {
|
|||||||
doneCh := make(chan struct{})
|
doneCh := make(chan struct{})
|
||||||
defer close(doneCh)
|
defer close(doneCh)
|
||||||
|
|
||||||
var report *bandwidth.Report
|
selectBuckets := b.SelectBuckets(strings.Split(bucketsString, ",")...)
|
||||||
selectBuckets := b.SelectAllBuckets()
|
report := globalBucketMonitor.GetReport(selectBuckets)
|
||||||
if bucketsString != "" {
|
|
||||||
selectBuckets = b.SelectBuckets(strings.Split(bucketsString, ",")...)
|
|
||||||
}
|
|
||||||
report = globalBucketMonitor.GetReport(selectBuckets)
|
|
||||||
|
|
||||||
enc := gob.NewEncoder(w)
|
enc := gob.NewEncoder(w)
|
||||||
if err := enc.Encode(report); err != nil {
|
if err := enc.Encode(report); err != nil {
|
||||||
|
@ -91,18 +91,16 @@ func NewMonitor(doneCh <-chan struct{}) *Monitor {
|
|||||||
// SelectionFunction for buckets
|
// SelectionFunction for buckets
|
||||||
type SelectionFunction func(bucket string) bool
|
type SelectionFunction func(bucket string) bool
|
||||||
|
|
||||||
// SelectAllBuckets will select all buckets
|
|
||||||
func SelectAllBuckets() SelectionFunction {
|
|
||||||
return func(bucket string) bool {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// SelectBuckets will select all the buckets passed in.
|
// SelectBuckets will select all the buckets passed in.
|
||||||
func SelectBuckets(buckets ...string) SelectionFunction {
|
func SelectBuckets(buckets ...string) SelectionFunction {
|
||||||
|
if len(buckets) == 0 {
|
||||||
|
return func(bucket string) bool {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
return func(bucket string) bool {
|
return func(bucket string) bool {
|
||||||
for _, b := range buckets {
|
for _, b := range buckets {
|
||||||
if b != "" && b == bucket {
|
if b == "" || b == bucket {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -160,7 +158,7 @@ func (m *Monitor) processAvg() {
|
|||||||
for _, bucketMeasurement := range m.activeBuckets {
|
for _, bucketMeasurement := range m.activeBuckets {
|
||||||
bucketMeasurement.updateExponentialMovingAverage(time.Now())
|
bucketMeasurement.updateExponentialMovingAverage(time.Now())
|
||||||
}
|
}
|
||||||
m.pubsub.Publish(m.getReport(SelectAllBuckets()))
|
m.pubsub.Publish(m.getReport(SelectBuckets()))
|
||||||
}
|
}
|
||||||
|
|
||||||
// track returns the measurement object for bucket and object
|
// track returns the measurement object for bucket and object
|
||||||
|
@ -142,13 +142,13 @@ func TestMonitor_GetReport(t *testing.T) {
|
|||||||
bucketThrottle: map[string]*throttle{"bucket": &thr},
|
bucketThrottle: map[string]*throttle{"bucket": &thr},
|
||||||
}
|
}
|
||||||
m.activeBuckets["bucket"].updateExponentialMovingAverage(tt.fields.endTime)
|
m.activeBuckets["bucket"].updateExponentialMovingAverage(tt.fields.endTime)
|
||||||
got := m.GetReport(SelectAllBuckets())
|
got := m.GetReport(SelectBuckets())
|
||||||
if !reflect.DeepEqual(got, tt.want) {
|
if !reflect.DeepEqual(got, tt.want) {
|
||||||
t.Errorf("GetReport() = %v, want %v", got, tt.want)
|
t.Errorf("GetReport() = %v, want %v", got, tt.want)
|
||||||
}
|
}
|
||||||
m.activeBuckets["bucket"].incrementBytes(tt.fields.update2)
|
m.activeBuckets["bucket"].incrementBytes(tt.fields.update2)
|
||||||
m.activeBuckets["bucket"].updateExponentialMovingAverage(tt.fields.endTime2)
|
m.activeBuckets["bucket"].updateExponentialMovingAverage(tt.fields.endTime2)
|
||||||
got = m.GetReport(SelectAllBuckets())
|
got = m.GetReport(SelectBuckets())
|
||||||
if !reflect.DeepEqual(got, tt.want2) {
|
if !reflect.DeepEqual(got, tt.want2) {
|
||||||
t.Errorf("GetReport() = %v, want %v", got, tt.want2)
|
t.Errorf("GetReport() = %v, want %v", got, tt.want2)
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user