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

Introduce Flow utilities and OffersMessageFlow implementation #3639

Draft
wants to merge 5 commits into
base: main
Choose a base branch
from

Conversation

shaavan
Copy link
Member

@shaavan shaavan commented Mar 3, 2025

This PR presents an alternative approach to #3412.

It introduces the Flow trait and its implementation, OffersMessageFlow, as a mid-level API that abstracts most Offers-related logic out of ChannelManager, while allowing ChannelManager to be parameterized over Flow.

This improves modularity and separation of concerns, making the Offers-related code more maintainable. Additionally, it provides a set of functional utilities for users who do not use ChannelManager in their workflow.

@ldk-reviews-bot
Copy link

ldk-reviews-bot commented Mar 3, 2025

👋 Thanks for assigning @TheBlueMatt as a reviewer!
I'll wait for their review and will help manage the review process.
Once they submit their review, I'll check if a second reviewer would be helpful.

@shaavan
Copy link
Member Author

shaavan commented Mar 3, 2025

Right now, this PR is a proof of concept for the approach—it introduces a basic set of utilities to move a significant chunk of Offers-related code out of ChannelManager.

What's in the current version:

  • Added initial utilities for creating Offers, Refunds, Blinded Paths, and handling message enqueue/dequeue directly within Flow.
  • Started using these utilities inside Offer payers and builders.

About the CI failures:

  • The CI is failing because of some documentation issues in this version.
  • All functional tests for the std case are passing locally.

If this approach looks good, I'll expand it to cover Offer handlers next!

@shaavan
Copy link
Member Author

shaavan commented Mar 3, 2025

@TheBlueMatt, @jkczyz
A gentle ping! :)

Copy link
Contributor

@jkczyz jkczyz left a comment

Choose a reason for hiding this comment

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

Let's iterate a bit more on this. We should have the MessageContext creation and thus the BlindedPath creation be the domain of OffersMessageFlow, IMO. This means passing a closure to the utility functions to create the blinded paths. Likewise with BlindedPaymentPaths.

message_router: MR,

our_network_pubkey: PublicKey,
highest_seen_timestamp: AtomicUsize,
Copy link
Contributor

Choose a reason for hiding this comment

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

We'd need to update this any time a block is processed. So either we need an Arc to share with ChannelManager or have Flow require chain::Listen such that ChannelManager (or whoever owns the Flow) can call block_connected.

Copy link
Member Author

Choose a reason for hiding this comment

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

That makes sense! I’ve implemented the chain::Listen trait for OffersMessageFlow in pr3639.02. This allows block_connected to be called as needed.

Let me know if this approach aligns with what you had in mind! Thanks a lot for the pointer!

Comment on lines +44 to +169
#[cfg(not(any(test, feature = "_test_utils")))]
pending_offers_messages: Mutex<Vec<(OffersMessage, MessageSendInstructions)>>,
#[cfg(any(test, feature = "_test_utils"))]
pub(crate) pending_offers_messages: Mutex<Vec<(OffersMessage, MessageSendInstructions)>>,

pending_async_payments_messages: Mutex<Vec<(AsyncPaymentsMessage, MessageSendInstructions)>>,

#[cfg(feature = "dnssec")]
pending_dns_onion_messages: Mutex<Vec<(DNSResolverMessage, MessageSendInstructions)>>,
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 idea here was to have the owner such as ChannelManager own the message queues since it would implement the message handlers.

Copy link
Member Author

Choose a reason for hiding this comment

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

I was deciding between two designs: having ChannelManager hold the queues or having OffersMessageFlow hold them. I chose the latter because:

  1. The long-term goal is to fully separate offers-related code from ChannelManager, and moving the queues out will be a major step in that direction.
  2. Keeping the queues within OffersMessageFlow helps maintain cleaner utility function signatures, as users don’t need to explicitly pass the queues as parameters.

Let me know your thoughts on this approach!

Copy link
Collaborator

Choose a reason for hiding this comment

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

I kinda like having the queues in the flow? It makes it more of a useful "object", rather than a "bag of functions".

