2016-12-14 18:41:48 +03:00
|
|
|
// Package metadata is a way of defining message headers
|
2016-01-28 20:55:28 +03:00
|
|
|
package metadata
|
|
|
|
|
|
|
|
import (
|
2018-03-03 14:53:52 +03:00
|
|
|
"context"
|
2020-09-30 16:14:54 +03:00
|
|
|
"net/textproto"
|
2021-01-21 18:35:31 +03:00
|
|
|
"sort"
|
2016-01-28 20:55:28 +03:00
|
|
|
)
|
|
|
|
|
2020-04-12 13:16:08 +03:00
|
|
|
type metadataKey struct{}
|
2016-01-28 20:55:28 +03:00
|
|
|
|
2016-01-31 00:17:44 +03:00
|
|
|
// Metadata is our way of representing request headers internally.
|
|
|
|
// They're used at the RPC level and translate back and forth
|
|
|
|
// from Transport headers.
|
2016-01-28 20:55:28 +03:00
|
|
|
type Metadata map[string]string
|
|
|
|
|
2020-10-12 17:17:59 +03:00
|
|
|
var (
|
2020-11-03 01:08:23 +03:00
|
|
|
// DefaultMetadataSize used when need to init new Metadata
|
2020-10-12 17:17:59 +03:00
|
|
|
DefaultMetadataSize = 6
|
|
|
|
)
|
|
|
|
|
2021-01-21 18:35:31 +03:00
|
|
|
type Iterator struct {
|
|
|
|
cur int
|
|
|
|
cnt int
|
|
|
|
keys []string
|
|
|
|
md Metadata
|
|
|
|
}
|
|
|
|
|
|
|
|
func (iter *Iterator) Next(k, v *string) bool {
|
|
|
|
if iter.cur+1 > iter.cnt {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
*k = iter.keys[iter.cur]
|
|
|
|
*v = iter.md[*k]
|
|
|
|
iter.cur++
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
// Iterate returns run user func with map key, val sorted by key
|
|
|
|
func (md Metadata) Iterator() *Iterator {
|
|
|
|
iter := &Iterator{md: md, cnt: len(md)}
|
|
|
|
iter.keys = make([]string, 0, iter.cnt)
|
|
|
|
for k := range md {
|
|
|
|
iter.keys = append(iter.keys, k)
|
|
|
|
}
|
|
|
|
sort.Strings(iter.keys)
|
|
|
|
return iter
|
|
|
|
}
|
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// Get returns value from metadata by key
|
2020-03-26 21:50:00 +03:00
|
|
|
func (md Metadata) Get(key string) (string, bool) {
|
2020-09-30 16:14:54 +03:00
|
|
|
// fast path
|
2020-03-26 21:50:00 +03:00
|
|
|
val, ok := md[key]
|
2020-09-30 16:14:54 +03:00
|
|
|
if !ok {
|
|
|
|
// slow path
|
|
|
|
val, ok = md[textproto.CanonicalMIMEHeaderKey(key)]
|
2020-03-26 21:50:00 +03:00
|
|
|
}
|
|
|
|
return val, ok
|
|
|
|
}
|
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// Set is used to store value in metadata
|
2020-04-12 13:17:23 +03:00
|
|
|
func (md Metadata) Set(key, val string) {
|
2020-09-30 16:14:54 +03:00
|
|
|
md[textproto.CanonicalMIMEHeaderKey(key)] = val
|
2020-04-12 13:17:23 +03:00
|
|
|
}
|
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// Del is used to remove value from metadata
|
2020-09-30 16:14:54 +03:00
|
|
|
func (md Metadata) Del(key string) {
|
2020-09-30 16:21:47 +03:00
|
|
|
// fast path
|
|
|
|
if _, ok := md[key]; ok {
|
|
|
|
delete(md, key)
|
|
|
|
} else {
|
|
|
|
// slow path
|
|
|
|
delete(md, textproto.CanonicalMIMEHeaderKey(key))
|
|
|
|
}
|
2020-03-31 22:55:33 +03:00
|
|
|
}
|
|
|
|
|
2019-10-26 01:27:59 +03:00
|
|
|
// Copy makes a copy of the metadata
|
2019-01-17 12:40:49 +03:00
|
|
|
func Copy(md Metadata) Metadata {
|
2020-10-12 17:20:52 +03:00
|
|
|
nmd := New(len(md))
|
2020-10-01 16:00:01 +03:00
|
|
|
for key, val := range md {
|
|
|
|
nmd.Set(key, val)
|
2019-01-17 12:40:49 +03:00
|
|
|
}
|
2020-09-30 16:14:54 +03:00
|
|
|
return nmd
|
2019-01-17 12:40:49 +03:00
|
|
|
}
|
|
|
|
|
2020-12-08 00:38:37 +03:00
|
|
|
// Del deletes key from metadata
|
2020-09-30 16:14:54 +03:00
|
|
|
func Del(ctx context.Context, key string) context.Context {
|
|
|
|
md, ok := FromContext(ctx)
|
|
|
|
if !ok {
|
2020-10-12 17:20:52 +03:00
|
|
|
md = New(0)
|
2020-09-30 16:14:54 +03:00
|
|
|
}
|
|
|
|
md.Del(key)
|
|
|
|
return context.WithValue(ctx, metadataKey{}, md)
|
2020-03-31 22:55:33 +03:00
|
|
|
}
|
|
|
|
|
2020-02-21 23:04:47 +03:00
|
|
|
// Set add key with val to metadata
|
2020-09-30 16:14:54 +03:00
|
|
|
func Set(ctx context.Context, key, val string) context.Context {
|
2020-02-21 23:04:47 +03:00
|
|
|
md, ok := FromContext(ctx)
|
|
|
|
if !ok {
|
2020-10-12 17:20:52 +03:00
|
|
|
md = New(0)
|
2020-02-21 23:04:47 +03:00
|
|
|
}
|
2020-09-30 16:14:54 +03:00
|
|
|
md.Set(key, val)
|
2020-04-12 13:16:08 +03:00
|
|
|
return context.WithValue(ctx, metadataKey{}, md)
|
2020-02-21 23:04:47 +03:00
|
|
|
}
|
|
|
|
|
2019-11-11 12:13:02 +03:00
|
|
|
// Get returns a single value from metadata in the context
|
|
|
|
func Get(ctx context.Context, key string) (string, bool) {
|
|
|
|
md, ok := FromContext(ctx)
|
|
|
|
if !ok {
|
|
|
|
return "", ok
|
|
|
|
}
|
2020-09-30 16:14:54 +03:00
|
|
|
return md.Get(key)
|
2019-11-11 12:13:02 +03:00
|
|
|
}
|
|
|
|
|
2020-10-12 17:17:59 +03:00
|
|
|
// New return new sized metadata
|
|
|
|
func New(size int) Metadata {
|
|
|
|
if size == 0 {
|
|
|
|
size = DefaultMetadataSize
|
|
|
|
}
|
|
|
|
return make(Metadata, size)
|
|
|
|
}
|
2021-01-23 00:09:07 +03:00
|
|
|
|
|
|
|
// Merge merges metadata to existing metadata, overwriting if specified
|
|
|
|
func Merge(omd Metadata, mmd Metadata, overwrite bool) Metadata {
|
|
|
|
nmd := Copy(omd)
|
|
|
|
for key, val := range mmd {
|
|
|
|
if _, ok := nmd[key]; ok && !overwrite {
|
|
|
|
// skip
|
|
|
|
} else if val != "" {
|
|
|
|
nmd.Set(key, val)
|
|
|
|
} else {
|
|
|
|
nmd.Del(key)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nmd
|
|
|
|
}
|