mirror of https://github.com/hashicorp/consul
Fix uint8 conversion issues for service config response maps.
parent
446abd7aa2
commit
8f5b16ebaf
|
@ -6,7 +6,6 @@ import (
|
||||||
|
|
||||||
"github.com/hashicorp/consul/acl"
|
"github.com/hashicorp/consul/acl"
|
||||||
"github.com/hashicorp/consul/agent/structs"
|
"github.com/hashicorp/consul/agent/structs"
|
||||||
"github.com/hashicorp/consul/lib"
|
|
||||||
"github.com/hashicorp/consul/testrpc"
|
"github.com/hashicorp/consul/testrpc"
|
||||||
msgpackrpc "github.com/hashicorp/net-rpc-msgpackrpc"
|
msgpackrpc "github.com/hashicorp/net-rpc-msgpackrpc"
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
|
@ -681,15 +680,6 @@ func TestConfigEntry_ResolveServiceConfig(t *testing.T) {
|
||||||
}
|
}
|
||||||
var out structs.ServiceConfigResponse
|
var out structs.ServiceConfigResponse
|
||||||
require.NoError(msgpackrpc.CallWithCodec(codec, "ConfigEntry.ResolveServiceConfig", &args, &out))
|
require.NoError(msgpackrpc.CallWithCodec(codec, "ConfigEntry.ResolveServiceConfig", &args, &out))
|
||||||
// Hack to fix up the string encoding in the map[string]interface{}.
|
|
||||||
// msgpackRPC's codec doesn't use RawToString.
|
|
||||||
var err error
|
|
||||||
out.ProxyConfig, err = lib.MapWalk(out.ProxyConfig)
|
|
||||||
require.NoError(err)
|
|
||||||
for k := range out.UpstreamConfigs {
|
|
||||||
out.UpstreamConfigs[k], err = lib.MapWalk(out.UpstreamConfigs[k])
|
|
||||||
require.NoError(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
expected := structs.ServiceConfigResponse{
|
expected := structs.ServiceConfigResponse{
|
||||||
ProxyConfig: map[string]interface{}{
|
ProxyConfig: map[string]interface{}{
|
||||||
|
|
|
@ -419,6 +419,54 @@ type ServiceConfigResponse struct {
|
||||||
QueryMeta
|
QueryMeta
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// MarshalBinary writes ServiceConfigResponse as msgpack encoded. It's only here
|
||||||
|
// because we need custom decoding of the raw interface{} values.
|
||||||
|
func (r *ServiceConfigResponse) MarshalBinary() (data []byte, err error) {
|
||||||
|
// bs will grow if needed but allocate enough to avoid reallocation in common
|
||||||
|
// case.
|
||||||
|
bs := make([]byte, 128)
|
||||||
|
enc := codec.NewEncoderBytes(&bs, msgpackHandle)
|
||||||
|
|
||||||
|
type Alias ServiceConfigResponse
|
||||||
|
|
||||||
|
if err := enc.Encode((*Alias)(r)); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return bs, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnmarshalBinary decodes msgpack encoded ServiceConfigResponse. It used
|
||||||
|
// default msgpack encoding but fixes up the uint8 strings and other problems we
|
||||||
|
// have with encoding map[string]interface{}.
|
||||||
|
func (r *ServiceConfigResponse) UnmarshalBinary(data []byte) error {
|
||||||
|
dec := codec.NewDecoderBytes(data, msgpackHandle)
|
||||||
|
|
||||||
|
type Alias ServiceConfigResponse
|
||||||
|
var a Alias
|
||||||
|
|
||||||
|
if err := dec.Decode(&a); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
*r = ServiceConfigResponse(a)
|
||||||
|
|
||||||
|
var err error
|
||||||
|
|
||||||
|
// Fix strings and maps in the returned maps
|
||||||
|
r.ProxyConfig, err = lib.MapWalk(r.ProxyConfig)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
for k := range r.UpstreamConfigs {
|
||||||
|
r.UpstreamConfigs[k], err = lib.MapWalk(r.UpstreamConfigs[k])
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// ConfigEntryResponse returns a single ConfigEntry
|
// ConfigEntryResponse returns a single ConfigEntry
|
||||||
type ConfigEntryResponse struct {
|
type ConfigEntryResponse struct {
|
||||||
Entry ConfigEntry
|
Entry ConfigEntry
|
||||||
|
|
|
@ -1,8 +1,10 @@
|
||||||
package structs
|
package structs
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/hashicorp/go-msgpack/codec"
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -100,3 +102,44 @@ func TestDecodeConfigEntry(t *testing.T) {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestServiceConfigResponse_MsgPack(t *testing.T) {
|
||||||
|
// TODO(banks) lib.MapWalker doesn't actually fix the map[interface{}] issue
|
||||||
|
// it claims to in docs yet. When it does uncomment those cases below.
|
||||||
|
a := ServiceConfigResponse{
|
||||||
|
ProxyConfig: map[string]interface{}{
|
||||||
|
"string": "foo",
|
||||||
|
// "map": map[string]interface{}{
|
||||||
|
// "baz": "bar",
|
||||||
|
// },
|
||||||
|
},
|
||||||
|
UpstreamConfigs: map[string]map[string]interface{}{
|
||||||
|
"a": map[string]interface{}{
|
||||||
|
"string": "aaaa",
|
||||||
|
// "map": map[string]interface{}{
|
||||||
|
// "baz": "aa",
|
||||||
|
// },
|
||||||
|
},
|
||||||
|
"b": map[string]interface{}{
|
||||||
|
"string": "bbbb",
|
||||||
|
// "map": map[string]interface{}{
|
||||||
|
// "baz": "bb",
|
||||||
|
// },
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var buf bytes.Buffer
|
||||||
|
|
||||||
|
// Encode as msgPack using a regular handle i.e. NOT one with RawAsString
|
||||||
|
// since our RPC codec doesn't use that.
|
||||||
|
enc := codec.NewEncoder(&buf, msgpackHandle)
|
||||||
|
require.NoError(t, enc.Encode(&a))
|
||||||
|
|
||||||
|
var b ServiceConfigResponse
|
||||||
|
|
||||||
|
dec := codec.NewDecoder(&buf, msgpackHandle)
|
||||||
|
require.NoError(t, dec.Decode(&b))
|
||||||
|
|
||||||
|
require.Equal(t, a, b)
|
||||||
|
}
|
||||||
|
|
|
@ -38,6 +38,20 @@ func TestMapWalk(t *testing.T) {
|
||||||
},
|
},
|
||||||
unexpected: true,
|
unexpected: true,
|
||||||
},
|
},
|
||||||
|
// TODO(banks): despite the doc comment, MapWalker doesn't actually fix
|
||||||
|
// these cases yet. Do that in a later PR.
|
||||||
|
// "map iface": tcase{
|
||||||
|
// input: map[string]interface{}{
|
||||||
|
// "foo": map[interface{}]interface{}{
|
||||||
|
// "bar": "baz",
|
||||||
|
// },
|
||||||
|
// },
|
||||||
|
// expected: map[string]interface{}{
|
||||||
|
// "foo": map[string]interface{}{
|
||||||
|
// "bar": "baz",
|
||||||
|
// },
|
||||||
|
// },
|
||||||
|
// },
|
||||||
}
|
}
|
||||||
|
|
||||||
for name, tcase := range cases {
|
for name, tcase := range cases {
|
||||||
|
|
Loading…
Reference in New Issue