// Copyright 2017-2019 Parity Technologies (UK) Ltd. // This file is part of Substrate. // Substrate 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. // Substrate 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 Substrate. If not, see . //! Substrate state API. use std::{ collections::{BTreeMap, HashMap}, ops::Range, sync::Arc, }; use error_chain::bail; use log::{warn, trace}; use client::{self, Client, CallExecutor, BlockchainEvents, runtime_api::Metadata}; use jsonrpc_derive::rpc; use jsonrpc_pubsub::{typed::Subscriber, SubscriptionId}; use primitives::{H256, Blake2Hasher, Bytes}; use primitives::hexdisplay::HexDisplay; use primitives::storage::{self, StorageKey, StorageData, StorageChangeSet}; use crate::rpc::Result as RpcResult; use crate::rpc::futures::{stream, Future, Sink, Stream}; use runtime_primitives::generic::BlockId; use runtime_primitives::traits::{ Block as BlockT, Header, ProvideRuntimeApi, NumberFor, SaturatedConversion }; use runtime_version::RuntimeVersion; use state_machine::{self, ExecutionStrategy}; use crate::subscriptions::Subscriptions; mod error; #[cfg(test)] mod tests; use self::error::Result; /// Substrate state API #[rpc] pub trait StateApi { /// RPC Metadata type Metadata; /// Call a contract at a block's state. #[rpc(name = "state_call", alias("state_callAt"))] fn call(&self, name: String, bytes: Bytes, hash: Option) -> Result; /// Returns the keys with prefix, leave empty to get all the keys #[rpc(name = "state_getKeys")] fn storage_keys(&self, prefix: StorageKey, hash: Option) -> Result>; /// Returns a storage entry at a specific block's state. #[rpc(name = "state_getStorage", alias("state_getStorageAt"))] fn storage(&self, key: StorageKey, hash: Option) -> Result>; /// Returns the hash of a storage entry at a block's state. #[rpc(name = "state_getStorageHash", alias("state_getStorageHashAt"))] fn storage_hash(&self, key: StorageKey, hash: Option) -> Result>; /// Returns the size of a storage entry at a block's state. #[rpc(name = "state_getStorageSize", alias("state_getStorageSizeAt"))] fn storage_size(&self, key: StorageKey, hash: Option) -> Result>; /// Returns the keys with prefix from a child storage, leave empty to get all the keys #[rpc(name = "state_getChildKeys")] fn child_storage_keys( &self, child_storage_key: StorageKey, prefix: StorageKey, hash: Option ) -> Result>; /// Returns a child storage entry at a specific block's state. #[rpc(name = "state_getChildStorage")] fn child_storage( &self, child_storage_key: StorageKey, key: StorageKey, hash: Option ) -> Result>; /// Returns the hash of a child storage entry at a block's state. #[rpc(name = "state_getChildStorageHash")] fn child_storage_hash( &self, child_storage_key: StorageKey, key: StorageKey, hash: Option ) -> Result>; /// Returns the size of a child storage entry at a block's state. #[rpc(name = "state_getChildStorageSize")] fn child_storage_size( &self, child_storage_key: StorageKey, key: StorageKey, hash: Option ) -> Result>; /// Returns the runtime metadata as an opaque blob. #[rpc(name = "state_getMetadata")] fn metadata(&self, hash: Option) -> Result; /// Get the runtime version. #[rpc(name = "state_getRuntimeVersion", alias("chain_getRuntimeVersion"))] fn runtime_version(&self, hash: Option) -> Result; /// Query historical storage entries (by key) starting from a block given as the second parameter. /// /// NOTE This first returned result contains the initial state of storage for all keys. /// Subsequent values in the vector represent changes to the previous state (diffs). #[rpc(name = "state_queryStorage")] fn query_storage( &self, keys: Vec, block: Hash, hash: Option ) -> Result>>; /// New runtime version subscription #[pubsub( subscription = "state_runtimeVersion", subscribe, name = "state_subscribeRuntimeVersion", alias("chain_subscribeRuntimeVersion") )] fn subscribe_runtime_version(&self, metadata: Self::Metadata, subscriber: Subscriber); /// Unsubscribe from runtime version subscription #[pubsub( subscription = "state_runtimeVersion", unsubscribe, name = "state_unsubscribeRuntimeVersion", alias("chain_unsubscribeRuntimeVersion") )] fn unsubscribe_runtime_version(&self, metadata: Option, id: SubscriptionId) -> RpcResult; /// New storage subscription #[pubsub(subscription = "state_storage", subscribe, name = "state_subscribeStorage")] fn subscribe_storage(&self, metadata: Self::Metadata, subscriber: Subscriber>, keys: Option>); /// Unsubscribe from storage subscription #[pubsub(subscription = "state_storage", unsubscribe, name = "state_unsubscribeStorage")] fn unsubscribe_storage(&self, metadata: Option, id: SubscriptionId) -> RpcResult; } /// State API with subscriptions support. pub struct State { /// Substrate client. client: Arc>, /// Current subscriptions. subscriptions: Subscriptions, } /// Ranges to query in state_queryStorage. struct QueryStorageRange { /// Hashes of all the blocks in the range. pub hashes: Vec, /// Number of the first block in the range. pub first_number: NumberFor, /// Blocks subrange ([begin; end) indices within `hashes`) where we should read keys at /// each state to get changes. pub unfiltered_range: Range, /// Blocks subrange ([begin; end) indices within `hashes`) where we could pre-filter /// blocks-with-changes by using changes tries. pub filtered_range: Option>, } impl State where Block: BlockT, B: client::backend::Backend, E: CallExecutor, { /// Create new State API RPC handler. pub fn new(client: Arc>, subscriptions: Subscriptions) -> Self { Self { client, subscriptions, } } /// Splits the `query_storage` block range into 'filtered' and 'unfiltered' subranges. /// Blocks that contain changes within filtered subrange could be filtered using changes tries. /// Blocks that contain changes within unfiltered subrange must be filtered manually. fn split_query_storage_range( &self, from: Block::Hash, to: Option ) -> Result> { let to = self.unwrap_or_best(to)?; let from_hdr = self.client.header(&BlockId::hash(from))?; let to_hdr = self.client.header(&BlockId::hash(to))?; match (from_hdr, to_hdr) { (Some(ref from), Some(ref to)) if from.number() <= to.number() => { // check if we can get from `to` to `from` by going through parent_hashes. let from_number = *from.number(); let blocks = { let mut blocks = vec![to.hash()]; let mut last = to.clone(); while *last.number() > from_number { if let Some(hdr) = self.client.header(&BlockId::hash(*last.parent_hash()))? { blocks.push(hdr.hash()); last = hdr; } else { bail!(invalid_block_range( Some(from), Some(to), format!("Parent of {} ({}) not found", last.number(), last.hash()), )) } } if last.hash() != from.hash() { bail!(invalid_block_range( Some(from), Some(to), format!("Expected to reach `from`, got {} ({})", last.number(), last.hash()), )) } blocks.reverse(); blocks }; // check if we can filter blocks-with-changes from some (sub)range using changes tries let changes_trie_range = self.client.max_key_changes_range(from_number, BlockId::Hash(to.hash()))?; let filtered_range_begin = changes_trie_range.map(|(begin, _)| (begin - from_number).saturated_into::()); let (unfiltered_range, filtered_range) = split_range(blocks.len(), filtered_range_begin); Ok(QueryStorageRange { hashes: blocks, first_number: from_number, unfiltered_range, filtered_range, }) }, (from, to) => bail!( invalid_block_range(from.as_ref(), to.as_ref(), "Invalid range or unknown block".into()) ), } } /// Iterates through range.unfiltered_range and check each block for changes of keys' values. fn query_storage_unfiltered( &self, range: &QueryStorageRange, keys: &[StorageKey], changes: &mut Vec>, ) -> Result<()> { let mut last_state: HashMap<_, Option<_>> = Default::default(); for block in range.unfiltered_range.start..range.unfiltered_range.end { let block_hash = range.hashes[block].clone(); let mut block_changes = StorageChangeSet { block: block_hash.clone(), changes: Vec::new() }; let id = BlockId::hash(block_hash); for key in keys { let (has_changed, data) = { let curr_data = self.client.storage(&id, key)?; let prev_data = last_state.get(key).and_then(|x| x.as_ref()); (curr_data.as_ref() != prev_data, curr_data) }; if has_changed { block_changes.changes.push((key.clone(), data.clone())); } last_state.insert(key.clone(), data); } changes.push(block_changes); } Ok(()) } /// Iterates through all blocks that are changing keys within range.filtered_range and collects these changes. fn query_storage_filtered( &self, range: &QueryStorageRange, keys: &[StorageKey], changes: &mut Vec>, ) -> Result<()> { let (begin, end) = match range.filtered_range { Some(ref filtered_range) => ( range.first_number + filtered_range.start.saturated_into(), BlockId::Hash(range.hashes[filtered_range.end - 1].clone()) ), None => return Ok(()), }; let mut changes_map: BTreeMap, StorageChangeSet> = BTreeMap::new(); for key in keys { let mut last_block = None; for (block, _) in self.client.key_changes(begin, end, key)? { if last_block == Some(block) { continue; } let block_hash = range.hashes[(block - range.first_number).saturated_into::()].clone(); let id = BlockId::Hash(block_hash); let value_at_block = self.client.storage(&id, key)?; changes_map.entry(block) .or_insert_with(|| StorageChangeSet { block: block_hash, changes: Vec::new() }) .changes.push((key.clone(), value_at_block)); last_block = Some(block); } } if let Some(additional_capacity) = changes_map.len().checked_sub(changes.len()) { changes.reserve(additional_capacity); } changes.extend(changes_map.into_iter().map(|(_, cs)| cs)); Ok(()) } } impl State where Block: BlockT, B: client::backend::Backend, E: CallExecutor, { fn unwrap_or_best(&self, hash: Option) -> Result { crate::helpers::unwrap_or_else(|| Ok(self.client.info()?.chain.best_hash), hash) } } impl StateApi for State where Block: BlockT + 'static, B: client::backend::Backend + Send + Sync + 'static, E: CallExecutor + Send + Sync + 'static + Clone, RA: Send + Sync + 'static, Client: ProvideRuntimeApi, as ProvideRuntimeApi>::Api: Metadata { type Metadata = crate::metadata::Metadata; fn call(&self, method: String, data: Bytes, block: Option) -> Result { let block = self.unwrap_or_best(block)?; trace!(target: "rpc", "Calling runtime at {:?} for method {} ({})", block, method, HexDisplay::from(&data.0)); let return_data = self.client .executor() .call( &BlockId::Hash(block), &method, &data.0, ExecutionStrategy::NativeElseWasm, state_machine::NeverOffchainExt::new(), )?; Ok(Bytes(return_data)) } fn storage_keys(&self, key_prefix: StorageKey, block: Option) -> Result> { let block = self.unwrap_or_best(block)?; trace!(target: "rpc", "Querying storage keys at {:?}", block); Ok(self.client.storage_keys(&BlockId::Hash(block), &key_prefix)?) } fn storage(&self, key: StorageKey, block: Option) -> Result> { let block = self.unwrap_or_best(block)?; trace!(target: "rpc", "Querying storage at {:?} for key {}", block, HexDisplay::from(&key.0)); Ok(self.client.storage(&BlockId::Hash(block), &key)?) } fn storage_hash(&self, key: StorageKey, block: Option) -> Result> { use runtime_primitives::traits::{Hash, Header as HeaderT}; Ok(self.storage(key, block)?.map(|x| ::Hashing::hash(&x.0))) } fn storage_size(&self, key: StorageKey, block: Option) -> Result> { Ok(self.storage(key, block)?.map(|x| x.0.len() as u64)) } fn child_storage( &self, child_storage_key: StorageKey, key: StorageKey, block: Option ) -> Result> { let block = self.unwrap_or_best(block)?; trace!(target: "rpc", "Querying child storage at {:?} for key {}", block, HexDisplay::from(&key.0)); Ok(self.client.child_storage(&BlockId::Hash(block), &child_storage_key, &key)?) } fn child_storage_keys( &self, child_storage_key: StorageKey, key_prefix: StorageKey, block: Option ) -> Result> { let block = self.unwrap_or_best(block)?; trace!(target: "rpc", "Querying child storage keys at {:?}", block); Ok(self.client.child_storage_keys(&BlockId::Hash(block), &child_storage_key, &key_prefix)?) } fn child_storage_hash( &self, child_storage_key: StorageKey, key: StorageKey, block: Option ) -> Result> { use runtime_primitives::traits::{Hash, Header as HeaderT}; Ok( self.child_storage(child_storage_key, key, block)? .map(|x| ::Hashing::hash(&x.0)) ) } fn child_storage_size( &self, child_storage_key: StorageKey, key: StorageKey, block: Option ) -> Result> { Ok(self.child_storage(child_storage_key, key, block)?.map(|x| x.0.len() as u64)) } fn metadata(&self, block: Option) -> Result { let block = self.unwrap_or_best(block)?; self.client.runtime_api().metadata(&BlockId::Hash(block)).map(Into::into).map_err(Into::into) } fn query_storage( &self, keys: Vec, from: Block::Hash, to: Option ) -> Result>> { let range = self.split_query_storage_range(from, to)?; let mut changes = Vec::new(); self.query_storage_unfiltered(&range, &keys, &mut changes)?; self.query_storage_filtered(&range, &keys, &mut changes)?; Ok(changes) } fn subscribe_storage( &self, _meta: Self::Metadata, subscriber: Subscriber>, keys: Option> ) { let keys = Into::>>::into(keys); let stream = match self.client.storage_changes_notification_stream(keys.as_ref().map(|x| &**x)) { Ok(stream) => stream, Err(err) => { let _ = subscriber.reject(error::Error::from(err).into()); return; }, }; // initial values let initial = stream::iter_result(keys .map(|keys| { let block = self.client.info().map(|info| info.chain.best_hash).unwrap_or_default(); let changes = keys .into_iter() .map(|key| self.storage(key.clone(), Some(block.clone()).into()) .map(|val| (key.clone(), val)) .unwrap_or_else(|_| (key, None)) ) .collect(); vec![Ok(Ok(StorageChangeSet { block, changes }))] }).unwrap_or_default()); self.subscriptions.add(subscriber, |sink| { let stream = stream .map_err(|e| warn!("Error creating storage notification stream: {:?}", e)) .map(|(block, changes)| Ok(StorageChangeSet { block, changes: changes.iter().cloned().collect(), })); sink .sink_map_err(|e| warn!("Error sending notifications: {:?}", e)) .send_all(initial.chain(stream)) // we ignore the resulting Stream (if the first stream is over we are unsubscribed) .map(|_| ()) }) } fn unsubscribe_storage(&self, _meta: Option, id: SubscriptionId) -> RpcResult { Ok(self.subscriptions.cancel(id)) } fn runtime_version(&self, at: Option) -> Result { let at = self.unwrap_or_best(at)?; Ok(self.client.runtime_version_at(&BlockId::Hash(at))?) } fn subscribe_runtime_version(&self, _meta: Self::Metadata, subscriber: Subscriber) { let stream = match self.client.storage_changes_notification_stream(Some(&[StorageKey(storage::well_known_keys::CODE.to_vec())])) { Ok(stream) => stream, Err(err) => { let _ = subscriber.reject(error::Error::from(err).into()); return; } }; self.subscriptions.add(subscriber, |sink| { let version = self.runtime_version(None.into()) .map_err(Into::into); let client = self.client.clone(); let mut previous_version = version.clone(); let stream = stream .map_err(|e| warn!("Error creating storage notification stream: {:?}", e)) .filter_map(move |_| { let version = client.info().and_then(|info| { client.runtime_version_at(&BlockId::hash(info.chain.best_hash)) }) .map_err(error::Error::from) .map_err(Into::into); if previous_version != version { previous_version = version.clone(); Some(version) } else { None } }); sink .sink_map_err(|e| warn!("Error sending notifications: {:?}", e)) .send_all( stream::iter_result(vec![Ok(version)]) .chain(stream) ) // we ignore the resulting Stream (if the first stream is over we are unsubscribed) .map(|_| ()) }); } fn unsubscribe_runtime_version(&self, _meta: Option, id: SubscriptionId) -> RpcResult { Ok(self.subscriptions.cancel(id)) } } /// Splits passed range into two subranges where: /// - first range has at least one element in it; /// - second range (optionally) starts at given `middle` element. pub(crate) fn split_range(size: usize, middle: Option) -> (Range, Option>) { // check if we can filter blocks-with-changes from some (sub)range using changes tries let range2_begin = match middle { // some of required changes tries are pruned => use available tries Some(middle) if middle != 0 => Some(middle), // all required changes tries are available, but we still want values at first block // => do 'unfiltered' read for the first block and 'filtered' for the rest Some(_) if size > 1 => Some(1), // range contains single element => do not use changes tries Some(_) => None, // changes tries are not available => do 'unfiltered' read for the whole range None => None, }; let range1 = 0..range2_begin.unwrap_or(size); let range2 = range2_begin.map(|begin| begin..size); (range1, range2) } fn invalid_block_range(from: Option<&H>, to: Option<&H>, reason: String) -> error::ErrorKind { let to_string = |x: Option<&H>| match x { None => "unknown hash".into(), Some(h) => format!("{} ({})", h.number(), h.hash()), }; error::ErrorKind::InvalidBlockRange(to_string(from), to_string(to), reason) }