Skip to content
This repository has been archived by the owner on Nov 15, 2023. It is now read-only.

runtime/disputes: slashing #5535

Merged
merged 95 commits into from
Sep 20, 2022
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
95 commits
Select commit Hold shift + click to select a range
d6a2eb2
disputes: runtime part of slashing
ordian May 6, 2022
fd2194a
disputes: reward winners
ordian May 9, 2022
9f8daef
disputes/slashing: validate_unsigned impl
ordian May 11, 2022
ee46d14
fmt
ordian May 11, 2022
7b158f6
disputes/slashing: report_dispute_lost_unsigned
ordian May 11, 2022
60a82a4
disputes/slashing: separate winners from losers and report winners
ordian May 13, 2022
b303ee4
disputes/slashing: refactoring
ordian May 13, 2022
6f1d724
impl HandleReports
ordian May 16, 2022
4a55be9
enable on Wenstend
ordian May 16, 2022
1941543
fmt
ordian May 16, 2022
04ad120
add slashing pallet to the mock and test runtimes
ordian May 16, 2022
3784591
fix a bug in report_dispute_lost_unsigned
ordian May 16, 2022
f49b55f
fmt
ordian May 16, 2022
cbcf176
disputes: remove new_participants from summary
ordian May 17, 2022
7b82f0d
disputes: remove punish_inconclusive
ordian May 17, 2022
f863baf
impl SlashingHandler for Pallet for type-safety
ordian May 17, 2022
b4fa682
do not impl slashing::Config on mainnets yet
ordian May 17, 2022
2739900
teach spellcheck deduplication
ordian May 17, 2022
62ff14f
simplify interfaces and resolve some TODOs
ordian May 17, 2022
4db3ab8
resolve some more TODOs
ordian May 17, 2022
02c069d
minor typos
ordian May 17, 2022
6fc8f5f
move slashing into a folder
ordian May 18, 2022
7f3ae20
remove unnecessary clone
ordian May 18, 2022
13c7c84
fix validator_set_count calculation
ordian May 18, 2022
5c98289
introduce ValidatorSetCount
ordian May 18, 2022
8263d16
store ValidatorSetCount
ordian May 18, 2022
c073e08
fmt
ordian May 18, 2022
dc3b6de
add the benchmark
ordian Jun 14, 2022
75f256d
fmt
ordian Jun 14, 2022
65851ad
unflatten slashing
ordian Jun 14, 2022
2210bad
post-rebase fixes
ordian Jun 14, 2022
6ab6b48
remove winners eagerly
ordian Jun 14, 2022
6a0ddfc
use real slashing weights for westend
ordian Jun 14, 2022
29376b1
remove bench test suite
ordian Jun 17, 2022
f76a697
Merge branch 'master' into ao-disputes-offences-runtime
ordian Jun 17, 2022
4800be1
zombinet: modify disputes test to check for an offence report
ordian Jun 17, 2022
4c0f83e
zombinet: add a timeout
ordian Jun 17, 2022
3086e26
add slashing pallet to Rococo
ordian Jun 20, 2022
677a262
zombienet: revert back to rococo-local
ordian Jun 20, 2022
e1ff282
Merge branch 'master' into ao-disputes-offences-runtime
ordian Jun 20, 2022
fe7495e
fmt
ordian Jun 20, 2022
461f37f
remove TODOs
ordian Jun 20, 2022
bae9912
revert some accidental changes
ordian Jun 20, 2022
15fcf48
slashing is submodule of disputes
ordian Jun 21, 2022
097d6c1
Merge branch 'master' into ao-disputes-offences-runtime
ordian Jun 21, 2022
65146ca
Merge branch 'master' into ao-disputes-offences-runtime
ordian Jun 28, 2022
650eb19
Merge branch 'master' into ao-disputes-offences-runtime
ordian Jul 5, 2022
95cd87d
Merge branch 'master' into ao-disputes-offences-runtime
ordian Jul 28, 2022
760bdc7
Change the log target
ordian Jul 30, 2022
d3405cc
Merge branch 'master' into ao-disputes-offences-runtime
ordian Aug 1, 2022
d5b3324
Merge branch 'ao-disputes-offences-runtime' of github.com:paritytech/…
ordian Aug 1, 2022
f50aa7d
wrap comments with rustfmt, more docs, constants
ordian Aug 1, 2022
248baa5
use Defensive trait
ordian Aug 3, 2022
b02bcef
cargo update -p sp-io
ordian Aug 3, 2022
e1c87c4
merge offence types, remove rewards for now
ordian Aug 3, 2022
37f9cd7
Merge branch 'master' into ao-disputes-offences-runtime
ordian Aug 3, 2022
b531490
cargo update -p sp-io
ordian Aug 3, 2022
454874f
benchmark fixes
ordian Aug 3, 2022
c0db743
fmt
ordian Aug 3, 2022
9fb1e38
unused var
ordian Aug 3, 2022
9c84b22
fix block_author impl
ordian Aug 3, 2022
19b7c6b
ressurect RewardValidators trait
ordian Aug 3, 2022
784e65e
remove outdated comment
ordian Aug 3, 2022
0c2559c
Merge branch 'master' into ao-disputes-offences-runtime
ordian Aug 4, 2022
3a2858d
Merge branch 'master' into ao-disputes-offences-runtime
ordian Aug 4, 2022
c22a943
more module docs
ordian Aug 5, 2022
884d9d8
introduce BenchmarkingConfig
ordian Aug 5, 2022
df81683
Merge branch 'master' into ao-disputes-offences-runtime
ordian Aug 5, 2022
7101379
typo fix
ordian Aug 5, 2022
ce879df
teach spellcheck unapplied
ordian Aug 5, 2022
b070f34
Merge branch 'master' into ao-disputes-offences-runtime
ordian Aug 31, 2022
563995b
Merge branch 'master' into ao-disputes-offences-runtime
ordian Aug 31, 2022
7d5171c
use Weight::new()
ordian Aug 31, 2022
b436249
fix mocking rewards
ordian Aug 31, 2022
397046e
use RefTimeWeight
ordian Aug 31, 2022
18965e3
".git/.scripts/bench-bot.sh" runtime westend-dev runtime_parachains::…
Aug 31, 2022
7269e18
refactor maybe_identify_validators
ordian Aug 31, 2022
399e0e6
no more ticket in disguise
ordian Aug 31, 2022
89fb5ed
remove outdated comments
ordian Aug 31, 2022
4f34bd7
lower against valid to 0.1%
ordian Sep 1, 2022
0d99787
bump zombienet version for debug
pepoviola Sep 1, 2022
9223d71
use from_perthousand
ordian Sep 1, 2022
c48aa45
Merge branch 'master' into ao-disputes-offences-runtime
ordian Sep 1, 2022
655fe98
post-merge fixes
ordian Sep 1, 2022
81578ff
Merge branch 'master' into ao-disputes-offences-runtime
ordian Sep 2, 2022
c5472c9
another day, another Weight changes
ordian Sep 2, 2022
6e5f372
Revert "bump zombienet version for debug"
ordian Sep 2, 2022
7bf295a
do not reward block authors
ordian Sep 5, 2022
8309493
Merge branch 'master' into ao-disputes-offences-runtime
ordian Sep 5, 2022
cd9aeb8
fix outdated comment
ordian Sep 6, 2022
3fe81e9
merge master once again
ordian Sep 6, 2022
e87cc9c
merge master once again II
ordian Sep 8, 2022
918b4ca
Merge branch 'master' into ao-disputes-offences-runtime
ordian Sep 19, 2022
b453b1a
use Pays from frame_support::dispatch::Pays
pepoviola Sep 19, 2022
65da473
add timeout to is up
pepoviola Sep 20, 2022
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
Prev Previous commit
Next Next commit
store ValidatorSetCount
  • Loading branch information
ordian committed Jun 14, 2022
commit 8263d16c8c69dab88fa971a097d7c89e942a0bf0
6 changes: 3 additions & 3 deletions runtime/parachains/src/disputes.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@

//! Runtime component for handling disputes of parachain candidates.

use crate::{configuration, initializer::SessionChangeNotification, session_info};
use crate::{configuration, initializer::{SessionChangeNotification, ValidatorSetCount}, session_info};
use bitvec::{bitvec, order::Lsb0 as BitOrderLsb0};
use frame_support::{ensure, traits::Get, weights::Weight};
use frame_system::pallet_prelude::*;
Expand Down Expand Up @@ -88,7 +88,7 @@ pub trait SlashingHandler<BlockNumber> {
fn initializer_finalize();

/// Called by the initializer to note that a new session has started.
fn initializer_on_new_session(notification: &SessionChangeNotification<BlockNumber>);
fn initializer_on_new_session(session_index: SessionIndex, count: ValidatorSetCount);
}

impl<BlockNumber> SlashingHandler<BlockNumber> for () {
Expand All @@ -114,7 +114,7 @@ impl<BlockNumber> SlashingHandler<BlockNumber> for () {

fn initializer_finalize() {}

fn initializer_on_new_session(_notification: &SessionChangeNotification<BlockNumber>) {}
fn initializer_on_new_session(_: SessionIndex, _: ValidatorSetCount) {}
}

/// Binary discriminator to determine if the expensive signature
Expand Down
77 changes: 46 additions & 31 deletions runtime/parachains/src/disputes/slashing/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@
//! because it is quite heavy.

use crate::{
initializer::SessionChangeNotification,
initializer::ValidatorSetCount,
session_info::{AccountId, IdentificationTuple},
};
use frame_support::{
Expand All @@ -50,7 +50,7 @@ use sp_runtime::{
};
use sp_session::{GetSessionNumber, GetValidatorCount};
use sp_staking::offence::{DisableStrategy, Kind, Offence, OffenceError, ReportOffence};
use sp_std::{collections::btree_set::BTreeSet, prelude::*};
use sp_std::{collections::btree_map::{BTreeMap, Entry}, prelude::*};

const LOG_TARGET: &str = "runtime::slashing";

Expand All @@ -72,9 +72,6 @@ impl DisputesTimeSlot {
}
}

/// Number of validators (not only parachain) in a session.
type ValidatorSetCount = u32;

/// An offence that is filed when a series of validators lost a dispute
/// about an invalid candidate.
#[derive(RuntimeDebug, TypeInfo)]
Expand Down Expand Up @@ -250,7 +247,7 @@ where
losers: impl IntoIterator<Item = ValidatorIndex>,
winners: impl IntoIterator<Item = ValidatorIndex>,
) {
let losers: Losers = losers.into_iter().collect();
let losers: Vec<ValidatorIndex> = losers.into_iter().collect();
if losers.is_empty() {
// Nothing to do
return
Expand Down Expand Up @@ -281,6 +278,9 @@ where
return
}

let losers: Losers = losers.into_iter()
.filter_map(|i| session_info.validators.get(i.0 as usize).cloned().map(|id| (i, id)))
.collect();
<PendingForInvalidLosers<T>>::insert(session_index, candidate_hash, losers);
<ForInvalidWinners<T>>::insert(session_index, candidate_hash, winners);
}
Expand All @@ -291,7 +291,7 @@ where
losers: impl IntoIterator<Item = ValidatorIndex>,
winners: impl IntoIterator<Item = ValidatorIndex>,
) {
let losers: Losers = losers.into_iter().collect();
let losers: Vec<ValidatorIndex> = losers.into_iter().collect();
if losers.is_empty() {
// Nothing to do
return
Expand Down Expand Up @@ -322,6 +322,9 @@ where
return
}

let losers: Losers = losers.into_iter()
.filter_map(|i| session_info.validators.get(i.0 as usize).cloned().map(|id| (i, id)))
.collect();
<PendingAgainstValidLosers<T>>::insert(session_index, candidate_hash, losers);
<AgainstValidWinners<T>>::insert(session_index, candidate_hash, winners);
}
Expand All @@ -334,8 +337,8 @@ where
Pallet::<T>::initializer_finalize()
}

fn initializer_on_new_session(notification: &SessionChangeNotification<T::BlockNumber>) {
Pallet::<T>::initializer_on_new_session(notification)
fn initializer_on_new_session(session_index: SessionIndex, count: ValidatorSetCount) {
Pallet::<T>::initializer_on_new_session(session_index, count)
}
}

Expand All @@ -361,8 +364,8 @@ pub struct DisputeProof {
pub validator_id: ValidatorId,
}

/// Indices of the validators who lost a dispute and are pending slashes.
pub type Losers = BTreeSet<ValidatorIndex>;
/// Indices and keys of the validators who lost a dispute and are pending slashes.
pub type Losers = BTreeMap<ValidatorIndex, ValidatorId>;
/// `AccountId`s of the validators who were on the winning side of a dispute.
pub type Winners<T> = Vec<AccountId<T>>;

Expand Down Expand Up @@ -513,6 +516,15 @@ pub mod pallet {
Winners<T>,
>;

/// `ValidatorSetCount` per session.
#[pallet::storage]
pub(super) type ValidatorSetCounts<T> = StorageMap<
_,
Twox64Concat,
SessionIndex,
ValidatorSetCount,
>;

/// Indices of the validators pending "against valid" dispute slashes.
#[pallet::storage]
pub(super) type PendingAgainstValidLosers<T> =
Expand Down Expand Up @@ -563,31 +575,28 @@ pub mod pallet {
let offender = T::KeyOwnerProofSystem::check_proof(key, key_owner_proof)
.ok_or(Error::<T>::InvalidKeyOwnershipProof)?;

// check that `validator_index` matches `validator_id`
let session_index = dispute_proof.time_slot.session_index;
let validator_set_count =
if let Some(info) = crate::session_info::Pallet::<T>::session_info(session_index) {
let i = dispute_proof.validator_index.0 as usize;
ensure!(
info.validators.get(i) == Some(&dispute_proof.validator_id),
Error::<T>::ValidatorIndexIdMismatch
);
// number of all validators (not only parachain) in the session
info.discovery_keys.len() as ValidatorSetCount
} else {
return Err(Error::<T>::InvalidSessionIndex.into())
};
let validator_set_count = <ValidatorSetCounts<T>>::get(session_index)
.ok_or(Error::<T>::InvalidSessionIndex)?;

// check that there is a pending slash for the given
// validator index and candidate hash
let candidate_hash = dispute_proof.time_slot.candidate_hash;
let try_remove = |v: &mut Option<Losers>| -> Result<(), DispatchError> {
let indices = v.as_mut().ok_or(Error::<T>::InvalidCandidateHash)?;

ensure!(
indices.remove(&dispute_proof.validator_index),
Error::<T>::InvalidValidatorIndex,
);
match indices.entry(dispute_proof.validator_index) {
Entry::Vacant(_) => {
return Err(Error::<T>::InvalidValidatorIndex.into())
}
// check that `validator_index` matches `validator_id`
Entry::Occupied(e) if e.get() != &dispute_proof.validator_id => {
return Err(Error::<T>::ValidatorIndexIdMismatch.into())
}
Entry::Occupied(e) => {
e.remove(); // all good
}
}

if indices.is_empty() {
*v = None;
Expand Down Expand Up @@ -673,13 +682,18 @@ impl<T: Config> Pallet<T> {
fn initializer_finalize() {}

/// Called by the initializer to note a new session in the disputes slashing pallet.
fn initializer_on_new_session(notification: &SessionChangeNotification<T::BlockNumber>) {
fn initializer_on_new_session(
session_index: SessionIndex,
validator_set_count: ValidatorSetCount,
) {
<ValidatorSetCounts<T>>::insert(session_index, validator_set_count);

let config = <super::configuration::Pallet<T>>::config();
if notification.session_index <= config.dispute_period + 1 {
if session_index <= config.dispute_period + 1 {
return
}

let old_session = notification.session_index - config.dispute_period - 1;
let old_session = session_index - config.dispute_period - 1;

match <PendingForInvalidLosers<T>>::remove_prefix(old_session, None) {
sp_io::KillStorageResult::AllRemoved(x) if x > 0 => {
Expand All @@ -703,6 +717,7 @@ impl<T: Config> Pallet<T> {
}
<ForInvalidWinners<T>>::remove_prefix(old_session, None);
<AgainstValidWinners<T>>::remove_prefix(old_session, None);
<ValidatorSetCounts<T>>::remove(old_session);
}
}

Expand Down
6 changes: 5 additions & 1 deletion runtime/parachains/src/initializer.rs
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,9 @@ pub struct SessionChangeNotification<BlockNumber> {
pub session_index: SessionIndex,
}

/// Number of validators (not only parachain) in a session.
pub type ValidatorSetCount = u32;

impl<BlockNumber: Default + From<u32>> Default for SessionChangeNotification<BlockNumber> {
fn default() -> Self {
Self {
Expand Down Expand Up @@ -242,6 +245,7 @@ impl<T: Config> Pallet<T> {
configuration::Pallet::<T>::initializer_on_new_session(&session_index);
let new_config = new_config.unwrap_or_else(|| prev_config.clone());

let validator_set_count = all_validators.len() as ValidatorSetCount;
let validators = shared::Pallet::<T>::initializer_on_new_session(
session_index,
random_seed.clone(),
Expand All @@ -263,7 +267,7 @@ impl<T: Config> Pallet<T> {
inclusion::Pallet::<T>::initializer_on_new_session(&notification);
session_info::Pallet::<T>::initializer_on_new_session(&notification);
T::DisputesHandler::initializer_on_new_session(&notification);
T::SlashingHandler::initializer_on_new_session(&notification);
T::SlashingHandler::initializer_on_new_session(session_index, validator_set_count);
dmp::Pallet::<T>::initializer_on_new_session(&notification, &outgoing_paras);
ump::Pallet::<T>::initializer_on_new_session(&notification, &outgoing_paras);
hrmp::Pallet::<T>::initializer_on_new_session(&notification, &outgoing_paras);
Expand Down
2 changes: 1 addition & 1 deletion runtime/parachains/src/mock.rs
Original file line number Diff line number Diff line change
Expand Up @@ -277,7 +277,7 @@ impl crate::disputes::SlashingHandler<BlockNumber> for Test {

fn initializer_finalize() {}

fn initializer_on_new_session(_: &initializer::SessionChangeNotification<BlockNumber>) {}
fn initializer_on_new_session(_: SessionIndex, _: u32) {}
}

impl crate::scheduler::Config for Test {}
Expand Down