Skip to content

Commit fdf2b5d

Browse files
committed
add functionality to periodically update routing scores from an external http source
1 parent e1c2fa4 commit fdf2b5d

File tree

5 files changed

+143
-34
lines changed

5 files changed

+143
-34
lines changed

Cargo.toml

+23-23
Original file line numberDiff line numberDiff line change
@@ -28,27 +28,27 @@ panic = 'abort' # Abort on panic
2828
default = []
2929

3030
[dependencies]
31-
lightning = { version = "0.1.0", features = ["std"] }
32-
lightning-types = { version = "0.2.0" }
33-
lightning-invoice = { version = "0.33.0", features = ["std"] }
34-
lightning-net-tokio = { version = "0.1.0" }
35-
lightning-persister = { version = "0.1.0" }
36-
lightning-background-processor = { version = "0.1.0", features = ["futures"] }
37-
lightning-rapid-gossip-sync = { version = "0.1.0" }
38-
lightning-block-sync = { version = "0.1.0", features = ["rpc-client", "tokio"] }
39-
lightning-transaction-sync = { version = "0.1.0", features = ["esplora-async-https", "time"] }
40-
lightning-liquidity = { version = "0.1.0", features = ["std"] }
41-
42-
#lightning = { git = "https://github.com/lightningdevkit/rust-lightning", branch = "main", features = ["std"] }
43-
#lightning-types = { git = "https://github.com/lightningdevkit/rust-lightning", branch = "main" }
44-
#lightning-invoice = { git = "https://github.com/lightningdevkit/rust-lightning", branch = "main", features = ["std"] }
45-
#lightning-net-tokio = { git = "https://github.com/lightningdevkit/rust-lightning", branch = "main" }
46-
#lightning-persister = { git = "https://github.com/lightningdevkit/rust-lightning", branch = "main" }
47-
#lightning-background-processor = { git = "https://github.com/lightningdevkit/rust-lightning", branch = "main", features = ["futures"] }
48-
#lightning-rapid-gossip-sync = { git = "https://github.com/lightningdevkit/rust-lightning", branch = "main" }
49-
#lightning-block-sync = { git = "https://github.com/lightningdevkit/rust-lightning", branch = "main", features = ["rpc-client", "tokio"] }
50-
#lightning-transaction-sync = { git = "https://github.com/lightningdevkit/rust-lightning", branch = "main", features = ["esplora-async-https", "time"] }
51-
#lightning-liquidity = { git = "https://github.com/lightningdevkit/rust-lightning", branch = "main" }
31+
#lightning = { version = "0.1.0", features = ["std"] }
32+
#lightning-types = { version = "0.2.0" }
33+
#lightning-invoice = { version = "0.33.0", features = ["std"] }
34+
#lightning-net-tokio = { version = "0.1.0" }
35+
#lightning-persister = { version = "0.1.0" }
36+
#lightning-background-processor = { version = "0.1.0", features = ["futures"] }
37+
#lightning-rapid-gossip-sync = { version = "0.1.0" }
38+
#lightning-block-sync = { version = "0.1.0", features = ["rpc-client", "tokio"] }
39+
#lightning-transaction-sync = { version = "0.1.0", features = ["esplora-async-https", "time"] }
40+
#lightning-liquidity = { version = "0.1.0", features = ["std"] }
41+
42+
lightning = { git = "https://github.com/joostjager/rust-lightning", branch = "merge-scores", features = ["std"] }
43+
lightning-types = { git = "https://github.com/joostjager/rust-lightning", branch = "merge-scores" }
44+
lightning-invoice = { git = "https://github.com/joostjager/rust-lightning", branch = "merge-scores", features = ["std"] }
45+
lightning-net-tokio = { git = "https://github.com/joostjager/rust-lightning", branch = "merge-scores" }
46+
lightning-persister = { git = "https://github.com/joostjager/rust-lightning", branch = "merge-scores" }
47+
lightning-background-processor = { git = "https://github.com/joostjager/rust-lightning", branch = "merge-scores", features = ["futures"] }
48+
lightning-rapid-gossip-sync = { git = "https://github.com/joostjager/rust-lightning", branch = "merge-scores" }
49+
lightning-block-sync = { git = "https://github.com/joostjager/rust-lightning", branch = "merge-scores", features = ["rpc-client", "tokio"] }
50+
lightning-transaction-sync = { git = "https://github.com/joostjager/rust-lightning", branch = "merge-scores", features = ["esplora-async-https", "time"] }
51+
lightning-liquidity = { git = "https://github.com/joostjager/rust-lightning", branch = "merge-scores" }
5252

