// Copyright 2013 Prometheus Team // 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 metric import ( "testing" "time" ) func TestGetValuesAtTimeOp(t *testing.T) { var scenarios = []struct { op getValuesAtTimeOp in Values out Values }{ // No values. { op: getValuesAtTimeOp{ baseOp: baseOp{current: testInstant}, }, }, // Operator time before single value. { op: getValuesAtTimeOp{ baseOp: baseOp{current: testInstant}, }, in: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, }, }, // Operator time exactly at single value. { op: getValuesAtTimeOp{ baseOp: baseOp{current: testInstant.Add(1 * time.Minute)}, }, in: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, }, }, // Operator time after single value. { op: getValuesAtTimeOp{ baseOp: baseOp{current: testInstant.Add(2 * time.Minute)}, }, in: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, }, }, // Operator time before two values. { op: getValuesAtTimeOp{ baseOp: baseOp{current: testInstant}, }, in: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, }, }, // Operator time at first of two values. { op: getValuesAtTimeOp{ baseOp: baseOp{current: testInstant.Add(1 * time.Minute)}, }, in: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, }, }, // Operator time between first and second of two values. { op: getValuesAtTimeOp{ baseOp: baseOp{current: testInstant.Add(90 * time.Second)}, }, in: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, }, }, // Operator time at second of two values. { op: getValuesAtTimeOp{ baseOp: baseOp{current: testInstant.Add(2 * time.Minute)}, }, in: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, }, }, // Operator time after second of two values. { op: getValuesAtTimeOp{ baseOp: baseOp{current: testInstant.Add(3 * time.Minute)}, }, in: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, }, }, } for i, scenario := range scenarios { actual := scenario.op.ExtractSamples(scenario.in) if len(actual) != len(scenario.out) { t.Fatalf("%d. expected length %d, got %d: %v", i, len(scenario.out), len(actual), scenario.op) t.Fatalf("%d. expected length %d, got %d", i, len(scenario.out), len(actual)) } for j, out := range scenario.out { if !out.Equal(&actual[j]) { t.Fatalf("%d. expected output %v, got %v", i, scenario.out, actual) } } } } func TestGetValuesAtIntervalOp(t *testing.T) { var scenarios = []struct { op getValuesAtIntervalOp in Values out Values }{ // No values. { op: getValuesAtIntervalOp{ baseOp: baseOp{current: testInstant}, through: testInstant.Add(1 * time.Minute), interval: 30 * time.Second, }, }, // Entire operator range before first value. { op: getValuesAtIntervalOp{ baseOp: baseOp{current: testInstant}, through: testInstant.Add(1 * time.Minute), interval: 30 * time.Second, }, in: Values{ { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(3 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, }, }, // Operator range starts before first value, ends within available values. { op: getValuesAtIntervalOp{ baseOp: baseOp{current: testInstant}, through: testInstant.Add(2 * time.Minute), interval: 30 * time.Second, }, in: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(3 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(3 * time.Minute), Value: 1, }, }, }, // Entire operator range is within available values. { op: getValuesAtIntervalOp{ baseOp: baseOp{current: testInstant.Add(1 * time.Minute)}, through: testInstant.Add(2 * time.Minute), interval: 30 * time.Second, }, in: Values{ { Timestamp: testInstant, Value: 1, }, { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(3 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(3 * time.Minute), Value: 1, }, }, }, // Operator range begins before first value, ends after last. { op: getValuesAtIntervalOp{ baseOp: baseOp{current: testInstant}, through: testInstant.Add(3 * time.Minute), interval: 30 * time.Second, }, in: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, }, }, // Operator range begins within available values, ends after the last value. { op: getValuesAtIntervalOp{ baseOp: baseOp{current: testInstant.Add(2 * time.Minute)}, through: testInstant.Add(4 * time.Minute), interval: 30 * time.Second, }, in: Values{ { Timestamp: testInstant, Value: 1, }, { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(3 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(3 * time.Minute), Value: 1, }, }, }, // Entire operator range after the last available value. { op: getValuesAtIntervalOp{ baseOp: baseOp{current: testInstant.Add(2 * time.Minute)}, through: testInstant.Add(3 * time.Minute), interval: 30 * time.Second, }, in: Values{ { Timestamp: testInstant, Value: 1, }, { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, }, }, // Operator interval skips over several values and ends past the last // available value. This is to verify that we still include the last value // of a series even if we target a time past it and haven't extracted that // value yet as part of a previous interval step (thus the necessity to // skip over values for the test). { op: getValuesAtIntervalOp{ baseOp: baseOp{current: testInstant.Add(30 * time.Second)}, through: testInstant.Add(4 * time.Minute), interval: 3 * time.Minute, }, in: Values{ { Timestamp: testInstant, Value: 1, }, { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(3 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant, Value: 1, }, { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(3 * time.Minute), Value: 1, }, }, }, } for i, scenario := range scenarios { actual := scenario.op.ExtractSamples(scenario.in) if len(actual) != len(scenario.out) { t.Fatalf("%d. expected length %d, got %d: %v", i, len(scenario.out), len(actual), actual) } if len(scenario.in) < 1 { continue } lastExtractedTime := scenario.out[len(scenario.out)-1].Timestamp if !scenario.op.Consumed() && scenario.op.CurrentTime().Before(lastExtractedTime) { t.Fatalf("%d. expected op to be consumed or with CurrentTime() after current chunk, %v, %v", i, scenario.op.CurrentTime(), scenario.out) } for j, out := range scenario.out { if !out.Equal(&actual[j]) { t.Fatalf("%d. expected output %v, got %v", i, scenario.out, actual) } } } } func TestGetValuesAlongRangeOp(t *testing.T) { var scenarios = []struct { op getValuesAlongRangeOp in Values out Values }{ // No values. { op: getValuesAlongRangeOp{ baseOp: baseOp{current: testInstant}, through: testInstant.Add(1 * time.Minute), }, }, // Entire operator range before first value. { op: getValuesAlongRangeOp{ baseOp: baseOp{current: testInstant}, through: testInstant.Add(1 * time.Minute), }, in: Values{ { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(3 * time.Minute), Value: 1, }, }, out: Values{}, }, // Operator range starts before first value, ends within available values. { op: getValuesAlongRangeOp{ baseOp: baseOp{current: testInstant}, through: testInstant.Add(2 * time.Minute), }, in: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(3 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, }, }, // Entire operator range is within available values. { op: getValuesAlongRangeOp{ baseOp: baseOp{current: testInstant.Add(1 * time.Minute)}, through: testInstant.Add(2 * time.Minute), }, in: Values{ { Timestamp: testInstant, Value: 1, }, { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(3 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, }, }, // Operator range begins before first value, ends after last. { op: getValuesAlongRangeOp{ baseOp: baseOp{current: testInstant}, through: testInstant.Add(3 * time.Minute), }, in: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, }, }, // Operator range begins within available values, ends after the last value. { op: getValuesAlongRangeOp{ baseOp: baseOp{current: testInstant.Add(2 * time.Minute)}, through: testInstant.Add(4 * time.Minute), }, in: Values{ { Timestamp: testInstant, Value: 1, }, { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(3 * time.Minute), Value: 1, }, }, out: Values{ { Timestamp: testInstant.Add(2 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(3 * time.Minute), Value: 1, }, }, }, // Entire operator range after the last available value. { op: getValuesAlongRangeOp{ baseOp: baseOp{current: testInstant.Add(2 * time.Minute)}, through: testInstant.Add(3 * time.Minute), }, in: Values{ { Timestamp: testInstant, Value: 1, }, { Timestamp: testInstant.Add(1 * time.Minute), Value: 1, }, }, out: Values{}, }, } for i, scenario := range scenarios { actual := scenario.op.ExtractSamples(scenario.in) if len(actual) != len(scenario.out) { t.Fatalf("%d. expected length %d, got %d: %v", i, len(scenario.out), len(actual), actual) } for j, out := range scenario.out { if !out.Equal(&actual[j]) { t.Fatalf("%d. expected output %v, got %v", i, scenario.out, actual) } } } } func TestGetValueRangeAtIntervalOp(t *testing.T) { testOp := getValueRangeAtIntervalOp{ baseOp: baseOp{current: testInstant.Add(-2 * time.Minute)}, rangeThrough: testInstant, rangeDuration: 2 * time.Minute, interval: 10 * time.Minute, through: testInstant.Add(20 * time.Minute), } var scenarios = []struct { op getValueRangeAtIntervalOp in Values out Values }{ // All values before the first range. { op: testOp, in: Values{ { Timestamp: testInstant.Add(-4 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(-3 * time.Minute), Value: 2, }, }, out: Values{}, }, // Values starting before first range, ending after last. { op: testOp, in: Values{ { Timestamp: testInstant.Add(-4 * time.Minute), Value: 1, }, { Timestamp: testInstant.Add(-3 * time.Minute), Value: 2, }, { Timestamp: testInstant.Add(-2 * time.Minute), Value: 3, }, { Timestamp: testInstant.Add(-1 * time.Minute), Value: 4, }, { Timestamp: testInstant.Add(0 * time.Minute), Value: 5, }, { Timestamp: testInstant.Add(5 * time.Minute), Value: 6, }, { Timestamp: testInstant.Add(8 * time.Minute), Value: 7, }, { Timestamp: testInstant.Add(9 * time.Minute), Value: 8, }, { Timestamp: testInstant.Add(10 * time.Minute), Value: 9, }, { Timestamp: testInstant.Add(15 * time.Minute), Value: 10, }, { Timestamp: testInstant.Add(18 * time.Minute), Value: 11, }, { Timestamp: testInstant.Add(19 * time.Minute), Value: 12, }, { Timestamp: testInstant.Add(20 * time.Minute), Value: 13, }, { Timestamp: testInstant.Add(21 * time.Minute), Value: 14, }, }, out: Values{ { Timestamp: testInstant.Add(-2 * time.Minute), Value: 3, }, { Timestamp: testInstant.Add(-1 * time.Minute), Value: 4, }, { Timestamp: testInstant.Add(0 * time.Minute), Value: 5, }, { Timestamp: testInstant.Add(8 * time.Minute), Value: 7, }, { Timestamp: testInstant.Add(9 * time.Minute), Value: 8, }, { Timestamp: testInstant.Add(10 * time.Minute), Value: 9, }, { Timestamp: testInstant.Add(18 * time.Minute), Value: 11, }, { Timestamp: testInstant.Add(19 * time.Minute), Value: 12, }, { Timestamp: testInstant.Add(20 * time.Minute), Value: 13, }, }, }, // Values starting after last range. { op: testOp, in: Values{ { Timestamp: testInstant.Add(21 * time.Minute), Value: 14, }, }, out: Values{}, }, } for i, scenario := range scenarios { actual := Values{} for !scenario.op.Consumed() { actual = append(actual, scenario.op.ExtractSamples(scenario.in)...) } if len(actual) != len(scenario.out) { t.Fatalf("%d. expected length %d, got %d: %v", i, len(scenario.out), len(actual), actual) } for j, out := range scenario.out { if !out.Equal(&actual[j]) { t.Fatalf("%d. expected output %v, got %v", i, scenario.out, actual) } } } }