mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-08 02:31:19 +00:00
d17996f8b0
* Update V3 from V4 * Fix build v3 -> v4 * Update ssz * Update beacon_chain.pb.go * Fix formatter import * Update update-mockgen.sh comment to v4 * Fix conflicts. Pass build and tests * Fix test
153 lines
4.9 KiB
Go
153 lines
4.9 KiB
Go
// Package depositsnapshot implements the EIP-4881 standard for minimal sparse Merkle tree.
|
|
// The format proposed by the EIP allows for the pruning of deposits that are no longer needed to participate fully in consensus.
|
|
// Full EIP-4881 specification can be found here: https://eips.ethereum.org/EIPS/eip-4881
|
|
package depositsnapshot
|
|
|
|
import (
|
|
"crypto/sha256"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/v4/encoding/bytesutil"
|
|
"github.com/prysmaticlabs/prysm/v4/math"
|
|
eth "github.com/prysmaticlabs/prysm/v4/proto/eth/v1"
|
|
)
|
|
|
|
var (
|
|
// ErrEmptyExecutionBlock occurs when the execution block is nil.
|
|
ErrEmptyExecutionBlock = errors.New("empty execution block")
|
|
// ErrInvalidSnapshotRoot occurs when the snapshot root does not match the calculated root.
|
|
ErrInvalidSnapshotRoot = errors.New("snapshot root is invalid")
|
|
// ErrInvalidMixInLength occurs when the value for mix in length is 0.
|
|
ErrInvalidMixInLength = errors.New("mixInLength should be greater than 0")
|
|
// ErrInvalidIndex occurs when the index is less than the number of finalized deposits.
|
|
ErrInvalidIndex = errors.New("index should be greater than finalizedDeposits - 1")
|
|
// ErrNoDeposits occurs when the number of deposits is 0.
|
|
ErrNoDeposits = errors.New("number of deposits should be greater than 0")
|
|
// ErrTooManyDeposits occurs when the number of deposits exceeds the capacity of the tree.
|
|
ErrTooManyDeposits = errors.New("number of deposits should not be greater than the capacity of the tree")
|
|
)
|
|
|
|
// DepositTree is the Merkle tree representation of deposits.
|
|
type DepositTree struct {
|
|
tree MerkleTreeNode
|
|
mixInLength uint64 // number of deposits in the tree, reference implementation calls this mix_in_length.
|
|
finalizedExecutionBlock executionBlock
|
|
}
|
|
|
|
type executionBlock struct {
|
|
Hash [32]byte
|
|
Depth uint64
|
|
}
|
|
|
|
// New creates an empty deposit tree.
|
|
//
|
|
//nolint:unused
|
|
func New() *DepositTree {
|
|
var leaves [][32]byte
|
|
merkle := create(leaves, DepositContractDepth)
|
|
return &DepositTree{
|
|
tree: merkle,
|
|
mixInLength: 0,
|
|
finalizedExecutionBlock: executionBlock{},
|
|
}
|
|
}
|
|
|
|
// getSnapshot returns a deposit tree snapshot.
|
|
//
|
|
//nolint:unused
|
|
func (d *DepositTree) getSnapshot() (DepositTreeSnapshot, error) {
|
|
if d.finalizedExecutionBlock == (executionBlock{}) {
|
|
return DepositTreeSnapshot{}, ErrEmptyExecutionBlock
|
|
}
|
|
var finalized [][32]byte
|
|
depositCount, finalized := d.tree.GetFinalized(finalized)
|
|
return fromTreeParts(finalized, depositCount, d.finalizedExecutionBlock)
|
|
}
|
|
|
|
// fromSnapshot returns a deposit tree from a deposit tree snapshot.
|
|
//
|
|
//nolint:unused
|
|
func fromSnapshot(snapshot DepositTreeSnapshot) (DepositTree, error) {
|
|
root, err := snapshot.CalculateRoot()
|
|
if err != nil {
|
|
return DepositTree{}, err
|
|
}
|
|
if snapshot.depositRoot != root {
|
|
return DepositTree{}, ErrInvalidSnapshotRoot
|
|
}
|
|
if snapshot.depositCount >= math.PowerOf2(uint64(DepositContractDepth)) {
|
|
return DepositTree{}, ErrTooManyDeposits
|
|
}
|
|
tree, err := fromSnapshotParts(snapshot.finalized, snapshot.depositCount, DepositContractDepth)
|
|
if err != nil {
|
|
return DepositTree{}, err
|
|
}
|
|
if snapshot.depositCount == 0 {
|
|
return DepositTree{}, ErrNoDeposits
|
|
}
|
|
return DepositTree{
|
|
tree: tree,
|
|
mixInLength: snapshot.depositCount,
|
|
finalizedExecutionBlock: snapshot.executionBlock,
|
|
}, nil
|
|
}
|
|
|
|
// finalize marks a deposit as finalized.
|
|
//
|
|
//nolint:unused
|
|
func (d *DepositTree) finalize(eth1data *eth.Eth1Data, executionBlockHeight uint64) error {
|
|
var blockHash [32]byte
|
|
copy(blockHash[:], eth1data.BlockHash)
|
|
d.finalizedExecutionBlock = executionBlock{
|
|
Hash: blockHash,
|
|
Depth: executionBlockHeight,
|
|
}
|
|
_, err := d.tree.Finalize(eth1data.DepositCount, DepositContractDepth)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// getProof returns the Deposit tree proof.
|
|
//
|
|
//nolint:unused
|
|
func (d *DepositTree) getProof(index uint64) ([32]byte, [][32]byte, error) {
|
|
if d.mixInLength <= 0 {
|
|
return [32]byte{}, nil, ErrInvalidMixInLength
|
|
}
|
|
finalizedDeposits, _ := d.tree.GetFinalized([][32]byte{})
|
|
if finalizedDeposits != 0 {
|
|
finalizedDeposits = finalizedDeposits - 1
|
|
}
|
|
if index <= finalizedDeposits {
|
|
return [32]byte{}, nil, ErrInvalidIndex
|
|
}
|
|
leaf, proof := generateProof(d.tree, index, DepositContractDepth)
|
|
var mixInLength [32]byte
|
|
copy(mixInLength[:], bytesutil.Uint64ToBytesLittleEndian32(d.mixInLength))
|
|
proof = append(proof, mixInLength)
|
|
return leaf, proof, nil
|
|
}
|
|
|
|
// getRoot returns the root of the deposit tree.
|
|
//
|
|
//nolint:unused
|
|
func (d *DepositTree) getRoot() [32]byte {
|
|
root := d.tree.GetRoot()
|
|
return sha256.Sum256(append(root[:], bytesutil.Uint64ToBytesLittleEndian32(d.mixInLength)...))
|
|
}
|
|
|
|
// pushLeaf adds a new leaf to the tree.
|
|
//
|
|
//nolint:unused
|
|
func (d *DepositTree) pushLeaf(leaf [32]byte) error {
|
|
var err error
|
|
d.tree, err = d.tree.PushLeaf(leaf, DepositContractDepth)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
d.mixInLength++
|
|
return nil
|
|
}
|