5353
#lightning = { path = "../rust-lightning/lightning", features = ["std"] }
5454
#lightning-types = { path = "../rust-lightning/lightning-types" }
@@ -89,8 +89,8 @@ prost = { version = "0.11.6", default-features = false}
8989
winapi = { version = "0.3", features = ["winbase"] }
9090

9191
[dev-dependencies]
92-
lightning = { version = "0.1.0", features = ["std", "_test_utils"] }
93-
#lightning = { git = "https://github.com/lightningdevkit/rust-lightning", branch="main", features = ["std", "_test_utils"] }
92+
#lightning = { version = "0.1.0", features = ["std", "_test_utils"] }
93+
lightning = { git = "https://github.com/joostjager/rust-lightning", branch="merge-scores", features = ["std", "_test_utils"] }
9494
#lightning = { path = "../rust-lightning/lightning", features = ["std", "_test_utils"] }
9595
electrum-client = { version = "0.21.0", default-features = true }
9696
bitcoincore-rpc = { version = "0.19.0", default-features = false }

src/builder.rs

+13-8
Original file line numberDiff line numberDiff line change
@@ -41,7 +41,8 @@ use lightning::ln::peer_handler::{IgnoringMessageHandler, MessageHandler};
4141
use lightning::routing::gossip::NodeAlias;
4242
use lightning::routing::router::DefaultRouter;
4343
use lightning::routing::scoring::{
44-
ProbabilisticScorer, ProbabilisticScoringDecayParameters, ProbabilisticScoringFeeParameters,
44+
CombinedScorer, ProbabilisticScorer, ProbabilisticScoringDecayParameters,
45+
ProbabilisticScoringFeeParameters,
4546
};
4647
use lightning::sign::EntropySource;
4748

@@ -322,6 +323,12 @@ impl NodeBuilder {
322323
self
323324
}
324325

326+
/// Configures the [`Node`] instance to source its external scores from the given URL.
327+
pub fn set_external_scores_source(&mut self, url: String) -> &mut Self {
328+
self.config.external_scores_url = Some(url);
329+
self
330+
}
331+
325332
/// Configures the [`Node`] instance to source its inbound liquidity from the given
326333
/// [LSPS2](https://github.com/BitcoinAndLightningLayerSpecs/lsp/blob/main/LSPS2/README.md)
327334
/// service.
@@ -950,26 +957,24 @@ fn build_with_store_internal(
950957
},
951958
};
952959

