// This file is part of Substrate. // Copyright (C) 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. //! This pallet demonstrates the use of the `pallet::task` api for service work. #![cfg_attr(not(feature = "std"), no_std)] use frame_support::dispatch::DispatchResult; use frame_system::offchain::SendTransactionTypes; #[cfg(feature = "experimental")] use frame_system::offchain::SubmitTransaction; // Re-export pallet items so that they can be accessed from the crate namespace. pub use pallet::*; pub mod mock; pub mod tests; #[cfg(feature = "runtime-benchmarks")] mod benchmarking; pub mod weights; pub use weights::*; #[cfg(feature = "experimental")] const LOG_TARGET: &str = "pallet-example-tasks"; #[frame_support::pallet(dev_mode)] pub mod pallet { use super::*; use frame_support::pallet_prelude::*; use frame_system::pallet_prelude::*; #[pallet::error] pub enum Error { /// The referenced task was not found. NotFound, } #[pallet::tasks_experimental] impl Pallet { /// Add a pair of numbers into the totals and remove them. #[pallet::task_list(Numbers::::iter_keys())] #[pallet::task_condition(|i| Numbers::::contains_key(i))] #[pallet::task_weight(T::WeightInfo::add_number_into_total())] #[pallet::task_index(0)] pub fn add_number_into_total(i: u32) -> DispatchResult { let v = Numbers::::take(i).ok_or(Error::::NotFound)?; Total::::mutate(|(total_keys, total_values)| { *total_keys += i; *total_values += v; }); Ok(()) } } #[pallet::hooks] impl Hooks> for Pallet { #[cfg(feature = "experimental")] fn offchain_worker(_block_number: BlockNumberFor) { if let Some(key) = Numbers::::iter_keys().next() { // Create a valid task let task = Task::::AddNumberIntoTotal { i: key }; let runtime_task = ::RuntimeTask::from(task); let call = frame_system::Call::::do_task { task: runtime_task.into() }; // Submit the task as an unsigned transaction let res = SubmitTransaction::>::submit_unsigned_transaction( call.into(), ); match res { Ok(_) => log::info!(target: LOG_TARGET, "Submitted the task."), Err(e) => log::error!(target: LOG_TARGET, "Error submitting task: {:?}", e), } } } } #[pallet::config] pub trait Config: SendTransactionTypes> + frame_system::Config { type RuntimeTask: frame_support::traits::Task + IsType<::RuntimeTask> + From>; type WeightInfo: WeightInfo; } #[pallet::pallet] pub struct Pallet(_); /// Some running total. #[pallet::storage] pub type Total = StorageValue<_, (u32, u32), ValueQuery>; /// Numbers to be added into the total. #[pallet::storage] pub type Numbers = StorageMap<_, Twox64Concat, u32, u32, OptionQuery>; }