Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

refactor(pallet-gear): Allow calculate_gas_info on terminated program. #3278

Merged
merged 5 commits into from
Sep 18, 2023
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
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
27 changes: 21 additions & 6 deletions pallets/gear/src/runtime_api.rs
Original file line number Diff line number Diff line change
Expand Up @@ -17,9 +17,10 @@
// along with this program. If not, see <https://www.gnu.org/licenses/>.

use super::*;
use crate::queue::QueueStep;
use crate::queue::{ActorResult, QueueStep};
use common::ActiveProgram;
use core::convert::TryFrom;
use core_processor::common::PrechargedDispatch;
use gear_core::{code::TryNewCodeConfig, pages::WasmPage};
use gear_wasm_instrument::syscalls::SysCallName;

Expand Down Expand Up @@ -126,11 +127,25 @@ where
break;
};

//let actor = ext_manager
// .get_actor(actor_id)
// .ok_or_else(|| b"Program not found in the storage".to_vec())?;
mqxf marked this conversation as resolved.
Show resolved Hide resolved

let actor_id = queued_dispatch.destination();
let dispatch_id = queued_dispatch.id();
let dispatch_reply = queued_dispatch.reply_details().is_some();

let balance = CurrencyOf::<T>::free_balance(&<T::AccountId as Origin>::from_origin(
actor_id.into_origin(),
));

let actor = ext_manager
.get_actor(actor_id)
.ok_or_else(|| b"Program not found in the storage".to_vec())?;
let get_actor_data = |precharged_dispatch: PrechargedDispatch| {
// At this point gas counters should be changed accordingly so fetch the program data.
match Self::get_active_actor_data(actor_id, dispatch_id, dispatch_reply) {
ActorResult::Data(data) => Ok((precharged_dispatch, data)),
ActorResult::Continue => Err(precharged_dispatch),
}
};

let dispatch_id = queued_dispatch.id();
let success_reply = queued_dispatch
Expand All @@ -148,8 +163,8 @@ where
ext_manager: &mut ext_manager,
gas_limit,
dispatch: queued_dispatch,
balance: actor.balance,
get_actor_data: |dispatch| Ok((dispatch, actor.executable_data)),
balance: balance.unique_saturated_into(),
get_actor_data,
};
let journal = step.execute().unwrap_or_else(|e| unreachable!("{e:?}"));

Expand Down
39 changes: 39 additions & 0 deletions pallets/gear/src/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15251,3 +15251,42 @@ mod utils {
<Test as pallet_gear_bank::Config>::GasMultiplier::get().gas_to_value(gas)
}
}

#[test]
fn test_gas_info_of_terminated_program() {
mqxf marked this conversation as resolved.
Show resolved Hide resolved
use demo_constructor::{Calls, Scheme};

init_logger();
new_test_ext().execute_with(|| {
// Dies in init
let init_dead = Calls::builder().panic("Die in init");
let handle_dead = Calls::builder().panic("Called after being terminated!");
let (_, pid_dead) = utils::submit_constructor_with_args(
USER_1,
b"salt1",
Scheme::predefined(init_dead, handle_dead, Calls::default()),
0,
);

// Sends in handle message do dead program
let handle_proxy = Calls::builder().send(pid_dead.into_bytes(), []);
let (_, proxy_pid) = utils::submit_constructor_with_args(
USER_1,
b"salt2",
Scheme::predefined(Calls::default(), handle_proxy, Calls::default()),
0,
);

run_to_next_block(None);

let _gas_info = Gear::calculate_gas_info(
USER_1.into_origin(),
HandleKind::Handle(proxy_pid),
EMPTY_PAYLOAD.to_vec(),
0,
true,
true,
)
.expect("failed getting gas info");
})
}