mirror of
				https://github.com/prometheus/prometheus.git
				synced 2025-11-04 10:21:02 +01:00 
			
		
		
		
	* feat: initial implement of createedTimestamp() with tests Signed-off-by: Manik Rana <manikrana54@gmail.com> * feat: return ct after finding it Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: remove unneeded test Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: add comments Signed-off-by: Manik Rana <manikrana54@gmail.com> * feat: multiple changes - implement changes from pair programming session - use newParse.val() - advance parser p if ct is found Signed-off-by: Manik Rana <manikrana54@gmail.com> * fix: check if err from p.Next() Signed-off-by: Manik Rana <manikrana54@gmail.com> * feat: advance parser and parse histograms + summary Signed-off-by: Manik Rana <manikrana54@gmail.com> * fix: restore previous tests Signed-off-by: Manik Rana <manikrana54@gmail.com> * fix: retore failing tests Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: remove unneeded comments Signed-off-by: Manik Rana <manikrana54@gmail.com> * fix: return nil when mtype doesn't match Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: update go fmt version Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: cleanup Signed-off-by: Manik Rana <manikrana54@gmail.com> * fix: comments Signed-off-by: Manik Rana <manikrana54@gmail.com> * feat: document deepcopyparser Co-authored-by: Arthur Silva Sens <arthursens2005@gmail.com> Signed-off-by: Manik Rana <Manikrana54@gmail.com> Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: lint Signed-off-by: Manik Rana <manikrana54@gmail.com> * fix: cover edgecase of `gauge_created` in CreatedTimestamp() Signed-off-by: Manik Rana <manikrana54@gmail.com> * refac: readability updates Signed-off-by: Manik Rana <manikrana54@gmail.com> * refac: dedeuplicate labeldiff checks Signed-off-by: Manik Rana <manikrana54@gmail.com> * tests: add tests for new label functions Signed-off-by: Manik Rana <manikrana54@gmail.com> * feat: document CreatedTimestamp func Signed-off-by: Manik Rana <manikrana54@gmail.com> * refac: optimize `CreatedTimestamp()` - Use refactored CreatedTimestamp function with bug fixes - Remove unused code in labels.go - Improve code documentation Signed-off-by: Manik Rana <manikrana54@gmail.com> Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: add tests and lint fixes Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: remove mName Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: lint Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: comments Signed-off-by: Manik Rana <manikrana54@gmail.com> * tests: add tests for CT parse failures and deepCopy Signed-off-by: Manik Rana <manikrana54@gmail.com> * refac: edit expectCT struct Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: lint Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: add new label in deepCopy Signed-off-by: Manik Rana <manikrana54@gmail.com> * fix: use p.builder in deepCopy Signed-off-by: Manik Rana <manikrana54@gmail.com> * fix: add NewMetricsParserWithOpts Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: lint Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: comments Co-authored-by: Arthur Silva Sens <arthursens2005@gmail.com> Signed-off-by: Manik Rana <Manikrana54@gmail.com> * chore: comments Co-authored-by: Arthur Silva Sens <arthursens2005@gmail.com> Signed-off-by: Manik Rana <Manikrana54@gmail.com> * chore: rename var Signed-off-by: Manik Rana <manikrana54@gmail.com> * fix: add condition for OM fuzzing Signed-off-by: Manik Rana <manikrana54@gmail.com> * fix: build tags Signed-off-by: Manik Rana <manikrana54@gmail.com> * refac: default skipCT to false Signed-off-by: Manik Rana <manikrana54@gmail.com> * refac: rename skipCT to skipCTSeries Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: formatting Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: comments and readability updates Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: comments Co-authored-by: Bartlomiej Plotka <bwplotka@gmail.com> Signed-off-by: Manik Rana <Manikrana54@gmail.com> * refac: remove NewOpenMetricsParserWithOpts Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: lint Signed-off-by: Manik Rana <manikrana54@gmail.com> * refac: extract skipCTSeries logic from parseMetricSuffix Signed-off-by: Manik Rana <manikrana54@gmail.com> * refac: inline create a NewOpenMetricsParser Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: comments Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: comments Co-authored-by: Bartlomiej Plotka <bwplotka@gmail.com> Signed-off-by: Manik Rana <Manikrana54@gmail.com> * refac: improve error handling Signed-off-by: Manik Rana <manikrana54@gmail.com> * fix: return error instead of nil Signed-off-by: Manik Rana <manikrana54@gmail.com> * fix: remove skipCT check from tBraceOpen Signed-off-by: Manik Rana <manikrana54@gmail.com> * Pair programming with Manik, Arthur and Daniel. Signed-off-by: bwplotka <bwplotka@gmail.com> * chore: comments and use helper funcs Signed-off-by: Manik Rana <manikrana54@gmail.com> * chore: lint Signed-off-by: Manik Rana <manikrana54@gmail.com> --------- Signed-off-by: Manik Rana <manikrana54@gmail.com> Signed-off-by: Manik Rana <Manikrana54@gmail.com> Signed-off-by: bwplotka <bwplotka@gmail.com> Co-authored-by: Arthur Silva Sens <arthursens2005@gmail.com> Co-authored-by: Bartlomiej Plotka <bwplotka@gmail.com>
		
			
				
	
	
		
			666 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			666 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// Copyright 2018 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.
 | 
						|
 | 
						|
