use super::circulating_supply::GenesisInfo;
use super::utils::structured;
use super::*;
use crate::interpreter::{ExecutionContext, IMPLICIT_MESSAGE_GAS_LIMIT, VM, VMTrace};
use crate::message::{MessageRead as _, MessageReadWrite as _};
use crate::rpc::state::{ApiInvocResult, InvocResult, MessageGasCost};
use crate::shim::executor::ApplyRet;
use crate::shim::message::Message;
use crate::state_migration::run_state_migrations;
use std::time::Duration;
use tracing::instrument;
impl StateManager {
#[instrument(skip(self))]
fn call_raw(
&self,
state_cid: Option<Cid>,
msg: &Message,
tipset: Option<Tipset>,
) -> Result<ApiInvocResult, Error> {
let mut msg = msg.clone();
let chain_config = self.chain_config();
let tipset = if let Some(ts) = tipset {
if ts.epoch() > 0 {
let parent = self
.chain_index()
.load_required_tipset(ts.parents())
.map_err(Error::other)?;
if chain_config.has_expensive_fork_between(parent.epoch(), ts.epoch() + 1) {
return Err(Error::ExpensiveFork);
}
}
ts
} else {
let mut heaviest_ts = self.heaviest_tipset();
while heaviest_ts.epoch() > 0 {
let parent = self
.chain_index()
.load_required_tipset(heaviest_ts.parents())
.map_err(Error::other)?;
if !chain_config.has_expensive_fork_between(parent.epoch(), heaviest_ts.epoch() + 1)
{
break;
}
heaviest_ts = parent;
}
heaviest_ts
};
let state_cid = state_cid.unwrap_or(*tipset.parent_state());
let state_cid = match run_state_migrations(
tipset.epoch(),
self.chain_config(),
self.db(),
&state_cid,
) {
Ok(Some(new_state)) => new_state,
Ok(None) => state_cid,
Err(e) => return Err(Error::other(e)),
};
let height = tipset.epoch();
let genesis_info = GenesisInfo::from_chain_config(self.chain_config().clone());
let mut vm = VM::new(
ExecutionContext {
heaviest_tipset: tipset.shallow_clone(),
state_tree_root: state_cid,
epoch: height,
rand: Box::new(self.chain_rand(tipset.shallow_clone())),
base_fee: tipset.block_headers().first().parent_base_fee.clone(),
circ_supply: genesis_info.get_vm_circulating_supply(
height,
self.db(),
&state_cid,
)?,
chain_config: self.chain_config().shallow_clone(),
chain_index: self.chain_index().shallow_clone(),
timestamp: tipset.min_timestamp(),
},
&self.engine,
VMTrace::Traced,
)?;
let tipset_messages = self
.chain_store()
.messages_for_tipset(&tipset)
.map_err(|err| Error::Other(err.to_string()))?;
let prior_messsages = tipset_messages
.iter()
.filter(|ts_msg| ts_msg.message().from() == msg.from());
for m in prior_messsages {
vm.apply_message(m)?;
}
let state_cid = vm.flush()?;
let state = StateTree::new_from_root(self.db(), &state_cid)?;
let from_actor = state
.get_actor(&msg.from())?
.ok_or_else(|| anyhow::anyhow!("actor not found"))?;
msg.set_sequence(from_actor.sequence);
let mut msg = msg.clone();
msg.gas_limit = IMPLICIT_MESSAGE_GAS_LIMIT as u64;
let (apply_ret, duration) = vm.apply_implicit_message(&msg)?;
Ok(ApiInvocResult {
msg: msg.clone(),
msg_rct: Some(apply_ret.msg_receipt()),
msg_cid: msg.cid(),
error: apply_ret.failure_info().unwrap_or_default(),
duration: duration.as_nanos().clamp(0, u128::from(u64::MAX)) as u64,
gas_cost: MessageGasCost::default(),
execution_trace: structured::parse_events(apply_ret.exec_trace()).unwrap_or_default(),
})
}
pub fn call(&self, message: &Message, tipset: Option<Tipset>) -> Result<ApiInvocResult, Error> {
self.call_raw(None, message, tipset)
}
pub fn call_on_state(
&self,
state_cid: Cid,
message: &Message,
tipset: Option<Tipset>,
) -> Result<ApiInvocResult, Error> {
self.call_raw(Some(state_cid), message, tipset)
}
pub async fn apply_on_state_with_gas(
&self,
tipset: Option<Tipset>,
msg: Message,
vm_flush: VMFlush,
) -> anyhow::Result<(ApiInvocResult, Option<Cid>)> {
let ts = tipset.unwrap_or_else(|| self.heaviest_tipset());
let from_a = self.resolve_to_deterministic_address(msg.from, &ts).await?;
let chain_msg = ChainMessage::for_gas_estimation(msg.clone(), from_a.protocol());
let (_invoc_res, apply_ret, duration, state_root) = self
.call_with_gas(chain_msg, Default::default(), Some(ts), vm_flush)
.await?;
Ok((
ApiInvocResult {
msg_cid: msg.cid(),
msg,
msg_rct: Some(apply_ret.msg_receipt()),
error: apply_ret.failure_info().unwrap_or_default(),
duration: duration.as_nanos().clamp(0, u128::from(u64::MAX)) as u64,
gas_cost: MessageGasCost::default(),
execution_trace: structured::parse_events(apply_ret.exec_trace())
.unwrap_or_default(),
},
state_root,
))
}
pub async fn call_with_gas(
&self,
mut message: ChainMessage,
prior_messages: Arc<Vec<ChainMessage>>,
tipset: Option<Tipset>,
vm_flush: VMFlush,
) -> Result<(InvocResult, ApplyRet, Duration, Option<Cid>), Error> {
let ts = tipset.unwrap_or_else(|| self.heaviest_tipset());
let TipsetState { state_root, .. } = self
.load_tipset_state(&ts)
.await
.map_err(|e| Error::Other(format!("Could not load tipset state: {e:#}")))?;
let chain_rand = self.chain_rand(ts.clone());
let epoch = ts.epoch() + 1;
let genesis_info = GenesisInfo::from_chain_config(self.chain_config().clone());
let this = self.shallow_clone();
tokio::task::spawn_blocking(move || {
let (ret, duration, state_cid) = stacker::grow(64 << 20, || -> anyhow::Result<_> {
let mut vm = VM::new(
ExecutionContext {
heaviest_tipset: ts.clone(),
state_tree_root: state_root,
epoch,
rand: Box::new(chain_rand),
base_fee: ts.block_headers().first().parent_base_fee.clone(),
circ_supply: genesis_info.get_vm_circulating_supply(
epoch,
this.chain_index().db(),
&state_root,
)?,
chain_config: this.chain_config().shallow_clone(),
chain_index: this.chain_index().shallow_clone(),
timestamp: ts.min_timestamp(),
},
&this.engine,
VMTrace::NotTraced,
)?;
for msg in prior_messages.iter() {
vm.apply_message(msg)?;
}
let from_actor = vm
.get_actor(&message.from())
.map_err(|e| Error::Other(format!("Could not get actor from state: {e:#}")))?
.ok_or_else(|| Error::Other("cant find actor in state tree".to_string()))?;
message.set_sequence(from_actor.sequence);
let (ret, duration) = vm.apply_message(&message)?;
let state_root = match vm_flush {
VMFlush::Flush => Some(vm.flush()?),
VMFlush::Skip => None,
};
Ok((ret, duration, state_root))
})?;
Ok((
InvocResult::new(message.message().clone(), &ret),
ret,
duration,
state_cid,
))
})
.await?
}
}