Skip to content

Commit ac6b669

Browse files
committed
Move JIT parameters to encrypted payment metadata
1 parent ec1b601 commit ac6b669

8 files changed

Lines changed: 303 additions & 144 deletions

File tree

CHANGELOG.md

Lines changed: 6 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,9 @@
1+
# Pending
2+
3+
## Compatibility Notes
4+
- Pending JIT-channel payments created before upgrading may fail after upgrade because the
5+
prior LSPS2 fee-limit state stored in `PaymentKind::Bolt11Jit` is not migrated.
6+
17
# 0.7.0 - Dec. 3, 2025
28
This seventh minor release introduces numerous new features, bug fixes, and API improvements. In particular, it adds support for channel Splicing, Async Payments, as well as sourcing chain data from a Bitcoin Core REST backend.
39

@@ -419,4 +425,3 @@ integrated LDK and BDK-based wallets.
419425

420426
**Note:** This release is still considered experimental, should not be run in
421427
production, and no compatibility guarantees are given until the release of 0.1.
422-

src/builder.rs

Lines changed: 12 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -75,6 +75,7 @@ use crate::lnurl_auth::LnurlAuth;
7575
use crate::logger::{log_error, LdkLogger, LogLevel, LogWriter, Logger};
7676
use crate::message_handler::NodeCustomMessageHandler;
7777
use crate::payment::asynchronous::om_mailbox::OnionMessageMailbox;
78+
use crate::payment::PaymentMetadataKeys;
7879
use crate::peer_store::PeerStore;
7980
use crate::runtime::{Runtime, RuntimeSpawner};
8081
use crate::tx_broadcaster::TransactionBroadcaster;
@@ -88,6 +89,7 @@ use crate::wallet::Wallet;
8889
use crate::{Node, NodeMetrics};
8990

9091
const LSPS_HARDENED_CHILD_INDEX: u32 = 577;
92+
const PAYMENT_METADATA_HARDENED_CHILD_INDEX: u32 = 578;
9193
const PERSISTER_MAX_PENDING_UPDATES: u64 = 100;
9294

9395
#[derive(Debug, Clone)]
@@ -2004,6 +2006,15 @@ fn build_with_store_internal(
20042006
};
20052007

20062008
let lnurl_auth = Arc::new(LnurlAuth::new(xprv, Arc::clone(&logger)));
2009+
let payment_metadata_keys = {
2010+
let payment_metadata_xpriv = derive_xprv(
2011+
Arc::clone(&config),
2012+
&seed_bytes,
2013+
PAYMENT_METADATA_HARDENED_CHILD_INDEX,
2014+
Arc::clone(&logger),
2015+
)?;
2016+
PaymentMetadataKeys::new(payment_metadata_xpriv.private_key.secret_bytes())
2017+
};
20072018

