kubernetes/test/instrumentation/decode_metric.go
Han Kang 49c08947f7 add support for parsing gauge func
Change-Id: Id0b9cd51dead5ee9f4adac804d62f5d9742320a7
2022-10-12 08:30:41 -07:00

609 lines
15 KiB
Go

/*
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 main
import (
"errors"
"fmt"
"go/ast"
"go/token"
"sort"
"strconv"
"strings"
"k8s.io/component-base/metrics"
)
func decodeMetricCalls(fs []*ast.CallExpr, metricsImportName string, variables map[string]ast.Expr) ([]metric, []error) {
finder := metricDecoder{
kubeMetricsImportName: metricsImportName,
variables: variables,
}
ms := make([]metric, 0, len(fs))
errors := []error{}
for _, f := range fs {
m, err := finder.decodeNewMetricCall(f)
if err != nil {
errors = append(errors, err)
continue
}
if m != nil {
ms = append(ms, *m)
}
}
return ms, errors
}
type metricDecoder struct {
kubeMetricsImportName string
variables map[string]ast.Expr
}
func (c *metricDecoder) decodeNewMetricCall(fc *ast.CallExpr) (*metric, error) {
var m metric
var err error
se, ok := fc.Fun.(*ast.SelectorExpr)
if !ok {
return nil, newDecodeErrorf(fc, errNotDirectCall)
}
functionName := se.Sel.String()
functionImport, ok := se.X.(*ast.Ident)
if !ok {
return nil, newDecodeErrorf(fc, errNotDirectCall)
}
if functionImport.String() != c.kubeMetricsImportName {
return nil, nil
}
switch functionName {
case "NewCounter", "NewGauge", "NewHistogram", "NewSummary", "NewTimingHistogram", "NewGaugeFunc":
m, err = c.decodeMetric(fc)
case "NewCounterVec", "NewGaugeVec", "NewHistogramVec", "NewSummaryVec", "NewTimingHistogramVec":
m, err = c.decodeMetricVec(fc)
case "Labels", "HandlerOpts", "HandlerFor", "HandlerWithReset":
return nil, nil
default:
return &m, newDecodeErrorf(fc, errNotDirectCall)
}
if err != nil {
return &m, err
}
m.Type = getMetricType(functionName)
return &m, nil
}
func getMetricType(functionName string) string {
switch functionName {
case "NewCounter", "NewCounterVec":
return counterMetricType
case "NewGauge", "NewGaugeVec", "NewGaugeFunc":
return gaugeMetricType
case "NewHistogram", "NewHistogramVec":
return histogramMetricType
case "NewSummary", "NewSummaryVec":
return summaryMetricType
case "NewTimingHistogram", "NewTimingHistogramVec":
return timingRatioHistogram
default:
panic("getMetricType expects correct function name")
}
}
func (c *metricDecoder) decodeMetric(call *ast.CallExpr) (metric, error) {
if len(call.Args) > 2 {
return metric{}, newDecodeErrorf(call, errInvalidNewMetricCall)
}
return c.decodeOpts(call.Args[0])
}
func (c *metricDecoder) decodeMetricVec(call *ast.CallExpr) (metric, error) {
if len(call.Args) != 2 {
return metric{}, newDecodeErrorf(call, errInvalidNewMetricCall)
}
m, err := c.decodeOpts(call.Args[0])
if err != nil {
return m, err
}
labels, err := c.decodeLabels(call.Args[1])
if err != nil {
return m, err
}
sort.Strings(labels)
m.Labels = labels
return m, nil
}
func (c *metricDecoder) decodeLabels(expr ast.Expr) ([]string, error) {
cl, ok := expr.(*ast.CompositeLit)
if !ok {
id, ok := expr.(*ast.Ident)
if !ok {
return nil, newDecodeErrorf(expr, errInvalidNewMetricCall)
}
variableExpr, found := c.variables[id.Name]
if !found {
return nil, newDecodeErrorf(expr, "couldn't find variable for labels")
}
cl2, ok := variableExpr.(*ast.CompositeLit)
if !ok {
return nil, newDecodeErrorf(expr, "couldn't interpret variable for labels")
}
cl = cl2
}
labels := make([]string, len(cl.Elts))
for i, el := range cl.Elts {
v, ok := el.(*ast.Ident)
if ok {
variableExpr, found := c.variables[v.Name]
if !found {
return nil, newDecodeErrorf(expr, errBadVariableAttribute)
}
bl, ok := variableExpr.(*ast.BasicLit)
if !ok {
return nil, newDecodeErrorf(expr, errNonStringAttribute)
}
value, err := stringValue(bl)
if err != nil {
return nil, err
}
labels[i] = value
continue
}
bl, ok := el.(*ast.BasicLit)
if !ok {
return nil, errors.New(errLabels)
}
value, err := stringValue(bl)
if err != nil {
return nil, err
}
labels[i] = value
}
return labels, nil
}
func (c *metricDecoder) decodeOpts(expr ast.Expr) (metric, error) {
m := metric{
Labels: []string{},
}
ue, ok := expr.(*ast.UnaryExpr)
if !ok {
return m, newDecodeErrorf(expr, errInvalidNewMetricCall)
}
cl, ok := ue.X.(*ast.CompositeLit)
if !ok {
return m, newDecodeErrorf(expr, errInvalidNewMetricCall)
}
for _, expr := range cl.Elts {
kv, ok := expr.(*ast.KeyValueExpr)
if !ok {
return m, newDecodeErrorf(expr, errPositionalArguments)
}
key := fmt.Sprintf("%v", kv.Key)
switch key {
case "Namespace", "Subsystem", "Name", "Help", "DeprecatedVersion":
var value string
var err error
switch v := kv.Value.(type) {
case *ast.BasicLit:
value, err = stringValue(v)
if err != nil {
return m, err
}
case *ast.Ident:
variableExpr, found := c.variables[v.Name]
if !found {
return m, newDecodeErrorf(expr, errBadVariableAttribute)
}
bl, ok := variableExpr.(*ast.BasicLit)
if !ok {
return m, newDecodeErrorf(expr, errNonStringAttribute)
}
value, err = stringValue(bl)
if err != nil {
return m, err
}
case *ast.SelectorExpr:
s, ok := v.X.(*ast.Ident)
if !ok {
return m, newDecodeErrorf(expr, errExprNotIdent, v.X)
}
variableExpr, found := c.variables[strings.Join([]string{s.Name, v.Sel.Name}, ".")]
if !found {
return m, newDecodeErrorf(expr, errBadImportedVariableAttribute)
}
bl, ok := variableExpr.(*ast.BasicLit)
if !ok {
return m, newDecodeErrorf(expr, errNonStringAttribute)
}
value, err = stringValue(bl)
if err != nil {
return m, err
}
case *ast.BinaryExpr:
var binaryExpr *ast.BinaryExpr
binaryExpr = v
var okay bool
okay = true
for okay {
yV, okay := binaryExpr.Y.(*ast.BasicLit)
if !okay {
return m, newDecodeErrorf(expr, errNonStringAttribute)
}
yVal, err := stringValue(yV)
if err != nil {
return m, newDecodeErrorf(expr, errNonStringAttribute)
}
value = fmt.Sprintf("%s%s", yVal, value)
x, okay := binaryExpr.X.(*ast.BinaryExpr)
if !okay {
// should be basicLit
xV, okay := binaryExpr.X.(*ast.BasicLit)
if !okay {
return m, newDecodeErrorf(expr, errNonStringAttribute)
}
xVal, err := stringValue(xV)
if err != nil {
return m, newDecodeErrorf(expr, errNonStringAttribute)
}
value = fmt.Sprintf("%s%s", xVal, value)
break
}
binaryExpr = x
}
default:
return m, newDecodeErrorf(expr, errNonStringAttribute)
}
switch key {
case "Namespace":
m.Namespace = value
case "Subsystem":
m.Subsystem = value
case "Name":
m.Name = value
case "DeprecatedVersion":
m.DeprecatedVersion = value
case "Help":
m.Help = value
}
case "Buckets":
buckets, err := c.decodeBuckets(kv.Value)
if err != nil {
return m, err
}
sort.Float64s(buckets)
m.Buckets = buckets
case "StabilityLevel":
level, err := decodeStabilityLevel(kv.Value, c.kubeMetricsImportName)
if err != nil {
return m, err
}
m.StabilityLevel = string(*level)
case "ConstLabels":
labels, err := c.decodeConstLabels(kv.Value)
if err != nil {
return m, err
}
m.ConstLabels = labels
case "AgeBuckets", "BufCap":
uintVal, err := c.decodeUint32(kv.Value)
if err != nil {
print(key)
return m, err
}
if key == "AgeBuckets" {
m.AgeBuckets = uintVal
}
if key == "BufCap" {
m.BufCap = uintVal
}
case "Objectives":
obj, err := c.decodeObjectives(kv.Value)
if err != nil {
print(key)
return m, err
}
m.Objectives = obj
case "MaxAge":
int64Val, err := c.decodeInt64(kv.Value)
if err != nil {
print(key)
return m, err
}
m.MaxAge = int64Val
default:
return m, newDecodeErrorf(expr, errFieldNotSupported, key)
}
}
return m, nil
}
func stringValue(bl *ast.BasicLit) (string, error) {
if bl.Kind != token.STRING {
return "", newDecodeErrorf(bl, errNonStringAttribute)
}
return strings.Trim(bl.Value, `"`), nil
}
func (c *metricDecoder) decodeBuckets(expr ast.Expr) ([]float64, error) {
switch v := expr.(type) {
case *ast.Ident:
variableExpr, found := c.variables[v.Name]
if !found {
return nil, fmt.Errorf("couldn't find variable for bucket")
}
v2, ok := variableExpr.(*ast.CompositeLit)
if !ok {
return nil, fmt.Errorf("couldn't find variable for bucket")
}
return decodeListOfFloats(v2, v2.Elts)
case *ast.CompositeLit:
return decodeListOfFloats(v, v.Elts)
case *ast.SelectorExpr:
variableName := v.Sel.String()
importName, ok := v.X.(*ast.Ident)
if ok && importName.String() == c.kubeMetricsImportName && variableName == "DefBuckets" {
return metrics.DefBuckets, nil
}
case *ast.CallExpr:
se, ok := v.Fun.(*ast.SelectorExpr)
if !ok {
return nil, newDecodeErrorf(v, errBuckets)
}
functionName := se.Sel.String()
functionImport, ok := se.X.(*ast.Ident)
if !ok {
return nil, newDecodeErrorf(v, errBuckets)
}
if functionImport.String() != c.kubeMetricsImportName {
return nil, newDecodeErrorf(v, errBuckets)
}
firstArg, secondArg, thirdArg, err := decodeBucketArguments(v)
if err != nil {
return nil, err
}
switch functionName {
case "LinearBuckets":
return metrics.LinearBuckets(firstArg, secondArg, thirdArg), nil
case "ExponentialBuckets":
return metrics.ExponentialBuckets(firstArg, secondArg, thirdArg), nil
}
}
return nil, newDecodeErrorf(expr, errBuckets)
}
func (c *metricDecoder) decodeObjectives(expr ast.Expr) (map[float64]float64, error) {
switch v := expr.(type) {
case *ast.CompositeLit:
return decodeFloatMap(v.Elts)
case *ast.Ident:
variableExpr, found := c.variables[v.Name]
if !found {
return nil, newDecodeErrorf(expr, errBadVariableAttribute)
}
return decodeFloatMap(variableExpr.(*ast.CompositeLit).Elts)
}
return nil, newDecodeErrorf(expr, errObjectives)
}
func (c *metricDecoder) decodeUint32(expr ast.Expr) (uint32, error) {
switch v := expr.(type) {
case *ast.BasicLit:
if v.Kind != token.FLOAT && v.Kind != token.INT {
print(v.Kind)
}
value, err := strconv.ParseUint(v.Value, 10, 32)
if err != nil {
return 0, err
}
return uint32(value), nil
case *ast.SelectorExpr:
variableName := v.Sel.String()
importName, ok := v.X.(*ast.Ident)
if ok && importName.String() == c.kubeMetricsImportName {
if variableName == "DefAgeBuckets" {
// hardcode this for now
return 5, nil
}
if variableName == "DefBufCap" {
// hardcode this for now
return 500, nil
}
}
case *ast.CallExpr:
_, ok := v.Fun.(*ast.SelectorExpr)
if !ok {
return 0, newDecodeErrorf(v, errDecodeUint32)
}
return 0, nil
}
return 0, newDecodeErrorf(expr, errDecodeUint32)
}
func (c *metricDecoder) decodeInt64(expr ast.Expr) (int64, error) {
switch v := expr.(type) {
case *ast.BasicLit:
if v.Kind != token.FLOAT && v.Kind != token.INT {
print(v.Kind)
}
value, err := strconv.ParseInt(v.Value, 10, 64)
if err != nil {
return 0, err
}
return value, nil
case *ast.SelectorExpr:
variableName := v.Sel.String()
importName, ok := v.X.(*ast.Ident)
if ok && importName.String() == c.kubeMetricsImportName {
if variableName == "DefMaxAge" {
// hardcode this for now. This is a duration but we'll output it as
// an int64 representing nanoseconds.
return 1000 * 1000 * 1000 * 60 * 10, nil
}
}
case *ast.CallExpr:
_, ok := v.Fun.(*ast.SelectorExpr)
if !ok {
return 0, newDecodeErrorf(v, errDecodeInt64)
}
return 0, nil
}
return 0, newDecodeErrorf(expr, errDecodeInt64)
}
func decodeFloatMap(exprs []ast.Expr) (map[float64]float64, error) {
buckets := map[float64]float64{}
for _, elt := range exprs {
bl, ok := elt.(*ast.KeyValueExpr)
if !ok {
return nil, newDecodeErrorf(bl, errObjectives)
}
keyExpr, ok := bl.Key.(*ast.BasicLit)
if !ok {
return nil, newDecodeErrorf(bl, errObjectives)
}
valueExpr, ok := bl.Value.(*ast.BasicLit)
if !ok {
return nil, newDecodeErrorf(bl, errObjectives)
}
valueForKey, err := strconv.ParseFloat(keyExpr.Value, 64)
if err != nil {
return nil, newDecodeErrorf(bl, errObjectives)
}
valueForValue, err := strconv.ParseFloat(valueExpr.Value, 64)
if err != nil {
return nil, newDecodeErrorf(bl, errObjectives)
}
buckets[valueForKey] = valueForValue
}
return buckets, nil
}
func decodeListOfFloats(expr ast.Expr, exprs []ast.Expr) ([]float64, error) {
buckets := make([]float64, len(exprs))
for i, elt := range exprs {
bl, ok := elt.(*ast.BasicLit)
if !ok {
return nil, newDecodeErrorf(expr, errBuckets)
}
if bl.Kind != token.FLOAT && bl.Kind != token.INT {
return nil, newDecodeErrorf(bl, errBuckets)
}
value, err := strconv.ParseFloat(bl.Value, 64)
if err != nil {
return nil, err
}
buckets[i] = value
}
return buckets, nil
}
func decodeBucketArguments(fc *ast.CallExpr) (float64, float64, int, error) {
if len(fc.Args) != 3 {
return 0, 0, 0, newDecodeErrorf(fc, errBuckets)
}
strArgs := make([]string, len(fc.Args))
for i, elt := range fc.Args {
bl, ok := elt.(*ast.BasicLit)
if !ok {
return 0, 0, 0, newDecodeErrorf(bl, errBuckets)
}
if bl.Kind != token.FLOAT && bl.Kind != token.INT {
return 0, 0, 0, newDecodeErrorf(bl, errBuckets)
}
strArgs[i] = bl.Value
}
firstArg, err := strconv.ParseFloat(strArgs[0], 64)
if err != nil {
return 0, 0, 0, newDecodeErrorf(fc.Args[0], errBuckets)
}
secondArg, err := strconv.ParseFloat(strArgs[1], 64)
if err != nil {
return 0, 0, 0, newDecodeErrorf(fc.Args[1], errBuckets)
}
thirdArg, err := strconv.ParseInt(strArgs[2], 10, 64)
if err != nil {
return 0, 0, 0, newDecodeErrorf(fc.Args[2], errBuckets)
}
return firstArg, secondArg, int(thirdArg), nil
}
func decodeStabilityLevel(expr ast.Expr, metricsFrameworkImportName string) (*metrics.StabilityLevel, error) {
se, ok := expr.(*ast.SelectorExpr)
if !ok {
return nil, newDecodeErrorf(expr, errStabilityLevel)
}
s, ok := se.X.(*ast.Ident)
if !ok {
return nil, newDecodeErrorf(expr, errStabilityLevel)
}
if s.String() != metricsFrameworkImportName {
return nil, newDecodeErrorf(expr, errStabilityLevel)
}
stability := metrics.StabilityLevel(se.Sel.Name)
return &stability, nil
}
func (c *metricDecoder) decodeConstLabels(expr ast.Expr) (map[string]string, error) {
retval := map[string]string{}
switch v := expr.(type) {
case *ast.CompositeLit:
for _, e2 := range v.Elts {
kv := e2.(*ast.KeyValueExpr)
key := ""
switch k := kv.Key.(type) {
case *ast.Ident:
variableExpr, found := c.variables[k.Name]
if !found {
return nil, newDecodeErrorf(expr, errBadVariableAttribute)
}
bl, ok := variableExpr.(*ast.BasicLit)
if !ok {
return nil, newDecodeErrorf(expr, errNonStringAttribute)
}
k2, err := stringValue(bl)
if err != nil {
return nil, err
}
key = k2
case *ast.BasicLit:
k2, err := stringValue(k)
if err != nil {
return nil, err
}
key = k2
}
val, err := stringValue(kv.Value.(*ast.BasicLit))
if err != nil {
return nil, err
}
retval[key] = val
}
}
return retval, nil
}