2022-02-08 01:57:56 -08:00
|
|
|
// Copyright 2022 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 textparse
|
|
|
|
|
|
|
|
import (
|
2024-10-07 04:17:44 -07:00
|
|
|
"errors"
|
|
|
|
"io"
|
2022-02-08 01:57:56 -08:00
|
|
|
"testing"
|
|
|
|
|
2024-10-07 04:17:44 -07:00
|
|
|
"github.com/google/go-cmp/cmp"
|
|
|
|
"github.com/prometheus/common/model"
|
2022-02-08 01:57:56 -08:00
|
|
|
"github.com/stretchr/testify/require"
|
2023-03-25 06:31:24 -07:00
|
|
|
|
2024-10-18 08:12:31 -07:00
|
|
|
"github.com/prometheus/prometheus/config"
|
2024-10-07 04:17:44 -07:00
|
|
|
"github.com/prometheus/prometheus/model/exemplar"
|
|
|
|
"github.com/prometheus/prometheus/model/histogram"
|
2023-03-25 06:31:24 -07:00
|
|
|
"github.com/prometheus/prometheus/model/labels"
|
2024-10-07 04:17:44 -07:00
|
|
|
"github.com/prometheus/prometheus/util/testutil"
|
2022-02-08 01:57:56 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestNewParser(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
2024-10-18 08:12:31 -07:00
|
|
|
requireNilParser := func(t *testing.T, p Parser) {
|
|
|
|
require.Nil(t, p)
|
|
|
|
}
|
|
|
|
|
2022-02-08 01:57:56 -08:00
|
|
|
requirePromParser := func(t *testing.T, p Parser) {
|
|
|
|
require.NotNil(t, p)
|
|
|
|
_, ok := p.(*PromParser)
|
|
|
|
require.True(t, ok)
|
|
|
|
}
|
|
|
|
|
|
|
|
requireOpenMetricsParser := func(t *testing.T, p Parser) {
|
|
|
|
require.NotNil(t, p)
|
|
|
|
_, ok := p.(*OpenMetricsParser)
|
|
|
|
require.True(t, ok)
|
|
|
|
}
|
|
|
|
|
2024-10-18 08:12:31 -07:00
|
|
|
requireProtobufParser := func(t *testing.T, p Parser) {
|
|
|
|
require.NotNil(t, p)
|
|
|
|
_, ok := p.(*ProtobufParser)
|
|
|
|
require.True(t, ok)
|
|
|
|
}
|
|
|
|
|
2022-02-08 01:57:56 -08:00
|
|
|
for name, tt := range map[string]*struct {
|
2024-10-18 08:12:31 -07:00
|
|
|
contentType string
|
|
|
|
fallbackScrapeProtocol config.ScrapeProtocol
|
|
|
|
validateParser func(*testing.T, Parser)
|
|
|
|
err string
|
2022-02-08 01:57:56 -08:00
|
|
|
}{
|
|
|
|
"empty-string": {
|
2024-10-18 08:12:31 -07:00
|
|
|
validateParser: requireNilParser,
|
|
|
|
err: "non-compliant scrape target sending blank Content-Type and no fallback_scrape_protocol specified for target",
|
|
|
|
},
|
|
|
|
"empty-string-fallback-text-plain": {
|
|
|
|
validateParser: requirePromParser,
|
|
|
|
fallbackScrapeProtocol: config.PrometheusText0_0_4,
|
|
|
|
err: "non-compliant scrape target sending blank Content-Type, using fallback_scrape_protocol \"text/plain\"",
|
2022-02-08 01:57:56 -08:00
|
|
|
},
|
|
|
|
"invalid-content-type-1": {
|
|
|
|
contentType: "invalid/",
|
2024-10-18 08:12:31 -07:00
|
|
|
validateParser: requireNilParser,
|
2022-02-08 02:01:37 -08:00
|
|
|
err: "expected token after slash",
|
2022-02-08 01:57:56 -08:00
|
|
|
},
|
2024-10-18 08:12:31 -07:00
|
|
|
"invalid-content-type-1-fallback-text-plain": {
|
|
|
|
contentType: "invalid/",
|
|
|
|
validateParser: requirePromParser,
|
|
|
|
fallbackScrapeProtocol: config.PrometheusText0_0_4,
|
|
|
|
err: "expected token after slash",
|
|
|
|
},
|
|
|
|
"invalid-content-type-1-fallback-openmetrics": {
|
|
|
|
contentType: "invalid/",
|
|
|
|
validateParser: requireOpenMetricsParser,
|
|
|
|
fallbackScrapeProtocol: config.OpenMetricsText0_0_1,
|
|
|
|
err: "expected token after slash",
|
|
|
|
},
|
|
|
|
"invalid-content-type-1-fallback-protobuf": {
|
|
|
|
contentType: "invalid/",
|
|
|
|
validateParser: requireProtobufParser,
|
|
|
|
fallbackScrapeProtocol: config.PrometheusProto,
|
|
|
|
err: "expected token after slash",
|
|
|
|
},
|
2022-02-08 01:57:56 -08:00
|
|
|
"invalid-content-type-2": {
|
|
|
|
contentType: "invalid/invalid/invalid",
|
2024-10-18 08:12:31 -07:00
|
|
|
validateParser: requireNilParser,
|
2022-02-08 02:01:37 -08:00
|
|
|
err: "unexpected content after media subtype",
|
2022-02-08 01:57:56 -08:00
|
|
|
},
|
2024-10-18 08:12:31 -07:00
|
|
|
"invalid-content-type-2-fallback-text-plain": {
|
|
|
|
contentType: "invalid/invalid/invalid",
|
|
|
|
validateParser: requirePromParser,
|
|
|
|
fallbackScrapeProtocol: config.PrometheusText1_0_0,
|
|
|
|
err: "unexpected content after media subtype",
|
|
|
|
},
|
2022-02-08 01:57:56 -08:00
|
|
|
"invalid-content-type-3": {
|
|
|
|
contentType: "/",
|
2024-10-18 08:12:31 -07:00
|
|
|
validateParser: requireNilParser,
|
2022-02-08 02:01:37 -08:00
|
|
|
err: "no media type",
|
2022-02-08 01:57:56 -08:00
|
|
|
},
|
2024-10-18 08:12:31 -07:00
|
|
|
"invalid-content-type-3-fallback-text-plain": {
|
|
|
|
contentType: "/",
|
|
|
|
validateParser: requirePromParser,
|
|
|
|
fallbackScrapeProtocol: config.PrometheusText1_0_0,
|
|
|
|
err: "no media type",
|
|
|
|
},
|
2022-02-08 01:57:56 -08:00
|
|
|
"invalid-content-type-4": {
|
|
|
|
contentType: "application/openmetrics-text; charset=UTF-8; charset=utf-8",
|
2024-10-18 08:12:31 -07:00
|
|
|
validateParser: requireNilParser,
|
2022-02-08 02:01:37 -08:00
|
|
|
err: "duplicate parameter name",
|
2022-02-08 01:57:56 -08:00
|
|
|
},
|
2024-10-18 08:12:31 -07:00
|
|
|
"invalid-content-type-4-fallback-open-metrics": {
|
|
|
|
contentType: "application/openmetrics-text; charset=UTF-8; charset=utf-8",
|
|
|
|
validateParser: requireOpenMetricsParser,
|
|
|
|
fallbackScrapeProtocol: config.OpenMetricsText1_0_0,
|
|
|
|
err: "duplicate parameter name",
|
|
|
|
},
|
2022-02-08 01:57:56 -08:00
|
|
|
"openmetrics": {
|
|
|
|
contentType: "application/openmetrics-text",
|
|
|
|
validateParser: requireOpenMetricsParser,
|
|
|
|
},
|
|
|
|
"openmetrics-with-charset": {
|
|
|
|
contentType: "application/openmetrics-text; charset=utf-8",
|
|
|
|
validateParser: requireOpenMetricsParser,
|
|
|
|
},
|
|
|
|
"openmetrics-with-charset-and-version": {
|
|
|
|
contentType: "application/openmetrics-text; version=1.0.0; charset=utf-8",
|
|
|
|
validateParser: requireOpenMetricsParser,
|
|
|
|
},
|
|
|
|
"plain-text": {
|
|
|
|
contentType: "text/plain",
|
|
|
|
validateParser: requirePromParser,
|
|
|
|
},
|
2024-10-18 08:12:31 -07:00
|
|
|
"protobuf": {
|
|
|
|
contentType: "application/vnd.google.protobuf",
|
|
|
|
validateParser: requireProtobufParser,
|
|
|
|
},
|
2022-02-08 01:57:56 -08:00
|
|
|
"plain-text-with-version": {
|
|
|
|
contentType: "text/plain; version=0.0.4",
|
|
|
|
validateParser: requirePromParser,
|
|
|
|
},
|
|
|
|
"some-other-valid-content-type": {
|
|
|
|
contentType: "text/html",
|
2024-10-18 08:12:31 -07:00
|
|
|
validateParser: requireNilParser,
|
|
|
|
err: "received unsupported Content-Type \"text/html\" and no fallback_scrape_protocol specified for target",
|
|
|
|
},
|
|
|
|
"some-other-valid-content-type-fallback-text-plain": {
|
|
|
|
contentType: "text/html",
|
|
|
|
validateParser: requirePromParser,
|
|
|
|
fallbackScrapeProtocol: config.PrometheusText0_0_4,
|
|
|
|
err: "received unsupported Content-Type \"text/html\", using fallback_scrape_protocol \"text/plain\"",
|
2022-02-08 01:57:56 -08:00
|
|
|
},
|
|
|
|
} {
|
|
|
|
t.Run(name, func(t *testing.T) {
|
|
|
|
tt := tt // Copy to local variable before going parallel.
|
|
|
|
t.Parallel()
|
|
|
|
|
2024-10-18 08:12:31 -07:00
|
|
|
fallbackProtoMediaType := tt.fallbackScrapeProtocol.HeaderMediaType()
|
|
|
|
|
|
|
|
p, err := New([]byte{}, tt.contentType, fallbackProtoMediaType, false, false, labels.NewSymbolTable())
|
2022-02-08 01:57:56 -08:00
|
|
|
tt.validateParser(t, p)
|
2022-02-08 02:01:37 -08:00
|
|
|
if tt.err == "" {
|
2022-02-08 01:57:56 -08:00
|
|
|
require.NoError(t, err)
|
|
|
|
} else {
|
2024-10-06 09:35:29 -07:00
|
|
|
require.ErrorContains(t, err, tt.err)
|
2022-02-08 01:57:56 -08:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2024-10-07 04:17:44 -07:00
|
|
|
|
|
|
|
// parsedEntry represents data that is parsed for each entry.
|
|
|
|
type parsedEntry struct {
|
|
|
|
// In all but EntryComment, EntryInvalid.
|
|
|
|
m string
|
|
|
|
|
|
|
|
// In EntryHistogram.
|
|
|
|
shs *histogram.Histogram
|
|
|
|
fhs *histogram.FloatHistogram
|
|
|
|
|
|
|
|
// In EntrySeries.
|
|
|
|
v float64
|
|
|
|
|
|
|
|
// In EntrySeries and EntryHistogram.
|
|
|
|
lset labels.Labels
|
|
|
|
t *int64
|
|
|
|
es []exemplar.Exemplar
|
|
|
|
ct *int64
|
|
|
|
|
|
|
|
// In EntryType.
|
|
|
|
typ model.MetricType
|
|
|
|
// In EntryHelp.
|
|
|
|
help string
|
|
|
|
// In EntryUnit.
|
|
|
|
unit string
|
|
|
|
// In EntryComment.
|
|
|
|
comment string
|
|
|
|
}
|
|
|
|
|
|
|
|
func requireEntries(t *testing.T, exp, got []parsedEntry) {
|
|
|
|
t.Helper()
|
|
|
|
|
|
|
|
testutil.RequireEqualWithOptions(t, exp, got, []cmp.Option{
|
|
|
|
cmp.AllowUnexported(parsedEntry{}),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func testParse(t *testing.T, p Parser) (ret []parsedEntry) {
|
|
|
|
t.Helper()
|
|
|
|
|
|
|
|
for {
|
|
|
|
et, err := p.Next()
|
|
|
|
if errors.Is(err, io.EOF) {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
var got parsedEntry
|
|
|
|
var m []byte
|
|
|
|
switch et {
|
|
|
|
case EntryInvalid:
|
|
|
|
t.Fatal("entry invalid not expected")
|
|
|
|
case EntrySeries, EntryHistogram:
|
|
|
|
if et == EntrySeries {
|
|
|
|
m, got.t, got.v = p.Series()
|
|
|
|
got.m = string(m)
|
|
|
|
} else {
|
|
|
|
m, got.t, got.shs, got.fhs = p.Histogram()
|
|
|
|
got.m = string(m)
|
|
|
|
}
|
|
|
|
|
|
|
|
p.Metric(&got.lset)
|
|
|
|
for e := (exemplar.Exemplar{}); p.Exemplar(&e); {
|
|
|
|
got.es = append(got.es, e)
|
|
|
|
}
|
|
|
|
// Parser reuses int pointer.
|
|
|
|
if ct := p.CreatedTimestamp(); ct != nil {
|
|
|
|
got.ct = int64p(*ct)
|
|
|
|
}
|
|
|
|
case EntryType:
|
|
|
|
m, got.typ = p.Type()
|
|
|
|
got.m = string(m)
|
|
|
|
|
|
|
|
case EntryHelp:
|
|
|
|
m, h := p.Help()
|
|
|
|
got.m = string(m)
|
|
|
|
got.help = string(h)
|
|
|
|
|
|
|
|
case EntryUnit:
|
|
|
|
m, u := p.Unit()
|
|
|
|
got.m = string(m)
|
|
|
|
got.unit = string(u)
|
|
|
|
|
|
|
|
case EntryComment:
|
|
|
|
got.comment = string(p.Comment())
|
|
|
|
}
|
|
|
|
ret = append(ret, got)
|
|
|
|
}
|
|
|
|
return ret
|
|
|
|
}
|