-
Notifications
You must be signed in to change notification settings - Fork 39.6k
/
csi_metrics.go
135 lines (115 loc) · 3.71 KB
/
csi_metrics.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
/*
Copyright 2019 The Kubernetes Authors.
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 csi
import (
"context"
"fmt"
"time"
"google.golang.org/grpc"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
servermetrics "k8s.io/kubernetes/pkg/kubelet/server/metrics"
"k8s.io/kubernetes/pkg/volume"
volumeutil "k8s.io/kubernetes/pkg/volume/util"
)
var _ volume.MetricsProvider = &metricsCsi{}
// metricsCsi represents a MetricsProvider that calculates the used,free and
// capacity information for volume using volume path.
type metricsCsi struct {
// the directory path the volume is mounted to.
targetPath string
// Volume handle or id
volumeID string
//csiClient with cache
csiClientGetter
}
// NewMetricsCsi creates a new metricsCsi with the Volume ID and path.
func NewMetricsCsi(volumeID string, targetPath string, driverName csiDriverName) volume.MetricsProvider {
mc := &metricsCsi{volumeID: volumeID, targetPath: targetPath}
mc.csiClientGetter.driverName = driverName
return mc
}
func (mc *metricsCsi) GetMetrics() (*volume.Metrics, error) {
startTime := time.Now()
defer servermetrics.CollectVolumeStatCalDuration(string(mc.csiClientGetter.driverName), startTime)
currentTime := metav1.Now()
ctx, cancel := context.WithTimeout(context.Background(), csiTimeout)
defer cancel()
// Get CSI client
csiClient, err := mc.csiClientGetter.Get()
if err != nil {
return nil, err
}
// Check whether "GET_VOLUME_STATS" is set
volumeStatsSet, err := csiClient.NodeSupportsVolumeStats(ctx)
if err != nil {
return nil, err
}
// if plugin doesnot support volume status, return.
if !volumeStatsSet {
return nil, volume.NewNotSupportedErrorWithDriverName(
string(mc.csiClientGetter.driverName))
}
// Get Volumestatus
metrics, err := csiClient.NodeGetVolumeStats(ctx, mc.volumeID, mc.targetPath)
if err != nil {
return nil, err
}
if metrics == nil {
return nil, fmt.Errorf("csi.NodeGetVolumeStats returned nil metrics for volume %s", mc.volumeID)
}
//set recorded time
metrics.Time = currentTime
return metrics, nil
}
// MetricsManager defines the metrics manager for CSI operation
type MetricsManager struct {
driverName string
}
// NewCSIMetricsManager creates a CSIMetricsManager object
func NewCSIMetricsManager(driverName string) *MetricsManager {
cmm := MetricsManager{
driverName: driverName,
}
return &cmm
}
type additionalInfo struct {
Migrated string
}
type additionalInfoKeyType struct{}
var additionalInfoKey additionalInfoKeyType
// RecordMetricsInterceptor is a grpc interceptor that is used to
// record CSI operation
func (cmm *MetricsManager) RecordMetricsInterceptor(
ctx context.Context,
method string,
req, reply interface{},
cc *grpc.ClientConn,
invoker grpc.UnaryInvoker,
opts ...grpc.CallOption) error {
start := time.Now()
err := invoker(ctx, method, req, reply, cc, opts...)
duration := time.Since(start)
// Check if this is migrated operation
additionalInfoVal := ctx.Value(additionalInfoKey)
migrated := "false"
if additionalInfoVal != nil {
additionalInfoVal, ok := additionalInfoVal.(additionalInfo)
if !ok {
return err
}
migrated = additionalInfoVal.Migrated
}
// Record the metric latency
volumeutil.RecordCSIOperationLatencyMetrics(cmm.driverName, method, err, duration, migrated)
return err
}