prysm-pulse/shared/ssz/example_and_test.go
2019-01-02 11:14:12 -08:00

148 lines
3.3 KiB
Go

package ssz
import (
"bytes"
"io"
"reflect"
"testing"
)
type exampleStruct1 struct {
Field1 uint8
Field2 []byte
}
func (e *exampleStruct1) EncodeSSZ(w io.Writer) error {
// Need to pass value of struct for Encode function
// Later we can enhance the ssz implementation to support passing pointer, if necessary
return Encode(w, *e)
}
func (e *exampleStruct1) EncodeSSZSize() (uint32, error) {
return EncodeSize(*e)
}
func (e *exampleStruct1) DecodeSSZ(r io.Reader) error {
// Need to pass pointer of struct for Decode function
return Decode(r, e)
}
func (e *exampleStruct1) TreeHashSSZ() ([32]byte, error) {
return TreeHash(e)
}
type exampleStruct2 struct {
Field1 uint8 // a volatile, or host-specific field that doesn't need to be exported
Field2 []byte
}
// You can use a helper struct to only encode/decode custom fields of your struct
type exampleStruct2Export struct {
Field2 []byte
}
func (e *exampleStruct2) EncodeSSZ(w io.Writer) error {
return Encode(w, exampleStruct2Export{
e.Field2,
})
}
func (e *exampleStruct2) EncodeSSZSize() (uint32, error) {
return EncodeSize(exampleStruct2Export{
e.Field2,
})
}
func (e *exampleStruct2) DecodeSSZ(r io.Reader) error {
ee := new(exampleStruct2Export)
if err := Decode(r, ee); err != nil {
return err
}
e.Field2 = ee.Field2
return nil
}
func (e *exampleStruct2) TreeHashSSZ() ([32]byte, error) {
return TreeHash(exampleStruct2Export{
e.Field2,
})
}
func TestEncodeDecodeExampleStruct1(t *testing.T) {
var err error
e1 := &exampleStruct1{
Field1: 10,
Field2: []byte{1, 2, 3, 4},
}
wBuf := new(bytes.Buffer)
if err = e1.EncodeSSZ(wBuf); err != nil {
t.Fatalf("failed to encode: %v", err)
}
encoding := wBuf.Bytes()
e2 := new(exampleStruct1)
rBuf := bytes.NewReader(encoding)
if err = e2.DecodeSSZ(rBuf); err != nil {
t.Fatalf("failed to decode: %v", err)
}
if !reflect.DeepEqual(*e1, *e2) {
t.Error("encode/decode algorithm don't match")
}
encodeSize := uint32(0)
if encodeSize, err = e1.EncodeSSZSize(); err != nil {
t.Errorf("failed to get encode size: %v", err)
}
if encodeSize != 13 {
t.Error("wrong encode size calculation result")
}
hash, err := e1.TreeHashSSZ()
if err != nil {
t.Fatalf("failed to hash: %v", err)
}
if !bytes.Equal(hash[:], unhex("cc9e553a3fae873a919e6ab62dc0f461199fb5880d38d124178c209c1f5d78cd")) {
t.Errorf("wrong hash result")
}
}
func TestEncodeDecodeExampleStruct2(t *testing.T) {
var err error
e1 := &exampleStruct2{
Field1: 10,
Field2: []byte{1, 2, 3, 4},
}
wBuf := new(bytes.Buffer)
if err = e1.EncodeSSZ(wBuf); err != nil {
t.Fatalf("failed to encode: %v", err)
}
encoding := wBuf.Bytes()
e2 := new(exampleStruct2)
rBuf := bytes.NewReader(encoding)
if err = e2.DecodeSSZ(rBuf); err != nil {
t.Fatalf("failed to decode: %v", err)
}
if !reflect.DeepEqual(e1.Field2, e2.Field2) {
t.Error("encode/decode algorithm don't match")
}
encodeSize := uint32(0)
if encodeSize, err = e1.EncodeSSZSize(); err != nil {
t.Errorf("failed to get encode size: %v", err)
}
if encodeSize != 12 {
t.Error("wrong encode size calculation result")
}
hash, err := e1.TreeHashSSZ()
if err != nil {
t.Fatalf("failed to hash: %v", err)
}
if !bytes.Equal(hash[:], unhex("d74e2ddcd0ec52ae08d6e041af3eec1a78955533577e8320c06af4ad923eba0d")) {
t.Errorf("wrong hash result")
}
}