Skip to content
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

Trampoline Payload Construction Method #3386

Open
wants to merge 11 commits into
base: main
Choose a base branch
from

Conversation

arik-so
Copy link
Contributor

@arik-so arik-so commented Oct 28, 2024

Introduce a method for constructing Trampoline payloads.

This will allow the non-Trampoline method to be modified to generate a Trampoline onion and finish its own regular payload construction.

Builds on top of #3333.

lightning/src/routing/router.rs Outdated Show resolved Hide resolved
lightning/src/routing/router.rs Outdated Show resolved Hide resolved
@@ -454,6 +454,9 @@ impl_writeable_tlv_based!(BlindedTail, {
pub struct Path {
/// The list of unblinded hops in this [`Path`]. Must be at least length one.
pub hops: Vec<RouteHop>,
/// The list of unblinded Trampoline hops. If present, must be at least one.
/// The public key of the first Trampoline hop must match the public key of the last regular hop.
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't see why this needs to be true. Can we just have each trampoline hop be next hops starting after the last unblinded node.

lightning/src/routing/router.rs Show resolved Hide resolved
@arik-so arik-so force-pushed the arik/trampoline/payload-construction branch 2 times, most recently from 7cad33e to 856ce97 Compare November 19, 2024 23:04
@arik-so arik-so force-pushed the arik/trampoline/payload-construction branch from 856ce97 to 5ea3ada Compare November 21, 2024 17:15
Copy link

codecov bot commented Nov 21, 2024

Codecov Report

Attention: Patch coverage is 93.42105% with 35 lines in your changes missing coverage. Please review.

Project coverage is 90.44%. Comparing base (1a8bf62) to head (9535d78).
Report is 19 commits behind head on main.

Files with missing lines Patch % Lines
lightning/src/ln/onion_utils.rs 89.21% 16 Missing and 13 partials ⚠️
lightning/src/events/mod.rs 25.00% 6 Missing ⚠️
Additional details and impacted files
@@            Coverage Diff             @@
##             main    #3386      +/-   ##
==========================================
+ Coverage   89.74%   90.44%   +0.69%     
==========================================
  Files         130      130              
  Lines      107793   113400    +5607     
  Branches   107793   113400    +5607     
==========================================
+ Hits        96743   102569    +5826     
+ Misses       8651     8495     -156     
+ Partials     2399     2336      -63     

☔ View full report in Codecov by Sentry.
📢 Have feedback on the report? Share it here.

Copy link
Collaborator

@TheBlueMatt TheBlueMatt left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ah I guess my previous comments still need to be addressed.

@@ -404,13 +404,16 @@ pub struct BlindedTail {
pub excess_final_cltv_expiry_delta: u32,
/// The total amount paid on this [`Path`], excluding the fees.
pub final_value_msat: u64,
/// Used for determining the type of Trampoline path to construct
pub final_hop_supports_trampoline: bool
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should we instead consider storing a *Features of some form?

lightning/src/routing/router.rs Outdated Show resolved Hide resolved
@arik-so arik-so force-pushed the arik/trampoline/payload-construction branch from 5ea3ada to af1818b Compare December 5, 2024 01:19
@arik-so arik-so mentioned this pull request Dec 5, 2024
30 tasks
@arik-so arik-so force-pushed the arik/trampoline/payload-construction branch 4 times, most recently from ee68783 to 4dd0fea Compare December 9, 2024 05:31
Ok((res, value_msat, cltv))
}

fn build_trampoline_onion_payloads_callback<'a, H, B, F>(
Copy link
Contributor

@valentinewallace valentinewallace Dec 10, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is there a way to DRY this with build_onion_payloads_callback, via implementing a trait on OutboundTrampolinePayload and OutboundOnionPayload that the common method is parameterized by? Kinda like what we do for decode_next_hop and how it works for both onion message and onion payment packets.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure the payloads_callback is super DRYable, because all the response types that are being pushed vary strongly between RouteHop and TrampolineHop, but I think I was able to at least DRY up the onion_keys callback.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I ended up looking into this a bit more to see if we could avoid all the C&P, got this patch to compile: valentinewallace@974f212

It needs to be filled out a bit but take a look!

Copy link
Contributor Author

@arik-so arik-so Dec 18, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

thanks! I incorporated it for the most part. The impl is a bit duplicative the way I did it, but the borrow checker and I are looking into simplifying it a bit

@arik-so arik-so force-pushed the arik/trampoline/payload-construction branch 4 times, most recently from dc679c0 to 7477b1a Compare December 12, 2024 17:14
lightning/src/events/mod.rs Outdated Show resolved Hide resolved
lightning/src/ln/onion_utils.rs Outdated Show resolved Hide resolved
cur_value_msat += final_value_msat;
callback(
PayloadCallbackAction::PushBack,
msgs::OutboundTrampolinePayload::BlindedReceive {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It looks like in https://github.com/lightning/bolts/blob/fa0594ac2af3531d734f1d707a146d6e13679451/bolt04/trampoline-payment-onion-test.json there can be final trampoline hops that are unblinded, but that doesn't look supported in the code atm. Are we explicitly only supporting trampoline to blinded paths?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yup, we deliberately are. We originally supported non-blinded destinations but decided not to.

lightning/src/ln/onion_route_tests.rs Show resolved Hide resolved
@arik-so arik-so force-pushed the arik/trampoline/payload-construction branch from 7477b1a to ba71d6e Compare December 13, 2024 10:34
lightning/src/ln/onion_route_tests.rs Show resolved Hide resolved
lightning/src/events/mod.rs Outdated Show resolved Hide resolved
lightning/src/events/mod.rs Outdated Show resolved Hide resolved
@arik-so arik-so force-pushed the arik/trampoline/payload-construction branch 2 times, most recently from b934379 to 6569246 Compare December 13, 2024 22:54
@arik-so arik-so force-pushed the arik/trampoline/payload-construction branch 2 times, most recently from 09c7f70 to adb0b88 Compare December 15, 2024 22:58
@valentinewallace
Copy link
Contributor

Haven't looked closely yet but it looks like the non-test changes in the latest commit belong in earlier commit(s). Feel free to squash IMO!

@arik-so arik-so force-pushed the arik/trampoline/payload-construction branch from adb0b88 to 9535d78 Compare December 16, 2024 19:38
@arik-so
Copy link
Contributor Author

arik-so commented Dec 16, 2024

I think the squashing should have taken care of that, though let me know if you'd like any other changes move around.

@@ -1757,3 +1762,83 @@ fn route_blinding_spec_test_vector() {
_ => panic!("Unexpected error")
}
}

#[test]
fn combined_trampoline_onion_creation_test() {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Could you add a comment or adjust the test name to document that these are from the spec test vectors?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

done

lightning/src/ln/blinded_payment_tests.rs Outdated Show resolved Hide resolved
lightning/src/ln/onion_utils.rs Outdated Show resolved Hide resolved
@@ -551,7 +585,7 @@ impl Readable for Route {
if hops.is_empty() { return Err(DecodeError::InvalidValue); }
min_final_cltv_expiry_delta =
cmp::min(min_final_cltv_expiry_delta, hops.last().unwrap().cltv_expiry_delta);
paths.push(Path { hops, blinded_tail: None });
paths.push(Path { hops, trampoline_hops: vec![], blinded_tail: None });
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Trampoline hops need ser

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This was marked as resolved but trampoline hops still aren't serialized in Routes, I think?

}

impl_writeable_tlv_based!(BlindedTail, {
(0, hops, required_vec),
(2, blinding_point, required),
(4, excess_final_cltv_expiry_delta, required),
(6, final_value_msat, required),
(8, final_hop_supports_trampoline, required),
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Older versions won't be able to read these now since it's a required TLV, probably want to use an odd TLV and default_value

/// The node_announcement features of the node at this hop. For the last hop, these may be
/// amended to match the features present in the invoice this node generated.
pub node_features: NodeFeatures,
/// The fee taken on this hop (for paying for the use of the *next* channel in the path).
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Isn't it the fee taken for the next several hops until the next trampoline node, or the intro node? I think the docs for this field might need a glance to make sure they're accurate for trampoline, same for cltv_expiry_delta

excess_final_cltv_expiry_delta: bt.excess_final_cltv_expiry_delta,
});

// don't include blinded tail when Trampoline hops are present
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Probably should include why we're doing this

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

done

Ok((res, value_msat, cltv))
}

fn build_trampoline_onion_payloads_callback<'a, H, B, F>(
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I ended up looking into this a bit more to see if we could avoid all the C&P, got this patch to compile: valentinewallace@974f212

It needs to be filled out a bit but take a look!

@arik-so arik-so force-pushed the arik/trampoline/payload-construction branch from 9535d78 to ab320ba Compare December 18, 2024 03:15
@arik-so arik-so force-pushed the arik/trampoline/payload-construction branch from ab320ba to e191a3d Compare December 18, 2024 07:18
@@ -1670,6 +1671,7 @@ impl Writeable for Event {
(2, payment_hash, option),
(4, path.hops, required_vec),
(6, path.blinded_tail, option),
(8, path.trampoline_hops, optional_vec),
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I am trying to understand how types get assigned in LDK - it's an optional field here so shouldn't it have an odd type value ? thanks for clarifying.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

those concepts are a bit orthogonal. Even means if it's present, you cannot ignore it, i.e. you have to understand how to parse it, whereas odd means it's fine to just skip that value if you don't know what to do with it.

Optional means that the value may not need to be set. So you can have optional even fields like here, where there may not be a value, but if there is one, it's important to understand it correctly.

The only scenario I cannot really think of a good use case for is required odd fields, which would suggest that the value must always be present, but is fine to skip when parsing.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

thank you I understand now.

@@ -668,7 +668,7 @@ impl HTLCSource {
pub fn dummy() -> Self {
assert!(cfg!(not(feature = "grind_signatures")));
HTLCSource::OutboundRoute {
path: Path { hops: Vec::new(), blinded_tail: None },
path: Path { hops: Vec::new(), trampoline_hops: Vec::new(), blinded_tail: None },
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: every other vec in this patch is initialized with vec![], perhaps make this consistent ? your call feel free to dismiss.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

oh, good point!

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ah, actually, the other element had already been initialized with Vec::new(), so I think that's more consistent in this instance

@arik-so arik-so force-pushed the arik/trampoline/payload-construction branch 2 times, most recently from 249a1a0 to 549fdd5 Compare December 19, 2024 07:42
@arik-so arik-so force-pushed the arik/trampoline/payload-construction branch from 549fdd5 to 91d60ac Compare January 7, 2025 16:33
Copy link
Contributor

@valentinewallace valentinewallace left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nearly LGTM to me after a second reviewer

_recipient_onion: &'a RecipientOnionFields, _keysend_preimage: Option<PaymentPreimage>,
_sender_intended_htlc_amt_msat: u64, _total_msat: u64, _cltv_expiry_height: u32,
) -> Self {
unreachable!("Unblinded receiving is not supported for Trampoline!")
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This does look unreachable right now but would still slightly prefer returning an error since it's a bit of a buried panic at the moment, in case we do add support for this case later

amt_to_forward: value_msat,
outgoing_cltv_value: cltv,
};
let payload = OP::new_forward(last_hop_id.unwrap(), value_msat, cltv);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: I think it would read a little nicer to avoid unwrap here

Comment on lines +397 to +398
/// The node_announcement features of the node at this hop. For the last hop, these may be
/// amended to match the features present in the invoice this node generated.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think the second sentence can be deleted since IIUC it only applies to unblinded recipients

@@ -551,7 +585,7 @@ impl Readable for Route {
if hops.is_empty() { return Err(DecodeError::InvalidValue); }
min_final_cltv_expiry_delta =
cmp::min(min_final_cltv_expiry_delta, hops.last().unwrap().cltv_expiry_delta);
paths.push(Path { hops, blinded_tail: None });
paths.push(Path { hops, trampoline_hops: vec![], blinded_tail: None });
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This was marked as resolved but trampoline hops still aren't serialized in Routes, I think?

@@ -1670,6 +1671,7 @@ impl Writeable for Event {
(2, payment_hash, option),
(4, path.hops, required_vec),
(6, path.blinded_tail, option),
(8, path.trampoline_hops, optional_vec),
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

For this field and in other events, I'm not 100% certain but I think we'd prefer to make it odd to avoid breaking downgrades and include a release note that downgrading may result in path events missing their trampoline hops.

#[allow(unused_imports)]
use crate::prelude::*;
use core::ops::Deref;
use types::payment::PaymentSecret;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: include this in the other types::payment import above and alphabetize imports

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants