@@ -230,6 +230,10 @@ pub struct PaymentParameters {
230
230
/// channel, as a power of 1/2. A higher value prefers to send the payment using more MPP parts
231
231
/// whereas a lower value prefers to send larger MPP parts, potentially saturating channels.
232
232
///
233
+ /// Note that this restriction will be relaxed during pathfinding after paths which meet this
234
+ /// restriction have been found. While paths which meet this criteria will be searched for, it
235
+ /// is ultimately up to the scorer to select them over other paths.
236
+ ///
233
237
/// A value of 0 will allow payments up to and including a channel's total announced usable
234
238
/// capacity, a value of one will only use up to half its capacity, two 1/4, etc.
235
239
///
@@ -257,10 +261,6 @@ impl PaymentParameters {
257
261
expiry_time : None ,
258
262
max_total_cltv_expiry_delta : DEFAULT_MAX_TOTAL_CLTV_EXPIRY_DELTA ,
259
263
max_path_count : DEFAULT_MAX_PATH_COUNT ,
260
- #[ cfg( test) ] // Many tests were written prior to the introduction of this parameter, so
261
- // we leave it as 0 by default in tests, and change it for a few.
262
- max_channel_saturation_power_of_half : 0 ,
263
- #[ cfg( not( test) ) ]
264
264
max_channel_saturation_power_of_half : 1 ,
265
265
}
266
266
}
@@ -304,6 +304,13 @@ impl PaymentParameters {
304
304
pub fn with_max_path_count ( self , max_path_count : u8 ) -> Self {
305
305
Self { max_path_count, ..self }
306
306
}
307
+
308
+ /// Includes a limit for the maximum number of payment paths that may be used.
309
+ ///
310
+ /// (C-not exported) since bindings don't support move semantics
311
+ pub fn with_max_channel_saturation_power_of_half ( self , max_channel_saturation_power_of_half : u8 ) -> Self {
312
+ Self { max_channel_saturation_power_of_half, ..self }
313
+ }
307
314
}
308
315
309
316
/// A list of hops along a payment path terminating with a channel to the recipient.
@@ -488,6 +495,17 @@ fn max_htlc_from_capacity(capacity: EffectiveCapacity, max_channel_saturation_po
488
495
}
489
496
}
490
497
498
+ fn iter_equal < I1 : Iterator , I2 : Iterator > ( mut iter_a : I1 , mut iter_b : I2 )
499
+ -> bool where I1 :: Item : PartialEq < I2 :: Item > {
500
+ loop {
501
+ let a = iter_a. next ( ) ;
502
+ let b = iter_b. next ( ) ;
503
+ if a. is_none ( ) && b. is_none ( ) { return true ; }
504
+ if a. is_none ( ) || b. is_none ( ) { return false ; }
505
+ if a. unwrap ( ) . ne ( & b. unwrap ( ) ) { return false ; }
506
+ }
507
+ }
508
+
491
509
/// It's useful to keep track of the hops associated with the fees required to use them,
492
510
/// so that we can choose cheaper paths (as per Dijkstra's algorithm).
493
511
/// Fee values should be updated only in the context of the whole path, see update_value_and_recompute_fees.
@@ -595,10 +613,9 @@ impl<'a> PaymentPath<'a> {
595
613
// to the fees being paid not lining up with the actual limits.
596
614
//
597
615
// Note that this function is not aware of the available_liquidity limit, and thus does not
598
- // support increasing the value being transferred.
616
+ // support increasing the value being transferred beyond what was selected during the initial
617
+ // routing passes.
599
618
fn update_value_and_recompute_fees ( & mut self , value_msat : u64 ) {
600
- assert ! ( value_msat <= self . hops. last( ) . unwrap( ) . 0 . fee_msat) ;
601
-
602
619
let mut total_fee_paid_msat = 0 as u64 ;
603
620
for i in ( 0 ..self . hops . len ( ) ) . rev ( ) {
604
621
let last_hop = i == self . hops . len ( ) - 1 ;
@@ -906,6 +923,11 @@ where L::Target: Logger {
906
923
final_value_msat
907
924
} ;
908
925
926
+ // When we start collecting routes we enforce the max_channel_saturation_power_of_half
927
+ // requirement strictly. After we've collected enough (or if we fail to find new routes) we
928
+ // drop the requirement by setting this to 0.
929
+ let mut channel_saturation_pow_half = payment_params. max_channel_saturation_power_of_half ;
930
+
909
931
// Keep track of how much liquidity has been used in selected channels. Used to determine
910
932
// if the channel can be used by additional MPP paths or to inform path finding decisions. It is
911
933
// aware of direction *only* to ensure that the correct htlc_maximum_msat value is used. Hence,
@@ -959,7 +981,7 @@ where L::Target: Logger {
959
981
if $src_node_id != $dest_node_id {
960
982
let short_channel_id = $candidate. short_channel_id( ) ;
961
983
let effective_capacity = $candidate. effective_capacity( ) ;
962
- let htlc_maximum_msat = max_htlc_from_capacity( effective_capacity, payment_params . max_channel_saturation_power_of_half ) ;
984
+ let htlc_maximum_msat = max_htlc_from_capacity( effective_capacity, channel_saturation_pow_half ) ;
963
985
964
986
// It is tricky to subtract $next_hops_fee_msat from available liquidity here.
965
987
// It may be misleading because we might later choose to reduce the value transferred
@@ -1531,8 +1553,7 @@ where L::Target: Logger {
1531
1553
. and_modify ( |used_liquidity_msat| * used_liquidity_msat += spent_on_hop_msat)
1532
1554
. or_insert ( spent_on_hop_msat) ;
1533
1555
let hop_capacity = hop. candidate . effective_capacity ( ) ;
1534
- let hop_max_msat = max_htlc_from_capacity ( hop_capacity,
1535
- payment_params. max_channel_saturation_power_of_half ) ;
1556
+ let hop_max_msat = max_htlc_from_capacity ( hop_capacity, channel_saturation_pow_half) ;
1536
1557
if * used_liquidity_msat == hop_max_msat {
1537
1558
// If this path used all of this channel's available liquidity, we know
1538
1559
// this path will not be selected again in the next loop iteration.
@@ -1579,6 +1600,10 @@ where L::Target: Logger {
1579
1600
}
1580
1601
1581
1602
if !allow_mpp {
1603
+ if !found_new_path && channel_saturation_pow_half != 0 {
1604
+ channel_saturation_pow_half = 0 ;
1605
+ continue ' paths_collection;
1606
+ }
1582
1607
// If we don't support MPP, no use trying to gather more value ever.
1583
1608
break ' paths_collection;
1584
1609
}
@@ -1588,7 +1613,9 @@ where L::Target: Logger {
1588
1613
// iteration.
1589
1614
// In the latter case, making another path finding attempt won't help,
1590
1615
// because we deterministically terminated the search due to low liquidity.
1591
- if already_collected_value_msat >= recommended_value_msat || !found_new_path {
1616
+ if !found_new_path && channel_saturation_pow_half != 0 {
1617
+ channel_saturation_pow_half = 0 ;
1618
+ } else if already_collected_value_msat >= recommended_value_msat || !found_new_path {
1592
1619
log_trace ! ( logger, "Have now collected {} msat (seeking {} msat) in paths. Last path loop {} a new path." ,
1593
1620
already_collected_value_msat, recommended_value_msat, if found_new_path { "found" } else { "did not find" } ) ;
1594
1621
break ' paths_collection;
@@ -1704,8 +1731,32 @@ where L::Target: Logger {
1704
1731
// Step (9).
1705
1732
// Select the best route by lowest total cost.
1706
1733
drawn_routes. sort_unstable_by_key ( |paths| paths. iter ( ) . map ( |path| path. get_cost_msat ( ) ) . sum :: < u64 > ( ) ) ;
1734
+ let selected_route = drawn_routes. first_mut ( ) . unwrap ( ) ;
1735
+
1736
+ // Sort by the path itself and combine redundant paths.
1737
+ // Note that we sort by SCIDs alone as its simpler but when combining we have to ensure we
1738
+ // compare both SCIDs and NodeIds as individual nodes may use random aliases causing collisions
1739
+ // across nodes.
1740
+ selected_route. sort_unstable_by_key ( |path| {
1741
+ let mut key = [ 0u64 ; MAX_PATH_LENGTH_ESTIMATE as usize ] ;
1742
+ debug_assert ! ( path. hops. len( ) <= key. len( ) ) ;
1743
+ for ( scid, key) in path. hops . iter ( ) . map ( |h| h. 0 . candidate . short_channel_id ( ) ) . zip ( key. iter_mut ( ) ) {
1744
+ * key = scid;
1745
+ }
1746
+ key
1747
+ } ) ;
1748
+ for idx in 0 ..( selected_route. len ( ) - 1 ) {
1749
+ if idx + 1 >= selected_route. len ( ) { break ; }
1750
+ if iter_equal ( selected_route[ idx ] . hops . iter ( ) . map ( |h : & ( PathBuildingHop < ' _ > , _ ) | ( h. 0 . candidate . short_channel_id ( ) , h. 0 . node_id ) ) ,
1751
+ selected_route[ idx + 1 ] . hops . iter ( ) . map ( |h : & ( PathBuildingHop < ' _ > , _ ) | ( h. 0 . candidate . short_channel_id ( ) , h. 0 . node_id ) ) ) {
1752
+ let new_value = selected_route[ idx] . get_value_msat ( ) + selected_route[ idx + 1 ] . get_value_msat ( ) ;
1753
+ selected_route[ idx] . update_value_and_recompute_fees ( new_value) ;
1754
+ selected_route. remove ( idx + 1 ) ;
1755
+ }
1756
+ }
1757
+
1707
1758
let mut selected_paths = Vec :: < Vec < Result < RouteHop , LightningError > > > :: new ( ) ;
1708
- for payment_path in drawn_routes . first ( ) . unwrap ( ) {
1759
+ for payment_path in selected_route {
1709
1760
let mut path = payment_path. hops . iter ( ) . map ( |( payment_hop, node_features) | {
1710
1761
Ok ( RouteHop {
1711
1762
pubkey : PublicKey :: from_slice ( payment_hop. node_id . as_slice ( ) ) . map_err ( |_| LightningError { err : format ! ( "Public key {:?} is invalid" , & payment_hop. node_id) , action : ErrorAction :: IgnoreAndLog ( Level :: Trace ) } ) ?,
@@ -4791,17 +4842,18 @@ mod tests {
4791
4842
4792
4843
// Get a route for 100 sats and check that we found the MPP route no problem and didn't
4793
4844
// overpay at all.
4794
- let route = get_route ( & our_id, & payment_params, & network_graph. read_only ( ) , None , 100_000 , 42 , Arc :: clone ( & logger) , & scorer, & random_seed_bytes) . unwrap ( ) ;
4845
+ let mut route = get_route ( & our_id, & payment_params, & network_graph. read_only ( ) , None , 100_000 , 42 , Arc :: clone ( & logger) , & scorer, & random_seed_bytes) . unwrap ( ) ;
4795
4846
assert_eq ! ( route. paths. len( ) , 2 ) ;
4796
- // Paths are somewhat randomly ordered, but:
4797
- // * the first is channel 2 (1 msat fee) -> channel 4 -> channel 42
4798
- // * the second is channel 1 (0 fee, but 99 sat maximum) -> channel 3 -> channel 42
4799
- assert_eq ! ( route. paths[ 0 ] [ 0 ] . short_channel_id, 2 ) ;
4800
- assert_eq ! ( route. paths[ 0 ] [ 0 ] . fee_msat, 1 ) ;
4801
- assert_eq ! ( route. paths[ 0 ] [ 2 ] . fee_msat, 1_000 ) ;
4802
- assert_eq ! ( route. paths[ 1 ] [ 0 ] . short_channel_id, 1 ) ;
4803
- assert_eq ! ( route. paths[ 1 ] [ 0 ] . fee_msat, 0 ) ;
4804
- assert_eq ! ( route. paths[ 1 ] [ 2 ] . fee_msat, 99_000 ) ;
4847
+ route. paths . sort_by_key ( |path| path[ 0 ] . short_channel_id ) ;
4848
+ // Paths are manually ordered ordered by SCID, so:
4849
+ // * the first is channel 1 (0 fee, but 99 sat maximum) -> channel 3 -> channel 42
4850
+ // * the second is channel 2 (1 msat fee) -> channel 4 -> channel 42
4851
+ assert_eq ! ( route. paths[ 0 ] [ 0 ] . short_channel_id, 1 ) ;
4852
+ assert_eq ! ( route. paths[ 0 ] [ 0 ] . fee_msat, 0 ) ;
4853
+ assert_eq ! ( route. paths[ 0 ] [ 2 ] . fee_msat, 99_000 ) ;
4854
+ assert_eq ! ( route. paths[ 1 ] [ 0 ] . short_channel_id, 2 ) ;
4855
+ assert_eq ! ( route. paths[ 1 ] [ 0 ] . fee_msat, 1 ) ;
4856
+ assert_eq ! ( route. paths[ 1 ] [ 2 ] . fee_msat, 1_000 ) ;
4805
4857
assert_eq ! ( route. get_total_fees( ) , 1 ) ;
4806
4858
assert_eq ! ( route. get_total_amount( ) , 100_000 ) ;
4807
4859
}
@@ -4815,7 +4867,8 @@ mod tests {
4815
4867
let scorer = test_utils:: TestScorer :: with_penalty ( 0 ) ;
4816
4868
let keys_manager = test_utils:: TestKeysInterface :: new ( & [ 0u8 ; 32 ] , Network :: Testnet ) ;
4817
4869
let random_seed_bytes = keys_manager. get_secure_random_bytes ( ) ;
4818
- let payment_params = PaymentParameters :: from_node_id ( nodes[ 2 ] ) . with_features ( InvoiceFeatures :: known ( ) ) ;
4870
+ let payment_params = PaymentParameters :: from_node_id ( nodes[ 2 ] ) . with_features ( InvoiceFeatures :: known ( ) )
4871
+ . with_max_channel_saturation_power_of_half ( 0 ) ;
4819
4872
4820
4873
// We need a route consisting of 3 paths:
4821
4874
// From our node to node2 via node0, node7, node1 (three paths one hop each).
@@ -5263,12 +5316,13 @@ mod tests {
5263
5316
assert_eq ! ( route. paths[ 0 ] . len( ) , 1 ) ;
5264
5317
assert_eq ! ( route. paths[ 1 ] . len( ) , 1 ) ;
5265
5318
5319
+ assert ! ( ( route. paths[ 0 ] [ 0 ] . short_channel_id == 3 && route. paths[ 1 ] [ 0 ] . short_channel_id == 2 ) ||
5320
+ ( route. paths[ 0 ] [ 0 ] . short_channel_id == 2 && route. paths[ 1 ] [ 0 ] . short_channel_id == 3 ) ) ;
5321
+
5266
5322
assert_eq ! ( route. paths[ 0 ] [ 0 ] . pubkey, nodes[ 0 ] ) ;
5267
- assert_eq ! ( route. paths[ 0 ] [ 0 ] . short_channel_id, 3 ) ;
5268
5323
assert_eq ! ( route. paths[ 0 ] [ 0 ] . fee_msat, 50_000 ) ;
5269
5324
5270
5325
assert_eq ! ( route. paths[ 1 ] [ 0 ] . pubkey, nodes[ 0 ] ) ;
5271
- assert_eq ! ( route. paths[ 1 ] [ 0 ] . short_channel_id, 2 ) ;
5272
5326
assert_eq ! ( route. paths[ 1 ] [ 0 ] . fee_msat, 50_000 ) ;
5273
5327
}
5274
5328
0 commit comments