mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-17 23:38:46 +00:00
f0fd29d367
* add synced_tips structure * create optimisticStore type * Add ForkChoice.Optimistic() Adds the logic implementing whether a synced block is optimistic or not. * Terence's bug catch * Reinforce warning * Preston's review * add comment on optimistic sync
203 lines
4.7 KiB
Go
203 lines
4.7 KiB
Go
package protoarray
|
|
|
|
import (
|
|
"context"
|
|
"testing"
|
|
|
|
types "github.com/prysmaticlabs/eth2-types"
|
|
"github.com/prysmaticlabs/prysm/encoding/bytesutil"
|
|
"github.com/prysmaticlabs/prysm/testing/require"
|
|
)
|
|
|
|
// We test the algorithm to check the optimistic status of a node. The
|
|
// status for this test is the following branching diagram
|
|
//
|
|
// -- E -- F
|
|
// /
|
|
// -- C -- D
|
|
// /
|
|
// 0 -- 1 -- A -- B -- J -- K
|
|
// \ /
|
|
// -- G -- H -- I
|
|
//
|
|
// Here nodes 0, 1, A, B, C, D are fully validated and nodes
|
|
// E, F, G, H, J, K are optimistic.
|
|
// Synced Tips are nodes B, C, D
|
|
// nodes 0 and 1 are outside the Fork Choice Store.
|
|
|
|
func TestOptimistic(t *testing.T) {
|
|
root0 := bytesutil.ToBytes32([]byte("hello0"))
|
|
slot0 := types.Slot(98)
|
|
root1 := bytesutil.ToBytes32([]byte("hello1"))
|
|
slot1 := types.Slot(99)
|
|
|
|
nodeA := &Node{
|
|
slot: types.Slot(100),
|
|
root: bytesutil.ToBytes32([]byte("helloA")),
|
|
bestChild: 1,
|
|
}
|
|
nodeB := &Node{
|
|
slot: types.Slot(101),
|
|
root: bytesutil.ToBytes32([]byte("helloB")),
|
|
bestChild: 2,
|
|
parent: 0,
|
|
}
|
|
nodeC := &Node{
|
|
slot: types.Slot(102),
|
|
root: bytesutil.ToBytes32([]byte("helloC")),
|
|
bestChild: 3,
|
|
parent: 1,
|
|
}
|
|
nodeD := &Node{
|
|
slot: types.Slot(103),
|
|
root: bytesutil.ToBytes32([]byte("helloD")),
|
|
bestChild: NonExistentNode,
|
|
parent: 2,
|
|
}
|
|
nodeE := &Node{
|
|
slot: types.Slot(103),
|
|
root: bytesutil.ToBytes32([]byte("helloE")),
|
|
bestChild: 5,
|
|
parent: 2,
|
|
}
|
|
nodeF := &Node{
|
|
slot: types.Slot(104),
|
|
root: bytesutil.ToBytes32([]byte("helloF")),
|
|
bestChild: NonExistentNode,
|
|
parent: 4,
|
|
}
|
|
nodeG := &Node{
|
|
slot: types.Slot(102),
|
|
root: bytesutil.ToBytes32([]byte("helloG")),
|
|
bestChild: 7,
|
|
parent: 1,
|
|
}
|
|
nodeH := &Node{
|
|
slot: types.Slot(103),
|
|
root: bytesutil.ToBytes32([]byte("helloH")),
|
|
bestChild: 8,
|
|
parent: 6,
|
|
}
|
|
nodeI := &Node{
|
|
slot: types.Slot(104),
|
|
root: bytesutil.ToBytes32([]byte("helloI")),
|
|
bestChild: NonExistentNode,
|
|
parent: 7,
|
|
}
|
|
nodeJ := &Node{
|
|
slot: types.Slot(103),
|
|
root: bytesutil.ToBytes32([]byte("helloJ")),
|
|
bestChild: 10,
|
|
parent: 6,
|
|
}
|
|
nodeK := &Node{
|
|
slot: types.Slot(104),
|
|
root: bytesutil.ToBytes32([]byte("helloK")),
|
|
bestChild: NonExistentNode,
|
|
parent: 9,
|
|
}
|
|
nodes := []*Node{
|
|
nodeA,
|
|
nodeB,
|
|
nodeC,
|
|
nodeD,
|
|
nodeE,
|
|
nodeF,
|
|
nodeG,
|
|
nodeH,
|
|
nodeI,
|
|
nodeJ,
|
|
nodeK,
|
|
}
|
|
ni := map[[32]byte]uint64{
|
|
nodeA.root: 0,
|
|
nodeB.root: 1,
|
|
nodeC.root: 2,
|
|
nodeD.root: 3,
|
|
nodeE.root: 4,
|
|
nodeF.root: 5,
|
|
nodeG.root: 6,
|
|
nodeH.root: 7,
|
|
nodeI.root: 8,
|
|
nodeJ.root: 9,
|
|
nodeK.root: 10,
|
|
}
|
|
|
|
s := &Store{
|
|
nodes: nodes,
|
|
nodesIndices: ni,
|
|
}
|
|
|
|
tips := map[[32]byte]types.Slot{
|
|
nodeB.root: nodeB.slot,
|
|
nodeC.root: nodeC.slot,
|
|
nodeD.root: nodeD.slot,
|
|
}
|
|
st := &optimisticStore{
|
|
validatedTips: tips,
|
|
}
|
|
f := &ForkChoice{
|
|
store: s,
|
|
syncedTips: st,
|
|
}
|
|
ctx := context.Background()
|
|
// We test the implementation of boundarySyncedTips
|
|
min, max := f.boundarySyncedTips()
|
|
require.Equal(t, min, types.Slot(101), "minimum tip slot is different")
|
|
require.Equal(t, max, types.Slot(103), "maximum tip slot is different")
|
|
|
|
// We test first nodes outside the Fork Choice store
|
|
op, err := f.Optimistic(ctx, root0, slot0)
|
|
require.NoError(t, err)
|
|
require.Equal(t, op, false)
|
|
|
|
op, err = f.Optimistic(ctx, root1, slot1)
|
|
require.NoError(t, err)
|
|
require.Equal(t, op, false)
|
|
|
|
// We check all nodes in the Fork Choice store.
|
|
op, err = f.Optimistic(ctx, nodeA.root, nodeA.slot)
|
|
require.NoError(t, err)
|
|
require.Equal(t, op, false)
|
|
|
|
op, err = f.Optimistic(ctx, nodeB.root, nodeB.slot)
|
|
require.NoError(t, err)
|
|
require.Equal(t, op, false)
|
|
|
|
op, err = f.Optimistic(ctx, nodeC.root, nodeC.slot)
|
|
require.NoError(t, err)
|
|
require.Equal(t, op, false)
|
|
|
|
op, err = f.Optimistic(ctx, nodeD.root, nodeD.slot)
|
|
require.NoError(t, err)
|
|
require.Equal(t, op, false)
|
|
|
|
op, err = f.Optimistic(ctx, nodeE.root, nodeE.slot)
|
|
require.NoError(t, err)
|
|
require.Equal(t, op, true)
|
|
|
|
op, err = f.Optimistic(ctx, nodeF.root, nodeF.slot)
|
|
require.NoError(t, err)
|
|
require.Equal(t, op, true)
|
|
|
|
op, err = f.Optimistic(ctx, nodeG.root, nodeG.slot)
|
|
require.NoError(t, err)
|
|
require.Equal(t, op, true)
|
|
|
|
op, err = f.Optimistic(ctx, nodeH.root, nodeH.slot)
|
|
require.NoError(t, err)
|
|
require.Equal(t, op, true)
|
|
|
|
op, err = f.Optimistic(ctx, nodeI.root, nodeI.slot)
|
|
require.NoError(t, err)
|
|
require.Equal(t, op, true)
|
|
|
|
op, err = f.Optimistic(ctx, nodeJ.root, nodeJ.slot)
|
|
require.NoError(t, err)
|
|
require.Equal(t, op, true)
|
|
|
|
op, err = f.Optimistic(ctx, nodeK.root, nodeK.slot)
|
|
require.NoError(t, err)
|
|
require.Equal(t, op, true)
|
|
}
|