Skip to content

Commit 889d34c

Browse files
committed
Add LSPS1 API
We add an `Lsps1Liquidity` API object, mirroring the approach we took with the `payment` APIs.
1 parent 5b2aaec commit 889d34c

File tree

2 files changed

+147
-5
lines changed

2 files changed

+147
-5
lines changed

src/lib.rs

+31-2
Original file line numberDiff line numberDiff line change
@@ -84,7 +84,7 @@ mod gossip;
8484
pub mod graph;
8585
mod hex_utils;
8686
pub mod io;
87-
mod liquidity;
87+
pub mod liquidity;
8888
pub mod logger;
8989
mod message_handler;
9090
pub mod payment;
@@ -100,6 +100,7 @@ pub use bip39;
100100
pub use bitcoin;
101101
pub use lightning;
102102
pub use lightning_invoice;
103+
pub use lightning_liquidity;
103104
pub use lightning_types;
104105
pub use vss_client;
105106

@@ -130,7 +131,7 @@ use event::{EventHandler, EventQueue};
130131
use gossip::GossipSource;
131132
use graph::NetworkGraph;
132133
use io::utils::write_node_metrics;
133-
use liquidity::LiquiditySource;
134+
use liquidity::{LSPS1Liquidity, LiquiditySource};
134135
use payment::store::PaymentStore;
135136
use payment::{
136137
Bolt11Payment, Bolt12Payment, OnchainPayment, PaymentDetails, SpontaneousPayment,
@@ -957,6 +958,34 @@ impl Node {
957958
))
958959
}
959960

