mirror of https://github.com/k3s-io/k3s
160 lines
4.8 KiB
Go
160 lines
4.8 KiB
Go
/*
|
|
Copyright 2015 The Kubernetes Authors All rights reserved.
|
|
|
|
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 metrics
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"reflect"
|
|
"strings"
|
|
|
|
"k8s.io/kubernetes/pkg/util/sets"
|
|
|
|
"github.com/golang/glog"
|
|
"github.com/prometheus/common/expfmt"
|
|
"github.com/prometheus/common/model"
|
|
)
|
|
|
|
var CommonMetrics = map[string][]string{
|
|
"get_token_count": {},
|
|
"get_token_fail_count": {},
|
|
"go_gc_duration_seconds": {"quantile"},
|
|
"go_gc_duration_seconds_count": {},
|
|
"go_gc_duration_seconds_sum": {},
|
|
"go_goroutines": {},
|
|
"http_request_duration_microseconds": {"handler", "quantile"},
|
|
"http_request_duration_microseconds_count": {"handler"},
|
|
"http_request_duration_microseconds_sum": {"handler"},
|
|
"http_request_size_bytes": {"handler", "quantile"},
|
|
"http_request_size_bytes_count": {"handler"},
|
|
"http_request_size_bytes_sum": {"handler"},
|
|
"http_requests_total": {"handler", "method", "code"},
|
|
"http_response_size_bytes": {"handler", "quantile"},
|
|
"http_response_size_bytes_count": {"handler"},
|
|
"http_response_size_bytes_sum": {"handler"},
|
|
"kubernetes_build_info": {"major", "minor", "gitCommit", "gitTreeState", "gitVersion"},
|
|
"process_cpu_seconds_total": {},
|
|
"process_max_fds": {},
|
|
"process_open_fds": {},
|
|
"process_resident_memory_bytes": {},
|
|
"process_start_time_seconds": {},
|
|
"process_virtual_memory_bytes": {},
|
|
"ssh_tunnel_open_count": {},
|
|
"ssh_tunnel_open_fail_count": {},
|
|
}
|
|
|
|
type Metrics map[string]model.Samples
|
|
|
|
func (m *Metrics) Equal(o Metrics) bool {
|
|
leftKeySet := []string{}
|
|
rightKeySet := []string{}
|
|
for k := range *m {
|
|
leftKeySet = append(leftKeySet, k)
|
|
}
|
|
for k := range o {
|
|
rightKeySet = append(rightKeySet, k)
|
|
}
|
|
if !reflect.DeepEqual(leftKeySet, rightKeySet) {
|
|
return false
|
|
}
|
|
for _, k := range leftKeySet {
|
|
if !(*m)[k].Equal(o[k]) {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
func PrintSample(sample *model.Sample) string {
|
|
buf := make([]string, 0)
|
|
// Id is a VERY special label. For 'normal' container it's usless, but it's necessary
|
|
// for 'system' containers (e.g. /docker-daemon, /kubelet, etc.). We know if that's the
|
|
// case by checking if there's a label "kubernetes_container_name" present. It's hacky
|
|
// but it works...
|
|
_, normalContainer := sample.Metric["kubernetes_container_name"]
|
|
for k, v := range sample.Metric {
|
|
if strings.HasPrefix(string(k), "__") || KubeletMetricsLabelsToSkip.Has(string(k)) {
|
|
continue
|
|
}
|
|
|
|
if string(k) == "id" && normalContainer {
|
|
continue
|
|
}
|
|
buf = append(buf, fmt.Sprintf("%v=%v", string(k), v))
|
|
}
|
|
return fmt.Sprintf("[%v] = %v", strings.Join(buf, ","), sample.Value)
|
|
}
|
|
|
|
func NewMetrics() Metrics {
|
|
result := make(Metrics)
|
|
for metric := range CommonMetrics {
|
|
result[metric] = make(model.Samples, 0)
|
|
}
|
|
return result
|
|
}
|
|
|
|
func parseMetrics(data string, knownMetrics map[string][]string, output *Metrics, unknownMetrics sets.String) error {
|
|
dec, err := expfmt.NewDecoder(strings.NewReader(data), expfmt.FmtText)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
decoder := expfmt.SampleDecoder{
|
|
Dec: dec,
|
|
Opts: &expfmt.DecodeOptions{},
|
|
}
|
|
|
|
for {
|
|
var v model.Vector
|
|
if err = decoder.Decode(&v); err != nil {
|
|
if err == io.EOF {
|
|
// Expected loop termination condition.
|
|
return nil
|
|
}
|
|
glog.Warningf("Invalid Decode. Skipping.")
|
|
continue
|
|
}
|
|
for _, metric := range v {
|
|
name := string(metric.Metric[model.MetricNameLabel])
|
|
_, isCommonMetric := CommonMetrics[name]
|
|
_, isKnownMetric := knownMetrics[name]
|
|
if isKnownMetric || isCommonMetric {
|
|
(*output)[name] = append((*output)[name], metric)
|
|
} else {
|
|
glog.Warningf("Unknown metric %v", metric)
|
|
if unknownMetrics != nil {
|
|
unknownMetrics.Insert(name)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (g *MetricsGrabber) getMetricsFromPod(podName string, namespace string, port int) (string, error) {
|
|
rawOutput, err := g.client.Get().
|
|
Prefix("proxy").
|
|
Namespace(namespace).
|
|
Resource("pods").
|
|
Name(fmt.Sprintf("%v:%v", podName, port)).
|
|
Suffix("metrics").
|
|
Do().Raw()
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
return string(rawOutput), nil
|
|
}
|