953-
let scorer = match io::utils::read_scorer(
960+
let local_scorer = match io::utils::read_scorer(
954961
Arc::clone(&kv_store),
955962
Arc::clone(&network_graph),
956963
Arc::clone(&logger),
957964
) {
958-
Ok(scorer) => Arc::new(Mutex::new(scorer)),
965+
Ok(scorer) => scorer,
959966
Err(e) => {
960967
if e.kind() == std::io::ErrorKind::NotFound {
961968
let params = ProbabilisticScoringDecayParameters::default();
962-
Arc::new(Mutex::new(ProbabilisticScorer::new(
963-
params,
964-
Arc::clone(&network_graph),
965-
Arc::clone(&logger),
966-
)))
969+
ProbabilisticScorer::new(params, Arc::clone(&network_graph), Arc::clone(&logger))
967970
} else {
968971
return Err(BuildError::ReadFailed);
969972
}
970973
},
971974
};
972975

976+
let scorer = Arc::new(Mutex::new(CombinedScorer::new(local_scorer)));
977+
973978
let scoring_fee_params = ProbabilisticScoringFeeParameters::default();
974979
let router = Arc::new(DefaultRouter::new(
975980
Arc::clone(&network_graph),

src/config.rs

+10
Original file line numberDiff line numberDiff line change
@@ -57,6 +57,9 @@ pub(crate) const PEER_RECONNECTION_INTERVAL: Duration = Duration::from_secs(10);
5757
// The time in-between RGS sync attempts.
5858
pub(crate) const RGS_SYNC_INTERVAL: Duration = Duration::from_secs(60 * 60);
5959

60+
// The time in-between external scores sync attempts.
61+
pub(crate) const EXTERNAL_SCORES_SYNC_INTERVAL: Duration = Duration::from_secs(60 * 60);
62+
6063
// The time in-between node announcement broadcast attempts.
6164
pub(crate) const NODE_ANN_BCAST_INTERVAL: Duration = Duration::from_secs(60 * 60);
6265

@@ -78,6 +81,9 @@ pub(crate) const TX_BROADCAST_TIMEOUT_SECS: u64 = 5;
7881
// The timeout after which we abort a RGS sync operation.
7982
pub(crate) const RGS_SYNC_TIMEOUT_SECS: u64 = 5;
8083

84+
// The timeout after which we abort a external scores sync operation.
85+
pub(crate) const EXTERNAL_SCORES_SYNC_TIMEOUT_SECS: u64 = 5;
86+
8187
// The length in bytes of our wallets' keys seed.
8288
pub(crate) const WALLET_KEYS_SEED_LEN: usize = 64;
8389

@@ -160,6 +166,9 @@ pub struct Config {
160166
/// **Note:** If unset, default parameters will be used, and you will be able to override the
161167
/// parameters on a per-payment basis in the corresponding method calls.
162168
pub sending_parameters: Option<SendingParameters>,
169+
170+
/// Optional URL to periodically fetch external scores from.
171+
pub external_scores_url: Option<String>,
163172
}
164173

165174
impl Default for Config {
@@ -173,6 +182,7 @@ impl Default for Config {
173182
anchor_channels_config: Some(AnchorChannelsConfig::default()),
174183
sending_parameters: None,
175184
node_alias: None,
185+
external_scores_url: None,
176186
}
177187
}
178188
}

src/lib.rs

+94-1
Original file line numberDiff line numberDiff line change
@@ -99,6 +99,8 @@ mod wallet;
9999
pub use bip39;
100100
pub use bitcoin;
101101
pub use lightning;
102+
use lightning::routing::scoring::{ChannelLiquidities, CombinedScorer};
103+
use lightning::util::ser::Readable;
102104
pub use lightning_invoice;
103105
pub use lightning_types;
104106
pub use vss_client;
@@ -122,7 +124,8 @@ pub use builder::NodeBuilder as Builder;
122124

123125
use chain::ChainSource;
124126
use config::{
125-
default_user_config, may_announce_channel, ChannelConfig, Config, NODE_ANN_BCAST_INTERVAL,
127+
default_user_config, may_announce_channel, ChannelConfig, Config,
128+
EXTERNAL_SCORES_SYNC_INTERVAL, EXTERNAL_SCORES_SYNC_TIMEOUT_SECS, NODE_ANN_BCAST_INTERVAL,
126129
PEER_RECONNECTION_INTERVAL, RGS_SYNC_INTERVAL,
127130
};
128131
use connection::ConnectionManager;
@@ -161,6 +164,7 @@ use bitcoin::secp256k1::PublicKey;
161164
use rand::Rng;
162165

163166
use std::default::Default;
167+
use std::io::Cursor;
164168
use std::net::ToSocketAddrs;
165169
use std::sync::atomic::{AtomicBool, Ordering};
166170
use std::sync::{Arc, Mutex, RwLock};
@@ -304,6 +308,10 @@ impl Node {
304308
});
305309
}
306310

