1
0
mirror of https://github.com/go-micro/go-micro.git synced 2025-01-05 10:20:53 +02:00
go-micro/metadata/metadata.go

80 lines
1.7 KiB
Go
Raw Normal View History

2016-12-14 17:41:48 +02:00
// Package metadata is a way of defining message headers
2016-01-28 19:55:28 +02:00
package metadata
import (
2018-03-03 13:53:52 +02:00
"context"
2019-12-31 15:37:29 +02:00
"strings"
2016-01-28 19:55:28 +02:00
)
2016-01-28 20:24:56 +02:00
type metaKey struct{}
2016-01-28 19:55:28 +02:00
2016-01-30 23:17:44 +02: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 19:55:28 +02:00
type Metadata map[string]string
2019-10-26 00:27:59 +02:00
// Copy makes a copy of the metadata
2019-01-17 11:40:49 +02:00
func Copy(md Metadata) Metadata {
cmd := make(Metadata)
for k, v := range md {
cmd[k] = v
}
return cmd
}
// 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
}
2019-12-31 15:37:29 +02:00
// attempt to get as is
val, ok := md[key]
2019-12-31 15:37:29 +02:00
if ok {
return val, ok
}
// attempt to get lower case
val, ok = md[strings.Title(key)]
return val, ok
}
2019-10-26 00:27:59 +02:00
// FromContext returns metadata from the given context
2016-01-28 19:55:28 +02:00
func FromContext(ctx context.Context) (Metadata, bool) {
2016-01-28 20:24:56 +02:00
md, ok := ctx.Value(metaKey{}).(Metadata)
2019-12-31 15:37:29 +02:00
if !ok {
return nil, ok
}
// capitalise all values
newMD := make(map[string]string)
for k, v := range md {
newMD[strings.Title(k)] = v
}
return newMD, ok
2016-01-28 19:55:28 +02:00
}
2019-10-26 00:27:59 +02:00
// NewContext creates a new context with the given metadata
2016-01-28 19:55:28 +02:00
func NewContext(ctx context.Context, md Metadata) context.Context {
2016-01-28 20:24:56 +02:00
return context.WithValue(ctx, metaKey{}, md)
2016-01-28 19:55:28 +02:00
}
2019-10-26 00:27:59 +02:00
// MergeContext merges metadata to existing metadata, overwriting if specified
func MergeContext(ctx context.Context, patchMd Metadata, overwrite bool) context.Context {
md, _ := ctx.Value(metaKey{}).(Metadata)
cmd := make(Metadata)
for k, v := range md {
cmd[k] = v
}
for k, v := range patchMd {
2019-10-25 17:27:28 +02:00
if _, ok := cmd[k]; ok && !overwrite {
// skip
} else {
cmd[k] = v
}
}
return context.WithValue(ctx, metaKey{}, cmd)
}