Support omitempty for nbt encoding

This commit is contained in:
Dmytro Manchynskyi
2022-05-27 01:54:22 +03:00
parent 691e507fcf
commit eb9d50b3ee
2 changed files with 79 additions and 2 deletions

View File

@ -197,6 +197,10 @@ func (e *Encoder) writeValue(val reflect.Value, tagType byte) error {
}
tagProps := parseTag(f, v, tag)
if tagProps.OmitEmpty && isEmptyValue(v) {
continue
}
if err := e.marshal(val.Field(i), tagProps.Type, tagProps.Name); err != nil {
return err
}
@ -318,11 +322,17 @@ func getTagTypeByType(vk reflect.Type) byte {
}
type tagProps struct {
Name string
Type byte
Name string
Type byte
OmitEmpty bool
}
func parseTag(f reflect.StructField, v reflect.Value, tagName string) (result tagProps) {
if strings.HasSuffix(tagName, ",omitempty") {
result.OmitEmpty = true
tagName = tagName[:len(tagName)-10]
}
if tagName != "" {
result.Name = tagName
} else {
@ -381,3 +391,22 @@ func (e *Encoder) writeInt64(n int64) error {
byte(n >> 24), byte(n >> 16), byte(n >> 8), byte(n)})
return err
}
// Copied from encoding/json/encode.go
func isEmptyValue(v reflect.Value) bool {
switch v.Kind() {
case reflect.Array, reflect.Map, reflect.Slice, reflect.String:
return v.Len() == 0
case reflect.Bool:
return !v.Bool()
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
return v.Int() == 0
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
return v.Uint() == 0
case reflect.Float32, reflect.Float64:
return v.Float() == 0
case reflect.Interface, reflect.Pointer:
return v.IsNil()
}
return false
}

View File

@ -273,3 +273,51 @@ func TestEncoder_Encode_textMarshaler(t *testing.T) {
t.Errorf("get %v, want %v", data, wants)
}
}
func TestEncoder_Encode_omitempty(t *testing.T) {
type Struct struct {
S string `nbt:"test,omitempty"`
B []byte `nbt:",omitempty"`
I int32 `nbt:",omitempty"`
}
tests := []struct {
name string
args any
want []byte
}{
{
name: "Empty struct",
args: Struct{},
want: []byte{
TagCompound, 0x00, 0x00,
TagEnd,
},
}, {
name: "Non-empty struct",
args: Struct{
S: "ab",
B: []byte{4, 5},
I: 9,
},
want: []byte{
TagCompound, 0x00, 0x00,
TagString, 0x00, 4, 't', 'e', 's', 't', 0, 2, 'a', 'b',
TagByteArray, 0x00, 1, 'B', 0x00, 0x00, 0, 2, 4, 5,
TagInt, 0x00, 1, 'I', 0x00, 0x00, 0x00, 0x09,
TagEnd,
},
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
data, err := Marshal(tt.args)
if err != nil {
t.Error(err)
} else if !bytes.Equal(data, tt.want) {
t.Errorf("Marshal([]struct{}) got = % 02x, want % 02x", data, tt.want)
return
}
})
}
}