Skip to content

Commit faf1e23

Browse files
committed
Fallback: add Address getter and use bitcoin types
1 parent 56146e7 commit faf1e23

File tree

5 files changed

+70
-30
lines changed

5 files changed

+70
-30
lines changed

lightning-invoice/Cargo.toml

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@ num-traits = { version = "0.2.8", default-features = false }
2727
bitcoin_hashes = { version = "0.11", default-features = false }
2828
hashbrown = { version = "0.8", optional = true }
2929
serde = { version = "1.0.118", optional = true }
30+
bitcoin = { version = "0.29.0", default-features = false }
3031

3132
[dev-dependencies]
3233
lightning = { version = "0.0.113", path = "../lightning", default-features = false, features = ["_test_utils"] }

lightning-invoice/src/de.rs

Lines changed: 20 additions & 17 deletions
Original file line numberDiff line numberDiff line change
@@ -5,10 +5,13 @@ use core::fmt::{Display, Formatter};
55
use core::num::ParseIntError;
66
use core::str;
77
use core::str::FromStr;
8+
use core::convert::TryFrom;
89

910
use bech32;
1011
use bech32::{u5, FromBase32};
1112

13+
use bitcoin::{PubkeyHash, ScriptHash};
14+
use bitcoin::util::address::WitnessVersion;
1215
use bitcoin_hashes::Hash;
1316
use bitcoin_hashes::sha256;
1417
use crate::prelude::*;
@@ -552,27 +555,24 @@ impl FromBase32 for Fallback {
552555
if bytes.len() < 2 || bytes.len() > 40 {
553556
return Err(ParseError::InvalidSegWitProgramLength);
554557
}
555-
558+
let version = WitnessVersion::try_from(version).expect("0 through 16 are valid SegWit versions");
556559
Ok(Fallback::SegWitProgram {
557560
version: version,
558561
program: bytes
559562
})
560563
},
561564
17 => {
562-
if bytes.len() != 20 {
563-
return Err(ParseError::InvalidPubKeyHashLength);
564-
}
565-
//TODO: refactor once const generics are available
566-
let mut pkh = [0u8; 20];
567-
pkh.copy_from_slice(&bytes);
565+
let pkh = match PubkeyHash::from_slice(&bytes) {
566+
Ok(pkh) => pkh,
567+
Err(bitcoin_hashes::Error::InvalidLength(_, _)) => return Err(ParseError::InvalidPubKeyHashLength),
568+
};
568569
Ok(Fallback::PubKeyHash(pkh))
569570
}
570571
18 => {
571-
if bytes.len() != 20 {
572-
return Err(ParseError::InvalidScriptHashLength);
573-
}
574-
let mut sh = [0u8; 20];
575-
sh.copy_from_slice(&bytes);
572+
let sh = match ScriptHash::from_slice(&bytes) {
573+
Ok(sh) => sh,
574+
Err(bitcoin_hashes::Error::InvalidLength(_, _)) => return Err(ParseError::InvalidScriptHashLength),
575+
};
576576
Ok(Fallback::ScriptHash(sh))
577577
}
578578
_ => Err(ParseError::Skip)
@@ -854,26 +854,29 @@ mod test {
854854
fn test_parse_fallback() {
855855
use crate::Fallback;
856856
use bech32::FromBase32;
857+
use bitcoin::{PubkeyHash, ScriptHash};
858+
use bitcoin::util::address::WitnessVersion;
859+
use bitcoin_hashes::Hash;
857860

858861
let cases = vec![
859862
(
860863
from_bech32("3x9et2e20v6pu37c5d9vax37wxq72un98".as_bytes()),
861-
Ok(Fallback::PubKeyHash([
864+
Ok(Fallback::PubKeyHash(PubkeyHash::from_slice(&[
862865
0x31, 0x72, 0xb5, 0x65, 0x4f, 0x66, 0x83, 0xc8, 0xfb, 0x14, 0x69, 0x59, 0xd3,
863866
0x47, 0xce, 0x30, 0x3c, 0xae, 0x4c, 0xa7
864-
]))
867+
]).unwrap()))
865868
),
866869
(
867870
from_bech32("j3a24vwu6r8ejrss3axul8rxldph2q7z9".as_bytes()),
868-
Ok(Fallback::ScriptHash([
871+
Ok(Fallback::ScriptHash(ScriptHash::from_slice(&[
869872
0x8f, 0x55, 0x56, 0x3b, 0x9a, 0x19, 0xf3, 0x21, 0xc2, 0x11, 0xe9, 0xb9, 0xf3,
870873
0x8c, 0xdf, 0x68, 0x6e, 0xa0, 0x78, 0x45
871-
]))
874+
]).unwrap()))
872875
),
873876
(
874877
from_bech32("qw508d6qejxtdg4y5r3zarvary0c5xw7k".as_bytes()),
875878
Ok(Fallback::SegWitProgram {
876-
version: u5::try_from_u8(0).unwrap(),
879+
version: WitnessVersion::V0,
877880
program: Vec::from(&[
878881
0x75u8, 0x1e, 0x76, 0xe8, 0x19, 0x91, 0x96, 0xd4, 0x54, 0x94, 0x1c, 0x45,
879882
0xd1, 0xb3, 0xa3, 0x23, 0xf1, 0x43, 0x3b, 0xd6

lightning-invoice/src/lib.rs

Lines changed: 41 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,12 @@ extern crate serde;
4545
use std::time::SystemTime;
4646

4747
use bech32::u5;
48+
use bitcoin::Address;
49+
use bitcoin::Network;
50+
use bitcoin::PubkeyHash;
51+
use bitcoin::ScriptHash;
52+
use bitcoin::util::address::Payload;
53+
use bitcoin::util::address::WitnessVersion;
4854
use bitcoin_hashes::Hash;
4955
use bitcoin_hashes::sha256;
5056
use lightning::ln::PaymentSecret;
@@ -442,17 +448,16 @@ pub struct ExpiryTime(Duration);
442448
#[derive(Clone, Debug, Hash, Eq, PartialEq)]
443449
pub struct MinFinalCltvExpiryDelta(pub u64);
444450

445-
// TODO: better types instead onf byte arrays
446451
/// Fallback address in case no LN payment is possible
447452
#[allow(missing_docs)]
448453
#[derive(Clone, Debug, Hash, Eq, PartialEq)]
449454
pub enum Fallback {
450455
SegWitProgram {
451-
version: u5,
456+
version: WitnessVersion,
452457
program: Vec<u8>,
453458
},
454-
PubKeyHash([u8; 20]),
455-
ScriptHash([u8; 20]),
459+
PubKeyHash(PubkeyHash),
460+
ScriptHash(ScriptHash),
456461
}
457462

458463
/// Recoverable signature
@@ -1258,6 +1263,33 @@ impl Invoice {
12581263
self.signed_invoice.fallbacks()
12591264
}
12601265

1266+
/// Returns a list of all fallback addresses as [`Address`]es
1267+
pub fn fallback_addresses(&self) -> Vec<Address> {
1268+
self.fallbacks().iter().map(|fallback| {
1269+
let network = match self.currency() {
1270+
Currency::Bitcoin => Network::Bitcoin,
1271+
Currency::BitcoinTestnet => Network::Testnet,
1272+
Currency::Regtest => Network::Regtest,
1273+
Currency::Simnet => Network::Regtest,
1274+
Currency::Signet => Network::Signet,
1275+
};
1276+
1277+
let payload = match fallback {
1278+
Fallback::SegWitProgram { version, program } => {
1279+
Payload::WitnessProgram { version: *version, program: program.to_vec() }
1280+
}
1281+
Fallback::PubKeyHash(pkh) => {
1282+
Payload::PubkeyHash(*pkh)
1283+
}
1284+
Fallback::ScriptHash(sh) => {
1285+
Payload::ScriptHash(*sh)
1286+
}
1287+
};
1288+
1289+
Address { payload, network }
1290+
}).collect()
1291+
}
1292+
12611293
/// Returns a list of all routes included in the invoice
12621294
pub fn private_routes(&self) -> Vec<&PrivateRoute> {
12631295
self.signed_invoice.private_routes()
@@ -1567,6 +1599,7 @@ impl<'de> Deserialize<'de> for Invoice {
15671599

15681600
#[cfg(test)]
15691601
mod test {
1602+
use bitcoin::Script;
15701603
use bitcoin_hashes::hex::FromHex;
15711604
use bitcoin_hashes::sha256;
15721605

@@ -1930,7 +1963,7 @@ mod test {
19301963
.payee_pub_key(public_key.clone())
19311964
.expiry_time(Duration::from_secs(54321))
19321965
.min_final_cltv_expiry_delta(144)
1933-
.fallback(Fallback::PubKeyHash([0;20]))
1966+
.fallback(Fallback::PubKeyHash(PubkeyHash::from_slice(&[0;20]).unwrap()))
19341967
.private_route(route_1.clone())
19351968
.private_route(route_2.clone())
19361969
.description_hash(sha256::Hash::from_slice(&[3;32][..]).unwrap())
@@ -1956,7 +1989,9 @@ mod test {
19561989
assert_eq!(invoice.payee_pub_key(), Some(&public_key));
19571990
assert_eq!(invoice.expiry_time(), Duration::from_secs(54321));
19581991
assert_eq!(invoice.min_final_cltv_expiry_delta(), 144);
1959-
assert_eq!(invoice.fallbacks(), vec![&Fallback::PubKeyHash([0;20])]);
1992+
assert_eq!(invoice.fallbacks(), vec![&Fallback::PubKeyHash(PubkeyHash::from_slice(&[0;20]).unwrap())]);
1993+
let address = Address::from_script(&Script::new_p2pkh(&PubkeyHash::from_slice(&[0;20]).unwrap()), Network::Testnet).unwrap();
1994+
assert_eq!(invoice.fallback_addresses(), vec![address]);
19601995
assert_eq!(invoice.private_routes(), vec![&PrivateRoute(route_1), &PrivateRoute(route_2)]);
19611996
assert_eq!(
19621997
invoice.description(),

lightning-invoice/src/ser.rs

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -329,7 +329,7 @@ impl ToBase32 for Fallback {
329329
fn write_base32<W: WriteBase32>(&self, writer: &mut W) -> Result<(), <W as WriteBase32>::Err> {
330330
match *self {
331331
Fallback::SegWitProgram {version: v, program: ref p} => {
332-
writer.write_u5(v)?;
332+
writer.write_u5(Into::<u5>::into(v))?;
333333
p.write_base32(writer)
334334
},
335335
Fallback::PubKeyHash(ref hash) => {

lightning-invoice/tests/ser_de.rs

Lines changed: 7 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -5,9 +5,10 @@ extern crate lightning_invoice;
55
extern crate secp256k1;
66
extern crate hex;
77

8+
use bitcoin::util::address::WitnessVersion;
9+
use bitcoin::{PubkeyHash, ScriptHash};
810
use bitcoin_hashes::hex::FromHex;
911
use bitcoin_hashes::{sha256, Hash};
10-
use bech32::u5;
1112
use lightning::ln::PaymentSecret;
1213
use lightning::routing::gossip::RoutingFees;
1314
use lightning::routing::router::{RouteHint, RouteHintHop};
@@ -115,7 +116,7 @@ fn get_test_tuples() -> Vec<(String, SignedRawInvoice, bool, bool)> {
115116
.payment_hash(sha256::Hash::from_hex(
116117
"0001020304050607080900010203040506070809000102030405060708090102"
117118
).unwrap())
118-
.fallback(Fallback::PubKeyHash([49, 114, 181, 101, 79, 102, 131, 200, 251, 20, 105, 89, 211, 71, 206, 48, 60, 174, 76, 167]))
119+
.fallback(Fallback::PubKeyHash(PubkeyHash::from_slice(&[49, 114, 181, 101, 79, 102, 131, 200, 251, 20, 105, 89, 211, 71, 206, 48, 60, 174, 76, 167]).unwrap()))
119120
.build_raw()
120121
.unwrap()
121122
.sign(|_| {
@@ -137,7 +138,7 @@ fn get_test_tuples() -> Vec<(String, SignedRawInvoice, bool, bool)> {
137138
.payment_hash(sha256::Hash::from_hex(
138139
"0001020304050607080900010203040506070809000102030405060708090102"
139140
).unwrap())
140-
.fallback(Fallback::PubKeyHash([4, 182, 31, 125, 193, 234, 13, 201, 148, 36, 70, 76, 196, 6, 77, 197, 100, 217, 30, 137]))
141+
.fallback(Fallback::PubKeyHash(PubkeyHash::from_slice(&[4, 182, 31, 125, 193, 234, 13, 201, 148, 36, 70, 76, 196, 6, 77, 197, 100, 217, 30, 137]).unwrap()))
141142
.private_route(RouteHint(vec![RouteHintHop {
142143
src_node_id: PublicKey::from_slice(&hex::decode(
143144
"029e03a901b85534ff1e92c43c74431f7ce72046060fcf7a95c37e148f78c77255"
@@ -176,7 +177,7 @@ fn get_test_tuples() -> Vec<(String, SignedRawInvoice, bool, bool)> {
176177
.payment_hash(sha256::Hash::from_hex(
177178
"0001020304050607080900010203040506070809000102030405060708090102"
178179
).unwrap())
179-
.fallback(Fallback::ScriptHash([143, 85, 86, 59, 154, 25, 243, 33, 194, 17, 233, 185, 243, 140, 223, 104, 110, 160, 120, 69]))
180+
.fallback(Fallback::ScriptHash(ScriptHash::from_slice(&[143, 85, 86, 59, 154, 25, 243, 33, 194, 17, 233, 185, 243, 140, 223, 104, 110, 160, 120, 69]).unwrap()))
180181
.build_raw()
181182
.unwrap()
182183
.sign(|_| {
@@ -198,7 +199,7 @@ fn get_test_tuples() -> Vec<(String, SignedRawInvoice, bool, bool)> {
198199
.payment_hash(sha256::Hash::from_hex(
199200
"0001020304050607080900010203040506070809000102030405060708090102"
200201
).unwrap())
201-
.fallback(Fallback::SegWitProgram { version: u5::try_from_u8(0).unwrap(),
202+
.fallback(Fallback::SegWitProgram { version: WitnessVersion::V0,
202203
program: vec![117, 30, 118, 232, 25, 145, 150, 212, 84, 148, 28, 69, 209, 179, 163, 35, 241, 67, 59, 214]
203204
})
204205
.build_raw()
@@ -222,7 +223,7 @@ fn get_test_tuples() -> Vec<(String, SignedRawInvoice, bool, bool)> {
222223
.payment_hash(sha256::Hash::from_hex(
223224
"0001020304050607080900010203040506070809000102030405060708090102"
224225
).unwrap())
225-
.fallback(Fallback::SegWitProgram { version: u5::try_from_u8(0).unwrap(),
226+
.fallback(Fallback::SegWitProgram { version: WitnessVersion::V0,
226227
program: vec![24, 99, 20, 60, 20, 197, 22, 104, 4, 189, 25, 32, 51, 86, 218, 19, 108, 152, 86, 120, 205, 77, 39, 161, 184, 198, 50, 150, 4, 144, 50, 98]
227228
})
228229
.build_raw()

0 commit comments

Comments
 (0)