Newer
Older
// Copyright 2018-2021 Parity Technologies (UK) Ltd.
//
// 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.
use super::{
hashing,
Account,
EnvInstance,
};
hash::{
Blake2x128,
Blake2x256,
CryptoHash,
HashOutput,
Keccak256,
Sha2x256,
},
EnvBackend,
Environment,
Error,
impl EnvInstance {
/// Returns the callee account.
fn callee_account(&self) -> &Account {
let callee = self
.exec_context()
.expect("uninitialized execution context")
.callee
.clone();
self.accounts
.get_account_off(&callee)
.expect("callee account does not exist")
}
/// Returns the callee account as mutable reference.
fn callee_account_mut(&mut self) -> &mut Account {
let callee = self
.exec_context()
.expect("uninitialized execution context")
.callee
.clone();
self.accounts
.get_account_off_mut(&callee)
.expect("callee account does not exist")
}
}
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
impl CryptoHash for Blake2x128 {
fn hash(input: &[u8], output: &mut <Self as HashOutput>::Type) {
type OutputType = [u8; 16];
static_assertions::assert_type_eq_all!(
<Blake2x128 as HashOutput>::Type,
OutputType
);
let output: &mut OutputType = arrayref::array_mut_ref!(output, 0, 16);
hashing::blake2b_128(input, output);
}
}
impl CryptoHash for Blake2x256 {
fn hash(input: &[u8], output: &mut <Self as HashOutput>::Type) {
type OutputType = [u8; 32];
static_assertions::assert_type_eq_all!(
<Blake2x256 as HashOutput>::Type,
OutputType
);
let output: &mut OutputType = arrayref::array_mut_ref!(output, 0, 32);
hashing::blake2b_256(input, output);
}
}
impl CryptoHash for Sha2x256 {
fn hash(input: &[u8], output: &mut <Self as HashOutput>::Type) {
type OutputType = [u8; 32];
static_assertions::assert_type_eq_all!(
<Sha2x256 as HashOutput>::Type,
OutputType
);
let output: &mut OutputType = arrayref::array_mut_ref!(output, 0, 32);
hashing::sha2_256(input, output);
}
}
impl CryptoHash for Keccak256 {
fn hash(input: &[u8], output: &mut <Self as HashOutput>::Type) {
type OutputType = [u8; 32];
static_assertions::assert_type_eq_all!(
<Keccak256 as HashOutput>::Type,
OutputType
);
let output: &mut OutputType = arrayref::array_mut_ref!(output, 0, 32);
hashing::keccak_256(input, output);
}
}
impl EnvBackend for EnvInstance {
fn set_contract_storage<V>(&mut self, key: &Key, value: &V)
where
V: scale::Encode,
{
self.callee_account_mut()
.set_storage(*key, value)
.expect("callee account is not a smart contract");
}
fn get_contract_storage<R>(&mut self, key: &Key) -> Result<Option<R>>
where
R: scale::Decode,
{
self.callee_account()
.get_storage::<R>(*key)
fn clear_contract_storage(&mut self, key: &Key) {
if !self.clear_storage_disabled {
self.callee_account_mut()
.clear_storage(*key)
.expect("callee account is not a smart contract");
}
fn decode_input<T>(&mut self) -> Result<T>
where
T: scale::Decode,
{
self.exec_context()
.map(|exec_ctx| &exec_ctx.call_data)
.map(|call_data| scale::Encode::encode(call_data))
.and_then(|encoded| {
<T as scale::Decode>::decode(&mut &encoded[..])
.map_err(|_| scale::Error::from("could not decode input call data"))
.map_err(Into::into)
})
fn return_value<R>(&mut self, flags: ReturnFlags, return_value: &R) -> !
where
R: scale::Encode,
{
let ctx = self
.exec_context_mut()
.expect("uninitialized execution context");
ctx.output = Some(return_value.encode());
}
fn println(&mut self, content: &str) {
self.console.println(content)
}
fn hash_bytes<H>(&mut self, input: &[u8], output: &mut <H as HashOutput>::Type)
where
H: CryptoHash,
{
<H as CryptoHash>::hash(input, output)
fn hash_encoded<H, T>(&mut self, input: &T, output: &mut <H as HashOutput>::Type)
where
H: CryptoHash,
T: scale::Encode,
{
let encoded = input.encode();
self.hash_bytes::<H>(&encoded[..], output)
fn call_chain_extension<I, T, E, ErrorCode, F, D>(
&mut self,
func_id: u32,
input: &I,
status_to_result: F,
decode_to_result: D,
) -> ::core::result::Result<T, E>
I: scale::Encode,
T: scale::Decode,
E: From<ErrorCode>,
F: FnOnce(u32) -> ::core::result::Result<(), ErrorCode>,
D: FnOnce(&[u8]) -> ::core::result::Result<T, E>,
let (status_code, mut output) = self
.chain_extension_handler
.eval(func_id, &encoded_input)
.expect("encountered unexpected missing chain extension method");
status_to_result(status_code)?;
let decoded = decode_to_result(&mut output)?;
destination: &T::AccountId,
{
let src_id = self.account_id::<T>()?;
let src_value = self
.accounts
.get_account::<T>(&src_id)
.expect("account of executed contract must exist")
.balance::<T>()?;
if src_value < value {
return Err(Error::TransferFailed)
.get_or_create_account::<T>(destination)
.balance::<T>()?;
self.accounts
.get_account_mut::<T>(&src_id)
.expect("account of executed contract must exist")
.set_balance::<T>(src_value - value)?;
self.accounts
.get_account_mut::<T>(destination)
.expect("the account must exist already or has just been created")
.set_balance::<T>(dst_value + value)?;
Ok(())
}
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
// Remove the calling account and transfer remaining balance.
//
// This function never returns. Either the termination was successful and the
// execution of the destroyed contract is halted. Or it failed during the termination
// which is considered fatal.
fn terminate_contract_impl<T>(&mut self, beneficiary: T::AccountId) -> !
where
T: Environment,
{
// Send the remaining balance to the beneficiary
let all: T::Balance = self.balance::<T>().expect("could not decode balance");
self.transfer_impl::<T>(&beneficiary, all)
.expect("transfer did not work ");
// Remove account
let contract_id = self.account_id::<T>().expect("could not decode account id");
self.accounts.remove_account::<T>(contract_id);
// The on-chain implementation would set a tombstone with a code hash here
// and remove the contract storage subsequently. Both is not easily achievable
// with our current off-chain env, hence we left it out here for the moment.
// Encode the result of the termination and panic with it.
// This enables testing for the proper result and makes sure this
// method returns `Never`.
let res = crate::test::ContractTerminationResult::<T> {
beneficiary,
transferred: all,
};
panic!("{:?}", scale::Encode::encode(&res));
impl TypedEnvBackend for EnvInstance {
fn caller<T: Environment>(&mut self) -> Result<T::AccountId> {
self.exec_context()
.expect("uninitialized execution context")
.caller::<T>()
.map_err(|_| scale::Error::from("could not decode caller"))
.map_err(Into::into)
}
fn transferred_balance<T: Environment>(&mut self) -> Result<T::Balance> {
self.exec_context()
.expect("uninitialized execution context")
.transferred_value::<T>()
.map_err(|_| scale::Error::from("could not decode transferred balance"))
.map_err(Into::into)
}
/// Emulates gas price calculation
fn weight_to_fee<T: Environment>(&mut self, gas: u64) -> Result<T::Balance> {
use crate::arithmetic::Saturating as _;
.map_err(|_| scale::Error::from("could not decode gas price"))?;
Ok(gas_price
.saturating_mul(gas.try_into().unwrap_or_else(|_| Bounded::max_value())))
fn gas_left<T: Environment>(&mut self) -> Result<T::Balance> {
self.exec_context()
.expect("uninitialized execution context")
.gas::<T>()
.map_err(|_| scale::Error::from("could not decode gas left"))
.map_err(Into::into)
}
fn block_timestamp<T: Environment>(&mut self) -> Result<T::Timestamp> {
self.current_block()
.expect("uninitialized execution context")
.timestamp::<T>()
.map_err(|_| scale::Error::from("could not decode block time"))
.map_err(Into::into)
}
fn account_id<T: Environment>(&mut self) -> Result<T::AccountId> {
self.exec_context()
.expect("uninitialized execution context")
.callee::<T>()
.map_err(|_| scale::Error::from("could not decode callee"))
.map_err(Into::into)
}
fn balance<T: Environment>(&mut self) -> Result<T::Balance> {
self.callee_account()
.balance::<T>()
.map_err(|_| scale::Error::from("could not decode callee balance"))
.map_err(Into::into)
}
fn rent_allowance<T: Environment>(&mut self) -> Result<T::Balance> {
self.callee_account()
.rent_allowance::<T>()
.map_err(|_| scale::Error::from("could not decode callee rent allowance"))
.map_err(Into::into)
}
fn block_number<T: Environment>(&mut self) -> Result<T::BlockNumber> {
self.current_block()
.expect("uninitialized execution context")
.number::<T>()
.map_err(|_| scale::Error::from("could not decode block number"))
.map_err(Into::into)
}
fn minimum_balance<T: Environment>(&mut self) -> Result<T::Balance> {
self.chain_spec
.minimum_balance::<T>()
.map_err(|_| scale::Error::from("could not decode minimum balance"))
.map_err(Into::into)
}
fn tombstone_deposit<T: Environment>(&mut self) -> Result<T::Balance> {
self.chain_spec
.tombstone_deposit::<T>()
.map_err(|_| scale::Error::from("could not decode tombstone deposit"))
.map_err(Into::into)
}
fn emit_event<T, Event>(&mut self, new_event: Event)
where
{
self.emitted_events.record::<T, Event>(new_event)
}
fn set_rent_allowance<T>(&mut self, new_rent_allowance: T::Balance)
where
{
self.callee_account_mut()
.set_rent_allowance::<T>(new_rent_allowance)
.expect("could not encode rent allowance")
}
Hero Bird
committed
fn invoke_contract<T, Args>(
&mut self,
_call_params: &CallParams<T, Args, ()>,
) -> Result<()>
Hero Bird
committed
Args: scale::Encode,
unimplemented!("off-chain environment does not support contract invocation")
Hero Bird
committed
fn eval_contract<T, Args, R>(
Hero Bird
committed
_call_params: &CallParams<T, Args, ReturnType<R>>,
Hero Bird
committed
Args: scale::Encode,
R: scale::Decode,
{
unimplemented!("off-chain environment does not support contract evaluation")
}
fn instantiate_contract<T, Args, Salt, C>(
_params: &CreateParams<T, Args, Salt, C>,
) -> Result<T::AccountId>
where
Hero Bird
committed
Args: scale::Encode,
{
unimplemented!("off-chain environment does not support contract instantiation")
}
fn terminate_contract<T>(&mut self, beneficiary: T::AccountId) -> !
self.terminate_contract_impl::<T>(beneficiary)
fn restore_contract<T>(
&mut self,
_account_id: T::AccountId,
_code_hash: T::Hash,
_rent_allowance: T::Balance,
_filtered_keys: &[Key],
) where
{
unimplemented!("off-chain environment does not support contract restoration")
}
fn transfer<T>(&mut self, destination: T::AccountId, value: T::Balance) -> Result<()>
where
self.transfer_impl::<T>(&destination, value)
fn random<T>(&mut self, subject: &[u8]) -> Result<T::Hash>
where