mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-09 03:01:19 +00:00
cf0bd633f0
* fork/version detection and unmarshaling support * Update config/params/config.go Co-authored-by: Radosław Kapka <rkapka@wp.pl> * Update proto/detect/configfork.go Co-authored-by: Radosław Kapka <rkapka@wp.pl> * PR feedback * move ssz initialization into the detect package * clarify comment * VersionForEpoch is much simpler/clearer in reverse * simpler VersionForEpoch; build AllConfigs in init * use fieldparams for Version * Update proto/detect/configfork_test.go Co-authored-by: Radosław Kapka <rkapka@wp.pl> * remove custom ForkName type, use runtime/version * pr cleanup * random fix from bad gh ui suggestion; privatize * privatize fieldSpec methods; + unit tests * Update proto/detect/configfork.go Co-authored-by: Potuz <potuz@prysmaticlabs.com> * fix bad github ui suggestion * ensure unique versions for simpler config match * fmt & adding unit test for ByState() * table-driven unit test for ByState * TestUnmarshalState * OrderedSchedule -> network/forks per PR feedback * goimports * lint fixes * move proto/detect -> ssz/encoding/detect * use typeUndefined in String * backport config tests from e2e PR * fix config parity test; make debugging it easier * lint * fix fork schedule initialization * cleanup * fix build * fix big ole derp * anything for you, deep source * goimportsss * InitializeForkSchedule in LoadChainConfigFile * PR feedback Co-authored-by: kasey <kasey@users.noreply.github.com> Co-authored-by: Radosław Kapka <rkapka@wp.pl> Co-authored-by: Potuz <potuz@prysmaticlabs.com>
180 lines
6.7 KiB
Go
180 lines
6.7 KiB
Go
package detect
|
|
|
|
import (
|
|
"fmt"
|
|
|
|
"github.com/prysmaticlabs/prysm/encoding/bytesutil"
|
|
"github.com/prysmaticlabs/prysm/network/forks"
|
|
|
|
ssz "github.com/ferranbt/fastssz"
|
|
"github.com/pkg/errors"
|
|
types "github.com/prysmaticlabs/eth2-types"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/state"
|
|
v1 "github.com/prysmaticlabs/prysm/beacon-chain/state/v1"
|
|
v2 "github.com/prysmaticlabs/prysm/beacon-chain/state/v2"
|
|
v3 "github.com/prysmaticlabs/prysm/beacon-chain/state/v3"
|
|
fieldparams "github.com/prysmaticlabs/prysm/config/fieldparams"
|
|
"github.com/prysmaticlabs/prysm/config/params"
|
|
ethpb "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1"
|
|
"github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1/block"
|
|
"github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1/wrapper"
|
|
"github.com/prysmaticlabs/prysm/runtime/version"
|
|
"github.com/prysmaticlabs/prysm/time/slots"
|
|
)
|
|
|
|
// VersionedUnmarshaler represents the intersection of Configuration (eg mainnet, testnet) and Fork (eg phase0, altair).
|
|
// Using a detected VersionedUnmarshaler, a BeaconState or SignedBeaconBlock can be correctly unmarshaled without the need to
|
|
// hard code a concrete type in paths where only the marshaled bytes, or marshaled bytes and a version, are available.
|
|
type VersionedUnmarshaler struct {
|
|
Config *params.BeaconChainConfig
|
|
// Fork aligns with the fork names in config/params/values.go
|
|
Fork int
|
|
// Version corresponds to the Version type defined in the beacon-chain spec, aka a "fork version number":
|
|
// https://github.com/ethereum/consensus-specs/blob/dev/specs/phase0/beacon-chain.md#custom-types
|
|
Version [fieldparams.VersionLength]byte
|
|
}
|
|
|
|
var beaconStateCurrentVersion = fieldSpec{
|
|
// 52 = 8 (genesis_time) + 32 (genesis_validators_root) + 8 (slot) + 4 (previous_version)
|
|
offset: 52,
|
|
t: typeBytes4,
|
|
}
|
|
|
|
// FromState exploits the fixed-size lower-order bytes in a BeaconState as a heuristic to obtain the value of the
|
|
// state.version field without first unmarshaling the BeaconState. The Version is then internally used to lookup
|
|
// the correct ConfigVersion.
|
|
func FromState(marshaled []byte) (*VersionedUnmarshaler, error) {
|
|
cv, err := beaconStateCurrentVersion.bytes4(marshaled)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return FromForkVersion(cv)
|
|
}
|
|
|
|
var ErrForkNotFound = errors.New("version found in fork schedule but can't be matched to a named fork")
|
|
|
|
// FromForkVersion uses a lookup table to resolve a Version (from a beacon node api for instance, or obtained by peeking at
|
|
// the bytes of a marshaled BeaconState) to a VersionedUnmarshaler.
|
|
func FromForkVersion(cv [fieldparams.VersionLength]byte) (*VersionedUnmarshaler, error) {
|
|
cfg, err := params.ConfigForVersion(cv)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var fork int
|
|
switch cv {
|
|
case bytesutil.ToBytes4(cfg.GenesisForkVersion):
|
|
fork = version.Phase0
|
|
case bytesutil.ToBytes4(cfg.AltairForkVersion):
|
|
fork = version.Altair
|
|
case bytesutil.ToBytes4(cfg.BellatrixForkVersion):
|
|
fork = version.Bellatrix
|
|
default:
|
|
return nil, errors.Wrapf(ErrForkNotFound, "version=%#x", cv)
|
|
}
|
|
return &VersionedUnmarshaler{
|
|
Config: cfg,
|
|
Fork: fork,
|
|
Version: cv,
|
|
}, nil
|
|
}
|
|
|
|
// UnmarshalBeaconState uses internal knowledge in the VersionedUnmarshaler to pick the right concrete BeaconState type,
|
|
// then Unmarshal()s the type and returns an instance of state.BeaconState if successful.
|
|
func (cf *VersionedUnmarshaler) UnmarshalBeaconState(marshaled []byte) (s state.BeaconState, err error) {
|
|
forkName := version.String(cf.Fork)
|
|
switch fork := cf.Fork; fork {
|
|
case version.Phase0:
|
|
st := ðpb.BeaconState{}
|
|
err = st.UnmarshalSSZ(marshaled)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed to unmarshal state, detected fork=%s", forkName)
|
|
}
|
|
s, err = v1.InitializeFromProtoUnsafe(st)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed to init state trie from state, detected fork=%s", forkName)
|
|
}
|
|
case version.Altair:
|
|
st := ðpb.BeaconStateAltair{}
|
|
err = st.UnmarshalSSZ(marshaled)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed to unmarshal state, detected fork=%s", forkName)
|
|
}
|
|
s, err = v2.InitializeFromProtoUnsafe(st)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed to init state trie from state, detected fork=%s", forkName)
|
|
}
|
|
case version.Bellatrix:
|
|
st := ðpb.BeaconStateBellatrix{}
|
|
err = st.UnmarshalSSZ(marshaled)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed to unmarshal state, detected fork=%s", forkName)
|
|
}
|
|
s, err = v3.InitializeFromProtoUnsafe(st)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed to init state trie from state, detected fork=%s", forkName)
|
|
}
|
|
default:
|
|
return nil, fmt.Errorf("unable to initialize BeaconState for fork version=%s", forkName)
|
|
}
|
|
|
|
return s, nil
|
|
}
|
|
|
|
var beaconBlockSlot = fieldSpec{
|
|
// ssz variable length offset (not to be confused with the fieldSpec offest) is a uint32
|
|
// variable length. Offsets come before fixed length data, so that's 4 bytes at the beginning
|
|
// then signature is 96 bytes, 4+96 = 100
|
|
offset: 100,
|
|
t: typeUint64,
|
|
}
|
|
|
|
func slotFromBlock(marshaled []byte) (types.Slot, error) {
|
|
slot, err := beaconBlockSlot.uint64(marshaled)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return types.Slot(slot), nil
|
|
}
|
|
|
|
var errBlockForkMismatch = errors.New("fork or config detected from state is different than block")
|
|
|
|
// UnmarshalBeaconBlock uses internal knowledge in the VersionedUnmarshaler to pick the right concrete SignedBeaconBlock type,
|
|
// then Unmarshal()s the type and returns an instance of block.SignedBeaconBlock if successful.
|
|
func (cf *VersionedUnmarshaler) UnmarshalBeaconBlock(marshaled []byte) (block.SignedBeaconBlock, error) {
|
|
slot, err := slotFromBlock(marshaled)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// heuristic to make sure block is from the same version as the VersionedUnmarshaler.
|
|
// Look up the version for the epoch that the block is from, then ensure that it matches the Version in the
|
|
// VersionedUnmarshaler.
|
|
epoch := slots.ToEpoch(slot)
|
|
fs := forks.NewOrderedSchedule(cf.Config)
|
|
ver, err := fs.VersionForEpoch(epoch)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if ver != cf.Version {
|
|
return nil, errors.Wrapf(errBlockForkMismatch, "slot=%d, epoch=%d, version=%#x", slot, epoch, ver)
|
|
}
|
|
|
|
var blk ssz.Unmarshaler
|
|
switch cf.Fork {
|
|
case version.Phase0:
|
|
blk = ðpb.SignedBeaconBlock{}
|
|
case version.Altair:
|
|
blk = ðpb.SignedBeaconBlockAltair{}
|
|
case version.Bellatrix:
|
|
blk = ðpb.SignedBeaconBlockBellatrix{}
|
|
default:
|
|
forkName := version.String(cf.Fork)
|
|
return nil, fmt.Errorf("unable to initialize BeaconBlock for fork version=%s at slot=%d", forkName, slot)
|
|
}
|
|
err = blk.UnmarshalSSZ(marshaled)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to unmarshal SignedBeaconBlock in UnmarshalSSZ")
|
|
}
|
|
return wrapper.WrappedSignedBeaconBlock(blk)
|
|
}
|