Comment on lines 186 to 342
fn create_offer_builder(
&self, nonce: Nonce,
) -> Result<OfferBuilder<DerivedMetadata, secp256k1::All>, Bolt12SemanticError> {
let node_id = self.get_our_node_id();
let expanded_key = &self.inbound_payment_key;
let secp_ctx = &self.secp_ctx;

let builder = OfferBuilder::deriving_signing_pubkey(node_id, expanded_key, nonce, secp_ctx)
.chain_hash(self.chain_hash);

Ok(builder)
}
Copy link
Contributor

Choose a reason for hiding this comment

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

I'd question what value utilities like these are really providing to the the user.

Copy link
Member Author

Choose a reason for hiding this comment

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

The idea I was exploring was to create a mid-level API that remains modular enough to be used in multiple contexts.

For example, self.flow.create_offer_builder simplifies both create_offer_builder and create_async_receive_offer_builder, making the API more reusable and reducing duplication.

Would love to hear your thoughts on whether this with indented design approach.

Copy link
Collaborator

Choose a reason for hiding this comment

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

I'm similarly skeptical of adding methods that do "nothing". In cases where we're delegating blinded message path building to the flow, fine, but here I'm pretty skeptical this is adding much value.

Copy link
Contributor

Choose a reason for hiding this comment

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

Note that the most recent version passes a closure for creating the paths. At very least we need this in flow so that the MessageContext creation resides there, I think. See also #3639 (comment) for discussion on Nonce creation.

MR::Target: MessageRouter,
{
fn create_offer_builder(
&self, nonce: Nonce,
Copy link
Contributor

@jkczyz jkczyz Mar 4, 2025

Choose a reason for hiding this comment

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

Seems like the Nonce could be generated from the EntropySource. Though I imagine we need the Nonce for the MessageContext. Don't we want OffersMessageFlow to determine the correct MessageContext to give to the BlindedPaths? Otherwise, the onus is on the user to pass the correct context. So this implies the blinded paths should be created within this method.

Copy link
Member Author

Choose a reason for hiding this comment

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

Makes sense! My initial reasoning was to keep offer building and blinded path creation separate, as this allowed us to use the same self.flow.create_offer_builder for both create_offer_builder (where we create & append the path later) and create_async_receive_offer_builder (where we already have a vector of paths to append).

That said, I think we can take a middle ground by passing an optional closure as input—if present, it would handle the creation and appending of paths. This keeps things flexible while reducing the onus on the user to manually pass the correct context.

Let me know what you think of this approach. Thanks!

Comment on lines 252 to 279
fn create_invoice_builder<'a>(
&'a self, refund: &'a Refund, payment_paths: Vec<BlindedPaymentPath>,
payment_hash: PaymentHash,
) -> Result<InvoiceBuilder<'a, DerivedSigningPubkey>, Bolt12SemanticError> {
Copy link
Contributor

@jkczyz jkczyz Mar 4, 2025

Choose a reason for hiding this comment

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

Note we need this not only for refunds but also for handling invoice requests. I'm wondering though if we'd rather want a method that does more similar to request_refund_payment and pay_for_offer?

Copy link
Member Author

Choose a reason for hiding this comment

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

Looking through the section of code where we create an invoice from an invoice request, it seems significantly different from invoice creation from a refund. Given our current trait design, introducing a separate function feels like a cleaner solution.

On the other hand, we could design the Flow trait to be more similar to request_refund_payment and pay_for_offer, but wouldn’t that make it too high-level of an API?

@TheBlueMatt, would love to hear your thoughts on this!

Copy link
Collaborator

Choose a reason for hiding this comment

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

Similarly, I feel like it'd be nice to have a higher-level API on the flow. If the goal is to move bolt12-specific logic out of ChannelManager as much as possible, we should target a high-level API and then make the Flow object configurable where we see demand for configuration, IMO.

Copy link
Contributor

Choose a reason for hiding this comment

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

Regarding a higher-level API, it seems we need ways to parameterize the following:

  • create blinded paths
  • add custom settings / TLVs when building messages
  • currency conversion / checks
  • general (possibly async?) checks before acting on a message (e.g., Fedimint needs to fetch a payment hash for creating an invoice and may need to do some check before paying an invoice)

Can these still be accomplished with a higher-level API? Seems type parameterization on OffersMessageFlow could help with some of these. But there are some lingering concerns:

  • Circular dependency between ChannelManager and OffersMessageFlow for blinded path creation would necessitate passing a closure instead of using a trait, IIUC.
  • Async interactions before acting upon a message (e.g., Fedimint) wouldn't be possible with trait parameterization without re-introducing an event.
  • Possible inconsistent API for adding custom settings / TLVs to messages (i.e., caller does so directly on result of create_offer_builder but would need a trait parameterization to do the same for invreqs and invoices).

Additionally, isn't a high-level API closer to what we were going for in #3412? The primary concern there was needing the trait parameter to know about payment management concerns. Could you say what a higher-level API in this PR would look like without re-introducing that?

Copy link
Collaborator

Choose a reason for hiding this comment

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

Circular dependency between ChannelManager and OffersMessageFlow for blinded path creation would necessitate passing a closure instead of using a trait, IIUC.

I'm confused, where is the circular dependency? If we do #3639 (comment) it wont be

Async interactions before acting upon a message (e.g., Fedimint) wouldn't be possible with trait parameterization without re-introducing an event.

There's a few ways to do it. We could have the flow parameterized by a trait, which receives a call saying "hey, process this" and then it can return "okay, i did, you process it now" or "okay, in process" (like we have for ChannelMonitorUpdate persistence). We could also do an "event" that is dedicated to the flow itself, rather than flowing through the normal event process we could have a dedicated path. I think either of those seem fine, and maybe preferrable to disintermediating the whole thing as a trait, but the trait approach is also fine with me.

Possible inconsistent API for adding custom settings / TLVs to messages (i.e., caller does so directly on result of create_offer_builder but would need a trait parameterization to do the same for invreqs and invoices).

Could take a similar approach as the above.

Additionally, isn't a high-level API closer to what we were going for in #3412? The primary concern there was needing the trait parameter to know about payment management concerns. Could you say what a higher-level API in this PR would look like without re-introducing that?

I wouldn't call #3412/an api where you have to expose the pending payment store a "high level API". I was thinking more what we have here, just with more moved into the flow (eg blinded path construction) and possibly with config knobs directly on the flow, rather than requiring interception.

}

fn create_blinded_paths(
&self, peers: Vec<MessageForwardNode>, context: MessageContext,
Copy link
Contributor

Choose a reason for hiding this comment

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

Again, it seems OffersMessageFlow should be the one determining which context to use.

Copy link
Collaborator

Choose a reason for hiding this comment

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

Right, hiding the context in the flow gives it some material value for the user cause that's a bit of complexity that the user wouldn't have to implement.


#[cfg(feature = "dnssec")]
use crate::onion_message::dns_resolution::DNSResolverMessage;

pub trait Flow {
Copy link
Contributor

Choose a reason for hiding this comment

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

After going through the PR, I'm not sure how useful the Flow trait is. Either the caller of the utilities can do any processing between calls or we can parameterize OffersMessageFlow with traits (e.g., currency conversion).

Copy link
Member Author

Choose a reason for hiding this comment

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

I think I agree with your point. We could make OffersMessageFlow a direct parameter of ChannelManager without needing the Flow trait as an intermediary. This way, we still remove Bolt12 responsibilities from ChannelManager while giving users the flexibility to implement their own handlers using OffersMessageFlow utilities.

@TheBlueMatt, I’d love to hear your thoughts on this approach!

Copy link
Collaborator

Choose a reason for hiding this comment

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

I think it depends a bit on where we land on the API itself - if its high-level and we allow the OffersMessageFlow to be configured where there is a need for options, great, we can drop the trait. If we end up with something where we want it to not be super configurable but rather overridable, we can keep the flow.

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.

Were we going to move most of the OffersMessageHandler logic into flow.rs?

@shaavan
Copy link
Member Author

shaavan commented Mar 6, 2025

Updated from pr3639.01 to pr3639.02 (diff):
Addressed @jkczyz comments

Changes:

  • Implement chain::Listen of OffersMessageFlow, to allow keep the highest_seen_timestamp synced.

@shaavan
Copy link
Member Author

shaavan commented Mar 6, 2025

@TheBlueMatt

Were we going to move most of the OffersMessageHandler logic into flow.rs?

Yes, that’s the goal! For this version, I’ve primarily abstracted out the Bolt12 payers/builders from ChannelManager. I wanted to first get a concept ACK on this approach before proceeding with abstracting the handlers out of ChannelManager.

@TheBlueMatt
Copy link
Collaborator

Yea, its just a bit hard to evaluate without that change - there's presumably some state that can move into the flow handler...

@shaavan
Copy link
Member Author

shaavan commented Mar 17, 2025

Updated from pr3639.02 to pr3639.03 (diff):
Addressed @jkczyz and @TheBlueMatt comments

Changes:

  • Completed the Flow approach by incorporating changes across all relevant functions, handlers, and queues.

Points of notice:

  1. In this approach, OffersMessageFlow is no longer parameterized on Route/MessageRouter. Instead, path creation is passed in as a closure to OffersMessageFlow. This removes the need for multiple layers of parameterization for Router/MessageRouter.
  2. I've aimed to keep this as a mid-level API that removes the need for users of ChannelManager to manually handle context creation.
  3. While the approach is functionally complete (all tests passed when I ran cargo test 🚀), I haven’t yet worked on documentation or run rustfmt, so CI will likely be sad :)

@shaavan shaavan requested review from TheBlueMatt and jkczyz March 17, 2025 19:05
@shaavan
Copy link
Member Author

shaavan commented Mar 17, 2025

Also, a gentle ping @TheBlueMatt and @jkczyz — thank you so much for all your inputs and ideas! Really grateful for them.
Looking forward to your thoughts on the completed approach! 🙌

Comment on lines +10314 to +10318
self.flow.enqueue_invoice_request(invoice_request.clone(), payment_id, Some(nonce), |context| {
self.create_blinded_paths(context)
})?;

Ok(())
create_pending_payment(&invoice_request, nonce)
Copy link
Contributor

@jkczyz jkczyz Mar 18, 2025

Choose a reason for hiding this comment

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

@TheBlueMatt Here there a is change such that we enqueue the invoice requests prior to creating the pending payment. Seems like this could be an issue?

@shaavan Could you remind me why the ordering needed to be changed here?

Copy link
Member Author

Choose a reason for hiding this comment

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

Sure!

I noticed that with the original ordering (create_pending_payment -> self.flow.enqueue_invoice_request), if we try to pay for an offer and fail during the Blinded Path creation stage, we still end up creating a pending payment with that PaymentId in pending_outbound_payments.

So, if we try to call pay_for_offer again with the same payment_id, we hit a DuplicatePaymentId error — because the payment was already enqueued in pending_outbound_payments during the first (failed) attempt.

Happy to share a branch that reproduces the error if it's helpful. Thanks!

Copy link
Collaborator

Choose a reason for hiding this comment

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

@TheBlueMatt Here there a is change such that we enqueue the invoice requests prior to creating the pending payment. Seems like this could be an issue?

Hmm, don't see why it would be? Its all in a single PersistenceNotifierGuard so the intermediate state cannot be written to disk (and we don't write the queue to disk anyway), and while in a theoretical edge case we could get the message out and get a response back faster than we can add the pending payment (causing us to consider the responding Invoice un-requested), I'm somewhat skeptical its worth worrying about here.

