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"
|
2019-12-31 13:37:29 +00:00
|
|
|
"strings"
|
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
|
|
|
|
|
2020-03-26 18:50:00 +00:00
|
|
|
func (md Metadata) Get(key string) (string, bool) {
|
|
|
|
// attempt to get as is
|
|
|
|
val, ok := md[key]
|
|
|
|
if ok {
|
|
|
|
return val, ok
|
|
|
|
}
|
|
|
|
|
|
|
|
// attempt to get lower case
|
|
|
|
val, ok = md[strings.Title(key)]
|
|
|
|
return val, ok
|
|
|
|
}
|
|
|
|
|
2020-04-12 11:17:23 +01:00
|
|
|
func (md Metadata) Set(key, val string) {
|
|
|
|
md[key] = val
|
|
|
|
}
|
|
|
|
|
2020-03-31 23:39:18 +03:00
|
|
|
func (md Metadata) Delete(key string) {
|
2020-03-31 22:55:33 +03:00
|
|
|
// delete key as-is
|
|
|
|
delete(md, key)
|
|
|
|
// delete also Title key
|
|
|
|
delete(md, strings.Title(key))
|
|
|
|
}
|
|
|
|
|
2022-09-30 16:27:07 +02:00
|
|
|
// Copy makes a copy of the metadata.
|
2019-01-17 09:40:49 +00:00
|
|
|
func Copy(md Metadata) Metadata {
|
2020-04-08 12:50:19 +03:00
|
|
|
cmd := make(Metadata, len(md))
|
2019-01-17 09:40:49 +00:00
|
|
|
for k, v := range md {
|
|
|
|
cmd[k] = v
|
|
|
|
}
|
|
|
|
return cmd
|
|
|
|
}
|
|
|
|
|
2022-09-30 16:27:07 +02:00
|
|
|
// Delete key from metadata.
|
2020-03-31 23:39:18 +03:00
|
|
|
func Delete(ctx context.Context, k string) context.Context {
|
2020-03-31 22:55:33 +03:00
|
|
|
return Set(ctx, k, "")
|
|
|
|
}
|
|
|
|
|
2022-09-30 16:27:07 +02:00
|
|
|
// Set add key with val to metadata.
|
2020-02-21 23:04:47 +03:00
|
|
|
func Set(ctx context.Context, k, v string) context.Context {
|
|
|
|
md, ok := FromContext(ctx)
|
|
|
|
if !ok {
|
|
|
|
md = make(Metadata)
|
|
|
|
}
|
2020-03-31 22:55:33 +03:00
|
|
|
if v == "" {
|
|
|
|
delete(md, k)
|
|
|
|
} else {
|
|
|
|
md[k] = v
|
|
|
|
}
|
2020-04-12 11:16:08 +01:00
|
|
|
return context.WithValue(ctx, metadataKey{}, md)
|
2020-02-21 23:04:47 +03:00
|
|
|
}
|
|
|
|
|
2022-09-30 16:27:07 +02:00
|
|
|
// Get returns a single value from metadata in the context.
|
2019-11-11 09:13:02 +00:00
|
|
|
func Get(ctx context.Context, key string) (string, bool) {
|
|
|
|
md, ok := FromContext(ctx)
|
|
|
|
if !ok {
|
|
|
|
return "", ok
|
|
|
|
}
|
2019-12-31 13:37:29 +00:00
|
|
|
// attempt to get as is
|
2019-11-11 09:13:02 +00:00
|
|
|
val, ok := md[key]
|
2019-12-31 13:37:29 +00:00
|
|
|
if ok {
|
|
|
|
return val, ok
|
|
|
|
}
|
|
|
|
|
|
|
|
// attempt to get lower case
|
|
|
|
val, ok = md[strings.Title(key)]
|
|
|
|
|
2019-11-11 09:13:02 +00:00
|
|
|
return val, ok
|
|
|
|
}
|
|
|
|
|
2022-09-30 16:27:07 +02: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)
|
2019-12-31 13:37:29 +00:00
|
|
|
if !ok {
|
|
|
|
return nil, ok
|
|
|
|
}
|
|
|
|
|
|
|
|
// capitalise all values
|
2020-04-08 12:50:19 +03:00
|
|
|
newMD := make(Metadata, len(md))
|
2019-12-31 13:37:29 +00:00
|
|
|
for k, v := range md {
|
|
|
|
newMD[strings.Title(k)] = v
|
|
|
|
}
|
|
|
|
|
|
|
|
return newMD, ok
|
2016-01-28 17:55:28 +00:00
|
|
|
}
|
|
|
|
|
2022-09-30 16:27:07 +02: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 {
|
2020-04-12 11:16:08 +01:00
|
|
|
return context.WithValue(ctx, metadataKey{}, md)
|
2016-01-28 17:55:28 +00:00
|
|
|
}
|
2019-10-24 17:51:54 -07:00
|
|
|
|
2022-09-30 16:27:07 +02:00
|
|
|
// MergeContext merges metadata to existing metadata, overwriting if specified.
|
2019-10-25 23:27:59 +01:00
|
|
|
func MergeContext(ctx context.Context, patchMd Metadata, overwrite bool) context.Context {
|
2020-02-06 17:22:16 +00:00
|
|
|
if ctx == nil {
|
|
|
|
ctx = context.Background()
|
|
|
|
}
|
2020-04-12 11:16:08 +01:00
|
|
|
md, _ := ctx.Value(metadataKey{}).(Metadata)
|
2020-04-08 12:50:19 +03:00
|
|
|
cmd := make(Metadata, len(md))
|
2019-10-24 17:51:54 -07:00
|
|
|
for k, v := range md {
|
|
|
|
cmd[k] = v
|
|
|
|
}
|
|
|
|
for k, v := range patchMd {
|
2019-10-25 08:27:28 -07:00
|
|
|
if _, ok := cmd[k]; ok && !overwrite {
|
|
|
|
// skip
|
2020-03-31 22:55:33 +03:00
|
|
|
} else if v != "" {
|
2019-10-25 08:27:28 -07:00
|
|
|
cmd[k] = v
|
2020-03-31 22:55:33 +03:00
|
|
|
} else {
|
|
|
|
delete(cmd, k)
|
2019-10-25 08:27:28 -07:00
|
|
|
}
|
2019-10-24 17:51:54 -07:00
|
|
|
}
|
2020-04-12 11:16:08 +01:00
|
|
|
return context.WithValue(ctx, metadataKey{}, cmd)
|
2019-10-24 17:51:54 -07:00
|
|
|
}
|