@@ -22,6 +22,7 @@ import (
22
22
"google.golang.org/protobuf/encoding/prototext"
23
23
24
24
"github.com/prometheus/common/internal/bitbucket.org/ww/goautoneg"
25
+ "github.com/prometheus/common/model"
25
26
26
27
dto "github.com/prometheus/client_model/go"
27
28
)
@@ -61,53 +62,73 @@ func (ec encoderCloser) Close() error {
61
62
// as the support is still experimental. To include the option to negotiate
62
63
// FmtOpenMetrics, use NegotiateOpenMetrics.
63
64
func Negotiate (h http.Header ) Format {
65
+ escapingScheme := Format (fmt .Sprintf ("; escaping=%s" , Format (model .NameEscapingScheme .String ())))
64
66
for _ , ac := range goautoneg .ParseAccept (h .Get (hdrAccept )) {
67
+ if escapeParam := ac .Params [model .EscapingKey ]; escapeParam != "" {
68
+ switch Format (escapeParam ) {
69
+ case model .AllowUTF8 , model .EscapeUnderscores , model .EscapeDots , model .EscapeValues :
70
+ escapingScheme = Format (fmt .Sprintf ("; escaping=%s" , escapeParam ))
71
+ default :
72
+ // If the escaping parameter is unknown, ignore it.
73
+ }
74
+ }
65
75
ver := ac .Params ["version" ]
66
76
if ac .Type + "/" + ac .SubType == ProtoType && ac .Params ["proto" ] == ProtoProtocol {
67
77
switch ac .Params ["encoding" ] {
68
78
case "delimited" :
69
- return FmtProtoDelim
79
+ return FmtProtoDelim + escapingScheme
70
80
case "text" :
71
- return FmtProtoText
81
+ return FmtProtoText + escapingScheme
72
82
case "compact-text" :
73
- return FmtProtoCompact
83
+ return FmtProtoCompact + escapingScheme
74
84
}
75
85
}
76
86
if ac .Type == "text" && ac .SubType == "plain" && (ver == TextVersion || ver == "" ) {
77
- return FmtText
87
+ return FmtText + escapingScheme
78
88
}
79
89
}
80
- return FmtText
90
+ return FmtText + escapingScheme
81
91
}
82
92
83
93
// NegotiateIncludingOpenMetrics works like Negotiate but includes
84
94
// FmtOpenMetrics as an option for the result. Note that this function is
85
95
// temporary and will disappear once FmtOpenMetrics is fully supported and as
86
96
// such may be negotiated by the normal Negotiate function.
87
97
func NegotiateIncludingOpenMetrics (h http.Header ) Format {
98
+ escapingScheme := Format (fmt .Sprintf ("; escaping=%s" , Format (model .NameEscapingScheme .String ())))
88
99
for _ , ac := range goautoneg .ParseAccept (h .Get (hdrAccept )) {
100
+ if escapeParam := ac .Params [model .EscapingKey ]; escapeParam != "" {
101
+ switch Format (escapeParam ) {
102
+ case model .AllowUTF8 , model .EscapeUnderscores , model .EscapeDots , model .EscapeValues :
103
+ escapingScheme = Format (fmt .Sprintf ("; escaping=%s" , escapeParam ))
104
+ default :
105
+ // If the escaping parameter is unknown, ignore it.
106
+ }
107
+ }
89
108
ver := ac .Params ["version" ]
90
109
if ac .Type + "/" + ac .SubType == ProtoType && ac .Params ["proto" ] == ProtoProtocol {
91
110
switch ac .Params ["encoding" ] {
92
111
case "delimited" :
93
- return FmtProtoDelim
112
+ return FmtProtoDelim + escapingScheme
94
113
case "text" :
95
- return FmtProtoText
114
+ return FmtProtoText + escapingScheme
96
115
case "compact-text" :
97
- return FmtProtoCompact
116
+ return FmtProtoCompact + escapingScheme
98
117
}
99
118
}
100
119
if ac .Type == "text" && ac .SubType == "plain" && (ver == TextVersion || ver == "" ) {
101
- return FmtText
120
+ return FmtText + escapingScheme
102
121
}
103
122
if ac .Type + "/" + ac .SubType == OpenMetricsType && (ver == OpenMetricsVersion_0_0_1 || ver == OpenMetricsVersion_1_0_0 || ver == "" ) {
104
- if ver == OpenMetricsVersion_1_0_0 {
105
- return FmtOpenMetrics_1_0_0
123
+ switch ver {
124
+ case OpenMetricsVersion_1_0_0 :
125
+ return FmtOpenMetrics_1_0_0 + escapingScheme
126
+ default :
127
+ return FmtOpenMetrics_0_0_1 + escapingScheme
106
128
}
107
- return FmtOpenMetrics_0_0_1
108
129
}
109
130
}
110
- return FmtText
131
+ return FmtText + escapingScheme
111
132
}
112
133
113
134
// NewEncoder returns a new encoder based on content type negotiation. All
@@ -116,44 +137,48 @@ func NegotiateIncludingOpenMetrics(h http.Header) Format {
116
137
// for FmtOpenMetrics, but a future (breaking) release will add the Close method
117
138
// to the Encoder interface directly. The current version of the Encoder
118
139
// interface is kept for backwards compatibility.
140
+ // In cases where the Format does not allow for UTF-8 names, the global
141
+ // NameEscapingScheme will be applied.
119
142
func NewEncoder (w io.Writer , format Format ) Encoder {
120
- switch format {
121
- case FmtProtoDelim :
143
+ escapingScheme := format .ToEscapingScheme ()
144
+
145
+ switch format .FormatType () {
146
+ case TypeProtoDelim :
122
147
return encoderCloser {
123
148
encode : func (v * dto.MetricFamily ) error {
124
149
_ , err := protodelim .MarshalTo (w , v )
125
150
return err
126
151
},
127
152
close : func () error { return nil },
128
153
}
129
- case FmtProtoCompact :
154
+ case TypeProtoCompact :
130
155
return encoderCloser {
131
156
encode : func (v * dto.MetricFamily ) error {
132
- _ , err := fmt .Fprintln (w , v .String ())
157
+ _ , err := fmt .Fprintln (w , model . EscapeMetricFamily ( v , escapingScheme ) .String ())
133
158
return err
134
159
},
135
160
close : func () error { return nil },
136
161
}
137
- case FmtProtoText :
162
+ case TypeProtoText :
138
163
return encoderCloser {
139
164
encode : func (v * dto.MetricFamily ) error {
140
- _ , err := fmt .Fprintln (w , prototext .Format (v ))
165
+ _ , err := fmt .Fprintln (w , prototext .Format (model . EscapeMetricFamily ( v , escapingScheme ) ))
141
166
return err
142
167
},
143
168
close : func () error { return nil },
144
169
}
145
- case FmtText :
170
+ case TypeTextPlain :
146
171
return encoderCloser {
147
172
encode : func (v * dto.MetricFamily ) error {
148
- _ , err := MetricFamilyToText (w , v )
173
+ _ , err := MetricFamilyToText (w , model . EscapeMetricFamily ( v , escapingScheme ) )
149
174
return err
150
175
},
151
176
close : func () error { return nil },
152
177
}
153
- case FmtOpenMetrics_0_0_1 , FmtOpenMetrics_1_0_0 :
178
+ case TypeOpenMetrics :
154
179
return encoderCloser {
155
180
encode : func (v * dto.MetricFamily ) error {
156
- _ , err := MetricFamilyToOpenMetrics (w , v )
181
+ _ , err := MetricFamilyToOpenMetrics (w , model . EscapeMetricFamily ( v , escapingScheme ) )
157
182
return err
158
183
},
159
184
close : func () error {
0 commit comments