Skip to content

Commit b677ddb

Browse files
committed
Invoice serialization in iterative style
1 parent f4c8340 commit b677ddb

File tree

6 files changed

+198
-347
lines changed

6 files changed

+198
-347
lines changed

fuzz/src/bolt11_deser.rs

-1
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,6 @@ use bech32::Fe32;
1212
use bitcoin::secp256k1::{Secp256k1, SecretKey};
1313
use lightning_invoice::{
1414
Bolt11Invoice, FromBase32, RawBolt11Invoice, RawDataPart, RawHrp, RawTaggedField, TaggedField,
15-
ToBase32,
1615
};
1716
use std::str::FromStr;
1817

lightning-invoice/src/de.rs

+6-7
Original file line numberDiff line numberDiff line change
@@ -40,22 +40,21 @@ impl FromBase32 for Vec<u8> {
4040
impl FromBase32 for PaymentSecret {
4141
type Err = CheckedHrpstringError;
4242

43-
fn from_base32(field_data: &[Fe32]) -> Result<PaymentSecret, CheckedHrpstringError> {
43+
fn from_base32(field_data: &[Fe32]) -> Result<Self, Self::Err> {
4444
if field_data.len() != 52 {
4545
return Err(CheckedHrpstringError::Checksum(ChecksumError::InvalidLength)) // TODO(bech32): not entirely accurate
46-
} else {
47-
let data_bytes = Vec::<u8>::from_base32(field_data)?;
48-
let mut payment_secret = [0; 32];
49-
payment_secret.copy_from_slice(&data_bytes);
50-
Ok(PaymentSecret(payment_secret))
5146
}
47+
let data_bytes = Vec::<u8>::from_base32(field_data)?;
48+
let mut payment_secret = [0; 32];
49+
payment_secret.copy_from_slice(&data_bytes);
50+
Ok(PaymentSecret(payment_secret))
5251
}
5352
}
5453

5554
impl FromBase32 for Bolt11InvoiceFeatures {
5655
type Err = CheckedHrpstringError;
5756

58-
fn from_base32(field_data: &[Fe32]) -> Result<Bolt11InvoiceFeatures, CheckedHrpstringError> {
57+
fn from_base32(field_data: &[Fe32]) -> Result<Self, Self::Err> {
5958
// Explanation for the "7": the normal way to round up when dividing is to add the divisor
6059
// minus one before dividing
6160
let length_bytes = (field_data.len() * 5 + 7) / 8 as usize;

lightning-invoice/src/lib.rs

+13-39
Original file line numberDiff line numberDiff line change
@@ -77,44 +77,7 @@ mod prelude {
7777

7878
use crate::prelude::*;
7979

80-
/// Interface to write `Fe32`s into a sink
81-
pub trait WriteBase32 {
82-
/// Write error
83-
type Err: fmt::Debug;
84-
85-
/// Write a `Fe32` slice
86-
fn write(&mut self, data: &Vec<Fe32>) -> Result<(), Self::Err> {
87-
for b in data {
88-
self.write_fe32(*b)?;
89-
}
90-
Ok(())
91-
}
92-
93-
/// Write a single `Fe32`
94-
fn write_fe32(&mut self, data: Fe32) -> Result<(), Self::Err>;
95-
}
96-
97-
/// A trait for converting a value to a type `T` that represents a `Fe32` slice.
98-
pub trait ToBase32 {
99-
/// Convert `Self` to base32 vector
100-
fn to_base32(&self) -> Vec<Fe32> {
101-
let mut vec = Vec::new();
102-
self.write_base32(&mut vec).unwrap();
103-
vec
104-
}
105-
106-
/// Encode as base32 and write it to the supplied writer
107-
/// Implementations shouldn't allocate.
108-
fn write_base32<W: WriteBase32>(&self, writer: &mut W) -> Result<(), <W as WriteBase32>::Err>;
109-
}
110-
111-
/// Interface to calculate the length of the base32 representation before actually serializing
112-
pub trait Base32Len: ToBase32 {
113-
/// Calculate the base32 serialized length
114-
fn base32_len(&self) -> usize;
115-
}
116-
117-
/// Trait for paring/converting base32 slice. It is the reciprocal of `ToBase32`.
80+
/// Trait for paring/converting base32 slice.
11881
pub trait FromBase32: Sized {
11982
/// The associated error which can be returned from parsing (e.g. because of bad padding).
12083
type Err;
@@ -347,6 +310,15 @@ pub struct RawHrp {
347310
pub si_prefix: Option<SiPrefix>,
348311
}
349312

313+
impl RawHrp {
314+
/// Convert to bech32::Hrp
315+
pub fn to_hrp(&self) -> bech32::Hrp {
316+
let hrp_str = self.to_string();
317+
let s = core::str::from_utf8(&hrp_str.as_bytes()).expect("asserted to be ASCII");
318+
bech32::Hrp::parse_unchecked(s)
319+
}
320+
}
321+
350322
/// Data of the [`RawBolt11Invoice`] that is encoded in the data part
351323
#[derive(Eq, PartialEq, Debug, Clone, Hash, Ord, PartialOrd)]
352324
pub struct RawDataPart {
@@ -1048,9 +1020,11 @@ impl RawBolt11Invoice {
10481020

10491021
/// Calculate the hash of the encoded `RawBolt11Invoice` which should be signed.
10501022
pub fn signable_hash(&self) -> [u8; 32] {
1023+
use crate::ser::Base32Iterable;
1024+
10511025
RawBolt11Invoice::hash_from_parts(
10521026
self.hrp.to_string().as_bytes(),
1053-
&self.data.to_base32()
1027+
&self.data.fe_iter().collect::<Vec<Fe32>>(),
10541028
)
10551029
}
10561030

0 commit comments

Comments
 (0)