mirror of
https://github.com/aljazceru/cdk.git
synced 2026-01-31 10:45:49 +01:00
* feat: add LDK Node Lightning backend with comprehensive integration - Add new cdk-ldk-node crate implementing Lightning backend using LDK Node - Extend MintPayment trait with start/stop methods for processor lifecycle management - Add LDK Node configuration support to cdk-mintd with chain source and gossip options - Enhance mint startup/shutdown to properly manage payment processor lifecycle --------- Co-authored-by: Erik <78821053+swedishfrenchpress@users.noreply.github.com>
464 lines
13 KiB
Rust
464 lines
13 KiB
Rust
use std::env;
|
|
use std::net::Ipv4Addr;
|
|
use std::path::{Path, PathBuf};
|
|
use std::sync::Arc;
|
|
|
|
use anyhow::Result;
|
|
use cdk::types::FeeReserve;
|
|
use cdk_cln::Cln as CdkCln;
|
|
use cdk_lnd::Lnd as CdkLnd;
|
|
use ldk_node::lightning::ln::msgs::SocketAddress;
|
|
use ldk_node::Node;
|
|
use ln_regtest_rs::bitcoin_client::BitcoinClient;
|
|
use ln_regtest_rs::bitcoind::Bitcoind;
|
|
use ln_regtest_rs::cln::Clnd;
|
|
use ln_regtest_rs::ln_client::{ClnClient, LightningClient, LndClient};
|
|
use ln_regtest_rs::lnd::Lnd;
|
|
use tokio::sync::oneshot::Sender;
|
|
use tokio::sync::Notify;
|
|
|
|
pub const BITCOIND_ADDR: &str = "127.0.0.1:18443";
|
|
pub const ZMQ_RAW_BLOCK: &str = "tcp://127.0.0.1:28332";
|
|
pub const ZMQ_RAW_TX: &str = "tcp://127.0.0.1:28333";
|
|
pub const BITCOIN_RPC_USER: &str = "testuser";
|
|
pub const BITCOIN_RPC_PASS: &str = "testpass";
|
|
|
|
const BITCOIN_DIR: &str = "bitcoin";
|
|
|
|
pub const LND_ADDR: &str = "0.0.0.0:18449";
|
|
pub const LND_RPC_ADDR: &str = "localhost:10009";
|
|
|
|
pub const LND_TWO_ADDR: &str = "0.0.0.0:18410";
|
|
pub const LND_TWO_RPC_ADDR: &str = "localhost:10010";
|
|
|
|
pub const CLN_ADDR: &str = "127.0.0.1:19846";
|
|
pub const CLN_TWO_ADDR: &str = "127.0.0.1:19847";
|
|
|
|
/// Configuration for regtest environment
|
|
pub struct RegtestConfig {
|
|
pub mint_addr: String,
|
|
pub cln_port: u16,
|
|
pub lnd_port: u16,
|
|
pub temp_dir: PathBuf,
|
|
}
|
|
|
|
impl Default for RegtestConfig {
|
|
fn default() -> Self {
|
|
Self {
|
|
mint_addr: "127.0.0.1".to_string(),
|
|
cln_port: 8085,
|
|
lnd_port: 8087,
|
|
temp_dir: std::env::temp_dir().join("cdk-itests-default"),
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn get_mint_url_with_config(config: &RegtestConfig, which: &str) -> String {
|
|
let port = match which {
|
|
"0" => config.cln_port,
|
|
"1" => config.lnd_port,
|
|
_ => panic!("Unknown mint identifier: {which}"),
|
|
};
|
|
format!("http://{}:{}", config.mint_addr, port)
|
|
}
|
|
|
|
pub fn get_mint_ws_url_with_config(config: &RegtestConfig, which: &str) -> String {
|
|
let port = match which {
|
|
"0" => config.cln_port,
|
|
"1" => config.lnd_port,
|
|
_ => panic!("Unknown mint identifier: {which}"),
|
|
};
|
|
format!("ws://{}:{}/v1/ws", config.mint_addr, port)
|
|
}
|
|
|
|
pub fn get_temp_dir() -> PathBuf {
|
|
let dir = env::var("CDK_ITESTS_DIR").expect("Temp dir not set");
|
|
std::fs::create_dir_all(&dir).unwrap();
|
|
dir.parse().expect("Valid path buf")
|
|
}
|
|
|
|
pub fn get_temp_dir_with_config(config: &RegtestConfig) -> &PathBuf {
|
|
&config.temp_dir
|
|
}
|
|
|
|
pub fn get_bitcoin_dir(temp_dir: &Path) -> PathBuf {
|
|
let dir = temp_dir.join(BITCOIN_DIR);
|
|
std::fs::create_dir_all(&dir).unwrap();
|
|
dir
|
|
}
|
|
|
|
pub fn init_bitcoind(work_dir: &Path) -> Bitcoind {
|
|
Bitcoind::new(
|
|
get_bitcoin_dir(work_dir),
|
|
BITCOIND_ADDR.parse().unwrap(),
|
|
BITCOIN_RPC_USER.to_string(),
|
|
BITCOIN_RPC_PASS.to_string(),
|
|
ZMQ_RAW_BLOCK.to_string(),
|
|
ZMQ_RAW_TX.to_string(),
|
|
)
|
|
}
|
|
|
|
pub fn init_bitcoin_client() -> Result<BitcoinClient> {
|
|
BitcoinClient::new(
|
|
"wallet".to_string(),
|
|
BITCOIND_ADDR.into(),
|
|
None,
|
|
Some(BITCOIN_RPC_USER.to_string()),
|
|
Some(BITCOIN_RPC_PASS.to_string()),
|
|
)
|
|
}
|
|
|
|
pub fn get_cln_dir(work_dir: &Path, name: &str) -> PathBuf {
|
|
let dir = work_dir.join("cln").join(name);
|
|
std::fs::create_dir_all(&dir).unwrap();
|
|
dir
|
|
}
|
|
|
|
pub fn get_lnd_dir(work_dir: &Path, name: &str) -> PathBuf {
|
|
let dir = work_dir.join("lnd").join(name);
|
|
std::fs::create_dir_all(&dir).unwrap();
|
|
dir
|
|
}
|
|
|
|
pub fn get_lnd_cert_file_path(lnd_dir: &Path) -> PathBuf {
|
|
lnd_dir.join("tls.cert")
|
|
}
|
|
|
|
pub fn get_lnd_macaroon_path(lnd_dir: &Path) -> PathBuf {
|
|
lnd_dir.join("data/chain/bitcoin/regtest/admin.macaroon")
|
|
}
|
|
|
|
pub async fn init_lnd(
|
|
work_dir: &Path,
|
|
lnd_dir: PathBuf,
|
|
lnd_addr: &str,
|
|
lnd_rpc_addr: &str,
|
|
) -> Lnd {
|
|
Lnd::new(
|
|
get_bitcoin_dir(work_dir),
|
|
lnd_dir,
|
|
lnd_addr.parse().unwrap(),
|
|
lnd_rpc_addr.to_string(),
|
|
BITCOIN_RPC_USER.to_string(),
|
|
BITCOIN_RPC_PASS.to_string(),
|
|
ZMQ_RAW_BLOCK.to_string(),
|
|
ZMQ_RAW_TX.to_string(),
|
|
)
|
|
}
|
|
|
|
pub fn generate_block(bitcoin_client: &BitcoinClient) -> Result<()> {
|
|
let mine_to_address = bitcoin_client.get_new_address()?;
|
|
let blocks = 10;
|
|
tracing::info!("Mining {blocks} blocks to {mine_to_address}");
|
|
|
|
bitcoin_client.generate_blocks(&mine_to_address, 10)?;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
pub async fn create_cln_backend(cln_client: &ClnClient) -> Result<CdkCln> {
|
|
let rpc_path = cln_client.rpc_path.clone();
|
|
|
|
let fee_reserve = FeeReserve {
|
|
min_fee_reserve: 1.into(),
|
|
percent_fee_reserve: 1.0,
|
|
};
|
|
|
|
Ok(CdkCln::new(rpc_path, fee_reserve).await?)
|
|
}
|
|
|
|
pub async fn create_lnd_backend(lnd_client: &LndClient) -> Result<CdkLnd> {
|
|
let fee_reserve = FeeReserve {
|
|
min_fee_reserve: 1.into(),
|
|
percent_fee_reserve: 1.0,
|
|
};
|
|
|
|
Ok(CdkLnd::new(
|
|
lnd_client.address.clone(),
|
|
lnd_client.cert_file.clone(),
|
|
lnd_client.macaroon_file.clone(),
|
|
fee_reserve,
|
|
)
|
|
.await?)
|
|
}
|
|
|
|
pub async fn fund_ln<C>(bitcoin_client: &BitcoinClient, ln_client: &C) -> Result<()>
|
|
where
|
|
C: LightningClient,
|
|
{
|
|
let ln_address = ln_client.get_new_onchain_address().await?;
|
|
|
|
bitcoin_client.send_to_address(&ln_address, 5_000_000)?;
|
|
|
|
ln_client.wait_chain_sync().await?;
|
|
|
|
let mine_to_address = bitcoin_client.get_new_address()?;
|
|
bitcoin_client.generate_blocks(&mine_to_address, 10)?;
|
|
|
|
ln_client.wait_chain_sync().await?;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
pub async fn open_channel<C1, C2>(cln_client: &C1, lnd_client: &C2) -> Result<()>
|
|
where
|
|
C1: LightningClient,
|
|
C2: LightningClient,
|
|
{
|
|
let cln_info = cln_client.get_connect_info().await?;
|
|
|
|
let cln_pubkey = cln_info.pubkey;
|
|
let cln_address = cln_info.address;
|
|
let cln_port = cln_info.port;
|
|
|
|
lnd_client
|
|
.connect_peer(cln_pubkey.to_string(), cln_address.to_string(), cln_port)
|
|
.await
|
|
.unwrap();
|
|
|
|
cln_client.wait_chain_sync().await?;
|
|
lnd_client.wait_chain_sync().await?;
|
|
|
|
lnd_client
|
|
.open_channel(1_500_000, &cln_pubkey.to_string(), Some(750_000))
|
|
.await
|
|
.unwrap();
|
|
|
|
Ok(())
|
|
}
|
|
|
|
pub async fn start_regtest_end(
|
|
work_dir: &Path,
|
|
sender: Sender<()>,
|
|
notify: Arc<Notify>,
|
|
ldk_node: Option<Arc<Node>>,
|
|
) -> anyhow::Result<()> {
|
|
let mut bitcoind = init_bitcoind(work_dir);
|
|
bitcoind.start_bitcoind()?;
|
|
|
|
let bitcoin_client = init_bitcoin_client()?;
|
|
bitcoin_client.create_wallet().ok();
|
|
bitcoin_client.load_wallet()?;
|
|
|
|
let new_add = bitcoin_client.get_new_address()?;
|
|
bitcoin_client.generate_blocks(&new_add, 200).unwrap();
|
|
|
|
let cln_one_dir = get_cln_dir(work_dir, "one");
|
|
let mut clnd = Clnd::new(
|
|
get_bitcoin_dir(work_dir),
|
|
cln_one_dir.clone(),
|
|
CLN_ADDR.into(),
|
|
BITCOIN_RPC_USER.to_string(),
|
|
BITCOIN_RPC_PASS.to_string(),
|
|
);
|
|
clnd.start_clnd()?;
|
|
|
|
let cln_client = ClnClient::new(cln_one_dir.clone(), None).await?;
|
|
|
|
cln_client.wait_chain_sync().await.unwrap();
|
|
|
|
fund_ln(&bitcoin_client, &cln_client).await.unwrap();
|
|
|
|
// Create second cln
|
|
let cln_two_dir = get_cln_dir(work_dir, "two");
|
|
let mut clnd_two = Clnd::new(
|
|
get_bitcoin_dir(work_dir),
|
|
cln_two_dir.clone(),
|
|
CLN_TWO_ADDR.into(),
|
|
BITCOIN_RPC_USER.to_string(),
|
|
BITCOIN_RPC_PASS.to_string(),
|
|
);
|
|
clnd_two.start_clnd()?;
|
|
|
|
let cln_two_client = ClnClient::new(cln_two_dir.clone(), None).await?;
|
|
|
|
cln_two_client.wait_chain_sync().await.unwrap();
|
|
|
|
fund_ln(&bitcoin_client, &cln_two_client).await.unwrap();
|
|
|
|
let lnd_dir = get_lnd_dir(work_dir, "one");
|
|
println!("{}", lnd_dir.display());
|
|
|
|
let mut lnd = init_lnd(work_dir, lnd_dir.clone(), LND_ADDR, LND_RPC_ADDR).await;
|
|
lnd.start_lnd().unwrap();
|
|
tracing::info!("Started lnd node");
|
|
|
|
let lnd_client = LndClient::new(
|
|
format!("https://{LND_RPC_ADDR}"),
|
|
get_lnd_cert_file_path(&lnd_dir),
|
|
get_lnd_macaroon_path(&lnd_dir),
|
|
)
|
|
.await?;
|
|
|
|
lnd_client.wait_chain_sync().await.unwrap();
|
|
|
|
if let Some(node) = ldk_node.as_ref() {
|
|
tracing::info!("Starting ldk node");
|
|
node.start()?;
|
|
let addr = node.onchain_payment().new_address().unwrap();
|
|
bitcoin_client.send_to_address(&addr.to_string(), 5_000_000)?;
|
|
}
|
|
|
|
fund_ln(&bitcoin_client, &lnd_client).await.unwrap();
|
|
|
|
// create second lnd node
|
|
let work_dir = get_temp_dir();
|
|
let lnd_two_dir = get_lnd_dir(&work_dir, "two");
|
|
let mut lnd_two = init_lnd(
|
|
&work_dir,
|
|
lnd_two_dir.clone(),
|
|
LND_TWO_ADDR,
|
|
LND_TWO_RPC_ADDR,
|
|
)
|
|
.await;
|
|
lnd_two.start_lnd().unwrap();
|
|
tracing::info!("Started second lnd node");
|
|
|
|
let lnd_two_client = LndClient::new(
|
|
format!("https://{LND_TWO_RPC_ADDR}"),
|
|
get_lnd_cert_file_path(&lnd_two_dir),
|
|
get_lnd_macaroon_path(&lnd_two_dir),
|
|
)
|
|
.await?;
|
|
|
|
lnd_two_client.wait_chain_sync().await.unwrap();
|
|
|
|
fund_ln(&bitcoin_client, &lnd_two_client).await.unwrap();
|
|
|
|
// Open channels concurrently
|
|
// Open channels
|
|
{
|
|
open_channel(&cln_client, &lnd_client).await.unwrap();
|
|
tracing::info!("Opened channel between cln and lnd one");
|
|
generate_block(&bitcoin_client)?;
|
|
// open_channel(&bitcoin_client, &cln_client, &cln_two_client)
|
|
// .await
|
|
// .unwrap();
|
|
// tracing::info!("Opened channel between cln and cln two");
|
|
|
|
open_channel(&lnd_client, &lnd_two_client).await.unwrap();
|
|
tracing::info!("Opened channel between lnd and lnd two");
|
|
generate_block(&bitcoin_client)?;
|
|
|
|
// open_channel(&cln_client, &lnd_two_client).await.unwrap();
|
|
// tracing::info!("Opened channel between cln and lnd two");
|
|
open_channel(&cln_two_client, &lnd_client).await.unwrap();
|
|
tracing::info!("Opened channel between cln two and lnd");
|
|
generate_block(&bitcoin_client)?;
|
|
|
|
open_channel(&cln_client, &lnd_two_client).await.unwrap();
|
|
tracing::info!("Opened channel between cln and lnd two");
|
|
generate_block(&bitcoin_client)?;
|
|
|
|
if let Some(node) = ldk_node {
|
|
let pubkey = node.node_id();
|
|
let listen_addr = node.listening_addresses();
|
|
let listen_addr = listen_addr.as_ref().unwrap().first().unwrap();
|
|
|
|
let (listen_addr, port) = match listen_addr {
|
|
SocketAddress::TcpIpV4 { addr, port } => (Ipv4Addr::from(*addr).to_string(), port),
|
|
_ => panic!(),
|
|
};
|
|
|
|
tracing::info!("Opening channel from cln to ldk");
|
|
|
|
cln_client
|
|
.connect_peer(pubkey.to_string(), listen_addr.clone(), *port)
|
|
.await?;
|
|
|
|
cln_client
|
|
.open_channel(1_500_000, &pubkey.to_string(), Some(750_000))
|
|
.await
|
|
.unwrap();
|
|
|
|
generate_block(&bitcoin_client)?;
|
|
|
|
let cln_two_info = cln_two_client.get_connect_info().await?;
|
|
|
|
cln_client
|
|
.connect_peer(cln_two_info.pubkey, listen_addr.clone(), cln_two_info.port)
|
|
.await?;
|
|
|
|
tracing::info!("Opening channel from lnd to ldk");
|
|
|
|
let cln_info = cln_client.get_connect_info().await?;
|
|
|
|
node.connect(
|
|
cln_info.pubkey.parse()?,
|
|
SocketAddress::TcpIpV4 {
|
|
addr: cln_info
|
|
.address
|
|
.split('.')
|
|
.map(|part| part.parse())
|
|
.collect::<Result<Vec<u8>, _>>()?
|
|
.try_into()
|
|
.unwrap(),
|
|
port: cln_info.port,
|
|
},
|
|
true,
|
|
)?;
|
|
|
|
let lnd_info = lnd_client.get_connect_info().await?;
|
|
|
|
node.connect(
|
|
lnd_info.pubkey.parse()?,
|
|
SocketAddress::TcpIpV4 {
|
|
addr: [127, 0, 0, 1],
|
|
port: lnd_info.port,
|
|
},
|
|
true,
|
|
)?;
|
|
|
|
// lnd_client
|
|
// .open_channel(1_500_000, &pubkey.to_string(), Some(750_000))
|
|
// .await
|
|
// .unwrap();
|
|
|
|
generate_block(&bitcoin_client)?;
|
|
lnd_client.wait_chain_sync().await?;
|
|
|
|
node.open_announced_channel(
|
|
lnd_info.pubkey.parse()?,
|
|
SocketAddress::TcpIpV4 {
|
|
addr: [127, 0, 0, 1],
|
|
port: lnd_info.port,
|
|
},
|
|
1_000_000,
|
|
Some(500_000_000),
|
|
None,
|
|
)?;
|
|
|
|
generate_block(&bitcoin_client)?;
|
|
|
|
tracing::info!("Ldk channels opened");
|
|
|
|
node.sync_wallets()?;
|
|
|
|
tracing::info!("Ldk wallet synced");
|
|
|
|
cln_client.wait_channels_active().await?;
|
|
|
|
lnd_client.wait_channels_active().await?;
|
|
|
|
node.stop()?;
|
|
} else {
|
|
cln_client.wait_channels_active().await?;
|
|
|
|
lnd_client.wait_channels_active().await?;
|
|
|
|
generate_block(&bitcoin_client)?;
|
|
}
|
|
}
|
|
|
|
tracing::info!("Regtest channels active");
|
|
|
|
// Send notification that regtest set up is complete
|
|
sender.send(()).expect("Could not send oneshot");
|
|
|
|
// Wait until we are told to shutdown
|
|
// If we return the bitcoind, lnd, and cln will be dropped and shutdown
|
|
notify.notified().await;
|
|
|
|
Ok(())
|
|
}
|