Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Disentangle Ethereum events logic from vote extensions #243

Merged
merged 25 commits into from
Aug 4, 2022
Merged
Show file tree
Hide file tree
Changes from 20 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 2 additions & 2 deletions apps/src/lib/node/ledger/protocol/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -14,10 +14,10 @@ use namada::ledger::storage::{DBIter, Storage, StorageHasher, DB};
use namada::ledger::treasury::TreasuryVp;
use namada::proto::{self, Tx};
use namada::types::address::{Address, InternalAddress};
use namada::types::ethereum_events::vote_extensions::VoteExtensionDigest;
use namada::types::storage;
use namada::types::transaction::protocol::{ProtocolTx, ProtocolTxType};
use namada::types::transaction::{DecryptedTx, TxResult, TxType, VpsResult};
use namada::types::vote_extensions::ethereum_events::EthEventsVextDigest;
use namada::vm::wasm::{TxCache, VpCache};
use namada::vm::{self, wasm, WasmCacheAccess};
use rayon::iter::{IntoParallelRefIterator, ParallelIterator};
Expand Down Expand Up @@ -159,7 +159,7 @@ where
}
TxType::Protocol(ProtocolTx {
tx:
ProtocolTxType::EthereumEvents(VoteExtensionDigest {
ProtocolTxType::EthereumEvents(EthEventsVextDigest {
events, ..
}),
..
Expand Down
109 changes: 59 additions & 50 deletions apps/src/lib/node/ledger/shell/prepare_proposal.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,10 +4,11 @@
mod prepare_block {
use std::collections::{BTreeMap, HashMap, HashSet};

use namada::types::ethereum_events::vote_extensions::{
FractionalVotingPower, MultiSignedEthEvent, VoteExtensionDigest,
};
use namada::types::transaction::protocol::ProtocolTxType;
use namada::types::vote_extensions::ethereum_events::{
EthEventsVextDigest, MultiSignedEthEvent,
};
use namada::types::voting_power::FractionalVotingPower;
use tendermint_proto::abci::{
ExtendedCommitInfo, ExtendedVoteInfo, TxRecord,
};
Expand Down Expand Up @@ -66,7 +67,7 @@ mod prepare_block {
}

/// Builds a batch of vote extension transactions, comprised of Ethereum
/// events
/// events and, optionally, a validator set update
sug0 marked this conversation as resolved.
Show resolved Hide resolved
fn build_vote_extensions_txs(
&mut self,
local_last_commit: Option<ExtendedCommitInfo>,
Expand All @@ -79,7 +80,7 @@ mod prepare_block {
let vote_extension_digest =
local_last_commit.and_then(|local_last_commit| {
let votes = local_last_commit.votes;
self.compress_vote_extensions(votes)
self.compress_ethereum_events(votes)
});
let vote_extension_digest =
match (vote_extension_digest, self.storage.last_height) {
Expand All @@ -88,22 +89,22 @@ mod prepare_block {
(Some(_), BlockHeight(0)) => {
unreachable!(
"We already handle this scenario in \
validate_vote_extension."
validate_eth_events_vext."
)
}
// handle block heights > 0
(Some(digest), _) => digest,
_ => unreachable!(
"Honest Namada validators will always sign a \
VoteExtension, even if no Ethereum events were \
observed at a given block height. In fact, a quorum \
of signed empty VoteExtension commits the fact no \
events were observed by a majority of validators. \
Likewise, a Tendermint quorum should never decide on \
a block including vote extensions reflecting less \
than or equal to 2/3 of the total stake. These \
scenarios are virtually impossible, so we will panic \
here."
"Honest Namada validators will always sign \
EthEventsVext instances, even if no Ethereum events \
were observed at a given block height. In fact, a \
quorum of signed empty EthEventsVext instances \
commits the fact no events were observed by a \
majority of validators. Likewise, a Tendermint \
quorum should never decide on a block including vote \
extensions reflecting less than or equal to 2/3 of \
the total stake. These scenarios are virtually \
impossible, so we will panic here."
),
};

Expand All @@ -112,6 +113,8 @@ mod prepare_block {
.to_bytes();
let tx_record = record::add(tx);

// TODO: include here a validator set update tx,
// if we are at the end of an epoch
vec![tx_record]
}

Expand Down Expand Up @@ -160,13 +163,16 @@ mod prepare_block {
.collect()
}

/// Compresses a set of vote extensions into a single
/// [`VoteExtensionDigest`], whilst filtering invalid
/// [`SignedExt`] instances in the process
fn compress_vote_extensions(
/// Compresses a set of signed Ethereum events into a single
/// [`EthEventsVextDigest`], whilst filtering invalid
/// [`SignedEthEventsVext`] instances in the process
// TODO: rename this as `compress_vote_extensions`, and return
// a `VoteExtensionDigest`, which will contain both digests of
// ethereum events and validator set update vote extensions
fn compress_ethereum_events(
&self,
vote_extensions: Vec<ExtendedVoteInfo>,
) -> Option<VoteExtensionDigest> {
) -> Option<EthEventsVextDigest> {
let events_epoch = self
.storage
.block
Expand Down Expand Up @@ -219,15 +225,15 @@ mod prepare_block {
?sig,
?validator_addr,
"Overwrote old signature from validator while \
constructing VoteExtensionDigest"
constructing EthEventsVextDigest"
);
}
}

if voting_power <= FractionalVotingPower::TWO_THIRDS {
tracing::error!(
"Tendermint has decided on a block including vote \
extensions reflecting <= 2/3 of the total stake"
"Tendermint has decided on a block including Ethereum \
events reflecting <= 2/3 of the total stake"
);
return None;
}
Expand All @@ -237,7 +243,7 @@ mod prepare_block {
.map(|(event, signers)| MultiSignedEthEvent { event, signers })
.collect();

Some(VoteExtensionDigest { events, signatures })
Some(EthEventsVextDigest { events, signatures })
}
}

Expand Down Expand Up @@ -286,18 +292,19 @@ mod prepare_block {
};
use namada::proto::SignedTxData;
use namada::types::address::xan;
use namada::types::ethereum_events::vote_extensions::VoteExtension;
use namada::types::ethereum_events::EthereumEvent;
use namada::types::key::common;
use namada::types::storage::{BlockHeight, Epoch};
use namada::types::transaction::protocol::ProtocolTxType;
use namada::types::transaction::{Fee, TxType};
use namada::types::vote_extensions::ethereum_events::{
EthEventsVext, SignedEthEventsVext,
};
use tendermint_proto::abci::tx_record::TxAction;
use tendermint_proto::abci::{
ExtendedCommitInfo, ExtendedVoteInfo, TxRecord,
};

use super::super::super::vote_extensions::SignedExt;
use super::*;
use crate::node::ledger::shell::test_utils::{
self, gen_keypair, TestShell,
Expand Down Expand Up @@ -326,18 +333,20 @@ mod prepare_block {
);
}

/// Serialize a [`SignedExt`] to an [`ExtendedVoteInfo`]
fn vote_extension_serialize(vext: SignedExt) -> ExtendedVoteInfo {
/// Serialize a [`SignedEthEventsVext`] to an [`ExtendedVoteInfo`]
fn vote_extension_serialize(
vext: SignedEthEventsVext,
) -> ExtendedVoteInfo {
ExtendedVoteInfo {
vote_extension: vext.try_to_vec().unwrap(),
..Default::default()
}
}

/// Check if we are filtering out an invalid vote extension `vext`
fn check_vote_extension_filtering(
fn check_eth_events_filtering(
shell: &mut TestShell,
vext: SignedExt,
vext: SignedEthEventsVext,
) {
let votes =
deserialize_vote_extensions(vec![vote_extension_serialize(
Expand All @@ -349,7 +358,7 @@ mod prepare_block {
assert_eq!(filtered_votes, vec![]);
}

/// Test if we are filtering out vote extensinos with bad
/// Test if we are filtering out Ethereum events with bad
/// signatures in a prepare proposal.
#[test]
fn test_prepare_proposal_filter_out_bad_vext_signatures() {
Expand All @@ -365,7 +374,7 @@ mod prepare_block {
let validator_addr = wallet::defaults::validator_address();

// generate a valid signature
let mut ext = VoteExtension {
let mut ext = EthEventsVext {
validator_addr,
block_height: LAST_HEIGHT,
ethereum_events: vec![],
Expand All @@ -378,10 +387,10 @@ mod prepare_block {
ext
};

check_vote_extension_filtering(&mut shell, signed_vote_extension);
check_eth_events_filtering(&mut shell, signed_vote_extension);
}

/// Test if we are filtering out vote extensinos for
/// Test if we are filtering out Ethereum events seen at
/// block heights different than the last height.
#[test]
fn test_prepare_proposal_filter_out_bad_vext_bheights() {
Expand All @@ -398,7 +407,7 @@ mod prepare_block {
let validator_addr = wallet::defaults::validator_address();

let signed_vote_extension = {
let ext = VoteExtension {
let ext = EthEventsVext {
validator_addr,
block_height: PRED_LAST_HEIGHT,
ethereum_events: vec![],
Expand All @@ -408,10 +417,10 @@ mod prepare_block {
ext
};

check_vote_extension_filtering(&mut shell, signed_vote_extension);
check_eth_events_filtering(&mut shell, signed_vote_extension);
}

/// Test if we are filtering out vote extensinos for
/// Test if we are filtering out Ethereum events seen by
/// non-validator nodes.
#[test]
fn test_prepare_proposal_filter_out_bad_vext_validators() {
Expand All @@ -429,7 +438,7 @@ mod prepare_block {
};

let signed_vote_extension = {
let ext = VoteExtension {
let ext = EthEventsVext {
validator_addr,
block_height: LAST_HEIGHT,
ethereum_events: vec![],
Expand All @@ -439,7 +448,7 @@ mod prepare_block {
ext
};

check_vote_extension_filtering(&mut shell, signed_vote_extension);
check_eth_events_filtering(&mut shell, signed_vote_extension);
}

/// Test if we are filtering out duped Ethereum events in
Expand All @@ -462,7 +471,7 @@ mod prepare_block {
};
let signed_vote_extension = {
let ev = ethereum_event;
let ext = VoteExtension {
let ext = EthEventsVext {
validator_addr,
block_height: LAST_HEIGHT,
ethereum_events: vec![ev.clone(), ev.clone(), ev],
Expand All @@ -475,7 +484,7 @@ mod prepare_block {
let maybe_digest = {
let votes =
vec![vote_extension_serialize(signed_vote_extension)];
shell.compress_vote_extensions(votes)
shell.compress_ethereum_events(votes)
};

// we should be filtering out the vote extension with
Expand All @@ -485,13 +494,13 @@ mod prepare_block {
assert!(maybe_digest.is_none());
}

/// Creates a vote extension digest manually, and encodes it as a
/// Creates an Ethereum events digest manually, and encodes it as a
/// [`TxRecord`].
fn manually_assemble_digest(
_protocol_key: &common::SecretKey,
ext: SignedExt,
ext: SignedEthEventsVext,
last_height: BlockHeight,
) -> VoteExtensionDigest {
) -> EthEventsVextDigest {
let events = vec![MultiSignedEthEvent {
event: ext.data.ethereum_events[0].clone(),
signers: {
Expand All @@ -507,7 +516,7 @@ mod prepare_block {
};

let vote_extension_digest =
VoteExtensionDigest { events, signatures };
EthEventsVextDigest { events, signatures };

assert_eq!(
vec![ext],
Expand All @@ -522,7 +531,7 @@ mod prepare_block {
// super::record::add(tx)
}

/// Test if vote extension validation and inclusion in a block
/// Test if Ethereum events validation and inclusion in a block
/// behaves as expected, considering honest validators.
#[test]
fn test_prepare_proposal_vext_normal_op() {
Expand All @@ -541,7 +550,7 @@ mod prepare_block {
transfers: vec![],
};
let signed_vote_extension = {
let ext = VoteExtension {
let ext = EthEventsVext {
validator_addr,
block_height: LAST_HEIGHT,
ethereum_events: vec![ethereum_event],
Expand Down Expand Up @@ -599,7 +608,7 @@ mod prepare_block {
// assert_eq!(rsp.tx_records, vec![digest]);
}

/// Test if vote extension validation and inclusion in a block
/// Test if Ethereum events validation and inclusion in a block
/// behaves as expected, considering <= 2/3 voting power.
#[test]
#[should_panic(expected = "Honest Namada validators")]
Expand Down Expand Up @@ -656,7 +665,7 @@ mod prepare_block {
transfers: vec![],
};
let signed_vote_extension = {
let ext = VoteExtension {
let ext = EthEventsVext {
validator_addr,
block_height: LAST_HEIGHT,
ethereum_events: vec![ethereum_event],
Expand Down
Loading