init_regtest.rs 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353
  1. use std::env;
  2. use std::path::{Path, PathBuf};
  3. use std::sync::Arc;
  4. use anyhow::Result;
  5. use cdk::types::FeeReserve;
  6. use cdk_cln::Cln as CdkCln;
  7. use cdk_lnd::Lnd as CdkLnd;
  8. use ln_regtest_rs::bitcoin_client::BitcoinClient;
  9. use ln_regtest_rs::bitcoind::Bitcoind;
  10. use ln_regtest_rs::cln::Clnd;
  11. use ln_regtest_rs::ln_client::{ClnClient, LightningClient, LndClient};
  12. use ln_regtest_rs::lnd::Lnd;
  13. use tokio::sync::oneshot::Sender;
  14. use tokio::sync::Notify;
  15. pub const BITCOIND_ADDR: &str = "127.0.0.1:18443";
  16. pub const ZMQ_RAW_BLOCK: &str = "tcp://127.0.0.1:28332";
  17. pub const ZMQ_RAW_TX: &str = "tcp://127.0.0.1:28333";
  18. pub const BITCOIN_RPC_USER: &str = "testuser";
  19. pub const BITCOIN_RPC_PASS: &str = "testpass";
  20. const BITCOIN_DIR: &str = "bitcoin";
  21. pub const LND_ADDR: &str = "0.0.0.0:18449";
  22. pub const LND_RPC_ADDR: &str = "localhost:10009";
  23. pub const LND_TWO_ADDR: &str = "0.0.0.0:18410";
  24. pub const LND_TWO_RPC_ADDR: &str = "localhost:10010";
  25. pub const CLN_ADDR: &str = "127.0.0.1:19846";
  26. pub const CLN_TWO_ADDR: &str = "127.0.0.1:19847";
  27. /// Configuration for regtest environment
  28. pub struct RegtestConfig {
  29. pub mint_addr: String,
  30. pub cln_port: u16,
  31. pub lnd_port: u16,
  32. pub temp_dir: PathBuf,
  33. }
  34. impl Default for RegtestConfig {
  35. fn default() -> Self {
  36. Self {
  37. mint_addr: "127.0.0.1".to_string(),
  38. cln_port: 8085,
  39. lnd_port: 8087,
  40. temp_dir: std::env::temp_dir().join("cdk-itests-default"),
  41. }
  42. }
  43. }
  44. pub fn get_mint_url_with_config(config: &RegtestConfig, which: &str) -> String {
  45. let port = match which {
  46. "0" => config.cln_port,
  47. "1" => config.lnd_port,
  48. _ => panic!("Unknown mint identifier: {which}"),
  49. };
  50. format!("http://{}:{}", config.mint_addr, port)
  51. }
  52. pub fn get_mint_ws_url_with_config(config: &RegtestConfig, which: &str) -> String {
  53. let port = match which {
  54. "0" => config.cln_port,
  55. "1" => config.lnd_port,
  56. _ => panic!("Unknown mint identifier: {which}"),
  57. };
  58. format!("ws://{}:{}/v1/ws", config.mint_addr, port)
  59. }
  60. pub fn get_temp_dir() -> PathBuf {
  61. let dir = env::var("CDK_ITESTS_DIR").expect("Temp dir not set");
  62. std::fs::create_dir_all(&dir).unwrap();
  63. dir.parse().expect("Valid path buf")
  64. }
  65. pub fn get_temp_dir_with_config(config: &RegtestConfig) -> &PathBuf {
  66. &config.temp_dir
  67. }
  68. pub fn get_bitcoin_dir(temp_dir: &Path) -> PathBuf {
  69. let dir = temp_dir.join(BITCOIN_DIR);
  70. std::fs::create_dir_all(&dir).unwrap();
  71. dir
  72. }
  73. pub fn init_bitcoind(work_dir: &Path) -> Bitcoind {
  74. Bitcoind::new(
  75. get_bitcoin_dir(work_dir),
  76. BITCOIND_ADDR.parse().unwrap(),
  77. BITCOIN_RPC_USER.to_string(),
  78. BITCOIN_RPC_PASS.to_string(),
  79. ZMQ_RAW_BLOCK.to_string(),
  80. ZMQ_RAW_TX.to_string(),
  81. )
  82. }
  83. pub fn init_bitcoin_client() -> Result<BitcoinClient> {
  84. BitcoinClient::new(
  85. "wallet".to_string(),
  86. BITCOIND_ADDR.into(),
  87. None,
  88. Some(BITCOIN_RPC_USER.to_string()),
  89. Some(BITCOIN_RPC_PASS.to_string()),
  90. )
  91. }
  92. pub fn get_cln_dir(work_dir: &Path, name: &str) -> PathBuf {
  93. let dir = work_dir.join("cln").join(name);
  94. std::fs::create_dir_all(&dir).unwrap();
  95. dir
  96. }
  97. pub fn get_lnd_dir(work_dir: &Path, name: &str) -> PathBuf {
  98. let dir = work_dir.join("lnd").join(name);
  99. std::fs::create_dir_all(&dir).unwrap();
  100. dir
  101. }
  102. pub fn get_lnd_cert_file_path(lnd_dir: &Path) -> PathBuf {
  103. lnd_dir.join("tls.cert")
  104. }
  105. pub fn get_lnd_macaroon_path(lnd_dir: &Path) -> PathBuf {
  106. lnd_dir.join("data/chain/bitcoin/regtest/admin.macaroon")
  107. }
  108. pub async fn init_lnd(
  109. work_dir: &Path,
  110. lnd_dir: PathBuf,
  111. lnd_addr: &str,
  112. lnd_rpc_addr: &str,
  113. ) -> Lnd {
  114. Lnd::new(
  115. get_bitcoin_dir(work_dir),
  116. lnd_dir,
  117. lnd_addr.parse().unwrap(),
  118. lnd_rpc_addr.to_string(),
  119. BITCOIN_RPC_USER.to_string(),
  120. BITCOIN_RPC_PASS.to_string(),
  121. ZMQ_RAW_BLOCK.to_string(),
  122. ZMQ_RAW_TX.to_string(),
  123. )
  124. }
  125. pub fn generate_block(bitcoin_client: &BitcoinClient) -> Result<()> {
  126. let mine_to_address = bitcoin_client.get_new_address()?;
  127. bitcoin_client.generate_blocks(&mine_to_address, 10)?;
  128. Ok(())
  129. }
  130. pub async fn create_cln_backend(cln_client: &ClnClient) -> Result<CdkCln> {
  131. let rpc_path = cln_client.rpc_path.clone();
  132. let fee_reserve = FeeReserve {
  133. min_fee_reserve: 1.into(),
  134. percent_fee_reserve: 1.0,
  135. };
  136. Ok(CdkCln::new(rpc_path, fee_reserve).await?)
  137. }
  138. pub async fn create_lnd_backend(lnd_client: &LndClient) -> Result<CdkLnd> {
  139. let fee_reserve = FeeReserve {
  140. min_fee_reserve: 1.into(),
  141. percent_fee_reserve: 1.0,
  142. };
  143. Ok(CdkLnd::new(
  144. lnd_client.address.clone(),
  145. lnd_client.cert_file.clone(),
  146. lnd_client.macaroon_file.clone(),
  147. fee_reserve,
  148. )
  149. .await?)
  150. }
  151. pub async fn fund_ln<C>(bitcoin_client: &BitcoinClient, ln_client: &C) -> Result<()>
  152. where
  153. C: LightningClient,
  154. {
  155. let ln_address = ln_client.get_new_onchain_address().await?;
  156. bitcoin_client.send_to_address(&ln_address, 5_000_000)?;
  157. ln_client.wait_chain_sync().await?;
  158. let mine_to_address = bitcoin_client.get_new_address()?;
  159. bitcoin_client.generate_blocks(&mine_to_address, 10)?;
  160. ln_client.wait_chain_sync().await?;
  161. Ok(())
  162. }
  163. pub async fn open_channel<C1, C2>(cln_client: &C1, lnd_client: &C2) -> Result<()>
  164. where
  165. C1: LightningClient,
  166. C2: LightningClient,
  167. {
  168. let cln_info = cln_client.get_connect_info().await?;
  169. let cln_pubkey = cln_info.pubkey;
  170. let cln_address = cln_info.address;
  171. let cln_port = cln_info.port;
  172. lnd_client
  173. .connect_peer(cln_pubkey.to_string(), cln_address.to_string(), cln_port)
  174. .await
  175. .unwrap();
  176. cln_client.wait_chain_sync().await?;
  177. lnd_client.wait_chain_sync().await?;
  178. lnd_client
  179. .open_channel(1_500_000, &cln_pubkey.to_string(), Some(750_000))
  180. .await
  181. .unwrap();
  182. Ok(())
  183. }
  184. pub async fn start_regtest_end(
  185. work_dir: &Path,
  186. sender: Sender<()>,
  187. notify: Arc<Notify>,
  188. ) -> anyhow::Result<()> {
  189. let mut bitcoind = init_bitcoind(work_dir);
  190. bitcoind.start_bitcoind()?;
  191. let bitcoin_client = init_bitcoin_client()?;
  192. bitcoin_client.create_wallet().ok();
  193. bitcoin_client.load_wallet()?;
  194. let new_add = bitcoin_client.get_new_address()?;
  195. bitcoin_client.generate_blocks(&new_add, 200).unwrap();
  196. let cln_one_dir = get_cln_dir(work_dir, "one");
  197. let mut clnd = Clnd::new(
  198. get_bitcoin_dir(work_dir),
  199. cln_one_dir.clone(),
  200. CLN_ADDR.into(),
  201. BITCOIN_RPC_USER.to_string(),
  202. BITCOIN_RPC_PASS.to_string(),
  203. );
  204. clnd.start_clnd()?;
  205. let cln_client = ClnClient::new(cln_one_dir.clone(), None).await?;
  206. cln_client.wait_chain_sync().await.unwrap();
  207. fund_ln(&bitcoin_client, &cln_client).await.unwrap();
  208. // Create second cln
  209. let cln_two_dir = get_cln_dir(work_dir, "two");
  210. let mut clnd_two = Clnd::new(
  211. get_bitcoin_dir(work_dir),
  212. cln_two_dir.clone(),
  213. CLN_TWO_ADDR.into(),
  214. BITCOIN_RPC_USER.to_string(),
  215. BITCOIN_RPC_PASS.to_string(),
  216. );
  217. clnd_two.start_clnd()?;
  218. let cln_two_client = ClnClient::new(cln_two_dir.clone(), None).await?;
  219. cln_two_client.wait_chain_sync().await.unwrap();
  220. fund_ln(&bitcoin_client, &cln_two_client).await.unwrap();
  221. let lnd_dir = get_lnd_dir(work_dir, "one");
  222. println!("{}", lnd_dir.display());
  223. let mut lnd = init_lnd(work_dir, lnd_dir.clone(), LND_ADDR, LND_RPC_ADDR).await;
  224. lnd.start_lnd().unwrap();
  225. tracing::info!("Started lnd node");
  226. let lnd_client = LndClient::new(
  227. format!("https://{LND_RPC_ADDR}"),
  228. get_lnd_cert_file_path(&lnd_dir),
  229. get_lnd_macaroon_path(&lnd_dir),
  230. )
  231. .await?;
  232. lnd_client.wait_chain_sync().await.unwrap();
  233. fund_ln(&bitcoin_client, &lnd_client).await.unwrap();
  234. // create second lnd node
  235. let work_dir = get_temp_dir();
  236. let lnd_two_dir = get_lnd_dir(&work_dir, "two");
  237. let mut lnd_two = init_lnd(
  238. &work_dir,
  239. lnd_two_dir.clone(),
  240. LND_TWO_ADDR,
  241. LND_TWO_RPC_ADDR,
  242. )
  243. .await;
  244. lnd_two.start_lnd().unwrap();
  245. tracing::info!("Started second lnd node");
  246. let lnd_two_client = LndClient::new(
  247. format!("https://{LND_TWO_RPC_ADDR}"),
  248. get_lnd_cert_file_path(&lnd_two_dir),
  249. get_lnd_macaroon_path(&lnd_two_dir),
  250. )
  251. .await?;
  252. lnd_two_client.wait_chain_sync().await.unwrap();
  253. fund_ln(&bitcoin_client, &lnd_two_client).await.unwrap();
  254. // Open channels concurrently
  255. // Open channels
  256. {
  257. open_channel(&cln_client, &lnd_client).await.unwrap();
  258. tracing::info!("Opened channel between cln and lnd one");
  259. generate_block(&bitcoin_client)?;
  260. // open_channel(&bitcoin_client, &cln_client, &cln_two_client)
  261. // .await
  262. // .unwrap();
  263. // tracing::info!("Opened channel between cln and cln two");
  264. open_channel(&lnd_client, &lnd_two_client).await.unwrap();
  265. tracing::info!("Opened channel between lnd and lnd two");
  266. generate_block(&bitcoin_client)?;
  267. // open_channel(&cln_client, &lnd_two_client).await.unwrap();
  268. // tracing::info!("Opened channel between cln and lnd two");
  269. open_channel(&cln_two_client, &lnd_client).await.unwrap();
  270. tracing::info!("Opened channel between cln two and lnd");
  271. generate_block(&bitcoin_client)?;
  272. open_channel(&cln_client, &lnd_two_client).await.unwrap();
  273. tracing::info!("Opened channel between cln and lnd two");
  274. generate_block(&bitcoin_client)?;
  275. cln_client.wait_channels_active().await?;
  276. cln_two_client.wait_channels_active().await?;
  277. lnd_client.wait_channels_active().await?;
  278. lnd_two_client.wait_channels_active().await?;
  279. }
  280. // Send notification that regtest set up is complete
  281. sender.send(()).expect("Could not send oneshot");
  282. // Wait until we are told to shutdown
  283. // If we return the bitcoind, lnd, and cln will be dropped and shutdown
  284. notify.notified().await;
  285. Ok(())
  286. }