breez_sdk_liquid/recover/handlers/
handle_chain_send_swap.rsuse anyhow::Result;
use log::{debug, error, warn};
use lwk_wollet::elements::Txid;
use crate::prelude::*;
use crate::recover::model::*;
pub(crate) struct ChainSendSwapHandler;
impl ChainSendSwapHandler {
pub fn should_skip_recovery(
chain_swap: &ChainSwap,
recovered_data: &RecoveredOnchainDataChainSend,
is_local_within_grace_period: bool,
) -> bool {
let swap_id = &chain_swap.id;
let lockup_is_cleared = chain_swap.user_lockup_tx_id.is_some()
&& recovered_data.lbtc_user_lockup_tx_id.is_none();
let refund_is_cleared =
chain_swap.refund_tx_id.is_some() && recovered_data.lbtc_refund_tx_id.is_none();
let claim_is_cleared =
chain_swap.claim_tx_id.is_some() && recovered_data.btc_claim_tx_id.is_none();
if is_local_within_grace_period
&& (lockup_is_cleared || refund_is_cleared || claim_is_cleared)
{
warn!(
"Local outgoing chain swap {swap_id} was updated recently - skipping recovery \
as it would clear a tx that may have been broadcasted by us. Lockup clear: \
{lockup_is_cleared} - Refund clear: {refund_is_cleared}"
);
return true;
}
false
}
pub async fn recover_swap(
chain_swap: &mut ChainSwap,
context: &RecoveryContext,
is_local_within_grace_period: bool,
) -> Result<()> {
let swap_id = &chain_swap.id.clone();
debug!("[Recover Chain Send] Recovering data for swap {swap_id}");
let claim_script = chain_swap
.get_claim_swap_script()
.ok()
.and_then(|script| script.as_bitcoin_script().ok())
.and_then(|script| script.funding_addrs.map(|addr| addr.script_pubkey()))
.ok_or_else(|| {
anyhow::anyhow!("BTC claim script not found for Onchain Send Swap {swap_id}")
})?;
let lockup_script = chain_swap
.get_lockup_swap_script()
.ok()
.and_then(|script| script.as_liquid_script().ok())
.and_then(|script| script.funding_addrs.map(|addr| addr.script_pubkey()))
.ok_or_else(|| {
anyhow::anyhow!("LBTC lockup script not found for Onchain Send Swap {swap_id}")
})?;
let history = &SendChainSwapHistory {
lbtc_lockup_script_history: context
.lbtc_script_to_history_map
.get(&lockup_script)
.cloned()
.unwrap_or_default(),
btc_claim_script_history: context
.btc_script_to_history_map
.get(&claim_script)
.cloned()
.unwrap_or_default(),
btc_claim_script_txs: context
.btc_script_to_txs_map
.get(&claim_script)
.cloned()
.unwrap_or_default(),
};
let recovered_data =
Self::recover_onchain_data(&context.tx_map, swap_id, history, &claim_script)?;
Self::update_swap(
chain_swap,
&recovered_data,
context.liquid_tip_height,
is_local_within_grace_period,
)
}
pub fn update_swap(
chain_swap: &mut ChainSwap,
recovered_data: &RecoveredOnchainDataChainSend,
current_block_height: u32,
is_local_within_grace_period: bool,
) -> Result<()> {
if Self::should_skip_recovery(chain_swap, recovered_data, is_local_within_grace_period) {
return Ok(());
}
let is_expired = current_block_height >= chain_swap.timeout_block_height;
if let Some(new_state) = recovered_data.derive_partial_state(is_expired) {
chain_swap.state = new_state;
}
chain_swap.user_lockup_tx_id = recovered_data
.lbtc_user_lockup_tx_id
.clone()
.map(|h| h.txid.to_string());
chain_swap.refund_tx_id = recovered_data
.lbtc_refund_tx_id
.clone()
.map(|h| h.txid.to_string());
chain_swap.server_lockup_tx_id = recovered_data
.btc_server_lockup_tx_id
.clone()
.map(|h| h.txid.to_string());
chain_swap.claim_tx_id = recovered_data
.btc_claim_tx_id
.clone()
.map(|h| h.txid.to_string());
Ok(())
}
fn recover_onchain_data(
tx_map: &TxMap,
swap_id: &str,
history: &SendChainSwapHistory,
claim_script: &BtcScript,
) -> Result<RecoveredOnchainDataChainSend> {
let lbtc_user_lockup_tx_id = history
.lbtc_lockup_script_history
.iter()
.find(|&tx| tx_map.outgoing_tx_map.contains_key::<Txid>(&tx.txid))
.cloned();
if lbtc_user_lockup_tx_id.is_none() {
error!("No lockup tx found when recovering data for Chain Send Swap {swap_id}");
}
let lbtc_refund_tx_id = history
.lbtc_lockup_script_history
.iter()
.find(|&tx| tx_map.incoming_tx_map.contains_key::<Txid>(&tx.txid))
.cloned();
let (btc_server_lockup_tx_id, btc_claim_tx_id) = match history
.btc_claim_script_history
.len()
{
1 => (Some(history.btc_claim_script_history[0].clone()), None),
2 => {
let first_tx = history.btc_claim_script_txs[0].clone();
let first_tx_id = history.btc_claim_script_history[0].clone();
let second_tx_id = history.btc_claim_script_history[1].clone();
let is_first_tx_lockup_tx = first_tx
.output
.iter()
.any(|out| matches!(&out.script_pubkey, x if x == claim_script));
match is_first_tx_lockup_tx {
true => (Some(first_tx_id), Some(second_tx_id)),
false => (Some(second_tx_id), Some(first_tx_id)),
}
}
n => {
warn!("BTC script history with length {n} found while recovering data for Chain Send Swap {swap_id}");
(None, None)
}
};
Ok(RecoveredOnchainDataChainSend {
lbtc_user_lockup_tx_id,
lbtc_refund_tx_id,
btc_server_lockup_tx_id,
btc_claim_tx_id,
})
}
}
pub(crate) struct RecoveredOnchainDataChainSend {
pub(crate) lbtc_user_lockup_tx_id: Option<LBtcHistory>,
pub(crate) lbtc_refund_tx_id: Option<LBtcHistory>,
pub(crate) btc_server_lockup_tx_id: Option<BtcHistory>,
pub(crate) btc_claim_tx_id: Option<BtcHistory>,
}
impl RecoveredOnchainDataChainSend {
pub(crate) fn derive_partial_state(&self, is_expired: bool) -> Option<PaymentState> {
match &self.lbtc_user_lockup_tx_id {
Some(_) => match (&self.btc_claim_tx_id, &self.lbtc_refund_tx_id) {
(Some(btc_claim_tx_id), None) => match btc_claim_tx_id.confirmed() {
true => Some(PaymentState::Complete),
false => Some(PaymentState::Pending),
},
(None, Some(lbtc_refund_tx_id)) => match lbtc_refund_tx_id.confirmed() {
true => Some(PaymentState::Failed),
false => Some(PaymentState::RefundPending),
},
(Some(btc_claim_tx_id), Some(lbtc_refund_tx_id)) => {
match btc_claim_tx_id.confirmed() {
true => match lbtc_refund_tx_id.confirmed() {
true => Some(PaymentState::Complete),
false => Some(PaymentState::RefundPending),
},
false => Some(PaymentState::Pending),
}
}
(None, None) => match is_expired {
true => Some(PaymentState::RefundPending),
false => Some(PaymentState::Pending),
},
},
None => match is_expired {
true => Some(PaymentState::Failed),
false => None,
},
}
}
}
#[derive(Clone)]
pub(crate) struct SendChainSwapHistory {
pub(crate) lbtc_lockup_script_history: Vec<LBtcHistory>,
pub(crate) btc_claim_script_history: Vec<BtcHistory>,
pub(crate) btc_claim_script_txs: Vec<bitcoin::Transaction>,
}