diff --git a/.gitignore b/.gitignore index ea8c4bf7..dac30f6d 100644 --- a/.gitignore +++ b/.gitignore @@ -1 +1,2 @@ -/target +# Rust +/target \ No newline at end of file diff --git a/Cargo.lock b/Cargo.lock index 005b6ef3..02d5e3e2 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -20,6 +20,7 @@ dependencies = [ "async-trait", "bytes", "clap", + "dirs", "educe", "ethereum-interfaces", "ethereum-types 0.14.0", diff --git a/Cargo.toml b/Cargo.toml index 6c051f99..98a03e73 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -11,6 +11,7 @@ anyhow = "1" async-trait = "0.1" bytes = { version = "1", features = ["serde"] } clap = { version = "4", features = ["derive"] } +dirs = "4.0" educe = { version = "0.4", features = ["Debug", "Default"] } ethereum-interfaces = { git = "https://github.com/ledgerwatch/interfaces" } ethereum-types = { version = "0.14", features = ["codec"] } @@ -45,3 +46,7 @@ name = "bundler-uopool" [[bin]] path = "bin/bundler-rpc.rs" name = "bundler-rpc" + +[[bin]] +path = "bin/create-wallet.rs" +name = "create-wallet" diff --git a/Makefile b/Makefile index 321e6425..1d589dac 100644 --- a/Makefile +++ b/Makefile @@ -1,5 +1,5 @@ run-bundler: - cargo run -- --mnemonic-file ./src/res/bundler/0xD00D3EEc454D05d3d9bB48532BabED0c89941f17 + cargo run -- --mnemonic-file ${HOME}/.aa-bundler/0x129D197b2a989C6798601A49D89a4AEC822A17a3 run-bundler-uopool: cargo run --bin bundler-uopool @@ -7,5 +7,8 @@ run-bundler-uopool: run-bundler-rpc: cargo run --bin bundler-rpc +run-create-wallet: + cargo run --bin create-wallet -- --output-path ${HOME}/.aa-bundler + cargo-fmt: cargo fmt --all diff --git a/README.md b/README.md index 01ca7cd4..00be94d3 100644 --- a/README.md +++ b/README.md @@ -12,19 +12,25 @@ For more information: https://hackmd.io/@Vid201/aa-bundler-rust ## How to run? -Bundler: +Create wallet for bundler: ```bash -cargo run -- --mnemonic-folder ./src/res/bundler +cargo run --bin create-wallet -- --output-path ${HOME}/.aa-bundler ``` -User operation pool: +Run bundler (with user operation pool and JSON-RPC API): + +```bash +cargo run -- --mnemonic-file ${HOME}/.aa-bundler/0x129D197b2a989C6798601A49D89a4AEC822A17a3 +``` + +Run only user operation pool: ```bash cargo run --bin bundler-uopool ``` -Bundler RPC: +Run only JSON-RPC API: ```bash cargo run --bin bundler-rpc diff --git a/bin/bundler-rpc.rs b/bin/bundler-rpc.rs index f3a81553..32a96215 100644 --- a/bin/bundler-rpc.rs +++ b/bin/bundler-rpc.rs @@ -9,7 +9,7 @@ use aa_bundler::rpc::{eth::EthApiServerImpl, eth_api::EthApiServer}; #[derive(Parser)] #[clap( name = "aa-bundler-rpc", - about = "RPC server for EIP-4337 Account Abstraction Bundler" + about = "JSON-RPC server for EIP-4337 Account Abstraction Bundler" )] pub struct Opt { #[clap(long, default_value = "127.0.0.1:4337")] @@ -36,7 +36,7 @@ async fn main() -> Result<()> { .unwrap(); let _jsonrpc_server_handle = jsonrpc_server.start(api.clone())?; - info!("JSONRPC server listening on {}", opt.rpc_listen_address); + info!("JSON-RPC server listening on {}", opt.rpc_listen_address); pending().await } diff --git a/bin/bundler.rs b/bin/bundler.rs index 6849d77a..4c3664b5 100644 --- a/bin/bundler.rs +++ b/bin/bundler.rs @@ -1,10 +1,10 @@ -use aa_bundler::rpc::{eth::EthApiServerImpl, eth_api::EthApiServer}; +use aa_bundler::{ + bundler::bundler::Bundler, + models::wallet::Wallet, + rpc::{eth::EthApiServerImpl, eth_api::EthApiServer}, +}; use anyhow::Result; use clap::Parser; -use ethers::{ - prelude::rand, - signers::{coins_bip39::English, MnemonicBuilder}, -}; use expanded_pathbuf::ExpandedPathBuf; use jsonrpsee::{core::server::rpc_module::Methods, server::ServerBuilder, tracing::info}; use std::{future::pending, panic}; @@ -16,10 +16,7 @@ use std::{future::pending, panic}; )] pub struct Opt { #[clap(long)] - pub mnemonic_file: Option, - - #[clap(long, default_value = "./src/res/bundler")] - pub mnemonic_folder: ExpandedPathBuf, + pub mnemonic_file: ExpandedPathBuf, // #[clap(long, default_value = "127.0.0.1:3000")] // pub grpc_listen_address: String, @@ -52,19 +49,10 @@ fn main() -> Result<()> { rt.block_on(async move { info!("Starting AA - Bundler"); - // TODO: move this to bundler package - let wallet = if let Some(mnemonic_file) = opt.mnemonic_file { - MnemonicBuilder::::default() - .phrase(mnemonic_file.to_path_buf()) - .build()? - } else { - let mut rng = rand::thread_rng(); - MnemonicBuilder::::default() - .write_to(opt.mnemonic_folder.to_path_buf()) - .build_random(&mut rng)? - }; + let wallet = Wallet::from_file(opt.mnemonic_file); + info!("{:?}", wallet.signer); - println!("{:?}", wallet); + let bundler = Bundler::new(wallet); if !opt.no_uopool { aa_bundler::uopool::run(opt.uopool_opts).await?; @@ -88,7 +76,7 @@ fn main() -> Result<()> { .unwrap(); let _jsonrpc_server_handle = jsonrpc_server.start(api.clone()).unwrap(); - info!("JSONRPC server listening on {}", opt.rpc_listen_address); + info!("JSON-RPC server listening on {}", opt.rpc_listen_address); pending::<()>().await } diff --git a/bin/create-wallet.rs b/bin/create-wallet.rs new file mode 100644 index 00000000..fd846ac5 --- /dev/null +++ b/bin/create-wallet.rs @@ -0,0 +1,35 @@ +use aa_bundler::models::wallet::Wallet; +use anyhow::Result; +use clap::Parser; +use dirs::home_dir; +use expanded_pathbuf::ExpandedPathBuf; +use jsonrpsee::tracing::info; +use std::str::FromStr; + +#[derive(Parser)] +#[clap( + name = "aa-bundler-create-wallet", + about = "Bundler's wallet creation for EIP-4337 Account Abstraction" +)] +pub struct Opt { + #[clap(long)] + pub output_path: Option, +} + +fn main() -> Result<()> { + let opt: Opt = Opt::parse(); + + tracing_subscriber::fmt::init(); + + let path = if let Some(output_path) = opt.output_path { + output_path + } else { + ExpandedPathBuf::from_str(home_dir().unwrap().join(".aa-bundler").to_str().unwrap()) + .unwrap() + }; + + let wallet = Wallet::new(path); + info!("{:?}", wallet.signer); + + Ok(()) +} diff --git a/src/bundler/bundler.rs b/src/bundler/bundler.rs new file mode 100644 index 00000000..b1e967a3 --- /dev/null +++ b/src/bundler/bundler.rs @@ -0,0 +1,11 @@ +use crate::models::wallet::Wallet; + +pub struct Bundler { + pub wallet: Wallet, +} + +impl Bundler { + pub fn new(wallet: Wallet) -> Self { + Self { wallet } + } +} diff --git a/src/bundler/mod.rs b/src/bundler/mod.rs new file mode 100644 index 00000000..f6b97184 --- /dev/null +++ b/src/bundler/mod.rs @@ -0,0 +1 @@ +pub mod bundler; diff --git a/src/lib.rs b/src/lib.rs index a38f7d4f..d9a8b45e 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -1,3 +1,4 @@ +pub mod bundler; pub mod models; pub mod rpc; pub mod types; diff --git a/src/models/mod.rs b/src/models/mod.rs index ea09c2f7..d6eaecd5 100644 --- a/src/models/mod.rs +++ b/src/models/mod.rs @@ -1 +1,2 @@ pub mod chainspec; +pub mod wallet; diff --git a/src/models/wallet.rs b/src/models/wallet.rs new file mode 100644 index 00000000..1dec1df1 --- /dev/null +++ b/src/models/wallet.rs @@ -0,0 +1,34 @@ +use ethers::{ + prelude::{k256::ecdsa::SigningKey, rand}, + signers::{coins_bip39::English, MnemonicBuilder}, +}; +use expanded_pathbuf::ExpandedPathBuf; +use std::fs; + +pub struct Wallet { + pub signer: ethers::signers::Wallet, +} + +impl Wallet { + pub fn new(output_path: ExpandedPathBuf) -> Self { + let mut rng = rand::thread_rng(); + + fs::create_dir_all(output_path.to_path_buf()).unwrap(); + + Self { + signer: MnemonicBuilder::::default() + .write_to(output_path.to_path_buf()) + .build_random(&mut rng) + .unwrap(), + } + } + + pub fn from_file(input_path: ExpandedPathBuf) -> Self { + Self { + signer: MnemonicBuilder::::default() + .phrase(input_path.to_path_buf()) + .build() + .unwrap(), + } + } +} diff --git a/src/res/bundler/.gitkeep b/src/res/bundler/.gitkeep deleted file mode 100644 index e69de29b..00000000 diff --git a/src/res/bundler/0xD00D3EEc454D05d3d9bB48532BabED0c89941f17 b/src/res/bundler/0xD00D3EEc454D05d3d9bB48532BabED0c89941f17 deleted file mode 100644 index a211a71c..00000000 --- a/src/res/bundler/0xD00D3EEc454D05d3d9bB48532BabED0c89941f17 +++ /dev/null @@ -1 +0,0 @@ -pass erase drink consider keen bid lottery country stool actress sword asthma \ No newline at end of file diff --git a/src/rpc/eth.rs b/src/rpc/eth.rs index c0871f1d..9106216e 100644 --- a/src/rpc/eth.rs +++ b/src/rpc/eth.rs @@ -1,7 +1,7 @@ use crate::{rpc::eth_api::EthApiServer, types::user_operation::UserOperation}; use async_trait::async_trait; use ethereum_types::{Address, U64}; -use jsonrpsee::core::RpcResult; +use jsonrpsee::{core::RpcResult, tracing::info}; pub struct EthApiServerImpl { pub call_gas_limit: u64, @@ -22,8 +22,8 @@ impl EthApiServer for EthApiServerImpl { user_operation: UserOperation, entry_point: Address, ) -> RpcResult { - println!("{:?}", user_operation); - println!("{:?}", entry_point); + info!("{:?}", user_operation); + info!("{:?}", entry_point); Ok(true) } }