-
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
Introduce custom TLV support for OnionMessage #2830
base: main
Are you sure you want to change the base?
Conversation
WalkthroughThe updates involve the integration of Changes
Thank you for using CodeRabbit. We offer it for free to the OSS community and would appreciate your support in helping us grow. If you find it useful, would you consider giving us a shout-out on your favorite social media? 🪧 TipsChatThere are 3 ways to chat with CodeRabbit:
Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments. CodeRabbit Commands (Invoked using PR comments)
Other keywords and placeholders
CodeRabbit Configuration File (
|
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.
Review Status
Actionable comments generated: 3
Configuration used: CodeRabbit UI
Files selected for processing (4)
- lightning/src/blinded_path/message.rs (4 hunks)
- lightning/src/onion_message/messenger.rs (4 hunks)
- lightning/src/onion_message/packet.rs (2 hunks)
- lightning/src/util/ser_macros.rs (2 hunks)
Additional comments: 13
lightning/src/blinded_path/message.rs (4)
- 25-26: The addition of the
custom_tlvs
field to theForwardTlvs
struct is consistent with the PR's objective to support custom TLVs.- 35-36: The addition of the
custom_tlvs
field to theReceiveTlvs
struct aligns with the PR's goal of extending TLV support.- 67-69: The update to the
blinded_hops
function correctly initializescustom_tlvs
with an empty vector, which is consistent with the changes made to theForwardTlvs
andReceiveTlvs
structs.- 87-87: The changes in the
advance_path_by_one
function correctly handle thecustom_tlvs
when reading theControlTlvs::Forward
variant, aligning with the PR's objectives to support custom TLVs.lightning/src/onion_message/packet.rs (2)
- 285-300: The custom TLV decoding logic added to the
ControlTlvs
enum is consistent with the PR's objective to support custom TLVs and appears to correctly handle the decoding process.- 312-317: The handling of
custom_tlvs
in theControlTlvs
enum when constructing the payload format is appropriate and aligns with the changes made to support custom TLVs.lightning/src/onion_message/messenger.rs (5)
- 630-630: The addition of
custom_tlvs
to theReceiveControlTlvs::Unblinded
variant aligns with the PR's objective to support custom TLVs.- 635-636: The addition of
custom_tlvs
to theForwardControlTlvs::Unblinded
variant is consistent with the changes made toReceiveControlTlvs
and supports the PR's goal for custom TLV support.- 1112-1112: The initialization of
custom_tlvs
withVec::new()
in theForwardTlvs
struct is correct and ensures that the vector is empty by default.- 1123-1123: The addition of
custom_tlvs
with an empty vector in theForwardTlvs
struct within theForwardControlTlvs::Unblinded
variant is consistent with the other changes and is correctly implemented.- 1157-1157: The
ReceiveTlvs
struct is correctly updated to includecustom_tlvs
with an empty vector, which is in line with the PR's objectives.lightning/src/util/ser_macros.rs (2)
- 475-475: The macro
decode_tlv_stream_with_custom_tlv_decode
is correctly defined and follows the established pattern for TLV stream decoding with the addition of custom TLV handling.- 829-845: The macro
_init_and_read_tlv_stream_with_custom_tlv_decode
is correctly defined and follows the established pattern for initializing and reading a TLV stream with custom TLV decoding.
With the current version of PR, I am seeking concept/approach ACKs before I start working on the test suite. A gentle ping, @jkczyz :) |
55098b4
to
c9d57f8
Compare
Updated from pr2380.01 to pr2830.02 (diff):
Note: |
@shaavan It sounds like you've made significant progress on the PR by addressing previous comments and making adjustments to the |
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.
Review Status
Actionable comments generated: 0
Configuration used: CodeRabbit UI
Files selected for processing (8)
- lightning/src/blinded_path/message.rs (3 hunks)
- lightning/src/blinded_path/payment.rs (10 hunks)
- lightning/src/ln/blinded_payment_tests.rs (3 hunks)
- lightning/src/ln/channelmanager.rs (1 hunks)
- lightning/src/ln/msgs.rs (1 hunks)
- lightning/src/onion_message/messenger.rs (2 hunks)
- lightning/src/onion_message/packet.rs (2 hunks)
- lightning/src/util/ser_macros.rs (2 hunks)
Files skipped from review as they are similar to previous changes (4)
- lightning/src/blinded_path/message.rs
- lightning/src/onion_message/messenger.rs
- lightning/src/onion_message/packet.rs
- lightning/src/util/ser_macros.rs
Additional comments: 17
lightning/src/blinded_path/payment.rs (9)
- 19-19: The
Read
trait has been imported fromcrate::io
.- 57-58: A
custom_tlvs
field of typeVec<(u64, Vec<u8>)>
has been added to theReceiveTlvs
struct to store custom TLV entries.- 127-127: The
Writeable
trait implementation forReceiveTlvs
has been updated to include thecustom_tlvs
field with anoptional_vec
encoding.- 145-173: > Note: This review was outside the patches, so it was mapped to the patch with the greatest overlap. Original lines [148-182]
The
Readable
trait implementation forBlindedPaymentTlvs
has been modified to decode custom TLVs and populate thecustom_tlvs
field inReceiveTlvs
.
- 344-344: The test
compute_payinfo
has been updated to include an emptycustom_tlvs
vector in theReceiveTlvs
struct.- 363-363: The test
compute_payinfo_1_hop
has been updated similarly to include an emptycustom_tlvs
vector in theReceiveTlvs
struct.- 417-417: The test
simple_aggregated_htlc_min
has been updated to include an emptycustom_tlvs
vector in theReceiveTlvs
struct.- 468-468: The test
aggregated_htlc_min
has been updated to include an emptycustom_tlvs
vector in theReceiveTlvs
struct.- 523-523: The test
aggregated_htlc_max
has been updated to include an emptycustom_tlvs
vector in theReceiveTlvs
struct.lightning/src/ln/blinded_payment_tests.rs (3)
- 59-59: The addition of
custom_tlvs: Vec::new()
initializes an empty vector for custom TLVs. This is consistent with the PR's objective to introduce custom TLV support. However, it's important to ensure that this new field is actually used somewhere in the code or tests, otherwise it's dead code.Verification successful
The
custom_tlvs
field is actively used throughout the codebase, including in tests, message handling, and payment processing. This confirms that the addition of this field is not dead code and aligns with the PR's objectives.* 93-93: The same `custom_tlvs: Vec::new()` addition here. The comment for the `get_blinded_route_parameters` function applies here as well. Ensure that the `custom_tlvs` field is utilized effectively in the codebase. * 136-136: Again, `custom_tlvs: Vec::new()` is added. The same verification is needed to ensure that this field is not just dead code and is actually being used as intended.Scripts Executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash # Verify if custom_tlvs are used elsewhere in the codebase rg "custom_tlvs" --type rustLength of output: 11880
Verification successful
The
custom_tlvs
field is actively used throughout the codebase, including in payment processing and onion message handling. The initialization ofcustom_tlvs
with an empty vector is consistent with its usage in other parts of the code, where it is later populated and utilized. Therefore, the addition ofcustom_tlvs: Vec::new(),
in thempp_to_one_hop_blinded_path
function aligns with the intended use of custom TLVs within the codebase.Scripts Executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash # Reusing the previous verification script as it applies here too rg "custom_tlvs" --type rustLength of output: 11880
lightning/src/ln/msgs.rs (3)
- 2400-2401: The
custom_tlvs
field in theReceiveTlvs
struct is being matched to_
, which indicates that any value is accepted but not used within the block. Please confirm if this is the intended behavior, as it may be necessary to handlecustom_tlvs
explicitly if they are to be used for processing or validation.- 2401-2401: The use of
unwrap_or(0)
fortotal_msat
assumes that a missingtotal_msat
can be treated as 0. Please ensure that this assumption is valid in all cases, as it could lead to unexpected behavior iftotal_msat
is expected to be provided but is not.- 2401-2401: The error handling for
amt_msat
usingok_or(DecodeError::InvalidValue)
is appropriate and ensures that a missingamt_msat
results in a decode error.lightning/src/ln/channelmanager.rs (2)
- 7942-7942: The addition of the
custom_tlvs
field initialized with an empty vector seems appropriate for optional or extendable features. Ensure that this field is properly populated and utilized in the downstream logic where the blinded payment paths are created.- 7942-7942: Verify if the
custom_tlvs
field should be passed to thecreate_blinded_payment_paths
function. If it is intended to be used for handling custom TLVs within the blinded payment paths, it may need to be included in the function call.
@shaavan Great to hear that you've addressed the comments from @TheBlueMatt and made the necessary changes to the Action performedReview triggered. |
The are still some conceptual kinks to be ironed out before finalising the custom TLV's implementation. |
0cdac04
to
07e73c5
Compare
Updated from pr2380.02 to pr2830.03 (diff): Changes:
Note:
I'm still refining the approach for custom TLVs within |
07e73c5
to
2668794
Compare
2668794
to
4c207d4
Compare
4c207d4
to
f9a8608
Compare
Codecov ReportAttention: Patch coverage is
Additional details and impacted files@@ Coverage Diff @@
## main #2830 +/- ##
==========================================
- Coverage 89.25% 89.23% -0.02%
==========================================
Files 130 130
Lines 106959 107116 +157
Branches 106959 107116 +157
==========================================
+ Hits 95464 95590 +126
- Misses 8706 8727 +21
- Partials 2789 2799 +10 ☔ View full report in Codecov by Sentry. |
lightning/src/ln/channelmanager.rs
Outdated
@@ -187,7 +187,9 @@ pub enum PendingHTLCRouting { | |||
/// For HTLCs received by LDK, this will ultimately be exposed in | |||
/// [`Event::PaymentClaimable::onion_fields`] as | |||
/// [`RecipientOnionFields::custom_tlvs`]. | |||
custom_tlvs: Vec<(u64, Vec<u8>)>, | |||
sender_custom_tlvs: Vec<(u64, Vec<u8>)>, | |||
/// Custom TLVs which were set by us through the reply 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.
"Sent by us" isn't really accurate. The sender is still sending them to us. It's just that we construct a blinded path for them to use which included the TLVs.
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've updated the docs to clarify how user_custom_tlvs
will be used in pr2830.08. Let me know if they look good! Thanks!
pub context: Option<MessageContext>, | ||
|
||
/// Custom Tlvs. A user can use this to send custom tlvs information back to themself. | ||
pub custom_tlvs: Option<Vec<u8>>, |
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.
Seems like this is just custom data rather than custom TLVs. Nothing is enforcing that this data is encoded as a TLV stream.
@TheBlueMatt Not sure if you have any thoughts on how if we should make this a Vec<(u64, Vec<u8>)>
instead? And, if so, if we should encode it inside type 65539
or place them directly in ReceiveTlvs
, restricting TLVs >= 2^16 and all the checks that that entails.
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'm not sure why it needs to be TLVs explicitly. ISTM users can just do whatever encoding of whatever data they want and we can store it in a single TLV. We should of course have some documentation that suggests users consider forward-compatibility in their encoding, pushing it all through TLVs on our end seems like overkill, I think? I don't feel incredibly strongly here, though, if we want to eat all the work of verifying TLV types and such we can go that route again 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 now, I've updated the documentation to include a clear note on Forward Compatibility.
Let me know if we prefer moving to the Vec<(u64, Vec<u8>)>
approach—I’d would love to integrate 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.
I mean I'm okay with being overridden here, but ISTM our users aren't gonna care too much about encoding ten different fields here, or if they do they can certainly use their own encoding for the data, and skipping TLVs simplifies our implementation, not to mention saves on allocations and indirection. TLVs are important when we're dealing with other codebases and need to have compatibility as each independently adds fields, but this is not the case here - its just the LDK user storing data for themselves later.
lightning/src/ln/msgs.rs
Outdated
} => { | ||
// We need to update [`ln::outbound_payment::RecipientOnionFields::with_custom_tlvs`] | ||
// to reject any reserved types in the experimental range if new ones are ever | ||
// standardized. | ||
let user_custom_tlv = (!user_custom_tlvs.is_empty()).then(|| (65537, user_custom_tlvs.to_vec())); |
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 do we need to allocate a new Vec
here?
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.
After discussion offline, we concluded that a vec allocation is necessary here.
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.
So... as it turns out, I was able to get this to work without the allocation. BUT it breaks the other use of _encode_varint_length_prefixed_tlv
by OutboundTrampolinePayload
.
Not sure if it is worth trying to get this to work given we are already allocating for both keysend preimage and the invoice request. Here is what I tried: (@TheBlueMatt Feel free to nerd snipe me on this.)
diff --git a/lightning/src/ln/msgs.rs b/lightning/src/ln/msgs.rs
index faffcdd62..2309e45cd 100644
--- a/lightning/src/ln/msgs.rs
+++ b/lightning/src/ln/msgs.rs
@@ -2748,12 +2748,13 @@ impl<'a> Writeable for OutboundOnionPayload<'a> {
// We need to update [`ln::outbound_payment::RecipientOnionFields::with_custom_tlvs`]
// to reject any reserved types in the experimental range if new ones are ever
// standardized.
- let user_custom_tlv = (!user_custom_tlvs.is_empty()).then(|| (65537, user_custom_tlvs.to_vec()));
+ let user_custom_tlv = (!user_custom_tlvs.is_empty()).then(|| (65537, user_custom_tlvs));
let keysend_tlv = keysend_preimage.map(|preimage| (5482373484, preimage.encode()));
- let mut custom_tlvs: Vec<&(u64, Vec<u8>)> = sender_custom_tlvs
+ let mut custom_tlvs: Vec<(u64, &Vec<u8>)> = sender_custom_tlvs
.iter()
- .chain(keysend_tlv.iter())
- .chain(user_custom_tlv.iter())
+ .map(|tlv| (tlv.0, &tlv.1))
+ .chain(keysend_tlv.iter().map(|tlv| (tlv.0, &tlv.1)))
+ .chain(user_custom_tlv.iter().map(|tlv| (tlv.0, *tlv.1)))
.collect();
custom_tlvs.sort_unstable_by_key(|(typ, _)| *typ);
_encode_varint_length_prefixed_tlv!(w, {
@@ -2776,14 +2777,15 @@ impl<'a> Writeable for OutboundOnionPayload<'a> {
// We need to update [`ln::outbound_payment::RecipientOnionFields::with_custom_tlvs`]
// to reject any reserved types in the experimental range if new ones are ever
// standardized.
- let user_custom_tlv = (!user_custom_tlvs.is_empty()).then(|| (65537, user_custom_tlvs.to_vec()));
+ let user_custom_tlv = (!user_custom_tlvs.is_empty()).then(|| (65537, user_custom_tlvs));
let invoice_request_tlv = invoice_request.map(|invreq| (77_777, invreq.encode())); // TODO: update TLV type once the async payments spec is merged
let keysend_tlv = keysend_preimage.map(|preimage| (5482373484, preimage.encode()));
- let mut custom_tlvs: Vec<&(u64, Vec<u8>)> = sender_custom_tlvs
+ let mut custom_tlvs: Vec<(u64, &Vec<u8>)> = sender_custom_tlvs
.iter()
- .chain(user_custom_tlv.iter())
- .chain(invoice_request_tlv.iter())
- .chain(keysend_tlv.iter())
+ .map(|tlv| (tlv.0, &tlv.1))
+ .chain(user_custom_tlv.iter().map(|tlv| (tlv.0, *tlv.1)))
+ .chain(invoice_request_tlv.iter().map(|tlv| (tlv.0, &tlv.1)))
+ .chain(keysend_tlv.iter().map(|tlv| (tlv.0, &tlv.1)))
.collect();
custom_tlvs.sort_unstable_by_key(|(typ, _)| *typ);
_encode_varint_length_prefixed_tlv!(w, {
@@ -2842,7 +2844,7 @@ impl<'a> Writeable for OutboundTrampolinePayload<'a> {
(12, intro_node_blinding_point, option),
(18, HighZeroBytesDroppedBigSize(*total_msat), required),
(20, keysend_preimage, option)
- }, custom_tlvs.iter());
+ }, custom_tlvs.iter().map(|tlv: &(u64, Vec<u8>)| (tlv.0, &tlv.1)));
}
}
Ok(())
diff --git a/lightning/src/util/ser_macros.rs b/lightning/src/util/ser_macros.rs
index 0703aac9e..964747a47 100644
--- a/lightning/src/util/ser_macros.rs
+++ b/lightning/src/util/ser_macros.rs
@@ -158,8 +158,8 @@ macro_rules! _encode_tlv_stream {
$crate::_encode_tlv!($stream, $type, $field, $fieldty);
)*
for tlv in $extra_tlvs {
- let (typ, value): &(u64, Vec<u8>) = tlv;
- $crate::_encode_tlv!($stream, *typ, *value, required_vec);
+ let (typ, value): &(u64, &Vec<u8>) = tlv;
+ $crate::_encode_tlv!($stream, *typ, **value, required_vec);
}
#[allow(unused_mut, unused_variables, unused_assignments)]
@@ -170,7 +170,7 @@ macro_rules! _encode_tlv_stream {
$crate::_check_encoded_tlv_order!(last_seen, $type, $fieldty);
)*
for tlv in $extra_tlvs {
- let (typ, _): &(u64, Vec<u8>) = tlv;
+ let (typ, _): &(u64, &Vec<u8>) = tlv;
$crate::_check_encoded_tlv_order!(last_seen, *typ, required_vec);
}
}
@@ -244,8 +244,8 @@ macro_rules! _encode_varint_length_prefixed_tlv {
$crate::_get_varint_length_prefixed_tlv_length!(len, $type, $field, $fieldty);
)*
for tlv in $extra_tlvs {
- let (typ, value): &(u64, Vec<u8>) = tlv;
- $crate::_get_varint_length_prefixed_tlv_length!(len, *typ, *value, required_vec);
+ let (typ, value): &(u64, &Vec<u8>) = tlv;
+ $crate::_get_varint_length_prefixed_tlv_length!(len, *typ, **value, required_vec);
}
len.0
};
f9a8608
to
d48b605
Compare
d48b605
to
7399605
Compare
7399605
to
ebcc448
Compare
Updated from pr2380.08 to pr2830.09 (diff): Changes:
|
- Introduces the ability for users to include custom data within blinded paths. - For paths used to reach them (e.g., offer or reply paths), users can attach custom data that is returned when the path is utilized.
- This update introduces a clear separation between two types of custom TLVs: those sent by the sender for the user, and those added by the user to the reply path, which are expected to return with the response. - This commit establishes this distinction in the codebase at relevant points. - The next commit will build on this by providing an interface for users to add their own custom TLVs to the reply path, allowing them to receive specific data back in the response. Note: 1. Similar to keysend, for serialization purposes, user_custom_tlv are assigned a specific TLV number. 2. For uniformity, user_custom_tlv are assigned the lowest possible odd number after (1 << 16), which is 65537.
- Building on the previous commit, this update allows users to include their own custom TLVs within the reply path of a sent onion message. - With this, users can attach custom data to the message, which will be returned in the response, providing more flexibility for custom use cases.
ebcc448
to
186e812
Compare
/// Similar to [`ForwardTlvs`], but these TLVs are for the final node. | ||
pub(crate) struct ReceiveTlvs { | ||
#[derive(Clone)] | ||
/// Similar to Forward Tlvs, but these TLVs are for the final node. |
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.
Let's not mention "Forward TLVs" if they are pub(crate)
. Instead, use similar wording from the ForwardTlvs
docs.
pub context: Option<MessageContext> | ||
pub context: Option<MessageContext>, | ||
|
||
/// Custom data set by the user. If `custom_data` is `Some`, it will be returned when the |
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.
s/returned/provided to the message recipient
/// Custom data set by the user. If `custom_data` is `Some`, it will be returned when the | ||
/// blinded path is used. | ||
/// | ||
/// This field allows encoding custom data intended to be retrieved when the blinded path is used. |
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.
s/retrieved/provided back
@@ -402,7 +405,7 @@ pub struct ResponseInstruction { | |||
/// [`Destination`] rather than an explicit [`BlindedMessagePath`] simplifies the logic in | |||
/// [`OnionMessenger::send_onion_message_internal`] somewhat. | |||
destination: Destination, | |||
context: Option<MessageContext>, | |||
reply_data: Option<(MessageContext, Vec<u8>)>, |
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.
Hmmm... we should make these separate fields (both using Option
) otherwise we'll set an empty Vec<u8>
when no custom data was provided. Better to have it be None
so no TLV overhead is added to the blinded path.
/// Custom data set by the receiver in the blinded path used to reach them. | ||
/// | ||
/// For HTLCs received by LDK, these will ultimately bubble back up as | ||
/// [`RecipientOnionFields::user_custom_data`]. |
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.
Could you add user_custom_data
in a separate commit from the rename? This would make the commits more atomic.
@@ -12095,7 +12105,7 @@ where | |||
let nonce = Nonce::from_entropy_source(&*self.entropy_source); | |||
let hmac = payment_hash.hmac_for_offer_payment(nonce, expanded_key); | |||
let context = MessageContext::Offers(OffersContext::InboundPayment { payment_hash, nonce, hmac }); | |||
Some((OffersMessage::Invoice(invoice), responder.respond_with_reply_path(context))) | |||
Some((OffersMessage::Invoice(invoice), responder.respond_with_reply_path(context, custom_data))) |
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 don't want to include the custom data here. That would just be sending it back in a reply. We really want the user to be able to examine it. That's probably outside the scope of this PR, but we should consider how to expose it in a follow-up to #3412.
@@ -1633,7 +1644,7 @@ where | |||
return | |||
} | |||
}; | |||
let response_instructions = self.offers_handler.handle_message(msg, context, responder); | |||
let response_instructions = self.offers_handler.handle_message(msg, context, custom_data, responder); |
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 need to give the data to all handlers, not just the offers handler.
@@ -10452,8 +10452,13 @@ where | |||
.map(|(node_id, _)| *node_id) | |||
.collect::<Vec<_>>(); | |||
|
|||
let receive_tlvs = message::ReceiveTlvs { | |||
context: Some(context), | |||
custom_data: 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.
We'll want to let users set this, too, right? Probably best to do in a follow-up after #3412 is landed, but we will eventually need ways for them to add it to offer/refund message paths and reply paths (e.g., when paying for an offer). Let's update the PR description to mention what is done in this PR and what will be done in a follow-up.
Part of #1970
TODO: