// This file is part of Substrate. // Copyright (C) Parity Technologies (UK) Ltd. // SPDX-License-Identifier: GPL-3.0-or-later WITH Classpath-exception-2.0 // This program 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. // This program 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 this program. If not, see . //! Execution extensions for runtime calls. //! //! This module is responsible for defining the execution //! strategy for the runtime calls and provide the right `Externalities` //! extensions to support APIs for particular execution context & capabilities. use parking_lot::RwLock; use sp_core::traits::{ReadRuntimeVersion, ReadRuntimeVersionExt}; use sp_externalities::{Extension, Extensions}; use sp_runtime::traits::{Block as BlockT, NumberFor}; use std::{marker::PhantomData, sync::Arc}; /// Generate the starting set of [`Extensions`]. /// /// These [`Extensions`] are passed to the environment a runtime is executed in. pub trait ExtensionsFactory: Send + Sync { /// Create [`Extensions`] for the given input. /// /// - `block_hash`: The hash of the block in the context that extensions will be used. /// - `block_number`: The number of the block in the context that extensions will be used. fn extensions_for(&self, block_hash: Block::Hash, block_number: NumberFor) -> Extensions; } impl ExtensionsFactory for () { fn extensions_for(&self, _: Block::Hash, _: NumberFor) -> Extensions { Extensions::new() } } impl> ExtensionsFactory for Vec { fn extensions_for( &self, block_hash: Block::Hash, block_number: NumberFor, ) -> Extensions { let mut exts = Extensions::new(); exts.extend(self.iter().map(|e| e.extensions_for(block_hash, block_number))); exts } } /// An [`ExtensionsFactory`] that registers an [`Extension`] before a certain block. pub struct ExtensionBeforeBlock { before: NumberFor, _marker: PhantomData Ext>, } impl ExtensionBeforeBlock { /// Create the extension factory. /// /// - `before`: The block number until the extension should be registered. pub fn new(before: NumberFor) -> Self { Self { before, _marker: PhantomData } } } impl ExtensionsFactory for ExtensionBeforeBlock { fn extensions_for(&self, _: Block::Hash, block_number: NumberFor) -> Extensions { let mut exts = Extensions::new(); if block_number < self.before { exts.register(Ext::default()); } exts } } /// A producer of execution extensions for offchain calls. /// /// This crate aggregates extensions available for the offchain calls /// and is responsible for producing a correct `Extensions` object. pub struct ExecutionExtensions { extensions_factory: RwLock>>, read_runtime_version: Arc, } impl ExecutionExtensions { /// Create new `ExecutionExtensions` given an `extensions_factory`. pub fn new( extensions_factory: Option>>, read_runtime_version: Arc, ) -> Self { Self { extensions_factory: extensions_factory .map(RwLock::new) .unwrap_or_else(|| RwLock::new(Box::new(()))), read_runtime_version, } } /// Set the new extensions_factory pub fn set_extensions_factory(&self, maker: impl ExtensionsFactory + 'static) { *self.extensions_factory.write() = Box::new(maker); } /// Produces default extensions based on the input parameters. pub fn extensions( &self, block_hash: Block::Hash, block_number: NumberFor, ) -> Extensions { let mut extensions = self.extensions_factory.read().extensions_for(block_hash, block_number); extensions.register(ReadRuntimeVersionExt::new(self.read_runtime_version.clone())); extensions } }