ttrpc/metadata_test.go
Jin Dong 430f734791 Add MD.Clone
Signed-off-by: Jin Dong <djdongjin95@gmail.com>
2024-12-27 16:12:45 +00:00

208 lines
4.9 KiB
Go

/*
Copyright The containerd Authors.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package ttrpc
import (
"context"
"fmt"
"sync"
"testing"
)
func TestMetadataGet(t *testing.T) {
metadata := make(MD)
metadata.Set("foo", "1", "2")
if list, ok := metadata.Get("foo"); !ok {
t.Error("key not found")
} else if len(list) != 2 {
t.Errorf("unexpected number of values: %d", len(list))
} else if list[0] != "1" {
t.Errorf("invalid metadata value at 0: %s", list[0])
} else if list[1] != "2" {
t.Errorf("invalid metadata value at 1: %s", list[1])
}
}
func TestMetadataGetInvalidKey(t *testing.T) {
metadata := make(MD)
metadata.Set("foo", "1", "2")
if _, ok := metadata.Get("invalid"); ok {
t.Error("found invalid key")
}
}
func TestMetadataUnset(t *testing.T) {
metadata := make(MD)
metadata.Set("foo", "1", "2")
metadata.Set("foo")
if _, ok := metadata.Get("foo"); ok {
t.Error("key not deleted")
}
}
func TestMetadataReplace(t *testing.T) {
metadata := make(MD)
metadata.Set("foo", "1", "2")
metadata.Set("foo", "3", "4")
if list, ok := metadata.Get("foo"); !ok {
t.Error("key not found")
} else if len(list) != 2 {
t.Errorf("unexpected number of values: %d", len(list))
} else if list[0] != "3" {
t.Errorf("invalid metadata value at 0: %s", list[0])
} else if list[1] != "4" {
t.Errorf("invalid metadata value at 1: %s", list[1])
}
}
func TestMetadataAppend(t *testing.T) {
metadata := make(MD)
metadata.Set("foo", "1")
metadata.Append("foo", "2")
metadata.Append("bar", "3")
if list, ok := metadata.Get("foo"); !ok {
t.Error("key not found")
} else if len(list) != 2 {
t.Errorf("unexpected number of values: %d", len(list))
} else if list[0] != "1" {
t.Errorf("invalid metadata value at 0: %s", list[0])
} else if list[1] != "2" {
t.Errorf("invalid metadata value at 1: %s", list[1])
}
if list, ok := metadata.Get("bar"); !ok {
t.Error("key not found")
} else if list[0] != "3" {
t.Errorf("invalid value: %s", list[0])
}
}
func TestMetadataContext(t *testing.T) {
metadata := make(MD)
metadata.Set("foo", "bar")
ctx := WithMetadata(context.Background(), metadata)
if bar, ok := GetMetadataValue(ctx, "foo"); !ok {
t.Error("metadata not found")
} else if bar != "bar" {
t.Errorf("invalid metadata value: %q", bar)
}
}
func TestMetadataClone(t *testing.T) {
var metadata MD
m2 := metadata.Clone()
if m2 != nil {
t.Error("MD.Clone() on nil metadata should return nil")
}
metadata = MD{"nil": nil, "foo": {"bar"}, "baz": {"qux", "quxx"}}
m2 = metadata.Clone()
if len(metadata) != len(m2) {
t.Errorf("unexpected number of keys: %d, expected: %d", len(m2), len(metadata))
}
for k, v := range metadata {
v2, ok := m2[k]
if !ok {
t.Errorf("key not found: %s", k)
}
if v == nil && v2 == nil {
continue
}
if v == nil || v2 == nil {
t.Errorf("unexpected nil value: %v, expected: %v", v2, v)
}
if len(v) != len(v2) {
t.Errorf("unexpected number of values: %d, expected: %d", len(v2), len(v))
}
for i := range v {
if v[i] != v2[i] {
t.Errorf("unexpected value: %s, expected: %s", v2[i], v[i])
}
}
}
}
func TestMetadataCloneConcurrent(t *testing.T) {
metadata := make(MD)
metadata.Set("foo", "bar")
var wg sync.WaitGroup
for i := 0; i < 20; i++ {
wg.Add(1)
go func() {
defer wg.Done()
m2 := metadata.Clone()
m2.Set("foo", "baz")
}()
}
wg.Wait()
// Concurrent modification should clone the metadata first to avoid panic
// due to concurrent map writes.
if val, ok := metadata.Get("foo"); !ok {
t.Error("metadata not found")
} else if val[0] != "bar" {
t.Errorf("invalid metadata value: %q", val[0])
}
}
func simpleClone(src MD) MD {
md := MD{}
for k, v := range src {
md[k] = append(md[k], v...)
}
return md
}
func BenchmarkMetadataClone(b *testing.B) {
for _, sz := range []int{5, 10, 20, 50} {
b.Run(fmt.Sprintf("size=%d", sz), func(b *testing.B) {
metadata := make(MD)
for i := 0; i < sz; i++ {
metadata.Set("foo"+fmt.Sprint(i), "bar"+fmt.Sprint(i))
}
for i := 0; i < b.N; i++ {
_ = metadata.Clone()
}
})
}
}
func BenchmarkSimpleMetadataClone(b *testing.B) {
for _, sz := range []int{5, 10, 20, 50} {
b.Run(fmt.Sprintf("size=%d", sz), func(b *testing.B) {
metadata := make(MD)
for i := 0; i < sz; i++ {
metadata.Set("foo"+fmt.Sprint(i), "bar"+fmt.Sprint(i))
}
for i := 0; i < b.N; i++ {
_ = simpleClone(metadata)
}
})
}
}