use super::{Call, *};
use frame_support::{
assert_err, assert_noop, assert_ok,
dispatch::{GetDispatchInfo, Pays},
traits::{Currency, EstimateNextSessionRotation, KeyOwnerProofSystem, OnFinalize},
};
use mock::*;
use pallet_session::ShouldEndSession;
use sp_consensus_babe::{
AllowedSlots, BabeEpochConfiguration, Slot, VrfSignature, RANDOMNESS_LENGTH,
};
use sp_core::crypto::Pair;
const EMPTY_RANDOMNESS: [u8; RANDOMNESS_LENGTH] = [
74, 25, 49, 128, 53, 97, 244, 49, 222, 202, 176, 2, 231, 66, 95, 10, 133, 49, 213, 228, 86,
161, 164, 127, 217, 153, 138, 37, 48, 192, 248, 0,
];
impl crate::migrations::BabePalletPrefix for Test {
fn pallet_prefix() -> &'static str {
"Babe"
}
}
#[test]
fn empty_randomness_is_correct() {
let s = compute_randomness([0; RANDOMNESS_LENGTH], 0, std::iter::empty(), None);
assert_eq!(s, EMPTY_RANDOMNESS);
}
#[test]
fn initial_values() {
build_and_execute(4, || {
assert_eq!(Authorities::<Test>::get().len(), 4);
})
}
#[test]
fn check_module() {
build_and_execute(4, || {
assert!(!Babe::should_end_session(0), "Genesis does not change sessions");
assert!(
!Babe::should_end_session(200000),
"BABE does not include the block number in epoch calculations"
);
})
}
#[test]
fn first_block_epoch_zero_start() {
build_and_execute_with_pairs(4, |pairs| {
let genesis_slot = Slot::from(100);
let (vrf_signature, vrf_randomness) =
make_vrf_signature_and_randomness(genesis_slot, &pairs[0]);
let pre_digest = make_primary_pre_digest(0, genesis_slot, vrf_signature);
assert_eq!(GenesisSlot::<Test>::get(), Slot::from(0));
System::reset_events();
System::initialize(&1, &Default::default(), &pre_digest);
assert!(!Babe::should_end_session(1));
assert_eq!(GenesisSlot::<Test>::get(), genesis_slot);
assert_eq!(CurrentSlot::<Test>::get(), genesis_slot);
assert_eq!(EpochIndex::<Test>::get(), 0);
Babe::on_finalize(1);
let header = System::finalize();
assert_eq!(AuthorVrfRandomness::<Test>::get(), Some(vrf_randomness));
assert_eq!(SegmentIndex::<Test>::get(), 0);
assert_eq!(UnderConstruction::<Test>::get(0), vec![vrf_randomness]);
assert_eq!(Randomness::<Test>::get(), [0; 32]);
assert_eq!(AuthorVrfRandomness::<Test>::get(), Some(vrf_randomness));
assert_eq!(NextRandomness::<Test>::get(), [0; 32]);
assert_eq!(header.digest.logs.len(), 2);
assert_eq!(pre_digest.logs.len(), 1);
assert_eq!(header.digest.logs[0], pre_digest.logs[0]);
let consensus_log = sp_consensus_babe::ConsensusLog::NextEpochData(
sp_consensus_babe::digests::NextEpochDescriptor {
authorities: Authorities::<Test>::get().into_inner(),
randomness: Randomness::<Test>::get(),
},
);
let consensus_digest = DigestItem::Consensus(BABE_ENGINE_ID, consensus_log.encode());
assert_eq!(header.digest.logs[1], consensus_digest.clone());
})
}
#[test]
fn current_slot_is_processed_on_initialization() {
build_and_execute_with_pairs(1, |pairs| {
let genesis_slot = Slot::from(10);
let (vrf_signature, vrf_randomness) =
make_vrf_signature_and_randomness(genesis_slot, &pairs[0]);
let pre_digest = make_primary_pre_digest(0, genesis_slot, vrf_signature);
System::reset_events();
System::initialize(&1, &Default::default(), &pre_digest);
assert_eq!(CurrentSlot::<Test>::get(), Slot::from(0));
assert!(Initialized::<Test>::get().is_none());
Babe::initialize(1);
assert_eq!(CurrentSlot::<Test>::get(), genesis_slot);
assert!(Initialized::<Test>::get().is_some());
assert_eq!(AuthorVrfRandomness::<Test>::get(), None);
Babe::on_finalize(1);
assert_eq!(AuthorVrfRandomness::<Test>::get(), Some(vrf_randomness));
})
}
fn test_author_vrf_output<F>(make_pre_digest: F)
where
F: Fn(sp_consensus_babe::AuthorityIndex, Slot, VrfSignature) -> sp_runtime::Digest,
{
build_and_execute_with_pairs(1, |pairs| {
let genesis_slot = Slot::from(10);
let (vrf_signature, vrf_randomness) =
make_vrf_signature_and_randomness(genesis_slot, &pairs[0]);
let pre_digest = make_pre_digest(0, genesis_slot, vrf_signature);
System::reset_events();
System::initialize(&1, &Default::default(), &pre_digest);
Babe::initialize(1);
assert_eq!(AuthorVrfRandomness::<Test>::get(), None);
Babe::on_finalize(1);
assert_eq!(AuthorVrfRandomness::<Test>::get(), Some(vrf_randomness));
System::finalize();
assert_eq!(AuthorVrfRandomness::<Test>::get(), Some(vrf_randomness));
})
}
#[test]
fn author_vrf_output_for_primary() {
test_author_vrf_output(make_primary_pre_digest);
}
#[test]
fn author_vrf_output_for_secondary_vrf() {
test_author_vrf_output(make_secondary_vrf_pre_digest);
}
#[test]
fn no_author_vrf_output_for_secondary_plain() {
build_and_execute(1, || {
let genesis_slot = Slot::from(10);
let secondary_plain_pre_digest = make_secondary_plain_pre_digest(0, genesis_slot);
System::reset_events();
System::initialize(&1, &Default::default(), &secondary_plain_pre_digest);
assert_eq!(AuthorVrfRandomness::<Test>::get(), None);
Babe::initialize(1);
assert_eq!(AuthorVrfRandomness::<Test>::get(), None);
Babe::on_finalize(1);
System::finalize();
assert_eq!(AuthorVrfRandomness::<Test>::get(), None);
})
}
#[test]
fn authority_index() {
build_and_execute(4, || {
assert_eq!(
Babe::find_author((&[(BABE_ENGINE_ID, &[][..])]).into_iter().cloned()),
None,
"Trivially invalid authorities are ignored"
)
})
}
#[test]
fn can_predict_next_epoch_change() {
build_and_execute(1, || {
assert_eq!(<Test as Config>::EpochDuration::get(), 3);
go_to_block(1, 6);
assert_eq!(*GenesisSlot::<Test>::get(), 6);
assert_eq!(*CurrentSlot::<Test>::get(), 6);
assert_eq!(EpochIndex::<Test>::get(), 0);
progress_to_block(5);
assert_eq!(EpochIndex::<Test>::get(), 5 / 3);
assert_eq!(*CurrentSlot::<Test>::get(), 10);
assert_eq!(*Babe::current_epoch_start(), 9); assert_eq!(Babe::next_expected_epoch_change(System::block_number()), Some(5 + 2));
})
}
#[test]
fn can_estimate_current_epoch_progress() {
build_and_execute(1, || {
assert_eq!(<Test as Config>::EpochDuration::get(), 3);
for i in 1u64..4 {
progress_to_block(i);
assert_eq!(Babe::estimate_next_session_rotation(i).0.unwrap(), 4);
if Babe::estimate_next_session_rotation(i).0.unwrap() - 1 == i {
assert_eq!(
Babe::estimate_current_session_progress(i).0.unwrap(),
Permill::from_percent(100)
);
} else {
assert!(
Babe::estimate_current_session_progress(i).0.unwrap() <
Permill::from_percent(100)
);
}
}
progress_to_block(4);
assert_eq!(
Babe::estimate_current_session_progress(4).0.unwrap(),
Permill::from_float(1.0 / 3.0),
);
})
}
#[test]
fn can_enact_next_config() {
build_and_execute(1, || {
assert_eq!(<Test as Config>::EpochDuration::get(), 3);
go_to_block(1, 6);
assert_eq!(*GenesisSlot::<Test>::get(), 6);
assert_eq!(*CurrentSlot::<Test>::get(), 6);
assert_eq!(EpochIndex::<Test>::get(), 0);
go_to_block(2, 7);
let current_config = BabeEpochConfiguration {
c: (0, 4),
allowed_slots: sp_consensus_babe::AllowedSlots::PrimarySlots,
};
let next_config = BabeEpochConfiguration {
c: (1, 4),
allowed_slots: sp_consensus_babe::AllowedSlots::PrimarySlots,
};
let next_next_config = BabeEpochConfiguration {
c: (2, 4),
allowed_slots: sp_consensus_babe::AllowedSlots::PrimarySlots,
};
EpochConfig::<Test>::put(current_config);
NextEpochConfig::<Test>::put(next_config.clone());
assert_eq!(NextEpochConfig::<Test>::get(), Some(next_config.clone()));
Babe::plan_config_change(
RuntimeOrigin::root(),
NextConfigDescriptor::V1 {
c: next_next_config.c,
allowed_slots: next_next_config.allowed_slots,
},
)
.unwrap();
progress_to_block(4);
Babe::on_finalize(9);
let header = System::finalize();
assert_eq!(EpochConfig::<Test>::get(), Some(next_config));
assert_eq!(NextEpochConfig::<Test>::get(), Some(next_next_config.clone()));
let consensus_log =
sp_consensus_babe::ConsensusLog::NextConfigData(NextConfigDescriptor::V1 {
c: next_next_config.c,
allowed_slots: next_next_config.allowed_slots,
});
let consensus_digest = DigestItem::Consensus(BABE_ENGINE_ID, consensus_log.encode());
assert_eq!(header.digest.logs[2], consensus_digest.clone())
});
}
#[test]
fn only_root_can_enact_config_change() {
use sp_runtime::DispatchError;
build_and_execute(1, || {
let next_config =
NextConfigDescriptor::V1 { c: (1, 4), allowed_slots: AllowedSlots::PrimarySlots };
let res = Babe::plan_config_change(RuntimeOrigin::none(), next_config.clone());
assert_noop!(res, DispatchError::BadOrigin);
let res = Babe::plan_config_change(RuntimeOrigin::signed(1), next_config.clone());
assert_noop!(res, DispatchError::BadOrigin);
let res = Babe::plan_config_change(RuntimeOrigin::root(), next_config);
assert!(res.is_ok());
});
}
#[test]
fn can_fetch_current_and_next_epoch_data() {
build_and_execute(5, || {
EpochConfig::<Test>::put(BabeEpochConfiguration {
c: (1, 4),
allowed_slots: sp_consensus_babe::AllowedSlots::PrimarySlots,
});
assert_eq!(Babe::current_epoch().authorities, Babe::next_epoch().authorities);
start_era(1);
let current_epoch = Babe::current_epoch();
assert_eq!(current_epoch.epoch_index, 3);
assert_eq!(*current_epoch.start_slot, 10);
assert_eq!(current_epoch.authorities.len(), 5);
let next_epoch = Babe::next_epoch();
assert_eq!(next_epoch.epoch_index, 4);
assert_eq!(*next_epoch.start_slot, 13);
assert_eq!(next_epoch.authorities.len(), 5);
assert!(current_epoch.randomness != next_epoch.randomness);
assert!(current_epoch.authorities == next_epoch.authorities);
});
}
#[test]
fn tracks_block_numbers_when_current_and_previous_epoch_started() {
build_and_execute(5, || {
progress_to_block(8);
let (last_epoch, current_epoch) = EpochStart::<Test>::get();
assert_eq!(last_epoch, 4);
assert_eq!(current_epoch, 7);
progress_to_block(10);
let (last_epoch, current_epoch) = EpochStart::<Test>::get();
assert_eq!(last_epoch, 7);
assert_eq!(current_epoch, 10);
});
}
#[test]
#[should_panic(
expected = "Validator with index 0 is disabled and should not be attempting to author blocks."
)]
fn disabled_validators_cannot_author_blocks() {
build_and_execute(4, || {
start_era(1);
Session::disable_index(1);
start_era(2);
assert_eq!(pallet_staking::CurrentEra::<Test>::get().unwrap(), 2);
Session::disable_index(0);
start_era(3);
});
}
#[test]
fn report_equivocation_current_session_works() {
build_and_execute_with_pairs(3, |pairs| {
start_era(1);
let authorities = Authorities::<Test>::get();
let validators = Session::validators();
for validator in &validators {
assert_eq!(Balances::total_balance(validator), 10_000_000);
assert_eq!(Staking::slashable_balance_of(validator), 10_000);
assert_eq!(
Staking::eras_stakers(1, &validator),
pallet_staking::Exposure { total: 10_000, own: 10_000, others: vec![] },
);
}
let offending_validator_index = 1;
let offending_validator_id = Session::validators()[offending_validator_index];
let offending_authority_pair = pairs
.into_iter()
.find(|p| p.public() == authorities[offending_validator_index].0)
.unwrap();
let equivocation_proof = generate_equivocation_proof(
offending_validator_index as u32,
&offending_authority_pair,
CurrentSlot::<Test>::get(),
);
let key = (sp_consensus_babe::KEY_TYPE, &offending_authority_pair.public());
let key_owner_proof = Historical::prove(key).unwrap();
Babe::report_equivocation_unsigned(
RuntimeOrigin::none(),
Box::new(equivocation_proof),
key_owner_proof,
)
.unwrap();
start_era(2);
assert_eq!(Balances::total_balance(&offending_validator_id), 10_000_000 - 10_000);
assert_eq!(Staking::slashable_balance_of(&offending_validator_id), 0);
assert_eq!(
Staking::eras_stakers(2, &offending_validator_id),
pallet_staking::Exposure { total: 0, own: 0, others: vec![] },
);
for validator in &validators {
if *validator == offending_validator_id {
continue;
}
assert_eq!(Balances::total_balance(validator), 10_000_000);
assert_eq!(Staking::slashable_balance_of(validator), 10_000);
assert_eq!(
Staking::eras_stakers(2, &validator),
pallet_staking::Exposure { total: 10_000, own: 10_000, others: vec![] },
);
}
})
}
#[test]
fn report_equivocation_old_session_works() {
build_and_execute_with_pairs(3, |pairs| {
start_era(1);
let authorities = Authorities::<Test>::get();
let offending_validator_index = 1;
let offending_validator_id = Session::validators()[offending_validator_index];
let offending_authority_pair = pairs
.into_iter()
.find(|p| p.public() == authorities[offending_validator_index].0)
.unwrap();
let equivocation_proof = generate_equivocation_proof(
offending_validator_index as u32,
&offending_authority_pair,
CurrentSlot::<Test>::get(),
);
let key = (sp_consensus_babe::KEY_TYPE, &offending_authority_pair.public());
let key_owner_proof = Historical::prove(key).unwrap();
start_era(2);
assert_eq!(Balances::total_balance(&offending_validator_id), 10_000_000);
assert_eq!(Staking::slashable_balance_of(&offending_validator_id), 10_000);
Babe::report_equivocation_unsigned(
RuntimeOrigin::none(),
Box::new(equivocation_proof),
key_owner_proof,
)
.unwrap();
start_era(3);
assert_eq!(Balances::total_balance(&offending_validator_id), 10_000_000 - 10_000);
assert_eq!(Staking::slashable_balance_of(&offending_validator_id), 0);
assert_eq!(
Staking::eras_stakers(3, &offending_validator_id),
pallet_staking::Exposure { total: 0, own: 0, others: vec![] },
);
})
}
#[test]
fn report_equivocation_invalid_key_owner_proof() {
build_and_execute_with_pairs(3, |pairs| {
start_era(1);
let authorities = Authorities::<Test>::get();
let offending_validator_index = 0;
let offending_authority_pair = pairs
.into_iter()
.find(|p| p.public() == authorities[offending_validator_index].0)
.unwrap();
let equivocation_proof = generate_equivocation_proof(
offending_validator_index as u32,
&offending_authority_pair,
CurrentSlot::<Test>::get(),
);
let key = (sp_consensus_babe::KEY_TYPE, &offending_authority_pair.public());
let mut key_owner_proof = Historical::prove(key).unwrap();
key_owner_proof.session = 0;
assert_err!(
Babe::report_equivocation_unsigned(
RuntimeOrigin::none(),
Box::new(equivocation_proof.clone()),
key_owner_proof
),
Error::<Test>::InvalidKeyOwnershipProof,
);
let key = (sp_consensus_babe::KEY_TYPE, &authorities[1].0);
let key_owner_proof = Historical::prove(key).unwrap();
start_era(2);
assert_err!(
Babe::report_equivocation_unsigned(
RuntimeOrigin::none(),
Box::new(equivocation_proof),
key_owner_proof,
),
Error::<Test>::InvalidKeyOwnershipProof,
);
})
}
#[test]
fn report_equivocation_invalid_equivocation_proof() {
use sp_runtime::traits::Header;
build_and_execute_with_pairs(3, |pairs| {
start_era(1);
let authorities = Authorities::<Test>::get();
let offending_validator_index = 0;
let offending_authority_pair = pairs
.into_iter()
.find(|p| p.public() == authorities[offending_validator_index].0)
.unwrap();
let key = (sp_consensus_babe::KEY_TYPE, &offending_authority_pair.public());
let key_owner_proof = Historical::prove(key).unwrap();
let assert_invalid_equivocation = |equivocation_proof| {
assert_err!(
Babe::report_equivocation_unsigned(
RuntimeOrigin::none(),
Box::new(equivocation_proof),
key_owner_proof.clone(),
),
Error::<Test>::InvalidEquivocationProof,
)
};
let mut equivocation_proof = generate_equivocation_proof(
offending_validator_index as u32,
&offending_authority_pair,
CurrentSlot::<Test>::get(),
);
equivocation_proof.second_header = equivocation_proof.first_header.clone();
assert_invalid_equivocation(equivocation_proof);
let mut equivocation_proof = generate_equivocation_proof(
offending_validator_index as u32,
&offending_authority_pair,
CurrentSlot::<Test>::get(),
);
equivocation_proof.first_header.digest_mut().logs.remove(0);
assert_invalid_equivocation(equivocation_proof);
let mut equivocation_proof = generate_equivocation_proof(
offending_validator_index as u32,
&offending_authority_pair,
CurrentSlot::<Test>::get(),
);
equivocation_proof.first_header.digest_mut().logs.remove(1);
assert_invalid_equivocation(equivocation_proof);
let mut equivocation_proof = generate_equivocation_proof(
offending_validator_index as u32,
&offending_authority_pair,
CurrentSlot::<Test>::get(),
);
equivocation_proof.slot = Slot::from(0);
assert_invalid_equivocation(equivocation_proof.clone());
let h1 = equivocation_proof.first_header;
let mut equivocation_proof = generate_equivocation_proof(
offending_validator_index as u32,
&offending_authority_pair,
CurrentSlot::<Test>::get() + 1,
);
equivocation_proof.first_header = h1.clone();
assert_invalid_equivocation(equivocation_proof.clone());
let mut equivocation_proof = generate_equivocation_proof(
offending_validator_index as u32,
&offending_authority_pair,
CurrentSlot::<Test>::get() + 1,
);
equivocation_proof.first_header.digest_mut().pop();
equivocation_proof
.first_header
.digest_mut()
.push(h1.digest().logs().last().unwrap().clone());
assert_invalid_equivocation(equivocation_proof.clone());
})
}
#[test]
fn report_equivocation_validate_unsigned_prevents_duplicates() {
use sp_runtime::transaction_validity::{
InvalidTransaction, TransactionPriority, TransactionSource, TransactionValidity,
ValidTransaction,
};
build_and_execute_with_pairs(3, |pairs| {
start_era(1);
let authorities = Authorities::<Test>::get();
let offending_validator_index = 0;
let offending_authority_pair = pairs
.into_iter()
.find(|p| p.public() == authorities[offending_validator_index].0)
.unwrap();
let equivocation_proof = generate_equivocation_proof(
offending_validator_index as u32,
&offending_authority_pair,
CurrentSlot::<Test>::get(),
);
let key = (sp_consensus_babe::KEY_TYPE, &offending_authority_pair.public());
let key_owner_proof = Historical::prove(key).unwrap();
let inner = Call::report_equivocation_unsigned {
equivocation_proof: Box::new(equivocation_proof.clone()),
key_owner_proof: key_owner_proof.clone(),
};
assert_eq!(
<Babe as sp_runtime::traits::ValidateUnsigned>::validate_unsigned(
TransactionSource::External,
&inner,
),
InvalidTransaction::Call.into(),
);
let tx_tag = (offending_authority_pair.public(), CurrentSlot::<Test>::get());
assert_eq!(
<Babe as sp_runtime::traits::ValidateUnsigned>::validate_unsigned(
TransactionSource::Local,
&inner,
),
TransactionValidity::Ok(ValidTransaction {
priority: TransactionPriority::max_value(),
requires: vec![],
provides: vec![("BabeEquivocation", tx_tag).encode()],
longevity: ReportLongevity::get(),
propagate: false,
})
);
assert_ok!(<Babe as sp_runtime::traits::ValidateUnsigned>::pre_dispatch(&inner));
Babe::report_equivocation_unsigned(
RuntimeOrigin::none(),
Box::new(equivocation_proof),
key_owner_proof,
)
.unwrap();
assert_err!(
<Babe as sp_runtime::traits::ValidateUnsigned>::validate_unsigned(
TransactionSource::Local,
&inner,
),
InvalidTransaction::Stale,
);
assert_err!(
<Babe as sp_runtime::traits::ValidateUnsigned>::pre_dispatch(&inner),
InvalidTransaction::Stale,
);
});
}
#[test]
fn report_equivocation_has_valid_weight() {
assert!((1..=100)
.map(|validators| <Test as Config>::WeightInfo::report_equivocation(validators, 1000))
.collect::<Vec<_>>()
.windows(2)
.all(|w| w[0] == w[1]));
assert!((100..=1000)
.map(|validators| <Test as Config>::WeightInfo::report_equivocation(validators, 1000))
.collect::<Vec<_>>()
.windows(2)
.all(|w| w[0].ref_time() < w[1].ref_time()));
}
#[test]
fn report_equivocation_after_skipped_epochs_works() {
build_and_execute_with_pairs(3, |pairs| {
let epoch_duration: u64 = <Test as Config>::EpochDuration::get();
let genesis_slot = 100;
go_to_block(1, genesis_slot);
assert_eq!(EpochIndex::<Test>::get(), 0);
go_to_block(System::block_number() + 1, genesis_slot + epoch_duration * 10);
assert_eq!(EpochIndex::<Test>::get(), 10);
assert_eq!(SkippedEpochs::<Test>::get(), vec![(10, 1)]);
let authorities = Authorities::<Test>::get();
let offending_validator_index = 1;
let offending_authority_pair = pairs
.into_iter()
.find(|p| p.public() == authorities[offending_validator_index].0)
.unwrap();
let equivocation_proof = generate_equivocation_proof(
offending_validator_index as u32,
&offending_authority_pair,
CurrentSlot::<Test>::get(),
);
let key = (sp_consensus_babe::KEY_TYPE, &offending_authority_pair.public());
let key_owner_proof = Historical::prove(key).unwrap();
assert_eq!(key_owner_proof.session, 1);
assert_ok!(Babe::report_equivocation_unsigned(
RuntimeOrigin::none(),
Box::new(equivocation_proof),
key_owner_proof
));
})
}
#[test]
fn valid_equivocation_reports_dont_pay_fees() {
build_and_execute_with_pairs(3, |pairs| {
start_era(1);
let offending_authority_pair = &pairs[0];
let equivocation_proof =
generate_equivocation_proof(0, &offending_authority_pair, CurrentSlot::<Test>::get());
let key_owner_proof =
Historical::prove((sp_consensus_babe::KEY_TYPE, &offending_authority_pair.public()))
.unwrap();
let info = Call::<Test>::report_equivocation_unsigned {
equivocation_proof: Box::new(equivocation_proof.clone()),
key_owner_proof: key_owner_proof.clone(),
}
.get_dispatch_info();
assert!(info.call_weight.ref_time() > 0);
assert_eq!(info.pays_fee, Pays::Yes);
let post_info = Babe::report_equivocation_unsigned(
RuntimeOrigin::none(),
Box::new(equivocation_proof.clone()),
key_owner_proof.clone(),
)
.unwrap();
assert!(post_info.actual_weight.is_none());
assert_eq!(post_info.pays_fee, Pays::No);
let post_info = Babe::report_equivocation_unsigned(
RuntimeOrigin::none(),
Box::new(equivocation_proof),
key_owner_proof,
)
.err()
.unwrap()
.post_info;
assert!(post_info.actual_weight.is_none());
assert_eq!(post_info.pays_fee, Pays::Yes);
})
}
#[test]
fn add_epoch_configurations_migration_works() {
use frame_support::storage::migration::{get_storage_value, put_storage_value};
build_and_execute(1, || {
let next_config_descriptor =
NextConfigDescriptor::V1 { c: (3, 4), allowed_slots: AllowedSlots::PrimarySlots };
put_storage_value(b"Babe", b"NextEpochConfig", &[], Some(next_config_descriptor.clone()));
assert!(get_storage_value::<Option<NextConfigDescriptor>>(
b"Babe",
b"NextEpochConfig",
&[],
)
.is_some());
let current_epoch = BabeEpochConfiguration {
c: (1, 4),
allowed_slots: sp_consensus_babe::AllowedSlots::PrimarySlots,
};
crate::migrations::add_epoch_configuration::<Test>(current_epoch.clone());
assert!(get_storage_value::<Option<NextConfigDescriptor>>(
b"Babe",
b"NextEpochConfig",
&[],
)
.is_none());
assert_eq!(EpochConfig::<Test>::get(), Some(current_epoch));
assert_eq!(PendingEpochConfigChange::<Test>::get(), Some(next_config_descriptor));
});
}
#[test]
fn generate_equivocation_report_blob() {
build_and_execute_with_pairs(3, |pairs| {
let offending_authority_index = 0;
let offending_authority_pair = &pairs[0];
start_era(1);
let equivocation_proof = generate_equivocation_proof(
offending_authority_index,
offending_authority_pair,
CurrentSlot::<Test>::get() + 1,
);
println!("equivocation_proof: {:?}", equivocation_proof);
println!("equivocation_proof.encode(): {:?}", equivocation_proof.encode());
});
}
#[test]
fn skipping_over_epochs_works() {
build_and_execute(3, || {
let epoch_duration: u64 = <Test as Config>::EpochDuration::get();
let genesis_slot = 100;
go_to_block(1, genesis_slot);
progress_to_block(epoch_duration + 1);
assert_eq!(EpochIndex::<Test>::get(), 1);
let randomness_for_epoch_2 = NextRandomness::<Test>::get();
assert!(randomness_for_epoch_2 != [0; 32]);
go_to_block(System::block_number() + 1, genesis_slot + epoch_duration * 4);
assert_eq!(EpochIndex::<Test>::get(), 4);
assert_eq!(Randomness::<Test>::get(), randomness_for_epoch_2);
assert_eq!(SkippedEpochs::<Test>::get(), vec![(4, 2)]);
assert_eq!(Babe::session_index_for_epoch(0), 0);
assert_eq!(Babe::session_index_for_epoch(1), 1);
assert_eq!(Babe::session_index_for_epoch(4), 2);
assert_eq!(Babe::session_index_for_epoch(5), 3);
});
}