mirror of
https://github.com/prometheus/prometheus.git
synced 2025-01-11 22:07:27 -08:00
fix(nhcb): created timestamp fails when keeping classic histograms (#15218)
Some checks are pending
buf.build / lint and publish (push) Waiting to run
CI / Go tests (push) Waiting to run
CI / More Go tests (push) Waiting to run
CI / Go tests with previous Go version (push) Waiting to run
CI / UI tests (push) Waiting to run
CI / Go tests on Windows (push) Waiting to run
CI / Mixins tests (push) Waiting to run
CI / Build Prometheus for common architectures (0) (push) Waiting to run
CI / Build Prometheus for common architectures (1) (push) Waiting to run
CI / Build Prometheus for common architectures (2) (push) Waiting to run
CI / Build Prometheus for all architectures (0) (push) Waiting to run
CI / Build Prometheus for all architectures (1) (push) Waiting to run
CI / Build Prometheus for all architectures (10) (push) Waiting to run
CI / Build Prometheus for all architectures (11) (push) Waiting to run
CI / Build Prometheus for all architectures (2) (push) Waiting to run
CI / Build Prometheus for all architectures (3) (push) Waiting to run
CI / Build Prometheus for all architectures (4) (push) Waiting to run
CI / Build Prometheus for all architectures (5) (push) Waiting to run
CI / Build Prometheus for all architectures (6) (push) Waiting to run
CI / Build Prometheus for all architectures (7) (push) Waiting to run
CI / Build Prometheus for all architectures (8) (push) Waiting to run
CI / Build Prometheus for all architectures (9) (push) Waiting to run
CI / Report status of build Prometheus for all architectures (push) Blocked by required conditions
CI / Check generated parser (push) Waiting to run
CI / golangci-lint (push) Waiting to run
CI / fuzzing (push) Waiting to run
CI / codeql (push) Waiting to run
CI / Publish main branch artifacts (push) Blocked by required conditions
CI / Publish release artefacts (push) Blocked by required conditions
CI / Publish UI on npm Registry (push) Blocked by required conditions
Scorecards supply-chain security / Scorecards analysis (push) Waiting to run
Some checks are pending
buf.build / lint and publish (push) Waiting to run
CI / Go tests (push) Waiting to run
CI / More Go tests (push) Waiting to run
CI / Go tests with previous Go version (push) Waiting to run
CI / UI tests (push) Waiting to run
CI / Go tests on Windows (push) Waiting to run
CI / Mixins tests (push) Waiting to run
CI / Build Prometheus for common architectures (0) (push) Waiting to run
CI / Build Prometheus for common architectures (1) (push) Waiting to run
CI / Build Prometheus for common architectures (2) (push) Waiting to run
CI / Build Prometheus for all architectures (0) (push) Waiting to run
CI / Build Prometheus for all architectures (1) (push) Waiting to run
CI / Build Prometheus for all architectures (10) (push) Waiting to run
CI / Build Prometheus for all architectures (11) (push) Waiting to run
CI / Build Prometheus for all architectures (2) (push) Waiting to run
CI / Build Prometheus for all architectures (3) (push) Waiting to run
CI / Build Prometheus for all architectures (4) (push) Waiting to run
CI / Build Prometheus for all architectures (5) (push) Waiting to run
CI / Build Prometheus for all architectures (6) (push) Waiting to run
CI / Build Prometheus for all architectures (7) (push) Waiting to run
CI / Build Prometheus for all architectures (8) (push) Waiting to run
CI / Build Prometheus for all architectures (9) (push) Waiting to run
CI / Report status of build Prometheus for all architectures (push) Blocked by required conditions
CI / Check generated parser (push) Waiting to run
CI / golangci-lint (push) Waiting to run
CI / fuzzing (push) Waiting to run
CI / codeql (push) Waiting to run
CI / Publish main branch artifacts (push) Blocked by required conditions
CI / Publish release artefacts (push) Blocked by required conditions
CI / Publish UI on npm Registry (push) Blocked by required conditions
Scorecards supply-chain security / Scorecards analysis (push) Waiting to run
The wrong source was used to return the created timestamp, leading to index out of bound panic. One line fix. Refactor the requirement test to be generic and be able to test OpenMetrics and Prom parsers as well. There are some differencies in what the parsers support, the Prom parser doesn't have created timestamp. The protobuf parser uses different formatting to identify the metric for the scrape loop. Each parser represents the sample timestamp differently. Signed-off-by: György Krajcsovits <gyorgy.krajcsovits@grafana.com>
This commit is contained in:
parent
bab587b9dc
commit
eb3b349024
|
@ -169,7 +169,7 @@ func (p *NHCBParser) CreatedTimestamp() *int64 {
|
|||
return p.parser.CreatedTimestamp()
|
||||
}
|
||||
case stateCollecting:
|
||||
return p.parser.CreatedTimestamp()
|
||||
return p.tempCT
|
||||
case stateEmitting:
|
||||
return p.ctNHCB
|
||||
}
|
||||
|
|
|
@ -524,9 +524,6 @@ something_bucket{a="b",le="+Inf"} 9 # {id="something-test"} 2e100 123.000
|
|||
// "classic" means the option "always_scrape_classic_histograms".
|
||||
// "nhcb" means the option "convert_classic_histograms_to_nhcb".
|
||||
//
|
||||
// Currently only with the ProtoBuf parser that supports exponential
|
||||
// histograms.
|
||||
//
|
||||
// Case 1. Only classic histogram is exposed.
|
||||
//
|
||||
// | Scrape Config | Expect classic | Expect exponential | Expect NHCB |.
|
||||
|
@ -550,7 +547,7 @@ something_bucket{a="b",le="+Inf"} 9 # {id="something-test"} 2e100 123.000
|
|||
// | classic=true, nhcb=false | NO | YES | NO |.
|
||||
// | classic=false, nhcb=true | NO | YES | NO |.
|
||||
// | classic=true, nhcb=true | NO | YES | NO |.
|
||||
func TestNHCBParserProtoBufParser_NoNHCBWhenExponential(t *testing.T) {
|
||||
func TestNHCBParser_NoNHCBWhenExponential(t *testing.T) {
|
||||
type requirement struct {
|
||||
expectClassic bool
|
||||
expectExponential bool
|
||||
|
@ -581,33 +578,90 @@ func TestNHCBParserProtoBufParser_NoNHCBWhenExponential(t *testing.T) {
|
|||
},
|
||||
}
|
||||
|
||||
// Create parser from keep classic option.
|
||||
type parserFactory func(bool) Parser
|
||||
|
||||
type testCase struct {
|
||||
name string
|
||||
parser parserFactory
|
||||
classic bool
|
||||
nhcb bool
|
||||
exp []parsedEntry
|
||||
}
|
||||
|
||||
type parserOptions struct {
|
||||
useUTF8sep bool
|
||||
hasCreatedTimeStamp bool
|
||||
}
|
||||
// Defines the parser name, the Parser factory and the test cases
|
||||
// supported by the parser and parser options.
|
||||
parsers := []func() (string, parserFactory, []int, parserOptions){
|
||||
func() (string, parserFactory, []int, parserOptions) {
|
||||
factory := func(keepClassic bool) Parser {
|
||||
inputBuf := createTestProtoBufHistogram(t)
|
||||
return NewProtobufParser(inputBuf.Bytes(), keepClassic, labels.NewSymbolTable())
|
||||
}
|
||||
return "ProtoBuf", factory, []int{1, 2, 3}, parserOptions{useUTF8sep: true, hasCreatedTimeStamp: true}
|
||||
},
|
||||
func() (string, parserFactory, []int, parserOptions) {
|
||||
factory := func(keepClassic bool) Parser {
|
||||
input := createTestOpenMetricsHistogram()
|
||||
return NewOpenMetricsParser([]byte(input), labels.NewSymbolTable(), WithOMParserCTSeriesSkipped())
|
||||
}
|
||||
return "OpenMetrics", factory, []int{1}, parserOptions{hasCreatedTimeStamp: true}
|
||||
},
|
||||
func() (string, parserFactory, []int, parserOptions) {
|
||||
factory := func(keepClassic bool) Parser {
|
||||
input := createTestPromHistogram()
|
||||
return NewPromParser([]byte(input), labels.NewSymbolTable())
|
||||
}
|
||||
return "Prometheus", factory, []int{1}, parserOptions{}
|
||||
},
|
||||
}
|
||||
|
||||
testCases := []testCase{}
|
||||
for _, parser := range parsers {
|
||||
for _, classic := range []bool{false, true} {
|
||||
for _, nhcb := range []bool{false, true} {
|
||||
parserName, parser, supportedCases, options := parser()
|
||||
requirementName := "classic=" + strconv.FormatBool(classic) + ", nhcb=" + strconv.FormatBool(nhcb)
|
||||
tc := testCase{
|
||||
name: "classic=" + strconv.FormatBool(classic) + ", nhcb=" + strconv.FormatBool(nhcb),
|
||||
name: "parser=" + parserName + ", " + requirementName,
|
||||
parser: parser,
|
||||
classic: classic,
|
||||
nhcb: nhcb,
|
||||
exp: []parsedEntry{},
|
||||
}
|
||||
for i, caseI := range cases {
|
||||
req := caseI[tc.name]
|
||||
metric := "test_histogram" + strconv.Itoa(i+1)
|
||||
for _, caseNumber := range supportedCases {
|
||||
caseI := cases[caseNumber-1]
|
||||
req, ok := caseI[requirementName]
|
||||
require.True(t, ok, "Case %d does not have requirement %s", caseNumber, requirementName)
|
||||
metric := "test_histogram" + strconv.Itoa(caseNumber)
|
||||
tc.exp = append(tc.exp, parsedEntry{
|
||||
m: metric,
|
||||
help: "Test histogram " + strconv.Itoa(i+1),
|
||||
help: "Test histogram " + strconv.Itoa(caseNumber),
|
||||
})
|
||||
tc.exp = append(tc.exp, parsedEntry{
|
||||
m: metric,
|
||||
typ: model.MetricTypeHistogram,
|
||||
})
|
||||
|
||||
var ct *int64
|
||||
if options.hasCreatedTimeStamp {
|
||||
ct = int64p(1000)
|
||||
}
|
||||
|
||||
var bucketForMetric func(string) string
|
||||
if options.useUTF8sep {
|
||||
bucketForMetric = func(s string) string {
|
||||
return "_bucket\xffle\xff" + s
|
||||
}
|
||||
} else {
|
||||
bucketForMetric = func(s string) string {
|
||||
return "_bucket{le=\"" + s + "\"}"
|
||||
}
|
||||
}
|
||||
|
||||
if req.expectExponential {
|
||||
// Always expect exponential histogram first.
|
||||
exponentialSeries := []parsedEntry{
|
||||
|
@ -626,7 +680,7 @@ func TestNHCBParserProtoBufParser_NoNHCBWhenExponential(t *testing.T) {
|
|||
},
|
||||
lset: labels.FromStrings("__name__", metric),
|
||||
t: int64p(1234568),
|
||||
ct: int64p(1000),
|
||||
ct: ct,
|
||||
},
|
||||
}
|
||||
tc.exp = append(tc.exp, exponentialSeries...)
|
||||
|
@ -639,42 +693,42 @@ func TestNHCBParserProtoBufParser_NoNHCBWhenExponential(t *testing.T) {
|
|||
v: 175,
|
||||
lset: labels.FromStrings("__name__", metric+"_count"),
|
||||
t: int64p(1234568),
|
||||
ct: int64p(1000),
|
||||
ct: ct,
|
||||
},
|
||||
{
|
||||
m: metric + "_sum",
|
||||
v: 0.0008280461746287094,
|
||||
lset: labels.FromStrings("__name__", metric+"_sum"),
|
||||
t: int64p(1234568),
|
||||
ct: int64p(1000),
|
||||
ct: ct,
|
||||
},
|
||||
{
|
||||
m: metric + "_bucket\xffle\xff-0.0004899999999999998",
|
||||
m: metric + bucketForMetric("-0.0004899999999999998"),
|
||||
v: 2,
|
||||
lset: labels.FromStrings("__name__", metric+"_bucket", "le", "-0.0004899999999999998"),
|
||||
t: int64p(1234568),
|
||||
ct: int64p(1000),
|
||||
ct: ct,
|
||||
},
|
||||
{
|
||||
m: metric + "_bucket\xffle\xff-0.0003899999999999998",
|
||||
m: metric + bucketForMetric("-0.0003899999999999998"),
|
||||
v: 4,
|
||||
lset: labels.FromStrings("__name__", metric+"_bucket", "le", "-0.0003899999999999998"),
|
||||
t: int64p(1234568),
|
||||
ct: int64p(1000),
|
||||
ct: ct,
|
||||
},
|
||||
{
|
||||
m: metric + "_bucket\xffle\xff-0.0002899999999999998",
|
||||
m: metric + bucketForMetric("-0.0002899999999999998"),
|
||||
v: 16,
|
||||
lset: labels.FromStrings("__name__", metric+"_bucket", "le", "-0.0002899999999999998"),
|
||||
t: int64p(1234568),
|
||||
ct: int64p(1000),
|
||||
ct: ct,
|
||||
},
|
||||
{
|
||||
m: metric + "_bucket\xffle\xff+Inf",
|
||||
m: metric + bucketForMetric("+Inf"),
|
||||
v: 175,
|
||||
lset: labels.FromStrings("__name__", metric+"_bucket", "le", "+Inf"),
|
||||
t: int64p(1234568),
|
||||
ct: int64p(1000),
|
||||
ct: ct,
|
||||
},
|
||||
}
|
||||
tc.exp = append(tc.exp, classicSeries...)
|
||||
|
@ -694,7 +748,7 @@ func TestNHCBParserProtoBufParser_NoNHCBWhenExponential(t *testing.T) {
|
|||
},
|
||||
lset: labels.FromStrings("__name__", metric),
|
||||
t: int64p(1234568),
|
||||
ct: int64p(1000),
|
||||
ct: ct,
|
||||
},
|
||||
}
|
||||
tc.exp = append(tc.exp, nhcbSeries...)
|
||||
|
@ -703,12 +757,11 @@ func TestNHCBParserProtoBufParser_NoNHCBWhenExponential(t *testing.T) {
|
|||
testCases = append(testCases, tc)
|
||||
}
|
||||
}
|
||||
|
||||
inputBuf := createTestProtoBufHistogram(t)
|
||||
}
|
||||
|
||||
for _, tc := range testCases {
|
||||
t.Run(tc.name, func(t *testing.T) {
|
||||
p := NewProtobufParser(inputBuf.Bytes(), tc.classic, labels.NewSymbolTable())
|
||||
p := tc.parser(tc.classic)
|
||||
if tc.nhcb {
|
||||
p = NewNHCBParser(p, labels.NewSymbolTable(), tc.classic)
|
||||
}
|
||||
|
@ -860,3 +913,27 @@ metric: <
|
|||
|
||||
return buf
|
||||
}
|
||||
|
||||
func createTestOpenMetricsHistogram() string {
|
||||
return `# HELP test_histogram1 Test histogram 1
|
||||
# TYPE test_histogram1 histogram
|
||||
test_histogram1_count 175 1234.568
|
||||
test_histogram1_sum 0.0008280461746287094 1234.568
|
||||
test_histogram1_bucket{le="-0.0004899999999999998"} 2 1234.568
|
||||
test_histogram1_bucket{le="-0.0003899999999999998"} 4 1234.568
|
||||
test_histogram1_bucket{le="-0.0002899999999999998"} 16 1234.568
|
||||
test_histogram1_bucket{le="+Inf"} 175 1234.568
|
||||
test_histogram1_created 1
|
||||
# EOF`
|
||||
}
|
||||
|
||||
func createTestPromHistogram() string {
|
||||
return `# HELP test_histogram1 Test histogram 1
|
||||
# TYPE test_histogram1 histogram
|
||||
test_histogram1_count 175 1234568
|
||||
test_histogram1_sum 0.0008280461746287094 1234768
|
||||
test_histogram1_bucket{le="-0.0004899999999999998"} 2 1234568
|
||||
test_histogram1_bucket{le="-0.0003899999999999998"} 4 1234568
|
||||
test_histogram1_bucket{le="-0.0002899999999999998"} 16 1234568
|
||||
test_histogram1_bucket{le="+Inf"} 175 1234568`
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue