Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(trezor): add withdraw eth with trezor #2005

Merged
merged 62 commits into from
Mar 8, 2024
Merged
Show file tree
Hide file tree
Changes from 58 commits
Commits
Show all changes
62 commits
Select commit Hold shift + click to select a range
f30842b
Original source for UDP transport to connect to Trezor emulator
dimxy Oct 10, 2023
76c8cc9
added modifications to udp.rs to integrate it
dimxy Oct 29, 2023
d47a43f
add tests for withdraw from witness and p2pkh with trezor, add trezor…
dimxy Oct 29, 2023
a76b189
refactor trezor ConnectableDeviceWrapper (use borrow instead of consu…
dimxy Oct 18, 2023
8e56aa9
add support for trezor evm initialisation with rpc task manager and w…
dimxy Nov 12, 2023
2d2894a
fix fmt
dimxy Nov 13, 2023
599e60a
remove unused commented code
dimxy Nov 13, 2023
9a39110
add Result to pubkey_from_xpub_str
dimxy Nov 13, 2023
1d8df3c
dispose extra chain_id replay protection, fix trimming zeros for trez…
dimxy Nov 15, 2023
aed8ef8
change amount in trezor test
dimxy Nov 15, 2023
da9c6a5
add error check of signature v created by trezor
dimxy Nov 15, 2023
719bff0
fix chunk length calc for trezor eth tx request
dimxy Nov 16, 2023
1b7993a
fix target all for bip32 lib for trezor
dimxy Nov 16, 2023
b0e2f58
add trezor eth network definitions structs
dimxy Nov 17, 2023
b4ccf3f
fix deadcode clippy
dimxy Nov 26, 2023
c9a6f5c
fix clippy errors
dimxy Nov 26, 2023
1c9fa22
add pin input in test trezor init loop, remove unsupported trezor tests
dimxy Nov 26, 2023
d17e605
move trezor tests in a module
dimxy Nov 27, 2023
7cbc46e
fix wasm build
dimxy Nov 27, 2023
958617c
added: init_enable_eth rpc set, check to disable init eth trezor in n…
dimxy Dec 8, 2023
e0a4565
fix fmt
dimxy Dec 10, 2023
da421b4
remove unused comments
dimxy Dec 13, 2023
bd4ec47
refactor xpub extraction by checking coin type
dimxy Dec 13, 2023
00aa339
add test code to try create new eth acc with trezor
dimxy Dec 13, 2023
10bfef1
fix eth gap_limit
dimxy Jan 18, 2024
4988565
fix getting address for derivation_path in eth withdraw for trezor
dimxy Jan 18, 2024
56fba06
allow eth withdraw from default account (no 'from' set) for trezor
dimxy Jan 18, 2024
3004711
fix clippy warnings
dimxy Jan 18, 2024
134c2e9
fix fmt
dimxy Jan 18, 2024
ddcb180
change get_enabled_address() to return HDAddress
dimxy Jan 21, 2024
1569184
change eth withdraw build to use cached from address (instead getting…
dimxy Jan 21, 2024
c7e5971
fix eth withdraw wasm build of refactored code
dimxy Jan 21, 2024
ff45c04
fix fmt
dimxy Jan 21, 2024
3925fe5
fix eth withdraw test mock
dimxy Jan 21, 2024
ad4cec8
fix fmt
dimxy Jan 21, 2024
6e664a9
fix eth withdraw test mock
dimxy Jan 22, 2024
055f9be
reorg imports
dimxy Feb 14, 2024
0e845bb
remove extra pub
dimxy Feb 14, 2024
bfe7a29
eliminate activated_pubkey in Trezor privkey policy; fix address for …
dimxy Feb 14, 2024
29b7835
mark unused trezor proto impl function as reserved for future
dimxy Feb 14, 2024
75c75a6
fix fmt
dimxy Feb 14, 2024
b98d380
fix default privkey policy in test helper activation request
dimxy Feb 14, 2024
5896b77
Merge branch 'evm-hd-wallet' into evm-hd-wallet-trezor
dimxy Feb 18, 2024
7d55c88
fix review notes: update signing status, get_public_key() return, sig…
dimxy Feb 20, 2024
950e349
fix fmt
dimxy Feb 22, 2024
049f025
refactor old DerivationMethod struct
dimxy Feb 22, 2024
1f897d7
new and refactor eth tests:
dimxy Feb 22, 2024
7f50a64
Merge remote-tracking branch 'evm-hd-wallet' into evm-hd-wallet-trezor
shamardy Mar 1, 2024
31295f3
Refactor EthPrivKeyBuildPolicy conversion and error handling in eth.rs
shamardy Mar 2, 2024
a4ac2ca
move eth on_generating_transaction after to and from address are checked
shamardy Mar 2, 2024
50e7933
add todo to add trezor to detect_priv_key_policy
shamardy Mar 2, 2024
58ca5f7
Remove JST distributor and related functions from eth/for_tests.rs
shamardy Mar 4, 2024
b520897
fix and remove some attributes
shamardy Mar 4, 2024
df2fa11
minor refactors
shamardy Mar 4, 2024
7481338
Removed the UnexpectedDerivationMethod variant from WithdrawError enu…
shamardy Mar 4, 2024
a0c33fd
Updated the to_response method in DerivationMethod enum in mm2src/coi…
shamardy Mar 4, 2024
c90489d
use for-tests feature for test_create_new_account_init_loop
shamardy Mar 4, 2024
504bd5e
Refactor coin activation statuses names
shamardy Mar 4, 2024
3dad3bf
fix `ETH_NETWORK_DEFS` typo and minor refactors
shamardy Mar 5, 2024
28bbba7
fix task_handle in re enable platform coin
dimxy Mar 5, 2024
097ba56
fix trezor tests for updated rpc_mode param
dimxy Mar 5, 2024
185dbb0
Merge remote-tracking branch 'evm-hd-wallet' into evm-hd-wallet-trezor
shamardy Mar 8, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions mm2src/coins/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ enable-solana = [
]
default = []
run-docker-tests = []
for-tests = []

[lib]
name = "coins"
Expand Down
259 changes: 61 additions & 198 deletions mm2src/coins/eth.rs

Large diffs are not rendered by default.

7 changes: 1 addition & 6 deletions mm2src/coins/eth/eth_hd_wallet.rs
Original file line number Diff line number Diff line change
Expand Up @@ -36,13 +36,8 @@ impl HDWalletCoinOps for EthCoin {
extended_pubkey: &Secp256k1ExtendedPublicKey,
derivation_path: DerivationPath,
) -> HDCoinHDAddress<Self> {
let serialized = extended_pubkey.public_key().serialize_uncompressed();
let mut pubkey = Public::default();
pubkey.as_mut().copy_from_slice(&serialized[1..65]);
drop_mutability!(pubkey);

let pubkey = pubkey_from_extended(extended_pubkey);
let address = public_to_address(&pubkey);

EthHDAddress {
address,
pubkey,
Expand Down
161 changes: 7 additions & 154 deletions mm2src/coins/eth/eth_tests.rs
Original file line number Diff line number Diff line change
@@ -1,13 +1,11 @@
use super::*;
use crate::eth::for_tests::eth_coin_for_test;
#[cfg(not(target_arch = "wasm32"))]
use crate::eth::for_tests::random_eth_coin_for_test;
use crate::{DexFee, IguanaPrivKey};
use common::{block_on, now_sec};
use crypto::privkey::key_pair_from_seed;
#[cfg(not(target_arch = "wasm32"))]
use ethkey::{Generator, Random};
use mm2_core::mm_ctx::{MmArc, MmCtxBuilder};
use mm2_test_helpers::{for_tests::{eth_jst_testnet_conf, eth_testnet_conf, ETH_DEV_NODE, ETH_DEV_NODES,
ETH_DEV_SWAP_CONTRACT, ETH_DEV_TOKEN_CONTRACT, ETH_MAINNET_NODE},
get_passphrase};
use mm2_core::mm_ctx::MmCtxBuilder;
use mm2_test_helpers::for_tests::{ETH_DEV_NODE, ETH_DEV_SWAP_CONTRACT, ETH_DEV_TOKEN_CONTRACT, ETH_MAINNET_NODE};
use mocktopus::mocking::*;

/// The gas price for the tests
Expand All @@ -21,156 +19,11 @@ const GAS_PRICE_APPROXIMATION_ON_TRADE_PREIMAGE: u64 = 53_500_000_000;

const TAKER_PAYMENT_SPEND_SEARCH_INTERVAL: f64 = 1.;

lazy_static! {
static ref ETH_DISTRIBUTOR: EthCoin = eth_distributor();
static ref JST_DISTRIBUTOR: EthCoin = jst_distributor();
static ref MM_CTX: MmArc = MmCtxBuilder::new().into_mm_arc();
}

fn check_sum(addr: &str, expected: &str) {
let actual = checksum_address(addr);
assert_eq!(expected, actual);
}

pub fn eth_distributor() -> EthCoin {
let req = json!({
"method": "enable",
"coin": "ETH",
"urls": ETH_DEV_NODES,
"swap_contract_address": ETH_DEV_SWAP_CONTRACT,
});
let seed = get_passphrase!(".env.client", "ALICE_PASSPHRASE").unwrap();
let keypair = key_pair_from_seed(&seed).unwrap();
let priv_key_policy = PrivKeyBuildPolicy::IguanaPrivKey(keypair.private().secret);
block_on(eth_coin_from_conf_and_request(
&MM_CTX,
"ETH",
&eth_testnet_conf(),
&req,
CoinProtocol::ETH,
priv_key_policy,
))
.unwrap()
}

pub fn jst_distributor() -> EthCoin {
let req = json!({
"method": "enable",
"coin": "ETH",
"urls": ETH_DEV_NODES,
"swap_contract_address": ETH_DEV_SWAP_CONTRACT,
});
let seed = get_passphrase!(".env.seed", "BOB_PASSPHRASE").unwrap();
let keypair = key_pair_from_seed(&seed).unwrap();
let priv_key_policy = PrivKeyBuildPolicy::IguanaPrivKey(keypair.private().secret);
block_on(eth_coin_from_conf_and_request(
&MM_CTX,
"ETH",
&eth_testnet_conf(),
&req,
CoinProtocol::ERC20 {
platform: "ETH".to_string(),
contract_address: ETH_DEV_TOKEN_CONTRACT.to_string(),
},
priv_key_policy,
))
.unwrap()
}

fn eth_coin_for_test(
coin_type: EthCoinType,
urls: &[&str],
fallback_swap_contract: Option<Address>,
) -> (MmArc, EthCoin) {
let key_pair = KeyPair::from_secret_slice(
&hex::decode("809465b17d0a4ddb3e4c69e8f23c2cabad868f51f8bed5c765ad1d6516c3306f").unwrap(),
)
.unwrap();
eth_coin_from_keypair(coin_type, urls, fallback_swap_contract, key_pair)
}

#[cfg(not(target_arch = "wasm32"))]
fn random_eth_coin_for_test(
coin_type: EthCoinType,
urls: &[&str],
fallback_swap_contract: Option<Address>,
) -> (MmArc, EthCoin) {
let key_pair = Random.generate().unwrap();
fill_eth(key_pair.address(), 0.001);
eth_coin_from_keypair(coin_type, urls, fallback_swap_contract, key_pair)
}

fn eth_coin_from_keypair(
coin_type: EthCoinType,
urls: &[&str],
fallback_swap_contract: Option<Address>,
key_pair: KeyPair,
) -> (MmArc, EthCoin) {
let mut web3_instances = vec![];
for url in urls.iter() {
let node = HttpTransportNode {
uri: url.parse().unwrap(),
gui_auth: false,
};

let transport = Web3Transport::new_http(node);
let web3 = Web3::new(transport);

web3_instances.push(Web3Instance { web3, is_parity: false });
}

drop_mutability!(web3_instances);

let conf = json!({
"coins":[
eth_testnet_conf(),
eth_jst_testnet_conf()
]
});
let ctx = MmCtxBuilder::new().with_conf(conf).into_mm_arc();
let ticker = match coin_type {
EthCoinType::Eth => "ETH".to_string(),
EthCoinType::Erc20 { .. } => "JST".to_string(),
};
let my_address = key_pair.address();

let eth_coin = EthCoin(Arc::new(EthCoinImpl {
coin_type,
decimals: 18,
gas_station_url: None,
gas_station_decimals: ETH_GAS_STATION_DECIMALS,
history_sync_state: Mutex::new(HistorySyncState::NotEnabled),
gas_station_policy: GasStationPricePolicy::MeanAverageFast,
sign_message_prefix: Some(String::from("Ethereum Signed Message:\n")),
priv_key_policy: key_pair.into(),
derivation_method: Arc::new(DerivationMethod::SingleAddress(my_address)),
swap_contract_address: Address::from_str(ETH_DEV_SWAP_CONTRACT).unwrap(),
fallback_swap_contract,
contract_supports_watchers: false,
ticker,
web3_instances: AsyncMutex::new(web3_instances),
ctx: ctx.weak(),
required_confirmations: 1.into(),
chain_id: None,
trezor_coin: None,
logs_block_range: DEFAULT_LOGS_BLOCK_RANGE,
nonce_lock: new_nonce_lock(),
erc20_tokens_infos: Default::default(),
abortable_system: AbortableQueue::default(),
}));
(ctx, eth_coin)
}

#[cfg(not(target_arch = "wasm32"))]
pub fn fill_eth(to_addr: Address, amount: f64) {
let wei_per_eth: u64 = 1_000_000_000_000_000_000;
let amount_in_wei = (amount * wei_per_eth as f64) as u64;
ETH_DISTRIBUTOR
.send_to_address(to_addr, amount_in_wei.into())
.wait()
.unwrap();
}

#[test]
/// https://github.com/ethereum/EIPs/blob/master/EIPS/eip-55.md#test-cases
fn test_check_sum_address() {
Expand Down Expand Up @@ -493,7 +346,7 @@ fn test_gas_station() {
fn test_withdraw_impl_manual_fee() {
let (_ctx, coin) = eth_coin_for_test(EthCoinType::Eth, &["http://dummy.dummy"], None);

EthCoin::my_balance.mock_safe(|_| {
EthCoin::address_balance.mock_safe(|_, _| {
let balance = wei_from_big_decimal(&1000000000.into(), 18).unwrap();
MockResult::Return(Box::new(futures01::future::ok(balance)))
});
Expand Down Expand Up @@ -538,7 +391,7 @@ fn test_withdraw_impl_fee_details() {
None,
);

EthCoin::my_balance.mock_safe(|_| {
EthCoin::address_balance.mock_safe(|_, _| {
let balance = wei_from_big_decimal(&1000000000.into(), 18).unwrap();
MockResult::Return(Box::new(futures01::future::ok(balance)))
});
Expand Down
27 changes: 26 additions & 1 deletion mm2src/coins/eth/eth_wasm_tests.rs
Original file line number Diff line number Diff line change
@@ -1,9 +1,11 @@
use super::*;
use crate::eth::for_tests::eth_coin_for_test;
use crate::lp_coininit;
use crypto::privkey::key_pair_from_seed;
use crypto::CryptoCtx;
use futures::compat::Future01CompatExt;
use mm2_core::mm_ctx::MmCtxBuilder;
use mm2_test_helpers::for_tests::{ETH_DEV_NODE, ETH_DEV_SWAP_CONTRACT};
use mm2_test_helpers::for_tests::{ETH_DEV_NODE, ETH_DEV_SWAP_CONTRACT, ETH_SEPOLIA_NODES};
use mm2_test_helpers::get_passphrase;
use wasm_bindgen_test::*;
use web_sys::console;
Expand Down Expand Up @@ -121,3 +123,26 @@ async fn wasm_test_sign_eth_tx() {
console::log_1(&format!("res={:?}", res).into());
assert!(res.is_ok());
}

#[wasm_bindgen_test]
async fn wasm_test_withdraw_eth() {
// we need to hold ref to _ctx until the end of the test (because of the weak ref to MmCtx in EthCoinImpl)
let (_ctx, coin) = eth_coin_for_test(EthCoinType::Eth, ETH_SEPOLIA_NODES, None); // use sepolia as usualy some amount exists there

let withdraw_req = WithdrawRequest {
amount: "0.00001".parse().unwrap(),
from: None,
to: "0x7Bc1bBDD6A0a722fC9bffC49c921B685ECB84b94".to_string(),
coin: "ETH".to_string(),
max: false,
fee: Some(WithdrawFee::EthGas {
gas: ETH_GAS,
gas_price: 1.into(),
}),
memo: None,
broadcast: false,
};
let res = coin.withdraw(withdraw_req).compat().await;
console::log_1(&format!("res={:?}", res).into());
assert!(res.is_ok());
}
Loading
Loading