311+
if let Some(url) = &self.config.external_scores_url {
312+
self.configure_external_scores_syncing(&runtime, url);
313+
}
314+
307315
if let Some(listening_addresses) = &self.config.listening_addresses {
308316
// Setup networking
309317
let peer_manager_connection_handler = Arc::clone(&self.peer_manager);
@@ -634,6 +642,41 @@ impl Node {
634642
Ok(())
635643
}
636644

645+
/// Spawn a background task to sync external scores from the given `url`.
646+
fn configure_external_scores_syncing(
647+
&self, runtime: &Arc<tokio::runtime::Runtime>, url: &String,
648+
) {
649+
log_trace!(self.logger, "External scores background syncing from {} enabled", url);
650+
651+
let external_scores_sync_logger = Arc::clone(&self.logger);
652+
let external_scores_scorer = Arc::clone(&self.scorer);
653+
let mut stop_sync = self.stop_sender.subscribe();
654+
let url = url.clone();
655+
656+
runtime.spawn(async move {
657+
let mut interval = tokio::time::interval(EXTERNAL_SCORES_SYNC_INTERVAL);
658+
loop {
659+
tokio::select! {
660+
_ = stop_sync.changed() => {
661+
log_trace!(
662+
external_scores_sync_logger,
663+
"Stopping background syncing external scores.",
664+
);
665+
return;
666+
}
667+
_ = interval.tick() => {
668+
log_trace!(
669+
external_scores_sync_logger,
670+
"Background sync of external scores started.",
671+
);
672+
673+
sync_external_scores(&external_scores_sync_logger, &external_scores_scorer, &url).await;
674+
}
675+
}
676+
}
677+
});
678+
}
679+
637680
/// Disconnects all peers, stops all running background tasks, and shuts down [`Node`].
638681
///
639682
/// After this returns most API methods will return [`Error::NotRunning`].
@@ -1610,3 +1653,53 @@ pub(crate) fn total_anchor_channels_reserve_sats(
16101653
* anchor_channels_config.per_channel_reserve_sats
16111654
})
16121655
}
1656+
1657+
async fn sync_external_scores(
1658+
logger: &Logger, scorer: &Mutex<CombinedScorer<Arc<Graph>, Arc<Logger>>>,
1659+
url: &String,
1660+
) -> () {
1661+
let response = tokio::time::timeout(
1662+
Duration::from_secs(EXTERNAL_SCORES_SYNC_TIMEOUT_SECS),
1663+
reqwest::get(url),
1664+
)
1665+
.await;
1666+
1667+
match response {
1668+
Ok(Ok(response)) => {
1669+
let body = response.bytes().await;
1670+
match body {
1671+
Err(e) => {
1672+
log_trace!(
1673+
logger,
1674+
"Failed to read external scores update from http source: {}",
1675+
e
1676+
);
1677+
return;
1678+
},
1679+
Ok(body) => {
1680+
let mut reader = Cursor::new(body);
1681+
match ChannelLiquidities::read(&mut reader) {
1682+
Ok(liquidities) => {
1683+
let duration_since_epoch =
1684+
SystemTime::now().duration_since(SystemTime::UNIX_EPOCH).unwrap();
1685+
scorer.lock().unwrap().merge(liquidities, duration_since_epoch);
1686+
log_trace!(logger, "External scores merged successfully",);
1687+
},
1688+
Err(e) => {
1689+
log_trace!(logger, "Failed to parse external scores update: {}", e);
1690+
return;
1691+
},
1692+
}
1693+
},
1694+
}
1695+
},
1696+
Err(e) => {
1697+
log_trace!(logger, "Retrieving external scores timed out: {}", e);
1698+
return;
1699+
},
1700+
Ok(Err(e)) => {
1701+
log_trace!(logger, "Failed to retrieve external scores update: {}", e);
1702+
return;
1703+
},
1704+
}
1705+
}

src/types.rs

+3-2
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,8 @@ use lightning::ln::peer_handler::IgnoringMessageHandler;
2121
use lightning::ln::types::ChannelId;
2222
use lightning::routing::gossip;
2323
use lightning::routing::router::DefaultRouter;
24-
use lightning::routing::scoring::{ProbabilisticScorer, ProbabilisticScoringFeeParameters};
24+
use lightning::routing::scoring::CombinedScorer;
25+
use lightning::routing::scoring::ProbabilisticScoringFeeParameters;
2526
use lightning::sign::InMemorySigner;
2627
use lightning::util::persist::KVStore;
2728
use lightning::util::ser::{Readable, Writeable, Writer};
@@ -88,7 +89,7 @@ pub(crate) type Router = DefaultRouter<
8889
ProbabilisticScoringFeeParameters,
8990
Scorer,
9091
>;
91-
pub(crate) type Scorer = ProbabilisticScorer<Arc<Graph>, Arc<Logger>>;
92+
pub(crate) type Scorer = CombinedScorer<Arc<Graph>, Arc<Logger>>;
9293

9394
pub(crate) type Graph = gossip::NetworkGraph<Arc<Logger>>;
9495

0 commit comments

Comments
 (0)