Add ssz enc/dec for beacon block

This commit is contained in:
Paul Hauner 2018-12-23 23:41:22 +11:00
parent 368a218af4
commit 6f039e6e6a
No known key found for this signature in database
GPG Key ID: 303E4494BB28068C
21 changed files with 422 additions and 181 deletions

View File

@ -2,10 +2,11 @@
name = "types" name = "types"
version = "0.1.0" version = "0.1.0"
authors = ["Paul Hauner <paul@paulhauner.com>"] authors = ["Paul Hauner <paul@paulhauner.com>"]
edition = "2018"
[dependencies] [dependencies]
bls = { path = "../utils/bls" } bls = { path = "../utils/bls" }
boolean-bitfield = { path = "../utils/boolean-bitfield" } boolean-bitfield = { path = "../utils/boolean-bitfield" }
ethereum-types = "0.4.0" ethereum-types = "0.4.0"
rand = "0.3" rand = "0.5.5"
ssz = { path = "../utils/ssz" } ssz = { path = "../utils/ssz" }

View File

@ -1,14 +1,8 @@
use super::attestation_data::SSZ_ATTESTION_DATA_LENGTH; use super::bls::AggregateSignature;
use super::bls::{AggregateSignature, BLS_AGG_SIG_BYTE_SIZE}; use super::ssz::{Decodable, DecodeError, Encodable, SszStream};
use super::ssz::{Decodable, DecodeError, Encodable, SszStream, LENGTH_BYTES};
use super::{AttestationData, Bitfield}; use super::{AttestationData, Bitfield};
use crate::random::TestRandom;
pub const MIN_SSZ_ATTESTION_RECORD_LENGTH: usize = { use rand::RngCore;
SSZ_ATTESTION_DATA_LENGTH + // data
5 + // participation_bitfield (assuming 1 byte of bitfield)
5 + // custody_bitfield (assuming 1 byte of bitfield)
LENGTH_BYTES + BLS_AGG_SIG_BYTE_SIZE // aggregate sig
};
#[derive(Debug, Clone, PartialEq)] #[derive(Debug, Clone, PartialEq)]
pub struct Attestation { pub struct Attestation {
@ -55,30 +49,31 @@ impl Attestation {
} }
} }
impl<T: RngCore> TestRandom<T> for Attestation {
fn random_for_test(rng: &mut T) -> Self {
Self {
data: <_>::random_for_test(rng),
participation_bitfield: <_>::random_for_test(rng),
custody_bitfield: <_>::random_for_test(rng),
aggregate_sig: <_>::random_for_test(rng),
}
}
}
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::super::ssz::ssz_encode; use super::super::ssz::ssz_encode;
use super::*; use super::*;
use crate::random::TestRandom;
use rand::{prng::XorShiftRng, SeedableRng};
#[test] #[test]
pub fn test_attestation_record_min_ssz_length() { pub fn test_ssz_round_trip() {
let ar = Attestation::zero(); let mut rng = XorShiftRng::from_seed([42; 16]);
let ssz = ssz_encode(&ar); let original = Attestation::random_for_test(&mut rng);
assert_eq!(ssz.len(), MIN_SSZ_ATTESTION_RECORD_LENGTH); let bytes = ssz_encode(&original);
} let (decoded, _) = <_>::ssz_decode(&bytes, 0).unwrap();
#[test]
pub fn test_attestation_record_ssz_round_trip() {
let original = Attestation {
data: AttestationData::zero(),
participation_bitfield: Bitfield::from_bytes(&vec![17; 42][..]),
custody_bitfield: Bitfield::from_bytes(&vec![18; 12][..]),
aggregate_sig: AggregateSignature::new(),
};
let ssz = ssz_encode(&original);
let (decoded, _) = Attestation::ssz_decode(&ssz, 0).unwrap();
assert_eq!(original, decoded); assert_eq!(original, decoded);
} }

View File

