@@ -35,6 +35,18 @@ import (
35
35
// sanity checks. If the input contains duplicate metrics or invalid metric or
36
36
// label names, the conversion will result in invalid text format output.
37
37
//
38
+ // If metric names conform to the legacy validation pattern, they will be placed
39
+ // outside the brackets in the traditional way, like `foo{}`. If the metric name
40
+ // fails the legacy validation check, it will be placed quoted inside the
41
+ // brackets: `{"foo"}`. As stated above, the input is assumed to be santized and
42
+ // no error will be thrown in this case.
43
+ //
44
+ // Similar to metric names, if label names conform to the legacy validation
45
+ // pattern, they will be unquoted as normal, like `foo{bar="baz"}`. If the label
46
+ // name fails the legacy validation check, it will be quoted:
47
+ // `foo{"bar"="baz"}`. As stated above, the input is assumed to be santized and
48
+ // no error will be thrown in this case.
49
+ //
38
50
// This function fulfills the type 'expfmt.encoder'.
39
51
//
40
52
// 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
98
110
if err != nil {
99
111
return
100
112
}
101
- n , err = w . WriteString ( shortName )
113
+ n , err = writeName ( w , shortName )
102
114
written += n
103
115
if err != nil {
104
116
return
@@ -124,7 +136,7 @@ func MetricFamilyToOpenMetrics(out io.Writer, in *dto.MetricFamily) (written int
124
136
if err != nil {
125
137
return
126
138
}
127
- n , err = w . WriteString ( shortName )
139
+ n , err = writeName ( w , shortName )
128
140
written += n
129
141
if err != nil {
130
142
return
@@ -303,21 +315,9 @@ func writeOpenMetricsSample(
303
315
floatValue float64 , intValue uint64 , useIntValue bool ,
304
316
exemplar * dto.Exemplar ,
305
317
) (int , error ) {
306
- var written int
307
- n , err := w .WriteString (name )
308
- written += n
309
- if err != nil {
310
- return written , err
311
- }
312
- if suffix != "" {
313
- n , err = w .WriteString (suffix )
314
- written += n
315
- if err != nil {
316
- return written , err
317
- }
318
- }
319
- n , err = writeOpenMetricsLabelPairs (
320
- w , metric .Label , additionalLabelName , additionalLabelValue ,
318
+ written := 0
319
+ n , err := writeOpenMetricsNameAndLabelPairs (
320
+ w , name + suffix , metric .Label , additionalLabelName , additionalLabelValue ,
321
321
)
322
322
written += n
323
323
if err != nil {
@@ -365,27 +365,58 @@ func writeOpenMetricsSample(
365
365
return written , nil
366
366
}
367
367
368
- // writeOpenMetricsLabelPairs works like writeOpenMetrics but formats the float
369
- // in OpenMetrics style.
370
- func writeOpenMetricsLabelPairs (
368
+ // writeOpenMetricsNameAndLabelPairs works like writeOpenMetricsSample but
369
+ // formats the float in OpenMetrics style.
370
+ func writeOpenMetricsNameAndLabelPairs (
371
371
w enhancedWriter ,
372
+ name string ,
372
373
in []* dto.LabelPair ,
373
374
additionalLabelName string , additionalLabelValue float64 ,
374
375
) (int , error ) {
375
- if len (in ) == 0 && additionalLabelName == "" {
376
- return 0 , nil
377
- }
378
376
var (
379
- written int
380
- separator byte = '{'
377
+ written int
378
+ separator byte = '{'
379
+ metricInsideBraces = false
381
380
)
381
+
382
+ if name != "" {
383
+ // If the name does not pass the legacy validity check, we must put the
384
+ // metric name inside the braces, quoted.
385
+ if ! model .IsValidLegacyMetricName (model .LabelValue (name )) {
386
+ metricInsideBraces = true
387
+ err := w .WriteByte (separator )
388
+ written ++
389
+ if err != nil {
390
+ return written , err
391
+ }
392
+ separator = ','
393
+ }
394
+
395
+ n , err := writeName (w , name )
396
+ written += n
397
+ if err != nil {
398
+ return written , err
399
+ }
400
+ }
401
+
402
+ if len (in ) == 0 && additionalLabelName == "" {
403
+ if metricInsideBraces {
404
+ err := w .WriteByte ('}' )
405
+ written ++
406
+ if err != nil {
407
+ return written , err
408
+ }
409
+ }
410
+ return written , nil
411
+ }
412
+
382
413
for _ , lp := range in {
383
414
err := w .WriteByte (separator )
384
415
written ++
385
416
if err != nil {
386
417
return written , err
387
418
}
388
- n , err := w . WriteString ( lp .GetName ())
419
+ n , err := writeName ( w , lp .GetName ())
389
420
written += n
390
421
if err != nil {
391
422
return written , err
@@ -451,7 +482,7 @@ func writeExemplar(w enhancedWriter, e *dto.Exemplar) (int, error) {
451
482
if err != nil {
452
483
return written , err
453
484
}
454
- n , err = writeOpenMetricsLabelPairs ( w , e .Label , "" , 0 )
485
+ n , err = writeOpenMetricsNameAndLabelPairs ( w , "" , e .Label , "" , 0 )
455
486
written += n
456
487
if err != nil {
457
488
return written , err
0 commit comments