@@ -11,6 +11,11 @@ import (
11
11
"sync"
12
12
"testing"
13
13
"time"
14
+ "unicode/utf8"
15
+
16
+ otlptranslate "github.com/prometheus/prometheus/storage/remote/otlptranslator/prometheus"
17
+
18
+ "github.com/grafana/loki/pkg/push"
14
19
15
20
"github.com/c2h5oh/datasize"
16
21
"github.com/go-kit/log"
@@ -46,6 +51,13 @@ import (
46
51
"github.com/grafana/loki/v3/pkg/validation"
47
52
)
48
53
54
+ const (
55
+ smValidName = "valid_name"
56
+ smInvalidName = "invalid-name"
57
+ smValidValue = "valid-value私"
58
+ smInvalidValue = "valid-value�"
59
+ )
60
+
49
61
var (
50
62
success = & logproto.PushResponse {}
51
63
ctx = user .InjectOrgID (context .Background (), "test" )
@@ -428,7 +440,7 @@ func TestDistributorPushConcurrently(t *testing.T) {
428
440
[]string {
429
441
fmt .Sprintf (`{app="foo-%d"}` , n ),
430
442
fmt .Sprintf (`{instance="bar-%d"}` , n ),
431
- },
443
+ }, false , false , false ,
432
444
)
433
445
response , err := distributors [n % len (distributors )].Push (ctx , request )
434
446
assert .NoError (t , err )
@@ -1226,17 +1238,58 @@ func Benchmark_Push(b *testing.B) {
1226
1238
limits .RejectOldSamplesMaxAge = model .Duration (24 * time .Hour )
1227
1239
limits .CreationGracePeriod = model .Duration (24 * time .Hour )
1228
1240
distributors , _ := prepare (& testing.T {}, 1 , 5 , limits , nil )
1229
- request := makeWriteRequest (100000 , 100 )
1230
-
1231
1241
b .ResetTimer ()
1232
1242
b .ReportAllocs ()
1233
1243
1234
- for n := 0 ; n < b .N ; n ++ {
1235
- _ , err := distributors [0 ].Push (ctx , request )
1236
- if err != nil {
1237
- require .NoError (b , err )
1244
+ b .Run ("no structured metadata" , func (b * testing.B ) {
1245
+ for n := 0 ; n < b .N ; n ++ {
1246
+ request := makeWriteRequestWithLabels (100000 , 100 , []string {`{foo="bar"}` }, false , false , false )
1247
+ _ , err := distributors [0 ].Push (ctx , request )
1248
+ if err != nil {
1249
+ require .NoError (b , err )
1250
+ }
1238
1251
}
1239
- }
1252
+ })
1253
+
1254
+ b .Run ("all valid structured metadata" , func (b * testing.B ) {
1255
+ for n := 0 ; n < b .N ; n ++ {
1256
+ request := makeWriteRequestWithLabels (100000 , 100 , []string {`{foo="bar"}` }, true , false , false )
1257
+ _ , err := distributors [0 ].Push (ctx , request )
1258
+ if err != nil {
1259
+ require .NoError (b , err )
1260
+ }
1261
+ }
1262
+ })
1263
+
1264
+ b .Run ("structured metadata with invalid names" , func (b * testing.B ) {
1265
+ for n := 0 ; n < b .N ; n ++ {
1266
+ request := makeWriteRequestWithLabels (100000 , 100 , []string {`{foo="bar"}` }, true , true , false )
1267
+ _ , err := distributors [0 ].Push (ctx , request )
1268
+ if err != nil {
1269
+ require .NoError (b , err )
1270
+ }
1271
+ }
1272
+ })
1273
+
1274
+ b .Run ("structured metadata with invalid values" , func (b * testing.B ) {
1275
+ for n := 0 ; n < b .N ; n ++ {
1276
+ request := makeWriteRequestWithLabels (100000 , 100 , []string {`{foo="bar"}` }, true , false , true )
1277
+ _ , err := distributors [0 ].Push (ctx , request )
1278
+ if err != nil {
1279
+ require .NoError (b , err )
1280
+ }
1281
+ }
1282
+ })
1283
+
1284
+ b .Run ("structured metadata with invalid names and values" , func (b * testing.B ) {
1285
+ for n := 0 ; n < b .N ; n ++ {
1286
+ request := makeWriteRequestWithLabels (100000 , 100 , []string {`{foo="bar"}` }, true , true , true )
1287
+ _ , err := distributors [0 ].Push (ctx , request )
1288
+ if err != nil {
1289
+ require .NoError (b , err )
1290
+ }
1291
+ }
1292
+ })
1240
1293
}
1241
1294
1242
1295
func TestShardCalculation (t * testing.T ) {
@@ -1696,7 +1749,7 @@ func makeWriteRequestWithLabelsWithLevel(lines, size int, labels []string, level
1696
1749
}
1697
1750
}
1698
1751
1699
- func makeWriteRequestWithLabels (lines , size int , labels []string ) * logproto.PushRequest {
1752
+ func makeWriteRequestWithLabels (lines , size int , labels []string , addStructuredMetadata , invalidName , invalidValue bool ) * logproto.PushRequest {
1700
1753
streams := make ([]logproto.Stream , len (labels ))
1701
1754
for i := 0 ; i < len (labels ); i ++ {
1702
1755
stream := logproto.Stream {Labels : labels [i ]}
@@ -1705,11 +1758,24 @@ func makeWriteRequestWithLabels(lines, size int, labels []string) *logproto.Push
1705
1758
// Construct the log line, honoring the input size
1706
1759
line := strconv .Itoa (j ) + strings .Repeat ("0" , size )
1707
1760
line = line [:size ]
1708
-
1709
- stream .Entries = append (stream .Entries , logproto.Entry {
1761
+ entry := logproto.Entry {
1710
1762
Timestamp : time .Now ().Add (time .Duration (j ) * time .Millisecond ),
1711
1763
Line : line ,
1712
- })
1764
+ }
1765
+ if addStructuredMetadata {
1766
+ name := smValidName
1767
+ value := smValidValue
1768
+ if invalidName {
1769
+ name = smInvalidName
1770
+ }
1771
+ if invalidValue {
1772
+ value = smInvalidValue
1773
+ }
1774
+ entry .StructuredMetadata = push.LabelsAdapter {
1775
+ {Name : name , Value : value },
1776
+ }
1777
+ }
1778
+ stream .Entries = append (stream .Entries , entry )
1713
1779
}
1714
1780
1715
1781
streams [i ] = stream
@@ -1721,7 +1787,7 @@ func makeWriteRequestWithLabels(lines, size int, labels []string) *logproto.Push
1721
1787
}
1722
1788
1723
1789
func makeWriteRequest (lines , size int ) * logproto.PushRequest {
1724
- return makeWriteRequestWithLabels (lines , size , []string {`{foo="bar"}` })
1790
+ return makeWriteRequestWithLabels (lines , size , []string {`{foo="bar"}` }, false , false , false )
1725
1791
}
1726
1792
1727
1793
type mockKafkaWriter struct {
@@ -1777,6 +1843,19 @@ func (i *mockIngester) Push(_ context.Context, in *logproto.PushRequest, _ ...gr
1777
1843
1778
1844
i .mu .Lock ()
1779
1845
defer i .mu .Unlock ()
1846
+ for _ , s := range in .Streams {
1847
+ for _ , e := range s .Entries {
1848
+ for _ , sm := range e .StructuredMetadata {
1849
+ if strings .ContainsRune (sm .Value , utf8 .RuneError ) {
1850
+ return nil , fmt .Errorf ("sm value was not sanitized before being pushed to ignester, invalid utf 8 rune %d" , utf8 .RuneError )
1851
+ }
1852
+ if sm .Name != otlptranslate .NormalizeLabel (sm .Name ) {
1853
+ return nil , fmt .Errorf ("sm name was not sanitized before being sent to ingester, contained characters %s" , sm .Name )
1854
+
1855
+ }
1856
+ }
1857
+ }
1858
+ }
1780
1859
1781
1860
i .pushed = append (i .pushed , in )
1782
1861
return nil , nil
@@ -1875,3 +1954,39 @@ func TestDistributorTee(t *testing.T) {
1875
1954
require .Equal (t , "test" , tee .tenant )
1876
1955
}
1877
1956
}
1957
+
1958
+ func TestDistributor_StructuredMetadataSanitization (t * testing.T ) {
1959
+ limits := & validation.Limits {}
1960
+ flagext .DefaultValues (limits )
1961
+ for _ , tc := range []struct {
1962
+ req * logproto.PushRequest
1963
+ expectedResponse * logproto.PushResponse
1964
+ }{
1965
+ {
1966
+ makeWriteRequestWithLabels (10 , 10 , []string {`{foo="bar"}` }, true , false , false ),
1967
+ success ,
1968
+ },
1969
+ {
1970
+ makeWriteRequestWithLabels (10 , 10 , []string {`{foo="bar"}` }, true , true , false ),
1971
+ success ,
1972
+ },
1973
+ {
1974
+ makeWriteRequestWithLabels (10 , 10 , []string {`{foo="bar"}` }, true , false , true ),
1975
+ success ,
1976
+ },
1977
+ {
1978
+ makeWriteRequestWithLabels (10 , 10 , []string {`{foo="bar"}` }, true , true , true ),
1979
+ success ,
1980
+ },
1981
+ } {
1982
+ distributors , _ := prepare (t , 1 , 5 , limits , nil )
1983
+
1984
+ var request logproto.PushRequest
1985
+ request .Streams = append (request .Streams , tc .req .Streams [0 ])
1986
+
1987
+ // the error would happen in the ingester mock, it's set to reject SM that has not been sanitized
1988
+ response , err := distributors [0 ].Push (ctx , & request )
1989
+ require .NoError (t , err )
1990
+ assert .Equal (t , tc .expectedResponse , response )
1991
+ }
1992
+ }
0 commit comments