mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-27 06:57:58 +00:00
93e8ffed55
* Introduce preimages module in traits * Multisize Preimages * Len not actually necessary * Tweaks to the preimage API * Fixes * Get Scheduler building with new API * Scheduler tests pass * Bounded Scheduler 🎉 * Use Agenda holes and introduce IncompleteSince to avoid need to reschedule * Tests pass with new weight system * New benchmarks * Add missing file * Drop preimage when permenantly overeight * Drop preimage when permenantly overeight * Referenda uses latest preimage API * Testing ok * Adding tests Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * fmt Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Add preimage migration Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Docs * Remove dbg * Refactor Democracy * Refactor Democracy * Add final MEL * Remove silly maps * Fixes * Minor refactor * Formatting * Fixes * Fixes * Fixes * Update frame/preimage/src/lib.rs Co-authored-by: Shawn Tabrizi <shawntabrizi@gmail.com> * Add migrations to Democracy * WIP Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Resolve conflicts Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Revert "Resolve conflicts" This reverts commit 734d66d69e54553471ffa54fa52e3e304dc8f106. * Undo wrong resolves... Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * WIP Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Make compile Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * massage clippy Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * More clippy Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * clippy annoyance Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * clippy annoyance Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Fix benchmarks Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * add missing file * Test <Preimage as QueryPreimage> Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * More tests Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Clippy harassment Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Add test Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * clippy Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Fixup tests Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Remove old stuff Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * fmt Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Test <Scheduler as Anon> trait functions Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Update pallet-ui tests Why is this needed? Should not be the case unless master is broken... Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * More scheduler trait test Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * More tests Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Apply review suggestion Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Beauty fixes Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Add Scheduler test migration_v3_to_v4_works Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Merge fixup Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Keep referenda benchmarks instantiatable Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Update weights Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Use new scheduler weight functions Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Use new democracy weight functions Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Use weight compare functions Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Update pallet-ui tests Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * More renaming… Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * More renaming… Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Add comment Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Implement OnRuntimeUpgrade for scheduler::v3_to_v4 migration Put the migration into a proper `MigrateToV4` struct and implement the OnRuntimeUpgrade hooks for it. Also move the test to use that instead. This should make it easier for adding it to Polkadot. Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Clippy Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Handle undecodable Agendas Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Remove trash Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Fix test Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Use new OnRuntimeUpgrade functions Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * fix test Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Fix BoundedSlice::truncate_from Co-authored-by: jakoblell Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Fix pre_upgrade hook return values Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Add more error logging Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Find too large preimages in the pre_upgrade hook Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Test that too large Calls in agendas are ignored Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Use new OnRuntimeUpgrade hooks Why did the CI not catch this?! Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * works fine - just more logs Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Fix staking migration Causing issues on Kusama... Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Fix UI tests No idea why this is needed. This is actually undoing an earlier change. Maybe the CI has different rustc versions!? Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Remove multisig's Calls (#12072) * Remove multisig's Calls * Multisig: Fix tests and re-introduce reserve logic (#12241) * Fix tests and re-introduce reserve logic * fix benches * add todo * remove irrelevant bench * [Feature] Add a migration that drains and refunds stored calls (#12313) * [Feature] Add a migration that drains and refunds stored calls * migration fixes * fixes * address review comments * consume the whole block weight * fix assertions * license header * fix interface Co-authored-by: parity-processbot <> Co-authored-by: parity-processbot <> Co-authored-by: Roman Useinov <roman.useinov@gmail.com> * Fix test Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Fix multisig benchmarks Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * ".git/.scripts/bench-bot.sh" pallet dev pallet_democracy * ".git/.scripts/bench-bot.sh" pallet dev pallet_scheduler * ".git/.scripts/bench-bot.sh" pallet dev pallet_preimage Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> Co-authored-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> Co-authored-by: Shawn Tabrizi <shawntabrizi@gmail.com> Co-authored-by: parity-processbot <> Co-authored-by: Roman Useinov <roman.useinov@gmail.com>
456 lines
16 KiB
Rust
456 lines
16 KiB
Rust
// This file is part of Substrate.
|
|
|
|
// Copyright (C) 2017-2022 Parity Technologies (UK) Ltd.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
//! # Preimage Pallet
|
|
//!
|
|
//! - [`Config`]
|
|
//! - [`Call`]
|
|
//!
|
|
//! ## Overview
|
|
//!
|
|
//! The Preimage pallet allows for the users and the runtime to store the preimage
|
|
//! of a hash on chain. This can be used by other pallets for storing and managing
|
|
//! large byte-blobs.
|
|
|
|
#![cfg_attr(not(feature = "std"), no_std)]
|
|
|
|
#[cfg(feature = "runtime-benchmarks")]
|
|
mod benchmarking;
|
|
pub mod migration;
|
|
#[cfg(test)]
|
|
mod mock;
|
|
#[cfg(test)]
|
|
mod tests;
|
|
pub mod weights;
|
|
|
|
use sp_runtime::traits::{BadOrigin, Hash, Saturating};
|
|
use sp_std::{borrow::Cow, prelude::*};
|
|
|
|
use codec::{Decode, Encode, MaxEncodedLen};
|
|
use frame_support::{
|
|
dispatch::Pays,
|
|
ensure,
|
|
pallet_prelude::Get,
|
|
traits::{
|
|
Currency, Defensive, FetchResult, Hash as PreimageHash, PreimageProvider,
|
|
PreimageRecipient, QueryPreimage, ReservableCurrency, StorePreimage,
|
|
},
|
|
BoundedSlice, BoundedVec,
|
|
};
|
|
use scale_info::TypeInfo;
|
|
pub use weights::WeightInfo;
|
|
|
|
use frame_support::pallet_prelude::*;
|
|
use frame_system::pallet_prelude::*;
|
|
|
|
pub use pallet::*;
|
|
|
|
/// A type to note whether a preimage is owned by a user or the system.
|
|
#[derive(Clone, Eq, PartialEq, Encode, Decode, TypeInfo, MaxEncodedLen, RuntimeDebug)]
|
|
pub enum RequestStatus<AccountId, Balance> {
|
|
/// The associated preimage has not yet been requested by the system. The given deposit (if
|
|
/// some) is being held until either it becomes requested or the user retracts the preimage.
|
|
Unrequested { deposit: (AccountId, Balance), len: u32 },
|
|
/// There are a non-zero number of outstanding requests for this hash by this chain. If there
|
|
/// is a preimage registered, then `len` is `Some` and it may be removed iff this counter
|
|
/// becomes zero.
|
|
Requested { deposit: Option<(AccountId, Balance)>, count: u32, len: Option<u32> },
|
|
}
|
|
|
|
type BalanceOf<T> =
|
|
<<T as Config>::Currency as Currency<<T as frame_system::Config>::AccountId>>::Balance;
|
|
|
|
/// Maximum size of preimage we can store is 4mb.
|
|
const MAX_SIZE: u32 = 4 * 1024 * 1024;
|
|
|
|
#[frame_support::pallet]
|
|
pub mod pallet {
|
|
use super::*;
|
|
|
|
/// The current storage version.
|
|
const STORAGE_VERSION: StorageVersion = StorageVersion::new(1);
|
|
|
|
#[pallet::config]
|
|
pub trait Config: frame_system::Config {
|
|
/// The overarching event type.
|
|
type RuntimeEvent: From<Event<Self>> + IsType<<Self as frame_system::Config>::RuntimeEvent>;
|
|
|
|
/// The Weight information for this pallet.
|
|
type WeightInfo: weights::WeightInfo;
|
|
|
|
/// Currency type for this pallet.
|
|
type Currency: ReservableCurrency<Self::AccountId>;
|
|
|
|
/// An origin that can request a preimage be placed on-chain without a deposit or fee, or
|
|
/// manage existing preimages.
|
|
type ManagerOrigin: EnsureOrigin<Self::RuntimeOrigin>;
|
|
|
|
/// The base deposit for placing a preimage on chain.
|
|
type BaseDeposit: Get<BalanceOf<Self>>;
|
|
|
|
/// The per-byte deposit for placing a preimage on chain.
|
|
type ByteDeposit: Get<BalanceOf<Self>>;
|
|
}
|
|
|
|
#[pallet::pallet]
|
|
#[pallet::generate_store(pub(super) trait Store)]
|
|
#[pallet::storage_version(STORAGE_VERSION)]
|
|
pub struct Pallet<T>(PhantomData<T>);
|
|
|
|
#[pallet::event]
|
|
#[pallet::generate_deposit(pub(super) fn deposit_event)]
|
|
pub enum Event<T: Config> {
|
|
/// A preimage has been noted.
|
|
Noted { hash: T::Hash },
|
|
/// A preimage has been requested.
|
|
Requested { hash: T::Hash },
|
|
/// A preimage has ben cleared.
|
|
Cleared { hash: T::Hash },
|
|
}
|
|
|
|
#[pallet::error]
|
|
pub enum Error<T> {
|
|
/// Preimage is too large to store on-chain.
|
|
TooBig,
|
|
/// Preimage has already been noted on-chain.
|
|
AlreadyNoted,
|
|
/// The user is not authorized to perform this action.
|
|
NotAuthorized,
|
|
/// The preimage cannot be removed since it has not yet been noted.
|
|
NotNoted,
|
|
/// A preimage may not be removed when there are outstanding requests.
|
|
Requested,
|
|
/// The preimage request cannot be removed since no outstanding requests exist.
|
|
NotRequested,
|
|
}
|
|
|
|
/// The request status of a given hash.
|
|
#[pallet::storage]
|
|
pub(super) type StatusFor<T: Config> =
|
|
StorageMap<_, Identity, T::Hash, RequestStatus<T::AccountId, BalanceOf<T>>>;
|
|
|
|
#[pallet::storage]
|
|
pub(super) type PreimageFor<T: Config> =
|
|
StorageMap<_, Identity, (T::Hash, u32), BoundedVec<u8, ConstU32<MAX_SIZE>>>;
|
|
|
|
#[pallet::call]
|
|
impl<T: Config> Pallet<T> {
|
|
/// Register a preimage on-chain.
|
|
///
|
|
/// If the preimage was previously requested, no fees or deposits are taken for providing
|
|
/// the preimage. Otherwise, a deposit is taken proportional to the size of the preimage.
|
|
#[pallet::weight(T::WeightInfo::note_preimage(bytes.len() as u32))]
|
|
pub fn note_preimage(origin: OriginFor<T>, bytes: Vec<u8>) -> DispatchResultWithPostInfo {
|
|
// We accept a signed origin which will pay a deposit, or a root origin where a deposit
|
|
// is not taken.
|
|
let maybe_sender = Self::ensure_signed_or_manager(origin)?;
|
|
let (system_requested, _) = Self::note_bytes(bytes.into(), maybe_sender.as_ref())?;
|
|
if system_requested || maybe_sender.is_none() {
|
|
Ok(Pays::No.into())
|
|
} else {
|
|
Ok(().into())
|
|
}
|
|
}
|
|
|
|
/// Clear an unrequested preimage from the runtime storage.
|
|
///
|
|
/// If `len` is provided, then it will be a much cheaper operation.
|
|
///
|
|
/// - `hash`: The hash of the preimage to be removed from the store.
|
|
/// - `len`: The length of the preimage of `hash`.
|
|
#[pallet::weight(T::WeightInfo::unnote_preimage())]
|
|
pub fn unnote_preimage(origin: OriginFor<T>, hash: T::Hash) -> DispatchResult {
|
|
let maybe_sender = Self::ensure_signed_or_manager(origin)?;
|
|
Self::do_unnote_preimage(&hash, maybe_sender)
|
|
}
|
|
|
|
/// Request a preimage be uploaded to the chain without paying any fees or deposits.
|
|
///
|
|
/// If the preimage requests has already been provided on-chain, we unreserve any deposit
|
|
/// a user may have paid, and take the control of the preimage out of their hands.
|
|
#[pallet::weight(T::WeightInfo::request_preimage())]
|
|
pub fn request_preimage(origin: OriginFor<T>, hash: T::Hash) -> DispatchResult {
|
|
T::ManagerOrigin::ensure_origin(origin)?;
|
|
Self::do_request_preimage(&hash);
|
|
Ok(())
|
|
}
|
|
|
|
/// Clear a previously made request for a preimage.
|
|
///
|
|
/// NOTE: THIS MUST NOT BE CALLED ON `hash` MORE TIMES THAN `request_preimage`.
|
|
#[pallet::weight(T::WeightInfo::unrequest_preimage())]
|
|
pub fn unrequest_preimage(origin: OriginFor<T>, hash: T::Hash) -> DispatchResult {
|
|
T::ManagerOrigin::ensure_origin(origin)?;
|
|
Self::do_unrequest_preimage(&hash)
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<T: Config> Pallet<T> {
|
|
/// Ensure that the origin is either the `ManagerOrigin` or a signed origin.
|
|
fn ensure_signed_or_manager(
|
|
origin: T::RuntimeOrigin,
|
|
) -> Result<Option<T::AccountId>, BadOrigin> {
|
|
if T::ManagerOrigin::ensure_origin(origin.clone()).is_ok() {
|
|
return Ok(None)
|
|
}
|
|
let who = ensure_signed(origin)?;
|
|
Ok(Some(who))
|
|
}
|
|
|
|
/// Store some preimage on chain.
|
|
///
|
|
/// If `maybe_depositor` is `None` then it is also requested. If `Some`, then it is not.
|
|
///
|
|
/// We verify that the preimage is within the bounds of what the pallet supports.
|
|
///
|
|
/// If the preimage was requested to be uploaded, then the user pays no deposits or tx fees.
|
|
fn note_bytes(
|
|
preimage: Cow<[u8]>,
|
|
maybe_depositor: Option<&T::AccountId>,
|
|
) -> Result<(bool, T::Hash), DispatchError> {
|
|
let hash = T::Hashing::hash(&preimage);
|
|
let len = preimage.len() as u32;
|
|
ensure!(len <= MAX_SIZE, Error::<T>::TooBig);
|
|
|
|
// We take a deposit only if there is a provided depositor and the preimage was not
|
|
// previously requested. This also allows the tx to pay no fee.
|
|
let status = match (StatusFor::<T>::get(hash), maybe_depositor) {
|
|
(Some(RequestStatus::Requested { count, deposit, .. }), _) =>
|
|
RequestStatus::Requested { count, deposit, len: Some(len) },
|
|
(Some(RequestStatus::Unrequested { .. }), Some(_)) =>
|
|
return Err(Error::<T>::AlreadyNoted.into()),
|
|
(Some(RequestStatus::Unrequested { len, deposit }), None) =>
|
|
RequestStatus::Requested { deposit: Some(deposit), count: 1, len: Some(len) },
|
|
(None, None) => RequestStatus::Requested { count: 1, len: Some(len), deposit: None },
|
|
(None, Some(depositor)) => {
|
|
let length = preimage.len() as u32;
|
|
let deposit = T::BaseDeposit::get()
|
|
.saturating_add(T::ByteDeposit::get().saturating_mul(length.into()));
|
|
T::Currency::reserve(depositor, deposit)?;
|
|
RequestStatus::Unrequested { deposit: (depositor.clone(), deposit), len }
|
|
},
|
|
};
|
|
let was_requested = matches!(status, RequestStatus::Requested { .. });
|
|
StatusFor::<T>::insert(hash, status);
|
|
|
|
let _ = Self::insert(&hash, preimage)
|
|
.defensive_proof("Unable to insert. Logic error in `note_bytes`?");
|
|
|
|
Self::deposit_event(Event::Noted { hash });
|
|
|
|
Ok((was_requested, hash))
|
|
}
|
|
|
|
// This function will add a hash to the list of requested preimages.
|
|
//
|
|
// If the preimage already exists before the request is made, the deposit for the preimage is
|
|
// returned to the user, and removed from their management.
|
|
fn do_request_preimage(hash: &T::Hash) {
|
|
let (count, len, deposit) =
|
|
StatusFor::<T>::get(hash).map_or((1, None, None), |x| match x {
|
|
RequestStatus::Requested { mut count, len, deposit } => {
|
|
count.saturating_inc();
|
|
(count, len, deposit)
|
|
},
|
|
RequestStatus::Unrequested { deposit, len } => (1, Some(len), Some(deposit)),
|
|
});
|
|
StatusFor::<T>::insert(hash, RequestStatus::Requested { count, len, deposit });
|
|
if count == 1 {
|
|
Self::deposit_event(Event::Requested { hash: *hash });
|
|
}
|
|
}
|
|
|
|
// Clear a preimage from the storage of the chain, returning any deposit that may be reserved.
|
|
//
|
|
// If `len` is provided, it will be a much cheaper operation.
|
|
//
|
|
// If `maybe_owner` is provided, we verify that it is the correct owner before clearing the
|
|
// data.
|
|
fn do_unnote_preimage(
|
|
hash: &T::Hash,
|
|
maybe_check_owner: Option<T::AccountId>,
|
|
) -> DispatchResult {
|
|
match StatusFor::<T>::get(hash).ok_or(Error::<T>::NotNoted)? {
|
|
RequestStatus::Requested { deposit: Some((owner, deposit)), count, len } => {
|
|
ensure!(maybe_check_owner.map_or(true, |c| c == owner), Error::<T>::NotAuthorized);
|
|
T::Currency::unreserve(&owner, deposit);
|
|
StatusFor::<T>::insert(
|
|
hash,
|
|
RequestStatus::Requested { deposit: None, count, len },
|
|
);
|
|
Ok(())
|
|
},
|
|
RequestStatus::Requested { deposit: None, .. } => {
|
|
ensure!(maybe_check_owner.is_none(), Error::<T>::NotAuthorized);
|
|
Self::do_unrequest_preimage(hash)
|
|
},
|
|
RequestStatus::Unrequested { deposit: (owner, deposit), len } => {
|
|
ensure!(maybe_check_owner.map_or(true, |c| c == owner), Error::<T>::NotAuthorized);
|
|
T::Currency::unreserve(&owner, deposit);
|
|
StatusFor::<T>::remove(hash);
|
|
|
|
Self::remove(hash, len);
|
|
Self::deposit_event(Event::Cleared { hash: *hash });
|
|
Ok(())
|
|
},
|
|
}
|
|
}
|
|
|
|
/// Clear a preimage request.
|
|
fn do_unrequest_preimage(hash: &T::Hash) -> DispatchResult {
|
|
match StatusFor::<T>::get(hash).ok_or(Error::<T>::NotRequested)? {
|
|
RequestStatus::Requested { mut count, len, deposit } if count > 1 => {
|
|
count.saturating_dec();
|
|
StatusFor::<T>::insert(hash, RequestStatus::Requested { count, len, deposit });
|
|
},
|
|
RequestStatus::Requested { count, len, deposit } => {
|
|
debug_assert!(count == 1, "preimage request counter at zero?");
|
|
match (len, deposit) {
|
|
// Preimage was never noted.
|
|
(None, _) => StatusFor::<T>::remove(hash),
|
|
// Preimage was noted without owner - just remove it.
|
|
(Some(len), None) => {
|
|
Self::remove(hash, len);
|
|
StatusFor::<T>::remove(hash);
|
|
Self::deposit_event(Event::Cleared { hash: *hash });
|
|
},
|
|
// Preimage was noted with owner - move to unrequested so they can get refund.
|
|
(Some(len), Some(deposit)) => {
|
|
StatusFor::<T>::insert(hash, RequestStatus::Unrequested { deposit, len });
|
|
},
|
|
}
|
|
},
|
|
RequestStatus::Unrequested { .. } => return Err(Error::<T>::NotRequested.into()),
|
|
}
|
|
Ok(())
|
|
}
|
|
|
|
fn insert(hash: &T::Hash, preimage: Cow<[u8]>) -> Result<(), ()> {
|
|
BoundedSlice::<u8, ConstU32<MAX_SIZE>>::try_from(preimage.as_ref())
|
|
.map(|s| PreimageFor::<T>::insert((hash, s.len() as u32), s))
|
|
}
|
|
|
|
fn remove(hash: &T::Hash, len: u32) {
|
|
PreimageFor::<T>::remove((hash, len))
|
|
}
|
|
|
|
fn have(hash: &T::Hash) -> bool {
|
|
Self::len(hash).is_some()
|
|
}
|
|
|
|
fn len(hash: &T::Hash) -> Option<u32> {
|
|
use RequestStatus::*;
|
|
match StatusFor::<T>::get(hash) {
|
|
Some(Requested { len: Some(len), .. }) | Some(Unrequested { len, .. }) => Some(len),
|
|
_ => None,
|
|
}
|
|
}
|
|
|
|
fn fetch(hash: &T::Hash, len: Option<u32>) -> FetchResult {
|
|
let len = len.or_else(|| Self::len(hash)).ok_or(DispatchError::Unavailable)?;
|
|
PreimageFor::<T>::get((hash, len))
|
|
.map(|p| p.into_inner())
|
|
.map(Into::into)
|
|
.ok_or(DispatchError::Unavailable)
|
|
}
|
|
}
|
|
|
|
impl<T: Config> PreimageProvider<T::Hash> for Pallet<T> {
|
|
fn have_preimage(hash: &T::Hash) -> bool {
|
|
Self::have(hash)
|
|
}
|
|
|
|
fn preimage_requested(hash: &T::Hash) -> bool {
|
|
matches!(StatusFor::<T>::get(hash), Some(RequestStatus::Requested { .. }))
|
|
}
|
|
|
|
fn get_preimage(hash: &T::Hash) -> Option<Vec<u8>> {
|
|
Self::fetch(hash, None).ok().map(Cow::into_owned)
|
|
}
|
|
|
|
fn request_preimage(hash: &T::Hash) {
|
|
Self::do_request_preimage(hash)
|
|
}
|
|
|
|
fn unrequest_preimage(hash: &T::Hash) {
|
|
let res = Self::do_unrequest_preimage(hash);
|
|
debug_assert!(res.is_ok(), "do_unrequest_preimage failed - counter underflow?");
|
|
}
|
|
}
|
|
|
|
impl<T: Config> PreimageRecipient<T::Hash> for Pallet<T> {
|
|
type MaxSize = ConstU32<MAX_SIZE>; // 2**22
|
|
|
|
fn note_preimage(bytes: BoundedVec<u8, Self::MaxSize>) {
|
|
// We don't really care if this fails, since that's only the case if someone else has
|
|
// already noted it.
|
|
let _ = Self::note_bytes(bytes.into_inner().into(), None);
|
|
}
|
|
|
|
fn unnote_preimage(hash: &T::Hash) {
|
|
// Should never fail if authorization check is skipped.
|
|
let res = Self::do_unrequest_preimage(hash);
|
|
debug_assert!(res.is_ok(), "unnote_preimage failed - request outstanding?");
|
|
}
|
|
}
|
|
|
|
impl<T: Config<Hash = PreimageHash>> QueryPreimage for Pallet<T> {
|
|
fn len(hash: &T::Hash) -> Option<u32> {
|
|
Pallet::<T>::len(hash)
|
|
}
|
|
|
|
fn fetch(hash: &T::Hash, len: Option<u32>) -> FetchResult {
|
|
Pallet::<T>::fetch(hash, len)
|
|
}
|
|
|
|
fn is_requested(hash: &T::Hash) -> bool {
|
|
matches!(StatusFor::<T>::get(hash), Some(RequestStatus::Requested { .. }))
|
|
}
|
|
|
|
fn request(hash: &T::Hash) {
|
|
Self::do_request_preimage(hash)
|
|
}
|
|
|
|
fn unrequest(hash: &T::Hash) {
|
|
let res = Self::do_unrequest_preimage(hash);
|
|
debug_assert!(res.is_ok(), "do_unrequest_preimage failed - counter underflow?");
|
|
}
|
|
}
|
|
|
|
impl<T: Config<Hash = PreimageHash>> StorePreimage for Pallet<T> {
|
|
const MAX_LENGTH: usize = MAX_SIZE as usize;
|
|
|
|
fn note(bytes: Cow<[u8]>) -> Result<T::Hash, DispatchError> {
|
|
// We don't really care if this fails, since that's only the case if someone else has
|
|
// already noted it.
|
|
let maybe_hash = Self::note_bytes(bytes, None).map(|(_, h)| h);
|
|
// Map to the correct trait error.
|
|
if maybe_hash == Err(DispatchError::from(Error::<T>::TooBig)) {
|
|
Err(DispatchError::Exhausted)
|
|
} else {
|
|
maybe_hash
|
|
}
|
|
}
|
|
|
|
fn unnote(hash: &T::Hash) {
|
|
// Should never fail if authorization check is skipped.
|
|
let res = Self::do_unnote_preimage(hash, None);
|
|
debug_assert!(res.is_ok(), "unnote_preimage failed - request outstanding?");
|
|
}
|
|
}
|