mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-29 23:07:57 +00:00
b581604aa7
* Apply some clippy hints * Revert clippy ci changes * Update client/cli/src/commands/generate.rs Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com> * Update client/cli/src/commands/inspect_key.rs Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com> * Update client/db/src/bench.rs Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com> * Update client/db/src/bench.rs Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com> * Update client/service/src/client/block_rules.rs Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com> * Update client/service/src/client/block_rules.rs Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com> * Update client/network/src/transactions.rs Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com> * Update client/network/src/protocol.rs Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com> * Revert due to missing `or_default` function. * Fix compilation and simplify code * Undo change that corrupts benchmark. * fix clippy * Update client/service/test/src/lib.rs Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com> * Update client/state-db/src/noncanonical.rs Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com> * Update client/state-db/src/noncanonical.rs remove leftovers! * Update client/tracing/src/logging/directives.rs Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com> * Update utils/fork-tree/src/lib.rs Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com> * added needed ref * Update frame/referenda/src/benchmarking.rs * Simplify byte-vec creation * let's just not overlap the ranges * Correction * cargo fmt * Update utils/frame/benchmarking-cli/src/shared/stats.rs Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com> * Update utils/frame/benchmarking-cli/src/pallet/command.rs Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com> * Update utils/frame/benchmarking-cli/src/pallet/command.rs Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com> Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com> Co-authored-by: Giles Cope <gilescope@gmail.com>
205 lines
5.6 KiB
Rust
205 lines
5.6 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.
|
|
|
|
//! Externalities extensions storage.
|
|
//!
|
|
//! Externalities support to register a wide variety custom extensions. The [`Extensions`] provides
|
|
//! some convenience functionality to store and retrieve these extensions.
|
|
//!
|
|
//! It is required that each extension implements the [`Extension`] trait.
|
|
|
|
use crate::Error;
|
|
use sp_std::{
|
|
any::{Any, TypeId},
|
|
boxed::Box,
|
|
collections::btree_map::{BTreeMap, Entry},
|
|
ops::DerefMut,
|
|
};
|
|
|
|
/// Marker trait for types that should be registered as [`Externalities`](crate::Externalities)
|
|
/// extension.
|
|
///
|
|
/// As extensions are stored as `Box<Any>`, this trait should give more confidence that the correct
|
|
/// type is registered and requested.
|
|
pub trait Extension: Send + Any {
|
|
/// Return the extension as `&mut dyn Any`.
|
|
///
|
|
/// This is a trick to make the trait type castable into an `Any`.
|
|
fn as_mut_any(&mut self) -> &mut dyn Any;
|
|
}
|
|
|
|
/// Macro for declaring an extension that usable with [`Extensions`].
|
|
///
|
|
/// The extension will be an unit wrapper struct that implements [`Extension`], `Deref` and
|
|
/// `DerefMut`. The wrapped type is given by the user.
|
|
///
|
|
/// # Example
|
|
/// ```
|
|
/// # use sp_externalities::decl_extension;
|
|
/// decl_extension! {
|
|
/// /// Some test extension
|
|
/// struct TestExt(String);
|
|
/// }
|
|
/// ```
|
|
#[macro_export]
|
|
macro_rules! decl_extension {
|
|
(
|
|
$( #[ $attr:meta ] )*
|
|
$vis:vis struct $ext_name:ident ($inner:ty);
|
|
) => {
|
|
$( #[ $attr ] )*
|
|
$vis struct $ext_name (pub $inner);
|
|
|
|
impl $crate::Extension for $ext_name {
|
|
fn as_mut_any(&mut self) -> &mut dyn std::any::Any {
|
|
self
|
|
}
|
|
}
|
|
|
|
impl std::ops::Deref for $ext_name {
|
|
type Target = $inner;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
&self.0
|
|
}
|
|
}
|
|
|
|
impl std::ops::DerefMut for $ext_name {
|
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
|
&mut self.0
|
|
}
|
|
}
|
|
|
|
impl From<$inner> for $ext_name {
|
|
fn from(inner: $inner) -> Self {
|
|
Self(inner)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Something that provides access to the [`Extensions`] store.
|
|
///
|
|
/// This is a super trait of the [`Externalities`](crate::Externalities).
|
|
pub trait ExtensionStore {
|
|
/// Tries to find a registered extension by the given `type_id` and returns it as a `&mut dyn
|
|
/// Any`.
|
|
///
|
|
/// It is advised to use [`ExternalitiesExt::extension`](crate::ExternalitiesExt::extension)
|
|
/// instead of this function to get type system support and automatic type downcasting.
|
|
fn extension_by_type_id(&mut self, type_id: TypeId) -> Option<&mut dyn Any>;
|
|
|
|
/// Register extension `extension` with specified `type_id`.
|
|
///
|
|
/// It should return error if extension is already registered.
|
|
fn register_extension_with_type_id(
|
|
&mut self,
|
|
type_id: TypeId,
|
|
extension: Box<dyn Extension>,
|
|
) -> Result<(), Error>;
|
|
|
|
/// Deregister extension with speicifed 'type_id' and drop it.
|
|
///
|
|
/// It should return error if extension is not registered.
|
|
fn deregister_extension_by_type_id(&mut self, type_id: TypeId) -> Result<(), Error>;
|
|
}
|
|
|
|
/// Stores extensions that should be made available through the externalities.
|
|
#[derive(Default)]
|
|
pub struct Extensions {
|
|
extensions: BTreeMap<TypeId, Box<dyn Extension>>,
|
|
}
|
|
|
|
#[cfg(feature = "std")]
|
|
impl std::fmt::Debug for Extensions {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
write!(f, "Extensions: ({})", self.extensions.len())
|
|
}
|
|
}
|
|
|
|
impl Extensions {
|
|
/// Create new instance of `Self`.
|
|
pub fn new() -> Self {
|
|
Self::default()
|
|
}
|
|
|
|
/// Register the given extension.
|
|
pub fn register<E: Extension>(&mut self, ext: E) {
|
|
let type_id = ext.type_id();
|
|
self.extensions.insert(type_id, Box::new(ext));
|
|
}
|
|
|
|
/// Register extension `extension` using the given `type_id`.
|
|
pub fn register_with_type_id(
|
|
&mut self,
|
|
type_id: TypeId,
|
|
extension: Box<dyn Extension>,
|
|
) -> Result<(), Error> {
|
|
match self.extensions.entry(type_id) {
|
|
Entry::Vacant(vacant) => {
|
|
vacant.insert(extension);
|
|
Ok(())
|
|
},
|
|
Entry::Occupied(_) => Err(Error::ExtensionAlreadyRegistered),
|
|
}
|
|
}
|
|
|
|
/// Return a mutable reference to the requested extension.
|
|
pub fn get_mut(&mut self, ext_type_id: TypeId) -> Option<&mut dyn Any> {
|
|
self.extensions
|
|
.get_mut(&ext_type_id)
|
|
.map(DerefMut::deref_mut)
|
|
.map(Extension::as_mut_any)
|
|
}
|
|
|
|
/// Deregister extension for the given `type_id`.
|
|
///
|
|
/// Returns `true` when the extension was registered.
|
|
pub fn deregister(&mut self, type_id: TypeId) -> bool {
|
|
self.extensions.remove(&type_id).is_some()
|
|
}
|
|
|
|
/// Returns a mutable iterator over all extensions.
|
|
pub fn iter_mut(&mut self) -> impl Iterator<Item = (&TypeId, &mut Box<dyn Extension>)> {
|
|
self.extensions.iter_mut()
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
|
|
decl_extension! {
|
|
struct DummyExt(u32);
|
|
}
|
|
decl_extension! {
|
|
struct DummyExt2(u32);
|
|
}
|
|
|
|
#[test]
|
|
fn register_and_retrieve_extension() {
|
|
let mut exts = Extensions::new();
|
|
exts.register(DummyExt(1));
|
|
exts.register(DummyExt2(2));
|
|
|
|
let ext = exts.get_mut(TypeId::of::<DummyExt>()).expect("Extension is registered");
|
|
let ext_ty = ext.downcast_mut::<DummyExt>().expect("Downcasting works");
|
|
|
|
assert_eq!(ext_ty.0, 1);
|
|
}
|
|
}
|