| 
									
										
										
										
											2022-02-08 17: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 19:17:44 +08:00
										 |  |  | 	"errors" | 
					
						
							|  |  |  | 	"io" | 
					
						
							| 
									
										
										
										
											2022-02-08 17:57:56 +08:00
										 |  |  | 	"testing" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-07 19:17:44 +08:00
										 |  |  | 	"github.com/google/go-cmp/cmp" | 
					
						
							| 
									
										
										
										
											2025-02-13 18:38:35 +08:00
										 |  |  | 	"github.com/google/go-cmp/cmp/cmpopts" | 
					
						
							| 
									
										
										
										
											2024-10-07 19:17:44 +08:00
										 |  |  | 	"github.com/prometheus/common/model" | 
					
						
							| 
									
										
										
										
											2022-02-08 17:57:56 +08:00
										 |  |  | 	"github.com/stretchr/testify/require" | 
					
						
							| 
									
										
										
										
											2023-03-25 21:31:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-18 23:12:31 +08:00
										 |  |  | 	"github.com/prometheus/prometheus/config" | 
					
						
							| 
									
										
										
										
											2024-10-07 19:17:44 +08:00
										 |  |  | 	"github.com/prometheus/prometheus/model/exemplar" | 
					
						
							|  |  |  | 	"github.com/prometheus/prometheus/model/histogram" | 
					
						
							| 
									
										
										
										
											2023-03-25 21:31:24 +08:00
										 |  |  | 	"github.com/prometheus/prometheus/model/labels" | 
					
						
							| 
									
										
										
										
											2024-10-07 19:17:44 +08:00
										 |  |  | 	"github.com/prometheus/prometheus/util/testutil" | 
					
						
							| 
									
										
										
										
											2022-02-08 17:57:56 +08:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestNewParser(t *testing.T) { | 
					
						
							|  |  |  | 	t.Parallel() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-18 23:12:31 +08:00
										 |  |  | 	requireNilParser := func(t *testing.T, p Parser) { | 
					
						
							|  |  |  | 		require.Nil(t, p) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-08 17: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 23:12:31 +08:00
										 |  |  | 	requireProtobufParser := func(t *testing.T, p Parser) { | 
					
						
							|  |  |  | 		require.NotNil(t, p) | 
					
						
							|  |  |  | 		_, ok := p.(*ProtobufParser) | 
					
						
							|  |  |  | 		require.True(t, ok) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-08 17:57:56 +08:00
										 |  |  | 	for name, tt := range map[string]*struct { | 
					
						
							| 
									
										
										
										
											2024-10-18 23:12:31 +08:00
										 |  |  | 		contentType            string | 
					
						
							|  |  |  | 		fallbackScrapeProtocol config.ScrapeProtocol | 
					
						
							|  |  |  | 		validateParser         func(*testing.T, Parser) | 
					
						
							|  |  |  | 		err                    string | 
					
						
							| 
									
										
										
										
											2022-02-08 17:57:56 +08:00
										 |  |  | 	}{ | 
					
						
							|  |  |  | 		"empty-string": { | 
					
						
							| 
									
										
										
										
											2024-10-18 23:12:31 +08: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 17:57:56 +08:00
										 |  |  | 		}, | 
					
						
							|  |  |  | 		"invalid-content-type-1": { | 
					
						
							|  |  |  | 			contentType:    "invalid/", | 
					
						
							| 
									
										
										
										
											2024-10-18 23:12:31 +08:00
										 |  |  | 			validateParser: requireNilParser, | 
					
						
							| 
									
										
										
										
											2022-02-08 18:01:37 +08:00
										 |  |  | 			err:            "expected token after slash", | 
					
						
							| 
									
										
										
										
											2022-02-08 17:57:56 +08:00
										 |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2024-10-18 23:12:31 +08: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 17:57:56 +08:00
										 |  |  | 		"invalid-content-type-2": { | 
					
						
							|  |  |  | 			contentType:    "invalid/invalid/invalid", | 
					
						
							| 
									
										
										
										
											2024-10-18 23:12:31 +08:00
										 |  |  | 			validateParser: requireNilParser, | 
					
						
							| 
									
										
										
										
											2022-02-08 18:01:37 +08:00
										 |  |  | 			err:            "unexpected content after media subtype", | 
					
						
							| 
									
										
										
										
											2022-02-08 17:57:56 +08:00
										 |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2024-10-18 23:12:31 +08: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 17:57:56 +08:00
										 |  |  | 		"invalid-content-type-3": { | 
					
						
							|  |  |  | 			contentType:    "/", | 
					
						
							| 
									
										
										
										
											2024-10-18 23:12:31 +08:00
										 |  |  | 			validateParser: requireNilParser, | 
					
						
							| 
									
										
										
										
											2022-02-08 18:01:37 +08:00
										 |  |  | 			err:            "no media type", | 
					
						
							| 
									
										
										
										
											2022-02-08 17:57:56 +08:00
										 |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2024-10-18 23:12:31 +08:00
										 |  |  | 		"invalid-content-type-3-fallback-text-plain": { | 
					
						
							|  |  |  | 			contentType:            "/", | 
					
						
							|  |  |  | 			validateParser:         requirePromParser, | 
					
						
							|  |  |  | 			fallbackScrapeProtocol: config.PrometheusText1_0_0, | 
					
						
							|  |  |  | 			err:                    "no media type", | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2022-02-08 17:57:56 +08:00
										 |  |  | 		"invalid-content-type-4": { | 
					
						
							|  |  |  | 			contentType:    "application/openmetrics-text; charset=UTF-8; charset=utf-8", | 
					
						
							| 
									
										
										
										
											2024-10-18 23:12:31 +08:00
										 |  |  | 			validateParser: requireNilParser, | 
					
						
							| 
									
										
										
										
											2022-02-08 18:01:37 +08:00
										 |  |  | 			err:            "duplicate parameter name", | 
					
						
							| 
									
										
										
										
											2022-02-08 17:57:56 +08:00
										 |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2024-10-18 23:12:31 +08: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 17: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 23:12:31 +08:00
										 |  |  | 		"protobuf": { | 
					
						
							|  |  |  | 			contentType:    "application/vnd.google.protobuf", | 
					
						
							|  |  |  | 			validateParser: requireProtobufParser, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2022-02-08 17: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 23:12:31 +08: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 17: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 23:12:31 +08:00
										 |  |  | 			fallbackProtoMediaType := tt.fallbackScrapeProtocol.HeaderMediaType() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			p, err := New([]byte{}, tt.contentType, fallbackProtoMediaType, false, false, labels.NewSymbolTable()) | 
					
						
							| 
									
										
										
										
											2022-02-08 17:57:56 +08:00
										 |  |  | 			tt.validateParser(t, p) | 
					
						
							| 
									
										
										
										
											2022-02-08 18:01:37 +08:00
										 |  |  | 			if tt.err == "" { | 
					
						
							| 
									
										
										
										
											2022-02-08 17:57:56 +08:00
										 |  |  | 				require.NoError(t, err) | 
					
						
							|  |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2024-10-07 00:35:29 +08:00
										 |  |  | 				require.ErrorContains(t, err, tt.err) | 
					
						
							| 
									
										
										
										
											2022-02-08 17:57:56 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2024-10-07 19:17:44 +08: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 | 
					
						
							| 
									
										
										
										
											2025-03-07 20:43:13 +08:00
										 |  |  | 	ct   int64 | 
					
						
							| 
									
										
										
										
											2024-10-07 19:17:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// 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{ | 
					
						
							| 
									
										
										
										
											2025-02-13 18:38:35 +08:00
										 |  |  | 		// We reuse slices so we sometimes have empty vs nil differences
 | 
					
						
							|  |  |  | 		// we need to ignore with cmpopts.EquateEmpty().
 | 
					
						
							|  |  |  | 		// However we have to filter out labels, as only
 | 
					
						
							|  |  |  | 		// one comparer per type has to be specified,
 | 
					
						
							|  |  |  | 		// and RequireEqualWithOptions uses
 | 
					
						
							|  |  |  | 		// cmp.Comparer(labels.Equal).
 | 
					
						
							|  |  |  | 		cmp.FilterValues(func(x, y any) bool { | 
					
						
							|  |  |  | 			_, xIsLabels := x.(labels.Labels) | 
					
						
							|  |  |  | 			_, yIsLabels := y.(labels.Labels) | 
					
						
							|  |  |  | 			return !xIsLabels && !yIsLabels | 
					
						
							|  |  |  | 		}, cmpopts.EquateEmpty()), | 
					
						
							| 
									
										
										
										
											2024-10-07 19:17:44 +08:00
										 |  |  | 		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: | 
					
						
							| 
									
										
										
										
											2025-02-13 18:38:35 +08:00
										 |  |  | 			var ts *int64 | 
					
						
							| 
									
										
										
										
											2024-10-07 19:17:44 +08:00
										 |  |  | 			if et == EntrySeries { | 
					
						
							| 
									
										
										
										
											2025-02-13 18:38:35 +08:00
										 |  |  | 				m, ts, got.v = p.Series() | 
					
						
							| 
									
										
										
										
											2024-10-07 19:17:44 +08:00
										 |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2025-02-13 18:38:35 +08:00
										 |  |  | 				m, ts, got.shs, got.fhs = p.Histogram() | 
					
						
							| 
									
										
										
										
											2024-10-07 19:17:44 +08:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2025-02-13 18:38:35 +08:00
										 |  |  | 			if ts != nil { | 
					
						
							|  |  |  | 				// TODO(bwplotka): Change to 0 in the interface for set check to
 | 
					
						
							|  |  |  | 				// avoid pointer mangling.
 | 
					
						
							|  |  |  | 				got.t = int64p(*ts) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			got.m = string(m) | 
					
						
							| 
									
										
										
										
											2025-02-12 23:47:56 +08:00
										 |  |  | 			p.Labels(&got.lset) | 
					
						
							| 
									
										
										
										
											2025-03-07 20:43:13 +08:00
										 |  |  | 			got.ct = p.CreatedTimestamp() | 
					
						
							| 
									
										
										
										
											2025-02-13 18:38:35 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-24 13:38:58 +08:00
										 |  |  | 			for e := (exemplar.Exemplar{}); p.Exemplar(&e); { | 
					
						
							|  |  |  | 				got.es = append(got.es, e) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2024-10-07 19:17:44 +08:00
										 |  |  | 		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 | 
					
						
							|  |  |  | } |