mirror of
https://github.com/logos-co/nomos-specs.git
synced 2025-01-09 23:26:01 +00:00
181 lines
7.3 KiB
Python
181 lines
7.3 KiB
Python
import random
|
|
from typing import Dict, List
|
|
from unittest import TestCase
|
|
from itertools import chain
|
|
|
|
from blspy import PrivateKey
|
|
|
|
from carnot.carnot import Id, Carnot, Block, Overlay, Vote, StandardQc, NewView
|
|
from carnot.beacon import generate_random_sk, RandomBeacon, NormalMode, RecoveryMode
|
|
from carnot.beaconized_carnot import BeaconizedCarnot, BeaconizedBlock
|
|
from carnot.overlay import FlatOverlay, EntropyOverlay
|
|
from carnot.test_unhappy_path import parents_from_childs
|
|
|
|
|
|
def gen_node(sk: PrivateKey, overlay: Overlay, entropy: bytes = b""):
|
|
node = BeaconizedCarnot(sk, overlay, entropy)
|
|
return node.id, node
|
|
|
|
|
|
def succeed(nodes: Dict[Id, BeaconizedCarnot], proposed_block: BeaconizedBlock, overlay: EntropyOverlay) -> (List[Vote], EntropyOverlay):
|
|
overlay = overlay.advance(proposed_block.beacon.entropy())
|
|
# broadcast the block
|
|
for node in nodes.values():
|
|
node.receive_block(proposed_block)
|
|
|
|
votes = {}
|
|
childs_ids = list(chain.from_iterable(overlay.leaf_committees()))
|
|
leafs = [nodes[_id] for _id in childs_ids]
|
|
for node in leafs:
|
|
vote = node.approve_block(proposed_block, set()).payload
|
|
votes[node.id] = vote
|
|
|
|
while len(parents := parents_from_childs(overlay, childs_ids)) != 0:
|
|
for node_id in parents:
|
|
node = nodes[node_id]
|
|
child_votes = [votes[_id] for _id in votes.keys() if overlay.is_member_of_child_committee(node_id, _id)]
|
|
vote = node.approve_block(proposed_block, child_votes).payload
|
|
votes[node_id] = vote
|
|
childs_ids = list(set(parents))
|
|
|
|
root_votes = [
|
|
votes[node_id]
|
|
for node_id in nodes
|
|
if overlay.is_member_of_root_committee(node_id) or overlay.is_child_of_root_committee(node_id)
|
|
]
|
|
return root_votes, overlay
|
|
|
|
|
|
def fail(nodes: Dict[Id, BeaconizedCarnot], proposed_block: BeaconizedBlock, overlay: EntropyOverlay) -> (List[NewView], EntropyOverlay):
|
|
overlay = overlay.advance(proposed_block.beacon.entropy())
|
|
# broadcast the block
|
|
for node in nodes.values():
|
|
node.receive_block(proposed_block)
|
|
|
|
node: BeaconizedCarnot
|
|
timeouts = []
|
|
for node in (nodes[_id] for _id in nodes if overlay.is_member_of_root_committee(_id) or overlay.is_child_of_root_committee(_id)):
|
|
timeout = node.local_timeout().payload
|
|
timeouts.append(timeout)
|
|
|
|
root_member = next(nodes[_id] for _id in nodes if overlay.is_member_of_root_committee(_id))
|
|
timeouts = [timeouts[i] for i in range(overlay.leader_super_majority_threshold(root_member.id))]
|
|
timeout_qc = root_member.timeout_detected(timeouts).payload
|
|
|
|
for node in nodes.values():
|
|
node.receive_timeout_qc(timeout_qc)
|
|
|
|
overlay = next(iter(nodes.values())).overlay
|
|
|
|
votes = {}
|
|
childs_ids = list(chain.from_iterable(overlay.leaf_committees()))
|
|
leafs = [nodes[_id] for _id in childs_ids]
|
|
for node in leafs:
|
|
vote = node.approve_new_view(timeout_qc, set()).payload
|
|
votes[node.id] = vote
|
|
|
|
while len(parents := parents_from_childs(overlay, childs_ids)) != 0:
|
|
for node_id in parents:
|
|
node = nodes[node_id]
|
|
child_votes = [votes[_id] for _id in votes.keys() if overlay.is_member_of_child_committee(node_id, _id)]
|
|
if len(child_votes) == overlay.super_majority_threshold(node_id) and node_id not in votes:
|
|
vote = node.approve_new_view(timeout_qc, child_votes).payload
|
|
votes[node_id] = vote
|
|
childs_ids = list(set(parents))
|
|
|
|
root_votes = [
|
|
votes[node_id]
|
|
for node_id in nodes
|
|
if overlay.is_member_of_root_committee(node_id) or overlay.is_child_of_root_committee(node_id)
|
|
]
|
|
return root_votes, overlay
|
|
|
|
|
|
def add_genesis_block(carnot: BeaconizedCarnot, sk: PrivateKey) -> Block:
|
|
beacon = NormalMode.generate_beacon(sk, -1)
|
|
genesis_block = BeaconizedBlock(
|
|
view=0,
|
|
qc=StandardQc(block=b"", view=0),
|
|
_id=b"",
|
|
beacon=beacon,
|
|
pk=sk.get_g1()
|
|
)
|
|
carnot.safe_blocks[genesis_block.id()] = genesis_block
|
|
carnot.receive_block(genesis_block)
|
|
carnot.local_high_qc = genesis_block.qc
|
|
carnot.current_view = 1
|
|
carnot.overlay = carnot.overlay.advance(beacon.entropy())
|
|
return genesis_block
|
|
|
|
|
|
def initial_setup(test_case: TestCase, size: int) -> (Dict[Id, Carnot], Carnot, Block, EntropyOverlay):
|
|
keys = [generate_random_sk() for _ in range(size)]
|
|
nodes_ids = [bytes(key.get_g1()) for key in keys]
|
|
genesis_sk = generate_random_sk()
|
|
entropy = RecoveryMode.generate_beacon(bytes(genesis_sk), -1).entropy()
|
|
|
|
random.seed(a=entropy, version=2)
|
|
current_leader = random.choice(nodes_ids)
|
|
|
|
nodes = dict(gen_node(key, FlatOverlay(current_leader, nodes_ids, entropy), entropy) for key in keys)
|
|
genesis_block = None
|
|
overlay = FlatOverlay(current_leader, nodes_ids, entropy)
|
|
leader: Carnot = nodes[overlay.leader()]
|
|
for node in nodes.values():
|
|
genesis_block = add_genesis_block(node, genesis_sk)
|
|
# votes for genesis block
|
|
genesis_votes = set(
|
|
Vote(
|
|
block=genesis_block.id(),
|
|
view=0,
|
|
voter=nodes_ids[i],
|
|
qc=StandardQc(
|
|
block=genesis_block.id(),
|
|
view=0
|
|
),
|
|
) for i in range(overlay.leader_super_majority_threshold(overlay.leader()))
|
|
)
|
|
proposed_block = leader.propose_block(1, genesis_votes).payload
|
|
test_case.assertIsNotNone(proposed_block)
|
|
overlay = overlay.advance(genesis_block.beacon.entropy())
|
|
return nodes, leader, proposed_block, overlay
|
|
|
|
|
|
class TestBeaconizedCarnot(TestCase):
|
|
def test_interleave_success_fails(self):
|
|
"""
|
|
At the end of the timeout the highQC in the next leader's aggregatedQC should be the highestQC held by the
|
|
majority of nodes or a qc higher than th highestQC held by the majority of nodes.
|
|
Majority means more than two thirds of total number of nodes, randomly assigned to committees.
|
|
"""
|
|
leader: BeaconizedCarnot
|
|
nodes, leader, proposed_block, overlay = initial_setup(self, 5)
|
|
|
|
for view in range(2, 5):
|
|
root_votes, overlay = succeed(nodes, proposed_block, overlay)
|
|
leader = nodes[overlay.leader()]
|
|
proposed_block = leader.propose_block(view, root_votes).payload
|
|
|
|
root_votes, overlay = fail(nodes, proposed_block, overlay)
|
|
leader = nodes[overlay.leader()]
|
|
proposed_block = leader.propose_block(6, root_votes).payload
|
|
|
|
for view in range(7, 8):
|
|
root_votes, overlay = succeed(nodes, proposed_block, overlay)
|
|
leader = nodes[overlay.leader()]
|
|
proposed_block = leader.propose_block(view, root_votes).payload
|
|
|
|
root_votes, overlay = fail(nodes, proposed_block, overlay)
|
|
leader = nodes[overlay.leader()]
|
|
proposed_block = leader.propose_block(9, root_votes).payload
|
|
|
|
for view in range(10, 15):
|
|
root_votes, overlay = succeed(nodes, proposed_block, overlay)
|
|
leader = nodes[overlay.leader()]
|
|
proposed_block = leader.propose_block(view, root_votes).payload
|
|
|
|
committed_blocks = [view for view in range(1, 11) if view not in (4, 5, 7, 8, 9, 10, 11, 12, 13)]
|
|
for node in nodes.values():
|
|
for view in committed_blocks:
|
|
self.assertIn(view, [block.view for block in node.committed_blocks().values()])
|