- 
                Notifications
    You must be signed in to change notification settings 
- Fork 422
          BOLT 12 offer encoding and building
          #1719
        
          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
  
    BOLT 12 offer encoding and building
  
  #1719
              Conversation
7545c45    to
    8081c46      
    Compare
  
    | Codecov ReportBase: 90.79% // Head: 90.80% // Increases project coverage by  
 
 Additional details and impacted files@@            Coverage Diff             @@
##             main    #1719      +/-   ##
==========================================
+ Coverage   90.79%   90.80%   +0.01%     
==========================================
  Files          87       89       +2     
  Lines       47604    47971     +367     
  Branches    47604    47971     +367     
==========================================
+ Hits        43221    43562     +341     
- Misses       4383     4409      +26     
 Help us with your feedback. Take ten seconds to tell us how you rate us. Have a feature suggestion? Share it here. ☔ View full report at Codecov. | 
        
          
                lightning/src/util/ser_macros.rs
              
                Outdated
          
        
      | u8 | ||
| }; | ||
| (u16) => { | ||
| ::util::ser::HighZeroBytesDroppedBigSize<u16> | 
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 a bit uneasy about putting these directly in a struct - can we not just keep using them as wrappers at serialization-time rather than actually storing them anywhere?
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.
Pushed a version that only uses them for decoding / encoding. It does remove a lot of intos, which is nice. Though I'd be more uneasy that the macros don't read / write correctly, whereas mistakes in the "struct using wrapper" approach can by caught be the compiler, FWIW.
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 given its all hidden by a struct-defining macro we end up in the same place, 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.
Yeah, mostly just saying more care needs to be taken when implementing and updating the macros. While making the change, I had an intermediate state where encoding and decoding were not compatible.
        
          
                lightning/src/util/ser_macros.rs
              
                Outdated
          
        
      | tlv_record_import!($fieldty$(<$gen>)?); | ||
| )* | ||
|  | ||
| pub(crate) struct $name<'a> { | 
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.
Can't we just name this something slightly different rather than putting it in a module and having to guess the import path required (by listing the Rust prelude explicitly in our code)?
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.
Mainly was trying to keep the usage site looking like a struct. Made it explicit now. Let me know if that is what you were thinking of.
        
          
                lightning/src/util/ser.rs
              
                Outdated
          
        
      | impl<'a> From<&'a String> for WithoutLength<&'a String> { | ||
