github.com/m3db/m3@v1.5.0/src/query/graphite/common/transform.go (about)

     1  // Copyright (c) 2019 Uber Technologies, Inc.
     2  //
     3  // Permission is hereby granted, free of charge, to any person obtaining a copy
     4  // of this software and associated documentation files (the "Software"), to deal
     5  // in the Software without restriction, including without limitation the rights
     6  // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
     7  // copies of the Software, and to permit persons to whom the Software is
     8  // furnished to do so, subject to the following conditions:
     9  //
    10  // The above copyright notice and this permission notice shall be included in
    11  // all copies or substantial portions of the Software.
    12  //
    13  // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
    14  // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
    15  // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
    16  // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
    17  // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
    18  // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
    19  // THE SOFTWARE.
    20  
    21  package common
    22  
    23  import (
    24  	"fmt"
    25  	"math"
    26  
    27  	"github.com/m3db/m3/src/query/graphite/ts"
    28  	"github.com/m3db/m3/src/x/errors"
    29  )
    30  
    31  // TransformFunc is used by Transform to apply a function
    32  // to all values in a series.
    33  type TransformFunc func(float64) float64
    34  
    35  // TransformFuncFactory creates transformation functions
    36  type TransformFuncFactory func() TransformFunc
    37  
    38  // Transformer transforms a value
    39  type Transformer interface {
    40  	// Apply applies the transformation
    41  	Apply(value float64) float64
    42  
    43  	// Reset resets the state
    44  	Reset()
    45  }
    46  
    47  type statelessTransformer struct {
    48  	fn TransformFunc
    49  }
    50  
    51  // NewStatelessTransformer creates a new stateless transformer
    52  func NewStatelessTransformer(fn TransformFunc) Transformer {
    53  	return statelessTransformer{fn: fn}
    54  }
    55  
    56  func (t statelessTransformer) Apply(value float64) float64 {
    57  	return t.fn(value)
    58  }
    59  
    60  func (t statelessTransformer) Reset() {}
    61  
    62  // MaintainNaNTransformer only applies a given ValueTransformer to
    63  // non-NaN values.
    64  func MaintainNaNTransformer(f TransformFunc) TransformFunc {
    65  	return func(v float64) float64 {
    66  		if math.IsNaN(v) {
    67  			return v
    68  		}
    69  		return f(v)
    70  	}
    71  }
    72  
    73  // Scale multiplies each element of a series list by a given value.
    74  func Scale(scale float64) TransformFunc {
    75  	return MaintainNaNTransformer(func(v float64) float64 {
    76  		return v * scale
    77  	})
    78  }
    79  
    80  // Offset adds a value to each element of a series list.
    81  func Offset(factor float64) TransformFunc {
    82  	return MaintainNaNTransformer(func(v float64) float64 {
    83  		return v + factor
    84  	})
    85  }
    86  
    87  // TransformNull transforms all nulls in a series to a value.
    88  func TransformNull(value float64) TransformFunc {
    89  	return func(v float64) float64 {
    90  		if math.IsNaN(v) {
    91  			return value
    92  		}
    93  
    94  		return v
    95  	}
    96  }
    97  
    98  // IsNonNull replaces datapoints that are non-null with 1, and null values with 0.
    99  // This is useful for understanding which series have data at a given point in time (i.e. to count
   100  // which servers are alive).
   101  func IsNonNull() TransformFunc {
   102  	return func(v float64) float64 {
   103  		if math.IsNaN(v) {
   104  			return 0
   105  		}
   106  
   107  		return 1
   108  	}
   109  }
   110  
   111  // PredicateFn is a predicate function.
   112  type PredicateFn func(v float64) bool
   113  
   114  // Filter removes data that does not satisfy a given predicate.
   115  func Filter(fn PredicateFn) TransformFunc {
   116  	return MaintainNaNTransformer(func(v float64) float64 {
   117  		if !fn(v) {
   118  			return math.NaN()
   119  		}
   120  
   121  		return v
   122  	})
   123  }
   124  
   125  // Logarithm takes one series or a series list, and draws the y-axis in logarithmic format. Only support
   126  // base 10 logarithms.
   127  func Logarithm() TransformFunc {
   128  	return func(v float64) float64 {
   129  		if !math.IsNaN(v) && v > 0 {
   130  			return math.Log10(v)
   131  		}
   132  
   133  		return math.NaN()
   134  	}
   135  }
   136  
   137  // Integral returns a function that accumulates values it has seen
   138  func Integral() TransformFunc {
   139  	currentSum := 0.0
   140  
   141  	return func(v float64) float64 {
   142  		if !math.IsNaN(v) {
   143  			currentSum += v
   144  		} else {
   145  			return v
   146  		}
   147  		return currentSum
   148  	}
   149  }
   150  
   151  // Derivative returns a function that computes the derivative among the values
   152  // it has seen
   153  func Derivative() TransformFunc {
   154  	previousValue := math.NaN()
   155  
   156  	return func(v float64) float64 {
   157  		var r float64
   158  		if math.IsNaN(v) || math.IsNaN(previousValue) {
   159  			previousValue, r = v, math.NaN()
   160  		} else {
   161  			previousValue, r = v, v-previousValue
   162  		}
   163  		return r
   164  	}
   165  }
   166  
   167  // NonNegativeDerivative returns a function that computes the derivative among the
   168  // values it has seen but ignores datapoints that trend down
   169  func NonNegativeDerivative(maxValue float64) TransformFunc {
   170  	previousValue := math.NaN()
   171  
   172  	return func(v float64) float64 {
   173  		var r float64
   174  
   175  		if math.IsNaN(v) || math.IsNaN(previousValue) {
   176  			previousValue, r = v, math.NaN()
   177  		} else if difference := v - previousValue; difference >= 0 {
   178  			previousValue, r = v, difference
   179  		} else if !math.IsNaN(maxValue) && maxValue >= v {
   180  			previousValue, r = v, (maxValue-previousValue)+v+1.0
   181  		} else {
   182  			previousValue, r = v, math.NaN()
   183  		}
   184  		return r
   185  	}
   186  }
   187  
   188  // Transform applies a specified ValueTransform to all values in each series, renaming
   189  // each series with the given SeriesRenamer.
   190  func Transform(ctx *Context, in ts.SeriesList, t Transformer, renamer SeriesRenamer) (ts.SeriesList, error) {
   191  	results := make([]*ts.Series, in.Len())
   192  
   193  	for i, series := range in.Values {
   194  		t.Reset()
   195  		values := ts.NewValues(ctx, series.MillisPerStep(), series.Len())
   196  		for step := 0; step < series.Len(); step++ {
   197  			value := series.ValueAt(step)
   198  			values.SetValueAt(step, t.Apply(value))
   199  		}
   200  
   201  		results[i] = ts.NewSeries(ctx, renamer(series), series.StartTime(), values)
   202  	}
   203  
   204  	in.Values = results
   205  	return in, nil
   206  }
   207  
   208  // Stdev takes one metric or a wildcard seriesList followed by an integer N. Draw the standard deviation
   209  // of all metrics passed for the past N datapoints. If the ratio of null points in the window is greater than
   210  // windowTolerance, skip the calculation.
   211  func Stdev(ctx *Context, in ts.SeriesList, points int, windowTolerance float64, renamer RenamerWithNumPoints) (ts.SeriesList, error) {
   212  	if points <= 0 {
   213  		return ts.NewSeriesList(), errors.NewInvalidParamsError(fmt.Errorf("invalid window size, points=%d", points))
   214  	}
   215  	results := make([]*ts.Series, 0, in.Len())
   216  	for _, series := range in.Values {
   217  		stdevName := renamer(series, points)
   218  		stdevVals := ts.NewValues(ctx, series.MillisPerStep(), series.Len())
   219  		validPoints := 0
   220  		currentSum := 0.0
   221  		currentSumOfSquares := 0.0
   222  		for index := 0; index < series.Len(); index++ {
   223  			newValue := series.ValueAt(index)
   224  			var bootstrapping bool
   225  			var droppedValue float64
   226  
   227  			// Mark whether we've reached our window size, don't drop points out otherwise
   228  			if index < points {
   229  				bootstrapping = true
   230  				droppedValue = math.NaN()
   231  			} else {
   232  				bootstrapping = false
   233  				droppedValue = series.ValueAt(index - points)
   234  			}
   235  
   236  			// Remove the value that just dropped out of the window
   237  			if !bootstrapping && !math.IsNaN(droppedValue) {
   238  				validPoints--
   239  				currentSum -= droppedValue
   240  				currentSumOfSquares -= droppedValue * droppedValue
   241  			}
   242  
   243  			// Add in the value that just popped in the window
   244  			if !math.IsNaN(newValue) {
   245  				validPoints++
   246  				currentSum += newValue
   247  				currentSumOfSquares += newValue * newValue
   248  			}
   249  
   250  			if validPoints > 0 && float64(validPoints)/float64(points) >= windowTolerance {
   251  				deviation := math.Sqrt(float64(validPoints)*currentSumOfSquares-currentSum*currentSum) / float64(validPoints)
   252  				stdevVals.SetValueAt(index, deviation)
   253  			}
   254  		}
   255  		stdevSeries := ts.NewSeries(ctx, stdevName, series.StartTime(), stdevVals)
   256  		results = append(results, stdevSeries)
   257  	}
   258  	in.Values = results
   259  	return in, nil
   260  }
   261  
   262  // RenamerWithNumPoints is a signature for renaming a single series that is passed to Stdev
   263  type RenamerWithNumPoints func(series *ts.Series, points int) string
   264  
   265  // PerSecond computes the derivative between consecutive values in the a time series, taking into
   266  // account the time interval between the values. It skips missing values, and calculates the
   267  // derivative between consecutive non-missing values.
   268  func PerSecond(ctx *Context, in ts.SeriesList, renamer SeriesRenamer) (ts.SeriesList, error) {
   269  	results := make([]*ts.Series, 0, in.Len())
   270  
   271  	for _, series := range in.Values {
   272  		var (
   273  			vals             = ts.NewValues(ctx, series.MillisPerStep(), series.Len())
   274  			prev             = math.NaN()
   275  			secsPerStep      = float64(series.MillisPerStep()) / 1000
   276  			secsSinceLastVal = secsPerStep
   277  		)
   278  
   279  		for step := 0; step < series.Len(); step++ {
   280  			cur := series.ValueAt(step)
   281  
   282  			if math.IsNaN(prev) {
   283  				vals.SetValueAt(step, math.NaN())
   284  				prev = cur
   285  				continue
   286  			}
   287  
   288  			if math.IsNaN(cur) {
   289  				vals.SetValueAt(step, math.NaN())
   290  				secsSinceLastVal += secsPerStep
   291  				continue
   292  			}
   293  
   294  			diff := cur - prev
   295  
   296  			if diff >= 0 {
   297  				vals.SetValueAt(step, diff/secsSinceLastVal)
   298  			} else {
   299  				vals.SetValueAt(step, math.NaN())
   300  			}
   301  
   302  			prev = cur
   303  			secsSinceLastVal = secsPerStep
   304  		}
   305  
   306  		s := ts.NewSeries(ctx, renamer(series), series.StartTime(), vals)
   307  		results = append(results, s)
   308  	}
   309  
   310  	in.Values = results
   311  	return in, nil
   312  }