| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469 |
- 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_common::database::mint::DynMintKVStore;
- use cdk_lnd::Lnd as CdkLnd;
- use cdk_sqlite::mint::memory;
- 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,
- };
- let kv_store: DynMintKVStore = Arc::new(memory::empty().await?);
- Ok(CdkCln::new(rpc_path, fee_reserve, kv_store).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,
- };
- let kv_store: DynMintKVStore = Arc::new(memory::empty().await?);
- Ok(CdkLnd::new(
- lnd_client.address.clone(),
- lnd_client.cert_file.clone(),
- lnd_client.macaroon_file.clone(),
- fee_reserve,
- kv_store,
- )
- .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(())
- }
|