//go:generate go get -u modernc.org/golex
 | 
						|
//go:generate golex -o=openmetricslex.l.go openmetricslex.l
 | 
						|
 | 
						|
package textparse
 | 
						|
 | 
						|
import (
 | 
						|
	"errors"
 | 
						|
	"fmt"
 | 
						|
	"io"
 | 
						|
	"math"
 | 
						|
	"strings"
 | 
						|
	"unicode/utf8"
 | 
						|
 | 
						|
	"github.com/prometheus/common/model"
 | 
						|
 | 
						|
	"github.com/prometheus/prometheus/model/exemplar"
 | 
						|
	"github.com/prometheus/prometheus/model/histogram"
 | 
						|
	"github.com/prometheus/prometheus/model/labels"
 | 
						|
	"github.com/prometheus/prometheus/model/value"
 | 
						|
)
 | 
						|
 | 
						|
type openMetricsLexer struct {
 | 
						|
	b     []byte
 | 
						|
	i     int
 | 
						|
	start int
 | 
						|
	err   error
 | 
						|
	state int
 | 
						|
}
 | 
						|
 | 
						|
// buf returns the buffer of the current token.
 | 
						|
func (l *openMetricsLexer) buf() []byte {
 | 
						|
	return l.b[l.start:l.i]
 | 
						|
}
 | 
						|
 | 
						|
// next advances the openMetricsLexer to the next character.
 | 
						|
