mirror of
https://github.com/minio/minio.git
synced 2025-01-24 13:13:16 -05:00
b1ad99edbf
code of this form is always racy, when the map itself is being written to as well ``` func (r Map) retMap() map[string]string { .. lock .. return r.internalMap } func (r Map) addMap(k, v string) { .. lock .. r.internalMap[k] = v } ``` Anyone reading from `retMap()` is not protected because of locking and we need to make sure to avoid code in this manner. Always safe to copy the map and return.
194 lines
4.8 KiB
Go
194 lines
4.8 KiB
Go
/*
|
|
* MinIO Cloud Storage, (C) 2017 MinIO, Inc.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
package cmd
|
|
|
|
import (
|
|
"fmt"
|
|
"net/http"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
"go.uber.org/atomic"
|
|
)
|
|
|
|
// ConnStats - Network statistics
|
|
// Count total input/output transferred bytes during
|
|
// the server's life.
|
|
type ConnStats struct {
|
|
totalInputBytes atomic.Uint64
|
|
totalOutputBytes atomic.Uint64
|
|
s3InputBytes atomic.Uint64
|
|
s3OutputBytes atomic.Uint64
|
|
}
|
|
|
|
// Increase total input bytes
|
|
func (s *ConnStats) incInputBytes(n int) {
|
|
s.totalInputBytes.Add(uint64(n))
|
|
}
|
|
|
|
// Increase total output bytes
|
|
func (s *ConnStats) incOutputBytes(n int) {
|
|
s.totalOutputBytes.Add(uint64(n))
|
|
}
|
|
|
|
// Return total input bytes
|
|
func (s *ConnStats) getTotalInputBytes() uint64 {
|
|
return s.totalInputBytes.Load()
|
|
}
|
|
|
|
// Return total output bytes
|
|
func (s *ConnStats) getTotalOutputBytes() uint64 {
|
|
return s.totalOutputBytes.Load()
|
|
}
|
|
|
|
// Increase outbound input bytes
|
|
func (s *ConnStats) incS3InputBytes(n int) {
|
|
s.s3InputBytes.Add(uint64(n))
|
|
}
|
|
|
|
// Increase outbound output bytes
|
|
func (s *ConnStats) incS3OutputBytes(n int) {
|
|
s.s3OutputBytes.Add(uint64(n))
|
|
}
|
|
|
|
// Return outbound input bytes
|
|
func (s *ConnStats) getS3InputBytes() uint64 {
|
|
return s.s3InputBytes.Load()
|
|
}
|
|
|
|
// Return outbound output bytes
|
|
func (s *ConnStats) getS3OutputBytes() uint64 {
|
|
return s.s3OutputBytes.Load()
|
|
}
|
|
|
|
// Return connection stats (total input/output bytes and total s3 input/output bytes)
|
|
func (s *ConnStats) toServerConnStats() ServerConnStats {
|
|
return ServerConnStats{
|
|
TotalInputBytes: s.getTotalInputBytes(),
|
|
TotalOutputBytes: s.getTotalOutputBytes(),
|
|
S3InputBytes: s.getS3InputBytes(),
|
|
S3OutputBytes: s.getS3OutputBytes(),
|
|
}
|
|
}
|
|
|
|
// Prepare new ConnStats structure
|
|
func newConnStats() *ConnStats {
|
|
return &ConnStats{}
|
|
}
|
|
|
|
// HTTPAPIStats holds statistics information about
|
|
// a given API in the requests.
|
|
type HTTPAPIStats struct {
|
|
apiStats map[string]int
|
|
sync.RWMutex
|
|
}
|
|
|
|
// Inc increments the api stats counter.
|
|
func (stats *HTTPAPIStats) Inc(api string) {
|
|
stats.Lock()
|
|
defer stats.Unlock()
|
|
if stats == nil {
|
|
return
|
|
}
|
|
if stats.apiStats == nil {
|
|
stats.apiStats = make(map[string]int)
|
|
}
|
|
if _, ok := stats.apiStats[api]; ok {
|
|
stats.apiStats[api]++
|
|
return
|
|
}
|
|
stats.apiStats[api] = 1
|
|
}
|
|
|
|
// Dec increments the api stats counter.
|
|
func (stats *HTTPAPIStats) Dec(api string) {
|
|
stats.Lock()
|
|
defer stats.Unlock()
|
|
if stats == nil {
|
|
return
|
|
}
|
|
if val, ok := stats.apiStats[api]; ok && val > 0 {
|
|
stats.apiStats[api]--
|
|
}
|
|
}
|
|
|
|
// Load returns the recorded stats.
|
|
func (stats *HTTPAPIStats) Load() map[string]int {
|
|
stats.Lock()
|
|
defer stats.Unlock()
|
|
var apiStats = make(map[string]int, len(stats.apiStats))
|
|
for k, v := range stats.apiStats {
|
|
apiStats[k] = v
|
|
}
|
|
return apiStats
|
|
}
|
|
|
|
// HTTPStats holds statistics information about
|
|
// HTTP requests made by all clients
|
|
type HTTPStats struct {
|
|
currentS3Requests HTTPAPIStats
|
|
totalS3Requests HTTPAPIStats
|
|
totalS3Errors HTTPAPIStats
|
|
}
|
|
|
|
func durationStr(totalDuration, totalCount float64) string {
|
|
return fmt.Sprint(time.Duration(totalDuration/totalCount) * time.Second)
|
|
}
|
|
|
|
// Converts http stats into struct to be sent back to the client.
|
|
func (st *HTTPStats) toServerHTTPStats() ServerHTTPStats {
|
|
serverStats := ServerHTTPStats{}
|
|
|
|
serverStats.CurrentS3Requests = ServerHTTPAPIStats{
|
|
APIStats: st.currentS3Requests.Load(),
|
|
}
|
|
|
|
serverStats.TotalS3Requests = ServerHTTPAPIStats{
|
|
APIStats: st.totalS3Requests.Load(),
|
|
}
|
|
|
|
serverStats.TotalS3Errors = ServerHTTPAPIStats{
|
|
APIStats: st.totalS3Errors.Load(),
|
|
}
|
|
return serverStats
|
|
}
|
|
|
|
// Update statistics from http request and response data
|
|
func (st *HTTPStats) updateStats(api string, r *http.Request, w *recordAPIStats, durationSecs float64) {
|
|
// A successful request has a 2xx response code
|
|
successReq := (w.respStatusCode >= 200 && w.respStatusCode < 300)
|
|
|
|
if w.isS3Request && !strings.HasSuffix(r.URL.Path, prometheusMetricsPath) {
|
|
st.totalS3Requests.Inc(api)
|
|
if !successReq && w.respStatusCode != 0 {
|
|
st.totalS3Errors.Inc(api)
|
|
}
|
|
}
|
|
|
|
if w.isS3Request && r.Method == "GET" {
|
|
// Increment the prometheus http request response histogram with appropriate label
|
|
httpRequestsDuration.With(prometheus.Labels{"api": api}).Observe(durationSecs)
|
|
}
|
|
}
|
|
|
|
// Prepare new HTTPStats structure
|
|
func newHTTPStats() *HTTPStats {
|
|
return &HTTPStats{}
|
|
}
|