961+
/// Returns a liquidity handler allowing to request channels via the [bLIP-51 / LSPS1] protocol.
962+
///
963+
/// [bLIP-51 / LSPS1]: https://github.com/lightning/blips/blob/master/blip-0051.md
964+
#[cfg(not(feature = "uniffi"))]
965+
pub fn lsps1_liquidity(&self) -> LSPS1Liquidity {
966+
LSPS1Liquidity::new(
967+
Arc::clone(&self.runtime),
968+
Arc::clone(&self.wallet),
969+
Arc::clone(&self.connection_manager),
970+
self.liquidity_source.clone(),
971+
Arc::clone(&self.logger),
972+
)
973+
}
974+
975+
/// Returns a liquidity handler allowing to request channels via the [bLIP-51 / LSPS1] protocol.
976+
///
977+
/// [bLIP-51 / LSPS1]: https://github.com/lightning/blips/blob/master/blip-0051.md
978+
#[cfg(feature = "uniffi")]
979+
pub fn lsps1_liquidity(&self) -> Arc<LSPS1Liquidity> {
980+
Arc::new(LSPS1Liquidity::new(
981+
Arc::clone(&self.runtime),
982+
Arc::clone(&self.wallet),
983+
Arc::clone(&self.connection_manager),
984+
self.liquidity_source.clone(),
985+
Arc::clone(&self.logger),
986+
))
987+
}
988+
960989
/// Retrieve a list of known channels.
961990
pub fn list_channels(&self) -> Vec<ChannelDetails> {
962991
self.channel_manager.list_channels().into_iter().map(|c| c.into()).collect()

src/liquidity.rs

+116-3
Original file line numberDiff line numberDiff line change
@@ -5,9 +5,12 @@
55
// http://opensource.org/licenses/MIT>, at your option. You may not use this file except in
66
// accordance with one or both of these licenses.
77

8+
//! Objects related to liquidity management.
9+
810
use crate::chain::ChainSource;
9-
use crate::logger::{log_debug, log_error, log_info, LdkLogger};
10-
use crate::types::{ChannelManager, KeysManager, LiquidityManager, PeerManager};
11+
use crate::connection::ConnectionManager;
12+
use crate::logger::{log_debug, log_error, log_info, LdkLogger, Logger};
13+
use crate::types::{ChannelManager, KeysManager, LiquidityManager, PeerManager, Wallet};
1114
use crate::{Config, Error};
1215

1316
use lightning::ln::channelmanager::MIN_FINAL_CLTV_EXPIRY_DELTA;
@@ -34,7 +37,7 @@ use tokio::sync::oneshot;
3437

3538
use std::collections::HashMap;
3639
use std::ops::Deref;
37-
use std::sync::{Arc, Mutex};
40+
use std::sync::{Arc, Mutex, RwLock};
3841
use std::time::Duration;
3942

4043
const LIQUIDITY_REQUEST_TIMEOUT_SECS: u64 = 5;
@@ -896,3 +899,113 @@ pub(crate) struct LSPS2BuyResponse {
896899
intercept_scid: u64,
897900
cltv_expiry_delta: u32,
898901
}
902+
903+
/// A liquidity handler allowing to request channels via the [bLIP-51 / LSPS1] protocol.
904+
///
905+
/// Should be retrieved by calling [`Node::lsps1_liquidity`].
906+
///
907+
/// [bLIP-51 / LSPS1]: https://github.com/lightning/blips/blob/master/blip-0051.md
908+
/// [`Node::lsps1_liquidity`]: crate::Node::lsps1_liquidity
909+
#[derive(Clone)]
910+
pub struct LSPS1Liquidity {
911+
runtime: Arc<RwLock<Option<Arc<tokio::runtime::Runtime>>>>,
912+
wallet: Arc<Wallet>,
913+
connection_manager: Arc<ConnectionManager<Arc<Logger>>>,
914+
liquidity_source: Option<Arc<LiquiditySource<Arc<Logger>>>>,
915+
logger: Arc<Logger>,
916+
}
917+
918+
impl LSPS1Liquidity {
919+
pub(crate) fn new(
920+
runtime: Arc<RwLock<Option<Arc<tokio::runtime::Runtime>>>>, wallet: Arc<Wallet>,
921+
connection_manager: Arc<ConnectionManager<Arc<Logger>>>,
922+
liquidity_source: Option<Arc<LiquiditySource<Arc<Logger>>>>, logger: Arc<Logger>,
923+
) -> Self {
924+
Self { runtime, wallet, connection_manager, liquidity_source, logger }
925+
}
926+
927+
/// Connects to the configured LSP and places an order for an inbound channel.
928+
///
929+
/// The channel will be opened after one of the returned payment options has successfully been
930+
/// paid.
931+
pub fn request_channel(
932+
&self, lsp_balance_sat: u64, client_balance_sat: u64, channel_expiry_blocks: u32,
933+
announce_channel: bool,
934+
) -> Result<LSPS1OrderStatus, Error> {
935+
let liquidity_source =
936+
self.liquidity_source.as_ref().ok_or(Error::LiquiditySourceUnavailable)?;
937+
938+
let (lsp_node_id, lsp_address) = liquidity_source
939+
.get_lsps1_service_details()
940+
.ok_or(Error::LiquiditySourceUnavailable)?;
941+
942+
let rt_lock = self.runtime.read().unwrap();
943+
let runtime = rt_lock.as_ref().unwrap();
944+
945+
let con_node_id = lsp_node_id;
946+
let con_addr = lsp_address.clone();
947+
let con_cm = Arc::clone(&self.connection_manager);
948+
949+
// We need to use our main runtime here as a local runtime might not be around to poll
950+
// connection futures going forward.
951+
tokio::task::block_in_place(move || {
952+
runtime.block_on(async move {
953+
con_cm.connect_peer_if_necessary(con_node_id, con_addr).await
954+
})
955+
})?;
956+
957+
log_info!(self.logger, "Connected to LSP {}@{}. ", lsp_node_id, lsp_address);
958+
959+
let refund_address = self.wallet.get_new_address()?;
960+
961+
let liquidity_source = Arc::clone(&liquidity_source);
962+
let response = tokio::task::block_in_place(move || {
963+
runtime.block_on(async move {
964+
liquidity_source
965+
.lsps1_request_channel(
966+
lsp_balance_sat,
967+
client_balance_sat,
968+
channel_expiry_blocks,
969+
announce_channel,
970+
refund_address,
971+
)
972+
.await
973+
})
974+
})?;
975+
976+
Ok(response)
977+
}
978+
979+
/// Connects to the configured LSP and checks for the status of a previously-placed order.
980+
pub fn check_order_status(&self, order_id: OrderId) -> Result<LSPS1OrderStatus, Error> {
981+
let liquidity_source =
982+
self.liquidity_source.as_ref().ok_or(Error::LiquiditySourceUnavailable)?;
983+
984+
let (lsp_node_id, lsp_address) = liquidity_source
985+
.get_lsps1_service_details()
986+
.ok_or(Error::LiquiditySourceUnavailable)?;
987+
988+
let rt_lock = self.runtime.read().unwrap();
989+
let runtime = rt_lock.as_ref().unwrap();
990+
991+
let con_node_id = lsp_node_id;
992+
let con_addr = lsp_address.clone();
993+
let con_cm = Arc::clone(&self.connection_manager);
994+
995+
// We need to use our main runtime here as a local runtime might not be around to poll
996+
// connection futures going forward.
997+
tokio::task::block_in_place(move || {
998+
runtime.block_on(async move {
999+
con_cm.connect_peer_if_necessary(con_node_id, con_addr).await
1000+
})
1001+
})?;
1002+
1003+
let liquidity_source = Arc::clone(&liquidity_source);
1004+
let response = tokio::task::block_in_place(move || {
1005+
runtime
1006+
.block_on(async move { liquidity_source.lsps1_check_order_status(order_id).await })
1007+
})?;
1008+
1009+
Ok(response)
1010+
}
1011+
}

0 commit comments

Comments
 (0)