@ -1,5 +1,7 @@
use super::ssz::{Decodable, DecodeError, Encodable, SszStream}; use super::ssz::{Decodable, DecodeError, Encodable, SszStream};
use super::Hash256; use super::Hash256;
use crate::random::TestRandom;
use rand::RngCore;
pub const SSZ_ATTESTION_DATA_LENGTH: usize = { pub const SSZ_ATTESTION_DATA_LENGTH: usize = {
8 + // slot 8 + // slot
@ -83,27 +85,35 @@ impl Decodable for AttestationData {
} }
} }
impl<T: RngCore> TestRandom<T> for AttestationData {
fn random_for_test(rng: &mut T) -> Self {
Self {
slot: <_>::random_for_test(rng),
shard: <_>::random_for_test(rng),
beacon_block_hash: <_>::random_for_test(rng),
epoch_boundary_hash: <_>::random_for_test(rng),
shard_block_hash: <_>::random_for_test(rng),
latest_crosslink_hash: <_>::random_for_test(rng),
justified_slot: <_>::random_for_test(rng),
justified_block_hash: <_>::random_for_test(rng),
}
}
}
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::super::ssz::ssz_encode; use super::super::ssz::ssz_encode;
use super::*; use super::*;
use crate::random::TestRandom;
use rand::{prng::XorShiftRng, SeedableRng};
#[test] #[test]
pub fn test_attestation_record_ssz_round_trip() { pub fn test_ssz_round_trip() {
let original = AttestationData { let mut rng = XorShiftRng::from_seed([42; 16]);
slot: 42, let original = AttestationData::random_for_test(&mut rng);
shard: 16,
beacon_block_hash: Hash256::from("beacon".as_bytes()),
epoch_boundary_hash: Hash256::from("epoch".as_bytes()),
shard_block_hash: Hash256::from("shard".as_bytes()),
latest_crosslink_hash: Hash256::from("xlink".as_bytes()),
justified_slot: 8,
justified_block_hash: Hash256::from("justified".as_bytes()),
};
let ssz = ssz_encode(&original); let bytes = ssz_encode(&original);
let (decoded, _) = <_>::ssz_decode(&bytes, 0).unwrap();
let (decoded, _) = AttestationData::ssz_decode(&ssz, 0).unwrap();
assert_eq!(original, decoded); assert_eq!(original, decoded);
} }

View File

@ -1,6 +1,8 @@
use super::ssz::{Decodable, DecodeError, Encodable, SszStream}; use super::ssz::{Decodable, DecodeError, Encodable, SszStream};
use super::{BeaconBlockBody, Hash256}; use super::{BeaconBlockBody, Hash256};
use crate::random::TestRandom;
use bls::AggregateSignature; use bls::AggregateSignature;
use rand::RngCore;
#[derive(Debug, PartialEq, Clone, Default)] #[derive(Debug, PartialEq, Clone, Default)]
pub struct BeaconBlock { pub struct BeaconBlock {
@ -49,3 +51,36 @@ impl Decodable for BeaconBlock {
)) ))
} }
} }
impl<T: RngCore> TestRandom<T> for BeaconBlock {
fn random_for_test(rng: &mut T) -> Self {
Self {
slot: <_>::random_for_test(rng),
parent_root: <_>::random_for_test(rng),
state_root: <_>::random_for_test(rng),
randao_reveal: <_>::random_for_test(rng),
candidate_pow_receipt_root: <_>::random_for_test(rng),
signature: <_>::random_for_test(rng),
body: <_>::random_for_test(rng),
}
}
}
#[cfg(test)]
mod tests {
use super::super::ssz::ssz_encode;
use super::*;
use crate::random::TestRandom;
use rand::{prng::XorShiftRng, SeedableRng};
#[test]
pub fn test_ssz_round_trip() {
let mut rng = XorShiftRng::from_seed([42; 16]);
let original = BeaconBlock::random_for_test(&mut rng);
let bytes = ssz_encode(&original);
let (decoded, _) = <_>::ssz_decode(&bytes, 0).unwrap();
assert_eq!(original, decoded);
}
}

View File

