mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-07-02 09:27:24 +00:00
a534274c6f
* runtime-interface: Implement `register_only` functions The runtime interface supports versioning of functions. Currently, if you add a new function it will be used by the runtime automatically. This results in requiring all nodes of a network to upgrade before the runtime is upgraded, otherwise they will fail to instantiate the new runtime because of missing host functions. This pr introduces `register_only` functions. This can be used when a new runtime interface function should be introduced, but the actual usage can be deferred. This means that nodes will have the host function for this, but the runtime will still use the old version of the function when being compiled for wasm. However, when a runtime is enacted that uses the new host function, the "old nodes" will already have the host function and will continue to work. * Update primitives/runtime-interface/src/lib.rs Co-authored-by: cheme <emericchevalier.pro@gmail.com> * Update primitives/runtime-interface/proc-macro/src/utils.rs Co-authored-by: cheme <emericchevalier.pro@gmail.com> * FMT Co-authored-by: cheme <emericchevalier.pro@gmail.com>
304 lines
8.1 KiB
Rust
304 lines
8.1 KiB
Rust
// This file is part of Substrate.
|
|
|
|
// Copyright (C) 2019-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.
|
|
|
|
//! Tests for the runtime interface traits and proc macros.
|
|
|
|
#![cfg_attr(not(feature = "std"), no_std)]
|
|
|
|
use sp_runtime_interface::runtime_interface;
|
|
|
|
#[cfg(not(feature = "std"))]
|
|
use sp_std::{convert::TryFrom, mem, prelude::*};
|
|
|
|
use sp_core::{sr25519::Public, wasm_export_functions};
|
|
|
|
// Include the WASM binary
|
|
#[cfg(feature = "std")]
|
|
include!(concat!(env!("OUT_DIR"), "/wasm_binary.rs"));
|
|
|
|
/// Wasm binary unwrapped. If built with `SKIP_WASM_BUILD`, the function panics.
|
|
#[cfg(feature = "std")]
|
|
pub fn wasm_binary_unwrap() -> &'static [u8] {
|
|
WASM_BINARY.expect(
|
|
"Development wasm binary is not available. Testing is only \
|
|
supported with the flag disabled.",
|
|
)
|
|
}
|
|
|
|
/// Used in the `test_array_as_mutable_reference` test.
|
|
const TEST_ARRAY: [u8; 16] = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16];
|
|
|
|
#[runtime_interface]
|
|
pub trait TestApi {
|
|
/// Returns the input data as result.
|
|
fn return_input(data: Vec<u8>) -> Vec<u8> {
|
|
data
|
|
}
|
|
|
|
/// Returns 16kb data.
|
|
///
|
|
/// # Note
|
|
///
|
|
/// We return a `Vec<u32>` because this will use the code path that uses SCALE
|
|
/// to pass the data between native/wasm. (Vec<u8> is passed without encoding the
|
|
/// data)
|
|
fn return_16kb() -> Vec<u32> {
|
|
vec![0; 4 * 1024]
|
|
}
|
|
|
|
/// Set the storage at key with value.
|
|
fn set_storage(&mut self, key: &[u8], data: &[u8]) {
|
|
self.place_storage(key.to_vec(), Some(data.to_vec()));
|
|
}
|
|
|
|
/// Copy `hello` into the given mutable reference
|
|
fn return_value_into_mutable_reference(&self, data: &mut [u8]) {
|
|
let res = "hello";
|
|
data[..res.as_bytes().len()].copy_from_slice(res.as_bytes());
|
|
}
|
|
|
|
/// Returns the input data wrapped in an `Option` as result.
|
|
fn return_option_input(data: Vec<u8>) -> Option<Vec<u8>> {
|
|
Some(data)
|
|
}
|
|
|
|
/// Get an array as input and returns a subset of this array.
|
|
fn get_and_return_array(data: [u8; 34]) -> [u8; 16] {
|
|
let mut res = [0u8; 16];
|
|
res.copy_from_slice(&data[..16]);
|
|
res
|
|
}
|
|
|
|
/// Take and fill mutable array.
|
|
fn array_as_mutable_reference(data: &mut [u8; 16]) {
|
|
data.copy_from_slice(&TEST_ARRAY);
|
|
}
|
|
|
|
/// Returns the given public key as result.
|
|
fn return_input_public_key(key: Public) -> Public {
|
|
key
|
|
}
|
|
|
|
/// A function that is called with invalid utf8 data from the runtime.
|
|
///
|
|
/// This also checks that we accept `_` (wild card) argument names.
|
|
fn invalid_utf8_data(_: &str) {}
|
|
|
|
/// Overwrite the native implementation in wasm. The native implementation always returns
|
|
/// `false` and the replacement function will return always `true`.
|
|
fn overwrite_native_function_implementation() -> bool {
|
|
false
|
|
}
|
|
|
|
/// Gets an `u128` and returns this value
|
|
fn get_and_return_u128(val: u128) -> u128 {
|
|
val
|
|
}
|
|
|
|
/// Gets an `i128` and returns this value
|
|
fn get_and_return_i128(val: i128) -> i128 {
|
|
val
|
|
}
|
|
|
|
fn test_versionning(&self, data: u32) -> bool {
|
|
data == 42 || data == 50
|
|
}
|
|
|
|
#[version(2)]
|
|
fn test_versionning(&self, data: u32) -> bool {
|
|
data == 42
|
|
}
|
|
|
|
fn test_versionning_register_only(&self, data: u32) -> bool {
|
|
data == 80
|
|
}
|
|
|
|
#[version(2, register_only)]
|
|
fn test_versionning_register_only(&self, data: u32) -> bool {
|
|
data == 42
|
|
}
|
|
|
|
/// Returns the input values as tuple.
|
|
fn return_input_as_tuple(
|
|
a: Vec<u8>,
|
|
b: u32,
|
|
c: Option<Vec<u32>>,
|
|
d: u8,
|
|
) -> (Vec<u8>, u32, Option<Vec<u32>>, u8) {
|
|
(a, b, c, d)
|
|
}
|
|
}
|
|
|
|
/// This function is not used, but we require it for the compiler to include `sp-io`.
|
|
/// `sp-io` is required for its panic and oom handler.
|
|
#[no_mangle]
|
|
pub fn import_sp_io() {
|
|
sp_io::misc::print_utf8(&[]);
|
|
}
|
|
|
|
wasm_export_functions! {
|
|
fn test_return_data() {
|
|
let input = vec![1, 2, 3, 4, 5, 6];
|
|
let res = test_api::return_input(input.clone());
|
|
|
|
assert_eq!(input, res);
|
|
}
|
|
|
|
fn test_return_option_data() {
|
|
let input = vec![1, 2, 3, 4, 5, 6];
|
|
let res = test_api::return_option_input(input.clone());
|
|
|
|
assert_eq!(Some(input), res);
|
|
}
|
|
|
|
fn test_set_storage() {
|
|
let key = "hello";
|
|
let value = "world";
|
|
|
|
test_api::set_storage(key.as_bytes(), value.as_bytes());
|
|
}
|
|
|
|
fn test_return_value_into_mutable_reference() {
|
|
let mut data = vec![1, 2, 3, 4, 5, 6];
|
|
|
|
test_api::return_value_into_mutable_reference(&mut data);
|
|
|
|
let expected = "hello";
|
|
assert_eq!(expected.as_bytes(), &data[..expected.len()]);
|
|
}
|
|
|
|
fn test_get_and_return_array() {
|
|
let mut input = unsafe { mem::MaybeUninit::<[u8; 34]>::zeroed().assume_init() };
|
|
input.copy_from_slice(&[
|
|
24, 3, 23, 20, 2, 16, 32, 1, 12, 26, 27, 8, 29, 31, 6, 5, 4, 19, 10, 28, 34, 21, 18, 33, 9,
|
|
13, 22, 25, 15, 11, 30, 7, 14, 17,
|
|
]);
|
|
|
|
let res = test_api::get_and_return_array(input);
|
|
|
|
assert_eq!(&res, &input[..16]);
|
|
}
|
|
|
|
fn test_array_as_mutable_reference() {
|
|
let mut array = [0u8; 16];
|
|
test_api::array_as_mutable_reference(&mut array);
|
|
|
|
assert_eq!(array, TEST_ARRAY);
|
|
}
|
|
|
|
fn test_return_input_public_key() {
|
|
let key = Public::try_from(
|
|
&[
|
|
1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24,
|
|
25, 26, 27, 28, 29, 30, 31, 32,
|
|
][..],
|
|
).unwrap();
|
|
let ret_key = test_api::return_input_public_key(key.clone());
|
|
|
|
let key_data: &[u8] = key.as_ref();
|
|
let ret_key_data: &[u8] = ret_key.as_ref();
|
|
assert_eq!(key_data, ret_key_data);
|
|
}
|
|
|
|
fn test_invalid_utf8_data_should_return_an_error() {
|
|
let data = vec![0, 159, 146, 150];
|
|
// I'm an evil hacker, trying to hack!
|
|
let data_str = unsafe { sp_std::str::from_utf8_unchecked(&data) };
|
|
|
|
test_api::invalid_utf8_data(data_str);
|
|
}
|
|
|
|
fn test_overwrite_native_function_implementation() {
|
|
fn new_implementation() -> bool {
|
|
true
|
|
}
|
|
|
|
// Check native implementation
|
|
assert!(!test_api::overwrite_native_function_implementation());
|
|
|
|
let _guard = test_api::host_overwrite_native_function_implementation
|
|
.replace_implementation(new_implementation);
|
|
|
|
assert!(test_api::overwrite_native_function_implementation());
|
|
}
|
|
|
|
fn test_u128_i128_as_parameter_and_return_value() {
|
|
for val in &[u128::MAX, 1u128, 5000u128, u64::MAX as u128] {
|
|
assert_eq!(*val, test_api::get_and_return_u128(*val));
|
|
}
|
|
|
|
for val in &[i128::MAX, i128::MIN, 1i128, 5000i128, u64::MAX as i128] {
|
|
assert_eq!(*val, test_api::get_and_return_i128(*val));
|
|
}
|
|
}
|
|
|
|
fn test_vec_return_value_memory_is_freed() {
|
|
let mut len = 0;
|
|
for _ in 0..1024 {
|
|
len += test_api::return_16kb().len();
|
|
}
|
|
assert_eq!(1024 * 1024 * 4, len);
|
|
}
|
|
|
|
fn test_encoded_return_value_memory_is_freed() {
|
|
let mut len = 0;
|
|
for _ in 0..1024 {
|
|
len += test_api::return_option_input(vec![0; 16 * 1024]).map(|v| v.len()).unwrap();
|
|
}
|
|
assert_eq!(1024 * 1024 * 16, len);
|
|
}
|
|
|
|
fn test_array_return_value_memory_is_freed() {
|
|
let mut len = 0;
|
|
for _ in 0..1024 * 1024 {
|
|
len += test_api::get_and_return_array([0; 34])[1];
|
|
}
|
|
assert_eq!(0, len);
|
|
}
|
|
|
|
fn test_versionning_works() {
|
|
// we fix new api to accept only 42 as a proper input
|
|
// as opposed to sp-runtime-interface-test-wasm-deprecated::test_api::verify_input
|
|
// which accepted 42 and 50.
|
|
assert!(test_api::test_versionning(42));
|
|
|
|
assert!(!test_api::test_versionning(50));
|
|
assert!(!test_api::test_versionning(102));
|
|
}
|
|
|
|
fn test_versionning_register_only_works() {
|
|
// Ensure that we will import the version of the runtime interface function that
|
|
// isn't tagged with `register_only`.
|
|
assert!(!test_api::test_versionning_register_only(42));
|
|
assert!(test_api::test_versionning_register_only(80));
|
|
}
|
|
|
|
fn test_return_input_as_tuple() {
|
|
let a = vec![1, 3, 4, 5];
|
|
let b = 10000;
|
|
let c = Some(vec![2, 3]);
|
|
let d = 5;
|
|
|
|
let res = test_api::return_input_as_tuple(a.clone(), b, c.clone(), d);
|
|
|
|
assert_eq!(a, res.0);
|
|
assert_eq!(b, res.1);
|
|
assert_eq!(c, res.2);
|
|
assert_eq!(d, res.3);
|
|
}
|
|
}
|