mirror of
https://github.com/serai-dex/serai.git
synced 2025-01-07 03:19:30 +00:00
5501de1f3a
Also makes various tweaks as necessary.
105 lines
3.6 KiB
Rust
105 lines
3.6 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::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.
|
|
pub fn new(provider: Arc<RootProvider<SimpleRequest>>, address: [u8; 20]) -> Self {
|
|
Self(provider, 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.ok().flatten().ok_or(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)
|
|
}
|
|
}
|