regtest.rs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383
  1. use std::{fmt::Debug, str::FromStr, sync::Arc, time::Duration};
  2. use anyhow::{bail, Result};
  3. use bip39::Mnemonic;
  4. use cdk::{
  5. amount::{Amount, SplitTarget},
  6. cdk_database::WalletMemoryDatabase,
  7. nuts::{
  8. CurrencyUnit, MeltQuoteState, MintBolt11Request, MintQuoteState, NotificationPayload,
  9. PreMintSecrets, State,
  10. },
  11. wallet::{
  12. client::{HttpClient, HttpClientMethods},
  13. Wallet,
  14. },
  15. };
  16. use cdk_integration_tests::init_regtest::{
  17. get_mint_url, get_mint_ws_url, init_cln_client, init_lnd_client,
  18. };
  19. use futures::{SinkExt, StreamExt};
  20. use lightning_invoice::Bolt11Invoice;
  21. use ln_regtest_rs::InvoiceStatus;
  22. use serde_json::json;
  23. use tokio::time::{sleep, timeout};
  24. use tokio_tungstenite::{connect_async, tungstenite::protocol::Message};
  25. async fn get_notification<T: StreamExt<Item = Result<Message, E>> + Unpin, E: Debug>(
  26. reader: &mut T,
  27. timeout_to_wait: Duration,
  28. ) -> (String, NotificationPayload) {
  29. let msg = timeout(timeout_to_wait, reader.next())
  30. .await
  31. .expect("timeout")
  32. .unwrap()
  33. .unwrap();
  34. let mut response: serde_json::Value =
  35. serde_json::from_str(&msg.to_text().unwrap()).expect("valid json");
  36. let mut params_raw = response
  37. .as_object_mut()
  38. .expect("object")
  39. .remove("params")
  40. .expect("valid params");
  41. let params_map = params_raw.as_object_mut().expect("params is object");
  42. (
  43. params_map
  44. .remove("subId")
  45. .unwrap()
  46. .as_str()
  47. .unwrap()
  48. .to_string(),
  49. serde_json::from_value(params_map.remove("payload").unwrap()).unwrap(),
  50. )
  51. }
  52. #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
  53. async fn test_regtest_mint_melt_round_trip() -> Result<()> {
  54. let lnd_client = init_lnd_client().await.unwrap();
  55. let wallet = Wallet::new(
  56. &get_mint_url(),
  57. CurrencyUnit::Sat,
  58. Arc::new(WalletMemoryDatabase::default()),
  59. &Mnemonic::generate(12)?.to_seed_normalized(""),
  60. None,
  61. )?;
  62. let (ws_stream, _) = connect_async(get_mint_ws_url())
  63. .await
  64. .expect("Failed to connect");
  65. let (mut write, mut reader) = ws_stream.split();
  66. let mint_quote = wallet.mint_quote(100.into(), None).await?;
  67. lnd_client.pay_invoice(mint_quote.request).await?;
  68. let mint_amount = wallet
  69. .mint(&mint_quote.id, SplitTarget::default(), None)
  70. .await?;
  71. assert!(mint_amount == 100.into());
  72. let invoice = lnd_client.create_invoice(50).await?;
  73. let melt = wallet.melt_quote(invoice, None).await?;
  74. write
  75. .send(Message::Text(serde_json::to_string(&json!({
  76. "jsonrpc": "2.0",
  77. "id": 2,
  78. "method": "subscribe",
  79. "params": {
  80. "kind": "bolt11_melt_quote",
  81. "filters": [
  82. melt.id.clone(),
  83. ],
  84. "subId": "test-sub",
  85. }
  86. }))?))
  87. .await?;
  88. assert_eq!(
  89. reader.next().await.unwrap().unwrap().to_text().unwrap(),
  90. r#"{"jsonrpc":"2.0","result":{"status":"OK","subId":"test-sub"},"id":2}"#
  91. );
  92. let melt_response = wallet.melt(&melt.id).await.unwrap();
  93. assert!(melt_response.preimage.is_some());
  94. assert!(melt_response.state == MeltQuoteState::Paid);
  95. let (sub_id, payload) = get_notification(&mut reader, Duration::from_millis(15000)).await;
  96. assert_eq!("test-sub", sub_id);
  97. let payload = match payload {
  98. NotificationPayload::MeltQuoteBolt11Response(melt) => melt,
  99. _ => panic!("Wrong payload"),
  100. };
  101. assert_eq!(payload.amount + payload.fee_reserve, 100.into());
  102. assert_eq!(payload.quote, melt.id);
  103. assert_eq!(payload.state, MeltQuoteState::Paid);
  104. Ok(())
  105. }
  106. #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
  107. async fn test_regtest_mint_melt() -> Result<()> {
  108. let lnd_client = init_lnd_client().await?;
  109. let wallet = Wallet::new(
  110. &get_mint_url(),
  111. CurrencyUnit::Sat,
  112. Arc::new(WalletMemoryDatabase::default()),
  113. &Mnemonic::generate(12)?.to_seed_normalized(""),
  114. None,
  115. )?;
  116. let mint_amount = Amount::from(100);
  117. let mint_quote = wallet.mint_quote(mint_amount, None).await?;
  118. assert_eq!(mint_quote.amount, mint_amount);
  119. lnd_client.pay_invoice(mint_quote.request).await?;
  120. let mint_amount = wallet
  121. .mint(&mint_quote.id, SplitTarget::default(), None)
  122. .await?;
  123. assert!(mint_amount == 100.into());
  124. Ok(())
  125. }
  126. #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
  127. async fn test_restore() -> Result<()> {
  128. let lnd_client = init_lnd_client().await?;
  129. let seed = Mnemonic::generate(12)?.to_seed_normalized("");
  130. let wallet = Wallet::new(
  131. &get_mint_url(),
  132. CurrencyUnit::Sat,
  133. Arc::new(WalletMemoryDatabase::default()),
  134. &seed,
  135. None,
  136. )?;
  137. let mint_quote = wallet.mint_quote(100.into(), None).await?;
  138. lnd_client.pay_invoice(mint_quote.request).await?;
  139. let _mint_amount = wallet
  140. .mint(&mint_quote.id, SplitTarget::default(), None)
  141. .await?;
  142. assert!(wallet.total_balance().await? == 100.into());
  143. let wallet_2 = Wallet::new(
  144. &get_mint_url(),
  145. CurrencyUnit::Sat,
  146. Arc::new(WalletMemoryDatabase::default()),
  147. &seed,
  148. None,
  149. )?;
  150. assert!(wallet_2.total_balance().await? == 0.into());
  151. let restored = wallet_2.restore().await?;
  152. let proofs = wallet_2.get_unspent_proofs().await?;
  153. wallet_2
  154. .swap(None, SplitTarget::default(), proofs, None, false)
  155. .await?;
  156. assert!(restored == 100.into());
  157. assert!(wallet_2.total_balance().await? == 100.into());
  158. let proofs = wallet.get_unspent_proofs().await?;
  159. let states = wallet.check_proofs_spent(proofs).await?;
  160. for state in states {
  161. if state.state != State::Spent {
  162. bail!("All proofs should be spent");
  163. }
  164. }
  165. Ok(())
  166. }
  167. #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
  168. async fn test_pay_invoice_twice() -> Result<()> {
  169. let lnd_client = init_lnd_client().await?;
  170. let seed = Mnemonic::generate(12)?.to_seed_normalized("");
  171. let wallet = Wallet::new(
  172. &get_mint_url(),
  173. CurrencyUnit::Sat,
  174. Arc::new(WalletMemoryDatabase::default()),
  175. &seed,
  176. None,
  177. )?;
  178. let mint_quote = wallet.mint_quote(100.into(), None).await?;
  179. lnd_client.pay_invoice(mint_quote.request).await?;
  180. let mint_amount = wallet
  181. .mint(&mint_quote.id, SplitTarget::default(), None)
  182. .await?;
  183. assert_eq!(mint_amount, 100.into());
  184. let invoice = lnd_client.create_invoice(10).await?;
  185. let melt_quote = wallet.melt_quote(invoice.clone(), None).await?;
  186. let melt = wallet.melt(&melt_quote.id).await.unwrap();
  187. let melt_two = wallet.melt_quote(invoice, None).await?;
  188. let melt_two = wallet.melt(&melt_two.id).await;
  189. match melt_two {
  190. Err(err) => match err {
  191. cdk::Error::RequestAlreadyPaid => (),
  192. _ => {
  193. bail!("Wrong invoice already paid");
  194. }
  195. },
  196. Ok(_) => {
  197. bail!("Should not have allowed second payment");
  198. }
  199. }
  200. let balance = wallet.total_balance().await?;
  201. assert_eq!(balance, (Amount::from(100) - melt.fee_paid - melt.amount));
  202. Ok(())
  203. }
  204. #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
  205. async fn test_internal_payment() -> Result<()> {
  206. let lnd_client = init_lnd_client().await?;
  207. let seed = Mnemonic::generate(12)?.to_seed_normalized("");
  208. let wallet = Wallet::new(
  209. &get_mint_url(),
  210. CurrencyUnit::Sat,
  211. Arc::new(WalletMemoryDatabase::default()),
  212. &seed,
  213. None,
  214. )?;
  215. let mint_quote = wallet.mint_quote(100.into(), None).await?;
  216. lnd_client.pay_invoice(mint_quote.request).await?;
  217. let _mint_amount = wallet
  218. .mint(&mint_quote.id, SplitTarget::default(), None)
  219. .await?;
  220. assert!(wallet.total_balance().await? == 100.into());
  221. let seed = Mnemonic::generate(12)?.to_seed_normalized("");
  222. let wallet_2 = Wallet::new(
  223. &get_mint_url(),
  224. CurrencyUnit::Sat,
  225. Arc::new(WalletMemoryDatabase::default()),
  226. &seed,
  227. None,
  228. )?;
  229. let mint_quote = wallet_2.mint_quote(10.into(), None).await?;
  230. let melt = wallet.melt_quote(mint_quote.request.clone(), None).await?;
  231. assert_eq!(melt.amount, 10.into());
  232. let _melted = wallet.melt(&melt.id).await.unwrap();
  233. let _wallet_2_mint = wallet_2
  234. .mint(&mint_quote.id, SplitTarget::default(), None)
  235. .await
  236. .unwrap();
  237. let cln_client = init_cln_client().await?;
  238. let payment_hash = Bolt11Invoice::from_str(&mint_quote.request)?;
  239. let check_paid = cln_client
  240. .check_incoming_invoice(payment_hash.payment_hash().to_string())
  241. .await?;
  242. match check_paid {
  243. InvoiceStatus::Unpaid => (),
  244. _ => {
  245. bail!("Invoice has incorrect status: {:?}", check_paid);
  246. }
  247. }
  248. let wallet_2_balance = wallet_2.total_balance().await?;
  249. assert!(wallet_2_balance == 10.into());
  250. let wallet_1_balance = wallet.total_balance().await?;
  251. assert!(wallet_1_balance == 90.into());
  252. Ok(())
  253. }
  254. #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
  255. async fn test_cached_mint() -> Result<()> {
  256. let lnd_client = init_lnd_client().await.unwrap();
  257. let wallet = Wallet::new(
  258. &get_mint_url(),
  259. CurrencyUnit::Sat,
  260. Arc::new(WalletMemoryDatabase::default()),
  261. &Mnemonic::generate(12)?.to_seed_normalized(""),
  262. None,
  263. )?;
  264. let mint_amount = Amount::from(100);
  265. let quote = wallet.mint_quote(mint_amount, None).await?;
  266. lnd_client.pay_invoice(quote.request).await?;
  267. loop {
  268. let status = wallet.mint_quote_state(&quote.id).await.unwrap();
  269. println!("Quote status: {}", status.state);
  270. if status.state == MintQuoteState::Paid {
  271. break;
  272. }
  273. sleep(Duration::from_secs(5)).await;
  274. }
  275. let active_keyset_id = wallet.get_active_mint_keyset().await?.id;
  276. let http_client = HttpClient::new();
  277. let premint_secrets =
  278. PreMintSecrets::random(active_keyset_id, 31.into(), &SplitTarget::default()).unwrap();
  279. let request = MintBolt11Request {
  280. quote: quote.id,
  281. outputs: premint_secrets.blinded_messages(),
  282. };
  283. let response = http_client
  284. .post_mint(get_mint_url().as_str().parse()?, request.clone())
  285. .await?;
  286. let response1 = http_client
  287. .post_mint(get_mint_url().as_str().parse()?, request)
  288. .await?;
  289. assert!(response == response1);
  290. Ok(())
  291. }