use std::{collections::HashMap, env, str::FromStr};
use alloy::{
primitives::{Address, Bytes, U256},
providers::{Provider, ProviderBuilder},
sol_types::SolValue,
transports::{RpcError, TransportErrorKind},
};
use hex::FromHex;
use num_bigint::BigInt;
use revm::state::Bytecode;
use serde_json::Value;
use tycho_common::simulation::errors::SimulationError;
use crate::evm::{simulation::SimulationEngineError, ContractCompiler, SlotId};
pub(crate) fn coerce_error(
err: &SimulationEngineError,
pool_state: &str,
gas_limit: Option<u64>,
) -> SimulationError {
match err {
SimulationEngineError::TransactionError { ref data, ref gas_used }
if data.starts_with("0x") =>
{
let reason = parse_solidity_error_message(data);
let err = SimulationEngineError::TransactionError {
data: format!("Revert! Reason: {reason}"),
gas_used: *gas_used,
};
if let (Some(gas_limit), Some(gas_used)) = (gas_limit, gas_used) {
let usage = *gas_used as f64 / gas_limit as f64;
if usage >= 0.97 {
return SimulationError::InvalidInput(
format!(
"SimulationError: Likely out-of-gas. Used: {:.2}% of gas limit. \
Original error: {}. \
Pool state: {}",
usage * 100.0,
err,
pool_state,
),
None,
);
}
}
SimulationError::FatalError(format!("Simulation reverted for unknown reason: {reason}"))
}
SimulationEngineError::TransactionError { ref data, ref gas_used }
if data.contains("OutOfGas") =>
{
let usage_msg = if let (Some(gas_limit), Some(gas_used)) = (gas_limit, gas_used) {
let usage = *gas_used as f64 / gas_limit as f64;
format!("Used: {:.2}% of gas limit. ", usage * 100.0)
} else {
String::new()
};
SimulationError::InvalidInput(
format!(
"SimulationError: out-of-gas. {usage_msg} Original error: {data}. Pool state: {pool_state}"
),
None,
)
}
SimulationEngineError::TransactionError { ref data, .. } => {
SimulationError::FatalError(format!("TransactionError: {data}"))
}
SimulationEngineError::StorageError(message) => {
SimulationError::RecoverableError(message.clone())
}
_ => SimulationError::FatalError(err.clone().to_string()),
}
}
fn parse_solidity_error_message(data: &str) -> String {
if data.len() >= 10 {
let data_bytes = match Vec::from_hex(&data[2..]) {
Ok(bytes) => bytes,
Err(_) => return format!("Failed to decode: {data}"),
};
if data_bytes.starts_with(&[0x08, 0xc3, 0x79, 0xa0]) {
if let Ok(decoded) = String::abi_decode(&data_bytes[4..]) {
return decoded;
}
} else if data_bytes.starts_with(&[0x4e, 0x48, 0x7b, 0x71]) {
if let Ok(decoded) = U256::abi_decode(&data_bytes[4..]) {
let panic_codes = get_solidity_panic_codes();
return panic_codes
.get(&decoded.as_limbs()[0])
.cloned()
.unwrap_or_else(|| format!("Panic({decoded})"));
}
}
if let Ok(decoded) = String::abi_decode(&data_bytes) {
return decoded;
}
if let Ok(decoded) = String::abi_decode(&data_bytes[4..]) {
return decoded;
}
}
format!("Failed to decode: {data}")
}
pub fn get_storage_slot_index_at_key(
key: Address,
mapping_slot: SlotId,
compiler: ContractCompiler,
) -> SlotId {
let mut key_bytes = key.as_slice().to_vec();
if key_bytes.len() < 32 {
let padding = vec![0u8; 32 - key_bytes.len()];
key_bytes.splice(0..0, padding); }
let mapping_slot_bytes: [u8; 32] = mapping_slot.to_be_bytes();
compiler.compute_map_slot(&mapping_slot_bytes, &key_bytes)
}
fn get_solidity_panic_codes() -> HashMap<u64, String> {
let mut panic_codes = HashMap::new();
panic_codes.insert(0, "GenericCompilerPanic".to_string());
panic_codes.insert(1, "AssertionError".to_string());
panic_codes.insert(17, "ArithmeticOver/Underflow".to_string());
panic_codes.insert(18, "ZeroDivisionError".to_string());
panic_codes.insert(33, "UnknownEnumMember".to_string());
panic_codes.insert(34, "BadStorageByteArrayEncoding".to_string());
panic_codes.insert(51, "EmptyArray".to_string());
panic_codes.insert(0x32, "OutOfBounds".to_string());
panic_codes.insert(0x41, "OutOfMemory".to_string());
panic_codes.insert(0x51, "BadFunctionPointer".to_string());
panic_codes
}
pub(crate) async fn get_code_for_contract(
address: &str,
connection_string: Option<String>,
) -> Result<Bytecode, SimulationError> {
let connection_string = connection_string.or_else(|| env::var("RPC_URL").ok());
let connection_string = match connection_string {
Some(url) => url,
None => {
return Err(SimulationError::FatalError(
"RPC_URL environment variable is not set".to_string(),
))
}
};
let addr = Address::from_str(address)
.map_err(|_| SimulationError::FatalError(format!("Invalid address format: {address}")))?;
match sync_get_code(&connection_string, addr) {
Ok(code) if code.is_empty() => {
Err(SimulationError::FatalError("Empty code response from RPC".to_string()))
}
Ok(code) => {
let bytecode = Bytecode::new_raw(Bytes::from(code.to_vec()));
Ok(bytecode)
}
Err(e) => match e {
RpcError::Transport(err) => Err(SimulationError::RecoverableError(format!(
"Failed to get code for contract due to internal RPC error: {err:?}"
))),
_ => Err(SimulationError::FatalError(format!(
"Failed to get code for contract. Invalid response from RPC: {e:?}"
))),
},
}
}
fn sync_get_code(
connection_string: &str,
addr: Address,
) -> Result<Bytes, RpcError<TransportErrorKind>> {
tokio::task::block_in_place(|| {
tokio::runtime::Handle::current().block_on(async {
let provider = ProviderBuilder::new()
.connect(connection_string)
.await?;
provider.get_code_at(addr).await
})
})
}
pub fn string_to_bytes32(pool_id: &str) -> Result<[u8; 32], SimulationError> {
let pool_id_no_prefix =
if let Some(stripped) = pool_id.strip_prefix("0x") { stripped } else { pool_id };
let bytes = hex::decode(pool_id_no_prefix)
.map_err(|e| SimulationError::FatalError(format!("Invalid hex string: {e}")))?;
if bytes.len() > 32 {
return Err(SimulationError::FatalError(format!(
"Hex string exceeds 32 bytes: length {}",
bytes.len()
)));
}
let mut array = [0u8; 32];
array[..bytes.len()].copy_from_slice(&bytes);
Ok(array)
}
pub fn json_deserialize_address_list(input: &[u8]) -> Result<Vec<Vec<u8>>, SimulationError> {
let json_value: Value = serde_json::from_slice(input)
.map_err(|_| SimulationError::FatalError(format!("Invalid JSON: {input:?}")))?;
if let Value::Array(hex_strings) = json_value {
let mut result = Vec::new();
for val in hex_strings {
if let Value::String(hexstring) = val {
let bytes = hex::decode(hexstring.trim_start_matches("0x")).map_err(|_| {
SimulationError::FatalError(format!("Invalid hex string: {hexstring}"))
})?;
result.push(bytes);
} else {
return Err(SimulationError::FatalError("Array contains a non-string value".into()));
}
}
Ok(result)
} else {
Err(SimulationError::FatalError("Input is not a JSON array".into()))
}
}
pub fn json_deserialize_be_bigint_list(input: &[u8]) -> Result<Vec<BigInt>, SimulationError> {
let json_value: Value = serde_json::from_slice(input)
.map_err(|_| SimulationError::FatalError(format!("Invalid JSON: {input:?}")))?;
if let Value::Array(hex_strings) = json_value {
let mut result = Vec::new();
for val in hex_strings {
if let Value::String(hexstring) = val {
let bytes = hex::decode(hexstring.trim_start_matches("0x")).map_err(|_| {
SimulationError::FatalError(format!("Invalid hex string: {hexstring}"))
})?;
let bigint = BigInt::from_signed_bytes_be(&bytes);
result.push(bigint);
} else {
return Err(SimulationError::FatalError("Array contains a non-string value".into()));
}
}
Ok(result)
} else {
Err(SimulationError::FatalError("Input is not a JSON array".into()))
}
}
#[cfg(test)]
mod tests {
use dotenv::dotenv;
use super::*;
use crate::utils::hexstring_to_vec;
#[tokio::test(flavor = "multi_thread", worker_threads = 2)]
#[cfg_attr(not(feature = "network_tests"), ignore)]
async fn test_get_code_for_address() {
let rpc_url = env::var("RPC_URL").unwrap_or_else(|_| {
dotenv().expect("Missing .env file");
env::var("RPC_URL").expect("Missing RPC_URL in .env file")
});
let address = "0x88e6A0c2dDD26FEEb64F039a2c41296FcB3f5640";
let result = get_code_for_contract(address, Some(rpc_url)).await;
assert!(result.is_ok(), "Network call should not fail");
let code = result.unwrap();
assert!(!code.bytes().is_empty(), "Code should not be empty");
}
#[test]
fn test_maybe_coerce_error_revert_no_gas_info() {
let err = SimulationEngineError::TransactionError{
data: "0x08c379a000000000000000000000000000000000000000000000000000000000000000200000000000000000000000000000000000000000000000000000000000000011496e76616c6964206f7065726174696f6e000000000000000000000000000000".to_string(),
gas_used: None
};
let result = coerce_error(&err, "test_pool", None);
if let SimulationError::FatalError(msg) = result {
assert!(msg.contains("Simulation reverted for unknown reason: Invalid operation"));
} else {
panic!("Expected SolidityError error");
}
}
#[test]
fn test_maybe_coerce_error_out_of_gas() {
let err = SimulationEngineError::TransactionError{
data: "0x08c379a000000000000000000000000000000000000000000000000000000000000000200000000000000000000000000000000000000000000000000000000000000011496e76616c6964206f7065726174696f6e000000000000000000000000000000".to_string(),
gas_used: Some(980)
};
let result = coerce_error(&err, "test_pool", Some(1000));
if let SimulationError::InvalidInput(message, _partial_result) = result {
assert!(message.contains("Used: 98.00% of gas limit."));
assert!(message.contains("test_pool"));
} else {
panic!("Expected OutOfGas error");
}
}
#[test]
fn test_maybe_coerce_error_no_gas_limit_info() {
let err = SimulationEngineError::TransactionError {
data: "OutOfGas".to_string(),
gas_used: None,
};
let result = coerce_error(&err, "test_pool", None);
if let SimulationError::InvalidInput(message, _partial_result) = result {
assert!(message.contains("Original error: OutOfGas"));
assert!(message.contains("Pool state: test_pool"));
} else {
panic!("Expected RetryDifferentInput error");
}
}
#[test]
fn test_maybe_coerce_error_storage_error() {
let err = SimulationEngineError::StorageError("Storage error:".to_string());
let result = coerce_error(&err, "test_pool", None);
if let SimulationError::RecoverableError(message) = result {
assert_eq!(message, "Storage error:");
} else {
println!("{result:?}");
panic!("Expected RetryLater error");
}
}
#[test]
fn test_maybe_coerce_error_no_match() {
let err = SimulationEngineError::TransactionError {
data: "Some other error".to_string(),
gas_used: None,
};
let result = coerce_error(&err, "test_pool", None);
if let SimulationError::FatalError(message) = result {
assert_eq!(message, "TransactionError: Some other error");
} else {
panic!("Expected solidity error");
}
}
#[test]
fn test_parse_solidity_error_message_error_string() {
let data = "0x08c379a00000000000000000000000000000000000000000000000000000000000000020000000000000000000000000000000000000000000000000000000000000000e416d6f756e7420746f6f206c6f77000000000000000000000000000000000000";
let result = parse_solidity_error_message(data);
assert_eq!(result, "Amount too low");
}
#[test]
fn test_parse_solidity_error_message_panic_code() {
let data = "0x4e487b710000000000000000000000000000000000000000000000000000000000000001";
let result = parse_solidity_error_message(data);
assert_eq!(result, "AssertionError");
}
#[test]
fn test_parse_solidity_error_message_failed_to_decode() {
let data = "0x1234567890";
let result = parse_solidity_error_message(data);
assert!(result.contains("Failed to decode"));
}
#[test]
fn test_hexstring_to_vec() {
let hexstring = "0x68656c6c6f";
let expected = vec![0x68, 0x65, 0x6c, 0x6c, 0x6f];
let result = hexstring_to_vec(hexstring).unwrap();
assert_eq!(result, expected);
}
#[test]
fn test_hexstring_to_vec_no_prefix() {
let hexstring = "68656c6c6f";
let expected = vec![0x68, 0x65, 0x6c, 0x6c, 0x6f];
let result = hexstring_to_vec(hexstring).unwrap();
assert_eq!(result, expected);
}
#[test]
fn test_hexstring_to_vec_invalid_characters() {
let hexstring = "0x68656c6c6z"; let result = hexstring_to_vec(hexstring);
assert!(result.is_err());
if let Err(SimulationError::FatalError(msg)) = result {
assert!(msg.contains("Invalid hex string"));
} else {
panic!("Expected EncodingError");
}
}
#[test]
fn test_json_deserialize_address_list() {
let json_input = r#"["0x1234","0xabcd"]"#.as_bytes();
let result = json_deserialize_address_list(json_input).unwrap();
assert_eq!(result, vec![vec![0x12, 0x34], vec![0xab, 0xcd]]);
}
#[test]
fn test_json_deserialize_bigint_list() {
let json_input = r#"["0x0b1a2bc2ec500000","0x02c68af0bb140000"]"#.as_bytes();
let result = json_deserialize_be_bigint_list(json_input).unwrap();
assert_eq!(
result,
vec![BigInt::from(800000000000000000u64), BigInt::from(200000000000000000u64)]
);
}
#[test]
fn test_invalid_deserialize_address_list() {
let json_input = r#"["invalid_hex"]"#.as_bytes();
let result = json_deserialize_address_list(json_input);
assert!(result.is_err());
}
#[test]
fn test_invalid_deserialize_bigint_list() {
let json_input = r#"["invalid_hex"]"#.as_bytes();
let result = json_deserialize_be_bigint_list(json_input);
assert!(result.is_err());
}
}