diff options
| author | 2024-05-21 13:17:22 +0000 | |
|---|---|---|
| committer | 2024-05-21 14:17:22 +0100 | |
| commit | 6c0d93c6cb082992fbc3c9bcbd85a99807563519 (patch) | |
| tree | a9e86538cd911451bc927c304e1e20a68de00071 /vendor/github.com/prometheus/common/expfmt | |
| parent | [chore]: Bump github.com/gin-contrib/cors from 1.7.1 to 1.7.2 (#2912) (diff) | |
| download | gotosocial-6c0d93c6cb082992fbc3c9bcbd85a99807563519.tar.xz | |
[chore] dependabot updates (#2922)
* [chore]: Bump github.com/prometheus/client_golang from 1.18.0 to 1.19.1
Bumps [github.com/prometheus/client_golang](https://github.com/prometheus/client_golang) from 1.18.0 to 1.19.1.
- [Release notes](https://github.com/prometheus/client_golang/releases)
- [Changelog](https://github.com/prometheus/client_golang/blob/main/CHANGELOG.md)
- [Commits](https://github.com/prometheus/client_golang/compare/v1.18.0...v1.19.1)
---
updated-dependencies:
- dependency-name: github.com/prometheus/client_golang
  dependency-type: direct:production
  update-type: version-update:semver-minor
...
Signed-off-by: dependabot[bot] <support@github.com>
* [chore]: Bump github.com/KimMachineGun/automemlimit from 0.6.0 to 0.6.1
Bumps [github.com/KimMachineGun/automemlimit](https://github.com/KimMachineGun/automemlimit) from 0.6.0 to 0.6.1.
- [Release notes](https://github.com/KimMachineGun/automemlimit/releases)
- [Commits](https://github.com/KimMachineGun/automemlimit/compare/v0.6.0...v0.6.1)
---
updated-dependencies:
- dependency-name: github.com/KimMachineGun/automemlimit
  dependency-type: direct:production
  update-type: version-update:semver-patch
...
Signed-off-by: dependabot[bot] <support@github.com>
* [chore]: Bump github.com/tdewolff/minify/v2 from 2.20.20 to 2.20.24
Bumps [github.com/tdewolff/minify/v2](https://github.com/tdewolff/minify) from 2.20.20 to 2.20.24.
- [Release notes](https://github.com/tdewolff/minify/releases)
- [Commits](https://github.com/tdewolff/minify/compare/v2.20.20...v2.20.24)
---
updated-dependencies:
- dependency-name: github.com/tdewolff/minify/v2
  dependency-type: direct:production
  update-type: version-update:semver-patch
...
Signed-off-by: dependabot[bot] <support@github.com>
* [chore]: Bump github.com/go-swagger/go-swagger
Bumps [github.com/go-swagger/go-swagger](https://github.com/go-swagger/go-swagger) from 0.30.6-0.20240418033037-c46c303aaa02 to 0.31.0.
- [Release notes](https://github.com/go-swagger/go-swagger/releases)
- [Changelog](https://github.com/go-swagger/go-swagger/blob/master/.goreleaser.yml)
- [Commits](https://github.com/go-swagger/go-swagger/commits/v0.31.0)
---
updated-dependencies:
- dependency-name: github.com/go-swagger/go-swagger
  dependency-type: direct:production
  update-type: version-update:semver-minor
...
Signed-off-by: dependabot[bot] <support@github.com>
* [chore]: Bump github.com/gin-gonic/gin from 1.9.1 to 1.10.0
Bumps [github.com/gin-gonic/gin](https://github.com/gin-gonic/gin) from 1.9.1 to 1.10.0.
- [Release notes](https://github.com/gin-gonic/gin/releases)
- [Changelog](https://github.com/gin-gonic/gin/blob/master/CHANGELOG.md)
- [Commits](https://github.com/gin-gonic/gin/compare/v1.9.1...v1.10.0)
---
updated-dependencies:
- dependency-name: github.com/gin-gonic/gin
  dependency-type: direct:production
  update-type: version-update:semver-minor
...
Signed-off-by: dependabot[bot] <support@github.com>
---------
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
Diffstat (limited to 'vendor/github.com/prometheus/common/expfmt')
6 files changed, 351 insertions, 109 deletions
| diff --git a/vendor/github.com/prometheus/common/expfmt/decode.go b/vendor/github.com/prometheus/common/expfmt/decode.go index 0ca86a3dc..b2b89b017 100644 --- a/vendor/github.com/prometheus/common/expfmt/decode.go +++ b/vendor/github.com/prometheus/common/expfmt/decode.go @@ -14,6 +14,7 @@  package expfmt  import ( +	"bufio"  	"fmt"  	"io"  	"math" @@ -21,8 +22,8 @@ import (  	"net/http"  	dto "github.com/prometheus/client_model/go" +	"google.golang.org/protobuf/encoding/protodelim" -	"github.com/matttproud/golang_protobuf_extensions/v2/pbutil"  	"github.com/prometheus/common/model"  ) @@ -44,7 +45,7 @@ func ResponseFormat(h http.Header) Format {  	mediatype, params, err := mime.ParseMediaType(ct)  	if err != nil { -		return FmtUnknown +		return fmtUnknown  	}  	const textType = "text/plain" @@ -52,28 +53,28 @@ func ResponseFormat(h http.Header) Format {  	switch mediatype {  	case ProtoType:  		if p, ok := params["proto"]; ok && p != ProtoProtocol { -			return FmtUnknown +			return fmtUnknown  		}  		if e, ok := params["encoding"]; ok && e != "delimited" { -			return FmtUnknown +			return fmtUnknown  		} -		return FmtProtoDelim +		return fmtProtoDelim  	case textType:  		if v, ok := params["version"]; ok && v != TextVersion { -			return FmtUnknown +			return fmtUnknown  		} -		return FmtText +		return fmtText  	} -	return FmtUnknown +	return fmtUnknown  }  // NewDecoder returns a new decoder based on the given input format.  // If the input format does not imply otherwise, a text format decoder is returned.  func NewDecoder(r io.Reader, format Format) Decoder { -	switch format { -	case FmtProtoDelim: +	switch format.FormatType() { +	case TypeProtoDelim:  		return &protoDecoder{r: r}  	}  	return &textDecoder{r: r} @@ -86,8 +87,10 @@ type protoDecoder struct {  // Decode implements the Decoder interface.  func (d *protoDecoder) Decode(v *dto.MetricFamily) error { -	_, err := pbutil.ReadDelimited(d.r, v) -	if err != nil { +	opts := protodelim.UnmarshalOptions{ +		MaxSize: -1, +	} +	if err := opts.UnmarshalFrom(bufio.NewReader(d.r), v); err != nil {  		return err  	}  	if !model.IsValidMetricName(model.LabelValue(v.GetName())) { diff --git a/vendor/github.com/prometheus/common/expfmt/encode.go b/vendor/github.com/prometheus/common/expfmt/encode.go index ca2140600..8fd806184 100644 --- a/vendor/github.com/prometheus/common/expfmt/encode.go +++ b/vendor/github.com/prometheus/common/expfmt/encode.go @@ -18,10 +18,12 @@ import (  	"io"  	"net/http" -	"github.com/matttproud/golang_protobuf_extensions/v2/pbutil" -	"github.com/prometheus/common/internal/bitbucket.org/ww/goautoneg" +	"google.golang.org/protobuf/encoding/protodelim"  	"google.golang.org/protobuf/encoding/prototext" +	"github.com/prometheus/common/internal/bitbucket.org/ww/goautoneg" +	"github.com/prometheus/common/model" +  	dto "github.com/prometheus/client_model/go"  ) @@ -60,23 +62,32 @@ func (ec encoderCloser) Close() error {  // as the support is still experimental. To include the option to negotiate  // FmtOpenMetrics, use NegotiateOpenMetrics.  func Negotiate(h http.Header) Format { +	escapingScheme := Format(fmt.Sprintf("; escaping=%s", Format(model.NameEscapingScheme.String())))  	for _, ac := range goautoneg.ParseAccept(h.Get(hdrAccept)) { +		if escapeParam := ac.Params[model.EscapingKey]; escapeParam != "" { +			switch Format(escapeParam) { +			case model.AllowUTF8, model.EscapeUnderscores, model.EscapeDots, model.EscapeValues: +				escapingScheme = Format(fmt.Sprintf("; escaping=%s", escapeParam)) +			default: +				// If the escaping parameter is unknown, ignore it. +			} +		}  		ver := ac.Params["version"]  		if ac.Type+"/"+ac.SubType == ProtoType && ac.Params["proto"] == ProtoProtocol {  			switch ac.Params["encoding"] {  			case "delimited": -				return FmtProtoDelim +				return fmtProtoDelim + escapingScheme  			case "text": -				return FmtProtoText +				return fmtProtoText + escapingScheme  			case "compact-text": -				return FmtProtoCompact +				return fmtProtoCompact + escapingScheme  			}  		}  		if ac.Type == "text" && ac.SubType == "plain" && (ver == TextVersion || ver == "") { -			return FmtText +			return fmtText + escapingScheme  		}  	} -	return FmtText +	return fmtText + escapingScheme  }  // NegotiateIncludingOpenMetrics works like Negotiate but includes @@ -84,29 +95,40 @@ func Negotiate(h http.Header) Format {  // temporary and will disappear once FmtOpenMetrics is fully supported and as  // such may be negotiated by the normal Negotiate function.  func NegotiateIncludingOpenMetrics(h http.Header) Format { +	escapingScheme := Format(fmt.Sprintf("; escaping=%s", Format(model.NameEscapingScheme.String())))  	for _, ac := range goautoneg.ParseAccept(h.Get(hdrAccept)) { +		if escapeParam := ac.Params[model.EscapingKey]; escapeParam != "" { +			switch Format(escapeParam) { +			case model.AllowUTF8, model.EscapeUnderscores, model.EscapeDots, model.EscapeValues: +				escapingScheme = Format(fmt.Sprintf("; escaping=%s", escapeParam)) +			default: +				// If the escaping parameter is unknown, ignore it. +			} +		}  		ver := ac.Params["version"]  		if ac.Type+"/"+ac.SubType == ProtoType && ac.Params["proto"] == ProtoProtocol {  			switch ac.Params["encoding"] {  			case "delimited": -				return FmtProtoDelim +				return fmtProtoDelim + escapingScheme  			case "text": -				return FmtProtoText +				return fmtProtoText + escapingScheme  			case "compact-text": -				return FmtProtoCompact +				return fmtProtoCompact + escapingScheme  			}  		}  		if ac.Type == "text" && ac.SubType == "plain" && (ver == TextVersion || ver == "") { -			return FmtText +			return fmtText + escapingScheme  		}  		if ac.Type+"/"+ac.SubType == OpenMetricsType && (ver == OpenMetricsVersion_0_0_1 || ver == OpenMetricsVersion_1_0_0 || ver == "") { -			if ver == OpenMetricsVersion_1_0_0 { -				return FmtOpenMetrics_1_0_0 +			switch ver { +			case OpenMetricsVersion_1_0_0: +				return fmtOpenMetrics_1_0_0 + escapingScheme +			default: +				return fmtOpenMetrics_0_0_1 + escapingScheme  			} -			return FmtOpenMetrics_0_0_1  		}  	} -	return FmtText +	return fmtText + escapingScheme  }  // NewEncoder returns a new encoder based on content type negotiation. All @@ -115,44 +137,48 @@ func NegotiateIncludingOpenMetrics(h http.Header) Format {  // for FmtOpenMetrics, but a future (breaking) release will add the Close method  // to the Encoder interface directly. The current version of the Encoder  // interface is kept for backwards compatibility. +// In cases where the Format does not allow for UTF-8 names, the global +// NameEscapingScheme will be applied.  func NewEncoder(w io.Writer, format Format) Encoder { -	switch format { -	case FmtProtoDelim: +	escapingScheme := format.ToEscapingScheme() + +	switch format.FormatType() { +	case TypeProtoDelim:  		return encoderCloser{  			encode: func(v *dto.MetricFamily) error { -				_, err := pbutil.WriteDelimited(w, v) +				_, err := protodelim.MarshalTo(w, v)  				return err  			},  			close: func() error { return nil },  		} -	case FmtProtoCompact: +	case TypeProtoCompact:  		return encoderCloser{  			encode: func(v *dto.MetricFamily) error { -				_, err := fmt.Fprintln(w, v.String()) +				_, err := fmt.Fprintln(w, model.EscapeMetricFamily(v, escapingScheme).String())  				return err  			},  			close: func() error { return nil },  		} -	case FmtProtoText: +	case TypeProtoText:  		return encoderCloser{  			encode: func(v *dto.MetricFamily) error { -				_, err := fmt.Fprintln(w, prototext.Format(v)) +				_, err := fmt.Fprintln(w, prototext.Format(model.EscapeMetricFamily(v, escapingScheme)))  				return err  			},  			close: func() error { return nil },  		} -	case FmtText: +	case TypeTextPlain:  		return encoderCloser{  			encode: func(v *dto.MetricFamily) error { -				_, err := MetricFamilyToText(w, v) +				_, err := MetricFamilyToText(w, model.EscapeMetricFamily(v, escapingScheme))  				return err  			},  			close: func() error { return nil },  		} -	case FmtOpenMetrics_0_0_1, FmtOpenMetrics_1_0_0: +	case TypeOpenMetrics:  		return encoderCloser{  			encode: func(v *dto.MetricFamily) error { -				_, err := MetricFamilyToOpenMetrics(w, v) +				_, err := MetricFamilyToOpenMetrics(w, model.EscapeMetricFamily(v, escapingScheme))  				return err  			},  			close: func() error { diff --git a/vendor/github.com/prometheus/common/expfmt/expfmt.go b/vendor/github.com/prometheus/common/expfmt/expfmt.go index c4cb20f0d..6fc9555e3 100644 --- a/vendor/github.com/prometheus/common/expfmt/expfmt.go +++ b/vendor/github.com/prometheus/common/expfmt/expfmt.go @@ -14,30 +14,154 @@  // Package expfmt contains tools for reading and writing Prometheus metrics.  package expfmt +import ( +	"strings" + +	"github.com/prometheus/common/model" +) +  // Format specifies the HTTP content type of the different wire protocols.  type Format string -// Constants to assemble the Content-Type values for the different wire protocols. +// Constants to assemble the Content-Type values for the different wire +// protocols. The Content-Type strings here are all for the legacy exposition +// formats, where valid characters for metric names and label names are limited. +// Support for arbitrary UTF-8 characters in those names is already partially +// implemented in this module (see model.ValidationScheme), but to actually use +// it on the wire, new content-type strings will have to be agreed upon and +// added here.  const (  	TextVersion              = "0.0.4"  	ProtoType                = `application/vnd.google.protobuf`  	ProtoProtocol            = `io.prometheus.client.MetricFamily` -	ProtoFmt                 = ProtoType + "; proto=" + ProtoProtocol + ";" +	protoFmt                 = ProtoType + "; proto=" + ProtoProtocol + ";"  	OpenMetricsType          = `application/openmetrics-text`  	OpenMetricsVersion_0_0_1 = "0.0.1"  	OpenMetricsVersion_1_0_0 = "1.0.0" -	// The Content-Type values for the different wire protocols. -	FmtUnknown           Format = `<unknown>` -	FmtText              Format = `text/plain; version=` + TextVersion + `; charset=utf-8` -	FmtProtoDelim        Format = ProtoFmt + ` encoding=delimited` -	FmtProtoText         Format = ProtoFmt + ` encoding=text` -	FmtProtoCompact      Format = ProtoFmt + ` encoding=compact-text` -	FmtOpenMetrics_1_0_0 Format = OpenMetricsType + `; version=` + OpenMetricsVersion_1_0_0 + `; charset=utf-8` -	FmtOpenMetrics_0_0_1 Format = OpenMetricsType + `; version=` + OpenMetricsVersion_0_0_1 + `; charset=utf-8` +	// The Content-Type values for the different wire protocols. Note that these +	// values are now unexported. If code was relying on comparisons to these +	// constants, instead use FormatType(). +	fmtUnknown           Format = `<unknown>` +	fmtText              Format = `text/plain; version=` + TextVersion + `; charset=utf-8` +	fmtProtoDelim        Format = protoFmt + ` encoding=delimited` +	fmtProtoText         Format = protoFmt + ` encoding=text` +	fmtProtoCompact      Format = protoFmt + ` encoding=compact-text` +	fmtOpenMetrics_1_0_0 Format = OpenMetricsType + `; version=` + OpenMetricsVersion_1_0_0 + `; charset=utf-8` +	fmtOpenMetrics_0_0_1 Format = OpenMetricsType + `; version=` + OpenMetricsVersion_0_0_1 + `; charset=utf-8`  )  const (  	hdrContentType = "Content-Type"  	hdrAccept      = "Accept"  ) + +// FormatType is a Go enum representing the overall category for the given +// Format. As the number of Format permutations increases, doing basic string +// comparisons are not feasible, so this enum captures the most useful +// high-level attribute of the Format string. +type FormatType int + +const ( +	TypeUnknown = iota +	TypeProtoCompact +	TypeProtoDelim +	TypeProtoText +	TypeTextPlain +	TypeOpenMetrics +) + +// NewFormat generates a new Format from the type provided. Mostly used for +// tests, most Formats should be generated as part of content negotiation in +// encode.go. +func NewFormat(t FormatType) Format { +	switch t { +	case TypeProtoCompact: +		return fmtProtoCompact +	case TypeProtoDelim: +		return fmtProtoDelim +	case TypeProtoText: +		return fmtProtoText +	case TypeTextPlain: +		return fmtText +	case TypeOpenMetrics: +		return fmtOpenMetrics_1_0_0 +	default: +		return fmtUnknown +	} +} + +// FormatType deduces an overall FormatType for the given format. +func (f Format) FormatType() FormatType { +	toks := strings.Split(string(f), ";") +	if len(toks) < 2 { +		return TypeUnknown +	} + +	params := make(map[string]string) +	for i, t := range toks { +		if i == 0 { +			continue +		} +		args := strings.Split(t, "=") +		if len(args) != 2 { +			continue +		} +		params[strings.TrimSpace(args[0])] = strings.TrimSpace(args[1]) +	} + +	switch strings.TrimSpace(toks[0]) { +	case ProtoType: +		if params["proto"] != ProtoProtocol { +			return TypeUnknown +		} +		switch params["encoding"] { +		case "delimited": +			return TypeProtoDelim +		case "text": +			return TypeProtoText +		case "compact-text": +			return TypeProtoCompact +		default: +			return TypeUnknown +		} +	case OpenMetricsType: +		if params["charset"] != "utf-8" { +			return TypeUnknown +		} +		return TypeOpenMetrics +	case "text/plain": +		v, ok := params["version"] +		if !ok { +			return TypeTextPlain +		} +		if v == TextVersion { +			return TypeTextPlain +		} +		return TypeUnknown +	default: +		return TypeUnknown +	} +} + +// ToEscapingScheme returns an EscapingScheme depending on the Format. Iff the +// Format contains a escaping=allow-utf-8 term, it will select NoEscaping. If a valid +// "escaping" term exists, that will be used. Otherwise, the global default will +// be returned. +func (format Format) ToEscapingScheme() model.EscapingScheme { +	for _, p := range strings.Split(string(format), ";") { +		toks := strings.Split(p, "=") +		if len(toks) != 2 { +			continue +		} +		key, value := strings.TrimSpace(toks[0]), strings.TrimSpace(toks[1]) +		if key == model.EscapingKey { +			scheme, err := model.ToEscapingScheme(value) +			if err != nil { +				return model.NameEscapingScheme +			} +			return scheme +		} +	} +	return model.NameEscapingScheme +} diff --git a/vendor/github.com/prometheus/common/expfmt/openmetrics_create.go b/vendor/github.com/prometheus/common/expfmt/openmetrics_create.go index 21cdddcf0..5622578ed 100644 --- a/vendor/github.com/prometheus/common/expfmt/openmetrics_create.go +++ b/vendor/github.com/prometheus/common/expfmt/openmetrics_create.go @@ -35,6 +35,18 @@ import (  // sanity checks. If the input contains duplicate metrics or invalid metric or  // label names, the conversion will result in invalid text format output.  // +// If metric names conform to the legacy validation pattern, they will be placed +// outside the brackets in the traditional way, like `foo{}`. If the metric name +// fails the legacy validation check, it will be placed quoted inside the +// brackets: `{"foo"}`. As stated above, the input is assumed to be santized and +// no error will be thrown in this case. +// +// Similar to metric names, if label names conform to the legacy validation +// pattern, they will be unquoted as normal, like `foo{bar="baz"}`. If the label +// name fails the legacy validation check, it will be quoted: +// `foo{"bar"="baz"}`. As stated above, the input is assumed to be santized and +// no error will be thrown in this case. +//  // This function fulfills the type 'expfmt.encoder'.  //  // Note that OpenMetrics requires a final `# EOF` line. Since this function acts @@ -98,7 +110,7 @@ func MetricFamilyToOpenMetrics(out io.Writer, in *dto.MetricFamily) (written int  		if err != nil {  			return  		} -		n, err = w.WriteString(shortName) +		n, err = writeName(w, shortName)  		written += n  		if err != nil {  			return @@ -124,7 +136,7 @@ func MetricFamilyToOpenMetrics(out io.Writer, in *dto.MetricFamily) (written int  	if err != nil {  		return  	} -	n, err = w.WriteString(shortName) +	n, err = writeName(w, shortName)  	written += n  	if err != nil {  		return @@ -303,21 +315,9 @@ func writeOpenMetricsSample(  	floatValue float64, intValue uint64, useIntValue bool,  	exemplar *dto.Exemplar,  ) (int, error) { -	var written int -	n, err := w.WriteString(name) -	written += n -	if err != nil { -		return written, err -	} -	if suffix != "" { -		n, err = w.WriteString(suffix) -		written += n -		if err != nil { -			return written, err -		} -	} -	n, err = writeOpenMetricsLabelPairs( -		w, metric.Label, additionalLabelName, additionalLabelValue, +	written := 0 +	n, err := writeOpenMetricsNameAndLabelPairs( +		w, name+suffix, metric.Label, additionalLabelName, additionalLabelValue,  	)  	written += n  	if err != nil { @@ -365,27 +365,58 @@ func writeOpenMetricsSample(  	return written, nil  } -// writeOpenMetricsLabelPairs works like writeOpenMetrics but formats the float -// in OpenMetrics style. -func writeOpenMetricsLabelPairs( +// writeOpenMetricsNameAndLabelPairs works like writeOpenMetricsSample but +// formats the float in OpenMetrics style. +func writeOpenMetricsNameAndLabelPairs(  	w enhancedWriter, +	name string,  	in []*dto.LabelPair,  	additionalLabelName string, additionalLabelValue float64,  ) (int, error) { -	if len(in) == 0 && additionalLabelName == "" { -		return 0, nil -	}  	var ( -		written   int -		separator byte = '{' +		written            int +		separator          byte = '{' +		metricInsideBraces      = false  	) + +	if name != "" { +		// If the name does not pass the legacy validity check, we must put the +		// metric name inside the braces, quoted. +		if !model.IsValidLegacyMetricName(model.LabelValue(name)) { +			metricInsideBraces = true +			err := w.WriteByte(separator) +			written++ +			if err != nil { +				return written, err +			} +			separator = ',' +		} + +		n, err := writeName(w, name) +		written += n +		if err != nil { +			return written, err +		} +	} + +	if len(in) == 0 && additionalLabelName == "" { +		if metricInsideBraces { +			err := w.WriteByte('}') +			written++ +			if err != nil { +				return written, err +			} +		} +		return written, nil +	} +  	for _, lp := range in {  		err := w.WriteByte(separator)  		written++  		if err != nil {  			return written, err  		} -		n, err := w.WriteString(lp.GetName()) +		n, err := writeName(w, lp.GetName())  		written += n  		if err != nil {  			return written, err @@ -451,7 +482,7 @@ func writeExemplar(w enhancedWriter, e *dto.Exemplar) (int, error) {  	if err != nil {  		return written, err  	} -	n, err = writeOpenMetricsLabelPairs(w, e.Label, "", 0) +	n, err = writeOpenMetricsNameAndLabelPairs(w, "", e.Label, "", 0)  	written += n  	if err != nil {  		return written, err diff --git a/vendor/github.com/prometheus/common/expfmt/text_create.go b/vendor/github.com/prometheus/common/expfmt/text_create.go index 2946b8f1a..f9b8265a9 100644 --- a/vendor/github.com/prometheus/common/expfmt/text_create.go +++ b/vendor/github.com/prometheus/common/expfmt/text_create.go @@ -62,6 +62,18 @@ var (  // contains duplicate metrics or invalid metric or label names, the conversion  // will result in invalid text format output.  // +// If metric names conform to the legacy validation pattern, they will be placed +// outside the brackets in the traditional way, like `foo{}`. If the metric name +// fails the legacy validation check, it will be placed quoted inside the +// brackets: `{"foo"}`. As stated above, the input is assumed to be santized and +// no error will be thrown in this case. +// +// Similar to metric names, if label names conform to the legacy validation +// pattern, they will be unquoted as normal, like `foo{bar="baz"}`. If the label +// name fails the legacy validation check, it will be quoted: +// `foo{"bar"="baz"}`. As stated above, the input is assumed to be santized and +// no error will be thrown in this case. +//  // This method fulfills the type 'prometheus.encoder'.  func MetricFamilyToText(out io.Writer, in *dto.MetricFamily) (written int, err error) {  	// Fail-fast checks. @@ -98,7 +110,7 @@ func MetricFamilyToText(out io.Writer, in *dto.MetricFamily) (written int, err e  		if err != nil {  			return  		} -		n, err = w.WriteString(name) +		n, err = writeName(w, name)  		written += n  		if err != nil {  			return @@ -124,7 +136,7 @@ func MetricFamilyToText(out io.Writer, in *dto.MetricFamily) (written int, err e  	if err != nil {  		return  	} -	n, err = w.WriteString(name) +	n, err = writeName(w, name)  	written += n  	if err != nil {  		return @@ -280,21 +292,9 @@ func writeSample(  	additionalLabelName string, additionalLabelValue float64,  	value float64,  ) (int, error) { -	var written int -	n, err := w.WriteString(name) -	written += n -	if err != nil { -		return written, err -	} -	if suffix != "" { -		n, err = w.WriteString(suffix) -		written += n -		if err != nil { -			return written, err -		} -	} -	n, err = writeLabelPairs( -		w, metric.Label, additionalLabelName, additionalLabelValue, +	written := 0 +	n, err := writeNameAndLabelPairs( +		w, name+suffix, metric.Label, additionalLabelName, additionalLabelValue,  	)  	written += n  	if err != nil { @@ -330,32 +330,64 @@ func writeSample(  	return written, nil  } -// writeLabelPairs converts a slice of LabelPair proto messages plus the -// explicitly given additional label pair into text formatted as required by the -// text format and writes it to 'w'. An empty slice in combination with an empty -// string 'additionalLabelName' results in nothing being written. Otherwise, the -// label pairs are written, escaped as required by the text format, and enclosed -// in '{...}'. The function returns the number of bytes written and any error -// encountered. -func writeLabelPairs( +// writeNameAndLabelPairs converts a slice of LabelPair proto messages plus the +// explicitly given metric name and additional label pair into text formatted as +// required by the text format and writes it to 'w'. An empty slice in +// combination with an empty string 'additionalLabelName' results in nothing +// being written. Otherwise, the label pairs are written, escaped as required by +// the text format, and enclosed in '{...}'. The function returns the number of +// bytes written and any error encountered. If the metric name is not +// legacy-valid, it will be put inside the brackets as well. Legacy-invalid +// label names will also be quoted. +func writeNameAndLabelPairs(  	w enhancedWriter, +	name string,  	in []*dto.LabelPair,  	additionalLabelName string, additionalLabelValue float64,  ) (int, error) { -	if len(in) == 0 && additionalLabelName == "" { -		return 0, nil -	}  	var ( -		written   int -		separator byte = '{' +		written            int +		separator          byte = '{' +		metricInsideBraces      = false  	) + +	if name != "" { +		// If the name does not pass the legacy validity check, we must put the +		// metric name inside the braces. +		if !model.IsValidLegacyMetricName(model.LabelValue(name)) { +			metricInsideBraces = true +			err := w.WriteByte(separator) +			written++ +			if err != nil { +				return written, err +			} +			separator = ',' +		} +		n, err := writeName(w, name) +		written += n +		if err != nil { +			return written, err +		} +	} + +	if len(in) == 0 && additionalLabelName == "" { +		if metricInsideBraces { +			err := w.WriteByte('}') +			written++ +			if err != nil { +				return written, err +			} +		} +		return written, nil +	} +  	for _, lp := range in {  		err := w.WriteByte(separator)  		written++  		if err != nil {  			return written, err  		} -		n, err := w.WriteString(lp.GetName()) +		n, err := writeName(w, lp.GetName())  		written += n  		if err != nil {  			return written, err @@ -462,3 +494,27 @@ func writeInt(w enhancedWriter, i int64) (int, error) {  	numBufPool.Put(bp)  	return written, err  } + +// writeName writes a string as-is if it complies with the legacy naming +// scheme, or escapes it in double quotes if not. +func writeName(w enhancedWriter, name string) (int, error) { +	if model.IsValidLegacyMetricName(model.LabelValue(name)) { +		return w.WriteString(name) +	} +	var written int +	var err error +	err = w.WriteByte('"') +	written++ +	if err != nil { +		return written, err +	} +	var n int +	n, err = writeEscapedString(w, name, true) +	written += n +	if err != nil { +		return written, err +	} +	err = w.WriteByte('"') +	written++ +	return written, err +} diff --git a/vendor/github.com/prometheus/common/expfmt/text_parse.go b/vendor/github.com/prometheus/common/expfmt/text_parse.go index 35db1cc9d..26490211a 100644 --- a/vendor/github.com/prometheus/common/expfmt/text_parse.go +++ b/vendor/github.com/prometheus/common/expfmt/text_parse.go @@ -16,6 +16,7 @@ package expfmt  import (  	"bufio"  	"bytes" +	"errors"  	"fmt"  	"io"  	"math" @@ -24,8 +25,9 @@ import (  	dto "github.com/prometheus/client_model/go" -	"github.com/prometheus/common/model"  	"google.golang.org/protobuf/proto" + +	"github.com/prometheus/common/model"  )  // A stateFn is a function that represents a state in a state machine. By @@ -112,7 +114,7 @@ func (p *TextParser) TextToMetricFamilies(in io.Reader) (map[string]*dto.MetricF  	// stream. Turn this error into something nicer and more  	// meaningful. (io.EOF is often used as a signal for the legitimate end  	// of an input stream.) -	if p.err == io.EOF { +	if p.err != nil && errors.Is(p.err, io.EOF) {  		p.parseError("unexpected end of input stream")  	}  	return p.metricFamiliesByName, p.err @@ -146,7 +148,7 @@ func (p *TextParser) startOfLine() stateFn {  		// which is not an error but the signal that we are done.  		// Any other error that happens to align with the start of  		// a line is still an error. -		if p.err == io.EOF { +		if errors.Is(p.err, io.EOF) {  			p.err = nil  		}  		return nil | 