@ -1,5 +1,7 @@
use super::ssz::{decode_ssz_list, Decodable, DecodeError, Encodable, SszStream}; use super::ssz::{Decodable, DecodeError, Encodable, SszStream};
use super::{Attestation, CasperSlashing, Deposit, Exit, ProposerSlashing}; use super::{Attestation, CasperSlashing, Deposit, Exit, ProposerSlashing};
use crate::random::TestRandom;
use rand::RngCore;
#[derive(Debug, PartialEq, Clone, Default)] #[derive(Debug, PartialEq, Clone, Default)]
pub struct BeaconBlockBody { pub struct BeaconBlockBody {
@ -22,11 +24,11 @@ impl Encodable for BeaconBlockBody {
impl Decodable for BeaconBlockBody { impl Decodable for BeaconBlockBody {
fn ssz_decode(bytes: &[u8], i: usize) -> Result<(Self, usize), DecodeError> { fn ssz_decode(bytes: &[u8], i: usize) -> Result<(Self, usize), DecodeError> {
let (proposer_slashings, i) = decode_ssz_list(bytes, i)?; let (proposer_slashings, i) = <_>::ssz_decode(bytes, i)?;
let (casper_slashings, i) = decode_ssz_list(bytes, i)?; let (casper_slashings, i) = <_>::ssz_decode(bytes, i)?;
let (attestations, i) = decode_ssz_list(bytes, i)?; let (attestations, i) = <_>::ssz_decode(bytes, i)?;
let (deposits, i) = decode_ssz_list(bytes, i)?; let (deposits, i) = <_>::ssz_decode(bytes, i)?;
let (exits, i) = decode_ssz_list(bytes, i)?; let (exits, i) = <_>::ssz_decode(bytes, i)?;
Ok(( Ok((
Self { Self {
@ -40,3 +42,34 @@ impl Decodable for BeaconBlockBody {
)) ))
} }
} }
impl<T: RngCore> TestRandom<T> for BeaconBlockBody {
fn random_for_test(rng: &mut T) -> Self {
Self {
proposer_slashings: <_>::random_for_test(rng),
casper_slashings: <_>::random_for_test(rng),
attestations: <_>::random_for_test(rng),
deposits: <_>::random_for_test(rng),
exits: <_>::random_for_test(rng),
}
}
}
#[cfg(test)]
mod tests {
use super::super::ssz::ssz_encode;
use super::*;
use crate::random::TestRandom;
use rand::{prng::XorShiftRng, SeedableRng};
#[test]
pub fn test_ssz_round_trip() {
let mut rng = XorShiftRng::from_seed([42; 16]);
let original = BeaconBlockBody::random_for_test(&mut rng);
let bytes = ssz_encode(&original);
let (decoded, _) = <_>::ssz_decode(&bytes, 0).unwrap();
assert_eq!(original, decoded);
}
}

View File

@ -1,5 +1,7 @@
use super::ssz::{Decodable, DecodeError, Encodable, SszStream}; use super::ssz::{Decodable, DecodeError, Encodable, SszStream};
use super::SlashableVoteData; use super::SlashableVoteData;
use crate::random::TestRandom;
use rand::RngCore;
#[derive(Debug, PartialEq, Clone, Default)] #[derive(Debug, PartialEq, Clone, Default)]
pub struct CasperSlashing { pub struct CasperSlashing {
@ -10,7 +12,7 @@ pub struct CasperSlashing {
impl Encodable for CasperSlashing { impl Encodable for CasperSlashing {
fn ssz_append(&self, s: &mut SszStream) { fn ssz_append(&self, s: &mut SszStream) {
s.append(&self.slashable_vote_data_1); s.append(&self.slashable_vote_data_1);
s.append(&self.slashable_vote_data_1); s.append(&self.slashable_vote_data_2);
} }
} }
@ -28,3 +30,31 @@ impl Decodable for CasperSlashing {
)) ))
} }
} }
impl<T: RngCore> TestRandom<T> for CasperSlashing {
fn random_for_test(rng: &mut T) -> Self {
Self {
slashable_vote_data_1: <_>::random_for_test(rng),
slashable_vote_data_2: <_>::random_for_test(rng),
}
}
}
#[cfg(test)]
mod tests {
use super::super::ssz::ssz_encode;
use super::*;
use crate::random::TestRandom;
use rand::{prng::XorShiftRng, SeedableRng};
#[test]
pub fn test_ssz_round_trip() {
let mut rng = XorShiftRng::from_seed([42; 16]);
let original = CasperSlashing::random_for_test(&mut rng);
let bytes = ssz_encode(&original);
let (decoded, _) = <_>::ssz_decode(&bytes, 0).unwrap();
assert_eq!(original, decoded);
}
}

View File

@ -1,5 +1,7 @@
use super::ssz::{decode_ssz_list, Decodable, DecodeError, Encodable, SszStream}; use super::ssz::{Decodable, DecodeError, Encodable, SszStream};
use super::{DepositData, Hash256}; use super::{DepositData, Hash256};
use crate::random::TestRandom;
use rand::RngCore;
#[derive(Debug, PartialEq, Clone)] #[derive(Debug, PartialEq, Clone)]
pub struct Deposit { pub struct Deposit {
@ -18,7 +20,7 @@ impl Encodable for Deposit {
impl Decodable for Deposit { impl Decodable for Deposit {
fn ssz_decode(bytes: &[u8], i: usize) -> Result<(Self, usize), DecodeError> { fn ssz_decode(bytes: &[u8], i: usize) -> Result<(Self, usize), DecodeError> {
let (merkle_branch, i) = decode_ssz_list(bytes, i)?; let (merkle_branch, i) = <_>::ssz_decode(bytes, i)?;
let (merkle_tree_index, i) = <_>::ssz_decode(bytes, i)?; let (merkle_tree_index, i) = <_>::ssz_decode(bytes, i)?;
let (deposit_data, i) = <_>::ssz_decode(bytes, i)?; let (deposit_data, i) = <_>::ssz_decode(bytes, i)?;
@ -33,37 +35,30 @@ impl Decodable for Deposit {
} }
} }
impl<T: RngCore> TestRandom<T> for Deposit {
fn random_for_test(rng: &mut T) -> Self {
Self {
merkle_branch: <_>::random_for_test(rng),
merkle_tree_index: <_>::random_for_test(rng),
deposit_data: <_>::random_for_test(rng),
}
}
}
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::super::ssz::ssz_encode; use super::super::ssz::ssz_encode;
use super::super::{DepositInput, Hash256};
use super::*; use super::*;
use bls::{Keypair, Signature}; use crate::random::TestRandom;
use rand::{prng::XorShiftRng, SeedableRng};
#[test] #[test]
pub fn test_ssz_round_trip() { pub fn test_ssz_round_trip() {
let keypair = Keypair::random(); let mut rng = XorShiftRng::from_seed([42; 16]);
let original = Deposit::random_for_test(&mut rng);
let original = Deposit {
merkle_branch: vec![
Hash256::from("one".as_bytes()),
Hash256::from("two".as_bytes()),
],
merkle_tree_index: 19,
deposit_data: DepositData {
deposit_input: DepositInput {
pubkey: keypair.pk,
withdrawal_credentials: Hash256::from("cats".as_bytes()),
randao_commitment: Hash256::from("dogs".as_bytes()),
proof_of_possession: Signature::new(&[42, 42], &keypair.sk),
},
value: 12,
timestamp: 100,
},
};
let bytes = ssz_encode(&original); let bytes = ssz_encode(&original);
let (decoded, _) = Deposit::ssz_decode(&bytes, 0).unwrap(); let (decoded, _) = <_>::ssz_decode(&bytes, 0).unwrap();
assert_eq!(original, decoded); assert_eq!(original, decoded);
} }

View File

@ -1,5 +1,7 @@
use super::ssz::{Decodable, DecodeError, Encodable, SszStream}; use super::ssz::{Decodable, DecodeError, Encodable, SszStream};
use super::DepositInput; use super::DepositInput;
use crate::random::TestRandom;
use rand::RngCore;
#[derive(Debug, PartialEq, Clone)] #[derive(Debug, PartialEq, Clone)]
pub struct DepositData { pub struct DepositData {
@ -33,30 +35,30 @@ impl Decodable for DepositData {
} }
} }
impl<T: RngCore> TestRandom<T> for DepositData {
fn random_for_test(rng: &mut T) -> Self {
Self {
deposit_input: <_>::random_for_test(rng),
value: <_>::random_for_test(rng),
timestamp: <_>::random_for_test(rng),
}
}
}
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::super::ssz::ssz_encode; use super::super::ssz::ssz_encode;
use super::super::Hash256;
use super::*; use super::*;
use bls::{Keypair, Signature}; use crate::random::TestRandom;
use rand::{prng::XorShiftRng, SeedableRng};
#[test] #[test]
pub fn test_ssz_round_trip() { pub fn test_ssz_round_trip() {
let keypair = Keypair::random(); let mut rng = XorShiftRng::from_seed([42; 16]);
let original = DepositData::random_for_test(&mut rng);
let original = DepositData {
deposit_input: DepositInput {
pubkey: keypair.pk,
withdrawal_credentials: Hash256::from("cats".as_bytes()),
randao_commitment: Hash256::from("dogs".as_bytes()),
proof_of_possession: Signature::new(&[42, 42], &keypair.sk),
},
value: 12,
timestamp: 100,
};
let bytes = ssz_encode(&original); let bytes = ssz_encode(&original);
let (decoded, _) = DepositData::ssz_decode(&bytes, 0).unwrap(); let (decoded, _) = <_>::ssz_decode(&bytes, 0).unwrap();
assert_eq!(original, decoded); assert_eq!(original, decoded);
} }

View File

@ -1,6 +1,8 @@
use super::ssz::{decode_ssz_list, Decodable, DecodeError, Encodable, SszStream}; use super::ssz::{decode_ssz_list, Decodable, DecodeError, Encodable, SszStream};
use super::Hash256; use super::Hash256;
use crate::random::TestRandom;
use bls::{PublicKey, Signature}; use bls::{PublicKey, Signature};
use rand::RngCore;
#[derive(Debug, PartialEq, Clone)] #[derive(Debug, PartialEq, Clone)]
pub struct DepositInput { pub struct DepositInput {
@ -39,25 +41,31 @@ impl Decodable for DepositInput {
} }
} }
impl<T: RngCore> TestRandom<T> for DepositInput {
fn random_for_test(rng: &mut T) -> Self {
Self {
pubkey: <_>::random_for_test(rng),
withdrawal_credentials: <_>::random_for_test(rng),
randao_commitment: <_>::random_for_test(rng),
proof_of_possession: <_>::random_for_test(rng),
}
}
}
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::super::ssz::ssz_encode; use super::super::ssz::ssz_encode;
use super::*; use super::*;
use bls::{Keypair, Signature}; use crate::random::TestRandom;
use rand::{prng::XorShiftRng, SeedableRng};
#[test] #[test]
pub fn test_ssz_round_trip() { pub fn test_ssz_round_trip() {
let keypair = Keypair::random(); let mut rng = XorShiftRng::from_seed([42; 16]);
let original = DepositInput::random_for_test(&mut rng);
let original = DepositInput {
pubkey: keypair.pk,
withdrawal_credentials: Hash256::from("cats".as_bytes()),
randao_commitment: Hash256::from("dogs".as_bytes()),
proof_of_possession: Signature::new(&[42, 42], &keypair.sk),
};
let bytes = ssz_encode(&original); let bytes = ssz_encode(&original);
let (decoded, _) = DepositInput::ssz_decode(&bytes, 0).unwrap(); let (decoded, _) = <_>::ssz_decode(&bytes, 0).unwrap();
assert_eq!(original, decoded); assert_eq!(original, decoded);
} }

View File

@ -1,11 +1,13 @@
use super::ssz::{Decodable, DecodeError, Encodable, SszStream}; use super::ssz::{Decodable, DecodeError, Encodable, SszStream};
use bls::AggregateSignature; use crate::random::TestRandom;
use bls::Signature;
use rand::RngCore;
#[derive(Debug, PartialEq, Clone)] #[derive(Debug, PartialEq, Clone)]
pub struct Exit { pub struct Exit {
pub slot: u64, pub slot: u64,
pub validator_index: u32, pub validator_index: u32,
pub signature: AggregateSignature, pub signature: Signature,
} }
impl Encodable for Exit { impl Encodable for Exit {
@ -33,27 +35,30 @@ impl Decodable for Exit {
} }
} }
impl<T: RngCore> TestRandom<T> for Exit {
fn random_for_test(rng: &mut T) -> Self {
Self {
slot: <_>::random_for_test(rng),
validator_index: <_>::random_for_test(rng),
signature: <_>::random_for_test(rng),
}
}
}
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::super::ssz::ssz_encode; use super::super::ssz::ssz_encode;
use super::*; use super::*;
use bls::{AggregateSignature, Keypair, Signature}; use crate::random::TestRandom;
use rand::{prng::XorShiftRng, SeedableRng};
#[test] #[test]
pub fn test_ssz_round_trip() { pub fn test_ssz_round_trip() {
let keypair = Keypair::random(); let mut rng = XorShiftRng::from_seed([42; 16]);
let single_signature = Signature::new(&[42, 42], &keypair.sk); let original = Exit::random_for_test(&mut rng);
let mut signature = AggregateSignature::new();
signature.add(&single_signature);
let original = Exit {
slot: 42,
validator_index: 12,
signature,
};
let bytes = ssz_encode(&original); let bytes = ssz_encode(&original);
let (decoded, _) = Exit::ssz_decode(&bytes, 0).unwrap(); let (decoded, _) = <_>::ssz_decode(&bytes, 0).unwrap();
assert_eq!(original, decoded); assert_eq!(original, decoded);
} }

View File

@ -3,6 +3,8 @@ extern crate boolean_bitfield;
extern crate ethereum_types; extern crate ethereum_types;
extern crate ssz; extern crate ssz;
mod random;
pub mod active_state; pub mod active_state;
pub mod attestation_data; pub mod attestation_data;
pub mod attestation; pub mod attestation;
@ -32,29 +34,29 @@ pub mod validator_registration;
use self::ethereum_types::{H160, H256, U256}; use self::ethereum_types::{H160, H256, U256};
use std::collections::HashMap; use std::collections::HashMap;
pub use active_state::ActiveState; pub use crate::active_state::ActiveState;
pub use attestation_data::AttestationData; pub use crate::attestation_data::AttestationData;
pub use attestation::Attestation; pub use crate::attestation::Attestation;
pub use beacon_block::BeaconBlock; pub use crate::beacon_block::BeaconBlock;
pub use beacon_block_body::BeaconBlockBody; pub use crate::beacon_block_body::BeaconBlockBody;
pub use beacon_state::BeaconState; pub use crate::beacon_state::BeaconState;
pub use casper_slashing::CasperSlashing; pub use crate::casper_slashing::CasperSlashing;
pub use chain_config::ChainConfig; pub use crate::chain_config::ChainConfig;
pub use crosslink_record::CrosslinkRecord; pub use crate::crosslink_record::CrosslinkRecord;
pub use crystallized_state::CrystallizedState; pub use crate::crystallized_state::CrystallizedState;
pub use deposit::Deposit; pub use crate::deposit::Deposit;
pub use deposit_data::DepositData; pub use crate::deposit_data::DepositData;
pub use deposit_input::DepositInput; pub use crate::deposit_input::DepositInput;
pub use exit::Exit; pub use crate::exit::Exit;
pub use fork_data::ForkData; pub use crate::fork_data::ForkData;
pub use pending_attestation_record::PendingAttestationRecord; pub use crate::pending_attestation_record::PendingAttestationRecord;
pub use proposal_signed_data::ProposalSignedData; pub use crate::proposal_signed_data::ProposalSignedData;
pub use proposer_slashing::ProposerSlashing; pub use crate::proposer_slashing::ProposerSlashing;
pub use slashable_vote_data::SlashableVoteData; pub use crate::slashable_vote_data::SlashableVoteData;
pub use shard_and_committee::ShardAndCommittee; pub use crate::shard_and_committee::ShardAndCommittee;
pub use special_record::{SpecialRecord, SpecialRecordKind}; pub use crate::special_record::{SpecialRecord, SpecialRecordKind};
pub use validator_record::{ValidatorRecord, ValidatorStatus}; pub use crate::validator_record::{ValidatorRecord, ValidatorStatus};
pub use validator_registration::ValidatorRegistration; pub use crate::validator_registration::ValidatorRegistration;
pub type Hash256 = H256; pub type Hash256 = H256;
pub type Address = H160; pub type Address = H160;

View File

@ -1,5 +1,7 @@
use super::ssz::{Decodable, DecodeError, Encodable, SszStream}; use super::ssz::{Decodable, DecodeError, Encodable, SszStream};
use super::Hash256; use super::Hash256;
use crate::random::TestRandom;
use rand::RngCore;
#[derive(Debug, PartialEq, Clone, Default)] #[derive(Debug, PartialEq, Clone, Default)]
pub struct ProposalSignedData { pub struct ProposalSignedData {
@ -33,21 +35,30 @@ impl Decodable for ProposalSignedData {
} }
} }
impl<T: RngCore> TestRandom<T> for ProposalSignedData {
fn random_for_test(rng: &mut T) -> Self {
Self {
slot: <_>::random_for_test(rng),
shard: <_>::random_for_test(rng),
block_root: <_>::random_for_test(rng),
}
}
}
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::super::ssz::ssz_encode; use super::super::ssz::ssz_encode;
use super::*; use super::*;
use crate::random::TestRandom;
use rand::{prng::XorShiftRng, SeedableRng};
#[test] #[test]
pub fn test_ssz_round_trip() { pub fn test_ssz_round_trip() {
let original = ProposalSignedData { let mut rng = XorShiftRng::from_seed([42; 16]);
slot: 42, let original = ProposalSignedData::random_for_test(&mut rng);
shard: 120,
block_root: Hash256::from("cats".as_bytes()),
};
let bytes = ssz_encode(&original); let bytes = ssz_encode(&original);
let (decoded, _) = ProposalSignedData::ssz_decode(&bytes, 0).unwrap(); let (decoded, _) = <_>::ssz_decode(&bytes, 0).unwrap();
assert_eq!(original, decoded); assert_eq!(original, decoded);
} }

View File

@ -1,6 +1,8 @@
use super::ssz::{Decodable, DecodeError, Encodable, SszStream}; use super::ssz::{Decodable, DecodeError, Encodable, SszStream};
use super::ProposalSignedData; use super::ProposalSignedData;
use crate::random::TestRandom;
use bls::Signature; use bls::Signature;
use rand::RngCore;
#[derive(Debug, PartialEq, Clone, Default)] #[derive(Debug, PartialEq, Clone, Default)]
pub struct ProposerSlashing { pub struct ProposerSlashing {
@ -42,35 +44,32 @@ impl Decodable for ProposerSlashing {
} }
} }
impl<T: RngCore> TestRandom<T> for ProposerSlashing {
fn random_for_test(rng: &mut T) -> Self {
Self {
proposer_index: <_>::random_for_test(rng),
proposal_data_1: <_>::random_for_test(rng),
proposal_signature_1: <_>::random_for_test(rng),
proposal_data_2: <_>::random_for_test(rng),
proposal_signature_2: <_>::random_for_test(rng),
}
}
}
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::super::ssz::ssz_encode; use super::super::ssz::ssz_encode;
use super::super::Hash256;
use super::*; use super::*;
use bls::{Keypair, Signature}; use crate::random::TestRandom;
use rand::{prng::XorShiftRng, SeedableRng};
#[test] #[test]
pub fn test_ssz_round_trip() { pub fn test_ssz_round_trip() {
let keypair = Keypair::random(); let mut rng = XorShiftRng::from_seed([42; 16]);
let original = ProposerSlashing::random_for_test(&mut rng);
let original = ProposerSlashing {
proposer_index: 42,
proposal_data_1: ProposalSignedData {
slot: 45,
shard: 110,
block_root: Hash256::from("cats".as_bytes()),
},
proposal_signature_1: Signature::new(&[42, 42], &keypair.sk),
proposal_data_2: ProposalSignedData {
slot: 1,
shard: 260,
block_root: Hash256::from("lol".as_bytes()),
},
proposal_signature_2: Signature::new(&[7, 8], &keypair.sk),
};
let bytes = ssz_encode(&original); let bytes = ssz_encode(&original);
let (decoded, _) = ProposerSlashing::ssz_decode(&bytes, 0).unwrap(); let (decoded, _) = <_>::ssz_decode(&bytes, 0).unwrap();
assert_eq!(original, decoded); assert_eq!(original, decoded);
} }

View File

@ -0,0 +1,12 @@
use super::TestRandom;
use bls::{AggregateSignature, Signature};
use rand::RngCore;
impl<T: RngCore> TestRandom<T> for AggregateSignature {
fn random_for_test(rng: &mut T) -> Self {
let signature = Signature::random_for_test(rng);
let mut aggregate_signature = AggregateSignature::new();
aggregate_signature.add(&signature);
aggregate_signature
}
}

View File

@ -0,0 +1,11 @@
use super::super::Bitfield;
use super::TestRandom;
use rand::RngCore;
impl<T: RngCore> TestRandom<T> for Bitfield {
fn random_for_test(rng: &mut T) -> Self {
let mut raw_bytes = vec![0; 32];
rng.fill_bytes(&mut raw_bytes);
Bitfield::from_bytes(&raw_bytes)
}
}

View File

@ -0,0 +1,11 @@
use super::TestRandom;
use crate::Hash256;
use rand::RngCore;
impl<T: RngCore> TestRandom<T> for Hash256 {
fn random_for_test(rng: &mut T) -> Self {
let mut key_bytes = vec![0; 32];
rng.fill_bytes(&mut key_bytes);
Hash256::from(&key_bytes[..])
}
}

View File

@ -0,0 +1,38 @@
use rand::RngCore;
pub mod aggregate_signature;
pub mod bitfield;
pub mod hash256;
pub mod signature;
pub mod secret_key;
pub mod public_key;
pub trait TestRandom<T>
where T: RngCore
{
fn random_for_test(rng: &mut T) -> Self;
}
impl<T: RngCore> TestRandom<T> for u64 {
fn random_for_test(rng: &mut T) -> Self {
rng.next_u64()
}
}
impl<T: RngCore> TestRandom<T> for u32 {
fn random_for_test(rng: &mut T) -> Self {
rng.next_u32()
}
}
impl<T: RngCore, U> TestRandom<T> for Vec<U>
where U: TestRandom<T>
{
fn random_for_test(rng: &mut T) -> Self {
vec![
<U>::random_for_test(rng),
<U>::random_for_test(rng),
<U>::random_for_test(rng),
]
}
}

View File

@ -0,0 +1,10 @@
use super::TestRandom;
use bls::{PublicKey, SecretKey};
use rand::RngCore;
impl<T: RngCore> TestRandom<T> for PublicKey {
fn random_for_test(rng: &mut T) -> Self {
let secret_key = SecretKey::random_for_test(rng);
PublicKey::from_secret_key(&secret_key)
}
}

View File

@ -0,0 +1,19 @@
use super::TestRandom;
use bls::SecretKey;
use rand::RngCore;
impl<T: RngCore> TestRandom<T> for SecretKey {
fn random_for_test(rng: &mut T) -> Self {
let mut key_bytes = vec![0; 48];
rng.fill_bytes(&mut key_bytes);
/*
* An `unreachable!` is used here as there's no reason why you cannot constuct a key from a
* fixed-length byte slice. Also, this should only be used during testing so a panic is
* acceptable.
*/
match SecretKey::from_bytes(&key_bytes) {
Ok(key) => key,
Err(_) => unreachable!(),
}
}
}

View File

@ -0,0 +1,13 @@
use super::TestRandom;
use bls::{SecretKey, Signature};
use rand::RngCore;
impl<T: RngCore> TestRandom<T> for Signature {
fn random_for_test(rng: &mut T) -> Self {
let secret_key = SecretKey::random_for_test(rng);
let mut message = vec![0; 32];
rng.fill_bytes(&mut message);
Signature::new(&message, &secret_key)
}
}

View File

@ -1,6 +1,8 @@
use super::ssz::{decode_ssz_list, Decodable, DecodeError, Encodable, SszStream}; use super::ssz::{Decodable, DecodeError, Encodable, SszStream};
use super::AttestationData; use super::AttestationData;
use crate::random::TestRandom;
use bls::AggregateSignature; use bls::AggregateSignature;
use rand::RngCore;
#[derive(Debug, PartialEq, Clone, Default)] #[derive(Debug, PartialEq, Clone, Default)]
pub struct SlashableVoteData { pub struct SlashableVoteData {
@ -21,8 +23,8 @@ impl Encodable for SlashableVoteData {
impl Decodable for SlashableVoteData { impl Decodable for SlashableVoteData {
fn ssz_decode(bytes: &[u8], i: usize) -> Result<(Self, usize), DecodeError> { fn ssz_decode(bytes: &[u8], i: usize) -> Result<(Self, usize), DecodeError> {
let (aggregate_signature_poc_0_indices, i) = decode_ssz_list(bytes, i)?; let (aggregate_signature_poc_0_indices, i) = <_>::ssz_decode(bytes, i)?;
let (aggregate_signature_poc_1_indices, i) = decode_ssz_list(bytes, i)?; let (aggregate_signature_poc_1_indices, i) = <_>::ssz_decode(bytes, i)?;
let (data, i) = <_>::ssz_decode(bytes, i)?; let (data, i) = <_>::ssz_decode(bytes, i)?;
let (aggregate_signature, i) = <_>::ssz_decode(bytes, i)?; let (aggregate_signature, i) = <_>::ssz_decode(bytes, i)?;
@ -38,32 +40,31 @@ impl Decodable for SlashableVoteData {
} }
} }
impl<T: RngCore> TestRandom<T> for SlashableVoteData {
fn random_for_test(rng: &mut T) -> Self {
Self {
aggregate_signature_poc_0_indices: <_>::random_for_test(rng),
aggregate_signature_poc_1_indices: <_>::random_for_test(rng),
data: <_>::random_for_test(rng),
aggregate_signature: <_>::random_for_test(rng),
}
}
}
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::super::ssz::ssz_encode; use super::super::ssz::ssz_encode;
use super::super::Hash256;
use super::*; use super::*;
use crate::random::TestRandom;
use rand::{prng::XorShiftRng, SeedableRng};
#[test] #[test]
pub fn test_ssz_round_trip() { pub fn test_ssz_round_trip() {
let original = SlashableVoteData { let mut rng = XorShiftRng::from_seed([42; 16]);
aggregate_signature_poc_0_indices: vec![0, 1, 2], let original = SlashableVoteData::random_for_test(&mut rng);
aggregate_signature_poc_1_indices: vec![42, 42, 42],
data: AttestationData {
slot: 42,
shard: 16,
beacon_block_hash: Hash256::from("beacon".as_bytes()),
epoch_boundary_hash: Hash256::from("epoch".as_bytes()),
shard_block_hash: Hash256::from("shard".as_bytes()),
latest_crosslink_hash: Hash256::from("xlink".as_bytes()),
justified_slot: 8,
justified_block_hash: Hash256::from("justified".as_bytes()),
},
aggregate_signature: AggregateSignature::new(),
};
let bytes = ssz_encode(&original); let bytes = ssz_encode(&original);
let (decoded, _) = SlashableVoteData::ssz_decode(&bytes, 0).unwrap(); let (decoded, _) = <_>::ssz_decode(&bytes, 0).unwrap();
assert_eq!(original, decoded); assert_eq!(original, decoded);
} }