@@ -22,9 +22,9 @@ import (
22
22
"math"
23
23
"strconv"
24
24
"strings"
25
+ "unicode/utf8"
25
26
26
27
dto "github.com/prometheus/client_model/go"
27
-
28
28
"google.golang.org/protobuf/proto"
29
29
30
30
"github.com/prometheus/common/model"
@@ -60,6 +60,7 @@ type TextParser struct {
60
60
currentMF * dto.MetricFamily
61
61
currentMetric * dto.Metric
62
62
currentLabelPair * dto.LabelPair
63
+ currentLabelPairs []* dto.LabelPair // Temporarily stores label pairs while parsing a metric line.
63
64
64
65
// The remaining member variables are only used for summaries/histograms.
65
66
currentLabels map [string ]string // All labels including '__name__' but excluding 'quantile'/'le'
@@ -74,6 +75,7 @@ type TextParser struct {
74
75
// count and sum of that summary/histogram.
75
76
currentIsSummaryCount , currentIsSummarySum bool
76
77
currentIsHistogramCount , currentIsHistogramSum bool
78
+ currentMetricIsInsideBraces bool
77
79
}
78
80
79
81
// TextToMetricFamilies reads 'in' as the simple and flat text-based exchange
@@ -137,12 +139,14 @@ func (p *TextParser) reset(in io.Reader) {
137
139
}
138
140
p .currentQuantile = math .NaN ()
139
141
p .currentBucket = math .NaN ()
142
+ p .currentMF = nil
140
143
}
141
144
142
145
// startOfLine represents the state where the next byte read from p.buf is the
143
146
// start of a line (or whitespace leading up to it).
144
147
func (p * TextParser ) startOfLine () stateFn {
145
148
p .lineCount ++
149
+ p .currentMetricIsInsideBraces = false
146
150
if p .skipBlankTab (); p .err != nil {
147
151
// This is the only place that we expect to see io.EOF,
148
152
// which is not an error but the signal that we are done.
@@ -158,6 +162,9 @@ func (p *TextParser) startOfLine() stateFn {
158
162
return p .startComment
159
163
case '\n' :
160
164
return p .startOfLine // Empty line, start the next one.
165
+ case '{' :
166
+ p .currentMetricIsInsideBraces = true
167
+ return p .readingLabels
161
168
}
162
169
return p .readingMetricName
163
170
}
@@ -275,6 +282,8 @@ func (p *TextParser) startLabelName() stateFn {
275
282
return nil // Unexpected end of input.
276
283
}
277
284
if p .currentByte == '}' {
285
+ p .currentMetric .Label = append (p .currentMetric .Label , p .currentLabelPairs ... )
286
+ p .currentLabelPairs = nil
278
287
if p .skipBlankTab (); p .err != nil {
279
288
return nil // Unexpected end of input.
280
289
}
@@ -287,6 +296,38 @@ func (p *TextParser) startLabelName() stateFn {
287
296
p .parseError (fmt .Sprintf ("invalid label name for metric %q" , p .currentMF .GetName ()))
288
297
return nil
289
298
}
299
+ if p .skipBlankTabIfCurrentBlankTab (); p .err != nil {
300
+ return nil // Unexpected end of input.
301
+ }
302
+ if p .currentByte != '=' {
303
+ if p .currentMetricIsInsideBraces {
304
+ if p .currentMF != nil && p .currentMF .GetName () != p .currentToken .String () {
305
+ p .parseError (fmt .Sprintf ("multiple metric names %s %s" , p .currentMF .GetName (), p .currentToken .String ()))
306
+ return nil
307
+ }
308
+ switch p .currentByte {
309
+ case ',' :
310
+ p .setOrCreateCurrentMF ()
311
+ p .currentMetric = & dto.Metric {}
312
+ return p .startLabelName
313
+ case '}' :
314
+ p .setOrCreateCurrentMF ()
315
+ p .currentMetric = & dto.Metric {}
316
+ p .currentMetric .Label = append (p .currentMetric .Label , p .currentLabelPairs ... )
317
+ p .currentLabelPairs = nil
318
+ if p .skipBlankTab (); p .err != nil {
319
+ return nil // Unexpected end of input.
320
+ }
321
+ return p .readingValue
322
+ default :
323
+ p .parseError (fmt .Sprintf ("unexpected end of metric name %q" , p .currentByte ))
324
+ return nil
325
+ }
326
+ }
327
+ p .parseError (fmt .Sprintf ("expected '=' after label name, found %q" , p .currentByte ))
328
+ p .currentLabelPairs = nil
329
+ return nil
330
+ }
290
331
p .currentLabelPair = & dto.LabelPair {Name : proto .String (p .currentToken .String ())}
291
332
if p .currentLabelPair .GetName () == string (model .MetricNameLabel ) {
292
333
p .parseError (fmt .Sprintf ("label name %q is reserved" , model .MetricNameLabel ))
@@ -296,23 +337,17 @@ func (p *TextParser) startLabelName() stateFn {
296
337
// labels to 'real' labels.
297
338
if ! (p .currentMF .GetType () == dto .MetricType_SUMMARY && p .currentLabelPair .GetName () == model .QuantileLabel ) &&
298
339
! (p .currentMF .GetType () == dto .MetricType_HISTOGRAM && p .currentLabelPair .GetName () == model .BucketLabel ) {
299
- p .currentMetric .Label = append (p .currentMetric .Label , p .currentLabelPair )
300
- }
301
- if p .skipBlankTabIfCurrentBlankTab (); p .err != nil {
302
- return nil // Unexpected end of input.
303
- }
304
- if p .currentByte != '=' {
305
- p .parseError (fmt .Sprintf ("expected '=' after label name, found %q" , p .currentByte ))
306
- return nil
340
+ p .currentLabelPairs = append (p .currentLabelPairs , p .currentLabelPair )
307
341
}
308
342
// Check for duplicate label names.
309
343
labels := make (map [string ]struct {})
310
- for _ , l := range p .currentMetric . Label {
344
+ for _ , l := range p .currentLabelPairs {
311
345
lName := l .GetName ()
312
346
if _ , exists := labels [lName ]; ! exists {
313
347
labels [lName ] = struct {}{}
314
348
} else {
315
349
p .parseError (fmt .Sprintf ("duplicate label names for metric %q" , p .currentMF .GetName ()))
350
+ p .currentLabelPairs = nil
316
351
return nil
317
352
}
318
353
}
@@ -345,6 +380,7 @@ func (p *TextParser) startLabelValue() stateFn {
345
380
if p .currentQuantile , p .err = parseFloat (p .currentLabelPair .GetValue ()); p .err != nil {
346
381
// Create a more helpful error message.
347
382
p .parseError (fmt .Sprintf ("expected float as value for 'quantile' label, got %q" , p .currentLabelPair .GetValue ()))
383
+ p .currentLabelPairs = nil
348
384
return nil
349
385
}
350
386
} else {
@@ -371,12 +407,19 @@ func (p *TextParser) startLabelValue() stateFn {
371
407
return p .startLabelName
372
408
373
409
case '}' :
410
+ if p .currentMF == nil {
411
+ p .parseError ("invalid metric name" )
412
+ return nil
413
+ }
414
+ p .currentMetric .Label = append (p .currentMetric .Label , p .currentLabelPairs ... )
415
+ p .currentLabelPairs = nil
374
416
if p .skipBlankTab (); p .err != nil {
375
417
return nil // Unexpected end of input.
376
418
}
377
419
return p .readingValue
378
420
default :
379
421
p .parseError (fmt .Sprintf ("unexpected end of label value %q" , p .currentLabelPair .GetValue ()))
422
+ p .currentLabelPairs = nil
380
423
return nil
381
424
}
382
425
}
@@ -585,6 +628,8 @@ func (p *TextParser) readTokenUntilNewline(recognizeEscapeSequence bool) {
585
628
p .currentToken .WriteByte (p .currentByte )
586
629
case 'n' :
587
630
p .currentToken .WriteByte ('\n' )
631
+ case '"' :
632
+ p .currentToken .WriteByte ('"' )
588
633
default :
589
634
p .parseError (fmt .Sprintf ("invalid escape sequence '\\ %c'" , p .currentByte ))
590
635
return
@@ -610,13 +655,45 @@ func (p *TextParser) readTokenUntilNewline(recognizeEscapeSequence bool) {
610
655
// but not into p.currentToken.
611
656
func (p * TextParser ) readTokenAsMetricName () {
612
657
p .currentToken .Reset ()
658
+ // A UTF-8 metric name must be quoted and may have escaped characters.
659
+ quoted := false
660
+ escaped := false
613
661
if ! isValidMetricNameStart (p .currentByte ) {
614
662
return
615
663
}
616
- for {
617
- p .currentToken .WriteByte (p .currentByte )
664
+ for p .err == nil {
665
+ if escaped {
666
+ switch p .currentByte {
667
+ case '\\' :
668
+ p .currentToken .WriteByte (p .currentByte )
669
+ case 'n' :
670
+ p .currentToken .WriteByte ('\n' )
671
+ case '"' :
672
+ p .currentToken .WriteByte ('"' )
673
+ default :
674
+ p .parseError (fmt .Sprintf ("invalid escape sequence '\\ %c'" , p .currentByte ))
675
+ return
676
+ }
677
+ escaped = false
678
+ } else {
679
+ switch p .currentByte {
680
+ case '"' :
681
+ quoted = ! quoted
682
+ if ! quoted {
683
+ p .currentByte , p .err = p .buf .ReadByte ()
684
+ return
685
+ }
686
+ case '\n' :
687
+ p .parseError (fmt .Sprintf ("metric name %q contains unescaped new-line" , p .currentToken .String ()))
688
+ return
689
+ case '\\' :
690
+ escaped = true
691
+ default :
692
+ p .currentToken .WriteByte (p .currentByte )
693
+ }
694
+ }
618
695
p .currentByte , p .err = p .buf .ReadByte ()
619
- if p . err != nil || ! isValidMetricNameContinuation ( p .currentByte ) {
696
+ if ! isValidMetricNameContinuation ( p . currentByte , quoted ) || ( ! quoted && p .currentByte == ' ' ) {
620
697
return
621
698
}
622
699
}
@@ -628,13 +705,45 @@ func (p *TextParser) readTokenAsMetricName() {
628
705
// but not into p.currentToken.
629
706
func (p * TextParser ) readTokenAsLabelName () {
630
707
p .currentToken .Reset ()
708
+ // A UTF-8 label name must be quoted and may have escaped characters.
709
+ quoted := false
710
+ escaped := false
631
711
if ! isValidLabelNameStart (p .currentByte ) {
632
712
return
633
713
}
634
- for {
635
- p .currentToken .WriteByte (p .currentByte )
714
+ for p .err == nil {
715
+ if escaped {
716
+ switch p .currentByte {
717
+ case '\\' :
718
+ p .currentToken .WriteByte (p .currentByte )
719
+ case 'n' :
720
+ p .currentToken .WriteByte ('\n' )
721
+ case '"' :
722
+ p .currentToken .WriteByte ('"' )
723
+ default :
724
+ p .parseError (fmt .Sprintf ("invalid escape sequence '\\ %c'" , p .currentByte ))
725
+ return
726
+ }
727
+ escaped = false
728
+ } else {
729
+ switch p .currentByte {
730
+ case '"' :
731
+ quoted = ! quoted
732
+ if ! quoted {
733
+ p .currentByte , p .err = p .buf .ReadByte ()
734
+ return
735
+ }
736
+ case '\n' :
737
+ p .parseError (fmt .Sprintf ("label name %q contains unescaped new-line" , p .currentToken .String ()))
738
+ return
739
+ case '\\' :
740
+ escaped = true
741
+ default :
742
+ p .currentToken .WriteByte (p .currentByte )
743
+ }
744
+ }
636
745
p .currentByte , p .err = p .buf .ReadByte ()
637
- if p . err != nil || ! isValidLabelNameContinuation ( p .currentByte ) {
746
+ if ! isValidLabelNameContinuation ( p . currentByte , quoted ) || ( ! quoted && p .currentByte == '=' ) {
638
747
return
639
748
}
640
749
}
@@ -660,6 +769,7 @@ func (p *TextParser) readTokenAsLabelValue() {
660
769
p .currentToken .WriteByte ('\n' )
661
770
default :
662
771
p .parseError (fmt .Sprintf ("invalid escape sequence '\\ %c'" , p .currentByte ))
772
+ p .currentLabelPairs = nil
663
773
return
664
774
}
665
775
escaped = false
@@ -718,19 +828,19 @@ func (p *TextParser) setOrCreateCurrentMF() {
718
828
}
719
829
720
830
func isValidLabelNameStart (b byte ) bool {
721
- return (b >= 'a' && b <= 'z' ) || (b >= 'A' && b <= 'Z' ) || b == '_'
831
+ return (b >= 'a' && b <= 'z' ) || (b >= 'A' && b <= 'Z' ) || b == '_' || b == '"'
722
832
}
723
833
724
- func isValidLabelNameContinuation (b byte ) bool {
725
- return isValidLabelNameStart (b ) || (b >= '0' && b <= '9' )
834
+ func isValidLabelNameContinuation (b byte , quoted bool ) bool {
835
+ return isValidLabelNameStart (b ) || (b >= '0' && b <= '9' ) || ( quoted && utf8 . ValidString ( string ( b )))
726
836
}
727
837
728
838
func isValidMetricNameStart (b byte ) bool {
729
839
return isValidLabelNameStart (b ) || b == ':'
730
840
}
731
841
732
- func isValidMetricNameContinuation (b byte ) bool {
733
- return isValidLabelNameContinuation (b ) || b == ':'
842
+ func isValidMetricNameContinuation (b byte , quoted bool ) bool {
843
+ return isValidLabelNameContinuation (b , quoted ) || b == ':'
734
844
}
735
845
736
846
func isBlankOrTab (b byte ) bool {
0 commit comments