Skip to content

Commit 31beaf2

Browse files
committed
Update PaymentStore after each Wallet sync
We update the payment store whenever syncing the wallet state finished.
1 parent 7002481 commit 31beaf2

File tree

3 files changed

+179
-13
lines changed

3 files changed

+179
-13
lines changed

src/payment/store.rs

+5
Original file line numberDiff line numberDiff line change
@@ -284,6 +284,11 @@ impl_writeable_tlv_based_enum!(PaymentStatus,
284284
#[derive(Clone, Debug, PartialEq, Eq)]
285285
pub enum PaymentKind {
286286
/// An on-chain payment.
287+
///
288+
/// Payments of this kind will be considered pending until the respective transaction has
289+
/// reached [`ANTI_REORG_DELAY`] confirmations on-chain.
290+
///
291+
/// [`ANTI_REORG_DELAY`]: lightning::chain::channelmonitor::ANTI_REORG_DELAY
287292
Onchain {
288293
/// The transaction identifier of this payment.
289294
txid: Txid,

src/wallet/mod.rs

+86-2
Original file line numberDiff line numberDiff line change
@@ -11,13 +11,16 @@ use crate::logger::{log_debug, log_error, log_info, log_trace, FilesystemLogger,
1111

1212
use crate::event::EventQueue;
1313
use crate::fee_estimator::{ConfirmationTarget, FeeEstimator};
14-
use crate::payment::store::PaymentStore;
14+
use crate::payment::store::{ConfirmationStatus, PaymentStore};
15+
use crate::payment::{PaymentDetails, PaymentDirection, PaymentStatus};
1516
use crate::Error;
1617

1718
use lightning::chain::chaininterface::BroadcasterInterface;
19+
use lightning::chain::channelmonitor::ANTI_REORG_DELAY;
1820
use lightning::chain::{BestBlock, Listen};
1921

2022
use lightning::events::bump_transaction::{Utxo, WalletSource};
23+
use lightning::ln::channelmanager::PaymentId;
2124
use lightning::ln::inbound_payment::ExpandedKey;
2225
use lightning::ln::msgs::{DecodeError, UnsignedGossipMessage};
2326
use lightning::ln::script::ShutdownScript;
@@ -46,6 +49,7 @@ use bitcoin::{
4649

4750
use std::ops::Deref;
4851
use std::sync::{Arc, Mutex};
52+
use std::time::{Duration, SystemTime, UNIX_EPOCH};
4953

5054
pub(crate) enum OnchainSendAmount {
5155
ExactRetainingReserve { amount_sats: u64, cur_anchor_reserve_sats: u64 },
@@ -112,6 +116,11 @@ where
112116
Error::PersistenceFailed
113117
})?;
114118

119+
self.update_payment_store(&mut *locked_wallet).map_err(|e| {
120+
log_error!(self.logger, "Failed to update payment store: {}", e);
121+
Error::PersistenceFailed
122+
})?;
123+
115124
Ok(())
116125
},
117126
Err(e) => {
@@ -136,6 +145,76 @@ where
136145
Ok(())
137146
}
138147

148+
fn update_payment_store<'a>(
149+
&self, locked_wallet: &'a mut PersistedWallet<KVStoreWalletPersister>,
150+
) -> Result<(), Error> {
151+
let latest_update_timestamp = SystemTime::now()
152+
.duration_since(UNIX_EPOCH)
153+
.unwrap_or(Duration::from_secs(0))
154+
.as_secs();
155+
156+
for wtx in locked_wallet.transactions() {
157+
let id = PaymentId(wtx.tx_node.txid.to_byte_array());
158+
let txid = wtx.tx_node.txid;
159+
let (payment_status, confirmation_status) = match wtx.chain_position {
160+
bdk_chain::ChainPosition::Confirmed { anchor, .. } => {
161+
let confirmation_height = anchor.block_id.height;
162+
let cur_height = locked_wallet.latest_checkpoint().height();
163+
let payment_status = if cur_height >= confirmation_height + ANTI_REORG_DELAY - 1
164+
{
165+
PaymentStatus::Succeeded
166+
} else {
167+
PaymentStatus::Pending
168+
};
169+
let confirmation_status = ConfirmationStatus::Confirmed {
170+
block_hash: anchor.block_id.hash,
171+
height: confirmation_height,
172+
timestamp: anchor.confirmation_time,
173+
};
174+
(payment_status, confirmation_status)
175+
},
176+
bdk_chain::ChainPosition::Unconfirmed { .. } => {
177+
(PaymentStatus::Pending, ConfirmationStatus::Unconfirmed)
178+
},
179+
};
180+
// TODO: It would be great to introduce additional variants for
181+
// `ChannelFunding` and `ChannelClosing`. For the former, we could just
182+
// take a reference to `ChannelManager` here and check against
183+
// `list_channels`. But for the latter the best approach is much less
184+
// clear: for force-closes/HTLC spends we should be good querying
185+
// `OutputSweeper::tracked_spendable_outputs`, but regular channel closes
186+
// (i.e., `SpendableOutputDescriptor::StaticOutput` variants) are directly
187+
// spent to a wallet address. The only solution I can come up with is to
188+
// create and persist a list of 'static pending outputs' that we could use
189+
// here to determine the `PaymentKind`, but that's not really satisfactory, so
190+
// we're punting on it until we can come up with a better solution.
191+
let kind = crate::payment::PaymentKind::Onchain { txid, status: confirmation_status };
192+
let (sent, received) = locked_wallet.sent_and_received(&wtx.tx_node.tx);
193+
let (direction, amount_msat) = if sent > received {
194+
let direction = PaymentDirection::Outbound;
195+
let amount_msat = Some(sent.to_sat().saturating_sub(received.to_sat()) * 1000);
196+
(direction, amount_msat)
197+
} else {
198+
let direction = PaymentDirection::Inbound;
199+
let amount_msat = Some(received.to_sat().saturating_sub(sent.to_sat()) * 1000);
200+
(direction, amount_msat)
201+
};
202+
203+
let payment = PaymentDetails {
204+
id,
205+
kind,
206+
amount_msat,
207+
direction,
208+
status: payment_status,
209+
latest_update_timestamp,
210+
};
211+
212+
self.payment_store.insert_or_update(&payment)?;
213+
}
214+
215+
Ok(())
216+
}
217+
139218
pub(crate) fn create_funding_transaction(
140219
&self, output_script: ScriptBuf, amount: Amount, confirmation_target: ConfirmationTarget,
141220
locktime: LockTime,
@@ -480,7 +559,12 @@ where
480559
}
481560

482561
match locked_wallet.apply_block(block, height) {
483-
Ok(()) => (),
562+
Ok(()) => {
563+
if let Err(e) = self.update_payment_store(&mut *locked_wallet) {
564+
log_error!(self.logger, "Failed to update payment store: {}", e);
565+
return;
566+
}
567+
},
484568
Err(e) => {
485569
log_error!(
486570
self.logger,

tests/common/mod.rs

+88-11
Original file line numberDiff line numberDiff line change
@@ -483,6 +483,36 @@ pub(crate) fn do_channel_full_cycle<E: ElectrumApi>(
483483
assert_eq!(node_a.list_balances().spendable_onchain_balance_sats, premine_amount_sat);
484484
assert_eq!(node_b.list_balances().spendable_onchain_balance_sats, premine_amount_sat);
485485

486+
// Check we saw the node funding transactions.
487+
assert_eq!(
488+
node_a
489+
.list_payments_with_filter(|p| p.direction == PaymentDirection::Inbound
490+
&& matches!(p.kind, PaymentKind::Onchain { .. }))
491+
.len(),
492+
1
493+
);
494+
assert_eq!(
495+
node_a
496+
.list_payments_with_filter(|p| p.direction == PaymentDirection::Outbound
497+
&& matches!(p.kind, PaymentKind::Onchain { .. }))
498+
.len(),
499+
0
500+
);
501+
assert_eq!(
502+
node_b
503+
.list_payments_with_filter(|p| p.direction == PaymentDirection::Inbound
504+
&& matches!(p.kind, PaymentKind::Onchain { .. }))
505+
.len(),
506+
1
507+
);
508+
assert_eq!(
509+
node_b
510+
.list_payments_with_filter(|p| p.direction == PaymentDirection::Outbound
511+
&& matches!(p.kind, PaymentKind::Onchain { .. }))
512+
.len(),
513+
0
514+
);
515+
486516
// Check we haven't got any events yet
487517
assert_eq!(node_a.next_event(), None);
488518
assert_eq!(node_b.next_event(), None);
@@ -515,6 +545,15 @@ pub(crate) fn do_channel_full_cycle<E: ElectrumApi>(
515545
node_a.sync_wallets().unwrap();
516546
node_b.sync_wallets().unwrap();
517547

548+
// Check we now see the channel funding transaction as outbound.
549+
assert_eq!(
550+
node_a
551+
.list_payments_with_filter(|p| p.direction == PaymentDirection::Outbound
552+
&& matches!(p.kind, PaymentKind::Onchain { .. }))
553+
.len(),
554+
1
555+
);
556+
518557
let onchain_fee_buffer_sat = 5000;
519558
let node_a_anchor_reserve_sat = if expect_anchor_channel { 25_000 } else { 0 };
520559
let node_a_upper_bound_sat =
@@ -564,22 +603,26 @@ pub(crate) fn do_channel_full_cycle<E: ElectrumApi>(
564603
let payment_id = node_a.bolt11_payment().send(&invoice, None).unwrap();
565604
assert_eq!(node_a.bolt11_payment().send(&invoice, None), Err(NodeError::DuplicatePayment));
566605

567-
assert_eq!(node_a.list_payments().first().unwrap().id, payment_id);
606+
assert!(!node_a.list_payments_with_filter(|p| p.id == payment_id).is_empty());
568607

569-
let outbound_payments_a =
570-
node_a.list_payments_with_filter(|p| p.direction == PaymentDirection::Outbound);
608+
let outbound_payments_a = node_a.list_payments_with_filter(|p| {
609+
p.direction == PaymentDirection::Outbound && matches!(p.kind, PaymentKind::Bolt11 { .. })
610+
});
571611
assert_eq!(outbound_payments_a.len(), 1);
572612

573-
let inbound_payments_a =
574-
node_a.list_payments_with_filter(|p| p.direction == PaymentDirection::Inbound);
613+
let inbound_payments_a = node_a.list_payments_with_filter(|p| {
614+
p.direction == PaymentDirection::Inbound && matches!(p.kind, PaymentKind::Bolt11 { .. })
615+
});
575616
assert_eq!(inbound_payments_a.len(), 0);
576617

577-
let outbound_payments_b =
578-
node_b.list_payments_with_filter(|p| p.direction == PaymentDirection::Outbound);
618+
let outbound_payments_b = node_b.list_payments_with_filter(|p| {
619+
p.direction == PaymentDirection::Outbound && matches!(p.kind, PaymentKind::Bolt11 { .. })
620+
});
579621
assert_eq!(outbound_payments_b.len(), 0);
580622

581-
let inbound_payments_b =
582-
node_b.list_payments_with_filter(|p| p.direction == PaymentDirection::Inbound);
623+
let inbound_payments_b = node_b.list_payments_with_filter(|p| {
624+
p.direction == PaymentDirection::Inbound && matches!(p.kind, PaymentKind::Bolt11 { .. })
625+
});
583626
assert_eq!(inbound_payments_b.len(), 1);
584627

585628
expect_event!(node_a, PaymentSuccessful);
@@ -813,8 +856,26 @@ pub(crate) fn do_channel_full_cycle<E: ElectrumApi>(
813856
node_b.payment(&keysend_payment_id).unwrap().kind,
814857
PaymentKind::Spontaneous { .. }
815858
));
816-
assert_eq!(node_a.list_payments().len(), 6);
817-
assert_eq!(node_b.list_payments().len(), 7);
859+
assert_eq!(
860+
node_a.list_payments_with_filter(|p| matches!(p.kind, PaymentKind::Bolt11 { .. })).len(),
861+
5
862+
);
863+
assert_eq!(
864+
node_b.list_payments_with_filter(|p| matches!(p.kind, PaymentKind::Bolt11 { .. })).len(),
865+
6
866+
);
867+
assert_eq!(
868+
node_a
869+
.list_payments_with_filter(|p| matches!(p.kind, PaymentKind::Spontaneous { .. }))
870+
.len(),
871+
1
872+
);
873+
assert_eq!(
874+
node_b
875+
.list_payments_with_filter(|p| matches!(p.kind, PaymentKind::Spontaneous { .. }))
876+
.len(),
877+
1
878+
);
818879

819880
println!("\nB close_channel (force: {})", force_close);
820881
if force_close {
@@ -935,6 +996,22 @@ pub(crate) fn do_channel_full_cycle<E: ElectrumApi>(
935996
assert_eq!(node_a.list_balances().total_anchor_channels_reserve_sats, 0);
936997
assert_eq!(node_b.list_balances().total_anchor_channels_reserve_sats, 0);
937998

999+
// Now we should have seen the channel closing transaction on-chain.
1000+
assert_eq!(
1001+
node_a
1002+
.list_payments_with_filter(|p| p.direction == PaymentDirection::Inbound
1003+
&& matches!(p.kind, PaymentKind::Onchain { .. }))
1004+
.len(),
1005+
2
1006+
);
1007+
assert_eq!(
1008+
node_b
1009+
.list_payments_with_filter(|p| p.direction == PaymentDirection::Inbound
1010+
&& matches!(p.kind, PaymentKind::Onchain { .. }))
1011+
.len(),
1012+
2
1013+
);
1014+
9381015
// Check we handled all events
9391016
assert_eq!(node_a.next_event(), None);
9401017
assert_eq!(node_b.next_event(), None);

0 commit comments

Comments
 (0)