@@ -477,6 +477,7 @@ where L::Target: Logger {
477
477
channel_liquidities : ChannelLiquidities ,
478
478
}
479
479
/// ChannelLiquidities contains live and historical liquidity bounds for each channel.
480
+ #[ derive( Clone ) ]
480
481
pub struct ChannelLiquidities ( HashMap < u64 , ChannelLiquidity > ) ;
481
482
482
483
impl ChannelLiquidities {
@@ -885,6 +886,7 @@ impl ProbabilisticScoringDecayParameters {
885
886
/// first node in the ordering of the channel's counterparties. Thus, swapping the two liquidity
886
887
/// offset fields gives the opposite direction.
887
888
#[ repr( C ) ] // Force the fields in memory to be in the order we specify
889
+ #[ derive( Clone ) ]
888
890
pub struct ChannelLiquidity {
889
891
/// Lower channel liquidity bound in terms of an offset from zero.
890
892
min_liquidity_offset_msat : u64 ,
@@ -1155,6 +1157,15 @@ impl ChannelLiquidity {
1155
1157
}
1156
1158
}
1157
1159
1160
+ fn merge ( & mut self , other : & Self ) {
1161
+ // Take average for min/max liquidity offsets.
1162
+ self . min_liquidity_offset_msat = ( self . min_liquidity_offset_msat + other. min_liquidity_offset_msat ) / 2 ;
1163
+ self . max_liquidity_offset_msat = ( self . max_liquidity_offset_msat + other. max_liquidity_offset_msat ) / 2 ;
1164
+
1165
+ // Merge historical liquidity data.
1166
+ self . liquidity_history . merge ( & other. liquidity_history ) ;
1167
+ }
1168
+
1158
1169
/// Returns a view of the channel liquidity directed from `source` to `target` assuming
1159
1170
/// `capacity_msat`.
1160
1171
fn as_directed (
@@ -1688,6 +1699,91 @@ impl<G: Deref<Target = NetworkGraph<L>>, L: Deref> ScoreUpdate for Probabilistic
1688
1699
}
1689
1700
}
1690
1701
1702
+ /// A probabilistic scorer that combines local and external information to score channels. This scorer is
1703
+ /// shadow-tracking local only scores, so that it becomes possible to cleanly merge external scores when they become
1704
+ /// available.
1705
+ pub struct CombinedScorer < G : Deref < Target = NetworkGraph < L > > , L : Deref > where L :: Target : Logger {
1706
+ local_only_scorer : ProbabilisticScorer < G , L > ,
1707
+ scorer : ProbabilisticScorer < G , L > ,
1708
+ }
1709
+
1710
+ impl < G : Deref < Target = NetworkGraph < L > > + Clone , L : Deref + Clone > CombinedScorer < G , L > where L :: Target : Logger {
1711
+ /// Create a new combined scorer with the given local scorer.
1712
+ pub fn new ( local_scorer : ProbabilisticScorer < G , L > ) -> Self {
1713
+ let decay_params = local_scorer. decay_params ;
1714
+ let network_graph = local_scorer. network_graph . clone ( ) ;
1715
+ let logger = local_scorer. logger . clone ( ) ;
1716
+ let mut scorer = ProbabilisticScorer :: new ( decay_params, network_graph, logger) ;
1717
+
1718
+ scorer. channel_liquidities = local_scorer. channel_liquidities . clone ( ) ;
1719
+
1720
+ Self {
1721
+ local_only_scorer : local_scorer,
1722
+ scorer : scorer,
1723
+ }
1724
+ }
1725
+
1726
+ /// Merge external channel liquidity information into the scorer.
1727
+ pub fn merge ( & mut self , mut external_scores : ChannelLiquidities , duration_since_epoch : Duration ) {
1728
+ // Decay both sets of scores to make them comparable and mergeable.
1729
+ self . local_only_scorer . time_passed ( duration_since_epoch) ;
1730
+ external_scores. time_passed ( duration_since_epoch, self . local_only_scorer . decay_params ) ;
1731
+
1732
+ let local_scores = & self . local_only_scorer . channel_liquidities ;
1733
+
1734
+ // For each channel, merge the external liquidity information with the isolated local liquidity information.
1735
+ for ( scid, mut liquidity) in external_scores. 0 {
1736
+ if let Some ( local_liquidity) = local_scores. get ( & scid) {
1737
+ liquidity. merge ( local_liquidity) ;
1738
+ }
1739
+ self . scorer . channel_liquidities . insert ( scid, liquidity) ;
1740
+ }
1741
+ }
1742
+ }
1743
+
1744
+ impl < G : Deref < Target = NetworkGraph < L > > , L : Deref > ScoreLookUp for CombinedScorer < G , L > where L :: Target : Logger {
1745
+ type ScoreParams = ProbabilisticScoringFeeParameters ;
1746
+
1747
+ fn channel_penalty_msat (
1748
+ & self , candidate : & CandidateRouteHop , usage : ChannelUsage , score_params : & ProbabilisticScoringFeeParameters
1749
+ ) -> u64 {
1750
+ self . scorer . channel_penalty_msat ( candidate, usage, score_params)
1751
+ }
1752
+ }
1753
+
1754
+ impl < G : Deref < Target = NetworkGraph < L > > , L : Deref > ScoreUpdate for CombinedScorer < G , L > where L :: Target : Logger {
1755
+ fn payment_path_failed ( & mut self , path : & Path , short_channel_id : u64 , duration_since_epoch : Duration ) {
1756
+ self . local_only_scorer . payment_path_failed ( path, short_channel_id, duration_since_epoch) ;
1757
+ self . scorer . payment_path_failed ( path, short_channel_id, duration_since_epoch) ;
1758
+ }
1759
+
1760
+ fn payment_path_successful ( & mut self , path : & Path , duration_since_epoch : Duration ) {
1761
+ self . local_only_scorer . payment_path_successful ( path, duration_since_epoch) ;
1762
+ self . scorer . payment_path_successful ( path, duration_since_epoch) ;
1763
+ }
1764
+
1765
+ fn probe_failed ( & mut self , path : & Path , short_channel_id : u64 , duration_since_epoch : Duration ) {
1766
+ self . local_only_scorer . probe_failed ( path, short_channel_id, duration_since_epoch) ;
1767
+ self . scorer . probe_failed ( path, short_channel_id, duration_since_epoch) ;
1768
+ }
1769
+
1770
+ fn probe_successful ( & mut self , path : & Path , duration_since_epoch : Duration ) {
1771
+ self . local_only_scorer . probe_successful ( path, duration_since_epoch) ;
1772
+ self . scorer . probe_successful ( path, duration_since_epoch) ;
1773
+ }
1774
+
1775
+ fn time_passed ( & mut self , duration_since_epoch : Duration ) {
1776
+ self . local_only_scorer . time_passed ( duration_since_epoch) ;
1777
+ self . scorer . time_passed ( duration_since_epoch) ;
1778
+ }
1779
+ }
1780
+
1781
+ impl < G : Deref < Target = NetworkGraph < L > > , L : Deref > Writeable for CombinedScorer < G , L > where L :: Target : Logger {
1782
+ fn write < W : crate :: util:: ser:: Writer > ( & self , writer : & mut W ) -> Result < ( ) , crate :: io:: Error > {
1783
+ self . local_only_scorer . write ( writer)
1784
+ }
1785
+ }
1786
+
1691
1787
#[ cfg( c_bindings) ]
1692
1788
impl < G : Deref < Target = NetworkGraph < L > > , L : Deref > Score for ProbabilisticScorer < G , L >
1693
1789
where L :: Target : Logger { }
@@ -1867,6 +1963,13 @@ mod bucketed_history {
1867
1963
self . buckets [ bucket] = self . buckets [ bucket] . saturating_add ( BUCKET_FIXED_POINT_ONE ) ;
1868
1964
}
1869
1965
}
1966
+
1967
+ /// Returns the average of the buckets between the two trackers.
1968
+ pub ( crate ) fn merge ( & mut self , other : & Self ) -> ( ) {
1969
+ for ( index, bucket) in self . buckets . iter_mut ( ) . enumerate ( ) {
1970
+ * bucket = ( * bucket + other. buckets [ index] ) / 2 ;
1971
+ }
1972
+ }
1870
1973
}
1871
1974
1872
1975
impl_writeable_tlv_based ! ( HistoricalBucketRangeTracker , { ( 0 , buckets, required) } ) ;
@@ -1963,6 +2066,13 @@ mod bucketed_history {
1963
2066
-> DirectedHistoricalLiquidityTracker < & ' a mut HistoricalLiquidityTracker > {
1964
2067
DirectedHistoricalLiquidityTracker { source_less_than_target, tracker : self }
1965
2068
}
2069
+
2070
+ /// Merges the historical liquidity data from another tracker into this one.
2071
+ pub fn merge ( & mut self , other : & Self ) {
2072
+ self . min_liquidity_offset_history . merge ( & other. min_liquidity_offset_history ) ;
2073
+ self . max_liquidity_offset_history . merge ( & other. max_liquidity_offset_history ) ;
2074
+ self . recalculate_valid_point_count ( ) ;
2075
+ }
1966
2076
}
1967
2077
1968
2078
/// A set of buckets representing the history of where we've seen the minimum- and maximum-
@@ -2121,7 +2231,54 @@ mod bucketed_history {
2121
2231
Some ( ( cumulative_success_prob * ( 1024.0 * 1024.0 * 1024.0 ) ) as u64 )
2122
2232
}
2123
2233
}
2234
+
2235
+ #[ cfg( test) ]
2236
+ mod tests {
2237
+ use crate :: routing:: scoring:: ProbabilisticScoringFeeParameters ;
2238
+
2239
+ use super :: { HistoricalBucketRangeTracker , HistoricalLiquidityTracker } ;
2240
+ #[ test]
2241
+ fn historical_liquidity_bucket_merge ( ) {
2242
+ let mut bucket1 = HistoricalBucketRangeTracker :: new ( ) ;
2243
+ bucket1. track_datapoint ( 100 , 1000 ) ;
2244
+ assert_eq ! ( bucket1. buckets, [ 0u16 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 32 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ] ) ;
2245
+
2246
+ let mut bucket2 = HistoricalBucketRangeTracker :: new ( ) ;
2247
+ bucket2. track_datapoint ( 0 , 1000 ) ;
2248
+ assert_eq ! ( bucket2. buckets, [ 32u16 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ] ) ;
2249
+
2250
+ bucket1. merge ( & bucket2) ;
2251
+ assert_eq ! ( bucket1. buckets, [ 16u16 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 16 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ] ) ;
2252
+ }
2253
+
2254
+ #[ test]
2255
+ fn historical_liquidity_tracker_merge ( ) {
2256
+ let params = ProbabilisticScoringFeeParameters :: default ( ) ;
2257
+
2258
+ let probability1: Option < u64 > ;
2259
+ let mut tracker1 = HistoricalLiquidityTracker :: new ( ) ;
2260
+ {
2261
+ let mut directed_tracker1 = tracker1. as_directed_mut ( true ) ;
2262
+ directed_tracker1. track_datapoint ( 100 , 200 , 1000 ) ;
2263
+ probability1 = directed_tracker1. calculate_success_probability_times_billion ( & params, 500 , 1000 ) ;
2264
+ }
2265
+
2266
+ let mut tracker2 = HistoricalLiquidityTracker :: new ( ) ;
2267
+ {
2268
+ let mut directed_tracker2 = tracker2. as_directed_mut ( true ) ;
2269
+ directed_tracker2. track_datapoint ( 200 , 300 , 1000 ) ;
2270
+ }
2271
+
2272
+ tracker1. merge ( & tracker2) ;
2273
+
2274
+ let directed_tracker1 = tracker1. as_directed ( true ) ;
2275
+ let probability = directed_tracker1. calculate_success_probability_times_billion ( & params, 500 , 1000 ) ;
2276
+
2277
+ assert_ne ! ( probability1, probability) ;
2278
+ }
2279
+ }
2124
2280
}
2281
+
2125
2282
use bucketed_history:: { LegacyHistoricalBucketRangeTracker , HistoricalBucketRangeTracker , DirectedHistoricalLiquidityTracker , HistoricalLiquidityTracker } ;
2126
2283
use hashbrown:: hash_map:: Entry ;
2127
2284
@@ -2216,15 +2373,15 @@ impl Readable for ChannelLiquidity {
2216
2373
2217
2374
#[ cfg( test) ]
2218
2375
mod tests {
2219
- use super :: { ChannelLiquidity , HistoricalLiquidityTracker , ProbabilisticScoringFeeParameters , ProbabilisticScoringDecayParameters , ProbabilisticScorer } ;
2376
+ use super :: { ChannelLiquidity , HistoricalLiquidityTracker , ProbabilisticScorer , ProbabilisticScoringDecayParameters , ProbabilisticScoringFeeParameters } ;
2220
2377
use crate :: blinded_path:: BlindedHop ;
2221
2378
use crate :: util:: config:: UserConfig ;
2222
2379
2223
2380
use crate :: ln:: channelmanager;
2224
2381
use crate :: ln:: msgs:: { ChannelAnnouncement , ChannelUpdate , UnsignedChannelAnnouncement , UnsignedChannelUpdate } ;
2225
2382
use crate :: routing:: gossip:: { EffectiveCapacity , NetworkGraph , NodeId } ;
2226
2383
use crate :: routing:: router:: { BlindedTail , Path , RouteHop , CandidateRouteHop , PublicHopCandidate } ;
2227
- use crate :: routing:: scoring:: { ChannelUsage , ScoreLookUp , ScoreUpdate } ;
2384
+ use crate :: routing:: scoring:: { ChannelLiquidities , ChannelUsage , CombinedScorer , ScoreLookUp , ScoreUpdate } ;
2228
2385
use crate :: util:: ser:: { ReadableArgs , Writeable } ;
2229
2386
use crate :: util:: test_utils:: { self , TestLogger } ;
2230
2387
@@ -2234,6 +2391,7 @@ mod tests {
2234
2391
use bitcoin:: network:: Network ;
2235
2392
use bitcoin:: secp256k1:: { PublicKey , Secp256k1 , SecretKey } ;
2236
2393
use core:: time:: Duration ;
2394
+ use std:: rc:: Rc ;
2237
2395
use crate :: io;
2238
2396
2239
2397
fn source_privkey ( ) -> SecretKey {
@@ -3725,6 +3883,59 @@ mod tests {
3725
3883
assert_eq ! ( scorer. historical_estimated_payment_success_probability( 42 , & target, amount_msat, & params, false ) ,
3726
3884
Some ( 0.0 ) ) ;
3727
3885
}
3886
+
3887
+ #[ test]
3888
+ fn combined_scorer ( ) {
3889
+ let logger = TestLogger :: new ( ) ;
3890
+ let network_graph = network_graph ( & logger) ;
3891
+ let params = ProbabilisticScoringFeeParameters :: default ( ) ;
3892
+ let mut scorer = ProbabilisticScorer :: new ( ProbabilisticScoringDecayParameters :: default ( ) , & network_graph, & logger) ;
3893
+ scorer. payment_path_failed ( & payment_path_for_amount ( 600 ) , 42 , Duration :: ZERO ) ;
3894
+
3895
+ let mut combined_scorer = CombinedScorer :: new ( scorer) ;
3896
+
3897
+ // Verify that the combined_scorer has the correct liquidity range after a failed 600 msat payment.
3898
+ let liquidity_range = combined_scorer. scorer . estimated_channel_liquidity_range ( 42 , & target_node_id ( ) ) ;
3899
+ assert_eq ! ( liquidity_range. unwrap( ) , ( 0 , 600 ) ) ;
3900
+
3901
+ let source = source_node_id ( ) ;
3902
+ let usage = ChannelUsage {
3903
+ amount_msat : 750 ,
3904
+ inflight_htlc_msat : 0 ,
3905
+ effective_capacity : EffectiveCapacity :: Total { capacity_msat : 1_000 , htlc_maximum_msat : 1_000 } ,
3906
+ } ;
3907
+
3908
+ {
3909
+ let network_graph = network_graph. read_only ( ) ;
3910
+ let channel = network_graph. channel ( 42 ) . unwrap ( ) ;
3911
+ let ( info, _) = channel. as_directed_from ( & source) . unwrap ( ) ;
3912
+ let candidate = CandidateRouteHop :: PublicHop ( PublicHopCandidate {
3913
+ info,
3914
+ short_channel_id : 42 ,
3915
+ } ) ;
3916
+
3917
+ let penalty = combined_scorer. channel_penalty_msat ( & candidate, usage, & params) ;
3918
+
3919
+ let mut external_liquidity = ChannelLiquidity :: new ( Duration :: ZERO ) ;
3920
+ let logger_rc = Rc :: new ( & logger) ; // Why necessary and not above for the network graph?
3921
+ external_liquidity. as_directed_mut ( & source_node_id ( ) , & target_node_id ( ) , 1_000 ) .
3922
+ successful ( 1000 , Duration :: ZERO , format_args ! ( "test channel" ) , logger_rc. as_ref ( ) ) ;
3923
+
3924
+ let mut external_scores = ChannelLiquidities :: new ( ) ;
3925
+
3926
+ external_scores. insert ( 42 , external_liquidity) ;
3927
+ combined_scorer. merge ( external_scores, Duration :: ZERO ) ;
3928
+
3929
+ let penalty_after_merge = combined_scorer. channel_penalty_msat ( & candidate, usage, & params) ;
3930
+
3931
+ // Since the external source observed a successful payment, the penalty should be lower after the merge.
3932
+ assert ! ( penalty_after_merge < penalty) ;
3933
+ }
3934
+
3935
+ // Verify that after the merge with a successful payment, the liquidity range is increased.
3936
+ let liquidity_range = combined_scorer. scorer . estimated_channel_liquidity_range ( 42 , & target_node_id ( ) ) ;
3937
+ assert_eq ! ( liquidity_range. unwrap( ) , ( 0 , 300 ) ) ;
3938
+ }
3728
3939
}
3729
3940
3730
3941
#[ cfg( ldk_bench) ]
0 commit comments