Skip to content
This repository was archived by the owner on Jan 10, 2023. It is now read-only.

Commit bc6cb05

Browse files
committed
sanitize label values for invalid utf-8
1 parent 6cb4313 commit bc6cb05

File tree

3 files changed

+56
-3
lines changed

3 files changed

+56
-3
lines changed

go.mod

+1
Original file line numberDiff line numberDiff line change
@@ -11,4 +11,5 @@ require (
1111
github.com/prometheus/procfs v0.0.0-20190519111021-9935e8e0588d // indirect
1212
github.com/yuin/gopher-lua v0.0.0-20181214045814-db9ae37725ec // indirect
1313
golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6 // indirect
14+
golang.org/x/text v0.3.0
1415
)

metric.go

+20-1
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ import (
55
"log"
66
"strconv"
77
"strings"
8+
"unicode/utf8"
89

910
"github.com/prometheus/client_golang/prometheus"
1011
)
@@ -45,6 +46,10 @@ func infoCollect(
4546
) []prometheus.Metric {
4647
var res []prometheus.Metric
4748
stats := parseInfo(info)
49+
validLabelValues := make([]string, len(labelValues))
50+
for pos, lv := range labelValues {
51+
validLabelValues[pos] = sanitizeLabelValue(lv)
52+
}
4853
for key, m := range metrics {
4954
v, ok := stats[key]
5055
if !ok {
@@ -58,12 +63,26 @@ func infoCollect(
5863
}
5964
res = append(
6065
res,
61-
prometheus.MustNewConstMetric(m.desc, m.typ, f, labelValues...),
66+
prometheus.MustNewConstMetric(m.desc, m.typ, f, validLabelValues...),
6267
)
6368
}
6469
return res
6570
}
6671

72+
func sanitizeLabelValue(lv string) string {
73+
if utf8.ValidString(lv) {
74+
return lv
75+
}
76+
fixUtf := func(r rune) rune {
77+
if r == utf8.RuneError {
78+
return 65533
79+
}
80+
return r
81+
}
82+
83+
return strings.Map(fixUtf, lv)
84+
}
85+
6786
func parseInfo(s string) map[string]string {
6887
r := map[string]string{}
6988
for _, l := range strings.Split(s, ";") {

metric_test.go

+35-2
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,7 @@ func TestInfoCollect(t *testing.T) {
4444
payload string
4545
field string
4646
metric cmetric
47+
labels []string
4748
want string
4849
}
4950
for n, c := range []cas{
@@ -103,9 +104,41 @@ func TestInfoCollect(t *testing.T) {
103104
},
104105
want: `gauge:<value:1 > `,
105106
},
107+
{
108+
payload: "counter-1=3.14:gauge-1=6.12:flag=true:counter-2=6.66",
109+
field: "flag",
110+
metric: cmetric{
111+
typ: prometheus.GaugeValue,
112+
desc: prometheus.NewDesc(
113+
"c1",
114+
"My second flag",
115+
[]string{"namespace", "set"},
116+
nil,
117+
),
118+
},
119+
//labels: []string{"ns", ""},
120+
labels: []string{"ns", "\xC0"},
121+
want: `label:<name:"namespace" value:"ns" > label:<name:"set" value:"\357\277\275" > gauge:<value:1 > `,
122+
},
123+
{
124+
payload: "counter-1=3.14:gauge-1=6.12:flag=true:counter-2=6.66",
125+
field: "flag",
126+
metric: cmetric{
127+
typ: prometheus.GaugeValue,
128+
desc: prometheus.NewDesc(
129+
"c1",
130+
"My second flag",
131+
[]string{"namespace", "set"},
132+
nil,
133+
),
134+
},
135+
//labels: []string{"ns", ""},
136+
labels: []string{"ns", "ӕ"},
137+
want: `label:<name:"namespace" value:"ns" > label:<name:"set" value:"\323\225" > gauge:<value:1 > `,
138+
},
106139
} {
107140
metrics := cmetrics{c.field: c.metric}
108-
ms := infoCollect(metrics, c.payload)
141+
ms := infoCollect(metrics, c.payload, c.labels...)
109142

110143
if have, want := len(ms), 1; have != want {
111144
t.Fatalf("have %d, want %d", have, want)
@@ -117,4 +150,4 @@ func TestInfoCollect(t *testing.T) {
117150
t.Errorf("case %d: have %q, want %q", n, have, want)
118151
}
119152
}
120-
}
153+
}

0 commit comments

Comments
 (0)