-
Notifications
You must be signed in to change notification settings - Fork 376
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add static invoice creation utils to ChannelManager
#3408
base: main
Are you sure you want to change the base?
Add static invoice creation utils to ChannelManager
#3408
Conversation
86cfb96
to
8feb3cb
Compare
pub(crate) struct RetryableInvoiceRequest { | ||
pub(crate) invoice_request: InvoiceRequest, | ||
pub(crate) nonce: Nonce, | ||
pub(super) needs_retry: bool, | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Would it be worth not holding on to these for offers that don't support async payments? Can't recall if we can tell from the offer or if it isn't known until the static invoice is received.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
IIRC there is nothing in the offers, just at the invoice request level.
let amount_msat = offer.amount().and_then(|amount| { | ||
match amount { | ||
Amount::Bitcoin { amount_msats } => Some(amount_msats), | ||
Amount::Currency { .. } => None | ||
} | ||
}); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is quantity_max
allowed for offers paying often-offline nodes? If so, is the amount checked by them in some way? If not, should we prevent building a static invoice from such offers?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
quantity_max
is currently allowed. I'm not sure I follow your question though:
If so, is the amount checked by them in some way?
Who is "them"? It looks like the payer checks the amount/quantity when sending an invreq but not sure I'm following.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Does the recipient check that the amount is sufficient for the requested quantity before releasing the preimage?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yeah seems they will have to, if that isn't done already. Currently we don't support receiving to the static invoices created by these utils (just creating them), so this is definitely something to consider in the follow-up.
for path in message_paths_to_always_online_node { | ||
builder = builder.path(path); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If the recipient is online, does the InvoiceRequest
get forwarded along? If so, how does the recipient authenticate it?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If the recipient is online, does the InvoiceRequest get forwarded along?
Yep!
how does the recipient authenticate it?
Since the recipient issued the offer themselves, they authenticate it the same way as always-online recipients, i.e. via verify_using_recipient_data
, if I'm understanding you.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Right, so I guess that means the recipient must construct these paths since it needs to include the Nonce
used to derive the signing keys.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hmm... but the function is creating and returning the Nonce
. I think we'll need to pass it in?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
To be clear, we don't include metadata in the offer when we have blinded paths. Can't recall if we do something different when constructing an offer for use with async payments.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I brought this up during review club. Since payment path contains the offer nonce -- and the sender will include the invoice request with the payment -- we should be able to verify the invoice request is authentic with this nonce. But ISTM we still want the same nonce in the offer's message paths for the normal case when the recipient is online.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah I misunderstood at first, this is a good point.... I'm not sure yet what a good solution would be.
For context, message_paths_to_always_online_node
terminate at the always-online node. Currently in LDK the recipient expects to receive the invreq either over a blinded path from the offer or to their node id but with offer_metadata
set (IIUC), but neither of those are supported in this PR atm.
For more context, I think the always-online node is expected to be a direct peer of the async receiver, otherwise an onion message round-trip would be incurred to check whether the recipient is online when the always-online node receives the invreq.
So it seems like our two options here are to either set the offer metadata for async receive offers (which may be a privacy leak), or for the recipient to provide the always-online node with a one-hop blinded path to send the invreq over, separately from their static invoice. Either of these options should allow the recipient to verify the invreq if they happen to be online at the time.
Any thoughts on this?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I wonder how big that privacy leak is in practice? If we assume most BPs to async-recipients are to an LSP and probably 1-hop BPs for reliability, then its probably nothing (as such BPs are always to async-recipients). If we assume they're sometimes 2-hops, then its probably not huge but is at least something non-zero. It may also just be unintuitive that this leak exists and users might not notice it.
That said, how does the LSP currently pass the invreq to the recipient? Presumably there's some offer -> node registration process so the LSP has a DB with (offer, node, static_invoice)
tuples? Replacing the node with a one-hop BP doesn't sound like all that much work, I think, depending on what the API from LDK looks like?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That said, how does the LSP currently pass the invreq to the recipient? Presumably there's some offer -> node registration process so the LSP has a DB with (offer, node, static_invoice) tuples? Replacing the node with a one-hop BP doesn't sound like all that much work, I think, depending on what the API from LDK looks like?
Yep. I think I agree the blinded path route is the way to go. It's more flexible in case async receivers prefer their offer's corresponding always-online node to be a hop or two away (IMO this would introduce too much payment latency in practice, but who knows), and avoids increasing the size of the offer via offer_metadata
.
We may want a separate method on ChannelManager
to create this blinded path, though it's actually already supported via MessageRouter
since the recipient has the offer nonce. So I may punt on hashing out the details of this for now until LSP async payments support is more fleshed out.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nice, thanks!
/// | ||
/// [`Offer`]: crate::offers::offer::Offer | ||
#[derive(Clone, Debug, Eq, PartialEq)] | ||
pub struct AsyncBolt12OfferContext { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
#3427 applies here too, though not sure if it needs to be addressed in this PR.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Opened #3435.
8feb3cb
to
73a28ef
Compare
lightning/src/events/mod.rs
Outdated
@@ -203,6 +203,15 @@ impl PaymentPurpose { | |||
payment_context: context, | |||
} | |||
}, | |||
Some(PaymentContext::AsyncBolt12Offer(_context)) => { | |||
debug_assert!(false, "Receiving async payments is not yet supported"); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
73a28ef
to
63601e4
Compare
63601e4
to
00fceb2
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fix-ups LGTM. Just comments from reviewing the tests.
// Set the random bytes so we can predict the offer outbound payment context nonce. | ||
*nodes[0].keys_manager.override_random_bytes.lock().unwrap() = Some([42; 32]); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Shouldn't these tests go in offers_tests.rs
and use extract_offer_nonce
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We can't use extract_offer_nonce
because the nonce we're trying to extract comes from the reply path to the invreq, not the invreq itself. But I switched away from manually constructing the hmac, see the latest fixup!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
LGTM. Just some minor comments to address.
1320eb6
to
ac69d09
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Feel free to squash after fixing the doc comment
ac69d09
to
588e19f
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Aside from the comments here, LGTM.
/// one of our offers. | ||
/// | ||
/// [`Offer`]: crate::offers::offer::Offer | ||
offer_id: OfferId, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Remind me why we need this? The HeldHtlcAvailable
message contains almost nothing, right? What offer_id
are we authenticating against? We presumably don't even have an offer at the point where we receive this.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is basically just used as an input into the HMAC field 10 lines below, so it helps authenticate this context. But maybe the nonce is sufficient as a unique input? I was following the pattern of hmac_for_payment_id
used for regular bolt 12 invoices but using an offer_id
instead.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
For AsyncPaymentsContext::OutboundPayment
when we receive the ReleaseHtlcAvailable
message we need to look up the pending payment with its payment_id
, then we use the nonce
and hmac
to authenticate the field. Here, I don't see a use for the offer_id
?
Arguably it would be reasonable to reply to every inbound HeldHtlcAvailable
message by just asking that the HTLC be released. It would imply someone can trivially ask if/when you're online which isn't ideal, so we should have something here (a nonce/hmac pair seems to suffice), but ultimately we may get a bunch of these for any given offer, so its not like we can track them and remove offers after they're paid. We might want a timeout here so that we can stop replying to old ones that have expired, I guess?
In any case, not sure what use we'd have for the OfferId
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yeah, I didn't really have a sense of whether nonce + hmac was sufficient since we haven't used only those things previously, but since it is I went ahead and removed it. Tracked expiring these paths on the async payments issue.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why change the format later? Should be trivial to store the timeout here, no?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Arguably it would be reasonable to reply to every inbound
HeldHtlcAvailable
message by just asking that the HTLC be released. It would imply someone can trivially ask if/when you're online which isn't ideal, so we should have something here (a nonce/hmac pair seems to suffice), but ultimately we may get a bunch of these for any given offer, so its not like we can track them and remove offers after they're paid. We might want a timeout here so that we can stop replying to old ones that have expired, I guess?
IIUC, this context is in the static invoice's message paths. So couldn't an adversary simply request an invoice, get back the static invoice, and save the path to use whenever it wants? We could have it expire with the static invoice, but there doesn't seem to be anything that would stop the adversary from doing the same thing with a new static invoice. Am I missing something?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You're not missing anything IIUC. The only thing that limits the attack you're describing is also expiring the offer/its paths.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
But isn't this a message path from the static invoice? Would we need to add an expiration to this new context in order to expire it? For payment paths we'd use the payment secret, IIUC.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Discussed offline; it is the case that if a recipient publishes an offer that never expires, then anyone can pretty much always check if they're online under this protocol as long as the offer's message paths remain valid. Probably worth noting in documentation in the future.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Chatted offline. Just was trying to point out that if you always want to be paid by anyone, then anyone can check if you are online because presumably you will always have a fresh offer with unexpired paths published.
588e19f
to
4d9135e
Compare
Removed the diff --git a/lightning/src/blinded_path/payment.rs b/lightning/src/blinded_path/payment.rs
index 70ac9823e..d72c13353 100644
--- a/lightning/src/blinded_path/payment.rs
+++ b/lightning/src/blinded_path/payment.rs
@@ -362,10 +362,6 @@ pub struct Bolt12OfferContext {
/// [`Offer`]: crate::offers::offer::Offer
#[derive(Clone, Debug, Eq, PartialEq)]
pub struct AsyncBolt12OfferContext {
- /// The identifier of the [`Offer`].
- ///
- /// [`Offer`]: crate::offers::offer::Offer
- pub offer_id: OfferId,
/// The [`Nonce`] used to verify that an inbound [`InvoiceRequest`] corresponds to this static
/// invoice's offer.
///
@@ -650,8 +646,7 @@ impl_writeable_tlv_based!(Bolt12OfferContext, {
});
impl_writeable_tlv_based!(AsyncBolt12OfferContext, {
- (0, offer_id, required),
- (2, offer_nonce, required),
+ (0, offer_nonce, required),
});
impl_writeable_tlv_based!(Bolt12RefundContext, {});
diff --git a/lightning/src/ln/channelmanager.rs b/lightning/src/ln/channelmanager.rs
index ba21b87ac..ed31e00c6 100644
--- a/lightning/src/ln/channelmanager.rs
+++ b/lightning/src/ln/channelmanager.rs
@@ -9602,7 +9602,7 @@ where
let secp_ctx = &self.secp_ctx;
let payment_context = PaymentContext::AsyncBolt12Offer(
- AsyncBolt12OfferContext { offer_id: offer.id(), offer_nonce }
+ AsyncBolt12OfferContext { offer_nonce }
);
let amount_msat = offer.amount().and_then(|amount| {
match amount { |
4d9135e
to
2c50051
Compare
LGTM, but needs rebase now :/ |
Prior to this patch, we would take() the invoice request stored for AwaitingInvoice outbound payments when retrying sending the invoice request onion message. This doesn't work for async payments because we need to keep the invoice request stored for inclusion in the payment onion. Therefore, clone it instead of take()ing it.
Prior to this fix, we would attempt to mark outbound async payments as abandoned but silently fail because they were in state AwaitingInvoice, which the mark_abandoned utility doesn't currently work for. These payments would eventually be removed by the remove_stale_payments method, but there would be a delay in generating the PaymentFailed event. Move to manually removing the outbound payment entry.
Useful for creating payment paths for static invoices which are typically amount-less.
Will be useful for static invoices' blinded paths, which may have long expiries. Rather than having a default max_cltv_expiry, we now base it on the invoice expiry.
This context is stored in the blinded payment paths we put in static invoices and is useful to authenticate payments over these paths to the recipient. We can't reuse Bolt12OfferContext for this because we don't have access to the invoice request fields at static invoice creation time.
2c50051
to
1c2554a
Compare
Codecov ReportAttention: Patch coverage is
Additional details and impacted files@@ Coverage Diff @@
## main #3408 +/- ##
==========================================
- Coverage 89.73% 89.72% -0.02%
==========================================
Files 130 130
Lines 107793 107815 +22
Branches 107793 107815 +22
==========================================
+ Hits 96727 96732 +5
- Misses 8662 8677 +15
- Partials 2404 2406 +2 ☔ View full report in Codecov by Sentry. |
This context is included in static invoice's blinded message paths, provided back to us in HeldHtlcAvailable onion messages for blinded path authentication. In future work, we will check if this context is valid and respond with a ReleaseHeldHtlc message to release the upstream payment if so. We also add creation methods for the hmac used for authenticating said blinded path.
1c2554a
to
5a67fa3
Compare
5a67fa3
to
9a52c80
Compare
Was testing the follow-up and realized we weren't allowing MPP on the static invoices: diff --git a/lightning/src/ln/channelmanager.rs b/lightning/src/ln/channelmanager.rs
index 112b2a0fa..ef118b99b 100644
--- a/lightning/src/ln/channelmanager.rs
+++ b/lightning/src/ln/channelmanager.rs
@@ -10068,7 +10068,7 @@ where
StaticInvoiceBuilder::for_offer_using_derived_keys(
offer, payment_paths, async_receive_message_paths, created_at, expanded_key,
offer_nonce, secp_ctx
- )
+ ).map(|inv| inv.allow_mpp())
}
/// Pays for an [`Offer`] using the given parameters by creating an [`InvoiceRequest`] and |
Should we leave this to the user? Otherwise, there's no way for them to disable this. Then again, we don't provide them a way to disable it for normal invoices. |
Fair point... I definitely think it's a better UX to enable MPP by default. IIRC, the only reason we've ran into for disabling it has been phantom payments, which aren't currently supported. I could document a follow-up to add an API for disabling it, possibly. |
Went ahead and moved the tests added here (and a couple pre-existing tests) into a dedicated async payments tests file. A good amount more tests are getting added in the follow-up so figured this change makes sense. |
8effdab
to
2391993
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
LGTM. Feel free to squash IMO.
2391993
to
82e5a92
Compare
We can't use our regular offer creation util for receiving async payments because the recipient can't be relied on to be online to service invoice_requests. Therefore, add a new offer creation util that is parameterized by blinded message paths to another node on the network that *is* always-online and can serve static invoices on behalf of the often-offline recipient. Also add a utility for creating static invoices corresponding to these offers. See new utils' docs and BOLTs PR 1149 for more info.
Since adding support for creating static invoices from ChannelManager, it's easier to test these failure cases that went untested when we added support for paying static invoices.
Blinded keysend is only planned to be supported in the async payments context.
82e5a92
to
d3a7efa
Compare
Err, also wasn't setting the invoice's relative expiry outside of the blinded paths. diff --git a/lightning/src/ln/async_payments_tests.rs b/lightning/src/ln/async_payments_tests.rs
index 41c1ec33b..424d76da6 100644
--- a/lightning/src/ln/async_payments_tests.rs
+++ b/lightning/src/ln/async_payments_tests.rs
@@ -35,6 +35,7 @@ use crate::util::config::UserConfig;
use bitcoin::secp256k1::Secp256k1;
use core::convert::Infallible;
+use core::time::Duration;
#[test]
fn blinded_keysend() {
@@ -514,12 +515,15 @@ fn pays_static_invoice() {
let offer = offer_builder.build().unwrap();
let amt_msat = 5000;
let payment_id = PaymentId([1; 32]);
+ let relative_expiry = Duration::from_secs(1000);
let static_invoice = nodes[2]
.node
- .create_static_invoice_builder(&offer, offer_nonce, None)
+ .create_static_invoice_builder(&offer, offer_nonce, Some(relative_expiry))
.unwrap()
.build_and_sign(&secp_ctx)
.unwrap();
+ assert!(static_invoice.invoice_features().supports_basic_mpp());
+ assert_eq!(static_invoice.relative_expiry(), relative_expiry);
nodes[0]
.node
diff --git a/lightning/src/ln/channelmanager.rs b/lightning/src/ln/channelmanager.rs
index ef118b99b..755198a8e 100644
--- a/lightning/src/ln/channelmanager.rs
+++ b/lightning/src/ln/channelmanager.rs
@@ -10068,7 +10068,7 @@ where
StaticInvoiceBuilder::for_offer_using_derived_keys(
offer, payment_paths, async_receive_message_paths, created_at, expanded_key,
offer_nonce, secp_ctx
- ).map(|inv| inv.allow_mpp())
+ ).map(|inv| inv.allow_mpp().relative_expiry(relative_expiry_secs))
}
/// Pays for an [`Offer`] using the given parameters by creating an [`InvoiceRequest`] and |
/// The invoice's expiration will default to [`DEFAULT_RELATIVE_EXPIRY`] after `created_at` unless | ||
/// overridden by [`StaticInvoiceBuilder::relative_expiry`]. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
IIUC, this default is not defined by the spec (which uses 2 hours for any invoice), and thus if not set StaticInvoice::relative_expiry
would give 24 hours whereas another implementation would assume it is 2 hours.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good point, thanks. I'll need to update the spec for that (added to the tracking issue for now). Currently relative_expiry()
gives 2 weeks if not set, though we may want to revisit that.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do we still need to explicitly set it in the ChannelManager
utility then as was added in the recent push? I guess it doesn't matter much since size isn't a constraint, but just wondering if a test caught this in a follow-up?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yeah, caught this in testing: 7d239c4. Fair that we could avoid setting it if it's None
or the default, though, let me know if you'd prefer that. Since size isn't a constraint as you mention, it simplifies the code a bit to always set it so I'm neutral.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Might as well have it be explicit.
Add static invoice creation utilities as part of supporting the async payments BOLTs spec Support async payments in BOLT 12 lightning/bolts#1149.
Take this opportunity to more easily test some code added in Support paying static invoices #3140 that went untested at the time. This is the bulk of the diff.
Address a piece of feedback from Support paying static invoices #3140 regarding
InvoiceRequest
s being unavailable when the time comes to send the async payment, cc Support paying static invoices #3140 (comment)