| fn from(s: &'a String) -> Self { Self(s) } | ||
| } | ||
| impl From<WithoutLength<String>> for String { | 
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 we make these serialization-only wrappers we can drop the copy versions, which seems like the right thing to do.
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'd still need this to simplify deserialization, though it should be a move. I'd imagine it's equivalent to explicitly using .0.
        
          
                lightning/src/offers/offer.rs
              
                Outdated
          
        
      |  | ||
| //! Data structures and encoding for `offer` messages. | ||
| //! | ||
| //! An [`Offer`] is built by the merchant for the user-pays-merchants flow or may be parsed from a | 
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.
Offers are also good for user-pays-user "Cash App" flow, would be nice to amend the docs to include 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.
The "user-pays-merchant" terminology is from the BOLT and is used to differentiate from "merchant-pays-user" (e.g., refunds, ATM) flow, which I use in a future commit. I could use "recipient" instead of the first use of "merchant" here to make it more generic. But not sure if I want to deviate from the spec by adding a new "flow". Open to other ideas so long as we can make it work with the invoice_request docs (see #1597 for current draft), which will be expanded with another builder for the "merchant-pays-user" flow.
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.
Maybe we could do payer-offers-money vs payer-pays-recipient (maybe the word "offer" is confusing there)? ATM isn't really captured by "merchant" either.. I think we've historically not cared too much about sticking with spec naming/copy if there's a reason to deviate, like with wumbo, scid_privacy vs alias, etc.
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.
Maybe we could do
payer-offers-moneyvspayer-pays-recipient(maybe the word "offer" is confusing there)?
Hmm... payer-pays-recipient seems rather tautological. The "payer" is always the one paying a recipient. 😛
I guess "recipient" was a poor suggestion given we want to distinguish who is getting paid. Really, we are trying to distinguish who creates the "offer" (loosely since it could be an invoice_request offer) from who pays the invoice. Or even who creates the invoice.
- In the user-pays-merchantflow, the merchant creates both the offer and the invoice. So the user pays the invoice.
- In the merchant-pays-userflow, the merchant creates the "offer" but the user creates the invoice. So the merchant pays the invoice.
With that in mind, referring to them as "user" and "merchant" seems orthogonal to what we are trying to convey. That is, relative to who created the "offer", who is creating (or paying) the invoice? Or even, is the offer for a product/service or is the "offer" for money?
ATM isn't really captured by "merchant" either..
Well, the merchant owns the ATM, which accepts fiat and is using lighting to deliver the product (i.e. bitcoin). So I think the ATM and the refund cases are still that of the merchant paying bitcoin.
I think we've historically not cared too much about sticking with spec naming/copy if there's a reason to deviate, like with
wumbo,scid_privacyvsalias, etc.
True, though we could contribute back if we have a better alternative in this case. Note that the BOLT says:
Here we use "user" as shorthand for the individual user's lightning node and "merchant" as the shorthand for the node of someone who is selling or has sold something.
Not sure if "individual user's lightning node" is really any clearer, though.
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.
From my PoV, I'm fine with the BOLT copy because it's protocol-dev-facing and gets the point across. For wallet-dev-facing documentation, though, IMO there are different requirements and it'd be best to be inclusive of the range of use cases.
Given this, maybe we could have an expanded Use Cases section in the top-level docs, which include the user-pays-merchant/ATM/etc, and Offer could be documented as "an offer to be paid" and InvoiceRequest as "an offer of money", rather than each referencing a specific "flow"?
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.
From my PoV, I'm fine with the BOLT copy because it's protocol-dev-facing and gets the point across. For wallet-dev-facing documentation, though, IMO there are different requirements and it'd be best to be inclusive of the range of use cases.
Could you enumerate these use cases?
Given this, maybe we could have an expanded
Use Casessection in the top-level docs, which include the user-pays-merchant/ATM/etc, andOffercould be documented as "an offer to be paid" andInvoiceRequestas "an offer of money", rather than each referencing a specific "flow"?
Yeah, I agree that the offers top-level module should have an expanded explanation of use cases and how the various types relate to them. I'd prefer to wait until all the types exist before writing this.
In the offer submodule, I'd like to stick to showing basic example code, though, for the specific types involved. I've updated the wording a bit. Let me know what you think.
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.
The updates look good, and that plan sgtm! The missing use case from my PoV was the Cash App flow (wouldn't want to exclude the people getting tattoos lol).
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.
Basically looks good.
        
          
                lightning/src/offers/offer.rs
              
                Outdated
          
        
      | } | ||
|  | ||
| /// The recipient's public key used to sign invoices. | ||
| pub fn node_id(&self) -> PublicKey { | 
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 want to differentiate the naming here given we should encourage users to sign invoices with a different key than their node id?
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.
Maybe signing_node_id?
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.
Maybe just signing_public_key or so? I don't think we want to connect it to node ids at all?
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.
Changed but left as node_id in the TLV stream to match the spec, for now.
        
          
                lightning/src/offers/offer.rs
              
                Outdated
          
        
      | } | ||
|  | ||
| /// Builds an [`Offer`] from the builder's settings. | ||
| pub fn build(self) -> Offer { | 
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 should sign the offer here, right?
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.
Offers no longer have signatures. The node_id is for signing invoices.
| Needs rebase on the features changes, should be easy. | 
        
          
                lightning/src/offers/offer.rs
              
                Outdated
          
        
      | } | ||
|  | ||
| /// Builds an [`Offer`] from the builder's settings. | ||
| pub fn build(self) -> Offer { | 
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.
Offers no longer have signatures. The node_id is for signing invoices.
        
          
                lightning/src/offers/offer.rs
              
                Outdated
          
        
      | } | ||
|  | ||
| /// The recipient's public key used to sign invoices. | ||
| pub fn node_id(&self) -> PublicKey { | 
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.
Maybe signing_node_id?
        
          
                lightning/src/offers/offer.rs
              
                Outdated
          
        
      | impl Offer { | ||
| /// The chain used for paying the invoice. | ||
| pub fn chain(&self) -> ChainHash { | ||
| // TODO: Update once spec is finalized | 
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.
@TheBlueMatt FYI, I don't think there was any resolution on this: lightning/bolts#798 (comment). Though I'm not sure how you are thinking about negotiating this given the sender would need to know what chains the recipient supports. Currently, the sender states which chain in the invoice_request based on what's in the offer. If it were left out, wouldn't you need another roundtrip?
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.
FYI, also updated this to return a Vec and made some changes to rust-bitcoin to allow us to return a slice.
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 was figuring the invoice would simply contain a list of supported chains and the sender would just pick which they want to use from that list.
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.
Don't you need a chain to figure out how to send the invoice_request in the first place?
a8ff89a    to
    6150762      
    Compare
  
    | Pushed some changes mostly rearranging some of the fixups and cleaning up some things that had been missed. | 
6150762    to
    c4759e1      
    Compare
  
    | 
 Rebased | 
        
          
                lightning/src/offers/offer.rs
              
                Outdated
          
        
      |  | ||
| //! Data structures and encoding for `offer` messages. | ||
| //! | ||
| //! An [`Offer`] is built by the merchant for the user-pays-merchants flow or may be parsed from a | 
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.
Maybe we could do payer-offers-money vs payer-pays-recipient (maybe the word "offer" is confusing there)? ATM isn't really captured by "merchant" either.. I think we've historically not cared too much about sticking with spec naming/copy if there's a reason to deviate, like with wumbo, scid_privacy vs alias, etc.
c4759e1    to
    50d3156      
    Compare
  
    | /// | ||
| /// Successive calls to this method will override the previous setting. | ||
| #[cfg(test)] | ||
| pub fn features(mut self, features: OfferFeatures) -> Self { | 
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 the plan to replace this with a basic_mpp() method when that feature bit is resolved?
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 think so, but I do wonder if we'd prefer to default it somehow. Thoughts?
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.
Hm, I guess I don't have a strong preference between a setter and an un-setter.
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.
Oh, just reading through the updated BOLT 12 draft. There is a separate set of features for each message, so we only need to worry about this in the InvoiceBuilder.
| // You may not use this file except in accordance with one or both of these | ||
| // licenses. | ||
|  | ||
| //! Data structures and encoding for `offer` messages. | 
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.
"Message" implies it's a p2p message to me, is there a way to disambiguate?
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.
Meh... it's splitting hairs a bit, IMHO. invoice_request and invoice aren't really p2p messages either since they are delivered as part of an onion message. And invoice_request in the "send invoice" case doesn't use p2p either. But all of them are messages in a general sense. offer just has a different transport mechanism than the others. 😛
Do you have any alternative phrasing in mind?
        
          
                lightning/src/offers/offer.rs
              
                Outdated
          
        
      | metadata: Option<Vec<u8>>, | ||
| amount: Option<Amount>, | ||
| description: String, | ||
| features: Option<OfferFeatures>, | 
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 this just be empty() and then set features based on setters?
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 preferred using Options otherwise special handling is needed for serializing empty features / Vecs. That said, for Option<Vec<_>> we could hide the Option from the API by returning a slice if Some or the empty slice if None. Similarly with Option<String> using &str. Unfortunately, I don't think we can do the same with OfferFeatures without returning a temporary. 😕
Do we prefer to hide the Option whenever possible? Seems it may make the API somewhat inconsistent but I'm open to the idea.
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 clarify how that makes the API inconsistent? Seems a bit cleaner to avoid returning an Option where it makes sense. I thought the OfferTlvStreamRef could work around the serialization issues but maybe I'm 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.
Could you clarify how that makes the API inconsistent? Seems a bit cleaner to avoid returning an
Optionwhere it makes sense.
There's a few questions that we'd have to answer. If we return &OfferFeatures instead of Option<&OfferFeature>, should we return
- &stror- Option<&String>?
- &[T]or- Option<&Vec<T>>
- u64or- Option<&Amount>
- Durationor- Option<Duration>
I can see the argument for &[T], but I'm not as sure about &str. Is it better for the user to check against None or the empty string to determine if a string is set. Is it ok to treat the empty string as not set, and vice versa?
For amount, we'll eventually need an exchange rate interface, so this could work for each variant. But for None, should we return 0? That said, presumably a wallet would want to display the offer's native currency, so would likely need to stick with Option or add a third variant to Amount for None.
Should we stick with Option for absolute expiry or return the maximum duration if None?
I thought the
OfferTlvStreamRefcould work around the serialization issues but maybe I'm missing something
Yeah, I suppose it could by doing:
let features = if self.features == OfferFeatures::empty() { None } else { Some(&self.features) };But that raises a secondary concern. When should Option be used in OfferContents? In some places like amount it probably should be. But what about issuer or paths? We'd have to be comfortable with treating an empty TLV value as the same as if the TLV record had not been set. This also means an encoded TLV might not be equal with it after parsed and re-encoded because presumably we wouldn't write the record for empty values. Similarly for features.
My intuition is to stick with Option whenever possible, especially if InvoiceRequest or Invoice need to check against the presence. But we probably can go without Option around Vec and OfferFeatures with the above caveat in mind, assuming we're ok with additional conversion logic in as_tlv_stream here and try_from in later PRs. I'm non sure about String, regardless of what the API ultimately looks like.
Let me know what you think.
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, since we already don’t return Options for all getters I didn’t think of that as much of an API inconsistency(?).
- u64 or Option<&Amount>
- Duration or Option
I thought these two could make sense as options still. Certainly expiry being None makes sense IIUC? Fair point that amount could be weird though, is it possible to make Amounts contain NonZeroU64s?
&str or Option<&String>?
IIUC that’s only for issuer, IMO that one arguably makes sense as an Option<&String> for the reasons you stated, no strong feelings though.
features was the only one where I couldn’t see a reason for it to be None vs Some(OfferFeatures::empty() and therefore seemed weird in the public API.
We could get rid of the features() getter entirely and only have individual methods for individual features (offer.supports_mpp()). That way it could stay an Option under the hood so that parsing-and-reencoding always gets the same result. I admit I hadn’t considered that situation though.
Although, I thought since we save the offer bytes, parsing and re-encoding will always be the same anyway? Lmk what I'm missing, not sure I understood your point about how sticking with Option benefits InvReq/Inv
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, since we already don’t return
Options for all getters I didn’t think of that as much of an API inconsistency(?).
Yes, very true.
- u64 or Option<&Amount>
- Duration or Option
I thought these two could make sense as options still. Certainly expiry being
Nonemakes sense IIUC? Fair point that amount could be weird though, is it possible to makeAmounts containNonZeroU64s?
Hadn't thought about NonZeroU64 for amount. Honestly, it's a little tough to work with, and the BOLT doesn't seem to care if it is zero. Just says it's a "minimum".
&str or Option<&String>?
IIUC that’s only for
issuer, IMO that one arguably makes sense as anOption<&String>for the reasons you stated, no strong feelings though.
Yeah, just wasn't sure if we should try to do something similar to Vecs. I think I'm learning towards Option but Option<&str>, which removes some boilerplate from tests.
featureswas the only one where I couldn’t see a reason for it to beNonevsSome(OfferFeatures::empty()and therefore seemed weird in the public API.We could get rid of the
features()getter entirely and only have individual methods for individual features (offer.supports_mpp()). That way it could stay anOptionunder the hood so that parsing-and-reencoding always gets the same result. I admit I hadn’t considered that situation though.
Let's use OfferFeatures::empty() without the Option for now. Could be nice to know if there are odd, unknown features.
Although, I thought since we save the offer
bytes, parsing and re-encoding will always be the same anyway? Lmk what I'm missing, not sure I understood your point about how sticking withOptionbenefitsInvReq/Inv
Ah, right! Literally ran into this yesterday when adding parsing failure tests in the next PR. Had to manually modify the TLV stream to make the test fail. 😛
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.
Lmk what I'm missing, not sure I understood your point about how sticking with
OptionbenefitsInvReq/Inv
Sometime we need to check if a field in the Offer is present to determine how to handle the InvoiceRequest (e.g., amount). During parsing, it's simplest to convert the OfferTlvStream to OfferContents first before performing these checks. So having the Option in OfferContents can be useful.
| /// | ||
| /// Successive calls to this method will override the previous setting. | ||
| #[cfg(test)] | ||
| pub fn features(mut self, features: OfferFeatures) -> Self { | 
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.
Hm, I guess I don't have a strong preference between a setter and an un-setter.
        
          
                lightning/src/offers/offer.rs
              
                Outdated
          
        
      |  | ||
| //! Data structures and encoding for `offer` messages. | ||
| //! | ||
| //! An [`Offer`] is built by the merchant for the user-pays-merchants flow or may be parsed from a | 
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.
The updates look good, and that plan sgtm! The missing use case from my PoV was the Cash App flow (wouldn't want to exclude the people getting tattoos lol).
        
          
                lightning/src/offers/offer.rs
              
                Outdated
          
        
      | metadata: Option<Vec<u8>>, | ||
| amount: Option<Amount>, | ||
| description: String, | ||
| features: Option<OfferFeatures>, | 
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 clarify how that makes the API inconsistent? Seems a bit cleaner to avoid returning an Option where it makes sense. I thought the OfferTlvStreamRef could work around the serialization issues but maybe I'm missing something
3c999b9    to
    51f6195      
    Compare
  
    | @valentinewallace  | 
| This is looking pretty good to me, I think it would be good timing for a second reviewer to come in and I'll check out #1726 in the meantime | 
51f6195    to
    5a4d91e      
    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.
Some docs need more detail, but is looking good.
        
          
                lightning/src/offers/offer.rs
              
                Outdated
          
        
      | pub type CurrencyCode = [u8; 3]; | ||
|  | ||
| tlv_stream!(OfferTlvStream, OfferTlvStreamRef, { | ||
| (2, chains: Vec<ChainHash>), | 
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 these are all required, should we do something like we do for lightning-invoice in the builder to make it a compile-time error to miss a field?
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.
It depends on the context, though note that all the records use Option in  the TLV structs.
For Offer, only description and node_id are required. This is currently enforced at compile-time by OfferBuilder::new requiring both as parameters.
However, for InvoiceRequest used as a "send invoice" (e.g., refund), node_id MUST NOT be set.
I also avoided the lightning-invoice-style compile time checks because IIUC it can't be easily done in bindings since each builder function returns a different type, essentially.
        
          
                lightning/src/offers/offer.rs
              
                Outdated
          
        
      | .unwrap_or_else(|| vec![ChainHash::using_genesis_block(Network::Bitcoin)]) | ||
| } | ||
|  | ||
| /// Metadata set by the originator. Useful for authentication and validating fields. | 
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.
Some of the above comments apply in the getters too.
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.
Ended up expanding comments in Offer and linking to those methods in the OfferBuilder docs to avoid repetition.
| Oh feel free to squash, as far as I care. | 
        
          
                lightning/src/util/ser_macros.rs
              
                Outdated
          
        
      | u8 | ||
| }; | ||
| (u16) => { | ||
| ::util::ser::HighZeroBytesDroppedBigSize<u16> | 
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.
Another question here is how much magic we want in the macro - instead of having to write out a list of type -> wrapper structs here, we could just do it at the callsite, ensuring the callsite readably describes how things are being serialized, rather than it being hidden? That will also avoid having a list of things that could end up randomly doing the Wrong Thing in a surprising way.
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.
Are you suggesting we expand encode_tlv (the call site) such that there is a tlv_record matcher for each of these types? Could do that, but it would result in a lot of repetitive boilerplate. Also, wouldn't prevent doing thing surprising given the same matchers here would have to be used there.
Note that these three helper macros are only used in one place each. I think I could define them as nested macros at the respective calls sites. But then relative to each other they would be spread all about, and it's kinda nice having them all in one place.
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.
Well, I'm still a bit lost on why we need a new tlv_record thing at all - if all the fields are Option wrapped, everything should be option? Then its just a matter of having the tlv_stream macro understand what the field is, which would presumably mean passing u8, HighZeroBytesDroppedBigSize as separate parameters.
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.
ie that the tlv_stream macro will then wrap the self.$field as $wrapper(self.$field).
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.
(Note for discussion that u8 doesn't require HighZeroBytesDroppedBigSize since it is only one byte.)
Hmm... not sure if that's any better or all that straightforward. We'd still need special handling to distinguish between records that need a wrapper from those that don't (e.g., u8, char, and user-defined struct) when defining fields in each of the two TLV stream structs in the macro. This would require another (small) helper macro. Further, for the reference version, we'd still need to handle primitives differently, so tlv_record_ref_type would be needed still.
Additionally, for decoding (opposed to encoding), we aren't wrapping but rather specifying the type that must be used with Readable and then pulling out the inner value.
To your original question, the idea was for the tlv_stream macro to abstract away the encoding requirement of using minimal representations for integers and avoiding duplicating the length of strings and vectors. Seems making the caller explicitly state these puts the onus on them needlessly.
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.
Putting vec_type aside, I can iterate on this to get rid of the tlv_record literal in favor of some variation of option that also takes a wrapper. This is pretty straightforward for encode_tlv. But for decode_tlv we already have a variation for specifying which Readable* trait along with optional args. So the question here is how do we want to formulate the matchers such that a $fieldty:tt can be passed through decode_tlv_stream to decode_tlv and handle either case.
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.
Not quite sure I follow what you are thinking here. Currently, tlv_record assumes the underlying field is an Option, but generally that is not the case and specifically is not the case for vec_type.
Ah, right, I confused myself because its read into an Option but then unwrap'd like the required stuff is.
I can iterate on this to get rid of the tlv_record literal in favor of some variation of option that also takes a wrapper. This is pretty straightforward for encode_tlv
Yep, I'm pretty sure it is, given you already did :)
But for decode_tlv we already have a variation for specifying which Readable* trait along with optional args. So the question here is how do we want to formulate the matchers such that a $fieldty:tt can be passed through decode_tlv_stream to decode_tlv and handle either case.
Right, I don't think we want to pass the $fieldty:tt through, I think we'll want to split it like we do for the _tlv_stream_impl_writeable and then pass the wrapper through without the type itself. I had a patch to do this, but dropped it as I'd ended up rewriting more than necessary but it was pretty easy.
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, I don't think we want to pass the
$fieldty:ttthrough, I think we'll want to split it like we do for the_tlv_stream_impl_writeableand then pass the wrapper through without the type itself. I had a patch to do this, but dropped it as I'd ended up rewriting more than necessary but it was pretty easy.
Ah, I mean how should we pass it through to disambiguate from the matcher taking a trait:
($reader: expr, $field: ident, (option: $trait: ident $(, $read_arg: expr)?)) => {{There, (option: $trait: ident $(, $read_arg: expr)?) is a token tree needed by decode_tlv_stream. I get that we want to pull the wrapper from $fieldty:tt, as you stated, before giving it to decode_tlv_stream. My question is more about what the matcher should look like in decode_tlv to differentiate it from the above matcher.
Could just be something like:
($reader: expr, $field: ident, (option, $wrapper:ident)) => {{But the only difference between that and the earlier one is using a comma instead of a colon and dropping the optional read arg, which isn't very intuitive. Maybe use a wrapper literal?
($reader: expr, $field: ident, (option, wrapper: $wrapper:ident)) => {{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, yea, no super strong opinion here. A wrapper literal makes sense to me - we can revisit it when we have a use for a wrapper that needs an argument.
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.
Ok, I removed the tlv_record literals but wasn't able to pull out the wrapper from $fieldty:tt. I guess it can be done like _tlv_stream_impl_writeable, but I was hoping to avoid the extra () around any types not needing special encoding. Drawback is anyone using this outside of tlv_stream would have to give both the encoding and the type. I suppose it is at least explicit and should fail to compile if given the wrong type.
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.
Working on updating docs.
        
          
                lightning/src/util/ser_macros.rs
              
                Outdated
          
        
      | u8 | ||
| }; | ||
| (u16) => { | ||
| ::util::ser::HighZeroBytesDroppedBigSize<u16> | 
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.
Are you suggesting we expand encode_tlv (the call site) such that there is a tlv_record matcher for each of these types? Could do that, but it would result in a lot of repetitive boilerplate. Also, wouldn't prevent doing thing surprising given the same matchers here would have to be used there.
Note that these three helper macros are only used in one place each. I think I could define them as nested macros at the respective calls sites. But then relative to each other they would be spread all about, and it's kinda nice having them all in one place.
        
          
                lightning/src/offers/offer.rs
              
                Outdated
          
        
      | pub type CurrencyCode = [u8; 3]; | ||
|  | ||
| tlv_stream!(OfferTlvStream, OfferTlvStreamRef, { | ||
| (2, chains: Vec<ChainHash>), | 
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.
It depends on the context, though note that all the records use Option in  the TLV structs.
For Offer, only description and node_id are required. This is currently enforced at compile-time by OfferBuilder::new requiring both as parameters.
However, for InvoiceRequest used as a "send invoice" (e.g., refund), node_id MUST NOT be set.
I also avoided the lightning-invoice-style compile time checks because IIUC it can't be easily done in bindings since each builder function returns a different type, essentially.
5a4d91e    to
    b81931c      
    Compare
  
    d5eae96    to
    14431fd      
    Compare
  
    | This should be ready for review with the most recent spec changes, including disallowing  | 
| } | ||
| } | ||
|  | ||
| tlv_stream!(OfferTlvStream, OfferTlvStreamRef, { | 
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.
FYI, a chunk of unused warnings are introduced here and from Offer::bytes
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... I only see them on intermediate commits. Are you seeing them on the HEAD commit?
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.
Hmm... I cna't seem to reproduce this locally. Can you? Maybe CI is using some flag that catches 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 cannot :( may be related to cargo version as well
4e4b15f    to
    8274164      
    Compare
  
            
          
                lightning/src/offers/offer.rs
              
                Outdated
          
        
      | pub fn supported_quantity(mut self, quantity: Quantity) -> Self { | ||
| self.offer.supported_quantity = quantity; | ||
| self.offer.supported_quantity = match quantity { | ||
| Quantity::Max(n) if n.get() == 1 => Quantity::One, | 
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.
Wondering if I should instead have only two variants (Quantity::Max(n) and Quantity::Many) and default to Quantity::Max(1). Then would translate to the appropriate Option<u64> internally for the TLV stream. Thoughts?
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.
Yea, I'm not a huge fan of the redundant One and Max(n) in the API. We could even just do a NonZeroU64 entirely as far as I'm concerned - we could just use u64::max for the Many entry. I also just generally don't want to overthink the quantity stuff, I have a strong feeling its not going to be heavily 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.
FYI, I changed this to two variants Bounded(n) and Unbounded as discussed offline.
5ccd8c7    to
    dc69f15      
    Compare
  
    | Needs rebase, feel free to squash when you do so IMO. | 
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.
Basically LGTM.
The offer message in BOLT 12 contains a features TLV record. Add a corresponding OfferFeatures type where the length is not included in the serialization as it would be redundant with the record length. Otherwise, define the features to be the same as InvoiceFeatures.
Strings defined by third parties may contain control characters. Provide a wrapper such that these are replaced when displayed. Useful in node aliases and offer fields.
Define an interface for BOLT 12 `offer` messages. The underlying format consists of the original bytes and the parsed contents. The bytes are later needed when constructing an `invoice_request` message. This is because it must mirror all the `offer` TLV records, including unknown ones, which aren't represented in the contents. The contents will be used in `invoice_request` messages to avoid duplication. Some fields while required in a typical user-pays-merchant flow may not be necessary in the merchant-pays-user flow (i.e., refund).
When serializing variable-length types as part of a TLV stream, the length does not need to be serialized as it is already encoded in TLV records. Add a WithoutLength wrapper for this encoding. Replace VecReadWrapper and VecWriteWrapper with this single type to avoid redundant encoders.
BOLT 12's offer message is encoded as a TLV stream (i.e., a sequence of TLV records). impl_writeable_tlv_based can't be used because it writes the overall length of the struct, whereas TLV streams only include the length of each TLV record. Add a `tlv_stream` macro for defining structs used in encoding. TLV records containing a single variable-length type should not encode the types length in the value since it is redundant. Add a wrapper type that can be used within a TLV stream to support the correct behavior during serialization and de-serialization.
dc69f15    to
    e673125      
    Compare
  
    | Rebased and squashed. | 
        
          
                lightning/src/offers/offer.rs
              
                Outdated
          
        
      | /// Builds an [`Offer`] from the builder's settings. | ||
| pub fn build(self) -> Result<Offer, ()> { | ||
| if let Some(Amount::Currency { .. }) = self.offer.amount { | ||
| return Err(()); | 
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.
Are we gonna document that this doesn't work? Why fail here, why not just not bother checking the amount is > MAX_MSAT?
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, not sure why I'm erroring here. The next PR, which adds the error type, already skips the check. Looks like I just missed removing the error earlier.
        
          
                lightning/src/offers/offer.rs
              
                Outdated
          
        
      | /// Returns the amount in millisatoshi. | ||
| pub fn as_msats(&self) -> u64 { | ||
| match self { | ||
| Amount::Currency { .. } => unimplemented!(), | 
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.
Isn't this user-reachable if we deserialize an Offer and the user calls amount.map(as_msats)? Maybe we should just drop the as_msats thing entirely?
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's why it's unimplemented and not unreachable. 🙂  Initially, I wasn't sure how I wanted to handle this. But then I figured we'd probably parameterize this method with a Forex trait or the like. Given the module is private, this could wait until later. Or maybe it's just simple enough to do now. Thoughts?
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 I really understand why we're providing this method at all? The user can match the Amount and look at the msat value or do their own forex conversion. Providing a util method that does that seems like it doesn't accomplish anything.
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.
In InvoiceRequestBuilder, we need to check that the amount provided is sufficient for the Offer when accounting for both currency and quantity to be in accordance with the spec:
- MUST specify `invreq_amount`.`msat` as greater or equal to amount expected by `offer_amount` (and, if present, `offer_currency` and `invreq_quantity`)
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 IIUC that means we have to do currency conversion when we receive an invoicerequest and check that the msat amount matches the given-currency amount? And for that you want to have a util on the Offer to do the conversion, presumably with some trait for currency conversion?
For now, let's just leave it all our, skip the conversion function, and reject invoicerequests with a currency other than bitcoin? That avoids adding broken code here, and we can figure out how/if we want to support currency conversion in the onion-message-handling pipeline later.
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.
Yes, building an async trait to fetch the exchange rate in the middle of a sync message process is.... not going to be fun.
Could we do the same as we do for fee rates? Just have a background task that polls the latest exchange rates.
What I meant is, if the offer is for $20 USD, and we get an invoice_request for $10 in msats, we have to reject that, no?
Correct. Yeah, we're on the same page now. Needed both for a sender requesting an invoice from an offer denominated in USD and a recipient who publishes said offer when checking an invoice_request.
At least there we can force the user to provide it - the request_invoice method can require users pass in the msat value for offers that are in other currencies.
Sure, that's definitely an alternative API. But if we need to check an amount when handling an invoice_request, why not be consistent?
I don't buy this. If you post a permanent offer, it's probably not for a specific amount - you're asking for donations. If you have a specific amount in mind, it's for a specific payer, and probably only valid for some short time period (to pay an order).
Thinking more about it I'm not sure how much of a problem different currency denomination offers is solving. That horse is out of the barn now so not worth trying to revisit it, but I don't think we should bend over backwards building something clean for that use-case until we see how people are using offers in the wild.
I'd imagine it's useful for recurring payments / subscriptions.
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 we do the same as we do for fee rates? Just have a background task that polls the latest exchange rates.
We currently push that complexity on users, which kinda sucks. Further, for exchange rates things may be more complicated - you have to pre-fetch every exchange rate and store that, not just one three values but hundreds. Worse, I'm not sure what the appropriate answer is for a node that doesn't want to query some centralized website and give them a list of all users running lightning - how are we going to do exchange rate lookups for the sample node? We certainly can't query some centralized website, and we don't currently directly support SOCKS to pass it over tor, nor do I know that we want to require Tor to run it.
To make matters worse, most financial data APIs are paid, or at least paid over some rate threshold. A cursory glance at exchange rate data seems to indicate that's mostly true here too.
I'd imagine it's useful for recurring payments / subscriptions.
Mmm, recurring payments is a good point. IIRC that's been pushed from the v1 spec for now?
Sure, that's definitely an alternative API. But if we need to check an amount when handling an invoice_request, why not be consistent?
Right, my question is how to make this simple for users. On the receiving side, you only need to care if you ever provided an offer with a non-BTC currency, on the sending side you always need to care. I don't really have an answer here, mostly because I'm pretty confident all possible answers really really suck. I think it may be totally reasonable for a node to simply not support sending to non-BTC-denominated offers, sadly.
In any case, as for the current PR, I'd really rather not commit code that panics and can't really be fixed, and at least for the offer-issuance side we can trivially support only-btc-denominated offers for now, which seems like a simple way to go.
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 currently push that complexity on users, which kinda sucks. Further, for exchange rates things may be more complicated - you have to pre-fetch every exchange rate and store that, not just one three values but hundreds. Worse, I'm not sure what the appropriate answer is for a node that doesn't want to query some centralized website and give them a list of all users running lightning - how are we going to do exchange rate lookups for the sample node? We certainly can't query some centralized website, and we don't currently directly support SOCKS to pass it over tor, nor do I know that we want to require Tor to run it.
To make matters worse, most financial data APIs are paid, or at least paid over some rate threshold. A cursory glance at exchange rate data seems to indicate that's mostly true here too.
Fair points. Though, FWIW, a user only needs to poll for the currency in which they denominated offers in. For paying offers, the implementation can fetch on demand or return an error if unsupported.
Mmm, recurring payments is a good point. IIRC that's been pushed from the v1 spec for now?
Correct.
Right, my question is how to make this simple for users. On the receiving side, you only need to care if you ever provided an offer with a non-BTC currency, on the sending side you always need to care. I don't really have an answer here, mostly because I'm pretty confident all possible answers really really suck. I think it may be totally reasonable for a node to simply not support sending to non-BTC-denominated offers, sadly.
If InvoiceRequestBuilder::amount_msat (or the builder itself) is parameterized by the exchange trait, then we could define a default implementation that errors, I suppose, if Offer::amount is a Currency.
In any case, as for the current PR, I'd really rather not commit code that panics and can't really be fixed, and at least for the offer-issuance side we can trivially support only-btc-denominated offers for now, which seems like a simple way to go.
So essentially have InvoiceRequestBuilder error if attempting to pay a non-BTC-denominated offer?
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.
Changed OfferBuilder::amount to OfferBuilder::amount_msats and updated dependent PRs accordingly.
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.
Fair points. Though, FWIW, a user only needs to poll for the currency in which they denominated offers in. For paying offers, the implementation can fetch on demand or return an error if unsupported.
Right, but how do we capture that in an interface that makes sense and is easy to use and hard to mis-use :(
So essentially have InvoiceRequestBuilder error if attempting to pay a non-BTC-denominated offer?
Yea, for now.
20da3f3    to
    0846064      
    Compare
  
    | return Err(()); | ||
| } | ||
| }, | ||
| Some(Amount::Currency { .. }) => unreachable!(), | 
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.
Can we just not fail here? I agree I think its unreachable but there's no reason to actually panic vs generating an offer that we may reject later.
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.
FWIW, it already fails in the next PR when we have actual errors to differentiate (cadc69c). Do you mind if we just leave this as is until then to avoid more rebasing?
Add a builder for creating offers given a required description and node_id. Other settings are optional and duplicative settings will override previous settings for non-Vec fields.
0846064    to
    8ba09e0      
    Compare
  
    
Define an interface for BOLT 12
offermessages. The underlying format consists of the original bytes and the parsed contents.The bytes are later needed when constructing an
invoice_requestmessage. This is because it must mirror all theofferTLV records, including unknown ones, which aren't represented in the contents.The contents will be used in
invoice_requestmessages to avoid duplication. Some fields while required in a typical user-pays-merchant flow may not be necessary in the merchant-pays-user flow (i.e., refund).Also, defines a builder for constructing an offer given a required description and node id.
See #1597 for ongoing BOLT 12 work depending on this PR.