func (l *openMetricsLexer) next() byte {
 | 
						|
	l.i++
 | 
						|
	if l.i >= len(l.b) {
 | 
						|
		l.err = io.EOF
 | 
						|
		return byte(tEOF)
 | 
						|
	}
 | 
						|
	// Lex struggles with null bytes. If we are in a label value or help string, where
 | 
						|
	// they are allowed, consume them here immediately.
 | 
						|
	for l.b[l.i] == 0 && (l.state == sLValue || l.state == sMeta2 || l.state == sComment) {
 | 
						|
		l.i++
 | 
						|
		if l.i >= len(l.b) {
 | 
						|
			l.err = io.EOF
 | 
						|
			return byte(tEOF)
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return l.b[l.i]
 | 
						|
}
 | 
						|
 | 
						|
func (l *openMetricsLexer) Error(es string) {
 | 
						|
	l.err = errors.New(es)
 | 
						|
}
 | 
						|
 | 
						|
// OpenMetricsParser parses samples from a byte slice of samples in the official
 | 
						|
// OpenMetrics text exposition format.
 | 
						|
// This is based on the working draft https://docs.google.com/document/u/1/d/1KwV0mAXwwbvvifBvDKH_LU1YjyXE_wxCkHNoCGq1GX0/edit
 | 
						|
type OpenMetricsParser struct {
 | 
						|
	l       *openMetricsLexer
 | 
						|
	builder labels.ScratchBuilder
 | 
						|
	series  []byte
 | 
						|
	text    []byte
 | 
						|
	mtype   model.MetricType
 | 
						|
	val     float64
 | 
						|
	ts      int64
 | 
						|
	hasTS   bool
 | 
						|
	start   int
 | 
						|
	// offsets is a list of offsets into series that describe the positions
 | 
						|
	// of the metric name and label names and values for this series.
 | 
						|
	// p.offsets[0] is the start character of the metric name.
 | 
						|
	// p.offsets[1] is the end of the metric name.
 | 
						|
	// Subsequently, p.offsets is a pair of pair of offsets for the positions
 | 
						|
	// of the label name and value start and end characters.
 | 
						|
	offsets []int
 | 
						|
 | 
						|
	eOffsets      []int
 | 
						|
	exemplar      []byte
 | 
						|
	exemplarVal   float64
 | 
						|
	exemplarTs    int64
 | 
						|
	hasExemplarTs bool
 | 
						|
 | 
						|
	skipCTSeries bool
 | 
						|
}
 | 
						|
 | 
						|
type openMetricsParserOptions struct {
 | 
						|
	SkipCTSeries bool
 | 
						|
}
 | 
						|
 | 
						|
type OpenMetricsOption func(*openMetricsParserOptions)
 | 
						|
 | 
						|
// WithOMParserCTSeriesSkipped turns off exposing _created lines
 | 
						|
// as series, which makes those only used for parsing created timestamp
 | 
						|
// for `CreatedTimestamp` method purposes.
 | 
						|
//
 | 
						|
// It's recommended to use this option to avoid using _created lines for other
 | 
						|
// purposes than created timestamp, but leave false by default for the
 | 
						|
// best-effort compatibility.
 | 
						|
func WithOMParserCTSeriesSkipped() OpenMetricsOption {
 | 
						|
	return func(o *openMetricsParserOptions) {
 | 
						|
		o.SkipCTSeries = true
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// NewOpenMetricsParser returns a new parser for the byte slice with option to skip CT series parsing.
 | 
						|
func NewOpenMetricsParser(b []byte, st *labels.SymbolTable, opts ...OpenMetricsOption) Parser {
 | 
						|
	options := &openMetricsParserOptions{}
 | 
						|
 | 
						|
	for _, opt := range opts {
 | 
						|
		opt(options)
 | 
						|
	}
 | 
						|
 | 
						|
	parser := &OpenMetricsParser{
 | 
						|
		l:            &openMetricsLexer{b: b},
 | 
						|
		builder:      labels.NewScratchBuilderWithSymbolTable(st, 16),
 | 
						|
		skipCTSeries: options.SkipCTSeries,
 | 
						|
	}
 | 
						|
 | 
						|
	return parser
 | 
						|
}
 | 
						|
 | 
						|
// Series returns the bytes of the series, the timestamp if set, and the value
 | 
						|
// of the current sample.
 | 
						|
func (p *OpenMetricsParser) Series() ([]byte, *int64, float64) {
 | 
						|
	if p.hasTS {
 | 
						|
		ts := p.ts
 | 
						|
		return p.series, &ts, p.val
 | 
						|
	}
 | 
						|
	return p.series, nil, p.val
 | 
						|
}
 | 
						|
 | 
						|
// Histogram returns (nil, nil, nil, nil) for now because OpenMetrics does not
 | 
						|
// support sparse histograms yet.
 | 
						|
func (p *OpenMetricsParser) Histogram() ([]byte, *int64, *histogram.Histogram, *histogram.FloatHistogram) {
 | 
						|
	return nil, nil, nil, nil
 | 
						|
}
 | 
						|
 | 
						|
// Help returns the metric name and help text in the current entry.
 | 
						|
// Must only be called after Next returned a help entry.
 | 
						|
// The returned byte slices become invalid after the next call to Next.
 | 
						|
func (p *OpenMetricsParser) Help() ([]byte, []byte) {
 | 
						|
	m := p.l.b[p.offsets[0]:p.offsets[1]]
 | 
						|
 | 
						|
	// Replacer causes allocations. Replace only when necessary.
 | 
						|
	if strings.IndexByte(yoloString(p.text), byte('\\')) >= 0 {
 | 
						|
		// OpenMetrics always uses the Prometheus format label value escaping.
 | 
						|
		return m, []byte(lvalReplacer.Replace(string(p.text)))
 | 
						|
	}
 | 
						|
	return m, p.text
 | 
						|
}
 | 
						|
 | 
						|
// Type returns the metric name and type in the current entry.
 | 
						|
// Must only be called after Next returned a type entry.
 | 
						|
// The returned byte slices become invalid after the next call to Next.
 | 
						|
func (p *OpenMetricsParser) Type() ([]byte, model.MetricType) {
 | 
						|
	return p.l.b[p.offsets[0]:p.offsets[1]], p.mtype
 | 
						|
}
 | 
						|
 | 
						|
// Unit returns the metric name and unit in the current entry.
 | 
						|
// Must only be called after Next returned a unit entry.
 | 
						|
// The returned byte slices become invalid after the next call to Next.
 | 
						|
func (p *OpenMetricsParser) Unit() ([]byte, []byte) {
 | 
						|
	return p.l.b[p.offsets[0]:p.offsets[1]], p.text
 | 
						|
}
 | 
						|
 | 
						|
// Comment returns the text of the current comment.
 | 
						|
// Must only be called after Next returned a comment entry.
 | 
						|
// The returned byte slice becomes invalid after the next call to Next.
 | 
						|
func (p *OpenMetricsParser) Comment() []byte {
 | 
						|
	return p.text
 | 
						|
}
 | 
						|
 | 
						|
// Metric writes the labels of the current sample into the passed labels.
 | 
						|
// It returns the string from which the metric was parsed.
 | 
						|
func (p *OpenMetricsParser) Metric(l *labels.Labels) string {
 | 
						|
	// Copy the buffer to a string: this is only necessary for the return value.
 | 
						|
	s := string(p.series)
 | 
						|
 | 
						|
	p.builder.Reset()
 | 
						|
	metricName := unreplace(s[p.offsets[0]-p.start : p.offsets[1]-p.start])
 | 
						|
	p.builder.Add(labels.MetricName, metricName)
 | 
						|
 | 
						|
	for i := 2; i < len(p.offsets); i += 4 {
 | 
						|
		a := p.offsets[i] - p.start
 | 
						|
		b := p.offsets[i+1] - p.start
 | 
						|
		label := unreplace(s[a:b])
 | 
						|
		c := p.offsets[i+2] - p.start
 | 
						|
		d := p.offsets[i+3] - p.start
 | 
						|
		value := unreplace(s[c:d])
 | 
						|
 | 
						|
		p.builder.Add(label, value)
 | 
						|
	}
 | 
						|
 | 
						|
	p.builder.Sort()
 | 
						|
	*l = p.builder.Labels()
 | 
						|
 | 
						|
	return s
 | 
						|
}
 | 
						|
 | 
						|
// Exemplar writes the exemplar of the current sample into the passed exemplar.
 | 
						|
// It returns whether an exemplar exists. As OpenMetrics only ever has one
 | 
						|
// exemplar per sample, every call after the first (for the same sample) will
 | 
						|
// always return false.
 | 
						|
func (p *OpenMetricsParser) Exemplar(e *exemplar.Exemplar) bool {
 | 
						|
	if len(p.exemplar) == 0 {
 | 
						|
		return false
 | 
						|
	}
 | 
						|
 | 
						|
	// Allocate the full immutable string immediately, so we just
 | 
						|
	// have to create references on it below.
 | 
						|
	s := string(p.exemplar)
 | 
						|
 | 
						|
	e.Value = p.exemplarVal
 | 
						|
	if p.hasExemplarTs {
 | 
						|
		e.HasTs = true
 | 
						|
		e.Ts = p.exemplarTs
 | 
						|
	}
 | 
						|
 | 
						|
	p.builder.Reset()
 | 
						|
	for i := 0; i < len(p.eOffsets); i += 4 {
 | 
						|
		a := p.eOffsets[i] - p.start
 | 
						|
		b := p.eOffsets[i+1] - p.start
 | 
						|
		c := p.eOffsets[i+2] - p.start
 | 
						|
		d := p.eOffsets[i+3] - p.start
 | 
						|
 | 
						|
		p.builder.Add(s[a:b], s[c:d])
 | 
						|
	}
 | 
						|
 | 
						|
	p.builder.Sort()
 | 
						|
	e.Labels = p.builder.Labels()
 | 
						|
 | 
						|
	// Wipe exemplar so that future calls return false.
 | 
						|
	p.exemplar = p.exemplar[:0]
 | 
						|
	return true
 | 
						|
}
 | 
						|
 | 
						|
// CreatedTimestamp returns the created timestamp for a current Metric if exists or nil.
 | 
						|
// NOTE(Maniktherana): Might use additional CPU/mem resources due to deep copy of parser required for peeking given 1.0 OM specification on _created series.
 | 
						|
func (p *OpenMetricsParser) CreatedTimestamp() *int64 {
 | 
						|
	if !TypeRequiresCT(p.mtype) {
 | 
						|
		// Not a CT supported metric type, fast path.
 | 
						|
		return nil
 | 
						|
	}
 | 
						|
 | 
						|
	var (
 | 
						|
		currLset                labels.Labels
 | 
						|
		buf                     []byte
 | 
						|
		peekWithoutNameLsetHash uint64
 | 
						|
	)
 | 
						|
	p.Metric(&currLset)
 | 
						|
	currFamilyLsetHash, buf := currLset.HashWithoutLabels(buf, labels.MetricName, "le", "quantile")
 | 
						|
	// Search for the _created line for the currFamilyLsetHash using ephemeral parser until
 | 
						|
	// we see EOF or new metric family. We have to do it as we don't know where (and if)
 | 
						|
	// that CT line is.
 | 
						|
	// TODO(bwplotka): Make sure OM 1.1/2.0 pass CT via metadata or exemplar-like to avoid this.
 | 
						|
	peek := deepCopy(p)
 | 
						|
	for {
 | 
						|
		eType, err := peek.Next()
 | 
						|
		if err != nil {
 | 
						|
			// This means peek will give error too later on, so def no CT line found.
 | 
						|
			// This might result in partial scrape with wrong/missing CT, but only
 | 
						|
			// spec improvement would help.
 | 
						|
			// TODO(bwplotka): Make sure OM 1.1/2.0 pass CT via metadata or exemplar-like to avoid this.
 | 
						|
			return nil
 | 
						|
		}
 | 
						|
		if eType != EntrySeries {
 | 
						|
			// Assume we hit different family, no CT line found.
 | 
						|
			return nil
 | 
						|
		}
 | 
						|
 | 
						|
		var peekedLset labels.Labels
 | 
						|
		peek.Metric(&peekedLset)
 | 
						|
		peekedName := peekedLset.Get(model.MetricNameLabel)
 | 
						|
		if !strings.HasSuffix(peekedName, "_created") {
 | 
						|
			// Not a CT line, search more.
 | 
						|
			continue
 | 
						|
		}
 | 
						|
 | 
						|
		// We got a CT line here, but let's search if CT line is actually for our series, edge case.
 | 
						|
		peekWithoutNameLsetHash, _ = peekedLset.HashWithoutLabels(buf, labels.MetricName, "le", "quantile")
 | 
						|
		if peekWithoutNameLsetHash != currFamilyLsetHash {
 | 
						|
			// CT line for a different series, for our series no CT.
 | 
						|
			return nil
 | 
						|
		}
 | 
						|
		ct := int64(peek.val)
 | 
						|
		return &ct
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// TypeRequiresCT returns true if the metric type requires a _created timestamp.
 | 
						|
func TypeRequiresCT(t model.MetricType) bool {
 | 
						|
	switch t {
 | 
						|
	case model.MetricTypeCounter, model.MetricTypeSummary, model.MetricTypeHistogram:
 | 
						|
		return true
 | 
						|
	default:
 | 
						|
		return false
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// deepCopy creates a copy of a parser without re-using the slices' original memory addresses.
 | 
						|
func deepCopy(p *OpenMetricsParser) OpenMetricsParser {
 | 
						|
	newB := make([]byte, len(p.l.b))
 | 
						|
	copy(newB, p.l.b)
 | 
						|
 | 
						|
	newLexer := &openMetricsLexer{
 | 
						|
		b:     newB,
 | 
						|
		i:     p.l.i,
 | 
						|
		start: p.l.start,
 | 
						|
		err:   p.l.err,
 | 
						|
		state: p.l.state,
 | 
						|
	}
 | 
						|
 | 
						|
	newParser := OpenMetricsParser{
 | 
						|
		l:            newLexer,
 | 
						|
		builder:      p.builder,
 | 
						|
		mtype:        p.mtype,
 | 
						|
		val:          p.val,
 | 
						|
		skipCTSeries: false,
 | 
						|
	}
 | 
						|
	return newParser
 | 
						|
}
 | 
						|
 | 
						|
// nextToken returns the next token from the openMetricsLexer.
 | 
						|
func (p *OpenMetricsParser) nextToken() token {
 | 
						|
	tok := p.l.Lex()
 | 
						|
	return tok
 | 
						|
}
 | 
						|
 | 
						|
func (p *OpenMetricsParser) parseError(exp string, got token) error {
 | 
						|
	e := p.l.i + 1
 | 
						|
	if len(p.l.b) < e {
 | 
						|
		e = len(p.l.b)
 | 
						|
	}
 | 
						|
	return fmt.Errorf("%s, got %q (%q) while parsing: %q", exp, p.l.b[p.l.start:e], got, p.l.b[p.start:e])
 | 
						|
}
 | 
						|
 | 
						|
// Next advances the parser to the next sample.
 | 
						|
// It returns (EntryInvalid, io.EOF) if no samples were read.
 | 
						|
func (p *OpenMetricsParser) Next() (Entry, error) {
 | 
						|
	var err error
 | 
						|
 | 
						|
	p.start = p.l.i
 | 
						|
	p.offsets = p.offsets[:0]
 | 
						|
	p.eOffsets = p.eOffsets[:0]
 | 
						|
	p.exemplar = p.exemplar[:0]
 | 
						|
	p.exemplarVal = 0
 | 
						|
	p.hasExemplarTs = false
 | 
						|
 | 
						|
	switch t := p.nextToken(); t {
 | 
						|
	case tEOFWord:
 | 
						|
		if t := p.nextToken(); t != tEOF {
 | 
						|
			return EntryInvalid, errors.New("unexpected data after # EOF")
 | 
						|
		}
 | 
						|
		return EntryInvalid, io.EOF
 | 
						|
	case tEOF:
 | 
						|
		return EntryInvalid, errors.New("data does not end with # EOF")
 | 
						|
	case tHelp, tType, tUnit:
 | 
						|
		switch t2 := p.nextToken(); t2 {
 | 
						|
		case tMName:
 | 
						|
			mStart := p.l.start
 | 
						|
			mEnd := p.l.i
 | 
						|
			if p.l.b[mStart] == '"' && p.l.b[mEnd-1] == '"' {
 | 
						|
				mStart++
 | 
						|
				mEnd--
 | 
						|
			}
 | 
						|
			p.offsets = append(p.offsets, mStart, mEnd)
 | 
						|
		default:
 | 
						|
			return EntryInvalid, p.parseError("expected metric name after "+t.String(), t2)
 | 
						|
		}
 | 
						|
		switch t2 := p.nextToken(); t2 {
 | 
						|
		case tText:
 | 
						|
			if len(p.l.buf()) > 1 {
 | 
						|
				p.text = p.l.buf()[1 : len(p.l.buf())-1]
 | 
						|
			} else {
 | 
						|
				p.text = []byte{}
 | 
						|
			}
 | 
						|
		default:
 | 
						|
			return EntryInvalid, fmt.Errorf("expected text in %s", t.String())
 | 
						|
		}
 | 
						|
		switch t {
 | 
						|
		case tType:
 | 
						|
			switch s := yoloString(p.text); s {
 | 
						|
			case "counter":
 | 
						|
				p.mtype = model.MetricTypeCounter
 | 
						|
			case "gauge":
 | 
						|
				p.mtype = model.MetricTypeGauge
 | 
						|
			case "histogram":
 | 
						|
				p.mtype = model.MetricTypeHistogram
 | 
						|
			case "gaugehistogram":
 | 
						|
				p.mtype = model.MetricTypeGaugeHistogram
 | 
						|
			case "summary":
 | 
						|
				p.mtype = model.MetricTypeSummary
 | 
						|
			case "info":
 | 
						|
				p.mtype = model.MetricTypeInfo
 | 
						|
			case "stateset":
 | 
						|
				p.mtype = model.MetricTypeStateset
 | 
						|
			case "unknown":
 | 
						|
				p.mtype = model.MetricTypeUnknown
 | 
						|
			default:
 | 
						|
				return EntryInvalid, fmt.Errorf("invalid metric type %q", s)
 | 
						|
			}
 | 
						|
		case tHelp:
 | 
						|
			if !utf8.Valid(p.text) {
 | 
						|
				return EntryInvalid, fmt.Errorf("help text %q is not a valid utf8 string", p.text)
 | 
						|
			}
 | 
						|
		}
 | 
						|
		switch t {
 | 
						|
		case tHelp:
 | 
						|
			return EntryHelp, nil
 | 
						|
		case tType:
 | 
						|
			return EntryType, nil
 | 
						|
		case tUnit:
 | 
						|
			m := yoloString(p.l.b[p.offsets[0]:p.offsets[1]])
 | 
						|
			u := yoloString(p.text)
 | 
						|
			if len(u) > 0 {
 | 
						|
				if !strings.HasSuffix(m, u) || len(m) < len(u)+1 || p.l.b[p.offsets[1]-len(u)-1] != '_' {
 | 
						|
					return EntryInvalid, fmt.Errorf("unit %q not a suffix of metric %q", u, m)
 | 
						|
				}
 | 
						|
			}
 | 
						|
			return EntryUnit, nil
 | 
						|
		}
 | 
						|
 | 
						|
	case tBraceOpen:
 | 
						|
		// We found a brace, so make room for the eventual metric name. If these
 | 
						|
		// values aren't updated, then the metric name was not set inside the
 | 
						|
		// braces and we can return an error.
 | 
						|
		if len(p.offsets) == 0 {
 | 
						|
			p.offsets = []int{-1, -1}
 | 
						|
		}
 | 
						|
		if p.offsets, err = p.parseLVals(p.offsets, false); err != nil {
 | 
						|
			return EntryInvalid, err
 | 
						|
		}
 | 
						|
 | 
						|
		p.series = p.l.b[p.start:p.l.i]
 | 
						|
		if err := p.parseSeriesEndOfLine(p.nextToken()); err != nil {
 | 
						|
			return EntryInvalid, err
 | 
						|
		}
 | 
						|
		if p.skipCTSeries && p.isCreatedSeries() {
 | 
						|
			return p.Next()
 | 
						|
		}
 | 
						|
		return EntrySeries, nil
 | 
						|
	case tMName:
 | 
						|
		p.offsets = append(p.offsets, p.start, p.l.i)
 | 
						|
		p.series = p.l.b[p.start:p.l.i]
 | 
						|
 | 
						|
		t2 := p.nextToken()
 | 
						|
		if t2 == tBraceOpen {
 | 
						|
			p.offsets, err = p.parseLVals(p.offsets, false)
 | 
						|
			if err != nil {
 | 
						|
				return EntryInvalid, err
 | 
						|
			}
 | 
						|
			p.series = p.l.b[p.start:p.l.i]
 | 
						|
			t2 = p.nextToken()
 | 
						|
		}
 | 
						|
 | 
						|
		if err := p.parseSeriesEndOfLine(t2); err != nil {
 | 
						|
			return EntryInvalid, err
 | 
						|
		}
 | 
						|
		if p.skipCTSeries && p.isCreatedSeries() {
 | 
						|
			return p.Next()
 | 
						|
		}
 | 
						|
		return EntrySeries, nil
 | 
						|
	default:
 | 
						|
		err = p.parseError("expected a valid start token", t)
 | 
						|
	}
 | 
						|
	return EntryInvalid, err
 | 
						|
}
 | 
						|
 | 
						|
func (p *OpenMetricsParser) parseComment() error {
 | 
						|
	var err error
 | 
						|
	// Parse the labels.
 | 
						|
	p.eOffsets, err = p.parseLVals(p.eOffsets, true)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	p.exemplar = p.l.b[p.start:p.l.i]
 | 
						|
 | 
						|
	// Get the value.
 | 
						|
	p.exemplarVal, err = p.getFloatValue(p.nextToken(), "exemplar labels")
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	// Read the optional timestamp.
 | 
						|
	p.hasExemplarTs = false
 | 
						|
	switch t2 := p.nextToken(); t2 {
 | 
						|
	case tEOF:
 | 
						|
		return errors.New("data does not end with # EOF")
 | 
						|
	case tLinebreak:
 | 
						|
		break
 | 
						|
	case tTimestamp:
 | 
						|
		p.hasExemplarTs = true
 | 
						|
		var ts float64
 | 
						|
		// A float is enough to hold what we need for millisecond resolution.
 | 
						|
		if ts, err = parseFloat(yoloString(p.l.buf()[1:])); err != nil {
 | 
						|
			return fmt.Errorf("%w while parsing: %q", err, p.l.b[p.start:p.l.i])
 | 
						|
		}
 | 
						|
		if math.IsNaN(ts) || math.IsInf(ts, 0) {
 | 
						|
			return fmt.Errorf("invalid exemplar timestamp %f", ts)
 | 
						|
		}
 | 
						|
		p.exemplarTs = int64(ts * 1000)
 | 
						|
		switch t3 := p.nextToken(); t3 {
 | 
						|
		case tLinebreak:
 | 
						|
		default:
 | 
						|
			return p.parseError("expected next entry after exemplar timestamp", t3)
 | 
						|
		}
 | 
						|
	default:
 | 
						|
		return p.parseError("expected timestamp or comment", t2)
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (p *OpenMetricsParser) parseLVals(offsets []int, isExemplar bool) ([]int, error) {
 | 
						|
	t := p.nextToken()
 | 
						|
	for {
 | 
						|
		curTStart := p.l.start
 | 
						|
		curTI := p.l.i
 | 
						|
		switch t {
 | 
						|
		case tBraceClose:
 | 
						|
			return offsets, nil
 | 
						|
		case tLName:
 | 
						|
		case tQString:
 | 
						|
		default:
 | 
						|
			return nil, p.parseError("expected label name", t)
 | 
						|
		}
 | 
						|
 | 
						|
		t = p.nextToken()
 | 
						|
		// A quoted string followed by a comma or brace is a metric name. Set the
 | 
						|
		// offsets and continue processing. If this is an exemplar, this format
 | 
						|
		// is not allowed.
 | 
						|
		if t == tComma || t == tBraceClose {
 | 
						|
			if isExemplar {
 | 
						|
				return nil, p.parseError("expected label name", t)
 | 
						|
			}
 | 
						|
			if offsets[0] != -1 || offsets[1] != -1 {
 | 
						|
				return nil, fmt.Errorf("metric name already set while parsing: %q", p.l.b[p.start:p.l.i])
 | 
						|
			}
 | 
						|
			offsets[0] = curTStart + 1
 | 
						|
			offsets[1] = curTI - 1
 | 
						|
			if t == tBraceClose {
 | 
						|
				return offsets, nil
 | 
						|
			}
 | 
						|
			t = p.nextToken()
 | 
						|
			continue
 | 
						|
		}
 | 
						|
		// We have a label name, and it might be quoted.
 | 
						|
		if p.l.b[curTStart] == '"' {
 | 
						|
			curTStart++
 | 
						|
			curTI--
 | 
						|
		}
 | 
						|
		offsets = append(offsets, curTStart, curTI)
 | 
						|
 | 
						|
		if t != tEqual {
 | 
						|
			return nil, p.parseError("expected equal", t)
 | 
						|
		}
 | 
						|
		if t := p.nextToken(); t != tLValue {
 | 
						|
			return nil, p.parseError("expected label value", t)
 | 
						|
		}
 | 
						|
		if !utf8.Valid(p.l.buf()) {
 | 
						|
			return nil, fmt.Errorf("invalid UTF-8 label value: %q", p.l.buf())
 | 
						|
		}
 | 
						|
 | 
						|
		// The openMetricsLexer ensures the value string is quoted. Strip first
 | 
						|
		// and last character.
 | 
						|
		offsets = append(offsets, p.l.start+1, p.l.i-1)
 | 
						|
 | 
						|
		// Free trailing commas are allowed.
 | 
						|
		t = p.nextToken()
 | 
						|
		if t == tComma {
 | 
						|
			t = p.nextToken()
 | 
						|
		} else if t != tBraceClose {
 | 
						|
			return nil, p.parseError("expected comma or brace close", t)
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// isCreatedSeries returns true if the current series is a _created series.
 | 
						|
func (p *OpenMetricsParser) isCreatedSeries() bool {
 | 
						|
	var newLbs labels.Labels
 | 
						|
	p.Metric(&newLbs)
 | 
						|
	name := newLbs.Get(model.MetricNameLabel)
 | 
						|
	if TypeRequiresCT(p.mtype) && strings.HasSuffix(name, "_created") {
 | 
						|
		return true
 | 
						|
	}
 | 
						|
	return false
 | 
						|
}
 | 
						|
 | 
						|
// parseSeriesEndOfLine parses the series end of the line (value, optional
 | 
						|
// timestamp, commentary, etc.) after the metric name and labels.
 | 
						|
// It starts parsing with the provided token.
 | 
						|
func (p *OpenMetricsParser) parseSeriesEndOfLine(t token) error {
 | 
						|
	if p.offsets[0] == -1 {
 | 
						|
		return fmt.Errorf("metric name not set while parsing: %q", p.l.b[p.start:p.l.i])
 | 
						|
	}
 | 
						|
 | 
						|
	var err error
 | 
						|
	p.val, err = p.getFloatValue(t, "metric")
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	p.hasTS = false
 | 
						|
	switch t2 := p.nextToken(); t2 {
 | 
						|
	case tEOF:
 | 
						|
		return errors.New("data does not end with # EOF")
 | 
						|
	case tLinebreak:
 | 
						|
		break
 | 
						|
	case tComment:
 | 
						|
		if err := p.parseComment(); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
	case tTimestamp:
 | 
						|
		p.hasTS = true
 | 
						|
		var ts float64
 | 
						|
		// A float is enough to hold what we need for millisecond resolution.
 | 
						|
		if ts, err = parseFloat(yoloString(p.l.buf()[1:])); err != nil {
 | 
						|
			return fmt.Errorf("%w while parsing: %q", err, p.l.b[p.start:p.l.i])
 | 
						|
		}
 | 
						|
		if math.IsNaN(ts) || math.IsInf(ts, 0) {
 | 
						|
			return fmt.Errorf("invalid timestamp %f", ts)
 | 
						|
		}
 | 
						|
		p.ts = int64(ts * 1000)
 | 
						|
		switch t3 := p.nextToken(); t3 {
 | 
						|
		case tLinebreak:
 | 
						|
		case tComment:
 | 
						|
			if err := p.parseComment(); err != nil {
 | 
						|
				return err
 | 
						|
			}
 | 
						|
		default:
 | 
						|
			return p.parseError("expected next entry after timestamp", t3)
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (p *OpenMetricsParser) getFloatValue(t token, after string) (float64, error) {
 | 
						|
	if t != tValue {
 | 
						|
		return 0, p.parseError(fmt.Sprintf("expected value after %v", after), t)
 | 
						|
	}
 | 
						|
	val, err := parseFloat(yoloString(p.l.buf()[1:]))
 | 
						|
	if err != nil {
 | 
						|
		return 0, fmt.Errorf("%w while parsing: %q", err, p.l.b[p.start:p.l.i])
 | 
						|
	}
 | 
						|
	// Ensure canonical NaN value.
 | 
						|
	if math.IsNaN(p.exemplarVal) {
 | 
						|
		val = math.Float64frombits(value.NormalNaN)
 | 
						|
	}
 | 
						|
	return val, nil
 | 
						|
}
 |