mirror of
https://github.com/comit-network/xmr-btc-swap.git
synced 2025-12-01 13:04:48 -05:00
* feat(wallet): Use mempool.space as a secondary fee estimation source * fix: warn if mempool client cannot be instantiated * make clippy happy * nitpick: rename clippy_check to clippy in justfile * rename `estimate_fee_rate_from_mempool` to `estimate_fee_rate_from_histogram` for clarity * dprint fmt * make clippy happy * change teacing level back to debug! * change log levels * refactors * refactor: estimate_fee and min_relay_fee * serde camel case Co-authored-by: Byron Hambly <byron@hambly.dev> * refactors * Add comments, use Weight struct where possible * fmt, fix testrs * dont fallback to bitcoin::MAX, fail instead * make mempool space optional * fmt * refactor: use estimate_fee(...) in max_giveable(...) * refactor max_giveable(...) * refactor max_giveeable to return fee as well, remove safety margin for fee * fix compile * fmtr * fix(integration test): Use pre-calculated cancel / punish fees for assert_alice_punished * fix(integration test): Use real fees for asserts * sync wallet before transaction_fee call * split send_to_address into sweep_balance_to_address_dynamic_fee --------- Co-authored-by: Byron Hambly <byron@hambly.dev>
432 lines
17 KiB
Rust
432 lines
17 KiB
Rust
#![warn(
|
|
unused_extern_crates,
|
|
missing_copy_implementations,
|
|
clippy::cast_possible_truncation,
|
|
clippy::cast_sign_loss,
|
|
clippy::fallible_impl_from,
|
|
clippy::cast_precision_loss,
|
|
clippy::cast_possible_wrap,
|
|
clippy::dbg_macro
|
|
)]
|
|
#![forbid(unsafe_code)]
|
|
#![allow(non_snake_case)]
|
|
|
|
use anyhow::Result;
|
|
use std::env;
|
|
use swap::cli::command::{parse_args_and_apply_defaults, ParseResult};
|
|
|
|
#[tokio::main]
|
|
pub async fn main() -> Result<()> {
|
|
rustls::crypto::ring::default_provider()
|
|
.install_default()
|
|
.expect("failed to install default rustls provider");
|
|
|
|
match parse_args_and_apply_defaults(env::args_os()).await? {
|
|
ParseResult::Success(context) => {
|
|
context.tasks.wait_for_tasks().await?;
|
|
}
|
|
ParseResult::PrintAndExitZero { message } => {
|
|
println!("{}", message);
|
|
std::process::exit(0);
|
|
}
|
|
};
|
|
|
|
Ok(())
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
use ::bitcoin::Amount;
|
|
use bitcoin::address::NetworkUnchecked;
|
|
use bitcoin::Address;
|
|
use std::sync::{Arc, Mutex};
|
|
use std::time::Duration;
|
|
use swap::cli::api::request::determine_btc_to_swap;
|
|
use swap::network::quote::BidQuote;
|
|
use swap::tracing_ext::capture_logs;
|
|
use tracing::level_filters::LevelFilter;
|
|
|
|
#[tokio::test]
|
|
async fn given_no_balance_and_transfers_less_than_max_swaps_max_giveable() {
|
|
let writer = capture_logs(LevelFilter::INFO);
|
|
let givable = Arc::new(Mutex::new(MaxGiveable::new(vec![
|
|
(Amount::ZERO, Amount::from_sat(1000)),
|
|
(Amount::from_btc(0.0009).unwrap(), Amount::from_sat(1000)),
|
|
])));
|
|
|
|
let (amount, fees) = determine_btc_to_swap(
|
|
true,
|
|
quote_with_max(0.01),
|
|
get_dummy_address(),
|
|
|| async { Ok(Amount::from_btc(0.001)?) },
|
|
|| async {
|
|
let mut result = givable.lock().unwrap();
|
|
result.give()
|
|
},
|
|
|| async { Ok(()) },
|
|
None,
|
|
None,
|
|
)
|
|
.await
|
|
.unwrap();
|
|
|
|
let expected_amount = Amount::from_btc(0.0009).unwrap();
|
|
let expected_fees = Amount::from_btc(0.0001).unwrap();
|
|
|
|
assert_eq!((amount, fees), (expected_amount, expected_fees));
|
|
assert_eq!(
|
|
writer.captured(),
|
|
r" INFO swap::cli::api::request: Received quote price=0.00100000 BTC minimum_amount=0 BTC maximum_amount=0.01000000 BTC
|
|
INFO swap::cli::api::request: Deposit at least 0.00001000 BTC to cover the min quantity with fee!
|
|
INFO swap::cli::api::request: Waiting for Bitcoin deposit deposit_address=1PdfytjS7C8wwd9Lq5o4x9aXA2YRqaCpH6 min_deposit_until_swap_will_start=0.00001000 BTC max_deposit_until_maximum_amount_is_reached=0.01001000 BTC max_giveable=0 BTC minimum_amount=0 BTC maximum_amount=0.01000000 BTC min_bitcoin_lock_tx_fee=0.00001000 BTC price=0.00100000 BTC
|
|
INFO swap::cli::api::request: Received Bitcoin new_balance=0.00100000 BTC max_giveable=0.00090000 BTC
|
|
"
|
|
);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn given_no_balance_and_transfers_more_then_swaps_max_quantity_from_quote() {
|
|
let writer = capture_logs(LevelFilter::INFO);
|
|
let givable = Arc::new(Mutex::new(MaxGiveable::new(vec![
|
|
(Amount::ZERO, Amount::from_sat(1000)),
|
|
(Amount::from_btc(0.1).unwrap(), Amount::from_sat(1000)),
|
|
])));
|
|
|
|
let (amount, fees) = determine_btc_to_swap(
|
|
true,
|
|
quote_with_max(0.01),
|
|
get_dummy_address(),
|
|
|| async { Ok(Amount::from_btc(0.1001)?) },
|
|
|| async {
|
|
let mut result = givable.lock().unwrap();
|
|
result.give()
|
|
},
|
|
|| async { Ok(()) },
|
|
None,
|
|
None,
|
|
)
|
|
.await
|
|
.unwrap();
|
|
|
|
let expected_amount = Amount::from_btc(0.01).unwrap();
|
|
let expected_fees = Amount::from_btc(0.0001).unwrap();
|
|
|
|
assert_eq!((amount, fees), (expected_amount, expected_fees));
|
|
assert_eq!(
|
|
writer.captured(),
|
|
r" INFO swap::cli::api::request: Received quote price=0.00100000 BTC minimum_amount=0 BTC maximum_amount=0.01000000 BTC
|
|
INFO swap::cli::api::request: Deposit at least 0.00001000 BTC to cover the min quantity with fee!
|
|
INFO swap::cli::api::request: Waiting for Bitcoin deposit deposit_address=1PdfytjS7C8wwd9Lq5o4x9aXA2YRqaCpH6 min_deposit_until_swap_will_start=0.00001000 BTC max_deposit_until_maximum_amount_is_reached=0.01001000 BTC max_giveable=0 BTC minimum_amount=0 BTC maximum_amount=0.01000000 BTC min_bitcoin_lock_tx_fee=0.00001000 BTC price=0.00100000 BTC
|
|
INFO swap::cli::api::request: Received Bitcoin new_balance=0.10010000 BTC max_giveable=0.10000000 BTC
|
|
"
|
|
);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn given_initial_balance_below_max_quantity_swaps_max_giveable() {
|
|
let writer = capture_logs(LevelFilter::INFO);
|
|
let givable = Arc::new(Mutex::new(MaxGiveable::new(vec![
|
|
(Amount::from_btc(0.0049).unwrap(), Amount::from_sat(1000)),
|
|
(Amount::from_btc(99.9).unwrap(), Amount::from_sat(1000)),
|
|
])));
|
|
|
|
let (amount, fees) = determine_btc_to_swap(
|
|
true,
|
|
quote_with_max(0.01),
|
|
async { panic!("should not request new address when initial balance is > 0") },
|
|
|| async { Ok(Amount::from_btc(0.005)?) },
|
|
|| async {
|
|
let mut result = givable.lock().unwrap();
|
|
result.give()
|
|
},
|
|
|| async { Ok(()) },
|
|
None,
|
|
None,
|
|
)
|
|
.await
|
|
.unwrap();
|
|
|
|
let expected_amount = Amount::from_btc(0.0049).unwrap();
|
|
let expected_fees = Amount::from_btc(0.0001).unwrap();
|
|
|
|
assert_eq!((amount, fees), (expected_amount, expected_fees));
|
|
assert_eq!(
|
|
writer.captured(),
|
|
" INFO swap::cli::api::request: Received quote price=0.00100000 BTC minimum_amount=0 BTC maximum_amount=0.01000000 BTC\n"
|
|
);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn given_initial_balance_above_max_quantity_swaps_max_quantity() {
|
|
let writer = capture_logs(LevelFilter::INFO);
|
|
let givable = Arc::new(Mutex::new(MaxGiveable::new(vec![
|
|
(Amount::from_btc(0.1).unwrap(), Amount::from_sat(1000)),
|
|
(Amount::from_btc(99.9).unwrap(), Amount::from_sat(1000)),
|
|
])));
|
|
|
|
let (amount, fees) = determine_btc_to_swap(
|
|
true,
|
|
quote_with_max(0.01),
|
|
async { panic!("should not request new address when initial balance is > 0") },
|
|
|| async { Ok(Amount::from_btc(0.1001)?) },
|
|
|| async {
|
|
let mut result = givable.lock().unwrap();
|
|
result.give()
|
|
},
|
|
|| async { Ok(()) },
|
|
None,
|
|
None,
|
|
)
|
|
.await
|
|
.unwrap();
|
|
|
|
let expected_amount = Amount::from_btc(0.01).unwrap();
|
|
let expected_fees = Amount::from_btc(0.0001).unwrap();
|
|
|
|
assert_eq!((amount, fees), (expected_amount, expected_fees));
|
|
assert_eq!(
|
|
writer.captured(),
|
|
" INFO swap::cli::api::request: Received quote price=0.00100000 BTC minimum_amount=0 BTC maximum_amount=0.01000000 BTC\n"
|
|
);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn given_no_initial_balance_then_min_wait_for_sufficient_deposit() {
|
|
let writer = capture_logs(LevelFilter::INFO);
|
|
let givable = Arc::new(Mutex::new(MaxGiveable::new(vec![
|
|
(Amount::ZERO, Amount::from_sat(1000)),
|
|
(Amount::from_btc(0.01).unwrap(), Amount::from_sat(1000)),
|
|
])));
|
|
|
|
let (amount, fees) = determine_btc_to_swap(
|
|
true,
|
|
quote_with_min(0.01),
|
|
get_dummy_address(),
|
|
|| async { Ok(Amount::from_btc(0.0101)?) },
|
|
|| async {
|
|
let mut result = givable.lock().unwrap();
|
|
result.give()
|
|
},
|
|
|| async { Ok(()) },
|
|
None,
|
|
None,
|
|
)
|
|
.await
|
|
.unwrap();
|
|
|
|
let expected_amount = Amount::from_btc(0.01).unwrap();
|
|
let expected_fees = Amount::from_btc(0.0001).unwrap();
|
|
|
|
assert_eq!((amount, fees), (expected_amount, expected_fees));
|
|
assert_eq!(
|
|
writer.captured(),
|
|
r" INFO swap::cli::api::request: Received quote price=0.00100000 BTC minimum_amount=0.01000000 BTC maximum_amount=21000000 BTC
|
|
INFO swap::cli::api::request: Deposit at least 0.01001000 BTC to cover the min quantity with fee!
|
|
INFO swap::cli::api::request: Waiting for Bitcoin deposit deposit_address=1PdfytjS7C8wwd9Lq5o4x9aXA2YRqaCpH6 min_deposit_until_swap_will_start=0.01001000 BTC max_deposit_until_maximum_amount_is_reached=21000000.00001000 BTC max_giveable=0 BTC minimum_amount=0.01000000 BTC maximum_amount=21000000 BTC min_bitcoin_lock_tx_fee=0.00001000 BTC price=0.00100000 BTC
|
|
INFO swap::cli::api::request: Received Bitcoin new_balance=0.01010000 BTC max_giveable=0.01000000 BTC
|
|
"
|
|
);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn given_balance_less_then_min_wait_for_sufficient_deposit() {
|
|
let writer = capture_logs(LevelFilter::INFO);
|
|
let givable = Arc::new(Mutex::new(MaxGiveable::new(vec![
|
|
(Amount::from_btc(0.0001).unwrap(), Amount::from_sat(1000)),
|
|
(Amount::from_btc(0.01).unwrap(), Amount::from_sat(1000)),
|
|
])));
|
|
|
|
let (amount, fees) = determine_btc_to_swap(
|
|
true,
|
|
quote_with_min(0.01),
|
|
get_dummy_address(),
|
|
|| async { Ok(Amount::from_btc(0.0101)?) },
|
|
|| async {
|
|
let mut result = givable.lock().unwrap();
|
|
result.give()
|
|
},
|
|
|| async { Ok(()) },
|
|
None,
|
|
None,
|
|
)
|
|
.await
|
|
.unwrap();
|
|
|
|
let expected_amount = Amount::from_btc(0.01).unwrap();
|
|
let expected_fees = Amount::from_btc(0.0001).unwrap();
|
|
|
|
assert_eq!((amount, fees), (expected_amount, expected_fees));
|
|
assert_eq!(
|
|
writer.captured(),
|
|
r" INFO swap::cli::api::request: Received quote price=0.00100000 BTC minimum_amount=0.01000000 BTC maximum_amount=21000000 BTC
|
|
INFO swap::cli::api::request: Deposit at least 0.00991000 BTC to cover the min quantity with fee!
|
|
INFO swap::cli::api::request: Waiting for Bitcoin deposit deposit_address=1PdfytjS7C8wwd9Lq5o4x9aXA2YRqaCpH6 min_deposit_until_swap_will_start=0.00991000 BTC max_deposit_until_maximum_amount_is_reached=20999999.99991000 BTC max_giveable=0.00010000 BTC minimum_amount=0.01000000 BTC maximum_amount=21000000 BTC min_bitcoin_lock_tx_fee=0.00001000 BTC price=0.00100000 BTC
|
|
INFO swap::cli::api::request: Received Bitcoin new_balance=0.01010000 BTC max_giveable=0.01000000 BTC
|
|
"
|
|
);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn given_no_initial_balance_and_transfers_less_than_min_keep_waiting() {
|
|
let writer = capture_logs(LevelFilter::INFO);
|
|
let givable = Arc::new(Mutex::new(MaxGiveable::new(vec![
|
|
(Amount::ZERO, Amount::from_sat(1000)),
|
|
(Amount::from_btc(0.01).unwrap(), Amount::from_sat(1000)),
|
|
(Amount::from_btc(0.01).unwrap(), Amount::from_sat(1000)),
|
|
(Amount::from_btc(0.01).unwrap(), Amount::from_sat(1000)),
|
|
(Amount::from_btc(0.01).unwrap(), Amount::from_sat(1000)),
|
|
])));
|
|
|
|
let error = tokio::time::timeout(
|
|
Duration::from_secs(1),
|
|
determine_btc_to_swap(
|
|
true,
|
|
quote_with_min(0.1),
|
|
get_dummy_address(),
|
|
|| async { Ok(Amount::from_btc(0.0101)?) },
|
|
|| async {
|
|
let mut result = givable.lock().unwrap();
|
|
result.give()
|
|
},
|
|
|| async { Ok(()) },
|
|
None,
|
|
None,
|
|
),
|
|
)
|
|
.await
|
|
.unwrap_err();
|
|
|
|
assert!(matches!(error, tokio::time::error::Elapsed { .. }));
|
|
assert_eq!(
|
|
writer.captured(),
|
|
r" INFO swap::cli::api::request: Received quote price=0.00100000 BTC minimum_amount=0.10000000 BTC maximum_amount=21000000 BTC
|
|
INFO swap::cli::api::request: Deposit at least 0.10001000 BTC to cover the min quantity with fee!
|
|
INFO swap::cli::api::request: Waiting for Bitcoin deposit deposit_address=1PdfytjS7C8wwd9Lq5o4x9aXA2YRqaCpH6 min_deposit_until_swap_will_start=0.10001000 BTC max_deposit_until_maximum_amount_is_reached=21000000.00001000 BTC max_giveable=0 BTC minimum_amount=0.10000000 BTC maximum_amount=21000000 BTC min_bitcoin_lock_tx_fee=0.00001000 BTC price=0.00100000 BTC
|
|
INFO swap::cli::api::request: Received Bitcoin new_balance=0.01010000 BTC max_giveable=0.01000000 BTC
|
|
INFO swap::cli::api::request: Deposited amount is not enough to cover `min_quantity` when accounting for network fees
|
|
INFO swap::cli::api::request: Deposit at least 0.09001000 BTC to cover the min quantity with fee!
|
|
INFO swap::cli::api::request: Waiting for Bitcoin deposit deposit_address=1PdfytjS7C8wwd9Lq5o4x9aXA2YRqaCpH6 min_deposit_until_swap_will_start=0.09001000 BTC max_deposit_until_maximum_amount_is_reached=20999999.99001000 BTC max_giveable=0.01000000 BTC minimum_amount=0.10000000 BTC maximum_amount=21000000 BTC min_bitcoin_lock_tx_fee=0.00001000 BTC price=0.00100000 BTC
|
|
"
|
|
);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn given_longer_delay_until_deposit_should_not_spam_user() {
|
|
let writer = capture_logs(LevelFilter::INFO);
|
|
let givable = Arc::new(Mutex::new(MaxGiveable::new(vec![
|
|
(Amount::ZERO, Amount::from_sat(1000)),
|
|
(Amount::ZERO, Amount::from_sat(1000)),
|
|
(Amount::ZERO, Amount::from_sat(1000)),
|
|
(Amount::ZERO, Amount::from_sat(1000)),
|
|
(Amount::ZERO, Amount::from_sat(1000)),
|
|
(Amount::ZERO, Amount::from_sat(1000)),
|
|
(Amount::ZERO, Amount::from_sat(1000)),
|
|
(Amount::ZERO, Amount::from_sat(1000)),
|
|
(Amount::ZERO, Amount::from_sat(1000)),
|
|
(Amount::from_btc(0.2).unwrap(), Amount::from_sat(1000)),
|
|
])));
|
|
|
|
tokio::time::timeout(
|
|
Duration::from_secs(10),
|
|
determine_btc_to_swap(
|
|
true,
|
|
quote_with_min(0.1),
|
|
get_dummy_address(),
|
|
|| async { Ok(Amount::from_btc(0.21)?) },
|
|
|| async {
|
|
let mut result = givable.lock().unwrap();
|
|
|
|
result.give()
|
|
},
|
|
|| async { Ok(()) },
|
|
None,
|
|
None,
|
|
),
|
|
)
|
|
.await
|
|
.unwrap()
|
|
.unwrap();
|
|
|
|
assert_eq!(
|
|
writer.captured(),
|
|
r" INFO swap::cli::api::request: Received quote price=0.00100000 BTC minimum_amount=0.10000000 BTC maximum_amount=21000000 BTC
|
|
INFO swap::cli::api::request: Deposit at least 0.10001000 BTC to cover the min quantity with fee!
|
|
INFO swap::cli::api::request: Waiting for Bitcoin deposit deposit_address=1PdfytjS7C8wwd9Lq5o4x9aXA2YRqaCpH6 min_deposit_until_swap_will_start=0.10001000 BTC max_deposit_until_maximum_amount_is_reached=21000000.00001000 BTC max_giveable=0 BTC minimum_amount=0.10000000 BTC maximum_amount=21000000 BTC min_bitcoin_lock_tx_fee=0.00001000 BTC price=0.00100000 BTC
|
|
INFO swap::cli::api::request: Received Bitcoin new_balance=0.21000000 BTC max_giveable=0.20000000 BTC
|
|
"
|
|
);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn given_bid_quote_max_amount_0_return_error() {
|
|
let givable = Arc::new(Mutex::new(MaxGiveable::new(vec![
|
|
(Amount::from_btc(0.0001).unwrap(), Amount::from_sat(1000)),
|
|
(Amount::from_btc(0.01).unwrap(), Amount::from_sat(1000)),
|
|
])));
|
|
|
|
let determination_error = determine_btc_to_swap(
|
|
true,
|
|
quote_with_max(0.00),
|
|
get_dummy_address(),
|
|
|| async { Ok(Amount::from_btc(0.0101)?) },
|
|
|| async {
|
|
let mut result = givable.lock().unwrap();
|
|
result.give()
|
|
},
|
|
|| async { Ok(()) },
|
|
None,
|
|
None,
|
|
)
|
|
.await
|
|
.err()
|
|
.unwrap()
|
|
.to_string();
|
|
|
|
assert_eq!("Received quote of 0", determination_error);
|
|
}
|
|
|
|
struct MaxGiveable {
|
|
amounts: Vec<(Amount, Amount)>,
|
|
call_counter: usize,
|
|
}
|
|
|
|
impl MaxGiveable {
|
|
fn new(amounts: Vec<(Amount, Amount)>) -> Self {
|
|
Self {
|
|
amounts,
|
|
call_counter: 0,
|
|
}
|
|
}
|
|
fn give(&mut self) -> Result<(Amount, Amount)> {
|
|
let amount = self
|
|
.amounts
|
|
.get(self.call_counter)
|
|
.ok_or_else(|| anyhow::anyhow!("No more balances available"))?;
|
|
self.call_counter += 1;
|
|
Ok(*amount)
|
|
}
|
|
}
|
|
|
|
fn quote_with_max(btc: f64) -> BidQuote {
|
|
BidQuote {
|
|
price: Amount::from_btc(0.001).unwrap(),
|
|
max_quantity: Amount::from_btc(btc).unwrap(),
|
|
min_quantity: Amount::ZERO,
|
|
}
|
|
}
|
|
|
|
fn quote_with_min(btc: f64) -> BidQuote {
|
|
BidQuote {
|
|
price: Amount::from_btc(0.001).unwrap(),
|
|
max_quantity: Amount::MAX_MONEY,
|
|
min_quantity: Amount::from_btc(btc).unwrap(),
|
|
}
|
|
}
|
|
|
|
async fn get_dummy_address() -> Result<bitcoin::Address> {
|
|
Ok("1PdfytjS7C8wwd9Lq5o4x9aXA2YRqaCpH6"
|
|
.parse::<Address<NetworkUnchecked>>()?
|
|
.assume_checked())
|
|
}
|
|
}
|