2022-03-09 03:05:51 +00:00
|
|
|
package doublylinkedtree
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2022-04-06 14:18:30 +00:00
|
|
|
|
2022-06-05 17:48:21 +00:00
|
|
|
"github.com/pkg/errors"
|
2022-04-06 14:18:30 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/config/params"
|
2022-03-09 03:05:51 +00:00
|
|
|
)
|
|
|
|
|
2022-05-02 13:53:22 +00:00
|
|
|
func (s *Store) setOptimisticToInvalid(ctx context.Context, root, parentRoot, payloadHash [32]byte) ([][32]byte, error) {
|
2022-04-06 14:18:30 +00:00
|
|
|
s.nodesLock.Lock()
|
|
|
|
invalidRoots := make([][32]byte, 0)
|
|
|
|
node, ok := s.nodeByRoot[root]
|
2022-05-02 13:53:22 +00:00
|
|
|
if !ok {
|
|
|
|
node, ok = s.nodeByRoot[parentRoot]
|
|
|
|
if !ok || node == nil {
|
|
|
|
s.nodesLock.Unlock()
|
2022-06-05 17:48:21 +00:00
|
|
|
return invalidRoots, errors.Wrap(ErrNilNode, "could not set node to invalid")
|
2022-05-02 13:53:22 +00:00
|
|
|
}
|
|
|
|
// return early if the parent is LVH
|
|
|
|
if node.payloadHash == payloadHash {
|
|
|
|
s.nodesLock.Unlock()
|
|
|
|
return invalidRoots, nil
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if node == nil {
|
|
|
|
s.nodesLock.Unlock()
|
2022-06-05 17:48:21 +00:00
|
|
|
return invalidRoots, errors.Wrap(ErrNilNode, "could not set node to invalid")
|
2022-05-02 13:53:22 +00:00
|
|
|
}
|
|
|
|
if node.parent.root != parentRoot {
|
|
|
|
s.nodesLock.Unlock()
|
|
|
|
return invalidRoots, errInvalidParentRoot
|
|
|
|
}
|
2022-04-06 14:18:30 +00:00
|
|
|
}
|
|
|
|
// Check if last valid hash is an ancestor of the passed node.
|
|
|
|
lastValid, ok := s.nodeByPayload[payloadHash]
|
|
|
|
if !ok || lastValid == nil {
|
|
|
|
s.nodesLock.Unlock()
|
|
|
|
return invalidRoots, errUnknownPayloadHash
|
|
|
|
}
|
|
|
|
firstInvalid := node
|
|
|
|
for ; firstInvalid.parent != nil && firstInvalid.parent.payloadHash != payloadHash; firstInvalid = firstInvalid.parent {
|
|
|
|
if ctx.Err() != nil {
|
|
|
|
s.nodesLock.Unlock()
|
|
|
|
return invalidRoots, ctx.Err()
|
|
|
|
}
|
|
|
|
}
|
2022-05-02 13:53:22 +00:00
|
|
|
// Deal with the case that the last valid payload is in a different fork
|
|
|
|
// This means we are dealing with an EE that does not follow the spec
|
2022-04-06 14:18:30 +00:00
|
|
|
if firstInvalid.parent == nil {
|
2022-05-02 13:53:22 +00:00
|
|
|
// return early if the invalid node was not imported
|
|
|
|
if node.root == parentRoot {
|
|
|
|
s.nodesLock.Unlock()
|
|
|
|
return invalidRoots, nil
|
|
|
|
}
|
2022-04-06 14:18:30 +00:00
|
|
|
firstInvalid = node
|
|
|
|
}
|
|
|
|
s.nodesLock.Unlock()
|
|
|
|
return s.removeNode(ctx, firstInvalid)
|
|
|
|
}
|
|
|
|
|
2022-03-09 03:05:51 +00:00
|
|
|
// removeNode removes the node with the given root and all of its children
|
|
|
|
// from the Fork Choice Store.
|
2022-04-06 14:18:30 +00:00
|
|
|
func (s *Store) removeNode(ctx context.Context, node *Node) ([][32]byte, error) {
|
2022-03-09 03:05:51 +00:00
|
|
|
s.nodesLock.Lock()
|
|
|
|
defer s.nodesLock.Unlock()
|
2022-03-23 18:55:05 +00:00
|
|
|
invalidRoots := make([][32]byte, 0)
|
2022-03-09 03:05:51 +00:00
|
|
|
|
2022-04-06 14:18:30 +00:00
|
|
|
if node == nil {
|
2022-06-05 17:48:21 +00:00
|
|
|
return invalidRoots, errors.Wrap(ErrNilNode, "could not remove node")
|
2022-03-09 03:05:51 +00:00
|
|
|
}
|
|
|
|
if !node.optimistic || node.parent == nil {
|
2022-03-23 18:55:05 +00:00
|
|
|
return invalidRoots, errInvalidOptimisticStatus
|
2022-03-09 03:05:51 +00:00
|
|
|
}
|
2022-04-06 14:18:30 +00:00
|
|
|
|
2022-03-09 03:05:51 +00:00
|
|
|
children := node.parent.children
|
|
|
|
if len(children) == 1 {
|
|
|
|
node.parent.children = []*Node{}
|
|
|
|
} else {
|
|
|
|
for i, n := range children {
|
|
|
|
if n == node {
|
|
|
|
if i != len(children)-1 {
|
|
|
|
children[i] = children[len(children)-1]
|
|
|
|
}
|
2022-05-10 20:02:00 +00:00
|
|
|
node.parent.children = children[:len(children)-1]
|
2022-03-09 03:05:51 +00:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-03-23 18:55:05 +00:00
|
|
|
return s.removeNodeAndChildren(ctx, node, invalidRoots)
|
2022-03-09 03:05:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// removeNodeAndChildren removes `node` and all of its descendant from the Store
|
2022-03-23 18:55:05 +00:00
|
|
|
func (s *Store) removeNodeAndChildren(ctx context.Context, node *Node, invalidRoots [][32]byte) ([][32]byte, error) {
|
|
|
|
var err error
|
2022-03-09 03:05:51 +00:00
|
|
|
for _, child := range node.children {
|
|
|
|
if ctx.Err() != nil {
|
2022-03-23 18:55:05 +00:00
|
|
|
return invalidRoots, ctx.Err()
|
2022-03-09 03:05:51 +00:00
|
|
|
}
|
2022-03-23 18:55:05 +00:00
|
|
|
if invalidRoots, err = s.removeNodeAndChildren(ctx, child, invalidRoots); err != nil {
|
|
|
|
return invalidRoots, err
|
2022-03-09 03:05:51 +00:00
|
|
|
}
|
|
|
|
}
|
2022-03-23 18:55:05 +00:00
|
|
|
invalidRoots = append(invalidRoots, node.root)
|
2022-04-06 14:18:30 +00:00
|
|
|
s.proposerBoostLock.Lock()
|
|
|
|
if node.root == s.proposerBoostRoot {
|
|
|
|
s.proposerBoostRoot = [32]byte{}
|
|
|
|
}
|
|
|
|
if node.root == s.previousProposerBoostRoot {
|
|
|
|
s.previousProposerBoostRoot = params.BeaconConfig().ZeroHash
|
|
|
|
s.previousProposerBoostScore = 0
|
|
|
|
}
|
|
|
|
s.proposerBoostLock.Unlock()
|
2022-03-09 03:05:51 +00:00
|
|
|
delete(s.nodeByRoot, node.root)
|
2022-04-06 14:18:30 +00:00
|
|
|
delete(s.nodeByPayload, node.payloadHash)
|
2022-03-23 18:55:05 +00:00
|
|
|
return invalidRoots, nil
|
2022-03-09 03:05:51 +00:00
|
|
|
}
|