Files
breez-sdk-liquid/lib/core/tests/regtest/bitcoin.rs
Daniel Granhão f437c9194e Add regtest end-to-end tests (#816)
* Add regtest end-to-end tests

* Check that regtest environment is set up

* Add todo comment

* Add missing gloo-timers feature

* Link GH issues

* Update DEVELOPMENT.md

* Update boltz submodule

* Fix after rebase
2025-04-03 01:26:21 +01:00

277 lines
8.6 KiB
Rust

use std::time::Duration;
use breez_sdk_liquid::model::{
PayAmount, PaymentDetails, PaymentMethod, PaymentState, PaymentType, PreparePayOnchainRequest,
PrepareReceiveRequest, PrepareRefundRequest, ReceiveAmount, RefundRequest, SdkEvent,
};
use serial_test::serial;
use crate::regtest::{utils, SdkNodeHandle, TIMEOUT};
#[cfg(all(target_family = "wasm", target_os = "unknown"))]
wasm_bindgen_test::wasm_bindgen_test_configure!(run_in_browser);
#[sdk_macros::async_test_all]
#[serial]
async fn bitcoin() {
let mut handle = SdkNodeHandle::init_node().await.unwrap();
// --------------RECEIVE--------------
let payer_amount_sat = 100_000;
let (prepare_response, receive_response) = handle
.receive_payment(&PrepareReceiveRequest {
payment_method: PaymentMethod::BitcoinAddress,
amount: Some(ReceiveAmount::Bitcoin { payer_amount_sat }),
})
.await
.unwrap();
let receiver_amount_sat = payer_amount_sat - prepare_response.fees_sat;
assert!(matches!(
prepare_response.amount.unwrap(),
ReceiveAmount::Bitcoin { payer_amount_sat: amount_sat } if amount_sat == payer_amount_sat
));
assert!(prepare_response.fees_sat > 0);
let bip21 = receive_response.destination;
let address = bip21.split(':').nth(1).unwrap().split('?').next().unwrap();
utils::send_to_address_bitcoind(&address, payer_amount_sat)
.await
.unwrap();
// Confirm user lockup
utils::mine_blocks(1).await.unwrap();
// Wait for swapper to lock up funds
tokio::time::sleep(Duration::from_secs(5)).await;
// Confirm swapper lockup
utils::mine_blocks(1).await.unwrap();
handle
.wait_for_event(
|e| matches!(e, SdkEvent::PaymentWaitingConfirmation { .. }),
TIMEOUT,
)
.await
.unwrap();
assert_eq!(
handle.get_pending_receive_sat().await.unwrap(),
receiver_amount_sat
);
assert_eq!(handle.get_balance_sat().await.unwrap(), 0);
// Confirm claim tx
utils::mine_blocks(1).await.unwrap();
handle
.wait_for_event(|e| matches!(e, SdkEvent::PaymentSucceeded { .. }), TIMEOUT)
.await
.unwrap();
assert_eq!(handle.get_pending_receive_sat().await.unwrap(), 0);
assert_eq!(handle.get_balance_sat().await.unwrap(), receiver_amount_sat);
let payments = handle.get_payments().await.unwrap();
assert_eq!(payments.len(), 1);
let payment = &payments[0];
assert_eq!(payment.amount_sat, receiver_amount_sat);
assert_eq!(payment.fees_sat, prepare_response.fees_sat);
assert_eq!(payment.payment_type, PaymentType::Receive);
assert_eq!(payment.status, PaymentState::Complete);
assert!(matches!(payment.details, PaymentDetails::Bitcoin { .. }));
// --------------SEND--------------
let initial_balance_sat = handle.get_balance_sat().await.unwrap();
let address = utils::generate_address_bitcoind().await.unwrap();
let receiver_amount_sat = 50_000;
let (prepare_response, _) = handle
.send_onchain_payment(
&PreparePayOnchainRequest {
amount: PayAmount::Bitcoin {
receiver_amount_sat,
},
fee_rate_sat_per_vbyte: None,
},
address,
)
.await
.unwrap();
let fees_sat = prepare_response.total_fees_sat;
let sender_amount_sat = receiver_amount_sat + fees_sat;
// Confirm user lockup
utils::mine_blocks(1).await.unwrap();
// Wait for swapper to lock up funds
tokio::time::sleep(Duration::from_secs(5)).await;
// Confirm swapper lockup
utils::mine_blocks(1).await.unwrap();
// Wait for sdk to broadcast claim tx
handle
.wait_for_event(
|e| matches!(e, SdkEvent::PaymentWaitingConfirmation { .. }),
TIMEOUT,
)
.await
.unwrap();
// Confirm claim tx
utils::mine_blocks(1).await.unwrap();
handle
.wait_for_event(|e| matches!(e, SdkEvent::PaymentSucceeded { .. }), TIMEOUT)
.await
.unwrap();
assert_eq!(
handle.get_balance_sat().await.unwrap(),
initial_balance_sat - sender_amount_sat
);
let payments = handle.get_payments().await.unwrap();
assert_eq!(payments.len(), 2);
let payment = &payments[0];
assert_eq!(payment.amount_sat, receiver_amount_sat);
assert_eq!(payment.fees_sat, fees_sat);
assert_eq!(payment.payment_type, PaymentType::Send);
assert_eq!(payment.status, PaymentState::Complete);
assert!(matches!(payment.details, PaymentDetails::Bitcoin { .. }));
// ----------------REFUND--------------
let payer_amount_sat = 100_000;
let lockup_amount_sat = payer_amount_sat - 1;
let (_, receive_response) = handle
.receive_payment(&PrepareReceiveRequest {
payment_method: PaymentMethod::BitcoinAddress,
amount: Some(ReceiveAmount::Bitcoin { payer_amount_sat }),
})
.await
.unwrap();
let bip21 = receive_response.destination;
let address = bip21.split(':').nth(1).unwrap().split('?').next().unwrap();
utils::send_to_address_bitcoind(&address, lockup_amount_sat)
.await
.unwrap();
handle
.wait_for_event(|e| matches!(e, SdkEvent::PaymentRefundable { .. }), TIMEOUT)
.await
.unwrap();
let refundables = handle.sdk.list_refundables().await.unwrap();
assert_eq!(refundables.len(), 1);
let refundable = &refundables[0];
assert_eq!(refundable.amount_sat, lockup_amount_sat);
assert_eq!(refundable.swap_address, address);
assert!(refundable.last_refund_tx_id.is_none());
let refund_address = utils::generate_address_bitcoind().await.unwrap();
let refund_fee_rate = 1;
let refund_rbf_fee_rate = 2;
let refund_response = handle
.sdk
.refund(&RefundRequest {
swap_address: address.to_string(),
refund_address: refund_address.clone(),
fee_rate_sat_per_vbyte: refund_fee_rate,
})
.await
.unwrap();
handle
.wait_for_event(
|e| matches!(e, SdkEvent::PaymentRefundPending { .. }),
TIMEOUT,
)
.await
.unwrap();
let refundables = handle.sdk.list_refundables().await.unwrap();
assert_eq!(refundables.len(), 1);
let refundable = &refundables[0];
assert_eq!(refundable.amount_sat, lockup_amount_sat);
assert_eq!(refundable.swap_address, address);
assert_eq!(
refundable.last_refund_tx_id.as_ref().unwrap(),
&refund_response.refund_tx_id
);
let prepare_refund_rbf_response = handle
.sdk
.prepare_refund(&PrepareRefundRequest {
swap_address: address.to_string(),
refund_address: refund_address.clone(),
fee_rate_sat_per_vbyte: refund_rbf_fee_rate,
})
.await
.unwrap();
let refund_rbf_response = handle
.sdk
.refund(&RefundRequest {
swap_address: address.to_string(),
refund_address,
fee_rate_sat_per_vbyte: refund_rbf_fee_rate,
})
.await
.unwrap();
let refundables = handle.sdk.list_refundables().await.unwrap();
assert_eq!(refundables.len(), 1);
let refundable = &refundables[0];
assert_eq!(refundable.amount_sat, lockup_amount_sat);
assert_eq!(refundable.swap_address, address);
assert_eq!(
refundable.last_refund_tx_id.as_ref().unwrap(),
&refund_rbf_response.refund_tx_id
);
utils::mine_blocks(1).await.unwrap();
handle
.wait_for_event(|e| matches!(e, SdkEvent::PaymentFailed { .. }), TIMEOUT)
.await
.unwrap();
let refundables = handle.sdk.list_refundables().await.unwrap();
assert_eq!(refundables.len(), 0);
let payments = handle.get_payments().await.unwrap();
assert_eq!(payments.len(), 3);
let payment = &payments[0];
assert_eq!(payment.status, PaymentState::Failed);
println!("Payment details: {:?}", payment.details);
println!("Lockup amount sat: {}", lockup_amount_sat);
println!(
"Prepare refund rbf response: {:?}",
prepare_refund_rbf_response
);
// The following fails because the payment's refund_tx_amount_sat is None. Related issue: https://github.com/breez/breez-sdk-liquid/issues/773
// TODO: uncomment once the issue is fixed
/*assert!(matches!(
payment.details,
PaymentDetails::Bitcoin {
refund_tx_amount_sat: Some(refund_tx_amount_sat),
..
} if refund_tx_amount_sat == lockup_amount_sat - prepare_refund_rbf_response.tx_fee_sat
));*/
}