mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-05-30 09:21:04 +00:00
a5034dbe98
* use stronger types for HeadData and ValidationCode in runtime * fix weird debug compile error * fix runtime build * update invocations invalidation.rs * fix tests
234 lines
7.5 KiB
Rust
234 lines
7.5 KiB
Rust
// Copyright 2020 Parity Technologies (UK) Ltd.
|
|
// This file is part of Polkadot.
|
|
|
|
// Polkadot 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.
|
|
|
|
// Polkadot 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 Polkadot. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
//! Primitive types which are strictly necessary from a parachain-execution point
|
|
//! of view.
|
|
|
|
use sp_std::vec::Vec;
|
|
|
|
use codec::{Encode, Decode, CompactAs};
|
|
use sp_core::{RuntimeDebug, TypeId};
|
|
|
|
#[cfg(feature = "std")]
|
|
use serde::{Serialize, Deserialize};
|
|
|
|
#[cfg(feature = "std")]
|
|
use sp_core::bytes;
|
|
|
|
/// The block number of the relay chain.
|
|
/// 32-bits will allow for 136 years of blocks assuming 1 block per second.
|
|
pub type RelayChainBlockNumber = u32;
|
|
|
|
/// Parachain head data included in the chain.
|
|
#[derive(PartialEq, Eq, Clone, PartialOrd, Ord, Encode, Decode, RuntimeDebug)]
|
|
#[cfg_attr(feature = "std", derive(Serialize, Deserialize, Default))]
|
|
pub struct HeadData(#[cfg_attr(feature = "std", serde(with="bytes"))] pub Vec<u8>);
|
|
|
|
impl From<Vec<u8>> for HeadData {
|
|
fn from(head: Vec<u8>) -> Self {
|
|
HeadData(head)
|
|
}
|
|
}
|
|
|
|
/// Parachain validation code.
|
|
#[derive(Default, PartialEq, Eq, Clone, Encode, Decode, RuntimeDebug)]
|
|
#[cfg_attr(feature = "std", derive(Serialize, Deserialize))]
|
|
pub struct ValidationCode(#[cfg_attr(feature = "std", serde(with="bytes"))] pub Vec<u8>);
|
|
|
|
impl From<Vec<u8>> for ValidationCode {
|
|
fn from(code: Vec<u8>) -> Self {
|
|
ValidationCode(code)
|
|
}
|
|
}
|
|
|
|
/// Parachain block data.
|
|
///
|
|
/// Contains everything required to validate para-block, may contain block and witness data.
|
|
#[derive(PartialEq, Eq, Clone, Encode, Decode)]
|
|
#[cfg_attr(feature = "std", derive(Serialize, Deserialize, Debug))]
|
|
pub struct BlockData(#[cfg_attr(feature = "std", serde(with="bytes"))] pub Vec<u8>);
|
|
|
|
/// Unique identifier of a parachain.
|
|
#[derive(
|
|
Clone, CompactAs, Copy, Decode, Default, Encode, Eq,
|
|
Hash, Ord, PartialEq, PartialOrd, RuntimeDebug,
|
|
)]
|
|
#[cfg_attr(feature = "std", derive(serde::Serialize, serde::Deserialize, derive_more::Display))]
|
|
pub struct Id(u32);
|
|
|
|
impl TypeId for Id {
|
|
const TYPE_ID: [u8; 4] = *b"para";
|
|
}
|
|
|
|
impl From<Id> for u32 {
|
|
fn from(x: Id) -> Self { x.0 }
|
|
}
|
|
|
|
impl From<u32> for Id {
|
|
fn from(x: u32) -> Self { Id(x) }
|
|
}
|
|
|
|
const USER_INDEX_START: u32 = 1000;
|
|
|
|
/// The ID of the first user (non-system) parachain.
|
|
pub const LOWEST_USER_ID: Id = Id(USER_INDEX_START);
|
|
|
|
impl Id {
|
|
/// Create an `Id`.
|
|
pub const fn new(id: u32) -> Self {
|
|
Self(id)
|
|
}
|
|
|
|
/// Returns `true` if this parachain runs with system-level privileges.
|
|
pub fn is_system(&self) -> bool { self.0 < USER_INDEX_START }
|
|
}
|
|
|
|
impl sp_std::ops::Add<u32> for Id {
|
|
type Output = Self;
|
|
|
|
fn add(self, other: u32) -> Self {
|
|
Self(self.0 + other)
|
|
}
|
|
}
|
|
|
|
/// This type can be converted into and possibly from an AccountId (which itself is generic).
|
|
pub trait AccountIdConversion<AccountId>: Sized {
|
|
/// Convert into an account ID. This is infallible.
|
|
fn into_account(&self) -> AccountId;
|
|
|
|
/// Try to convert an account ID into this type. Might not succeed.
|
|
fn try_from_account(a: &AccountId) -> Option<Self>;
|
|
}
|
|
|
|
// TODO: Remove all of this, move sp-runtime::AccountIdConversion to own crate and and use that.
|
|
// #360
|
|
struct TrailingZeroInput<'a>(&'a [u8]);
|
|
impl<'a> codec::Input for TrailingZeroInput<'a> {
|
|
fn remaining_len(&mut self) -> Result<Option<usize>, codec::Error> {
|
|
Ok(None)
|
|
}
|
|
|
|
fn read(&mut self, into: &mut [u8]) -> Result<(), codec::Error> {
|
|
let len = into.len().min(self.0.len());
|
|
into[..len].copy_from_slice(&self.0[..len]);
|
|
for i in &mut into[len..] {
|
|
*i = 0;
|
|
}
|
|
self.0 = &self.0[len..];
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
/// Format is b"para" ++ encode(parachain ID) ++ 00.... where 00... is indefinite trailing
|
|
/// zeroes to fill AccountId.
|
|
impl<T: Encode + Decode + Default> AccountIdConversion<T> for Id {
|
|
fn into_account(&self) -> T {
|
|
(b"para", self).using_encoded(|b|
|
|
T::decode(&mut TrailingZeroInput(b))
|
|
).unwrap_or_default()
|
|
}
|
|
|
|
fn try_from_account(x: &T) -> Option<Self> {
|
|
x.using_encoded(|d| {
|
|
if &d[0..4] != b"para" { return None }
|
|
let mut cursor = &d[4..];
|
|
let result = Decode::decode(&mut cursor).ok()?;
|
|
if cursor.iter().all(|x| *x == 0) {
|
|
Some(result)
|
|
} else {
|
|
None
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
/// Which origin a parachain's message to the relay chain should be dispatched from.
|
|
#[derive(Clone, PartialEq, Eq, Encode, Decode)]
|
|
#[cfg_attr(feature = "std", derive(Debug))]
|
|
#[repr(u8)]
|
|
pub enum ParachainDispatchOrigin {
|
|
/// As a simple `Origin::Signed`, using `ParaId::account_id` as its value. This is good when
|
|
/// interacting with standard modules such as `balances`.
|
|
Signed,
|
|
/// As the special `Origin::Parachain(ParaId)`. This is good when interacting with parachain-
|
|
/// aware modules which need to succinctly verify that the origin is a parachain.
|
|
Parachain,
|
|
/// As the simple, superuser `Origin::Root`. This can only be done on specially permissioned
|
|
/// parachains.
|
|
Root,
|
|
}
|
|
|
|
impl sp_std::convert::TryFrom<u8> for ParachainDispatchOrigin {
|
|
type Error = ();
|
|
fn try_from(x: u8) -> core::result::Result<ParachainDispatchOrigin, ()> {
|
|
const SIGNED: u8 = ParachainDispatchOrigin::Signed as u8;
|
|
const PARACHAIN: u8 = ParachainDispatchOrigin::Parachain as u8;
|
|
Ok(match x {
|
|
SIGNED => ParachainDispatchOrigin::Signed,
|
|
PARACHAIN => ParachainDispatchOrigin::Parachain,
|
|
_ => return Err(()),
|
|
})
|
|
}
|
|
}
|
|
|
|
/// A message from a parachain to its Relay Chain.
|
|
#[derive(Clone, PartialEq, Eq, Encode, Decode)]
|
|
#[cfg_attr(
|
|
any(feature = "std", feature = "wasm-api"),
|
|
derive(sp_runtime_interface::pass_by::PassByCodec,
|
|
))]
|
|
#[cfg_attr(feature = "std", derive(Debug))]
|
|
pub struct UpwardMessage {
|
|
/// The origin for the message to be sent from.
|
|
pub origin: ParachainDispatchOrigin,
|
|
/// The message data.
|
|
pub data: Vec<u8>,
|
|
}
|
|
|
|
/// Validation parameters for evaluating the parachain validity function.
|
|
// TODO: balance downloads (https://github.com/paritytech/polkadot/issues/220)
|
|
#[derive(PartialEq, Eq, Decode)]
|
|
#[cfg_attr(feature = "std", derive(Debug, Encode))]
|
|
pub struct ValidationParams {
|
|
/// The collation body.
|
|
pub block_data: BlockData,
|
|
/// Previous head-data.
|
|
pub parent_head: HeadData,
|
|
/// The maximum code size permitted, in bytes.
|
|
pub max_code_size: u32,
|
|
/// The maximum head-data size permitted, in bytes.
|
|
pub max_head_data_size: u32,
|
|
/// The current relay-chain block number.
|
|
pub relay_chain_height: RelayChainBlockNumber,
|
|
/// Whether a code upgrade is allowed or not, and at which height the upgrade
|
|
/// would be applied after, if so. The parachain logic should apply any upgrade
|
|
/// issued in this block after the first block
|
|
/// with `relay_chain_height` at least this value, if `Some`. if `None`, issue
|
|
/// no upgrade.
|
|
pub code_upgrade_allowed: Option<RelayChainBlockNumber>,
|
|
}
|
|
|
|
/// The result of parachain validation.
|
|
// TODO: egress and balance uploads (https://github.com/paritytech/polkadot/issues/220)
|
|
#[derive(PartialEq, Eq, Encode)]
|
|
#[cfg_attr(feature = "std", derive(Debug, Decode))]
|
|
pub struct ValidationResult {
|
|
/// New head data that should be included in the relay chain state.
|
|
pub head_data: HeadData,
|
|
/// An update to the validation code that should be scheduled in the relay chain.
|
|
pub new_validation_code: Option<ValidationCode>,
|
|
}
|