#![warn(missing_docs)]
use codec::{Decode, Encode};
use error::{Error, Result};
use sc_executor::WasmExecutor;
use sp_core::{
traits::{CallContext, CodeExecutor, FetchRuntimeCode, RuntimeCode},
OpaqueMetadata,
};
use sp_state_machine::BasicExternalities;
use sp_wasm_interface::HostFunctions;
use std::borrow::Cow;
pub mod error;
pub fn fetch_latest_metadata_from_code_blob<HF: HostFunctions>(
executor: &WasmExecutor<HF>,
code_bytes: Cow<[u8]>,
) -> Result<OpaqueMetadata> {
let runtime_caller = RuntimeCaller::new(executor, code_bytes);
let version_result = runtime_caller.call("Metadata_metadata_versions", ());
match version_result {
Ok(supported_versions) => {
let supported_versions = Vec::<u32>::decode(&mut supported_versions.as_slice())?;
let latest_stable = supported_versions
.into_iter()
.filter(|v| *v != u32::MAX)
.max()
.ok_or(Error::StableMetadataVersionNotFound)?;
let encoded = runtime_caller.call("Metadata_metadata_at_version", latest_stable)?;
Option::<OpaqueMetadata>::decode(&mut encoded.as_slice())?
.ok_or(Error::OpaqueMetadataNotFound)
},
Err(_) => {
let encoded = runtime_caller.call("Metadata_metadata", ())?;
Decode::decode(&mut encoded.as_slice()).map_err(Into::into)
},
}
}
struct BasicCodeFetcher<'a> {
code: Cow<'a, [u8]>,
hash: Vec<u8>,
}
impl<'a> FetchRuntimeCode for BasicCodeFetcher<'a> {
fn fetch_runtime_code(&self) -> Option<Cow<[u8]>> {
Some(self.code.as_ref().into())
}
}
impl<'a> BasicCodeFetcher<'a> {
fn new(code: Cow<'a, [u8]>) -> Self {
Self { hash: sp_crypto_hashing::blake2_256(&code).to_vec(), code }
}
fn runtime_code(&'a self) -> RuntimeCode<'a> {
RuntimeCode {
code_fetcher: self as &'a dyn FetchRuntimeCode,
heap_pages: None,
hash: self.hash.clone(),
}
}
}
pub struct RuntimeCaller<'a, 'b, HF: HostFunctions> {
executor: &'b WasmExecutor<HF>,
code_fetcher: BasicCodeFetcher<'a>,
}
impl<'a, 'b, HF: HostFunctions> RuntimeCaller<'a, 'b, HF> {
pub fn new(executor: &'b WasmExecutor<HF>, code_bytes: Cow<'a, [u8]>) -> Self {
Self { executor, code_fetcher: BasicCodeFetcher::new(code_bytes) }
}
pub fn call(&self, method: &str, data: impl Encode) -> Result<Vec<u8>> {
let mut ext = BasicExternalities::default();
self.executor
.call(
&mut ext,
&self.code_fetcher.runtime_code(),
method,
&data.encode(),
CallContext::Offchain,
)
.0
.map_err(Into::into)
}
}
#[cfg(test)]
mod tests {
use codec::Decode;
use sc_executor::WasmExecutor;
use sp_version::RuntimeVersion;
type ParachainHostFunctions = (
cumulus_primitives_proof_size_hostfunction::storage_proof_size::HostFunctions,
sp_io::SubstrateHostFunctions,
);
#[test]
fn test_fetch_latest_metadata_from_blob_fetches_metadata() {
let executor: WasmExecutor<ParachainHostFunctions> = WasmExecutor::builder().build();
let code_bytes = cumulus_test_runtime::WASM_BINARY
.expect("To run this test, build the wasm binary of cumulus-test-runtime")
.to_vec();
let metadata = subxt::Metadata::decode(
&mut (*super::fetch_latest_metadata_from_code_blob(&executor, code_bytes.into())
.unwrap())
.as_slice(),
)
.unwrap();
assert!(metadata.pallet_by_name("ParachainInfo").is_some());
}
#[test]
fn test_runtime_caller_can_call_into_runtime() {
let executor: WasmExecutor<ParachainHostFunctions> = WasmExecutor::builder().build();
let code_bytes = cumulus_test_runtime::WASM_BINARY
.expect("To run this test, build the wasm binary of cumulus-test-runtime")
.to_vec();
let runtime_caller = super::RuntimeCaller::new(&executor, code_bytes.into());
let runtime_version = runtime_caller
.call("Core_version", ())
.expect("Should be able to call runtime_version");
let _runtime_version: RuntimeVersion = Decode::decode(&mut runtime_version.as_slice())
.expect("Should be able to decode runtime version");
}
}