1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
use std::{sync::Arc, collections::HashSet};

use alloy_core::primitives::{Address, B256, U256};

use alloy_sol_types::{SolInterface, SolEvent};

use alloy_rpc_types_eth::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)
  }
}