2021-08-20 16:47:37 +00:00
|
|
|
package ssz_static
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/hex"
|
|
|
|
"errors"
|
|
|
|
"path"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
fssz "github.com/ferranbt/fastssz"
|
|
|
|
"github.com/golang/snappy"
|
|
|
|
stateAltair "github.com/prysmaticlabs/prysm/beacon-chain/state/v2"
|
|
|
|
ethpb "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1"
|
2021-09-23 18:53:46 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/testing/require"
|
2021-09-14 16:02:58 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/testing/spectest/utils"
|
2021-09-23 18:53:46 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/testing/util"
|
2021-08-20 16:47:37 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// SSZRoots --
|
|
|
|
type SSZRoots struct {
|
|
|
|
Root string `json:"root"`
|
|
|
|
SigningRoot string `json:"signing_root"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// RunSSZStaticTests executes "ssz_static" tests.
|
|
|
|
func RunSSZStaticTests(t *testing.T, config string) {
|
|
|
|
require.NoError(t, utils.SetConfig(t, config))
|
|
|
|
|
|
|
|
testFolders, _ := utils.TestFolders(t, config, "altair", "ssz_static")
|
|
|
|
for _, folder := range testFolders {
|
|
|
|
innerPath := path.Join("ssz_static", folder.Name(), "ssz_random")
|
|
|
|
innerTestFolders, innerTestsFolderPath := utils.TestFolders(t, config, "altair", innerPath)
|
|
|
|
|
|
|
|
for _, innerFolder := range innerTestFolders {
|
|
|
|
t.Run(path.Join(folder.Name(), innerFolder.Name()), func(t *testing.T) {
|
2021-09-23 18:53:46 +00:00
|
|
|
serializedBytes, err := util.BazelFileBytes(innerTestsFolderPath, innerFolder.Name(), "serialized.ssz_snappy")
|
2021-08-20 16:47:37 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
serializedSSZ, err := snappy.Decode(nil /* dst */, serializedBytes)
|
|
|
|
require.NoError(t, err, "Failed to decompress")
|
|
|
|
object, err := UnmarshalledSSZ(t, serializedSSZ, folder.Name())
|
|
|
|
require.NoError(t, err, "Could not unmarshall serialized SSZ")
|
|
|
|
|
2021-09-23 18:53:46 +00:00
|
|
|
rootsYamlFile, err := util.BazelFileBytes(innerTestsFolderPath, innerFolder.Name(), "roots.yaml")
|
2021-08-20 16:47:37 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
rootsYaml := &SSZRoots{}
|
|
|
|
require.NoError(t, utils.UnmarshalYaml(rootsYamlFile, rootsYaml), "Failed to Unmarshal")
|
|
|
|
|
|
|
|
// Custom hash tree root for beacon state.
|
|
|
|
var htr func(interface{}) ([32]byte, error)
|
|
|
|
if _, ok := object.(*ethpb.BeaconStateAltair); ok {
|
|
|
|
htr = func(s interface{}) ([32]byte, error) {
|
|
|
|
beaconState, err := stateAltair.InitializeFromProto(s.(*ethpb.BeaconStateAltair))
|
|
|
|
require.NoError(t, err)
|
|
|
|
return beaconState.HashTreeRoot(context.Background())
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
htr = func(s interface{}) ([32]byte, error) {
|
|
|
|
sszObj, ok := s.(fssz.HashRoot)
|
|
|
|
if !ok {
|
|
|
|
return [32]byte{}, errors.New("could not get hash root, not compatible object")
|
|
|
|
}
|
|
|
|
return sszObj.HashTreeRoot()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
root, err := htr(object)
|
|
|
|
require.NoError(t, err)
|
|
|
|
rootBytes, err := hex.DecodeString(rootsYaml.Root[2:])
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.DeepEqual(t, rootBytes, root[:], "Did not receive expected hash tree root")
|
|
|
|
|
|
|
|
if rootsYaml.SigningRoot == "" {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
var signingRoot [32]byte
|
|
|
|
if v, ok := object.(fssz.HashRoot); ok {
|
|
|
|
signingRoot, err = v.HashTreeRoot()
|
|
|
|
} else {
|
|
|
|
t.Fatal("object does not meet fssz.HashRoot")
|
|
|
|
}
|
|
|
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
signingRootBytes, err := hex.DecodeString(rootsYaml.SigningRoot[2:])
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.DeepEqual(t, signingRootBytes, signingRoot[:], "Did not receive expected signing root")
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalledSSZ unmarshalls serialized input.
|
|
|
|
func UnmarshalledSSZ(t *testing.T, serializedBytes []byte, folderName string) (interface{}, error) {
|
|
|
|
var obj interface{}
|
|
|
|
switch folderName {
|
|
|
|
case "Attestation":
|
|
|
|
obj = ðpb.Attestation{}
|
|
|
|
case "AttestationData":
|
|
|
|
obj = ðpb.AttestationData{}
|
|
|
|
case "AttesterSlashing":
|
|
|
|
obj = ðpb.AttesterSlashing{}
|
|
|
|
case "AggregateAndProof":
|
|
|
|
obj = ðpb.AggregateAttestationAndProof{}
|
|
|
|
case "BeaconBlock":
|
|
|
|
obj = ðpb.BeaconBlockAltair{}
|
|
|
|
case "BeaconBlockBody":
|
|
|
|
obj = ðpb.BeaconBlockBodyAltair{}
|
|
|
|
case "BeaconBlockHeader":
|
|
|
|
obj = ðpb.BeaconBlockHeader{}
|
|
|
|
case "BeaconState":
|
|
|
|
obj = ðpb.BeaconStateAltair{}
|
|
|
|
case "Checkpoint":
|
|
|
|
obj = ðpb.Checkpoint{}
|
|
|
|
case "Deposit":
|
|
|
|
obj = ðpb.Deposit{}
|
|
|
|
case "DepositMessage":
|
|
|
|
obj = ðpb.DepositMessage{}
|
|
|
|
case "DepositData":
|
|
|
|
obj = ðpb.Deposit_Data{}
|
|
|
|
case "Eth1Data":
|
|
|
|
obj = ðpb.Eth1Data{}
|
|
|
|
case "Eth1Block":
|
|
|
|
t.Skip("Unused type")
|
|
|
|
return nil, nil
|
|
|
|
case "Fork":
|
|
|
|
obj = ðpb.Fork{}
|
|
|
|
case "ForkData":
|
|
|
|
obj = ðpb.ForkData{}
|
|
|
|
case "HistoricalBatch":
|
|
|
|
obj = ðpb.HistoricalBatch{}
|
|
|
|
case "IndexedAttestation":
|
|
|
|
obj = ðpb.IndexedAttestation{}
|
|
|
|
case "PendingAttestation":
|
|
|
|
obj = ðpb.PendingAttestation{}
|
|
|
|
case "ProposerSlashing":
|
|
|
|
obj = ðpb.ProposerSlashing{}
|
|
|
|
case "SignedAggregateAndProof":
|
|
|
|
obj = ðpb.SignedAggregateAttestationAndProof{}
|
|
|
|
case "SignedBeaconBlock":
|
|
|
|
obj = ðpb.SignedBeaconBlockAltair{}
|
|
|
|
case "SignedBeaconBlockHeader":
|
|
|
|
obj = ðpb.SignedBeaconBlockHeader{}
|
|
|
|
case "SignedVoluntaryExit":
|
|
|
|
obj = ðpb.SignedVoluntaryExit{}
|
|
|
|
case "SigningData":
|
|
|
|
obj = ðpb.SigningData{}
|
|
|
|
case "Validator":
|
|
|
|
obj = ðpb.Validator{}
|
|
|
|
case "VoluntaryExit":
|
|
|
|
obj = ðpb.VoluntaryExit{}
|
|
|
|
case "SyncCommitteeMessage":
|
|
|
|
obj = ðpb.SyncCommitteeMessage{}
|
|
|
|
case "SyncCommitteeContribution":
|
|
|
|
obj = ðpb.SyncCommitteeContribution{}
|
|
|
|
case "ContributionAndProof":
|
|
|
|
obj = ðpb.ContributionAndProof{}
|
|
|
|
case "SignedContributionAndProof":
|
|
|
|
obj = ðpb.SignedContributionAndProof{}
|
|
|
|
case "SyncAggregate":
|
|
|
|
obj = ðpb.SyncAggregate{}
|
|
|
|
case "SyncAggregatorSelectionData":
|
|
|
|
obj = ðpb.SyncAggregatorSelectionData{}
|
|
|
|
case "SyncCommittee":
|
2021-09-07 19:42:37 +00:00
|
|
|
obj = ðpb.SyncCommittee{}
|
2021-08-20 16:47:37 +00:00
|
|
|
case "LightClientSnapshot":
|
|
|
|
t.Skip("not a beacon node type, this is a light node type")
|
|
|
|
return nil, nil
|
|
|
|
case "LightClientUpdate":
|
|
|
|
t.Skip("not a beacon node type, this is a light node type")
|
|
|
|
return nil, nil
|
|
|
|
default:
|
|
|
|
return nil, errors.New("type not found")
|
|
|
|
}
|
|
|
|
var err error
|
|
|
|
if o, ok := obj.(fssz.Unmarshaler); ok {
|
|
|
|
err = o.UnmarshalSSZ(serializedBytes)
|
|
|
|
} else {
|
|
|
|
err = errors.New("could not unmarshal object, not a fastssz compatible object")
|
|
|
|
}
|
|
|
|
return obj, err
|
|
|
|
}
|