Skip to content

Commit

Permalink
setup the environnement to handle oz accounts (kkrt-labs#1360)
Browse files Browse the repository at this point in the history
* setup the environnement to handle oz accounts

* clean up

* clean up

* fix comments

* fix conflicts

* propagate error on AccountManager init

* fix error in async loop

* add eth_client in account manager struct

* clean up

* modify lock

* modify lock

* add error for empty accounts in account manager

* fix error handling

* clean up
  • Loading branch information
tcoratger authored Sep 9, 2024
1 parent 4bb98c1 commit a4ecc08
Show file tree
Hide file tree
Showing 12 changed files with 121 additions and 332 deletions.
6 changes: 0 additions & 6 deletions .env.example
Original file line number Diff line number Diff line change
Expand Up @@ -37,14 +37,8 @@ EVM_PRIVATE_KEY=0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff
# Number of Felt (bytes) allowed in a single call data
MAX_FELTS_IN_CALLDATA=22500

# Interval between retries of transactions (in seconds)
RETRY_TX_INTERVAL=10

# Comma separated list of white listed pre EIP-155 transaction hashes
WHITE_LISTED_EIP_155_TRANSACTION_HASHES=

# Maximum number of times a transaction can be retried
TRANSACTION_MAX_RETRIES=10

# Maximum number of logs to output for eth_getLogs RPC Method
MAX_LOGS=10000
2 changes: 0 additions & 2 deletions docker-compose.prod.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -125,9 +125,7 @@ services:
- UNINITIALIZED_ACCOUNT_CLASS_HASH=0x600f6862938312a05a0cfecba0dcaf37693efc9e4075a6adfb62e196022678e
- ACCOUNT_CONTRACT_CLASS_HASH=0x1276d0b017701646f8646b69de6c3b3584edce71879678a679f28c07a9971cf
- MAX_FELTS_IN_CALLDATA=30000
- TRANSACTION_MAX_RETRIES=10
- MAX_LOGS=10000
- RETRY_TX_INTERVAL=10
- WHITE_LISTED_EIP_155_TRANSACTION_HASHES=0xeddf9e61fb9d8f5111840daef55e5fde0041f5702856532cdbb5a02998033d26,0xb6274b80bc7cda162df89894c7748a5cb7ba2eaa6004183c41a1837c3b072f1e,0x07471adfe8f4ec553c1199f495be97fc8be8e0626ae307281c22534460184ed1,0xb95343413e459a0f97461812111254163ae53467855c0d73e0f1e7c5b8442fa3
restart: on-failure
volumes:
Expand Down
2 changes: 0 additions & 2 deletions docker-compose.staging.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -125,9 +125,7 @@ services:
- UNINITIALIZED_ACCOUNT_CLASS_HASH=0x600f6862938312a05a0cfecba0dcaf37693efc9e4075a6adfb62e196022678e
- ACCOUNT_CONTRACT_CLASS_HASH=0x1276d0b017701646f8646b69de6c3b3584edce71879678a679f28c07a9971cf
- MAX_FELTS_IN_CALLDATA=30000
- TRANSACTION_MAX_RETRIES=10
- MAX_LOGS=10000
- RETRY_TX_INTERVAL=10
- WHITE_LISTED_EIP_155_TRANSACTION_HASHES=0xeddf9e61fb9d8f5111840daef55e5fde0041f5702856532cdbb5a02998033d26,0xb6274b80bc7cda162df89894c7748a5cb7ba2eaa6004183c41a1837c3b072f1e,0x07471adfe8f4ec553c1199f495be97fc8be8e0626ae307281c22534460184ed1,0xb95343413e459a0f97461812111254163ae53467855c0d73e0f1e7c5b8442fa3
restart: on-failure
volumes:
Expand Down
2 changes: 0 additions & 2 deletions docker-compose.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -138,9 +138,7 @@ services:
- MONGO_CONNECTION_STRING=mongodb://mongo:mongo@mongo:27017
- MONGO_DATABASE_NAME=kakarot-local
- MAX_FELTS_IN_CALLDATA=30000
- TRANSACTION_MAX_RETRIES=10
- MAX_LOGS=10000
- RETRY_TX_INTERVAL=1
- WHITE_LISTED_EIP_155_TRANSACTION_HASHES=0xeddf9e61fb9d8f5111840daef55e5fde0041f5702856532cdbb5a02998033d26,0xb6274b80bc7cda162df89894c7748a5cb7ba2eaa6004183c41a1837c3b072f1e,0x07471adfe8f4ec553c1199f495be97fc8be8e0626ae307281c22534460184ed1,0xb95343413e459a0f97461812111254163ae53467855c0d73e0f1e7c5b8442fa3
- OTEL_EXPORTER_OTLP_ENDPOINT=http://otel:4317
volumes:
Expand Down
2 changes: 0 additions & 2 deletions docker/hive/Dockerfile
Original file line number Diff line number Diff line change
Expand Up @@ -144,9 +144,7 @@ ENV KAKAROT_RPC_URL=0.0.0.0:8545
ENV STARKNET_NETWORK=http://localhost:5050
ENV RUST_LOG=info
ENV MAX_FELTS_IN_CALLDATA=30000
ENV TRANSACTION_MAX_RETRIES=10
ENV MAX_LOGS=10000
ENV RETRY_TX_INTERVAL=10
ENV DEFAULT_BLOCK_GAS_LIMIT=7000000

HEALTHCHECK --interval=10s --timeout=10s --start-period=15s --retries=5 \
Expand Down
3 changes: 0 additions & 3 deletions src/eth_rpc/servers/kakarot_rpc.rs
Original file line number Diff line number Diff line change
@@ -1,7 +1,6 @@
use crate::{
config::KakarotRpcConfig,
eth_rpc::api::kakarot_api::KakarotApiServer,
pool::{get_retry_tx_interval, get_transaction_max_retries},
providers::eth_provider::{
constant::{Constant, MAX_LOGS},
error::{EthApiError, EthereumDataFormatError, SignatureError},
Expand Down Expand Up @@ -100,8 +99,6 @@ where
Ok(Constant {
max_logs: *MAX_LOGS,
starknet_network: String::from(starknet_config.network_url),
retry_tx_interval: get_retry_tx_interval(),
transaction_max_retries: get_transaction_max_retries(),
max_felts_in_calldata: *MAX_FELTS_IN_CALLDATA,
white_listed_eip_155_transaction_hashes: get_white_listed_eip_155_transaction_hashes(),
})
Expand Down
1 change: 1 addition & 0 deletions src/lib.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
#![cfg_attr(not(any(test, feature = "testing")), warn(unused_crate_dependencies))]
use opentelemetry as _;
use opentelemetry_otlp as _;
use opentelemetry_sdk as _;
use tracing_opentelemetry as _;
Expand Down
5 changes: 0 additions & 5 deletions src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@ use kakarot_rpc::{
client::EthClient,
config::KakarotRpcConfig,
eth_rpc::{config::RPCConfig, rpc::KakarotRpcModuleBuilder, run_server},
pool::RetryHandler,
providers::eth_provider::database::Database,
};
use mongodb::options::{DatabaseOptions, ReadConcern, WriteConcern};
Expand Down Expand Up @@ -50,10 +49,6 @@ async fn main() -> Result<()> {

let eth_client = EthClient::try_new(starknet_provider, db.clone()).await.expect("failed to start ethereum client");

// Setup the retry handler
let retry_handler = RetryHandler::new(eth_client.clone(), db);
retry_handler.start(&tokio::runtime::Handle::current());

// Setup the RPC module
let kakarot_rpc_module = KakarotRpcModuleBuilder::new(eth_client).rpc_module()?;

Expand Down
178 changes: 120 additions & 58 deletions src/pool/mempool.rs
Original file line number Diff line number Diff line change
@@ -1,12 +1,23 @@
use super::validate::KakarotTransactionValidator;
use crate::pool::EthClient;
use crate::{
client::EthClient,
into_via_wrapper,
models::felt::Felt252Wrapper,
providers::eth_provider::{
error::ExecutionError,
starknet::{ERC20Reader, STARKNET_NATIVE_TOKEN},
utils::{class_hash_not_declared, contract_not_found},
},
};
use reth_primitives::{BlockId, U256};
use reth_transaction_pool::{
blobstore::NoopBlobStore, CoinbaseTipOrdering, EthPooledTransaction, Pool, TransactionPool,
};
use serde_json::Value;
use starknet::core::types::Felt;
use std::{collections::HashSet, fs::File, io::Read, time::Duration};
use tokio::runtime::Handle;
use std::{collections::HashMap, fs::File, io::Read, sync::Arc, time::Duration};
use tokio::{runtime::Handle, sync::Mutex};
use tracing::Instrument;

/// A type alias for the Kakarot Transaction Validator.
/// Uses the Reth implementation [`TransactionValidationTaskExecutor`].
Expand All @@ -19,97 +30,148 @@ pub type TransactionOrdering = CoinbaseTipOrdering<EthPooledTransaction>;
/// A type alias for the Kakarot Sequencer Mempool.
pub type KakarotPool<Client> = Pool<Validator<Client>, TransactionOrdering, NoopBlobStore>;

#[derive(Debug, Default)]
pub struct AccountManager {
accounts: HashSet<Felt>,
/// Manages a collection of accounts and their associated nonces, interfacing with an Ethereum client.
///
/// This struct provides functionality to initialize account data from a file, monitor account balances,
/// and process transactions for accounts with sufficient balance.
#[derive(Debug)]
pub struct AccountManager<SP: starknet::providers::Provider + Send + Sync + Clone + 'static> {
/// A shared, mutable collection of accounts and their nonces.
accounts: Arc<Mutex<HashMap<Felt, Felt>>>,
/// The Ethereum client used to interact with the blockchain.
eth_client: Arc<EthClient<SP>>,
}

impl AccountManager {
pub fn new(path: &str) -> Self {
let mut accounts = HashSet::new();
impl<SP: starknet::providers::Provider + Send + Sync + Clone + 'static> AccountManager<SP> {
/// Creates a new [`AccountManager`] instance by initializing account data from a JSON file.
pub async fn new(path: &str, eth_client: Arc<EthClient<SP>>) -> eyre::Result<Self> {
let mut accounts = HashMap::new();

// Open the file specified by `path`
let Ok(mut file) = File::open(path) else {
return Self::default();
};
let mut file = File::open(path)?;

let mut contents = String::new();
if file.read_to_string(&mut contents).is_err() {
return Self::default();
}
file.read_to_string(&mut contents)?;

// Parse the file contents as JSON
let json: Value = match serde_json::from_str(&contents) {
Ok(json) => json,
Err(_) => {
return Self::default();
}
};
let json: Value = serde_json::from_str(&contents)?;

// Extract the account addresses from the JSON array
if let Some(array) = json.as_array() {
for item in array {
if let Some(address) = item.as_str() {
accounts.insert(Felt::from_hex_unchecked(address));
if let Some(account_address) = item.as_str() {
let felt_address = Felt::from_hex_unchecked(account_address);

let starknet_block_id = eth_client
.eth_provider()
.to_starknet_block_id(Some(BlockId::default()))
.await
.map_err(|e| eyre::eyre!("Error converting block ID: {:?}", e))?;

// Query the initial account_nonce for the account from the provider
accounts.insert(
felt_address,
eth_client
.eth_provider()
.starknet_provider()
.get_nonce(starknet_block_id, felt_address)
.await
.unwrap_or_default(),
);
}
}
}

Self { accounts }
if accounts.is_empty() {
return Err(eyre::eyre!("No accounts found in file"));
}

Ok(Self { accounts: Arc::new(Mutex::new(accounts)), eth_client })
}

pub fn start<SP>(&self, rt_handle: &Handle, eth_client: &'static EthClient<SP>)
where
SP: starknet::providers::Provider + Send + Sync + Clone + 'static,
{
/// Starts the account manager task that periodically checks account balances and processes transactions.
#[allow(clippy::significant_drop_tightening)]
pub fn start(&'static self, rt_handle: &Handle) {
let accounts = self.accounts.clone();

rt_handle.spawn(async move {
loop {
for address in &accounts {
Self::process_transaction(address, eth_client);
// Get account addresses first without acquiring the lock
let account_addresses: Vec<Felt> = {
let accounts = accounts.lock().await;
accounts.keys().copied().collect()
};

// Iterate over account addresses and check balances
for account_address in account_addresses {
// Fetch the balance and handle errors functionally
let balance = self
.get_balance(&account_address)
.await
.inspect_err(|err| {
tracing::error!(
"Error getting balance for account_address {:?}: {:?}",
account_address,
err
);
})
.unwrap_or_default();

if balance > U256::from(u128::pow(10, 18)) {
// Acquire lock only when necessary to modify account state
let mut accounts = accounts.lock().await;
if let Some(account_nonce) = accounts.get_mut(&account_address) {
self.process_transaction(&account_address, account_nonce);
}
}
}

tokio::time::sleep(Duration::from_secs(1)).await;
}
});
}

fn process_transaction<SP>(address: &Felt, eth_client: &EthClient<SP>)
where
SP: starknet::providers::Provider + Send + Sync + Clone + 'static,
{
let balance = Self::check_balance(address);
/// Retrieves the balance of the specified account address.
async fn get_balance(&self, account_address: &Felt) -> eyre::Result<U256> {
// Convert the optional Ethereum block ID to a Starknet block ID.
let starknet_block_id = self.eth_client.eth_provider().to_starknet_block_id(Some(BlockId::default())).await?;

if balance > Felt::ONE {
let best_hashes = eth_client.mempool().as_ref().best_transactions().map(|x| *x.hash()).collect::<Vec<_>>();
// Create a new `ERC20Reader` instance for the Starknet native token
let eth_contract = ERC20Reader::new(*STARKNET_NATIVE_TOKEN, self.eth_client.eth_provider().starknet_provider());

if let Some(best_hash) = best_hashes.first() {
eth_client.mempool().as_ref().remove_transactions(vec![*best_hash]);
}
// Call the `balanceOf` method on the contract for the given account_address and block ID, awaiting the result
let span = tracing::span!(tracing::Level::INFO, "sn::balance");
let res = eth_contract.balanceOf(account_address).block_id(starknet_block_id).call().instrument(span).await;

if contract_not_found(&res) || class_hash_not_declared(&res) {
return Err(eyre::eyre!("Contract not found or class hash not declared"));
}
}

const fn check_balance(_address: &Felt) -> Felt {
Felt::ONE
}
}
// Otherwise, extract the balance from the result, converting any errors to ExecutionError
let balance = res.map_err(ExecutionError::from)?.balance;

#[cfg(test)]
mod tests {
use super::*;
// Convert the low and high parts of the balance to U256
let low: U256 = into_via_wrapper!(balance.low);
let high: U256 = into_via_wrapper!(balance.high);

#[test]
fn test_account_manager_new() {
let account_manager = AccountManager::new("src/pool/accounts.json");
// Combine the low and high parts to form the final balance and return it
let balance = low + (high << 128);

let accounts = account_manager.accounts;
Ok(balance)
}

assert!(accounts
.contains(&Felt::from_hex_unchecked("0x00686735619287df0f11ec4cda22675f780886b52bf59cf899dd57fd5d5f4cad")));
assert!(accounts
.contains(&Felt::from_hex_unchecked("0x0332825a42ccbec3e2ceb6c242f4dff4682e7d16b8559104b5df8fd925ddda09")));
assert!(accounts
.contains(&Felt::from_hex_unchecked("0x003f5628053c2d6bdfc9e45ea8aeb14405b8917226d455a94b3225a9a7520559")));
/// Processes a transaction for the given account if the balance is sufficient.
fn process_transaction(&self, _account_address: &Felt, account_nonce: &mut Felt)
where
SP: starknet::providers::Provider + Send + Sync + Clone + 'static,
{
let best_hashes = self.eth_client.mempool().as_ref().best_transactions().map(|x| *x.hash()).collect::<Vec<_>>();

if let Some(best_hash) = best_hashes.first() {
self.eth_client.mempool().as_ref().remove_transactions(vec![*best_hash]);

// Increment account_nonce after sending a transaction
*account_nonce = *account_nonce + 1;
}
}
}
Loading

0 comments on commit a4ecc08

Please sign in to comment.