| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | // Copyright 2015 The Prometheus 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 promql | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | import ( | 
					
						
							|  |  |  | 	"math" | 
					
						
							| 
									
										
										
										
											2015-08-18 05:25:53 +08:00
										 |  |  | 	"regexp" | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	"sort" | 
					
						
							|  |  |  | 	"strconv" | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2016-08-23 03:57:33 +08:00
										 |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-20 23:18:46 +08:00
										 |  |  | 	"github.com/prometheus/common/model" | 
					
						
							| 
									
										
										
										
											2016-12-24 21:01:10 +08:00
										 |  |  | 	"github.com/prometheus/prometheus/pkg/labels" | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | // Function represents a function of the expression language and is
 | 
					
						
							|  |  |  | // used by function nodes.
 | 
					
						
							|  |  |  | type Function struct { | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 	Name       string | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 	ArgTypes   []ValueType | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 	Variadic   int | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 	ReturnType ValueType | 
					
						
							|  |  |  | 	Call       func(ev *evaluator, args Expressions) Value | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | // === time() float64 ===
 | 
					
						
							|  |  |  | func funcTime(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-12-24 17:44:04 +08:00
										 |  |  | 	return Scalar{ | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		V: float64(ev.Timestamp / 1000), | 
					
						
							|  |  |  | 		T: ev.Timestamp, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-29 05:13:41 +08:00
										 |  |  | // extrapolatedRate is a utility function for rate/increase/delta.
 | 
					
						
							|  |  |  | // It calculates the rate (allowing for counter resets if isCounter is true),
 | 
					
						
							|  |  |  | // extrapolates if the first/last sample is close to the boundary, and returns
 | 
					
						
							|  |  |  | // the result as either per-second (if isRate is true) or overall.
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func extrapolatedRate(ev *evaluator, arg Expr, isCounter bool, isRate bool) Value { | 
					
						
							| 
									
										
										
										
											2015-11-29 05:13:41 +08:00
										 |  |  | 	ms := arg.(*MatrixSelector) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	var ( | 
					
						
							|  |  |  | 		matrix       = ev.evalMatrix(ms) | 
					
						
							|  |  |  | 		rangeStart   = ev.Timestamp - durationMilliseconds(ms.Range+ms.Offset) | 
					
						
							|  |  |  | 		rangeEnd     = ev.Timestamp - durationMilliseconds(ms.Offset) | 
					
						
							|  |  |  | 		resultVector = make(Vector, 0, len(matrix)) | 
					
						
							|  |  |  | 	) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	for _, samples := range matrix { | 
					
						
							| 
									
										
										
										
											2015-11-29 05:13:41 +08:00
										 |  |  | 		// No sense in trying to compute a rate without at least two points. Drop
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 		// this Vector element.
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:29:39 +08:00
										 |  |  | 		if len(samples.Points) < 2 { | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-08-25 00:04:41 +08:00
										 |  |  | 		var ( | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 			counterCorrection float64 | 
					
						
							|  |  |  | 			lastValue         float64 | 
					
						
							| 
									
										
										
										
											2015-08-25 00:04:41 +08:00
										 |  |  | 		) | 
					
						
							| 
									
										
										
										
											2016-12-24 18:29:39 +08:00
										 |  |  | 		for _, sample := range samples.Points { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			if isCounter && sample.V < lastValue { | 
					
						
							| 
									
										
										
										
											2016-08-13 02:21:04 +08:00
										 |  |  | 				counterCorrection += lastValue | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			lastValue = sample.V | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-12-24 18:29:39 +08:00
										 |  |  | 		resultValue := lastValue - samples.Points[0].V + counterCorrection | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-29 05:13:41 +08:00
										 |  |  | 		// Duration between first/last samples and boundary of range.
 | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 		durationToStart := float64(samples.Points[0].T-rangeStart) / 1000 | 
					
						
							|  |  |  | 		durationToEnd := float64(rangeEnd-samples.Points[len(samples.Points)-1].T) / 1000 | 
					
						
							| 
									
										
										
										
											2015-11-29 05:13:41 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 		sampledInterval := float64(samples.Points[len(samples.Points)-1].T-samples.Points[0].T) / 1000 | 
					
						
							|  |  |  | 		averageDurationBetweenSamples := sampledInterval / float64(len(samples.Points)-1) | 
					
						
							| 
									
										
										
										
											2015-11-29 05:13:41 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:29:39 +08:00
										 |  |  | 		if isCounter && resultValue > 0 && samples.Points[0].V >= 0 { | 
					
						
							| 
									
										
										
										
											2015-11-29 05:13:41 +08:00
										 |  |  | 			// Counters cannot be negative. If we have any slope at
 | 
					
						
							|  |  |  | 			// all (i.e. resultValue went up), we can extrapolate
 | 
					
						
							|  |  |  | 			// the zero point of the counter. If the duration to the
 | 
					
						
							|  |  |  | 			// zero point is shorter than the durationToStart, we
 | 
					
						
							|  |  |  | 			// take the zero point as the start of the series,
 | 
					
						
							|  |  |  | 			// thereby avoiding extrapolation to negative counter
 | 
					
						
							|  |  |  | 			// values.
 | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 			durationToZero := sampledInterval * (samples.Points[0].V / resultValue) | 
					
						
							| 
									
										
										
										
											2015-11-29 05:13:41 +08:00
										 |  |  | 			if durationToZero < durationToStart { | 
					
						
							|  |  |  | 				durationToStart = durationToZero | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// If the first/last samples are close to the boundaries of the range,
 | 
					
						
							|  |  |  | 		// extrapolate the result. This is as we expect that another sample
 | 
					
						
							|  |  |  | 		// will exist given the spacing between samples we've seen thus far,
 | 
					
						
							|  |  |  | 		// with an allowance for noise.
 | 
					
						
							|  |  |  | 		extrapolationThreshold := averageDurationBetweenSamples * 1.1 | 
					
						
							|  |  |  | 		extrapolateToInterval := sampledInterval | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if durationToStart < extrapolationThreshold { | 
					
						
							|  |  |  | 			extrapolateToInterval += durationToStart | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			extrapolateToInterval += averageDurationBetweenSamples / 2 | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if durationToEnd < extrapolationThreshold { | 
					
						
							|  |  |  | 			extrapolateToInterval += durationToEnd | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			extrapolateToInterval += averageDurationBetweenSamples / 2 | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 		resultValue = resultValue * (extrapolateToInterval / sampledInterval) | 
					
						
							| 
									
										
										
										
											2015-11-29 05:13:41 +08:00
										 |  |  | 		if isRate { | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 			resultValue = resultValue / ms.Range.Seconds() | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:32:10 +08:00
										 |  |  | 		resultVector = append(resultVector, Sample{ | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 			Metric: dropMetricName(samples.Metric), | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			Point:  Point{V: resultValue, T: ev.Timestamp}, | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return resultVector | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:42:54 +08:00
										 |  |  | // === delta(Matrix ValueTypeMatrix) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcDelta(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2015-11-29 05:13:41 +08:00
										 |  |  | 	return extrapolatedRate(ev, args[0], false, false) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | // === rate(node ValueTypeMatrix) Vector ===
 | 
					
						
							|  |  |  | func funcRate(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2015-11-29 05:13:41 +08:00
										 |  |  | 	return extrapolatedRate(ev, args[0], true, true) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | // === increase(node ValueTypeMatrix) Vector ===
 | 
					
						
							|  |  |  | func funcIncrease(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2015-11-29 05:13:41 +08:00
										 |  |  | 	return extrapolatedRate(ev, args[0], true, false) | 
					
						
							| 
									
										
										
										
											2015-05-25 02:48:56 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | // === irate(node ValueTypeMatrix) Vector ===
 | 
					
						
							|  |  |  | func funcIrate(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-08-08 16:52:00 +08:00
										 |  |  | 	return instantValue(ev, args[0], true) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // === idelta(node model.ValMatric) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcIdelta(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-08-08 16:52:00 +08:00
										 |  |  | 	return instantValue(ev, args[0], false) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func instantValue(ev *evaluator, arg Expr, isRate bool) Value { | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 	resultVector := Vector{} | 
					
						
							| 
									
										
										
										
											2016-08-08 16:52:00 +08:00
										 |  |  | 	for _, samples := range ev.evalMatrix(arg) { | 
					
						
							| 
									
										
										
										
											2015-10-10 00:58:43 +08:00
										 |  |  | 		// No sense in trying to compute a rate without at least two points. Drop
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 		// this Vector element.
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:29:39 +08:00
										 |  |  | 		if len(samples.Points) < 2 { | 
					
						
							| 
									
										
										
										
											2015-10-10 00:58:43 +08:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:29:39 +08:00
										 |  |  | 		lastSample := samples.Points[len(samples.Points)-1] | 
					
						
							|  |  |  | 		previousSample := samples.Points[len(samples.Points)-2] | 
					
						
							| 
									
										
										
										
											2015-10-10 00:58:43 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		var resultValue float64 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		if isRate && lastSample.V < previousSample.V { | 
					
						
							| 
									
										
										
										
											2015-10-10 00:58:43 +08:00
										 |  |  | 			// Counter reset.
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			resultValue = lastSample.V | 
					
						
							| 
									
										
										
										
											2015-10-10 00:58:43 +08:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			resultValue = lastSample.V - previousSample.V | 
					
						
							| 
									
										
										
										
											2015-10-10 00:58:43 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		sampledInterval := lastSample.T - previousSample.T | 
					
						
							| 
									
										
										
										
											2015-10-10 00:58:43 +08:00
										 |  |  | 		if sampledInterval == 0 { | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 			// Avoid dividing by 0.float64
 | 
					
						
							| 
									
										
										
										
											2015-10-10 00:58:43 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-08-08 16:40:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-08 16:52:00 +08:00
										 |  |  | 		if isRate { | 
					
						
							|  |  |  | 			// Convert to per-second.
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 			resultValue /= float64(sampledInterval) / 1000 | 
					
						
							| 
									
										
										
										
											2016-08-08 16:40:50 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:32:10 +08:00
										 |  |  | 		resultVector = append(resultVector, Sample{ | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 			Metric: dropMetricName(samples.Metric), | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			Point:  Point{V: resultValue, T: ev.Timestamp}, | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2016-08-08 16:40:50 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return resultVector | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-10 11:29:02 +08:00
										 |  |  | // Calculate the trend value at the given index i in raw data d.
 | 
					
						
							|  |  |  | // This is somewhat analogous to the slope of the trend at the given index.
 | 
					
						
							|  |  |  | // The argument "s" is the set of computed smoothed values.
 | 
					
						
							|  |  |  | // The argument "b" is the set of computed trend factors.
 | 
					
						
							|  |  |  | // The argument "d" is the set of raw input values.
 | 
					
						
							|  |  |  | func calcTrendValue(i int, sf, tf float64, s, b, d []float64) float64 { | 
					
						
							|  |  |  | 	if i == 0 { | 
					
						
							|  |  |  | 		return b[0] | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	x := tf * (s[i] - s[i-1]) | 
					
						
							|  |  |  | 	y := (1 - tf) * b[i-1] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Cache the computed value.
 | 
					
						
							|  |  |  | 	b[i] = x + y | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return b[i] | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Holt-Winters is similar to a weighted moving average, where historical data has exponentially less influence on the current data.
 | 
					
						
							| 
									
										
										
										
											2016-05-12 22:14:48 +08:00
										 |  |  | // Holt-Winter also accounts for trends in data. The smoothing factor (0 < sf < 1) affects how historical data will affect the current
 | 
					
						
							|  |  |  | // data. A lower smoothing factor increases the influence of historical data. The trend factor (0 < tf < 1) affects
 | 
					
						
							|  |  |  | // how trends in historical data will affect the current data. A higher trend factor increases the influence.
 | 
					
						
							| 
									
										
										
										
											2016-03-10 11:29:02 +08:00
										 |  |  | // of trends. Algorithm taken from https://en.wikipedia.org/wiki/Exponential_smoothing titled: "Double exponential smoothing".
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcHoltWinters(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-03-10 11:29:02 +08:00
										 |  |  | 	mat := ev.evalMatrix(args[0]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// The smoothing factor argument.
 | 
					
						
							|  |  |  | 	sf := ev.evalFloat(args[1]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// The trend factor argument.
 | 
					
						
							|  |  |  | 	tf := ev.evalFloat(args[2]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Sanity check the input.
 | 
					
						
							|  |  |  | 	if sf <= 0 || sf >= 1 { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		ev.errorf("invalid smoothing factor. Expected: 0 < sf < 1 goT: %f", sf) | 
					
						
							| 
									
										
										
										
											2016-03-10 11:29:02 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if tf <= 0 || tf >= 1 { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		ev.errorf("invalid trend factor. Expected: 0 < tf < 1 goT: %f", sf) | 
					
						
							| 
									
										
										
										
											2016-03-10 11:29:02 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 	// Make an output Vector large enough to hold the entire result.
 | 
					
						
							|  |  |  | 	resultVector := make(Vector, 0, len(mat)) | 
					
						
							| 
									
										
										
										
											2016-03-10 11:29:02 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Create scratch values.
 | 
					
						
							|  |  |  | 	var s, b, d []float64 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var l int | 
					
						
							|  |  |  | 	for _, samples := range mat { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:29:39 +08:00
										 |  |  | 		l = len(samples.Points) | 
					
						
							| 
									
										
										
										
											2016-03-10 11:29:02 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// Can't do the smoothing operation with less than two points.
 | 
					
						
							|  |  |  | 		if l < 2 { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Resize scratch values.
 | 
					
						
							|  |  |  | 		if l != len(s) { | 
					
						
							|  |  |  | 			s = make([]float64, l) | 
					
						
							|  |  |  | 			b = make([]float64, l) | 
					
						
							|  |  |  | 			d = make([]float64, l) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Fill in the d values with the raw values from the input.
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:29:39 +08:00
										 |  |  | 		for i, v := range samples.Points { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			d[i] = v.V | 
					
						
							| 
									
										
										
										
											2016-03-10 11:29:02 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Set initial values.
 | 
					
						
							|  |  |  | 		s[0] = d[0] | 
					
						
							|  |  |  | 		b[0] = d[1] - d[0] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Run the smoothing operation.
 | 
					
						
							|  |  |  | 		var x, y float64 | 
					
						
							|  |  |  | 		for i := 1; i < len(d); i++ { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// Scale the raw value against the smoothing factor.
 | 
					
						
							|  |  |  | 			x = sf * d[i] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// Scale the last smoothed value with the trend at this point.
 | 
					
						
							|  |  |  | 			y = (1 - sf) * (s[i-1] + calcTrendValue(i-1, sf, tf, s, b, d)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			s[i] = x + y | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:32:10 +08:00
										 |  |  | 		resultVector = append(resultVector, Sample{ | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 			Metric: dropMetricName(samples.Metric), | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			Point:  Point{V: s[len(s)-1], T: ev.Timestamp}, // The last value in the Vector is the smoothed result.
 | 
					
						
							| 
									
										
										
										
											2016-03-10 11:29:02 +08:00
										 |  |  | 		}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return resultVector | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | // === sort(node ValueTypeVector) Vector ===
 | 
					
						
							|  |  |  | func funcSort(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2015-12-30 22:06:51 +08:00
										 |  |  | 	// NaN should sort to the bottom, so take descending sort with NaN first and
 | 
					
						
							|  |  |  | 	// reverse it.
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:37:16 +08:00
										 |  |  | 	byValueSorter := vectorByReverseValueHeap(ev.evalVector(args[0])) | 
					
						
							| 
									
										
										
										
											2015-12-30 22:06:51 +08:00
										 |  |  | 	sort.Sort(sort.Reverse(byValueSorter)) | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 	return Vector(byValueSorter) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | // === sortDesc(node ValueTypeVector) Vector ===
 | 
					
						
							|  |  |  | func funcSortDesc(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2015-12-30 22:06:51 +08:00
										 |  |  | 	// NaN should sort to the bottom, so take ascending sort with NaN first and
 | 
					
						
							|  |  |  | 	// reverse it.
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:37:16 +08:00
										 |  |  | 	byValueSorter := vectorByValueHeap(ev.evalVector(args[0])) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	sort.Sort(sort.Reverse(byValueSorter)) | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 	return Vector(byValueSorter) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | // === clamp_max(Vector ValueTypeVector, max Scalar) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcClampMax(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2015-11-20 08:42:35 +08:00
										 |  |  | 	vec := ev.evalVector(args[0]) | 
					
						
							|  |  |  | 	max := ev.evalFloat(args[1]) | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	for i := range vec { | 
					
						
							|  |  |  | 		el := &vec[i] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 		el.Metric = dropMetricName(el.Metric) | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		el.V = math.Min(max, float64(el.V)) | 
					
						
							| 
									
										
										
										
											2015-11-20 08:42:35 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return vec | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | // === clamp_min(Vector ValueTypeVector, min Scalar) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcClampMin(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2015-11-20 08:42:35 +08:00
										 |  |  | 	vec := ev.evalVector(args[0]) | 
					
						
							|  |  |  | 	min := ev.evalFloat(args[1]) | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	for i := range vec { | 
					
						
							|  |  |  | 		el := &vec[i] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 		el.Metric = dropMetricName(el.Metric) | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		el.V = math.Max(min, float64(el.V)) | 
					
						
							| 
									
										
										
										
											2015-11-20 08:42:35 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return vec | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | // === drop_common_labels(node ValueTypeVector) Vector ===
 | 
					
						
							|  |  |  | func funcDropCommonLabels(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2015-08-25 00:04:41 +08:00
										 |  |  | 	vec := ev.evalVector(args[0]) | 
					
						
							|  |  |  | 	if len(vec) < 1 { | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 		return Vector{} | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 	common := map[string]string{} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, l := range vec[0].Metric { | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		// TODO(julius): Should we also drop common metric names?
 | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 		if l.Name == labels.MetricName { | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		common[l.Name] = l.Value | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-25 00:04:41 +08:00
										 |  |  | 	for _, el := range vec[1:] { | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		for k, v := range common { | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 			for _, l := range el.Metric { | 
					
						
							|  |  |  | 				if l.Name == k && l.Value != v { | 
					
						
							|  |  |  | 					// Deletion of map entries while iterating over them is safe.
 | 
					
						
							|  |  |  | 					// From http://golang.org/ref/spec#For_statements:
 | 
					
						
							|  |  |  | 					// "If map entries that have not yet been reached are deleted during
 | 
					
						
							|  |  |  | 					// iteration, the corresponding iteration values will not be produced."
 | 
					
						
							|  |  |  | 					delete(common, k) | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 	cnames := []string{} | 
					
						
							|  |  |  | 	for n := range common { | 
					
						
							|  |  |  | 		cnames = append(cnames, n) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	for i := range vec { | 
					
						
							|  |  |  | 		el := &vec[i] | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 		el.Metric = labels.NewBuilder(el.Metric).Del(cnames...).Labels() | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-08-25 00:04:41 +08:00
										 |  |  | 	return vec | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | // === round(Vector ValueTypeVector, toNearest=1 Scalar) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcRound(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	// round returns a number rounded to toNearest.
 | 
					
						
							|  |  |  | 	// Ties are solved by rounding up.
 | 
					
						
							|  |  |  | 	toNearest := float64(1) | 
					
						
							|  |  |  | 	if len(args) >= 2 { | 
					
						
							|  |  |  | 		toNearest = ev.evalFloat(args[1]) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Invert as it seems to cause fewer floating point accuracy issues.
 | 
					
						
							|  |  |  | 	toNearestInverse := 1.0 / toNearest | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-25 00:04:41 +08:00
										 |  |  | 	vec := ev.evalVector(args[0]) | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	for i := range vec { | 
					
						
							|  |  |  | 		el := &vec[i] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 		el.Metric = dropMetricName(el.Metric) | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		el.V = math.Floor(float64(el.V)*toNearestInverse+0.5) / toNearestInverse | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-08-25 00:04:41 +08:00
										 |  |  | 	return vec | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:44:04 +08:00
										 |  |  | // === Scalar(node ValueTypeVector) Scalar ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcScalar(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	v := ev.evalVector(args[0]) | 
					
						
							|  |  |  | 	if len(v) != 1 { | 
					
						
							| 
									
										
										
										
											2016-12-24 17:44:04 +08:00
										 |  |  | 		return Scalar{ | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			V: math.NaN(), | 
					
						
							|  |  |  | 			T: ev.Timestamp, | 
					
						
							| 
									
										
										
										
											2015-09-18 22:49:32 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-24 17:44:04 +08:00
										 |  |  | 	return Scalar{ | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		V: v[0].V, | 
					
						
							|  |  |  | 		T: ev.Timestamp, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | // === count_scalar(Vector ValueTypeVector) float64 ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcCountScalar(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-12-24 17:44:04 +08:00
										 |  |  | 	return Scalar{ | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		V: float64(len(ev.evalVector(args[0]))), | 
					
						
							|  |  |  | 		T: ev.Timestamp, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | func aggrOverTime(ev *evaluator, args Expressions, aggrFn func([]Point) float64) Value { | 
					
						
							| 
									
										
										
										
											2015-08-25 00:04:41 +08:00
										 |  |  | 	mat := ev.evalMatrix(args[0]) | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 	resultVector := Vector{} | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-25 00:04:41 +08:00
										 |  |  | 	for _, el := range mat { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:29:39 +08:00
										 |  |  | 		if len(el.Points) == 0 { | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:32:10 +08:00
										 |  |  | 		resultVector = append(resultVector, Sample{ | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 			Metric: dropMetricName(el.Metric), | 
					
						
							| 
									
										
										
										
											2016-12-24 18:29:39 +08:00
										 |  |  | 			Point:  Point{V: aggrFn(el.Points), T: ev.Timestamp}, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return resultVector | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:42:54 +08:00
										 |  |  | // === avg_over_time(Matrix ValueTypeMatrix) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcAvgOverTime(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 	return aggrOverTime(ev, args, func(values []Point) float64 { | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		var sum float64 | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		for _, v := range values { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			sum += v.V | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		return sum / float64(len(values)) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:42:54 +08:00
										 |  |  | // === count_over_time(Matrix ValueTypeMatrix) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcCountOverTime(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 	return aggrOverTime(ev, args, func(values []Point) float64 { | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		return float64(len(values)) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | // === floor(Vector ValueTypeVector) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcFloor(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	vec := ev.evalVector(args[0]) | 
					
						
							|  |  |  | 	for i := range vec { | 
					
						
							|  |  |  | 		el := &vec[i] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 		el.Metric = dropMetricName(el.Metric) | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		el.V = math.Floor(float64(el.V)) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	return vec | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:42:54 +08:00
										 |  |  | // === max_over_time(Matrix ValueTypeMatrix) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcMaxOverTime(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 	return aggrOverTime(ev, args, func(values []Point) float64 { | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		max := math.Inf(-1) | 
					
						
							|  |  |  | 		for _, v := range values { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			max = math.Max(max, float64(v.V)) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		return max | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:42:54 +08:00
										 |  |  | // === min_over_time(Matrix ValueTypeMatrix) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcMinOverTime(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 	return aggrOverTime(ev, args, func(values []Point) float64 { | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		min := math.Inf(1) | 
					
						
							|  |  |  | 		for _, v := range values { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			min = math.Min(min, float64(v.V)) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		return min | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:42:54 +08:00
										 |  |  | // === sum_over_time(Matrix ValueTypeMatrix) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcSumOverTime(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 	return aggrOverTime(ev, args, func(values []Point) float64 { | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		var sum float64 | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		for _, v := range values { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			sum += v.V | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		return sum | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:42:54 +08:00
										 |  |  | // === quantile_over_time(Matrix ValueTypeMatrix) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcQuantileOverTime(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-07-08 20:22:22 +08:00
										 |  |  | 	q := ev.evalFloat(args[0]) | 
					
						
							|  |  |  | 	mat := ev.evalMatrix(args[1]) | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 	resultVector := Vector{} | 
					
						
							| 
									
										
										
										
											2016-07-08 20:22:22 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for _, el := range mat { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:29:39 +08:00
										 |  |  | 		if len(el.Points) == 0 { | 
					
						
							| 
									
										
										
										
											2016-07-08 20:22:22 +08:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 		el.Metric = dropMetricName(el.Metric) | 
					
						
							| 
									
										
										
										
											2016-12-24 18:37:16 +08:00
										 |  |  | 		values := make(vectorByValueHeap, 0, len(el.Points)) | 
					
						
							| 
									
										
										
										
											2016-12-24 18:29:39 +08:00
										 |  |  | 		for _, v := range el.Points { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:32:10 +08:00
										 |  |  | 			values = append(values, Sample{Point: Point{V: v.V}}) | 
					
						
							| 
									
										
										
										
											2016-07-08 20:33:20 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-12-24 18:32:10 +08:00
										 |  |  | 		resultVector = append(resultVector, Sample{ | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			Metric: el.Metric, | 
					
						
							|  |  |  | 			Point:  Point{V: quantile(q, values), T: ev.Timestamp}, | 
					
						
							| 
									
										
										
										
											2016-07-08 20:22:22 +08:00
										 |  |  | 		}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return resultVector | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:42:54 +08:00
										 |  |  | // === stddev_over_time(Matrix ValueTypeMatrix) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcStddevOverTime(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 	return aggrOverTime(ev, args, func(values []Point) float64 { | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		var sum, squaredSum, count float64 | 
					
						
							| 
									
										
										
										
											2016-07-16 07:34:44 +08:00
										 |  |  | 		for _, v := range values { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			sum += v.V | 
					
						
							|  |  |  | 			squaredSum += v.V * v.V | 
					
						
							| 
									
										
										
										
											2016-08-25 00:37:09 +08:00
										 |  |  | 			count++ | 
					
						
							| 
									
										
										
										
											2016-07-16 07:34:44 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		avg := sum / count | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		return math.Sqrt(float64(squaredSum/count - avg*avg)) | 
					
						
							| 
									
										
										
										
											2016-07-16 07:34:44 +08:00
										 |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:42:54 +08:00
										 |  |  | // === stdvar_over_time(Matrix ValueTypeMatrix) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcStdvarOverTime(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 	return aggrOverTime(ev, args, func(values []Point) float64 { | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		var sum, squaredSum, count float64 | 
					
						
							| 
									
										
										
										
											2016-07-16 07:34:44 +08:00
										 |  |  | 		for _, v := range values { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			sum += v.V | 
					
						
							|  |  |  | 			squaredSum += v.V * v.V | 
					
						
							| 
									
										
										
										
											2016-08-25 00:37:09 +08:00
										 |  |  | 			count++ | 
					
						
							| 
									
										
										
										
											2016-07-16 07:34:44 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		avg := sum / count | 
					
						
							|  |  |  | 		return squaredSum/count - avg*avg | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2016-07-08 20:48:48 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-07-16 07:34:44 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | // === abs(Vector ValueTypeVector) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcAbs(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	vec := ev.evalVector(args[0]) | 
					
						
							|  |  |  | 	for i := range vec { | 
					
						
							|  |  |  | 		el := &vec[i] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 		el.Metric = dropMetricName(el.Metric) | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		el.V = math.Abs(float64(el.V)) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	return vec | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | // === absent(Vector ValueTypeVector) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcAbsent(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	if len(ev.evalVector(args[0])) > 0 { | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 		return Vector{} | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 	m := []labels.Label{} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	if vs, ok := args[0].(*VectorSelector); ok { | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		for _, ma := range vs.LabelMatchers { | 
					
						
							| 
									
										
										
										
											2016-12-25 18:34:22 +08:00
										 |  |  | 			if ma.Type == labels.MatchEqual && ma.Name != labels.MetricName { | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 				m = append(m, labels.Label{Name: ma.Name, Value: ma.Value}) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 	return Vector{ | 
					
						
							| 
									
										
										
										
											2016-12-24 18:32:10 +08:00
										 |  |  | 		Sample{ | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			Metric: labels.New(m...), | 
					
						
							|  |  |  | 			Point:  Point{V: 1, T: ev.Timestamp}, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | // === ceil(Vector ValueTypeVector) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcCeil(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	vec := ev.evalVector(args[0]) | 
					
						
							|  |  |  | 	for i := range vec { | 
					
						
							|  |  |  | 		el := &vec[i] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 		el.Metric = dropMetricName(el.Metric) | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		el.V = math.Ceil(float64(el.V)) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	return vec | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | // === exp(Vector ValueTypeVector) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcExp(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	vec := ev.evalVector(args[0]) | 
					
						
							|  |  |  | 	for i := range vec { | 
					
						
							|  |  |  | 		el := &vec[i] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 		el.Metric = dropMetricName(el.Metric) | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		el.V = math.Exp(float64(el.V)) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	return vec | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | // === sqrt(Vector VectorNode) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcSqrt(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	vec := ev.evalVector(args[0]) | 
					
						
							|  |  |  | 	for i := range vec { | 
					
						
							|  |  |  | 		el := &vec[i] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 		el.Metric = dropMetricName(el.Metric) | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		el.V = math.Sqrt(float64(el.V)) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	return vec | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | // === ln(Vector ValueTypeVector) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcLn(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	vec := ev.evalVector(args[0]) | 
					
						
							|  |  |  | 	for i := range vec { | 
					
						
							|  |  |  | 		el := &vec[i] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 		el.Metric = dropMetricName(el.Metric) | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		el.V = math.Log(float64(el.V)) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	return vec | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | // === log2(Vector ValueTypeVector) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcLog2(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	vec := ev.evalVector(args[0]) | 
					
						
							|  |  |  | 	for i := range vec { | 
					
						
							|  |  |  | 		el := &vec[i] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 		el.Metric = dropMetricName(el.Metric) | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		el.V = math.Log2(float64(el.V)) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	return vec | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | // === log10(Vector ValueTypeVector) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcLog10(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	vec := ev.evalVector(args[0]) | 
					
						
							|  |  |  | 	for i := range vec { | 
					
						
							|  |  |  | 		el := &vec[i] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 		el.Metric = dropMetricName(el.Metric) | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		el.V = math.Log10(float64(el.V)) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	return vec | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-14 18:57:05 +08:00
										 |  |  | // === timestamp(Vector ValueTypeVector) Vector ===
 | 
					
						
							|  |  |  | func funcTimestamp(ev *evaluator, args Expressions) Value { | 
					
						
							|  |  |  | 	vec := ev.evalVector(args[0]) | 
					
						
							|  |  |  | 	for i := range vec { | 
					
						
							|  |  |  | 		el := &vec[i] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		el.Metric = dropMetricName(el.Metric) | 
					
						
							|  |  |  | 		el.V = float64(el.T) / 1000.0 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return vec | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-25 00:16:24 +08:00
										 |  |  | // linearRegression performs a least-square linear regression analysis on the
 | 
					
						
							|  |  |  | // provided SamplePairs. It returns the slope, and the intercept value at the
 | 
					
						
							|  |  |  | // provided time.
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | func linearRegression(samples []Point, interceptTime int64) (slope, intercept float64) { | 
					
						
							| 
									
										
										
										
											2016-02-25 00:16:24 +08:00
										 |  |  | 	var ( | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		n            float64 | 
					
						
							|  |  |  | 		sumX, sumY   float64 | 
					
						
							|  |  |  | 		sumXY, sumX2 float64 | 
					
						
							| 
									
										
										
										
											2016-02-25 00:16:24 +08:00
										 |  |  | 	) | 
					
						
							|  |  |  | 	for _, sample := range samples { | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 		x := float64(sample.T-interceptTime) / 1e3 | 
					
						
							| 
									
										
										
										
											2016-02-25 00:16:24 +08:00
										 |  |  | 		n += 1.0 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		sumY += sample.V | 
					
						
							| 
									
										
										
										
											2016-02-25 00:16:24 +08:00
										 |  |  | 		sumX += x | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		sumXY += x * sample.V | 
					
						
							| 
									
										
										
										
											2016-02-25 00:16:24 +08:00
										 |  |  | 		sumX2 += x * x | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	covXY := sumXY - sumX*sumY/n | 
					
						
							|  |  |  | 	varX := sumX2 - sumX*sumX/n | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	slope = covXY / varX | 
					
						
							|  |  |  | 	intercept = sumY/n - slope*sumX/n | 
					
						
							| 
									
										
										
										
											2016-03-09 22:06:00 +08:00
										 |  |  | 	return slope, intercept | 
					
						
							| 
									
										
										
										
											2016-02-25 00:16:24 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | // === deriv(node ValueTypeMatrix) Vector ===
 | 
					
						
							|  |  |  | func funcDeriv(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2015-08-25 00:04:41 +08:00
										 |  |  | 	mat := ev.evalMatrix(args[0]) | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 	resultVector := make(Vector, 0, len(mat)) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-25 00:04:41 +08:00
										 |  |  | 	for _, samples := range mat { | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		// No sense in trying to compute a derivative without at least two points.
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 		// Drop this Vector element.
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:29:39 +08:00
										 |  |  | 		if len(samples.Points) < 2 { | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-08-10 22:28:49 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-08 00:15:38 +08:00
										 |  |  | 		// We pass in an arbitrary timestamp that is near the values in use
 | 
					
						
							|  |  |  | 		// to avoid floating point accuracy issues, see
 | 
					
						
							|  |  |  | 		// https://github.com/prometheus/prometheus/issues/2674
 | 
					
						
							| 
									
										
										
										
											2017-08-10 22:28:49 +08:00
										 |  |  | 		slope, _ := linearRegression(samples.Points, samples.Points[0].T) | 
					
						
							| 
									
										
										
										
											2016-12-24 18:32:10 +08:00
										 |  |  | 		resultSample := Sample{ | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 			Metric: dropMetricName(samples.Metric), | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			Point:  Point{V: slope, T: ev.Timestamp}, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		resultVector = append(resultVector, resultSample) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return resultVector | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | // === predict_linear(node ValueTypeMatrix, k ValueTypeScalar) Vector ===
 | 
					
						
							|  |  |  | func funcPredictLinear(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-02-25 00:16:24 +08:00
										 |  |  | 	mat := ev.evalMatrix(args[0]) | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 	resultVector := make(Vector, 0, len(mat)) | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 	duration := ev.evalFloat(args[1]) | 
					
						
							| 
									
										
										
										
											2015-07-28 19:30:57 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-25 00:16:24 +08:00
										 |  |  | 	for _, samples := range mat { | 
					
						
							|  |  |  | 		// No sense in trying to predict anything without at least two points.
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 		// Drop this Vector element.
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:29:39 +08:00
										 |  |  | 		if len(samples.Points) < 2 { | 
					
						
							| 
									
										
										
										
											2015-07-28 19:30:57 +08:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-12-24 18:29:39 +08:00
										 |  |  | 		slope, intercept := linearRegression(samples.Points, ev.Timestamp) | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:32:10 +08:00
										 |  |  | 		resultVector = append(resultVector, Sample{ | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 			Metric: dropMetricName(samples.Metric), | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			Point:  Point{V: slope*duration + intercept, T: ev.Timestamp}, | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2015-07-28 19:30:57 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-02-25 00:16:24 +08:00
										 |  |  | 	return resultVector | 
					
						
							| 
									
										
										
										
											2015-07-28 19:30:57 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | // === histogram_quantile(k ValueTypeScalar, Vector ValueTypeVector) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcHistogramQuantile(ev *evaluator, args Expressions) Value { | 
					
						
							|  |  |  | 	q := ev.evalFloat(args[0]) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	inVec := ev.evalVector(args[1]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 	outVec := Vector{} | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	signatureToMetricWithBuckets := map[uint64]*metricWithBuckets{} | 
					
						
							|  |  |  | 	for _, el := range inVec { | 
					
						
							|  |  |  | 		upperBound, err := strconv.ParseFloat( | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 			el.Metric.Get(model.BucketLabel), 64, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			// Oops, no bucket label or malformed label value. Skip.
 | 
					
						
							|  |  |  | 			// TODO(beorn7): Issue a warning somehow.
 | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		hash := hashWithoutLabels(el.Metric, excludedLabels...) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		mb, ok := signatureToMetricWithBuckets[hash] | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		if !ok { | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 			el.Metric = labels.NewBuilder(el.Metric). | 
					
						
							|  |  |  | 				Del(labels.BucketLabel, labels.MetricName). | 
					
						
							|  |  |  | 				Labels() | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 			mb = &metricWithBuckets{el.Metric, nil} | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 			signatureToMetricWithBuckets[hash] = mb | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		mb.buckets = append(mb.buckets, bucket{upperBound, el.V}) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, mb := range signatureToMetricWithBuckets { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:32:10 +08:00
										 |  |  | 		outVec = append(outVec, Sample{ | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			Metric: mb.metric, | 
					
						
							|  |  |  | 			Point:  Point{V: bucketQuantile(q, mb.buckets), T: ev.Timestamp}, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return outVec | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:42:54 +08:00
										 |  |  | // === resets(Matrix ValueTypeMatrix) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcResets(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2015-05-26 23:47:52 +08:00
										 |  |  | 	in := ev.evalMatrix(args[0]) | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 	out := make(Vector, 0, len(in)) | 
					
						
							| 
									
										
										
										
											2015-05-26 23:47:52 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for _, samples := range in { | 
					
						
							|  |  |  | 		resets := 0 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:29:39 +08:00
										 |  |  | 		prev := samples.Points[0].V | 
					
						
							|  |  |  | 		for _, sample := range samples.Points[1:] { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			current := sample.V | 
					
						
							| 
									
										
										
										
											2015-05-26 23:47:52 +08:00
										 |  |  | 			if current < prev { | 
					
						
							|  |  |  | 				resets++ | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			prev = current | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:32:10 +08:00
										 |  |  | 		out = append(out, Sample{ | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 			Metric: dropMetricName(samples.Metric), | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			Point:  Point{V: float64(resets), T: ev.Timestamp}, | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2015-05-26 23:47:52 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return out | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:42:54 +08:00
										 |  |  | // === changes(Matrix ValueTypeMatrix) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcChanges(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2015-05-27 01:01:34 +08:00
										 |  |  | 	in := ev.evalMatrix(args[0]) | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 	out := make(Vector, 0, len(in)) | 
					
						
							| 
									
										
										
										
											2015-05-27 01:01:34 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for _, samples := range in { | 
					
						
							|  |  |  | 		changes := 0 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:29:39 +08:00
										 |  |  | 		prev := samples.Points[0].V | 
					
						
							|  |  |  | 		for _, sample := range samples.Points[1:] { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			current := sample.V | 
					
						
							| 
									
										
										
										
											2016-09-30 16:46:58 +08:00
										 |  |  | 			if current != prev && !(math.IsNaN(float64(current)) && math.IsNaN(float64(prev))) { | 
					
						
							| 
									
										
										
										
											2015-05-27 01:01:34 +08:00
										 |  |  | 				changes++ | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			prev = current | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:32:10 +08:00
										 |  |  | 		out = append(out, Sample{ | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 			Metric: dropMetricName(samples.Metric), | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			Point:  Point{V: float64(changes), T: ev.Timestamp}, | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2015-05-27 01:01:34 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return out | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | // === label_replace(Vector ValueTypeVector, dst_label, replacement, src_labelname, regex ValueTypeString) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcLabelReplace(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2015-08-18 05:25:53 +08:00
										 |  |  | 	var ( | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 		vector   = ev.evalVector(args[0]) | 
					
						
							| 
									
										
										
										
											2016-12-24 18:25:26 +08:00
										 |  |  | 		dst      = ev.evalString(args[1]).V | 
					
						
							|  |  |  | 		repl     = ev.evalString(args[2]).V | 
					
						
							|  |  |  | 		src      = ev.evalString(args[3]).V | 
					
						
							|  |  |  | 		regexStr = ev.evalString(args[4]).V | 
					
						
							| 
									
										
										
										
											2015-08-18 05:25:53 +08:00
										 |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-01 21:35:20 +08:00
										 |  |  | 	regex, err := regexp.Compile("^(?:" + regexStr + ")$") | 
					
						
							| 
									
										
										
										
											2015-08-18 05:25:53 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		ev.errorf("invalid regular expression in label_replace(): %s", regexStr) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-08-20 23:18:46 +08:00
										 |  |  | 	if !model.LabelNameRE.MatchString(string(dst)) { | 
					
						
							| 
									
										
										
										
											2015-08-18 05:25:53 +08:00
										 |  |  | 		ev.errorf("invalid destination label name in label_replace(): %s", dst) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	outSet := make(map[uint64]struct{}, len(vector)) | 
					
						
							|  |  |  | 	for i := range vector { | 
					
						
							|  |  |  | 		el := &vector[i] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		srcVal := el.Metric.Get(src) | 
					
						
							| 
									
										
										
										
											2015-08-18 05:25:53 +08:00
										 |  |  | 		indexes := regex.FindStringSubmatchIndex(srcVal) | 
					
						
							|  |  |  | 		// If there is no match, no replacement should take place.
 | 
					
						
							|  |  |  | 		if indexes == nil { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		res := regex.ExpandString([]byte{}, repl, srcVal, indexes) | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		lb := labels.NewBuilder(el.Metric).Del(dst) | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		if len(res) > 0 { | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 			lb.Set(dst, string(res)) | 
					
						
							| 
									
										
										
										
											2015-08-18 05:25:53 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 		el.Metric = lb.Labels() | 
					
						
							| 
									
										
										
										
											2015-08-18 05:25:53 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		h := el.Metric.Hash() | 
					
						
							|  |  |  | 		if _, ok := outSet[h]; ok { | 
					
						
							|  |  |  | 			ev.errorf("duplicated label set in output of label_replace(): %s", el.Metric) | 
					
						
							| 
									
										
										
										
											2015-08-18 05:25:53 +08:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 			outSet[h] = struct{}{} | 
					
						
							| 
									
										
										
										
											2015-08-18 05:25:53 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	return vector | 
					
						
							| 
									
										
										
										
											2015-08-18 05:25:53 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:44:04 +08:00
										 |  |  | // === Vector(s Scalar) Vector ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcVector(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 	return Vector{ | 
					
						
							| 
									
										
										
										
											2016-12-24 18:32:10 +08:00
										 |  |  | 		Sample{ | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 			Metric: labels.Labels{}, | 
					
						
							|  |  |  | 			Point:  Point{V: ev.evalFloat(args[0]), T: ev.Timestamp}, | 
					
						
							| 
									
										
										
										
											2015-09-11 19:09:34 +08:00
										 |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | // === label_join(vector model.ValVector, dest_labelname, separator, src_labelname...) Vector ===
 | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | func funcLabelJoin(ev *evaluator, args Expressions) Value { | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 	var ( | 
					
						
							|  |  |  | 		vector    = ev.evalVector(args[0]) | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		dst       = ev.evalString(args[1]).V | 
					
						
							|  |  |  | 		sep       = ev.evalString(args[2]).V | 
					
						
							|  |  |  | 		srcLabels = make([]string, len(args)-3) | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 	) | 
					
						
							|  |  |  | 	for i := 3; i < len(args); i++ { | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		src := ev.evalString(args[i]).V | 
					
						
							|  |  |  | 		if !model.LabelName(src).IsValid() { | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 			ev.errorf("invalid source label name in label_join(): %s", src) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		srcLabels[i-3] = src | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 	if !model.LabelName(dst).IsValid() { | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		ev.errorf("invalid destination label name in label_join(): %s", dst) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 	outSet := make(map[uint64]struct{}, len(vector)) | 
					
						
							|  |  |  | 	for i := range vector { | 
					
						
							|  |  |  | 		el := &vector[i] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		srcVals := make([]string, len(srcLabels)) | 
					
						
							|  |  |  | 		for i, src := range srcLabels { | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 			srcVals[i] = el.Metric.Get(src) | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		lb := labels.NewBuilder(el.Metric) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		strval := strings.Join(srcVals, sep) | 
					
						
							|  |  |  | 		if strval == "" { | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 			lb.Del(dst) | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 			lb.Set(dst, strval) | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		el.Metric = lb.Labels() | 
					
						
							|  |  |  | 		h := el.Metric.Hash() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if _, exists := outSet[h]; exists { | 
					
						
							|  |  |  | 			ev.errorf("duplicated label set in output of label_join(): %s", el.Metric) | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 			outSet[h] = struct{}{} | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return vector | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-30 01:22:12 +08:00
										 |  |  | // Common code for date related functions.
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func dateWrapper(ev *evaluator, args Expressions, f func(time.Time) float64) Value { | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 	var v Vector | 
					
						
							| 
									
										
										
										
											2016-08-30 01:22:12 +08:00
										 |  |  | 	if len(args) == 0 { | 
					
						
							| 
									
										
										
										
											2016-12-24 17:40:09 +08:00
										 |  |  | 		v = Vector{ | 
					
						
							| 
									
										
										
										
											2016-12-24 18:32:10 +08:00
										 |  |  | 			Sample{ | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 				Metric: labels.Labels{}, | 
					
						
							| 
									
										
										
										
											2017-09-04 20:09:21 +08:00
										 |  |  | 				Point:  Point{V: float64(ev.Timestamp) / 1000, T: ev.Timestamp}, | 
					
						
							| 
									
										
										
										
											2016-08-30 01:22:12 +08:00
										 |  |  | 			}, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		v = ev.evalVector(args[0]) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	for i := range v { | 
					
						
							|  |  |  | 		el := &v[i] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 21:35:24 +08:00
										 |  |  | 		el.Metric = dropMetricName(el.Metric) | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 		t := time.Unix(int64(el.V), 0).UTC() | 
					
						
							|  |  |  | 		el.V = f(t) | 
					
						
							| 
									
										
										
										
											2016-08-23 04:08:13 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-08-30 01:22:12 +08:00
										 |  |  | 	return v | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:44:04 +08:00
										 |  |  | // === days_in_month(v Vector) Scalar ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcDaysInMonth(ev *evaluator, args Expressions) Value { | 
					
						
							|  |  |  | 	return dateWrapper(ev, args, func(t time.Time) float64 { | 
					
						
							|  |  |  | 		return float64(32 - time.Date(t.Year(), t.Month(), 32, 0, 0, 0, 0, time.UTC).Day()) | 
					
						
							| 
									
										
										
										
											2016-08-30 01:22:12 +08:00
										 |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2016-08-23 04:08:13 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:44:04 +08:00
										 |  |  | // === day_of_month(v Vector) Scalar ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcDayOfMonth(ev *evaluator, args Expressions) Value { | 
					
						
							|  |  |  | 	return dateWrapper(ev, args, func(t time.Time) float64 { | 
					
						
							|  |  |  | 		return float64(t.Day()) | 
					
						
							| 
									
										
										
										
											2016-08-30 01:22:12 +08:00
										 |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2016-08-23 03:57:33 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:44:04 +08:00
										 |  |  | // === day_of_week(v Vector) Scalar ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcDayOfWeek(ev *evaluator, args Expressions) Value { | 
					
						
							|  |  |  | 	return dateWrapper(ev, args, func(t time.Time) float64 { | 
					
						
							|  |  |  | 		return float64(t.Weekday()) | 
					
						
							| 
									
										
										
										
											2016-08-30 01:22:12 +08:00
										 |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2016-08-23 03:57:33 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:44:04 +08:00
										 |  |  | // === hour(v Vector) Scalar ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcHour(ev *evaluator, args Expressions) Value { | 
					
						
							|  |  |  | 	return dateWrapper(ev, args, func(t time.Time) float64 { | 
					
						
							|  |  |  | 		return float64(t.Hour()) | 
					
						
							| 
									
										
										
										
											2016-08-30 01:22:12 +08:00
										 |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2016-08-23 03:57:33 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:44:04 +08:00
										 |  |  | // === minute(v Vector) Scalar ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcMinute(ev *evaluator, args Expressions) Value { | 
					
						
							|  |  |  | 	return dateWrapper(ev, args, func(t time.Time) float64 { | 
					
						
							|  |  |  | 		return float64(t.Minute()) | 
					
						
							| 
									
										
										
										
											2016-09-13 03:29:44 +08:00
										 |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:44:04 +08:00
										 |  |  | // === month(v Vector) Scalar ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcMonth(ev *evaluator, args Expressions) Value { | 
					
						
							|  |  |  | 	return dateWrapper(ev, args, func(t time.Time) float64 { | 
					
						
							|  |  |  | 		return float64(t.Month()) | 
					
						
							| 
									
										
										
										
											2016-08-30 01:22:12 +08:00
										 |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2016-08-23 03:57:33 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 17:44:04 +08:00
										 |  |  | // === year(v Vector) Scalar ===
 | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | func funcYear(ev *evaluator, args Expressions) Value { | 
					
						
							|  |  |  | 	return dateWrapper(ev, args, func(t time.Time) float64 { | 
					
						
							|  |  |  | 		return float64(t.Year()) | 
					
						
							| 
									
										
										
										
											2016-08-30 01:22:12 +08:00
										 |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2016-08-23 03:57:33 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | var functions = map[string]*Function{ | 
					
						
							|  |  |  | 	"abs": { | 
					
						
							|  |  |  | 		Name:       "abs", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcAbs, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	"absent": { | 
					
						
							|  |  |  | 		Name:       "absent", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcAbsent, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	"avg_over_time": { | 
					
						
							|  |  |  | 		Name:       "avg_over_time", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeMatrix}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcAvgOverTime, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	"ceil": { | 
					
						
							|  |  |  | 		Name:       "ceil", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcCeil, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2015-05-27 01:01:34 +08:00
										 |  |  | 	"changes": { | 
					
						
							|  |  |  | 		Name:       "changes", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeMatrix}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-05-27 01:01:34 +08:00
										 |  |  | 		Call:       funcChanges, | 
					
						
							|  |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2015-11-20 08:42:35 +08:00
										 |  |  | 	"clamp_max": { | 
					
						
							|  |  |  | 		Name:       "clamp_max", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector, ValueTypeScalar}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-11-20 08:42:35 +08:00
										 |  |  | 		Call:       funcClampMax, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	"clamp_min": { | 
					
						
							|  |  |  | 		Name:       "clamp_min", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector, ValueTypeScalar}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-11-20 08:42:35 +08:00
										 |  |  | 		Call:       funcClampMin, | 
					
						
							|  |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	"count_over_time": { | 
					
						
							|  |  |  | 		Name:       "count_over_time", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeMatrix}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcCountOverTime, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	"count_scalar": { | 
					
						
							|  |  |  | 		Name:       "count_scalar", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeScalar, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcCountScalar, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2016-08-23 04:08:13 +08:00
										 |  |  | 	"days_in_month": { | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Name:       "days_in_month", | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Variadic:   1, | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Call:       funcDaysInMonth, | 
					
						
							| 
									
										
										
										
											2016-08-23 04:08:13 +08:00
										 |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2016-08-23 03:57:33 +08:00
										 |  |  | 	"day_of_month": { | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Name:       "day_of_month", | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Variadic:   1, | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Call:       funcDayOfMonth, | 
					
						
							| 
									
										
										
										
											2016-08-23 03:57:33 +08:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	"day_of_week": { | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Name:       "day_of_week", | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Variadic:   1, | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Call:       funcDayOfWeek, | 
					
						
							| 
									
										
										
										
											2016-08-23 03:57:33 +08:00
										 |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	"delta": { | 
					
						
							| 
									
										
										
										
											2015-10-10 22:41:23 +08:00
										 |  |  | 		Name:       "delta", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeMatrix}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-10-10 22:41:23 +08:00
										 |  |  | 		Call:       funcDelta, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	"deriv": { | 
					
						
							|  |  |  | 		Name:       "deriv", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeMatrix}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcDeriv, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	"drop_common_labels": { | 
					
						
							|  |  |  | 		Name:       "drop_common_labels", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcDropCommonLabels, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	"exp": { | 
					
						
							|  |  |  | 		Name:       "exp", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcExp, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	"floor": { | 
					
						
							|  |  |  | 		Name:       "floor", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcFloor, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	"histogram_quantile": { | 
					
						
							|  |  |  | 		Name:       "histogram_quantile", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeScalar, ValueTypeVector}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcHistogramQuantile, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2016-03-10 11:29:02 +08:00
										 |  |  | 	"holt_winters": { | 
					
						
							|  |  |  | 		Name:       "holt_winters", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeMatrix, ValueTypeScalar, ValueTypeScalar}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2016-03-10 11:29:02 +08:00
										 |  |  | 		Call:       funcHoltWinters, | 
					
						
							|  |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2016-08-24 05:34:22 +08:00
										 |  |  | 	"hour": { | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Name:       "hour", | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Variadic:   1, | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Call:       funcHour, | 
					
						
							| 
									
										
										
										
											2015-10-10 00:58:43 +08:00
										 |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2016-08-08 16:40:50 +08:00
										 |  |  | 	"idelta": { | 
					
						
							|  |  |  | 		Name:       "idelta", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeMatrix}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2016-08-08 16:40:50 +08:00
										 |  |  | 		Call:       funcIdelta, | 
					
						
							|  |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2016-08-23 04:00:53 +08:00
										 |  |  | 	"increase": { | 
					
						
							|  |  |  | 		Name:       "increase", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeMatrix}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2016-08-23 04:00:53 +08:00
										 |  |  | 		Call:       funcIncrease, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	"irate": { | 
					
						
							|  |  |  | 		Name:       "irate", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeMatrix}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2016-08-23 04:00:53 +08:00
										 |  |  | 		Call:       funcIrate, | 
					
						
							|  |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2015-08-18 05:25:53 +08:00
										 |  |  | 	"label_replace": { | 
					
						
							|  |  |  | 		Name:       "label_replace", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector, ValueTypeString, ValueTypeString, ValueTypeString, ValueTypeString}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-08-18 05:25:53 +08:00
										 |  |  | 		Call:       funcLabelReplace, | 
					
						
							|  |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 	"label_join": { | 
					
						
							|  |  |  | 		Name:       "label_join", | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector, ValueTypeString, ValueTypeString, ValueTypeString}, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Variadic:   -1, | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Call:       funcLabelJoin, | 
					
						
							|  |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	"ln": { | 
					
						
							|  |  |  | 		Name:       "ln", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcLn, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	"log10": { | 
					
						
							|  |  |  | 		Name:       "log10", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcLog10, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	"log2": { | 
					
						
							|  |  |  | 		Name:       "log2", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcLog2, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	"max_over_time": { | 
					
						
							|  |  |  | 		Name:       "max_over_time", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeMatrix}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcMaxOverTime, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	"min_over_time": { | 
					
						
							|  |  |  | 		Name:       "min_over_time", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeMatrix}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcMinOverTime, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2016-09-13 03:29:44 +08:00
										 |  |  | 	"minute": { | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Name:       "minute", | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Variadic:   1, | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Call:       funcMinute, | 
					
						
							| 
									
										
										
										
											2016-09-13 03:29:44 +08:00
										 |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2016-08-24 05:34:22 +08:00
										 |  |  | 	"month": { | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Name:       "month", | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Variadic:   1, | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Call:       funcMonth, | 
					
						
							| 
									
										
										
										
											2016-08-23 03:57:33 +08:00
										 |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2015-07-28 19:30:57 +08:00
										 |  |  | 	"predict_linear": { | 
					
						
							|  |  |  | 		Name:       "predict_linear", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeMatrix, ValueTypeScalar}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-07-28 19:30:57 +08:00
										 |  |  | 		Call:       funcPredictLinear, | 
					
						
							|  |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2016-07-08 20:22:22 +08:00
										 |  |  | 	"quantile_over_time": { | 
					
						
							|  |  |  | 		Name:       "quantile_over_time", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeScalar, ValueTypeMatrix}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2016-07-08 20:22:22 +08:00
										 |  |  | 		Call:       funcQuantileOverTime, | 
					
						
							|  |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	"rate": { | 
					
						
							|  |  |  | 		Name:       "rate", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeMatrix}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcRate, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2015-05-26 23:47:52 +08:00
										 |  |  | 	"resets": { | 
					
						
							|  |  |  | 		Name:       "resets", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeMatrix}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-05-26 23:47:52 +08:00
										 |  |  | 		Call:       funcResets, | 
					
						
							|  |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	"round": { | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Name:       "round", | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector, ValueTypeScalar}, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Variadic:   1, | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Call:       funcRound, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	"scalar": { | 
					
						
							|  |  |  | 		Name:       "scalar", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeScalar, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcScalar, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	"sort": { | 
					
						
							|  |  |  | 		Name:       "sort", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcSort, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	"sort_desc": { | 
					
						
							|  |  |  | 		Name:       "sort_desc", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcSortDesc, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	"sqrt": { | 
					
						
							|  |  |  | 		Name:       "sqrt", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcSqrt, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2016-07-16 07:34:44 +08:00
										 |  |  | 	"stddev_over_time": { | 
					
						
							|  |  |  | 		Name:       "stddev_over_time", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeMatrix}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2016-07-16 07:34:44 +08:00
										 |  |  | 		Call:       funcStddevOverTime, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	"stdvar_over_time": { | 
					
						
							|  |  |  | 		Name:       "stdvar_over_time", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeMatrix}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2016-07-16 07:34:44 +08:00
										 |  |  | 		Call:       funcStdvarOverTime, | 
					
						
							|  |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	"sum_over_time": { | 
					
						
							|  |  |  | 		Name:       "sum_over_time", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeMatrix}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcSumOverTime, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	"time": { | 
					
						
							|  |  |  | 		Name:       "time", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeScalar, | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		Call:       funcTime, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2017-04-14 18:57:05 +08:00
										 |  |  | 	"timestamp": { | 
					
						
							|  |  |  | 		Name:       "timestamp", | 
					
						
							|  |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							|  |  |  | 		Call:       funcTimestamp, | 
					
						
							|  |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	"vector": { | 
					
						
							|  |  |  | 		Name:       "vector", | 
					
						
							| 
									
										
										
										
											2016-12-23 20:51:59 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeScalar}, | 
					
						
							|  |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2015-09-16 20:33:12 +08:00
										 |  |  | 		Call:       funcVector, | 
					
						
							| 
									
										
										
										
											2015-09-11 19:09:34 +08:00
										 |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2016-08-23 03:57:33 +08:00
										 |  |  | 	"year": { | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Name:       "year", | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		ArgTypes:   []ValueType{ValueTypeVector}, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Variadic:   1, | 
					
						
							| 
									
										
										
										
											2017-06-23 19:15:44 +08:00
										 |  |  | 		ReturnType: ValueTypeVector, | 
					
						
							| 
									
										
										
										
											2017-06-16 21:51:22 +08:00
										 |  |  | 		Call:       funcYear, | 
					
						
							| 
									
										
										
										
											2016-08-23 03:57:33 +08:00
										 |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | // getFunction returns a predefined Function object for the given name.
 | 
					
						
							|  |  |  | func getFunction(name string) (*Function, bool) { | 
					
						
							| 
									
										
										
										
											2015-03-31 00:12:51 +08:00
										 |  |  | 	function, ok := functions[name] | 
					
						
							|  |  |  | 	return function, ok | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:37:16 +08:00
										 |  |  | type vectorByValueHeap Vector | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:37:16 +08:00
										 |  |  | func (s vectorByValueHeap) Len() int { | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	return len(s) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:37:16 +08:00
										 |  |  | func (s vectorByValueHeap) Less(i, j int) bool { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 	if math.IsNaN(float64(s[i].V)) { | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 		return true | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 	return s[i].V < s[j].V | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:37:16 +08:00
										 |  |  | func (s vectorByValueHeap) Swap(i, j int) { | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	s[i], s[j] = s[j], s[i] | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:37:16 +08:00
										 |  |  | func (s *vectorByValueHeap) Push(x interface{}) { | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	*s = append(*s, *(x.(*Sample))) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:37:16 +08:00
										 |  |  | func (s *vectorByValueHeap) Pop() interface{} { | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	old := *s | 
					
						
							|  |  |  | 	n := len(old) | 
					
						
							|  |  |  | 	el := old[n-1] | 
					
						
							|  |  |  | 	*s = old[0 : n-1] | 
					
						
							|  |  |  | 	return el | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:37:16 +08:00
										 |  |  | type vectorByReverseValueHeap Vector | 
					
						
							| 
									
										
										
										
											2015-12-22 21:53:43 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:37:16 +08:00
										 |  |  | func (s vectorByReverseValueHeap) Len() int { | 
					
						
							| 
									
										
										
										
											2015-12-22 21:53:43 +08:00
										 |  |  | 	return len(s) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:37:16 +08:00
										 |  |  | func (s vectorByReverseValueHeap) Less(i, j int) bool { | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 	if math.IsNaN(float64(s[i].V)) { | 
					
						
							| 
									
										
										
										
											2015-12-22 21:53:43 +08:00
										 |  |  | 		return true | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-24 18:23:06 +08:00
										 |  |  | 	return s[i].V > s[j].V | 
					
						
							| 
									
										
										
										
											2015-12-22 21:53:43 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:37:16 +08:00
										 |  |  | func (s vectorByReverseValueHeap) Swap(i, j int) { | 
					
						
							| 
									
										
										
										
											2015-12-22 21:53:43 +08:00
										 |  |  | 	s[i], s[j] = s[j], s[i] | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:37:16 +08:00
										 |  |  | func (s *vectorByReverseValueHeap) Push(x interface{}) { | 
					
						
							| 
									
										
										
										
											2016-12-28 16:16:48 +08:00
										 |  |  | 	*s = append(*s, *(x.(*Sample))) | 
					
						
							| 
									
										
										
										
											2015-12-22 21:53:43 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 18:37:16 +08:00
										 |  |  | func (s *vectorByReverseValueHeap) Pop() interface{} { | 
					
						
							| 
									
										
										
										
											2015-12-22 21:53:43 +08:00
										 |  |  | 	old := *s | 
					
						
							|  |  |  | 	n := len(old) | 
					
						
							|  |  |  | 	el := old[n-1] | 
					
						
							|  |  |  | 	*s = old[0 : n-1] | 
					
						
							|  |  |  | 	return el | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } |