mirror of
https://github.com/k3s-io/kubernetes.git
synced 2025-08-03 09:22:44 +00:00
Merge pull request #125790 from benluddy/cbor-fieldsv1
KEP-4222: Support either JSON or CBOR in FieldsV1.
This commit is contained in:
commit
ccbbbc0f1f
@ -24,8 +24,10 @@ import (
|
|||||||
|
|
||||||
"k8s.io/apimachinery/pkg/fields"
|
"k8s.io/apimachinery/pkg/fields"
|
||||||
"k8s.io/apimachinery/pkg/labels"
|
"k8s.io/apimachinery/pkg/labels"
|
||||||
|
cbor "k8s.io/apimachinery/pkg/runtime/serializer/cbor/direct"
|
||||||
"k8s.io/apimachinery/pkg/selection"
|
"k8s.io/apimachinery/pkg/selection"
|
||||||
"k8s.io/apimachinery/pkg/types"
|
"k8s.io/apimachinery/pkg/types"
|
||||||
|
utiljson "k8s.io/apimachinery/pkg/util/json"
|
||||||
)
|
)
|
||||||
|
|
||||||
// LabelSelectorAsSelector converts the LabelSelector api type into a struct that implements
|
// LabelSelectorAsSelector converts the LabelSelector api type into a struct that implements
|
||||||
@ -280,13 +282,20 @@ func (f FieldsV1) MarshalJSON() ([]byte, error) {
|
|||||||
if f.Raw == nil {
|
if f.Raw == nil {
|
||||||
return []byte("null"), nil
|
return []byte("null"), nil
|
||||||
}
|
}
|
||||||
|
if f.getContentType() == fieldsV1InvalidOrValidCBORObject {
|
||||||
|
var u map[string]interface{}
|
||||||
|
if err := cbor.Unmarshal(f.Raw, &u); err != nil {
|
||||||
|
return nil, fmt.Errorf("metav1.FieldsV1 cbor invalid: %w", err)
|
||||||
|
}
|
||||||
|
return utiljson.Marshal(u)
|
||||||
|
}
|
||||||
return f.Raw, nil
|
return f.Raw, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// UnmarshalJSON implements json.Unmarshaler
|
// UnmarshalJSON implements json.Unmarshaler
|
||||||
func (f *FieldsV1) UnmarshalJSON(b []byte) error {
|
func (f *FieldsV1) UnmarshalJSON(b []byte) error {
|
||||||
if f == nil {
|
if f == nil {
|
||||||
return errors.New("metav1.Fields: UnmarshalJSON on nil pointer")
|
return errors.New("metav1.FieldsV1: UnmarshalJSON on nil pointer")
|
||||||
}
|
}
|
||||||
if !bytes.Equal(b, []byte("null")) {
|
if !bytes.Equal(b, []byte("null")) {
|
||||||
f.Raw = append(f.Raw[0:0], b...)
|
f.Raw = append(f.Raw[0:0], b...)
|
||||||
@ -296,3 +305,75 @@ func (f *FieldsV1) UnmarshalJSON(b []byte) error {
|
|||||||
|
|
||||||
var _ json.Marshaler = FieldsV1{}
|
var _ json.Marshaler = FieldsV1{}
|
||||||
var _ json.Unmarshaler = &FieldsV1{}
|
var _ json.Unmarshaler = &FieldsV1{}
|
||||||
|
|
||||||
|
func (f FieldsV1) MarshalCBOR() ([]byte, error) {
|
||||||
|
if f.Raw == nil {
|
||||||
|
return cbor.Marshal(nil)
|
||||||
|
}
|
||||||
|
if f.getContentType() == fieldsV1InvalidOrValidJSONObject {
|
||||||
|
var u map[string]interface{}
|
||||||
|
if err := utiljson.Unmarshal(f.Raw, &u); err != nil {
|
||||||
|
return nil, fmt.Errorf("metav1.FieldsV1 json invalid: %w", err)
|
||||||
|
}
|
||||||
|
return cbor.Marshal(u)
|
||||||
|
}
|
||||||
|
return f.Raw, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var cborNull = []byte{0xf6}
|
||||||
|
|
||||||
|
func (f *FieldsV1) UnmarshalCBOR(b []byte) error {
|
||||||
|
if f == nil {
|
||||||
|
return errors.New("metav1.FieldsV1: UnmarshalCBOR on nil pointer")
|
||||||
|
}
|
||||||
|
if !bytes.Equal(b, cborNull) {
|
||||||
|
f.Raw = append(f.Raw[0:0], b...)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
// fieldsV1InvalidOrEmpty indicates that a FieldsV1 either contains no raw bytes or its raw
|
||||||
|
// bytes don't represent an allowable value in any supported encoding.
|
||||||
|
fieldsV1InvalidOrEmpty = iota
|
||||||
|
|
||||||
|
// fieldsV1InvalidOrValidJSONObject indicates that a FieldV1 either contains raw bytes that
|
||||||
|
// are a valid JSON encoding of an allowable value or don't represent an allowable value in
|
||||||
|
// any supported encoding.
|
||||||
|
fieldsV1InvalidOrValidJSONObject
|
||||||
|
|
||||||
|
// fieldsV1InvalidOrValidCBORObject indicates that a FieldV1 either contains raw bytes that
|
||||||
|
// are a valid CBOR encoding of an allowable value or don't represent an allowable value in
|
||||||
|
// any supported encoding.
|
||||||
|
fieldsV1InvalidOrValidCBORObject
|
||||||
|
)
|
||||||
|
|
||||||
|
// getContentType returns one of fieldsV1InvalidOrEmpty, fieldsV1InvalidOrValidJSONObject,
|
||||||
|
// fieldsV1InvalidOrValidCBORObject based on the value of Raw.
|
||||||
|
//
|
||||||
|
// Raw can be encoded in JSON or CBOR and is only valid if it is empty, null, or an object (map)
|
||||||
|
// value. It is invalid if it contains a JSON string, number, boolean, or array. If Raw is nonempty
|
||||||
|
// and represents an allowable value, then the initial byte unambiguously distinguishes a
|
||||||
|
// JSON-encoded value from a CBOR-encoded value.
|
||||||
|
//
|
||||||
|
// A valid JSON-encoded value can begin with any of the four JSON whitespace characters, the first
|
||||||
|
// character 'n' of null, or '{' (0x09, 0x0a, 0x0d, 0x20, 0x6e, or 0x7b, respectively). A valid
|
||||||
|
// CBOR-encoded value can begin with the null simple value, an initial byte with major type "map",
|
||||||
|
// or, if a tag-enclosed map, an initial byte with major type "tag" (0xf6, 0xa0...0xbf, or
|
||||||
|
// 0xc6...0xdb). The two sets of valid initial bytes don't intersect.
|
||||||
|
func (f FieldsV1) getContentType() int {
|
||||||
|
if len(f.Raw) > 0 {
|
||||||
|
p := f.Raw[0]
|
||||||
|
switch p {
|
||||||
|
case 'n', '{', '\t', '\r', '\n', ' ':
|
||||||
|
return fieldsV1InvalidOrValidJSONObject
|
||||||
|
case 0xf6: // null
|
||||||
|
return fieldsV1InvalidOrValidCBORObject
|
||||||
|
default:
|
||||||
|
if p >= 0xa0 && p <= 0xbf /* map */ || p >= 0xc6 && p <= 0xdb /* tag */ {
|
||||||
|
return fieldsV1InvalidOrValidCBORObject
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return fieldsV1InvalidOrEmpty
|
||||||
|
}
|
||||||
|
@ -249,3 +249,245 @@ func TestSetMetaDataLabel(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestFieldsV1MarshalJSON(t *testing.T) {
|
||||||
|
for _, tc := range []struct {
|
||||||
|
Name string
|
||||||
|
FieldsV1 FieldsV1
|
||||||
|
Want []byte
|
||||||
|
Error string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
Name: "nil encodes as json null",
|
||||||
|
FieldsV1: FieldsV1{},
|
||||||
|
Want: []byte(`null`),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "empty invalid json is returned as-is",
|
||||||
|
FieldsV1: FieldsV1{Raw: []byte{}},
|
||||||
|
Want: []byte{},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "cbor null is transcoded to json null",
|
||||||
|
FieldsV1: FieldsV1{Raw: []byte{0xf6}}, // null
|
||||||
|
Want: []byte(`null`),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "valid non-map cbor and valid non-object json is returned as-is",
|
||||||
|
FieldsV1: FieldsV1{Raw: []byte{0x30}},
|
||||||
|
Want: []byte{0x30}, // Valid CBOR encoding of -17 and JSON encoding of 0!
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "self-described cbor map is transcoded to json map",
|
||||||
|
FieldsV1: FieldsV1{Raw: []byte{0xd9, 0xd9, 0xf7, 0xa1, 0x43, 'f', 'o', 'o', 0x43, 'b', 'a', 'r'}}, // 55799({"foo":"bar"})
|
||||||
|
Want: []byte(`{"foo":"bar"}`),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "json object is returned as-is",
|
||||||
|
FieldsV1: FieldsV1{Raw: []byte(" \t\r\n{\"foo\":\"bar\"}")},
|
||||||
|
Want: []byte(" \t\r\n{\"foo\":\"bar\"}"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "invalid json is returned as-is",
|
||||||
|
FieldsV1: FieldsV1{Raw: []byte(`{{`)},
|
||||||
|
Want: []byte(`{{`),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "invalid cbor fails to transcode to json",
|
||||||
|
FieldsV1: FieldsV1{Raw: []byte{0xa1}},
|
||||||
|
Error: "metav1.FieldsV1 cbor invalid: unexpected EOF",
|
||||||
|
},
|
||||||
|
} {
|
||||||
|
t.Run(tc.Name, func(t *testing.T) {
|
||||||
|
got, err := tc.FieldsV1.MarshalJSON()
|
||||||
|
if err != nil {
|
||||||
|
if tc.Error == "" {
|
||||||
|
t.Fatalf("unexpected error: %v", err)
|
||||||
|
}
|
||||||
|
if msg := err.Error(); msg != tc.Error {
|
||||||
|
t.Fatalf("expected error %q, got %q", tc.Error, msg)
|
||||||
|
}
|
||||||
|
} else if tc.Error != "" {
|
||||||
|
t.Fatalf("expected error %q, got nil", tc.Error)
|
||||||
|
}
|
||||||
|
if diff := cmp.Diff(tc.Want, got); diff != "" {
|
||||||
|
t.Errorf("unexpected diff:\n%s", diff)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFieldsV1MarshalCBOR(t *testing.T) {
|
||||||
|
for _, tc := range []struct {
|
||||||
|
Name string
|
||||||
|
FieldsV1 FieldsV1
|
||||||
|
Want []byte
|
||||||
|
Error string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
Name: "nil encodes as cbor null",
|
||||||
|
FieldsV1: FieldsV1{},
|
||||||
|
Want: []byte{0xf6}, // null
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "empty invalid cbor is returned as-is",
|
||||||
|
FieldsV1: FieldsV1{Raw: []byte{}},
|
||||||
|
Want: []byte{},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "json null is transcoded to cbor null",
|
||||||
|
FieldsV1: FieldsV1{Raw: []byte(`null`)},
|
||||||
|
Want: []byte{0xf6}, // null
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "valid non-map cbor and valid non-object json is returned as-is",
|
||||||
|
FieldsV1: FieldsV1{Raw: []byte{0x30}},
|
||||||
|
Want: []byte{0x30}, // Valid CBOR encoding of -17 and JSON encoding of 0!
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "json object is transcoded to cbor map",
|
||||||
|
FieldsV1: FieldsV1{Raw: []byte(" \t\r\n{\"foo\":\"bar\"}")},
|
||||||
|
Want: []byte{0xa1, 0x43, 'f', 'o', 'o', 0x43, 'b', 'a', 'r'},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "self-described cbor map is returned as-is",
|
||||||
|
FieldsV1: FieldsV1{Raw: []byte{0xd9, 0xd9, 0xf7, 0xa1, 0x43, 'f', 'o', 'o', 0x43, 'b', 'a', 'r'}}, // 55799({"foo":"bar"})
|
||||||
|
Want: []byte{0xd9, 0xd9, 0xf7, 0xa1, 0x43, 'f', 'o', 'o', 0x43, 'b', 'a', 'r'}, // 55799({"foo":"bar"})
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "invalid json fails to transcode to cbor",
|
||||||
|
FieldsV1: FieldsV1{Raw: []byte(`{{`)},
|
||||||
|
Error: "metav1.FieldsV1 json invalid: invalid character '{' looking for beginning of object key string",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "invalid cbor is returned as-is",
|
||||||
|
FieldsV1: FieldsV1{Raw: []byte{0xa1}},
|
||||||
|
Want: []byte{0xa1},
|
||||||
|
},
|
||||||
|
} {
|
||||||
|
t.Run(tc.Name, func(t *testing.T) {
|
||||||
|
got, err := tc.FieldsV1.MarshalCBOR()
|
||||||
|
if err != nil {
|
||||||
|
if tc.Error == "" {
|
||||||
|
t.Fatalf("unexpected error: %v", err)
|
||||||
|
}
|
||||||
|
if msg := err.Error(); msg != tc.Error {
|
||||||
|
t.Fatalf("expected error %q, got %q", tc.Error, msg)
|
||||||
|
}
|
||||||
|
} else if tc.Error != "" {
|
||||||
|
t.Fatalf("expected error %q, got nil", tc.Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
if diff := cmp.Diff(tc.Want, got); diff != "" {
|
||||||
|
t.Errorf("unexpected diff:\n%s", diff)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFieldsV1UnmarshalJSON(t *testing.T) {
|
||||||
|
for _, tc := range []struct {
|
||||||
|
Name string
|
||||||
|
JSON []byte
|
||||||
|
Into *FieldsV1
|
||||||
|
Want *FieldsV1
|
||||||
|
Error string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
Name: "nil receiver returns error",
|
||||||
|
Into: nil,
|
||||||
|
Error: "metav1.FieldsV1: UnmarshalJSON on nil pointer",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "json null does not modify receiver", // conventional for json.Unmarshaler
|
||||||
|
JSON: []byte(`null`),
|
||||||
|
Into: &FieldsV1{Raw: []byte(`unmodified`)},
|
||||||
|
Want: &FieldsV1{Raw: []byte(`unmodified`)},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "valid input is copied verbatim",
|
||||||
|
JSON: []byte("{\"foo\":\"bar\"} \t\r\n"),
|
||||||
|
Into: &FieldsV1{},
|
||||||
|
Want: &FieldsV1{Raw: []byte("{\"foo\":\"bar\"} \t\r\n")},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "invalid input is copied verbatim",
|
||||||
|
JSON: []byte("{{"),
|
||||||
|
Into: &FieldsV1{},
|
||||||
|
Want: &FieldsV1{Raw: []byte("{{")},
|
||||||
|
},
|
||||||
|
} {
|
||||||
|
t.Run(tc.Name, func(t *testing.T) {
|
||||||
|
got := tc.Into.DeepCopy()
|
||||||
|
err := got.UnmarshalJSON(tc.JSON)
|
||||||
|
if err != nil {
|
||||||
|
if tc.Error == "" {
|
||||||
|
t.Fatalf("unexpected error: %v", err)
|
||||||
|
}
|
||||||
|
if msg := err.Error(); msg != tc.Error {
|
||||||
|
t.Fatalf("expected error %q, got %q", tc.Error, msg)
|
||||||
|
}
|
||||||
|
} else if tc.Error != "" {
|
||||||
|
t.Fatalf("expected error %q, got nil", tc.Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
if diff := cmp.Diff(tc.Want, got); diff != "" {
|
||||||
|
t.Errorf("unexpected diff:\n%s", diff)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFieldsV1UnmarshalCBOR(t *testing.T) {
|
||||||
|
for _, tc := range []struct {
|
||||||
|
Name string
|
||||||
|
CBOR []byte
|
||||||
|
Into *FieldsV1
|
||||||
|
Want *FieldsV1
|
||||||
|
Error string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
Name: "nil receiver returns error",
|
||||||
|
Into: nil,
|
||||||
|
Want: nil,
|
||||||
|
Error: "metav1.FieldsV1: UnmarshalCBOR on nil pointer",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "cbor null does not modify receiver",
|
||||||
|
CBOR: []byte{0xf6},
|
||||||
|
Into: &FieldsV1{Raw: []byte(`unmodified`)},
|
||||||
|
Want: &FieldsV1{Raw: []byte(`unmodified`)},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "valid input is copied verbatim",
|
||||||
|
CBOR: []byte{0xa1, 0x43, 'f', 'o', 'o', 0x43, 'b', 'a', 'r'},
|
||||||
|
Into: &FieldsV1{},
|
||||||
|
Want: &FieldsV1{Raw: []byte{0xa1, 0x43, 'f', 'o', 'o', 0x43, 'b', 'a', 'r'}},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "invalid input is copied verbatim",
|
||||||
|
CBOR: []byte{0xff}, // UnmarshalCBOR should never be called with malformed input, testing anyway.
|
||||||
|
Into: &FieldsV1{},
|
||||||
|
Want: &FieldsV1{Raw: []byte{0xff}},
|
||||||
|
},
|
||||||
|
} {
|
||||||
|
t.Run(tc.Name, func(t *testing.T) {
|
||||||
|
got := tc.Into.DeepCopy()
|
||||||
|
err := got.UnmarshalCBOR(tc.CBOR)
|
||||||
|
if err != nil {
|
||||||
|
if tc.Error == "" {
|
||||||
|
t.Fatalf("unexpected error: %v", err)
|
||||||
|
}
|
||||||
|
if msg := err.Error(); msg != tc.Error {
|
||||||
|
t.Fatalf("expected error %q, got %q", tc.Error, msg)
|
||||||
|
}
|
||||||
|
} else if tc.Error != "" {
|
||||||
|
t.Fatalf("expected error %q, got nil", tc.Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
if diff := cmp.Diff(tc.Want, got); diff != "" {
|
||||||
|
t.Errorf("unexpected diff:\n%s", diff)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -121,15 +121,27 @@ func (p *anyObject) UnmarshalCBOR(in []byte) error {
|
|||||||
return modes.Decode.Unmarshal(in, &p.Value)
|
return modes.Decode.Unmarshal(in, &p.Value)
|
||||||
}
|
}
|
||||||
|
|
||||||
type structWithRawExtensionField struct {
|
type structWithRawFields struct {
|
||||||
Extension runtime.RawExtension `json:"extension"`
|
FieldsV1 metav1.FieldsV1 `json:"f"`
|
||||||
|
FieldsV1Pointer *metav1.FieldsV1 `json:"fp"`
|
||||||
|
RawExtension runtime.RawExtension `json:"r"`
|
||||||
|
RawExtensionPointer *runtime.RawExtension `json:"rp"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p structWithRawExtensionField) GetObjectKind() schema.ObjectKind {
|
func (structWithRawFields) GetObjectKind() schema.ObjectKind {
|
||||||
return schema.EmptyObjectKind
|
return schema.EmptyObjectKind
|
||||||
}
|
}
|
||||||
|
|
||||||
func (structWithRawExtensionField) DeepCopyObject() runtime.Object {
|
func (structWithRawFields) DeepCopyObject() runtime.Object {
|
||||||
|
panic("unimplemented")
|
||||||
|
}
|
||||||
|
|
||||||
|
type structWithEmbeddedMetas struct {
|
||||||
|
metav1.TypeMeta `json:",inline"`
|
||||||
|
metav1.ObjectMeta `json:"metadata,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (structWithEmbeddedMetas) DeepCopyObject() runtime.Object {
|
||||||
panic("unimplemented")
|
panic("unimplemented")
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -277,13 +289,35 @@ func TestDecode(t *testing.T) {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "rawextension transcoded",
|
name: "raw types transcoded",
|
||||||
data: []byte{0xa1, 0x49, 'e', 'x', 't', 'e', 'n', 's', 'i', 'o', 'n', 0xa1, 0x41, 'a', 0x01},
|
data: []byte{0xa4, 0x41, 'f', 0xa1, 0x41, 'a', 0x01, 0x42, 'f', 'p', 0xa1, 0x41, 'z', 0x02, 0x41, 'r', 0xa1, 0x41, 'b', 0x03, 0x42, 'r', 'p', 0xa1, 0x41, 'y', 0x04},
|
||||||
gvk: &schema.GroupVersionKind{},
|
gvk: &schema.GroupVersionKind{},
|
||||||
metaFactory: stubMetaFactory{gvk: &schema.GroupVersionKind{}},
|
metaFactory: stubMetaFactory{gvk: &schema.GroupVersionKind{}},
|
||||||
typer: stubTyper{gvks: []schema.GroupVersionKind{{Group: "x", Version: "y", Kind: "z"}}},
|
typer: stubTyper{gvks: []schema.GroupVersionKind{{Group: "x", Version: "y", Kind: "z"}}},
|
||||||
into: &structWithRawExtensionField{},
|
into: &structWithRawFields{},
|
||||||
expectedObj: &structWithRawExtensionField{Extension: runtime.RawExtension{Raw: []byte(`{"a":1}`)}},
|
expectedObj: &structWithRawFields{
|
||||||
|
FieldsV1: metav1.FieldsV1{Raw: []byte(`{"a":1}`)},
|
||||||
|
FieldsV1Pointer: &metav1.FieldsV1{Raw: []byte(`{"z":2}`)},
|
||||||
|
RawExtension: runtime.RawExtension{Raw: []byte(`{"b":3}`)},
|
||||||
|
RawExtensionPointer: &runtime.RawExtension{Raw: []byte(`{"y":4}`)},
|
||||||
|
},
|
||||||
|
expectedGVK: &schema.GroupVersionKind{Group: "x", Version: "y", Kind: "z"},
|
||||||
|
assertOnError: func(t *testing.T, err error) {
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("expected nil error, got: %v", err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "object with embedded typemeta and objectmeta",
|
||||||
|
data: []byte("\xa2\x48metadata\xa1\x44name\x43foo\x44spec\xa0"), // {"metadata": {"name": "foo"}}
|
||||||
|
gvk: &schema.GroupVersionKind{},
|
||||||
|
metaFactory: stubMetaFactory{gvk: &schema.GroupVersionKind{}},
|
||||||
|
typer: stubTyper{gvks: []schema.GroupVersionKind{{Group: "x", Version: "y", Kind: "z"}}},
|
||||||
|
into: &structWithEmbeddedMetas{},
|
||||||
|
expectedObj: &structWithEmbeddedMetas{
|
||||||
|
ObjectMeta: metav1.ObjectMeta{Name: "foo"},
|
||||||
|
},
|
||||||
expectedGVK: &schema.GroupVersionKind{Group: "x", Version: "y", Kind: "z"},
|
expectedGVK: &schema.GroupVersionKind{Group: "x", Version: "y", Kind: "z"},
|
||||||
assertOnError: func(t *testing.T, err error) {
|
assertOnError: func(t *testing.T, err error) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -21,6 +21,7 @@ import (
|
|||||||
"reflect"
|
"reflect"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
||||||
"k8s.io/apimachinery/pkg/runtime"
|
"k8s.io/apimachinery/pkg/runtime"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -44,6 +45,23 @@ var rawTypeTranscodeFuncs = map[reflect.Type]func(reflect.Value) error{
|
|||||||
re.Raw = j
|
re.Raw = j
|
||||||
return nil
|
return nil
|
||||||
},
|
},
|
||||||
|
reflect.TypeFor[metav1.FieldsV1](): func(rv reflect.Value) error {
|
||||||
|
if !rv.CanAddr() {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
fields := rv.Addr().Interface().(*metav1.FieldsV1)
|
||||||
|
if fields.Raw == nil {
|
||||||
|
// When Raw is nil it encodes to null. Don't change nil Raw values during
|
||||||
|
// transcoding, they would have unmarshalled from JSON as nil too.
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
j, err := fields.MarshalJSON()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to transcode FieldsV1 to JSON: %w", err)
|
||||||
|
}
|
||||||
|
fields.Raw = j
|
||||||
|
return nil
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
func transcodeRawTypes(v interface{}) error {
|
func transcodeRawTypes(v interface{}) error {
|
||||||
|
@ -20,6 +20,7 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
||||||
"k8s.io/apimachinery/pkg/runtime"
|
"k8s.io/apimachinery/pkg/runtime"
|
||||||
|
|
||||||
"github.com/google/go-cmp/cmp"
|
"github.com/google/go-cmp/cmp"
|
||||||
@ -112,6 +113,68 @@ func TestTranscodeRawTypes(t *testing.T) {
|
|||||||
In: &map[string][]runtime.RawExtension{"hello": {{Raw: []byte{0xd9, 0xd9, 0xf7, 0x07}}}},
|
In: &map[string][]runtime.RawExtension{"hello": {{Raw: []byte{0xd9, 0xd9, 0xf7, 0x07}}}},
|
||||||
Out: &map[string][]runtime.RawExtension{"hello": {{Raw: []byte(`7`)}}},
|
Out: &map[string][]runtime.RawExtension{"hello": {{Raw: []byte(`7`)}}},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
In: &metav1.FieldsV1{Raw: []byte{0xa0}},
|
||||||
|
Out: &metav1.FieldsV1{Raw: []byte(`{}`)},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
In: &metav1.FieldsV1{},
|
||||||
|
Out: &metav1.FieldsV1{},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
In: metav1.FieldsV1{Raw: []byte{0xa0}},
|
||||||
|
Out: metav1.FieldsV1{Raw: []byte{0xa0}}, // not addressable
|
||||||
|
},
|
||||||
|
{
|
||||||
|
In: &[...]metav1.FieldsV1{{Raw: []byte{0xa0}}, {Raw: []byte{0xf6}}},
|
||||||
|
Out: &[...]metav1.FieldsV1{{Raw: []byte(`{}`)}, {Raw: []byte(`null`)}},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
In: &[0]metav1.FieldsV1{},
|
||||||
|
Out: &[0]metav1.FieldsV1{},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
In: &[]metav1.FieldsV1{{Raw: []byte{0xa0}}, {Raw: []byte{0xf6}}},
|
||||||
|
Out: &[]metav1.FieldsV1{{Raw: []byte(`{}`)}, {Raw: []byte(`null`)}},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
In: &[]metav1.FieldsV1{},
|
||||||
|
Out: &[]metav1.FieldsV1{},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
In: (*metav1.FieldsV1)(nil),
|
||||||
|
Out: (*metav1.FieldsV1)(nil),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
In: &struct{ I fmt.Stringer }{I: &metav1.FieldsV1{Raw: []byte{0xa0}}},
|
||||||
|
Out: &struct{ I fmt.Stringer }{I: &metav1.FieldsV1{Raw: []byte(`{}`)}},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
In: &struct {
|
||||||
|
E metav1.FieldsV1
|
||||||
|
I int64
|
||||||
|
}{E: metav1.FieldsV1{Raw: []byte{0xa0}}, I: 7},
|
||||||
|
Out: &struct {
|
||||||
|
E metav1.FieldsV1
|
||||||
|
I int64
|
||||||
|
}{E: metav1.FieldsV1{Raw: []byte(`{}`)}, I: 7},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
In: &struct {
|
||||||
|
metav1.FieldsV1
|
||||||
|
}{FieldsV1: metav1.FieldsV1{Raw: []byte{0xa0}}},
|
||||||
|
Out: &struct {
|
||||||
|
metav1.FieldsV1
|
||||||
|
}{FieldsV1: metav1.FieldsV1{Raw: []byte(`{}`)}},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
In: &map[string]metav1.FieldsV1{"hello": {Raw: []byte{0xa0}}},
|
||||||
|
Out: &map[string]metav1.FieldsV1{"hello": {Raw: []byte{0xa0}}}, // not addressable
|
||||||
|
},
|
||||||
|
{
|
||||||
|
In: &map[string][]metav1.FieldsV1{"hello": {{Raw: []byte{0xa0}}}},
|
||||||
|
Out: &map[string][]metav1.FieldsV1{"hello": {{Raw: []byte(`{}`)}}},
|
||||||
|
},
|
||||||
} {
|
} {
|
||||||
t.Run(fmt.Sprintf("%#v", tc.In), func(t *testing.T) {
|
t.Run(fmt.Sprintf("%#v", tc.In), func(t *testing.T) {
|
||||||
if err := transcodeRawTypes(tc.In); err != nil {
|
if err := transcodeRawTypes(tc.In); err != nil {
|
||||||
|
Loading…
Reference in New Issue
Block a user