20082019
let (stop_sender, _) = tokio::sync::watch::channel(());
20092020
let (background_processor_stop_sender, _) = tokio::sync::watch::channel(());
@@ -2050,6 +2061,7 @@ fn build_with_store_internal(
20502061
scorer,
20512062
peer_store,
20522063
payment_store,
2064+
payment_metadata_keys,
20532065
lnurl_auth,
20542066
is_running,
20552067
node_metrics,

src/event.rs

Lines changed: 121 additions & 44 deletions
Original file line numberDiff line numberDiff line change
@@ -50,6 +50,7 @@ use crate::payment::asynchronous::static_invoice_store::StaticInvoiceStore;
5050
use crate::payment::store::{
5151
PaymentDetails, PaymentDetailsUpdate, PaymentDirection, PaymentKind, PaymentStatus,
5252
};
53+
use crate::payment::{EncryptedPaymentMetadata, PaymentMetadata, PaymentMetadataKeys};
5354
use crate::runtime::Runtime;
5455
use crate::types::{
5556
CustomTlvRecord, DynStore, KeysManager, OnionMessenger, PaymentStore, Sweeper, Wallet,
@@ -537,6 +538,7 @@ where
537538
payment_store: Arc<PaymentStore>,
538539
peer_store: Arc<PeerStore<L>>,
539540
keys_manager: Arc<KeysManager>,
541+
payment_metadata_keys: PaymentMetadataKeys,
540542
runtime: Arc<Runtime>,
541543
logger: L,
542544
config: Arc<Config>,
@@ -556,9 +558,10 @@ where
556558
output_sweeper: Arc<Sweeper>, network_graph: Arc<Graph>,
557559
liquidity_source: Option<Arc<LiquiditySource<Arc<Logger>>>>,
558560
payment_store: Arc<PaymentStore>, peer_store: Arc<PeerStore<L>>,
559-
keys_manager: Arc<KeysManager>, static_invoice_store: Option<StaticInvoiceStore>,
560-
onion_messenger: Arc<OnionMessenger>, om_mailbox: Option<Arc<OnionMessageMailbox>>,
561-
runtime: Arc<Runtime>, logger: L, config: Arc<Config>,
561+
keys_manager: Arc<KeysManager>, payment_metadata_keys: PaymentMetadataKeys,
562+
static_invoice_store: Option<StaticInvoiceStore>, onion_messenger: Arc<OnionMessenger>,
563+
om_mailbox: Option<Arc<OnionMessageMailbox>>, runtime: Arc<Runtime>, logger: L,
564+
config: Arc<Config>,
562565
) -> Self {
563566
Self {
564567
event_queue,
@@ -572,6 +575,7 @@ where
572575
payment_store,
573576
peer_store,
574577
keys_manager,
578+
payment_metadata_keys,
575579
logger,
576580
runtime,
577581
config,
@@ -581,6 +585,36 @@ where
581585
}
582586
}
583587

588+
fn fail_claimable_payment(
589+
&self, payment_id: PaymentId, payment_hash: &PaymentHash,
590+
) -> Result<(), ReplayEvent> {
591+
self.channel_manager.fail_htlc_backwards(payment_hash);
592+
593+
let update = PaymentDetailsUpdate {
594+
status: Some(PaymentStatus::Failed),
595+
..PaymentDetailsUpdate::new(payment_id)
596+
};
597+
match self.payment_store.update(update) {
598+
Ok(_) => Ok(()),
599+
Err(e) => {
600+
log_error!(self.logger, "Failed to access payment store: {}", e);
601+
Err(ReplayEvent())
602+
},
603+
}
604+
}
605+
606+
fn lsps2_max_total_opening_fee_msat(
607+
metadata: &PaymentMetadata, amount_msat: u64,
608+
) -> Option<u64> {
609+
let lsps2_parameters = metadata.lsps2_parameters?;
610+
lsps2_parameters.max_total_opening_fee_msat.or_else(|| {
611+
lsps2_parameters.max_proportional_opening_fee_ppm_msat.and_then(|max_prop_fee| {
612+
// If it's a variable amount payment, compute the actual fee.
613+
compute_opening_fee(amount_msat, 0, max_prop_fee)
614+
})
615+
})
616+
}
617+
584618
pub async fn handle_event(&self, event: LdkEvent) -> Result<(), ReplayEvent> {
585619
match event {
586620
LdkEvent::FundingGenerationReady {
@@ -694,7 +728,8 @@ where
694728
..
695729
} => {
696730
let payment_id = PaymentId(payment_hash.0);
697-
if let Some(info) = self.payment_store.get(&payment_id) {
731+
let payment_info = self.payment_store.get(&payment_id);
732+
if let Some(info) = payment_info.as_ref() {
698733
if info.direction == PaymentDirection::Outbound {
699734
log_info!(
700735
self.logger,
@@ -717,14 +752,13 @@ where
717752
}
718753

719754
if info.status == PaymentStatus::Succeeded
720-
|| matches!(info.kind, PaymentKind::Spontaneous { .. })
755+
|| matches!(&info.kind, PaymentKind::Spontaneous { .. })
721756
{
722-
let stored_preimage = match info.kind {
757+
let stored_preimage = match &info.kind {
723758
PaymentKind::Bolt11 { preimage, .. }
724-
| PaymentKind::Bolt11Jit { preimage, .. }
725759
| PaymentKind::Bolt12Offer { preimage, .. }
726760
| PaymentKind::Bolt12Refund { preimage, .. }
727-
| PaymentKind::Spontaneous { preimage, .. } => preimage,
761+
| PaymentKind::Spontaneous { preimage, .. } => *preimage,
728762
_ => None,
729763
};
730764

@@ -759,22 +793,35 @@ where
759793
},
760794
};
761795
}
796+
}
762797

763-
let max_total_opening_fee_msat = match info.kind {
764-
PaymentKind::Bolt11Jit { lsp_fee_limits, .. } => {
765-
lsp_fee_limits
766-
.max_total_opening_fee_msat
767-
.or_else(|| {
768-
lsp_fee_limits.max_proportional_opening_fee_ppm_msat.and_then(
769-
|max_prop_fee| {
770-
// If it's a variable amount payment, compute the actual fee.
771-
compute_opening_fee(amount_msat, 0, max_prop_fee)
772-
},
773-
)
774-
})
775-
.unwrap_or(0)
776-
},
777-
_ => 0,
798+
if counterparty_skimmed_fee_msat > 0 {
799+
let max_total_opening_fee_msat = match &purpose {
800+
PaymentPurpose::Bolt11InvoicePayment { payment_secret, .. } => onion_fields
801+
.as_ref()
802+
.and_then(|fields| fields.payment_metadata.as_ref())
803+
.and_then(|metadata| {
804+
EncryptedPaymentMetadata::from_raw(metadata.clone()).decrypt(
805+
&self.payment_metadata_keys,
806+
&payment_hash,
807+
payment_secret,
808+
)
809+
})
810+
.and_then(|metadata| {
811+
Self::lsps2_max_total_opening_fee_msat(&metadata, amount_msat)
812+
}),
813+
_ => None,
814+
};
815+
816+
let Some(max_total_opening_fee_msat) = max_total_opening_fee_msat else {
817+
log_info!(
818+
self.logger,
819+
"Refusing inbound payment with hash {} as the counterparty withheld {}msat without valid BOLT11 LSPS2 payment metadata",
820+
hex_utils::to_string(&payment_hash.0),
821+
counterparty_skimmed_fee_msat,
822+
);
823+
self.fail_claimable_payment(payment_id, &payment_hash)?;
824+
return Ok(());
778825
};
779826

780827
if counterparty_skimmed_fee_msat > max_total_opening_fee_msat {
@@ -785,26 +832,13 @@ where
785832
counterparty_skimmed_fee_msat,
786833
max_total_opening_fee_msat,
787834
);
788-
self.channel_manager.fail_htlc_backwards(&payment_hash);
789-
790-
let update = PaymentDetailsUpdate {
791-
hash: Some(Some(payment_hash)),
792-
status: Some(PaymentStatus::Failed),
793-
..PaymentDetailsUpdate::new(payment_id)
794-
};
795-
match self.payment_store.update(update) {
796-
Ok(_) => return Ok(()),
797-
Err(e) => {
798-
log_error!(self.logger, "Failed to access payment store: {}", e);
799-
return Err(ReplayEvent());
800-
},
801-
};
835+
self.fail_claimable_payment(payment_id, &payment_hash)?;
836+
return Ok(());
802837
}
803838

804-
// If the LSP skimmed anything, update our stored payment.
805-
if counterparty_skimmed_fee_msat > 0 {
806-
match info.kind {
807-
PaymentKind::Bolt11Jit { .. } => {
839+
if let Some(info) = payment_info.as_ref() {
840+
match &info.kind {
841+
PaymentKind::Bolt11 { .. } => {
808842
let update = PaymentDetailsUpdate {
809843
counterparty_skimmed_fee_msat: Some(Some(counterparty_skimmed_fee_msat)),
810844
..PaymentDetailsUpdate::new(payment_id)
@@ -817,16 +851,17 @@ where
817851
},
818852
};
819853
}
820-
_ => debug_assert!(false, "We only expect the counterparty to get away with withholding fees for JIT payments."),
854+
_ => debug_assert!(false, "We only expect the counterparty to get away with withholding fees for BOLT11 payments."),
821855
}
822856
}
857+
}
823858

859+
if let Some(info) = payment_info {
824860
// If this is known by the store but ChannelManager doesn't know the preimage,
825861
// the payment has been registered via `_for_hash` variants and needs to be manually claimed via
826862
// user interaction.
827863
match info.kind {
828-
PaymentKind::Bolt11 { preimage, .. }
829-
| PaymentKind::Bolt11Jit { preimage, .. } => {
864+
PaymentKind::Bolt11 { preimage, .. } => {
830865
if purpose.preimage().is_none() {
831866
debug_assert!(
832867
preimage.is_none(),
@@ -1897,8 +1932,50 @@ mod tests {
18971932

18981933
use super::*;
18991934
use crate::io::test_utils::InMemoryStore;
1935+
use crate::payment::store::LSPS2Parameters;
19001936
use crate::types::DynStoreWrapper;
19011937

1938+
#[test]
1939+
fn lsps2_payment_metadata_decodes_total_fee_limit() {
1940+
let metadata = PaymentMetadata {
1941+
lsps2_parameters: Some(LSPS2Parameters {
1942+
max_total_opening_fee_msat: Some(42_000),
1943+
max_proportional_opening_fee_ppm_msat: None,
1944+
}),
1945+
};
1946+
1947+
assert_eq!(
1948+
EventHandler::<Arc<TestLogger>>::lsps2_max_total_opening_fee_msat(&metadata, 100_000),
1949+
Some(42_000)
1950+
);
1951+
}
1952+
1953+
#[test]
1954+
fn lsps2_payment_metadata_missing_limit_is_rejected() {
1955+
let empty_metadata = PaymentMetadata { lsps2_parameters: None };
1956+
let metadata_without_fee_limit = PaymentMetadata {
1957+
lsps2_parameters: Some(LSPS2Parameters {
1958+
max_total_opening_fee_msat: None,
1959+
max_proportional_opening_fee_ppm_msat: None,
1960+
}),
1961+
};
1962+
1963+
assert_eq!(
1964+
EventHandler::<Arc<TestLogger>>::lsps2_max_total_opening_fee_msat(
1965+
&empty_metadata,
1966+
100_000
1967+
),
1968+
None
1969+
);
1970+
assert_eq!(
1971+
EventHandler::<Arc<TestLogger>>::lsps2_max_total_opening_fee_msat(
1972+
&metadata_without_fee_limit,
1973+
100_000
1974+
),
1975+
None
1976+
);
1977+
}
1978+
19021979
#[tokio::test]
19031980
async fn event_queue_persistence() {
19041981
let store: Arc<DynStore> = Arc::new(DynStoreWrapper(InMemoryStore::new()));

src/lib.rs

Lines changed: 6 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -166,8 +166,8 @@ use logger::{log_debug, log_error, log_info, log_trace, LdkLogger, Logger};
166166
use payment::asynchronous::om_mailbox::OnionMessageMailbox;
167167
use payment::asynchronous::static_invoice_store::StaticInvoiceStore;
168168
use payment::{
169-
Bolt11Payment, Bolt12Payment, OnchainPayment, PaymentDetails, SpontaneousPayment,
170-
UnifiedPayment,
169+
Bolt11Payment, Bolt12Payment, OnchainPayment, PaymentDetails, PaymentMetadataKeys,
170+
SpontaneousPayment, UnifiedPayment,
171171
};
172172
use peer_store::{PeerInfo, PeerStore};
173173
use runtime::Runtime;
@@ -233,6 +233,7 @@ pub struct Node {
233233
scorer: Arc<Mutex<Scorer>>,
234234
peer_store: Arc<PeerStore<Arc<Logger>>>,
235235
payment_store: Arc<PaymentStore>,
236+
payment_metadata_keys: PaymentMetadataKeys,
236237
lnurl_auth: Arc<LnurlAuth>,
237238
is_running: Arc<RwLock<bool>>,
238239
node_metrics: Arc<RwLock<NodeMetrics>>,
@@ -593,6 +594,7 @@ impl Node {
593594
Arc::clone(&self.payment_store),
594595
Arc::clone(&self.peer_store),
595596
Arc::clone(&self.keys_manager),
597+
self.payment_metadata_keys,
596598
static_invoice_store,
597599
Arc::clone(&self.onion_messenger),
598600
self.om_mailbox.clone(),
@@ -885,6 +887,7 @@ impl Node {
885887
self.liquidity_source.clone(),
886888
Arc::clone(&self.payment_store),
887889
Arc::clone(&self.peer_store),
890+
self.payment_metadata_keys,
888891
Arc::clone(&self.config),
889892
Arc::clone(&self.is_running),
890893
Arc::clone(&self.logger),
@@ -903,6 +906,7 @@ impl Node {
903906
self.liquidity_source.clone(),
904907
Arc::clone(&self.payment_store),
905908
Arc::clone(&self.peer_store),
909+
self.payment_metadata_keys,
906910
Arc::clone(&self.config),
907911
Arc::clone(&self.is_running),
908912
Arc::clone(&self.logger),

0 commit comments

Comments
 (0)