2020-11-28 07:58:33 -08:00
// Copyright 2020 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 main
import (
"context"
"math"
2022-04-27 02:24:36 -07:00
"os"
2021-02-24 09:42:31 -08:00
"path/filepath"
2020-11-28 07:58:33 -08:00
"testing"
"time"
v1 "github.com/prometheus/client_golang/api/prometheus/v1"
"github.com/prometheus/common/model"
2024-09-09 18:41:53 -07:00
"github.com/prometheus/common/promslog"
2021-10-22 01:19:38 -07:00
"github.com/stretchr/testify/require"
2021-11-08 06:23:17 -08:00
"github.com/prometheus/prometheus/model/labels"
2020-11-28 07:58:33 -08:00
"github.com/prometheus/prometheus/tsdb"
2021-11-28 23:54:23 -08:00
"github.com/prometheus/prometheus/tsdb/chunkenc"
2020-11-28 07:58:33 -08:00
)
2021-02-24 09:42:31 -08:00
type mockQueryRangeAPI struct {
samples model . Matrix
}
2020-11-28 07:58:33 -08:00
2023-10-31 04:35:13 -07:00
func ( mockAPI mockQueryRangeAPI ) QueryRange ( _ context . Context , query string , r v1 . Range , opts ... v1 . Option ) ( model . Value , v1 . Warnings , error ) {
2021-02-24 09:42:31 -08:00
return mockAPI . samples , v1 . Warnings { } , nil
2020-11-28 07:58:33 -08:00
}
2021-10-21 14:28:37 -07:00
const defaultBlockDuration = time . Duration ( tsdb . DefaultBlockDuration ) * time . Millisecond
2020-11-28 07:58:33 -08:00
// TestBackfillRuleIntegration is an integration test that runs all the rule importer code to confirm the parts work together.
func TestBackfillRuleIntegration ( t * testing . T ) {
2021-03-20 12:38:30 -07:00
const (
testMaxSampleCount = 50
testValue = 123
testValue2 = 98
)
var (
2021-10-21 14:28:37 -07:00
start = time . Date ( 2009 , time . November , 10 , 6 , 34 , 0 , 0 , time . UTC )
testTime = model . Time ( start . Add ( - 9 * time . Hour ) . Unix ( ) )
testTime2 = model . Time ( start . Add ( - 8 * time . Hour ) . Unix ( ) )
twentyFourHourDuration , _ = time . ParseDuration ( "24h" )
2021-03-20 12:38:30 -07:00
)
2021-10-22 01:06:44 -07:00
testCases := [ ] struct {
2021-02-24 09:42:31 -08:00
name string
runcount int
2021-10-21 14:28:37 -07:00
maxBlockDuration time . Duration
2021-02-24 09:42:31 -08:00
expectedBlockCount int
expectedSeriesCount int
expectedSampleCount int
samples [ ] * model . SampleStream
} {
2021-10-21 14:28:37 -07:00
{ "no samples" , 1 , defaultBlockDuration , 0 , 0 , 0 , [ ] * model . SampleStream { } } ,
{ "run importer once" , 1 , defaultBlockDuration , 8 , 4 , 4 , [ ] * model . SampleStream { { Metric : model . Metric { "name1" : "val1" } , Values : [ ] model . SamplePair { { Timestamp : testTime , Value : testValue } } } } } ,
{ "run importer with dup name label" , 1 , defaultBlockDuration , 8 , 4 , 4 , [ ] * model . SampleStream { { Metric : model . Metric { "__name__" : "val1" , "name1" : "val1" } , Values : [ ] model . SamplePair { { Timestamp : testTime , Value : testValue } } } } } ,
{ "one importer twice" , 2 , defaultBlockDuration , 8 , 4 , 8 , [ ] * model . SampleStream { { Metric : model . Metric { "name1" : "val1" } , Values : [ ] model . SamplePair { { Timestamp : testTime , Value : testValue } , { Timestamp : testTime2 , Value : testValue2 } } } } } ,
{ "run importer once with larger blocks" , 1 , twentyFourHourDuration , 4 , 4 , 4 , [ ] * model . SampleStream { { Metric : model . Metric { "name1" : "val1" } , Values : [ ] model . SamplePair { { Timestamp : testTime , Value : testValue } } } } } ,
2021-02-24 09:42:31 -08:00
}
for _ , tt := range testCases {
t . Run ( tt . name , func ( t * testing . T ) {
2021-11-23 17:09:28 -08:00
tmpDir := t . TempDir ( )
2021-02-24 09:42:31 -08:00
ctx := context . Background ( )
// Execute the test more than once to simulate running the rule importer twice with the same data.
2021-03-20 12:38:30 -07:00
// We expect duplicate blocks with the same series are created when run more than once.
2021-02-24 09:42:31 -08:00
for i := 0 ; i < tt . runcount ; i ++ {
2021-10-21 14:28:37 -07:00
ruleImporter , err := newTestRuleImporter ( ctx , start , tmpDir , tt . samples , tt . maxBlockDuration )
2021-02-24 09:42:31 -08:00
require . NoError ( t , err )
path1 := filepath . Join ( tmpDir , "test.file" )
require . NoError ( t , createSingleRuleTestFiles ( path1 ) )
path2 := filepath . Join ( tmpDir , "test2.file" )
require . NoError ( t , createMultiRuleTestFiles ( path2 ) )
// Confirm that the rule files were loaded in correctly.
errs := ruleImporter . loadGroups ( ctx , [ ] string { path1 , path2 } )
for _ , err := range errs {
require . NoError ( t , err )
}
2023-12-07 03:35:01 -08:00
require . Len ( t , ruleImporter . groups , 3 )
2021-02-24 09:42:31 -08:00
group1 := ruleImporter . groups [ path1 + ";group0" ]
require . NotNil ( t , group1 )
const defaultInterval = 60
2021-07-28 01:03:46 -07:00
require . Equal ( t , defaultInterval * time . Second , group1 . Interval ( ) )
2021-02-24 09:42:31 -08:00
gRules := group1 . Rules ( )
2023-12-07 03:35:01 -08:00
require . Len ( t , gRules , 1 )
2021-02-24 09:42:31 -08:00
require . Equal ( t , "rule1" , gRules [ 0 ] . Name ( ) )
require . Equal ( t , "ruleExpr" , gRules [ 0 ] . Query ( ) . String ( ) )
2022-02-27 06:37:04 -08:00
require . Equal ( t , 1 , gRules [ 0 ] . Labels ( ) . Len ( ) )
2021-02-24 09:42:31 -08:00
group2 := ruleImporter . groups [ path2 + ";group2" ]
require . NotNil ( t , group2 )
2021-07-28 01:03:46 -07:00
require . Equal ( t , defaultInterval * time . Second , group2 . Interval ( ) )
2021-02-24 09:42:31 -08:00
g2Rules := group2 . Rules ( )
2023-12-07 03:35:01 -08:00
require . Len ( t , g2Rules , 2 )
2021-02-24 09:42:31 -08:00
require . Equal ( t , "grp2_rule1" , g2Rules [ 0 ] . Name ( ) )
require . Equal ( t , "grp2_rule1_expr" , g2Rules [ 0 ] . Query ( ) . String ( ) )
2022-02-27 06:37:04 -08:00
require . Equal ( t , 0 , g2Rules [ 0 ] . Labels ( ) . Len ( ) )
2021-02-24 09:42:31 -08:00
2021-03-15 12:44:58 -07:00
// Backfill all recording rules then check the blocks to confirm the correct data was created.
2021-02-24 09:42:31 -08:00
errs = ruleImporter . importAll ( ctx )
for _ , err := range errs {
require . NoError ( t , err )
}
opts := tsdb . DefaultOptions ( )
2021-06-05 07:29:32 -07:00
db , err := tsdb . Open ( tmpDir , nil , nil , opts , nil )
2021-02-24 09:42:31 -08:00
require . NoError ( t , err )
blocks := db . Blocks ( )
2023-12-07 03:35:01 -08:00
require . Len ( t , blocks , ( i + 1 ) * tt . expectedBlockCount )
2021-02-24 09:42:31 -08:00
2023-09-12 03:37:38 -07:00
q , err := db . Querier ( math . MinInt64 , math . MaxInt64 )
2021-02-24 09:42:31 -08:00
require . NoError ( t , err )
2023-09-12 03:37:38 -07:00
selectedSeries := q . Select ( ctx , false , nil , labels . MustNewMatcher ( labels . MatchRegexp , "" , ".*" ) )
2021-02-24 09:42:31 -08:00
var seriesCount , samplesCount int
for selectedSeries . Next ( ) {
seriesCount ++
series := selectedSeries . At ( )
2022-02-27 06:37:04 -08:00
if series . Labels ( ) . Len ( ) != 3 {
require . Equal ( t , 2 , series . Labels ( ) . Len ( ) )
2022-07-21 09:50:48 -07:00
x := labels . FromStrings ( "__name__" , "grp2_rule1" , "name1" , "val1" )
2021-02-24 09:42:31 -08:00
require . Equal ( t , x , series . Labels ( ) )
} else {
2022-02-27 06:37:04 -08:00
require . Equal ( t , 3 , series . Labels ( ) . Len ( ) )
2021-02-24 09:42:31 -08:00
}
2022-09-20 10:16:45 -07:00
it := series . Iterator ( nil )
2021-11-28 23:54:23 -08:00
for it . Next ( ) == chunkenc . ValFloat {
2021-02-24 09:42:31 -08:00
samplesCount ++
ts , v := it . At ( )
2021-03-20 12:38:30 -07:00
if v == testValue {
require . Equal ( t , int64 ( testTime ) , ts )
} else {
require . Equal ( t , int64 ( testTime2 ) , ts )
}
2021-02-24 09:42:31 -08:00
}
require . NoError ( t , it . Err ( ) )
}
require . NoError ( t , selectedSeries . Err ( ) )
require . Equal ( t , tt . expectedSeriesCount , seriesCount )
require . Equal ( t , tt . expectedSampleCount , samplesCount )
require . NoError ( t , q . Close ( ) )
require . NoError ( t , db . Close ( ) )
2020-11-28 07:58:33 -08:00
}
2021-02-24 09:42:31 -08:00
} )
2020-11-28 07:58:33 -08:00
}
}
2023-04-12 04:05:41 -07:00
func newTestRuleImporter ( _ context . Context , start time . Time , tmpDir string , testSamples model . Matrix , maxBlockDuration time . Duration ) ( * ruleImporter , error ) {
2024-09-09 18:41:53 -07:00
logger := promslog . NewNopLogger ( )
2020-11-28 07:58:33 -08:00
cfg := ruleImporterConfig {
2021-10-21 14:28:37 -07:00
outputDir : tmpDir ,
start : start . Add ( - 10 * time . Hour ) ,
end : start . Add ( - 7 * time . Hour ) ,
evalInterval : 60 * time . Second ,
maxBlockDuration : maxBlockDuration ,
2020-11-28 07:58:33 -08:00
}
2021-02-24 09:42:31 -08:00
return newRuleImporter ( logger , cfg , mockQueryRangeAPI {
samples : testSamples ,
2021-03-15 12:44:58 -07:00
} ) , nil
2020-11-28 07:58:33 -08:00
}
2021-02-24 09:42:31 -08:00
func createSingleRuleTestFiles ( path string ) error {
recordingRules := ` groups :
- name : group0
2020-11-28 07:58:33 -08:00
rules :
2021-02-24 09:42:31 -08:00
- record : rule1
2021-03-15 12:44:58 -07:00
expr : ruleExpr
2020-11-28 07:58:33 -08:00
labels :
2021-02-24 09:42:31 -08:00
testlabel11 : testlabelvalue11
`
2022-04-27 02:24:36 -07:00
return os . WriteFile ( path , [ ] byte ( recordingRules ) , 0 o777 )
2021-02-24 09:42:31 -08:00
}
func createMultiRuleTestFiles ( path string ) error {
recordingRules := ` groups :
- name : group1
rules :
- record : grp1_rule1
expr : grp1_rule1_expr
labels :
2021-09-11 09:31:40 -07:00
testlabel11 : testlabelvalue12
2021-02-24 09:42:31 -08:00
- name : group2
rules :
- record : grp2_rule1
expr : grp2_rule1_expr
- record : grp2_rule2
expr : grp2_rule2_expr
labels :
2021-09-11 09:31:40 -07:00
testlabel11 : testlabelvalue13
2021-02-24 09:42:31 -08:00
`
2022-04-27 02:24:36 -07:00
return os . WriteFile ( path , [ ] byte ( recordingRules ) , 0 o777 )
2020-11-28 07:58:33 -08:00
}
2021-10-16 08:06:05 -07:00
// TestBackfillLabels confirms that the labels in the rule file override the labels from the metrics
// received from Prometheus Query API, including the __name__ label.
func TestBackfillLabels ( t * testing . T ) {
2021-11-23 17:09:28 -08:00
tmpDir := t . TempDir ( )
2021-10-16 08:06:05 -07:00
ctx := context . Background ( )
start := time . Date ( 2009 , time . November , 10 , 6 , 34 , 0 , 0 , time . UTC )
mockAPISamples := [ ] * model . SampleStream {
{
2021-10-17 08:24:31 -07:00
Metric : model . Metric { "name1" : "override-me" , "__name__" : "override-me-too" } ,
2021-10-16 08:06:05 -07:00
Values : [ ] model . SamplePair { { Timestamp : model . TimeFromUnixNano ( start . UnixNano ( ) ) , Value : 123 } } ,
} ,
}
2021-10-21 14:28:37 -07:00
ruleImporter , err := newTestRuleImporter ( ctx , start , tmpDir , mockAPISamples , defaultBlockDuration )
2021-10-22 15:45:29 -07:00
require . NoError ( t , err )
2021-10-16 08:06:05 -07:00
path := filepath . Join ( tmpDir , "test.file" )
recordingRules := ` groups :
- name : group0
rules :
2021-10-17 08:24:31 -07:00
- record : rulename
2021-10-16 08:06:05 -07:00
expr : ruleExpr
labels :
2021-10-17 08:24:31 -07:00
name1 : value - from - rule
2021-10-16 08:06:05 -07:00
`
2022-04-27 02:24:36 -07:00
require . NoError ( t , os . WriteFile ( path , [ ] byte ( recordingRules ) , 0 o777 ) )
2021-10-16 08:06:05 -07:00
errs := ruleImporter . loadGroups ( ctx , [ ] string { path } )
for _ , err := range errs {
require . NoError ( t , err )
}
errs = ruleImporter . importAll ( ctx )
for _ , err := range errs {
require . NoError ( t , err )
}
opts := tsdb . DefaultOptions ( )
db , err := tsdb . Open ( tmpDir , nil , nil , opts , nil )
require . NoError ( t , err )
2023-09-12 03:37:38 -07:00
q , err := db . Querier ( math . MinInt64 , math . MaxInt64 )
2021-10-16 08:06:05 -07:00
require . NoError ( t , err )
t . Run ( "correct-labels" , func ( t * testing . T ) {
2023-09-12 03:37:38 -07:00
selectedSeries := q . Select ( ctx , false , nil , labels . MustNewMatcher ( labels . MatchRegexp , "" , ".*" ) )
2021-10-16 08:06:05 -07:00
for selectedSeries . Next ( ) {
series := selectedSeries . At ( )
2022-07-21 09:50:48 -07:00
expectedLabels := labels . FromStrings ( "__name__" , "rulename" , "name1" , "value-from-rule" )
2021-10-16 08:06:05 -07:00
require . Equal ( t , expectedLabels , series . Labels ( ) )
}
require . NoError ( t , selectedSeries . Err ( ) )
require . NoError ( t , q . Close ( ) )
require . NoError ( t , db . Close ( ) )
} )
}