micro/metadata/metadata.go

125 lines
2.6 KiB
Go
Raw Normal View History

2016-12-14 15:41:48 +00:00
// Package metadata is a way of defining message headers
2016-01-28 17:55:28 +00:00
package metadata
import (
2018-03-03 11:53:52 +00:00
"context"
"net/textproto"
2016-01-28 17:55:28 +00:00
)
2020-04-12 11:16:08 +01:00
type metadataKey struct{}
2016-01-28 17:55:28 +00:00
2016-01-30 21:17:44 +00: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 17:55:28 +00:00
type Metadata map[string]string
var (
DefaultMetadataSize = 6
)
2020-03-26 18:50:00 +00:00
func (md Metadata) Get(key string) (string, bool) {
// fast path
2020-03-26 18:50:00 +00:00
val, ok := md[key]
if !ok {
// slow path
val, ok = md[textproto.CanonicalMIMEHeaderKey(key)]
2020-03-26 18:50:00 +00:00
}
return val, ok
}
2020-04-12 11:17:23 +01:00
func (md Metadata) Set(key, val string) {
md[textproto.CanonicalMIMEHeaderKey(key)] = val
2020-04-12 11:17:23 +01:00
}
func (md Metadata) Del(key string) {
// fast path
if _, ok := md[key]; ok {
delete(md, key)
} else {
// slow path
delete(md, textproto.CanonicalMIMEHeaderKey(key))
}
}
2019-10-25 23:27:59 +01:00
// Copy makes a copy of the metadata
2019-01-17 09:40:49 +00:00
func Copy(md Metadata) Metadata {
nmd := New(len(md))
for key, val := range md {
nmd.Set(key, val)
2019-01-17 09:40:49 +00:00
}
return nmd
2019-01-17 09:40:49 +00:00
}
func Del(ctx context.Context, key string) context.Context {
md, ok := FromContext(ctx)
if !ok {
md = New(0)
}
md.Del(key)
return context.WithValue(ctx, metadataKey{}, md)
}
// Set add key with val to metadata
func Set(ctx context.Context, key, val string) context.Context {
md, ok := FromContext(ctx)
if !ok {
md = New(0)
}
md.Set(key, val)
2020-04-12 11:16:08 +01:00
return context.WithValue(ctx, metadataKey{}, md)
}
// 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
}
return md.Get(key)
}
2019-10-25 23:27:59 +01:00
// FromContext returns metadata from the given context
2016-01-28 17:55:28 +00:00
func FromContext(ctx context.Context) (Metadata, bool) {
2020-04-12 11:16:08 +01:00
md, ok := ctx.Value(metadataKey{}).(Metadata)
if !ok {
return nil, ok
}
nmd := Copy(md)
return nmd, ok
2016-01-28 17:55:28 +00:00
}
// New return new sized metadata
func New(size int) Metadata {
if size == 0 {
size = DefaultMetadataSize
}
return make(Metadata, size)
}
2019-10-25 23:27:59 +01:00
// NewContext creates a new context with the given metadata
2016-01-28 17:55:28 +00:00
func NewContext(ctx context.Context, md Metadata) context.Context {
return context.WithValue(ctx, metadataKey{}, Copy(md))
2016-01-28 17:55:28 +00:00
}
2019-10-25 23:27:59 +01:00
// MergeContext merges metadata to existing metadata, overwriting if specified
func MergeContext(ctx context.Context, pmd Metadata, overwrite bool) context.Context {
if ctx == nil {
ctx = context.Background()
}
md, ok := FromContext(ctx)
if !ok {
return context.WithValue(ctx, metadataKey{}, Copy(pmd))
}
nmd := Copy(md)
for key, val := range pmd {
if _, ok := nmd[key]; ok && !overwrite {
2019-10-25 08:27:28 -07:00
// skip
} else if val != "" {
nmd.Set(key, val)
} else {
nmd.Del(key)
2019-10-25 08:27:28 -07:00
}
}
return context.WithValue(ctx, metadataKey{}, nmd)
}