diff --git a/src/rpc/methods/eth.rs b/src/rpc/methods/eth.rs index 47375c6854eb..e3776b678d4a 100644 --- a/src/rpc/methods/eth.rs +++ b/src/rpc/methods/eth.rs @@ -31,7 +31,7 @@ use cid::{ Cid, }; use fvm_ipld_blockstore::Blockstore; -use fvm_ipld_encoding::{CBOR, DAG_CBOR, IPLD_RAW}; +use fvm_ipld_encoding::{RawBytes, CBOR, DAG_CBOR, IPLD_RAW}; use itertools::Itertools; use keccak_hash::keccak; use num_bigint::{self, Sign}; @@ -144,15 +144,6 @@ pub struct Int64( lotus_json_with_self!(Int64); -#[derive(PartialEq, Debug, Deserialize, Serialize, Default, Clone, JsonSchema)] -pub struct Bytes( - #[schemars(with = "String")] - #[serde(with = "crate::lotus_json::hexify_vec_bytes")] - pub Vec, -); - -lotus_json_with_self!(Bytes); - #[derive(PartialEq, Debug, Deserialize, Serialize, Default, Clone, JsonSchema)] pub struct Hash(#[schemars(with = "String")] pub ethereum_types::H256); @@ -300,7 +291,7 @@ pub struct Block { pub gas_limit: Uint64, pub gas_used: Uint64, pub timestamp: Uint64, - pub extra_data: Bytes, + pub extra_data: EthBytes, pub mix_hash: Hash, pub nonce: Nonce, pub base_fee_per_gas: BigInt, @@ -347,7 +338,7 @@ pub struct Tx { pub to: Option, pub value: BigInt, pub r#type: Uint64, - pub input: Bytes, + pub input: EthBytes, pub gas: Uint64, pub max_fee_per_gas: BigInt, pub max_priority_fee_per_gas: BigInt, @@ -881,7 +872,7 @@ fn eth_tx_from_signed_eth_message(smsg: &SignedMessage, chain_id: u32) -> Result v, r, s, - input: Bytes(tx_args.input), + input: EthBytes(tx_args.input), ..Tx::default() }) } @@ -933,10 +924,10 @@ fn encode_filecoin_params_as_abi( method: MethodNum, codec: u64, params: &fvm_ipld_encoding::RawBytes, -) -> Result { +) -> Result { let mut buffer: Vec = vec![0x86, 0x8e, 0x10, 0xc4]; buffer.append(&mut encode_filecoin_returns_as_abi(method, codec, params)); - Ok(Bytes(buffer)) + Ok(EthBytes(buffer)) } fn encode_filecoin_returns_as_abi( @@ -984,16 +975,16 @@ fn encode_as_abi_helper(param1: u64, param2: u64, data: &[u8]) -> Vec { } /// Decodes the payload using the given codec. -fn decode_payload(payload: &fvm_ipld_encoding::RawBytes, codec: u64) -> Result { +fn decode_payload(payload: &fvm_ipld_encoding::RawBytes, codec: u64) -> Result { match codec { DAG_CBOR | CBOR => { let result: Result, _> = serde_ipld_dagcbor::de::from_reader(payload.reader()); match result { - Ok(buffer) => Ok(Bytes(buffer)), + Ok(buffer) => Ok(EthBytes(buffer)), Err(err) => bail!("decode_payload: failed to decode cbor payload: {err}"), } } - IPLD_RAW => Ok(Bytes(payload.to_vec())), + IPLD_RAW => Ok(EthBytes(payload.to_vec())), _ => bail!("decode_payload: unsupported codec {codec}"), } } @@ -1275,7 +1266,7 @@ impl RpcMethod<2> for EthGetCode { const PERMISSION: Permission = Permission::Read; type Params = (EthAddress, BlockNumberOrHash); - type Ok = Bytes; + type Ok = EthBytes; async fn handle( ctx: Ctx, @@ -1299,18 +1290,17 @@ impl RpcMethod<2> for EthGetCode { gas_limit: BLOCK_GAS_LIMIT, ..Default::default() }; - let mut api_invoc_result = None; - for ts in ts.chain_arc(ctx.store()) { - match ctx.state_manager.call(&message, Some(ts)) { - Ok(res) => { - api_invoc_result = Some(res); - break; + + let api_invoc_result = 'invoc: { + for ts in ts.chain_arc(ctx.store()) { + match ctx.state_manager.call(&message, Some(ts)) { + Ok(res) => { + break 'invoc res; + } + Err(e) => tracing::warn!(%e), } - Err(e) => tracing::warn!(%e), } - } - let Some(api_invoc_result) = api_invoc_result else { - return Err(anyhow::anyhow!("no message receipt").into()); + return Err(anyhow::anyhow!("Call failed").into()); }; let Some(msg_rct) = api_invoc_result.msg_rct else { return Err(anyhow::anyhow!("no message receipt").into()); @@ -1322,13 +1312,86 @@ impl RpcMethod<2> for EthGetCode { let get_bytecode_return: GetBytecodeReturn = fvm_ipld_encoding::from_slice(msg_rct.return_data().as_slice())?; if let Some(cid) = get_bytecode_return.0 { - Ok(Bytes(ctx.store().get_required(&cid)?)) + Ok(EthBytes(ctx.store().get_required(&cid)?)) } else { Ok(Default::default()) } } } +pub enum EthGetStorageAt {} +impl RpcMethod<3> for EthGetStorageAt { + const NAME: &'static str = "Filecoin.EthGetStorageAt"; + const PARAM_NAMES: [&'static str; 3] = ["eth_address", "position", "block_number_or_hash"]; + const API_VERSION: ApiVersion = ApiVersion::V1; + const PERMISSION: Permission = Permission::Read; + + type Params = (EthAddress, EthBytes, BlockNumberOrHash); + type Ok = EthBytes; + + async fn handle( + ctx: Ctx, + (eth_address, position, block_number_or_hash): Self::Params, + ) -> Result { + let make_empty_result = || EthBytes(vec![0; EVM_WORD_LENGTH]); + + let ts = tipset_by_block_number_or_hash(&ctx.chain_store, block_number_or_hash)?; + let to_address = FilecoinAddress::try_from(ð_address)?; + let Some(actor) = ctx + .state_manager + .get_actor(&to_address, *ts.parent_state())? + else { + return Ok(make_empty_result()); + }; + + if !fil_actor_interface::is_evm_actor(&actor.code) { + return Ok(make_empty_result()); + } + + let params = RawBytes::new(GetStorageAtParams::new(position.0)?.serialize_params()?); + let message = Message { + from: FilecoinAddress::SYSTEM_ACTOR, + to: to_address, + method_num: METHOD_GET_STORAGE_AT, + gas_limit: BLOCK_GAS_LIMIT, + params, + ..Default::default() + }; + let api_invoc_result = 'invoc: { + for ts in ts.chain_arc(ctx.store()) { + match ctx.state_manager.call(&message, Some(ts)) { + Ok(res) => { + break 'invoc res; + } + Err(e) => tracing::warn!(%e), + } + } + return Err(anyhow::anyhow!("Call failed").into()); + }; + let Some(msg_rct) = api_invoc_result.msg_rct else { + return Err(anyhow::anyhow!("no message receipt").into()); + }; + if !msg_rct.exit_code().is_success() || !api_invoc_result.error.is_empty() { + return Err(anyhow::anyhow!( + "failed to lookup storage slot: {}", + api_invoc_result.error + ) + .into()); + } + + let mut ret = fvm_ipld_encoding::from_slice::(msg_rct.return_data().as_slice())? + .bytes() + .to_vec(); + if ret.len() < EVM_WORD_LENGTH { + let mut with_padding = vec![0; EVM_WORD_LENGTH.saturating_sub(ret.len())]; + with_padding.append(&mut ret); + Ok(EthBytes(with_padding)) + } else { + Ok(EthBytes(ret)) + } + } +} + #[cfg(test)] mod test { use super::*; diff --git a/src/rpc/methods/eth/types.rs b/src/rpc/methods/eth/types.rs index 757a9ef76d2b..198e3137af3c 100644 --- a/src/rpc/methods/eth/types.rs +++ b/src/rpc/methods/eth/types.rs @@ -4,17 +4,51 @@ use super::*; pub const METHOD_GET_BYTE_CODE: u64 = 3; +pub const METHOD_GET_STORAGE_AT: u64 = 5; + +#[derive(PartialEq, Debug, Deserialize, Serialize, Default, Clone, JsonSchema)] +pub struct EthBytes( + #[schemars(with = "String")] + #[serde(with = "crate::lotus_json::hexify_vec_bytes")] + pub Vec, +); +lotus_json_with_self!(EthBytes); #[derive(Debug, Deserialize, Serialize)] pub struct GetBytecodeReturn(pub Option); +const GET_STORAGE_AT_PARAMS_ARRAY_LENGTH: usize = 32; + +#[derive(Debug, Clone)] +pub struct GetStorageAtParams(pub [u8; GET_STORAGE_AT_PARAMS_ARRAY_LENGTH]); + +impl GetStorageAtParams { + pub fn new(position: Vec) -> anyhow::Result { + if position.len() > GET_STORAGE_AT_PARAMS_ARRAY_LENGTH { + anyhow::bail!("supplied storage key is too long"); + } + let mut bytes = [0; GET_STORAGE_AT_PARAMS_ARRAY_LENGTH]; + bytes + .get_mut(GET_STORAGE_AT_PARAMS_ARRAY_LENGTH.saturating_sub(position.len())..) + .expect("Infallible") + .copy_from_slice(&position); + Ok(Self(bytes)) + } + + pub fn serialize_params(&self) -> anyhow::Result> { + const LENGTH_BUF_GET_STORAGE_AT_PARAMS: u8 = 129; + let mut encoded = fvm_ipld_encoding::to_vec(&RawBytes::new(self.0.to_vec()))?; + encoded.insert(0, LENGTH_BUF_GET_STORAGE_AT_PARAMS); + Ok(encoded) + } +} + #[derive(PartialEq, Debug, Deserialize, Serialize, Default, Clone, JsonSchema)] pub struct EthAddress( #[schemars(with = "String")] #[serde(with = "crate::lotus_json::hexify_bytes")] pub ethereum_types::Address, ); - lotus_json_with_self!(EthAddress); impl EthAddress { @@ -152,4 +186,13 @@ mod tests { let ser = fvm_ipld_encoding::to_vec(&des).unwrap(); assert_eq!(ser, bytes); } + + #[test] + fn get_storage_at_params() { + let param = GetStorageAtParams::new(vec![0xa]).unwrap(); + assert_eq!( + &hex::encode(param.serialize_params().unwrap()), + "815820000000000000000000000000000000000000000000000000000000000000000a" + ); + } } diff --git a/src/rpc/mod.rs b/src/rpc/mod.rs index 4530a291eb06..3ae83d20ac25 100644 --- a/src/rpc/mod.rs +++ b/src/rpc/mod.rs @@ -69,6 +69,7 @@ macro_rules! for_each_method { $callback!(crate::rpc::eth::EthBlockNumber); $callback!(crate::rpc::eth::EthChainId); $callback!(crate::rpc::eth::EthGetCode); + $callback!(crate::rpc::eth::EthGetStorageAt); $callback!(crate::rpc::eth::EthGasPrice); $callback!(crate::rpc::eth::EthGetBalance); $callback!(crate::rpc::eth::EthGetBlockByNumber); diff --git a/src/shim/error.rs b/src/shim/error.rs index c63faaf4c006..512279c09e40 100644 --- a/src/shim/error.rs +++ b/src/shim/error.rs @@ -31,6 +31,10 @@ impl ExitCode { pub fn value(&self) -> u32 { self.0.value() } + + pub fn is_success(&self) -> bool { + self.0.is_success() + } } impl From for ExitCode { diff --git a/src/tool/subcommands/api_cmd.rs b/src/tool/subcommands/api_cmd.rs index 6854a3eaf0e4..839a1cb7d16c 100644 --- a/src/tool/subcommands/api_cmd.rs +++ b/src/tool/subcommands/api_cmd.rs @@ -14,7 +14,7 @@ use crate::message::{Message as _, SignedMessage}; use crate::message_pool::{MessagePool, MpoolRpcProvider}; use crate::networks::{parse_bootstrap_peers, ChainConfig, NetworkChain}; use crate::rpc::beacon::BeaconGetEntry; -use crate::rpc::eth::types::EthAddress; +use crate::rpc::eth::types::{EthAddress, EthBytes}; use crate::rpc::gas::GasEstimateGasLimit; use crate::rpc::miner::BlockTemplate; use crate::rpc::types::{ApiTipsetKey, MessageFilter, MessageLookup, SectorOnChainInfo}; @@ -1071,6 +1071,15 @@ fn eth_tests_with_tipset(shared_tipset: &Tipset) -> Vec { RpcTest::identity( EthGetBlockTransactionCountByNumber::request((Int64(shared_tipset.epoch()),)).unwrap(), ), + RpcTest::identity( + EthGetStorageAt::request(( + // https://filfox.info/en/address/f410fpoidg73f7krlfohnla52dotowde5p2sejxnd4mq + EthAddress::from_str("0x7B90337f65fAA2B2B8ed583ba1Ba6EB0C9D7eA44").unwrap(), + EthBytes(vec![0xa]), + BlockNumberOrHash::BlockNumber(shared_tipset.epoch()), + )) + .unwrap(), + ), RpcTest::identity( EthGetCode::request(( // https://filfox.info/en/address/f410fpoidg73f7krlfohnla52dotowde5p2sejxnd4mq