Make optional.Option[T] type serializable (#29282)
make the generic `Option` type de-/serializable for json and yaml --------- Co-authored-by: KN4CK3R <admin@oldschoolhack.me> (cherry picked from commit 53c7d8908e5ef35818b72b8c3d873b509269bc1a)
This commit is contained in:
parent
7bf905a43c
commit
7db422d989
3 changed files with 248 additions and 10 deletions
|
@ -1,47 +1,49 @@
|
||||||
// Copyright 2024 The Gitea Authors. All rights reserved.
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
||||||
// SPDX-License-Identifier: MIT
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
package optional
|
package optional_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"code.gitea.io/gitea/modules/optional"
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestOption(t *testing.T) {
|
func TestOption(t *testing.T) {
|
||||||
var uninitialized Option[int]
|
var uninitialized optional.Option[int]
|
||||||
assert.False(t, uninitialized.Has())
|
assert.False(t, uninitialized.Has())
|
||||||
assert.Equal(t, int(0), uninitialized.Value())
|
assert.Equal(t, int(0), uninitialized.Value())
|
||||||
assert.Equal(t, int(1), uninitialized.ValueOrDefault(1))
|
assert.Equal(t, int(1), uninitialized.ValueOrDefault(1))
|
||||||
|
|
||||||
none := None[int]()
|
none := optional.None[int]()
|
||||||
assert.False(t, none.Has())
|
assert.False(t, none.Has())
|
||||||
assert.Equal(t, int(0), none.Value())
|
assert.Equal(t, int(0), none.Value())
|
||||||
assert.Equal(t, int(1), none.ValueOrDefault(1))
|
assert.Equal(t, int(1), none.ValueOrDefault(1))
|
||||||
|
|
||||||
some := Some[int](1)
|
some := optional.Some[int](1)
|
||||||
assert.True(t, some.Has())
|
assert.True(t, some.Has())
|
||||||
assert.Equal(t, int(1), some.Value())
|
assert.Equal(t, int(1), some.Value())
|
||||||
assert.Equal(t, int(1), some.ValueOrDefault(2))
|
assert.Equal(t, int(1), some.ValueOrDefault(2))
|
||||||
|
|
||||||
var ptr *int
|
var ptr *int
|
||||||
assert.False(t, FromPtr(ptr).Has())
|
assert.False(t, optional.FromPtr(ptr).Has())
|
||||||
|
|
||||||
int1 := 1
|
int1 := 1
|
||||||
opt1 := FromPtr(&int1)
|
opt1 := optional.FromPtr(&int1)
|
||||||
assert.True(t, opt1.Has())
|
assert.True(t, opt1.Has())
|
||||||
assert.Equal(t, int(1), opt1.Value())
|
assert.Equal(t, int(1), opt1.Value())
|
||||||
|
|
||||||
assert.False(t, FromNonDefault("").Has())
|
assert.False(t, optional.FromNonDefault("").Has())
|
||||||
|
|
||||||
opt2 := FromNonDefault("test")
|
opt2 := optional.FromNonDefault("test")
|
||||||
assert.True(t, opt2.Has())
|
assert.True(t, opt2.Has())
|
||||||
assert.Equal(t, "test", opt2.Value())
|
assert.Equal(t, "test", opt2.Value())
|
||||||
|
|
||||||
assert.False(t, FromNonDefault(0).Has())
|
assert.False(t, optional.FromNonDefault(0).Has())
|
||||||
|
|
||||||
opt3 := FromNonDefault(1)
|
opt3 := optional.FromNonDefault(1)
|
||||||
assert.True(t, opt3.Has())
|
assert.True(t, opt3.Has())
|
||||||
assert.Equal(t, int(1), opt3.Value())
|
assert.Equal(t, int(1), opt3.Value())
|
||||||
}
|
}
|
||||||
|
|
46
modules/optional/serialization.go
Normal file
46
modules/optional/serialization.go
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package optional
|
||||||
|
|
||||||
|
import (
|
||||||
|
"code.gitea.io/gitea/modules/json"
|
||||||
|
|
||||||
|
"gopkg.in/yaml.v3"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (o *Option[T]) UnmarshalJSON(data []byte) error {
|
||||||
|
var v *T
|
||||||
|
if err := json.Unmarshal(data, &v); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
*o = FromPtr(v)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o Option[T]) MarshalJSON() ([]byte, error) {
|
||||||
|
if !o.Has() {
|
||||||
|
return []byte("null"), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return json.Marshal(o.Value())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *Option[T]) UnmarshalYAML(value *yaml.Node) error {
|
||||||
|
var v *T
|
||||||
|
if err := value.Decode(&v); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
*o = FromPtr(v)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o Option[T]) MarshalYAML() (interface{}, error) {
|
||||||
|
if !o.Has() {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
value := new(yaml.Node)
|
||||||
|
err := value.Encode(o.Value())
|
||||||
|
return value, err
|
||||||
|
}
|
190
modules/optional/serialization_test.go
Normal file
190
modules/optional/serialization_test.go
Normal file
|
@ -0,0 +1,190 @@
|
||||||
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package optional_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
std_json "encoding/json" //nolint:depguard
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"code.gitea.io/gitea/modules/json"
|
||||||
|
"code.gitea.io/gitea/modules/optional"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"gopkg.in/yaml.v3"
|
||||||
|
)
|
||||||
|
|
||||||
|
type testSerializationStruct struct {
|
||||||
|
NormalString string `json:"normal_string" yaml:"normal_string"`
|
||||||
|
NormalBool bool `json:"normal_bool" yaml:"normal_bool"`
|
||||||
|
OptBool optional.Option[bool] `json:"optional_bool,omitempty" yaml:"optional_bool,omitempty"`
|
||||||
|
OptString optional.Option[string] `json:"optional_string,omitempty" yaml:"optional_string,omitempty"`
|
||||||
|
OptTwoBool optional.Option[bool] `json:"optional_two_bool" yaml:"optional_two_bool"`
|
||||||
|
OptTwoString optional.Option[string] `json:"optional_twostring" yaml:"optional_two_string"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestOptionalToJson(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
obj *testSerializationStruct
|
||||||
|
want string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "empty",
|
||||||
|
obj: new(testSerializationStruct),
|
||||||
|
want: `{"normal_string":"","normal_bool":false,"optional_two_bool":null,"optional_twostring":null}`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "some",
|
||||||
|
obj: &testSerializationStruct{
|
||||||
|
NormalString: "a string",
|
||||||
|
NormalBool: true,
|
||||||
|
OptBool: optional.Some(false),
|
||||||
|
OptString: optional.Some(""),
|
||||||
|
OptTwoBool: optional.None[bool](),
|
||||||
|
OptTwoString: optional.None[string](),
|
||||||
|
},
|
||||||
|
want: `{"normal_string":"a string","normal_bool":true,"optional_bool":false,"optional_string":"","optional_two_bool":null,"optional_twostring":null}`,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, tc := range tests {
|
||||||
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
|
b, err := json.Marshal(tc.obj)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.EqualValues(t, tc.want, string(b), "gitea json module returned unexpected")
|
||||||
|
|
||||||
|
b, err = std_json.Marshal(tc.obj)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.EqualValues(t, tc.want, string(b), "std json module returned unexpected")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestOptionalFromJson(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
data string
|
||||||
|
want testSerializationStruct
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "empty",
|
||||||
|
data: `{}`,
|
||||||
|
want: testSerializationStruct{
|
||||||
|
NormalString: "",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "some",
|
||||||
|
data: `{"normal_string":"a string","normal_bool":true,"optional_bool":false,"optional_string":"","optional_two_bool":null,"optional_twostring":null}`,
|
||||||
|
want: testSerializationStruct{
|
||||||
|
NormalString: "a string",
|
||||||
|
NormalBool: true,
|
||||||
|
OptBool: optional.Some(false),
|
||||||
|
OptString: optional.Some(""),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, tc := range tests {
|
||||||
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
|
var obj1 testSerializationStruct
|
||||||
|
err := json.Unmarshal([]byte(tc.data), &obj1)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.EqualValues(t, tc.want, obj1, "gitea json module returned unexpected")
|
||||||
|
|
||||||
|
var obj2 testSerializationStruct
|
||||||
|
err = std_json.Unmarshal([]byte(tc.data), &obj2)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.EqualValues(t, tc.want, obj2, "std json module returned unexpected")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestOptionalToYaml(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
obj *testSerializationStruct
|
||||||
|
want string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "empty",
|
||||||
|
obj: new(testSerializationStruct),
|
||||||
|
want: `normal_string: ""
|
||||||
|
normal_bool: false
|
||||||
|
optional_two_bool: null
|
||||||
|
optional_two_string: null
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "some",
|
||||||
|
obj: &testSerializationStruct{
|
||||||
|
NormalString: "a string",
|
||||||
|
NormalBool: true,
|
||||||
|
OptBool: optional.Some(false),
|
||||||
|
OptString: optional.Some(""),
|
||||||
|
},
|
||||||
|
want: `normal_string: a string
|
||||||
|
normal_bool: true
|
||||||
|
optional_bool: false
|
||||||
|
optional_string: ""
|
||||||
|
optional_two_bool: null
|
||||||
|
optional_two_string: null
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, tc := range tests {
|
||||||
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
|
b, err := yaml.Marshal(tc.obj)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.EqualValues(t, tc.want, string(b), "yaml module returned unexpected")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestOptionalFromYaml(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
data string
|
||||||
|
want testSerializationStruct
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "empty",
|
||||||
|
data: ``,
|
||||||
|
want: testSerializationStruct{},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "empty but init",
|
||||||
|
data: `normal_string: ""
|
||||||
|
normal_bool: false
|
||||||
|
optional_bool:
|
||||||
|
optional_two_bool:
|
||||||
|
optional_two_string:
|
||||||
|
`,
|
||||||
|
want: testSerializationStruct{},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "some",
|
||||||
|
data: `
|
||||||
|
normal_string: a string
|
||||||
|
normal_bool: true
|
||||||
|
optional_bool: false
|
||||||
|
optional_string: ""
|
||||||
|
optional_two_bool: null
|
||||||
|
optional_twostring: null
|
||||||
|
`,
|
||||||
|
want: testSerializationStruct{
|
||||||
|
NormalString: "a string",
|
||||||
|
NormalBool: true,
|
||||||
|
OptBool: optional.Some(false),
|
||||||
|
OptString: optional.Some(""),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, tc := range tests {
|
||||||
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
|
var obj testSerializationStruct
|
||||||
|
err := yaml.Unmarshal([]byte(tc.data), &obj)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.EqualValues(t, tc.want, obj, "yaml module returned unexpected")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue