1c0e57d984
This commit systematically rebrands various references from Parity Technologies' Polkadot/Substrate ecosystem to PezkuwiChain within the kurdistan-sdk. Key changes include: - Updated external repository URLs (zombienet-sdk, parity-db, parity-scale-codec, wasm-instrument) to point to pezkuwichain forks. - Modified internal documentation and code comments to reflect PezkuwiChain naming and structure. - Replaced direct references to with or specific paths within the for XCM, Pezkuwi, and other modules. - Cleaned up deprecated issue and PR references in various and files, particularly in and modules. - Adjusted image and logo URLs in documentation to point to PezkuwiChain assets. - Removed or rephrased comments related to external Polkadot/Substrate PRs and issues. This is a significant step towards fully customizing the SDK for the PezkuwiChain ecosystem.
344 lines
11 KiB
Rust
344 lines
11 KiB
Rust
// Copyright (C) Parity Technologies (UK) Ltd.
|
|
// This file is part of Pezkuwi.
|
|
|
|
// Pezkuwi is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
|
|
// Pezkuwi is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with Pezkuwi. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
use std::collections::{
|
|
btree_map::{Entry as Bentry, Keys as Bkeys},
|
|
BTreeMap, BTreeSet,
|
|
};
|
|
|
|
use codec::{Decode, Encode};
|
|
|
|
use pezsp_application_crypto::AppCrypto;
|
|
use pezsp_keystore::{Error as KeystoreError, KeystorePtr};
|
|
|
|
use pezkuwi_primitives::{
|
|
CandidateHash, CandidateReceiptV2 as CandidateReceipt, CompactStatement, DisputeStatement,
|
|
EncodeAs, InvalidDisputeStatementKind, SessionIndex, SigningContext, UncheckedSigned,
|
|
ValidDisputeStatementKind, ValidatorId, ValidatorIndex, ValidatorSignature,
|
|
};
|
|
|
|
/// `DisputeMessage` and related types.
|
|
mod message;
|
|
pub use message::{DisputeMessage, Error as DisputeMessageCheckError, UncheckedDisputeMessage};
|
|
mod status;
|
|
pub use status::{dispute_is_inactive, DisputeStatus, Timestamp, ACTIVE_DURATION_SECS};
|
|
|
|
/// A checked dispute statement from an associated validator.
|
|
#[derive(Debug, Clone)]
|
|
pub struct SignedDisputeStatement {
|
|
dispute_statement: DisputeStatement,
|
|
candidate_hash: CandidateHash,
|
|
validator_public: ValidatorId,
|
|
validator_signature: ValidatorSignature,
|
|
session_index: SessionIndex,
|
|
}
|
|
|
|
/// Errors encountered while signing a dispute statement
|
|
#[derive(Debug)]
|
|
pub enum SignedDisputeStatementError {
|
|
/// Encountered a keystore error while signing
|
|
KeyStoreError(KeystoreError),
|
|
/// Could not generate signing payload
|
|
PayloadError,
|
|
}
|
|
|
|
/// Tracked votes on candidates, for the purposes of dispute resolution.
|
|
#[derive(Debug, Clone)]
|
|
pub struct CandidateVotes {
|
|
/// The receipt of the candidate itself.
|
|
pub candidate_receipt: CandidateReceipt,
|
|
/// Votes of validity, sorted by validator index.
|
|
pub valid: ValidCandidateVotes,
|
|
/// Votes of invalidity, sorted by validator index.
|
|
pub invalid: BTreeMap<ValidatorIndex, (InvalidDisputeStatementKind, ValidatorSignature)>,
|
|
}
|
|
|
|
/// Type alias for retrieving valid votes from `CandidateVotes`
|
|
pub type ValidVoteData = (ValidatorIndex, (ValidDisputeStatementKind, ValidatorSignature));
|
|
|
|
/// Type alias for retrieving invalid votes from `CandidateVotes`
|
|
pub type InvalidVoteData = (ValidatorIndex, (InvalidDisputeStatementKind, ValidatorSignature));
|
|
|
|
impl CandidateVotes {
|
|
/// Get the set of all validators who have votes in the set, ascending.
|
|
pub fn voted_indices(&self) -> BTreeSet<ValidatorIndex> {
|
|
let mut keys: BTreeSet<_> = self.valid.keys().cloned().collect();
|
|
keys.extend(self.invalid.keys().cloned());
|
|
keys
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Clone)]
|
|
/// Valid candidate votes.
|
|
///
|
|
/// Prefer backing votes over other votes.
|
|
pub struct ValidCandidateVotes {
|
|
votes: BTreeMap<ValidatorIndex, (ValidDisputeStatementKind, ValidatorSignature)>,
|
|
}
|
|
|
|
impl ValidCandidateVotes {
|
|
/// Create new empty `ValidCandidateVotes`
|
|
pub fn new() -> Self {
|
|
Self { votes: BTreeMap::new() }
|
|
}
|
|
/// Insert a vote, replacing any already existing vote.
|
|
///
|
|
/// Except, for backing votes: Backing votes are always kept, and will never get overridden.
|
|
/// Import of other king of `valid` votes, will be ignored if a backing vote is already
|
|
/// present. Any already existing `valid` vote, will be overridden by any given backing vote.
|
|
///
|
|
/// Returns: true, if the insert had any effect.
|
|
pub fn insert_vote(
|
|
&mut self,
|
|
validator_index: ValidatorIndex,
|
|
kind: ValidDisputeStatementKind,
|
|
sig: ValidatorSignature,
|
|
) -> bool {
|
|
match self.votes.entry(validator_index) {
|
|
Bentry::Vacant(vacant) => {
|
|
vacant.insert((kind, sig));
|
|
true
|
|
},
|
|
Bentry::Occupied(mut occupied) => match occupied.get().0 {
|
|
ValidDisputeStatementKind::BackingValid(_) |
|
|
ValidDisputeStatementKind::BackingSeconded(_) => false,
|
|
ValidDisputeStatementKind::Explicit |
|
|
ValidDisputeStatementKind::ApprovalChecking |
|
|
ValidDisputeStatementKind::ApprovalCheckingMultipleCandidates(_) => {
|
|
occupied.insert((kind.clone(), sig));
|
|
kind != occupied.get().0
|
|
},
|
|
},
|
|
}
|
|
}
|
|
|
|
/// Retain any votes that match the given criteria.
|
|
pub fn retain<F>(&mut self, f: F)
|
|
where
|
|
F: FnMut(&ValidatorIndex, &mut (ValidDisputeStatementKind, ValidatorSignature)) -> bool,
|
|
{
|
|
self.votes.retain(f)
|
|
}
|
|
|
|
/// Get all the validator indices we have votes for.
|
|
pub fn keys(
|
|
&self,
|
|
) -> Bkeys<'_, ValidatorIndex, (ValidDisputeStatementKind, ValidatorSignature)> {
|
|
self.votes.keys()
|
|
}
|
|
|
|
/// Get read only direct access to underlying map.
|
|
pub fn raw(
|
|
&self,
|
|
) -> &BTreeMap<ValidatorIndex, (ValidDisputeStatementKind, ValidatorSignature)> {
|
|
&self.votes
|
|
}
|
|
}
|
|
|
|
impl FromIterator<(ValidatorIndex, (ValidDisputeStatementKind, ValidatorSignature))>
|
|
for ValidCandidateVotes
|
|
{
|
|
fn from_iter<T>(iter: T) -> Self
|
|
where
|
|
T: IntoIterator<Item = (ValidatorIndex, (ValidDisputeStatementKind, ValidatorSignature))>,
|
|
{
|
|
Self { votes: BTreeMap::from_iter(iter) }
|
|
}
|
|
}
|
|
|
|
impl From<ValidCandidateVotes>
|
|
for BTreeMap<ValidatorIndex, (ValidDisputeStatementKind, ValidatorSignature)>
|
|
{
|
|
fn from(wrapped: ValidCandidateVotes) -> Self {
|
|
wrapped.votes
|
|
}
|
|
}
|
|
impl IntoIterator for ValidCandidateVotes {
|
|
type Item = (ValidatorIndex, (ValidDisputeStatementKind, ValidatorSignature));
|
|
type IntoIter = <BTreeMap<ValidatorIndex, (ValidDisputeStatementKind, ValidatorSignature)> as IntoIterator>::IntoIter;
|
|
|
|
fn into_iter(self) -> Self::IntoIter {
|
|
self.votes.into_iter()
|
|
}
|
|
}
|
|
|
|
impl SignedDisputeStatement {
|
|
/// Create a new `SignedDisputeStatement` from information
|
|
/// that is available on-chain, and hence already can be trusted.
|
|
///
|
|
/// Attention: Not to be used other than with guaranteed fetches.
|
|
pub fn new_unchecked_from_trusted_source(
|
|
dispute_statement: DisputeStatement,
|
|
candidate_hash: CandidateHash,
|
|
session_index: SessionIndex,
|
|
validator_public: ValidatorId,
|
|
validator_signature: ValidatorSignature,
|
|
) -> Self {
|
|
SignedDisputeStatement {
|
|
dispute_statement,
|
|
candidate_hash,
|
|
validator_public,
|
|
validator_signature,
|
|
session_index,
|
|
}
|
|
}
|
|
|
|
/// Create a new `SignedDisputeStatement`, which is only possible by checking the signature.
|
|
pub fn new_checked(
|
|
dispute_statement: DisputeStatement,
|
|
candidate_hash: CandidateHash,
|
|
session_index: SessionIndex,
|
|
validator_public: ValidatorId,
|
|
validator_signature: ValidatorSignature,
|
|
) -> Result<Self, ()> {
|
|
dispute_statement
|
|
.check_signature(&validator_public, candidate_hash, session_index, &validator_signature)
|
|
.map(|_| SignedDisputeStatement {
|
|
dispute_statement,
|
|
candidate_hash,
|
|
validator_public,
|
|
validator_signature,
|
|
session_index,
|
|
})
|
|
}
|
|
|
|
/// Sign this statement with the given keystore and key. Pass `valid = true` to
|
|
/// indicate validity of the candidate, and `valid = false` to indicate invalidity.
|
|
pub fn sign_explicit(
|
|
keystore: &KeystorePtr,
|
|
valid: bool,
|
|
candidate_hash: CandidateHash,
|
|
session_index: SessionIndex,
|
|
validator_public: ValidatorId,
|
|
) -> Result<Option<Self>, SignedDisputeStatementError> {
|
|
let dispute_statement = if valid {
|
|
DisputeStatement::Valid(ValidDisputeStatementKind::Explicit)
|
|
} else {
|
|
DisputeStatement::Invalid(InvalidDisputeStatementKind::Explicit)
|
|
};
|
|
|
|
let data = dispute_statement
|
|
.payload_data(candidate_hash, session_index)
|
|
.map_err(|_| SignedDisputeStatementError::PayloadError)?;
|
|
let signature = keystore
|
|
.sr25519_sign(ValidatorId::ID, validator_public.as_ref(), &data)
|
|
.map_err(SignedDisputeStatementError::KeyStoreError)?
|
|
.map(|sig| Self {
|
|
dispute_statement,
|
|
candidate_hash,
|
|
validator_public,
|
|
validator_signature: sig.into(),
|
|
session_index,
|
|
});
|
|
Ok(signature)
|
|
}
|
|
|
|
/// Access the underlying dispute statement
|
|
pub fn statement(&self) -> &DisputeStatement {
|
|
&self.dispute_statement
|
|
}
|
|
|
|
/// Access the underlying candidate hash.
|
|
pub fn candidate_hash(&self) -> &CandidateHash {
|
|
&self.candidate_hash
|
|
}
|
|
|
|
/// Access the underlying validator public key.
|
|
pub fn validator_public(&self) -> &ValidatorId {
|
|
&self.validator_public
|
|
}
|
|
|
|
/// Access the underlying validator signature.
|
|
pub fn validator_signature(&self) -> &ValidatorSignature {
|
|
&self.validator_signature
|
|
}
|
|
|
|
/// Consume self to return the signature.
|
|
pub fn into_validator_signature(self) -> ValidatorSignature {
|
|
self.validator_signature
|
|
}
|
|
|
|
/// Access the underlying session index.
|
|
pub fn session_index(&self) -> SessionIndex {
|
|
self.session_index
|
|
}
|
|
|
|
/// Convert a unchecked backing statement to a [`SignedDisputeStatement`]
|
|
///
|
|
/// As the unchecked backing statement contains only the validator index and
|
|
/// not the validator public key, the public key must be passed as well,
|
|
/// along with the signing context.
|
|
///
|
|
/// This does signature checks again with the data provided.
|
|
pub fn from_backing_statement<T>(
|
|
backing_statement: &UncheckedSigned<T, CompactStatement>,
|
|
signing_context: SigningContext,
|
|
validator_public: ValidatorId,
|
|
) -> Result<Self, ()>
|
|
where
|
|
for<'a> &'a T: Into<CompactStatement>,
|
|
T: EncodeAs<CompactStatement>,
|
|
{
|
|
let (statement_kind, candidate_hash) = match backing_statement.unchecked_payload().into() {
|
|
CompactStatement::Seconded(candidate_hash) => (
|
|
ValidDisputeStatementKind::BackingSeconded(signing_context.parent_hash),
|
|
candidate_hash,
|
|
),
|
|
CompactStatement::Valid(candidate_hash) => (
|
|
ValidDisputeStatementKind::BackingValid(signing_context.parent_hash),
|
|
candidate_hash,
|
|
),
|
|
};
|
|
|
|
let dispute_statement = DisputeStatement::Valid(statement_kind);
|
|
Self::new_checked(
|
|
dispute_statement,
|
|
candidate_hash,
|
|
signing_context.session_index,
|
|
validator_public,
|
|
backing_statement.unchecked_signature().clone(),
|
|
)
|
|
}
|
|
}
|
|
|
|
/// Any invalid vote (currently only explicit).
|
|
#[derive(Clone, Encode, Decode, Debug)]
|
|
pub struct InvalidDisputeVote {
|
|
/// The voting validator index.
|
|
pub validator_index: ValidatorIndex,
|
|
|
|
/// The validator signature, that can be verified when constructing a
|
|
/// `SignedDisputeStatement`.
|
|
pub signature: ValidatorSignature,
|
|
|
|
/// Kind of dispute statement.
|
|
pub kind: InvalidDisputeStatementKind,
|
|
}
|
|
|
|
/// Any valid vote (backing, approval, explicit).
|
|
#[derive(Clone, Encode, Decode, Debug)]
|
|
pub struct ValidDisputeVote {
|
|
/// The voting validator index.
|
|
pub validator_index: ValidatorIndex,
|
|
|
|
/// The validator signature, that can be verified when constructing a
|
|
/// `SignedDisputeStatement`.
|
|
pub signature: ValidatorSignature,
|
|
|
|
/// Kind of dispute statement.
|
|
pub kind: ValidDisputeStatementKind,
|
|
}
|