mirror of
https://github.com/serai-dex/serai.git
synced 2025-01-04 09:59:42 +00:00
119 lines
4 KiB
Rust
119 lines
4 KiB
Rust
|
use std::{sync::Arc, collections::HashSet};
|
||
|
|
||
|
use alloy_core::primitives::{Address, B256, U256};
|
||
|
|
||
|
use alloy_sol_types::{SolInterface, SolEvent};
|
||
|
|
||
|
use alloy_rpc_types::{BlockNumberOrTag, Filter};
|
||
|
use alloy_simple_request_transport::SimpleRequest;
|
||
|
use alloy_provider::{Provider, RootProvider};
|
||
|
|
||
|
use crate::Error;
|
||
|
pub use crate::abi::erc20 as abi;
|
||
|
use abi::{IERC20Calls, Transfer, transferCall, transferFromCall};
|
||
|
|
||
|
#[derive(Clone, Debug)]
|
||
|
pub struct TopLevelErc20Transfer {
|
||
|
pub id: [u8; 32],
|
||
|
pub from: [u8; 20],
|
||
|
pub amount: U256,
|
||
|
pub data: Vec<u8>,
|
||
|
}
|
||
|
|
||
|
/// A view for an ERC20 contract.
|
||
|
#[derive(Clone, Debug)]
|
||
|
pub struct ERC20(Arc<RootProvider<SimpleRequest>>, Address);
|
||
|
impl ERC20 {
|
||
|
/// Construct a new view of the specified ERC20 contract.
|
||
|
///
|
||
|
/// This checks a contract is deployed at that address yet does not check the contract is
|
||
|
/// actually an ERC20.
|
||
|
pub async fn new(
|
||
|
provider: Arc<RootProvider<SimpleRequest>>,
|
||
|
address: [u8; 20],
|
||
|
) -> Result<Option<Self>, Error> {
|
||
|
let code = provider
|
||
|
.get_code_at(address.into(), BlockNumberOrTag::Finalized.into())
|
||
|
.await
|
||
|
.map_err(|_| Error::ConnectionError)?;
|
||
|
// Contract has yet to be deployed
|
||
|
if code.is_empty() {
|
||
|
return Ok(None);
|
||
|
}
|
||
|
Ok(Some(Self(provider.clone(), Address::from(&address))))
|
||
|
}
|
||
|
|
||
|
pub async fn top_level_transfers(
|
||
|
&self,
|
||
|
block: u64,
|
||
|
to: [u8; 20],
|
||
|
) -> Result<Vec<TopLevelErc20Transfer>, Error> {
|
||
|
let filter = Filter::new().from_block(block).to_block(block).address(self.1);
|
||
|
let filter = filter.event_signature(Transfer::SIGNATURE_HASH);
|
||
|
let mut to_topic = [0; 32];
|
||
|
to_topic[12 ..].copy_from_slice(&to);
|
||
|
let filter = filter.topic2(B256::from(to_topic));
|
||
|
let logs = self.0.get_logs(&filter).await.map_err(|_| Error::ConnectionError)?;
|
||
|
|
||
|
let mut handled = HashSet::new();
|
||
|
|
||
|
let mut top_level_transfers = vec![];
|
||
|
for log in logs {
|
||
|
// Double check the address which emitted this log
|
||
|
if log.address() != self.1 {
|
||
|
Err(Error::ConnectionError)?;
|
||
|
}
|
||
|
|
||
|
let tx_id = log.transaction_hash.ok_or(Error::ConnectionError)?;
|
||
|
let tx = self.0.get_transaction_by_hash(tx_id).await.map_err(|_| Error::ConnectionError)?;
|
||
|
|
||
|
// If this is a top-level call...
|
||
|
if tx.to == Some(self.1) {
|
||
|
// And we recognize the call...
|
||
|
// Don't validate the encoding as this can't be re-encoded to an identical bytestring due
|
||
|
// to the InInstruction appended
|
||
|
if let Ok(call) = IERC20Calls::abi_decode(&tx.input, false) {
|
||
|
// Extract the top-level call's from/to/value
|
||
|
let (from, call_to, value) = match call {
|
||
|
IERC20Calls::transfer(transferCall { to: call_to, value }) => (tx.from, call_to, value),
|
||
|
IERC20Calls::transferFrom(transferFromCall { from, to: call_to, value }) => {
|
||
|
(from, call_to, value)
|
||
|
}
|
||
|
// Treat any other function selectors as unrecognized
|
||
|
_ => continue,
|
||
|
};
|
||
|
|
||
|
let log = log.log_decode::<Transfer>().map_err(|_| Error::ConnectionError)?.inner.data;
|
||
|
|
||
|
// Ensure the top-level transfer is equivalent, and this presumably isn't a log for an
|
||
|
// internal transfer
|
||
|
if (log.from != from) || (call_to != to) || (value != log.value) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// Now that the top-level transfer is confirmed to be equivalent to the log, ensure it's
|
||
|
// the only log we handle
|
||
|
if handled.contains(&tx_id) {
|
||
|
continue;
|
||
|
}
|
||
|
handled.insert(tx_id);
|
||
|
|
||
|
// Read the data appended after
|
||
|
let encoded = call.abi_encode();
|
||
|
let data = tx.input.as_ref()[encoded.len() ..].to_vec();
|
||
|
|
||
|
// Push the transfer
|
||
|
top_level_transfers.push(TopLevelErc20Transfer {
|
||
|
// Since we'll only handle one log for this TX, set the ID to the TX ID
|
||
|
id: *tx_id,
|
||
|
from: *log.from.0,
|
||
|
amount: log.value,
|
||
|
data,
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
Ok(top_level_transfers)
|
||
|
}
|
||
|
}
|