debug_assert!(false);
return Err(Bolt12SemanticError::MissingIssuerSigningPubkey);
}
self.flow.enqueue_invoice_request(invoice_request.clone(), payment_id, Some(nonce), |context| {
Copy link
Contributor

Choose a reason for hiding this comment

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

It's unfortunate that the user needs to know that they must pass the same nonce here as they had passed to create_invoice_request_builder.

Copy link
Member Author

Choose a reason for hiding this comment

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

That's true! Maybe we can add a short note in the enqueue_invoice_request documentation mentioning that the nonce the user passes here must exactly match the nonce used to create the invoice_request.

Alternatively, we could consider renaming the parameter to invoice_request_nonce to make the connection clearer.

Let me know what you think — Thanks!

@TheBlueMatt TheBlueMatt removed their request for review March 18, 2025 19:13
Err(()) => {
self.abandon_payment_with_reason(payment_id, PaymentFailureReason::BlindedPathCreationFailed);
if self.flow.enqueue_async_payment_messages(invoice, payment_id, |context| {
self.create_blinded_paths(context)
Copy link
Collaborator

Choose a reason for hiding this comment

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

Rather than passing a lambda for create_blinded_paths to the flow, can we just pass the output of self.list_usable_channels() and let the flow handle the rest? Seems like that would further reduce bolt12-specific code in ChannelManager. Somewhat awkwardly, LNDK probably needs to be able to specify its own blinded payment paths, but would probably prefer the flow specify the blinded message paths. If we end up making this change that probably means we should add a separate method for LNDK to provide its own blinded path.

Copy link
Contributor

Choose a reason for hiding this comment

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

Yeah, it does seem a little hacky for LNDK, but maybe we can live with this.

Regarding blinded message paths, I don't believe we can fully do this in the flow. Currently, we have ChannelManager-specific logic to determine if a peer should be included and which scid should be preferred for compact paths.

let peers = self.per_peer_state.read().unwrap()
.iter()
.map(|(node_id, peer_state)| (node_id, peer_state.lock().unwrap()))
.filter(|(_, peer)| peer.is_connected)
.filter(|(_, peer)| peer.latest_features.supports_onion_messages())
.map(|(node_id, peer)| MessageForwardNode {
node_id: *node_id,
short_channel_id: peer.channel_by_id
.iter()
.filter(|(_, channel)| channel.context().is_usable())
.min_by_key(|(_, channel)| channel.context().channel_creation_height)
.and_then(|(_, channel)| channel.context().get_short_channel_id()),
})
.collect::<Vec<_>>();

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 understand why this requires that channelmanager.rs create the blinded path? (a) we could pass "list of connected peers that support onion messaging" to the flow and (b) we probably actually shouldn't be using "list of connected peers" as the criteria at all but rather be filtering by "peers that we have a channel with (that are connected)", which can be determined by looking at a ChannelDetails list.

Comment on lines +44 to +169
#[cfg(not(any(test, feature = "_test_utils")))]
pending_offers_messages: Mutex<Vec<(OffersMessage, MessageSendInstructions)>>,
#[cfg(any(test, feature = "_test_utils"))]
pub(crate) pending_offers_messages: Mutex<Vec<(OffersMessage, MessageSendInstructions)>>,

pending_async_payments_messages: Mutex<Vec<(AsyncPaymentsMessage, MessageSendInstructions)>>,

#[cfg(feature = "dnssec")]
pending_dns_onion_messages: Mutex<Vec<(DNSResolverMessage, MessageSendInstructions)>>,
Copy link
Collaborator

Choose a reason for hiding this comment

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

I kinda like having the queues in the flow? It makes it more of a useful "object", rather than a "bag of functions".

Comment on lines 186 to 342
fn create_offer_builder(
&self, nonce: Nonce,
) -> Result<OfferBuilder<DerivedMetadata, secp256k1::All>, Bolt12SemanticError> {
let node_id = self.get_our_node_id();
let expanded_key = &self.inbound_payment_key;
let secp_ctx = &self.secp_ctx;

let builder = OfferBuilder::deriving_signing_pubkey(node_id, expanded_key, nonce, secp_ctx)
.chain_hash(self.chain_hash);

Ok(builder)
}
Copy link
Collaborator

Choose a reason for hiding this comment

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

I'm similarly skeptical of adding methods that do "nothing". In cases where we're delegating blinded message path building to the flow, fine, but here I'm pretty skeptical this is adding much value.

Comment on lines 252 to 279
fn create_invoice_builder<'a>(
&'a self, refund: &'a Refund, payment_paths: Vec<BlindedPaymentPath>,
payment_hash: PaymentHash,
) -> Result<InvoiceBuilder<'a, DerivedSigningPubkey>, Bolt12SemanticError> {
Copy link
Collaborator

Choose a reason for hiding this comment

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

Similarly, I feel like it'd be nice to have a higher-level API on the flow. If the goal is to move bolt12-specific logic out of ChannelManager as much as possible, we should target a high-level API and then make the Flow object configurable where we see demand for configuration, IMO.


#[cfg(feature = "dnssec")]
use crate::onion_message::dns_resolution::DNSResolverMessage;

pub trait Flow {
Copy link
Collaborator

Choose a reason for hiding this comment

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

I think it depends a bit on where we land on the API itself - if its high-level and we allow the OffersMessageFlow to be configured where there is a need for options, great, we can drop the trait. If we end up with something where we want it to not be super configurable but rather overridable, we can keep the flow.

Comment on lines +10314 to +10318
self.flow.enqueue_invoice_request(invoice_request.clone(), payment_id, Some(nonce), |context| {
self.create_blinded_paths(context)
})?;

Ok(())
create_pending_payment(&invoice_request, nonce)
Copy link
Collaborator

Choose a reason for hiding this comment

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

@TheBlueMatt Here there a is change such that we enqueue the invoice requests prior to creating the pending payment. Seems like this could be an issue?

Hmm, don't see why it would be? Its all in a single PersistenceNotifierGuard so the intermediate state cannot be written to disk (and we don't write the queue to disk anyway), and while in a theoretical edge case we could get the message out and get a response back faster than we can add the pending payment (causing us to consider the responding Invoice un-requested), I'm somewhat skeptical its worth worrying about here.

Copy link
Contributor

@jkczyz jkczyz left a comment

Choose a reason for hiding this comment

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

Chatted a bit with @shaavan offline about the open comments. Added some responses where we had concerns. @TheBlueMatt could you take a look?

Err(()) => {
self.abandon_payment_with_reason(payment_id, PaymentFailureReason::BlindedPathCreationFailed);
if self.flow.enqueue_async_payment_messages(invoice, payment_id, |context| {
self.create_blinded_paths(context)
Copy link
Contributor

Choose a reason for hiding this comment

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

Yeah, it does seem a little hacky for LNDK, but maybe we can live with this.

Regarding blinded message paths, I don't believe we can fully do this in the flow. Currently, we have ChannelManager-specific logic to determine if a peer should be included and which scid should be preferred for compact paths.

let peers = self.per_peer_state.read().unwrap()
.iter()
.map(|(node_id, peer_state)| (node_id, peer_state.lock().unwrap()))
.filter(|(_, peer)| peer.is_connected)
.filter(|(_, peer)| peer.latest_features.supports_onion_messages())
.map(|(node_id, peer)| MessageForwardNode {
node_id: *node_id,
short_channel_id: peer.channel_by_id
.iter()
.filter(|(_, channel)| channel.context().is_usable())
.min_by_key(|(_, channel)| channel.context().channel_creation_height)
.and_then(|(_, channel)| channel.context().get_short_channel_id()),
})
.collect::<Vec<_>>();

Comment on lines 186 to 342
fn create_offer_builder(
&self, nonce: Nonce,
) -> Result<OfferBuilder<DerivedMetadata, secp256k1::All>, Bolt12SemanticError> {
let node_id = self.get_our_node_id();
let expanded_key = &self.inbound_payment_key;
let secp_ctx = &self.secp_ctx;

let builder = OfferBuilder::deriving_signing_pubkey(node_id, expanded_key, nonce, secp_ctx)
.chain_hash(self.chain_hash);

Ok(builder)
}
Copy link
Contributor

Choose a reason for hiding this comment

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

Note that the most recent version passes a closure for creating the paths. At very least we need this in flow so that the MessageContext creation resides there, I think. See also #3639 (comment) for discussion on Nonce creation.

Comment on lines 252 to 279
fn create_invoice_builder<'a>(
&'a self, refund: &'a Refund, payment_paths: Vec<BlindedPaymentPath>,
payment_hash: PaymentHash,
) -> Result<InvoiceBuilder<'a, DerivedSigningPubkey>, Bolt12SemanticError> {
Copy link
Contributor

Choose a reason for hiding this comment

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

Regarding a higher-level API, it seems we need ways to parameterize the following:

  • create blinded paths
  • add custom settings / TLVs when building messages
  • currency conversion / checks
  • general (possibly async?) checks before acting on a message (e.g., Fedimint needs to fetch a payment hash for creating an invoice and may need to do some check before paying an invoice)

Can these still be accomplished with a higher-level API? Seems type parameterization on OffersMessageFlow could help with some of these. But there are some lingering concerns:

  • Circular dependency between ChannelManager and OffersMessageFlow for blinded path creation would necessitate passing a closure instead of using a trait, IIUC.
  • Async interactions before acting upon a message (e.g., Fedimint) wouldn't be possible with trait parameterization without re-introducing an event.
  • Possible inconsistent API for adding custom settings / TLVs to messages (i.e., caller does so directly on result of create_offer_builder but would need a trait parameterization to do the same for invreqs and invoices).

Additionally, isn't a high-level API closer to what we were going for in #3412? The primary concern there was needing the trait parameter to know about payment management concerns. Could you say what a higher-level API in this PR would look like without re-introducing that?

@jkczyz jkczyz requested a review from TheBlueMatt April 1, 2025 17:09
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.

I think I responded to all the discussions

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