mirror of
https://github.com/minio/minio.git
synced 2024-12-31 17:43:21 -05:00
91805bcab6
allow non-inlined on disk to be inlined via an unversioned ReadVersion() call, we only need ReadXL() to resolve objects with multiple versions only. The choice of this block makes it to be dynamic and chosen by the user via `mc admin config set` Other bonus things - Start measuring internode TTFB performance. - Set TCP_NODELAY, TCP_CORK for low latency
91 lines
2.8 KiB
Go
91 lines
2.8 KiB
Go
// Copyright (c) 2015-2022 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 rest
|
|
|
|
import (
|
|
"net/http"
|
|
"net/http/httptrace"
|
|
"sync/atomic"
|
|
"time"
|
|
|
|
"github.com/minio/minio/internal/logger"
|
|
)
|
|
|
|
var globalStats = struct {
|
|
errs uint64
|
|
|
|
tcpDialErrs uint64
|
|
tcpDialCount uint64
|
|
tcpDialTotalDur uint64
|
|
tcpTimeForFirstByteTotalDur uint64
|
|
}{}
|
|
|
|
// RPCStats holds information about the DHCP/TCP metrics and errors
|
|
type RPCStats struct {
|
|
Errs uint64
|
|
|
|
DialAvgDuration uint64
|
|
TTFBAvgDuration uint64
|
|
DialErrs uint64
|
|
}
|
|
|
|
// GetRPCStats returns RPC stats, include calls errors and dhcp/tcp metrics
|
|
func GetRPCStats() RPCStats {
|
|
s := RPCStats{
|
|
Errs: atomic.LoadUint64(&globalStats.errs),
|
|
DialErrs: atomic.LoadUint64(&globalStats.tcpDialErrs),
|
|
}
|
|
if v := atomic.LoadUint64(&globalStats.tcpDialCount); v > 0 {
|
|
s.DialAvgDuration = atomic.LoadUint64(&globalStats.tcpDialTotalDur) / v
|
|
s.TTFBAvgDuration = atomic.LoadUint64(&globalStats.tcpTimeForFirstByteTotalDur) / v
|
|
}
|
|
return s
|
|
}
|
|
|
|
// Return a function which update the global stats related to tcp connections
|
|
func setupReqStatsUpdate(req *http.Request) (*http.Request, func()) {
|
|
var dialStart, dialEnd int64
|
|
start := time.Now()
|
|
trace := &httptrace.ClientTrace{
|
|
GotFirstResponseByte: func() {
|
|
atomic.AddUint64(&globalStats.tcpTimeForFirstByteTotalDur, uint64(time.Since(start)))
|
|
},
|
|
ConnectStart: func(network, addr string) {
|
|
atomic.StoreInt64(&dialStart, time.Now().UnixNano())
|
|
},
|
|
ConnectDone: func(network, addr string, err error) {
|
|
if err == nil {
|
|
atomic.StoreInt64(&dialEnd, time.Now().UnixNano())
|
|
} else {
|
|
logger.LogOnceIf(req.Context(), logSubsys, err, req.URL.Hostname())
|
|
}
|
|
},
|
|
}
|
|
|
|
return req.WithContext(httptrace.WithClientTrace(req.Context(), trace)), func() {
|
|
if ds := atomic.LoadInt64(&dialStart); ds > 0 {
|
|
if de := atomic.LoadInt64(&dialEnd); de == 0 {
|
|
atomic.AddUint64(&globalStats.tcpDialErrs, 1)
|
|
} else if de >= ds {
|
|
atomic.AddUint64(&globalStats.tcpDialCount, 1)
|
|
atomic.AddUint64(&globalStats.tcpDialTotalDur, uint64(dialEnd-dialStart))
|
|
}
|
|
}
|
|
}
|
|
}
|