Skip to content
This repository has been archived by the owner on Nov 15, 2023. It is now read-only.

Test Fix Cross-contract calling: simple debugger #14711

Closed
wants to merge 42 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
42 commits
Select commit Hold shift + click to select a range
1361556
Provide basic breakpoints
pmikolajczyk41 Jul 28, 2023
4ce4d35
Rename to Observer
pmikolajczyk41 Jul 28, 2023
f1bbb0c
Rename feature. Single trait. Borrow-checker
pmikolajczyk41 Jul 31, 2023
1e8707d
: frame_system::Config
pmikolajczyk41 Jul 31, 2023
2c1dd10
Confused type name
pmikolajczyk41 Jul 31, 2023
76c2c8c
Minor bugs
pmikolajczyk41 Jul 31, 2023
c75d8d5
pub trait
pmikolajczyk41 Jul 31, 2023
7a89df0
No unnecessary cloning
pmikolajczyk41 Aug 1, 2023
39bff7d
Make node compile with all features
pmikolajczyk41 Aug 1, 2023
bd32939
Move everything debug-related to a single module
pmikolajczyk41 Aug 1, 2023
a6e6d6e
Add docs and implementation for ()
pmikolajczyk41 Aug 1, 2023
386c7ee
fmt
pmikolajczyk41 Aug 1, 2023
19e9da0
Make it feature-gated or for tests
pmikolajczyk41 Aug 1, 2023
9d28acc
Prepare testing kit
pmikolajczyk41 Aug 1, 2023
55962ea
Testcase
pmikolajczyk41 Aug 1, 2023
9d96ff4
Merge remote-tracking branch 'origin/master' into pmikolajczyk41/cont…
pmikolajczyk41 Aug 1, 2023
52b267d
Fmt
pmikolajczyk41 Aug 1, 2023
8387b60
Use feature in dev-deps
pmikolajczyk41 Aug 2, 2023
91cc170
Merge remote-tracking branch 'origin/master' into pmikolajczyk41/cont…
pmikolajczyk41 Aug 2, 2023
04780a7
?
pmikolajczyk41 Aug 2, 2023
725cfcc
feature propagation
pmikolajczyk41 Aug 2, 2023
232f12c
AAAA
pmikolajczyk41 Aug 2, 2023
e8ac68a
lol, that doesn't make much sense to me
pmikolajczyk41 Aug 2, 2023
d1a8027
Turn on
pmikolajczyk41 Aug 2, 2023
5a95151
clippy
pmikolajczyk41 Aug 2, 2023
539e6b6
Remove self dep
pmikolajczyk41 Aug 2, 2023
eb8f34c
fmt, feature-gating test
pmikolajczyk41 Aug 2, 2023
9dd8b33
Noop to trigger CI
pmikolajczyk41 Aug 3, 2023
3a3aa45
idk
pmikolajczyk41 Aug 3, 2023
cae093f
add feature to pipeline
pmikolajczyk41 Aug 3, 2023
17a25ba
Corrupt test to see if it is actually being run
pmikolajczyk41 Aug 3, 2023
a3f8461
Revert change
pmikolajczyk41 Aug 3, 2023
986e9e2
Doc for conf type
pmikolajczyk41 Aug 3, 2023
23b03b7
Review
pmikolajczyk41 Aug 3, 2023
5970e8a
Imports
pmikolajczyk41 Aug 3, 2023
19c74f9
...
pmikolajczyk41 Aug 3, 2023
b2aa12f
Merge remote-tracking branch 'origin/master' into pmikolajczyk41/cont…
pmikolajczyk41 Aug 4, 2023
6485e4d
Remove debug for kitchen-sink
pmikolajczyk41 Aug 4, 2023
01ef583
Move test
pmikolajczyk41 Aug 4, 2023
1eaabb1
Fix imports
pmikolajczyk41 Aug 4, 2023
92f29be
fix build issue
pgherveou Aug 4, 2023
02b0d5b
Probably need that as well...
pgherveou Aug 4, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions bin/node/runtime/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -134,6 +134,7 @@ substrate-wasm-builder = { version = "5.0.0-dev", path = "../../../utils/wasm-bu
[features]
default = ["std"]
with-tracing = ["frame-executive/with-tracing"]
unsafe-debug = ["pallet-contracts/unsafe-debug"]
std = [
"pallet-whitelist/std",
"pallet-offences-benchmarking?/std",
Expand Down
2 changes: 2 additions & 0 deletions bin/node/runtime/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1256,6 +1256,8 @@ impl pallet_contracts::Config for Runtime {
type Migrations = ();
#[cfg(feature = "runtime-benchmarks")]
type Migrations = pallet_contracts::migration::codegen::BenchMigrations;
#[cfg(feature = "unsafe-debug")]
type Debug = ();
type MaxDelegateDependencies = ConstU32<32>;
type CodeHashLockupDepositPercent = CodeHashLockupDepositPercent;
}
Expand Down
3 changes: 2 additions & 1 deletion frame/contracts/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -112,5 +112,6 @@ try-runtime = [
"pallet-proxy/try-runtime",
"pallet-timestamp/try-runtime",
"pallet-utility/try-runtime",
"sp-runtime/try-runtime"
"sp-runtime/try-runtime",
]
unsafe-debug = []
14 changes: 13 additions & 1 deletion frame/contracts/src/exec.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,8 @@
// See the License for the specific language governing permissions and
// limitations under the License.

#[cfg(feature = "unsafe-debug")]
use crate::unsafe_debug::ExecutionObserver;
use crate::{
gas::GasMeter,
storage::{self, DepositAccount, WriteOutcome},
Expand Down Expand Up @@ -344,7 +346,7 @@ pub trait Ext: sealing::Sealed {
}

/// Describes the different functions that can be exported by an [`Executable`].
#[derive(Clone, Copy, PartialEq)]
#[derive(Clone, Copy, PartialEq, Eq, Debug)]
pub enum ExportedFunction {
/// The constructor function which is executed on deployment of a contract.
Constructor,
Expand Down Expand Up @@ -904,11 +906,21 @@ where
// Every non delegate call or instantiate also optionally transfers the balance.
self.initial_transfer()?;

#[cfg(feature = "unsafe-debug")]
let (code_hash, input_clone) = {
let code_hash = *executable.code_hash();
T::Debug::before_call(&code_hash, entry_point, &input_data);
(code_hash, input_data.clone())
};

// Call into the Wasm blob.
let output = executable
.execute(self, &entry_point, input_data)
.map_err(|e| ExecError { error: e.error, origin: ErrorOrigin::Callee })?;

#[cfg(feature = "unsafe-debug")]
T::Debug::after_call(&code_hash, entry_point, input_clone, &output);

// Avoid useless work that would be reverted anyways.
if output.did_revert() {
return Ok(output)
Expand Down
9 changes: 9 additions & 0 deletions frame/contracts/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -97,6 +97,7 @@ mod wasm;

pub mod chain_extension;
pub mod migration;
pub mod unsafe_debug;
pub mod weights;

#[cfg(test)]
Expand Down Expand Up @@ -351,6 +352,14 @@ pub mod pallet {
/// type Migrations = (v10::Migration<Runtime, Currency>,);
/// ```
type Migrations: MigrateSequence;

/// Type that provides debug handling for the contract execution process.
///
/// # Warning
///
/// Do **not** use it in a production environment or for benchmarking purposes.
#[cfg(feature = "unsafe-debug")]
type Debug: unsafe_debug::UnsafeDebug<Self>;
}

#[pallet::hooks]
Expand Down
3 changes: 3 additions & 0 deletions frame/contracts/src/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@
// limitations under the License.

mod pallet_dummy;
mod unsafe_debug;

use self::test_utils::{ensure_stored, expected_deposit, hash};
use crate::{
Expand Down Expand Up @@ -464,6 +465,8 @@ impl Config for Test {
type Migrations = crate::migration::codegen::BenchMigrations;
type CodeHashLockupDepositPercent = CodeHashLockupDepositPercent;
type MaxDelegateDependencies = MaxDelegateDependencies;
#[cfg(feature = "unsafe-debug")]
type Debug = unsafe_debug::TestDebugger;
}

pub const ALICE: AccountId32 = AccountId32::new([1u8; 32]);
Expand Down
138 changes: 138 additions & 0 deletions frame/contracts/src/tests/unsafe_debug.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,138 @@
#![cfg(feature = "unsafe-debug")]

use super::*;
use crate::unsafe_debug::{ExecutionObserver, ExportedFunction};
use frame_support::traits::Currency;
use pallet_contracts_primitives::ExecReturnValue;
use pretty_assertions::assert_eq;
use std::cell::RefCell;

#[derive(Clone, PartialEq, Eq, Debug)]
struct DebugFrame {
code_hash: CodeHash<Test>,
call: ExportedFunction,
input: Vec<u8>,
result: Option<Vec<u8>>,
}

thread_local! {
static DEBUG_EXECUTION_TRACE: RefCell<Vec<DebugFrame>> = RefCell::new(Vec::new());
}

pub struct TestDebugger;

impl ExecutionObserver<CodeHash<Test>> for TestDebugger {
fn before_call(code_hash: &CodeHash<Test>, entry_point: ExportedFunction, input_data: &[u8]) {
DEBUG_EXECUTION_TRACE.with(|d| {
d.borrow_mut().push(DebugFrame {
code_hash: code_hash.clone(),
call: entry_point,
input: input_data.to_vec(),
result: None,
})
});
}

fn after_call(
code_hash: &CodeHash<Test>,
entry_point: ExportedFunction,
input_data: Vec<u8>,
output: &ExecReturnValue,
) {
DEBUG_EXECUTION_TRACE.with(|d| {
d.borrow_mut().push(DebugFrame {
code_hash: code_hash.clone(),
call: entry_point,
input: input_data,
result: Some(output.data.clone()),
})
});
}
}

#[test]
fn unsafe_debugging_works() {
let (wasm_caller, code_hash_caller) = compile_module::<Test>("call").unwrap();
let (wasm_callee, code_hash_callee) = compile_module::<Test>("store_call").unwrap();

fn current_stack() -> Vec<DebugFrame> {
DEBUG_EXECUTION_TRACE.with(|stack| stack.borrow().clone())
}

fn deploy(wasm: Vec<u8>) -> AccountId32 {
Contracts::bare_instantiate(
ALICE,
0,
GAS_LIMIT,
None,
Code::Upload(wasm),
vec![],
vec![],
DebugInfo::Skip,
CollectEvents::Skip,
)
.result
.unwrap()
.account_id
}

fn constructor_frame(hash: CodeHash<Test>, after: bool) -> DebugFrame {
DebugFrame {
code_hash: hash,
call: ExportedFunction::Constructor,
input: vec![],
result: if after { Some(vec![]) } else { None },
}
}

fn call_frame(hash: CodeHash<Test>, args: Vec<u8>, after: bool) -> DebugFrame {
DebugFrame {
code_hash: hash,
call: ExportedFunction::Call,
input: args,
result: if after { Some(vec![]) } else { None },
}
}

ExtBuilder::default().existential_deposit(200).build().execute_with(|| {
let _ = Balances::deposit_creating(&ALICE, 1_000_000);

assert_eq!(current_stack(), vec![]);

let addr_caller = deploy(wasm_caller);
let addr_callee = deploy(wasm_callee);

assert_eq!(
current_stack(),
vec![
constructor_frame(code_hash_caller, false),
constructor_frame(code_hash_caller, true),
constructor_frame(code_hash_callee, false),
constructor_frame(code_hash_callee, true),
]
);

let main_args = (100u32, &addr_callee).encode();
let inner_args = (100u32).encode();

assert_ok!(Contracts::call(
RuntimeOrigin::signed(ALICE),
addr_caller,
0,
GAS_LIMIT,
None,
main_args.clone()
));

let stack_top = current_stack()[4..].to_vec();
assert_eq!(
stack_top,
vec![
call_frame(code_hash_caller, main_args.clone(), false),
call_frame(code_hash_callee, inner_args.clone(), false),
call_frame(code_hash_callee, inner_args, true),
call_frame(code_hash_caller, main_args, true),
]
);
});
}
47 changes: 47 additions & 0 deletions frame/contracts/src/unsafe_debug.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
#![cfg(feature = "unsafe-debug")]

pub use crate::exec::ExportedFunction;
use crate::{CodeHash, Vec};
use pallet_contracts_primitives::ExecReturnValue;

/// Umbrella trait for all interfaces that serves for debugging, but are not suitable for any
/// production or benchmarking use.
pub trait UnsafeDebug<T: frame_system::Config>: ExecutionObserver<CodeHash<T>> {}

impl<T: frame_system::Config, D> UnsafeDebug<T> for D where D: ExecutionObserver<CodeHash<T>> {}

/// Defines the interface between pallet contracts and the outside observer.
///
/// The intended use is the environment, where the observer holds directly the whole runtime
/// (externalities) and thus can react to the execution breakpoints synchronously.
///
/// This definitely *should not* be used in any production or benchmarking setting, since handling
/// callbacks might be arbitrarily expensive and thus significantly influence performance.
pub trait ExecutionObserver<CodeHash> {
/// Called just before the execution of a contract.
///
/// # Arguments
///
/// * `code_hash` - The code hash of the contract being called.
/// * `entry_point` - Describes whether the call is the constructor or a regular call.
/// * `input_data` - The raw input data of the call.
fn before_call(_code_hash: &CodeHash, _entry_point: ExportedFunction, _input_data: &[u8]) {}

/// Called just after the execution of a contract.
///
/// # Arguments
///
/// * `code_hash` - The code hash of the contract being called.
/// * `entry_point` - Describes whether the call was the constructor or a regular call.
/// * `input_data` - The raw input data of the call.
/// * `output` - The raw output of the call.
fn after_call(
_code_hash: &CodeHash,
_entry_point: ExportedFunction,
_input_data: Vec<u8>,
_output: &ExecReturnValue,
) {
}
}

impl<CodeHash> ExecutionObserver<CodeHash> for () {}
2 changes: 1 addition & 1 deletion scripts/ci/gitlab/pipeline/test.yml
Original file line number Diff line number Diff line change
Expand Up @@ -221,7 +221,7 @@ test-linux-stable:
--locked
--release
--verbose
--features runtime-benchmarks,try-runtime,experimental
--features runtime-benchmarks,try-runtime,experimental,unsafe-debug
--manifest-path ./bin/node/cli/Cargo.toml
--partition count:${CI_NODE_INDEX}/${CI_NODE_TOTAL}
# we need to update cache only from one job
Expand Down