regtest.rs 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  1. use std::{str::FromStr, sync::Arc};
  2. use anyhow::{bail, Result};
  3. use bip39::Mnemonic;
  4. use cdk::{
  5. amount::{Amount, SplitTarget},
  6. cdk_database::WalletMemoryDatabase,
  7. nuts::{CurrencyUnit, MeltQuoteState, State},
  8. wallet::Wallet,
  9. };
  10. use cdk_integration_tests::init_regtest::{get_mint_url, init_cln_client, init_lnd_client};
  11. use lightning_invoice::Bolt11Invoice;
  12. use ln_regtest_rs::InvoiceStatus;
  13. #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
  14. async fn test_regtest_mint_melt_round_trip() -> Result<()> {
  15. let lnd_client = init_lnd_client().await.unwrap();
  16. let wallet = Wallet::new(
  17. &get_mint_url(),
  18. CurrencyUnit::Sat,
  19. Arc::new(WalletMemoryDatabase::default()),
  20. &Mnemonic::generate(12)?.to_seed_normalized(""),
  21. None,
  22. )?;
  23. let mint_quote = wallet.mint_quote(100.into(), None).await?;
  24. lnd_client.pay_invoice(mint_quote.request).await?;
  25. let mint_amount = wallet
  26. .mint(&mint_quote.id, SplitTarget::default(), None)
  27. .await?;
  28. assert!(mint_amount == 100.into());
  29. let invoice = lnd_client.create_invoice(50).await?;
  30. let melt = wallet.melt_quote(invoice, None).await?;
  31. let melt = wallet.melt(&melt.id).await.unwrap();
  32. assert!(melt.preimage.is_some());
  33. assert!(melt.state == MeltQuoteState::Paid);
  34. Ok(())
  35. }
  36. #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
  37. async fn test_regtest_mint_melt() -> Result<()> {
  38. let lnd_client = init_lnd_client().await?;
  39. let wallet = Wallet::new(
  40. &get_mint_url(),
  41. CurrencyUnit::Sat,
  42. Arc::new(WalletMemoryDatabase::default()),
  43. &Mnemonic::generate(12)?.to_seed_normalized(""),
  44. None,
  45. )?;
  46. let mint_amount = Amount::from(100);
  47. let mint_quote = wallet.mint_quote(mint_amount, None).await?;
  48. assert_eq!(mint_quote.amount, mint_amount);
  49. lnd_client.pay_invoice(mint_quote.request).await?;
  50. let mint_amount = wallet
  51. .mint(&mint_quote.id, SplitTarget::default(), None)
  52. .await?;
  53. assert!(mint_amount == 100.into());
  54. Ok(())
  55. }
  56. #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
  57. async fn test_restore() -> Result<()> {
  58. let lnd_client = init_lnd_client().await?;
  59. let seed = Mnemonic::generate(12)?.to_seed_normalized("");
  60. let wallet = Wallet::new(
  61. &get_mint_url(),
  62. CurrencyUnit::Sat,
  63. Arc::new(WalletMemoryDatabase::default()),
  64. &seed,
  65. None,
  66. )?;
  67. let mint_quote = wallet.mint_quote(100.into(), None).await?;
  68. lnd_client.pay_invoice(mint_quote.request).await?;
  69. let _mint_amount = wallet
  70. .mint(&mint_quote.id, SplitTarget::default(), None)
  71. .await?;
  72. assert!(wallet.total_balance().await? == 100.into());
  73. let wallet_2 = Wallet::new(
  74. &get_mint_url(),
  75. CurrencyUnit::Sat,
  76. Arc::new(WalletMemoryDatabase::default()),
  77. &seed,
  78. None,
  79. )?;
  80. assert!(wallet_2.total_balance().await? == 0.into());
  81. let restored = wallet_2.restore().await?;
  82. let proofs = wallet_2.get_proofs().await?;
  83. wallet_2
  84. .swap(None, SplitTarget::default(), proofs, None, false)
  85. .await?;
  86. assert!(restored == 100.into());
  87. assert!(wallet_2.total_balance().await? == 100.into());
  88. let proofs = wallet.get_proofs().await?;
  89. let states = wallet.check_proofs_spent(proofs).await?;
  90. for state in states {
  91. if state.state != State::Spent {
  92. bail!("All proofs should be spent");
  93. }
  94. }
  95. Ok(())
  96. }
  97. #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
  98. async fn test_pay_invoice_twice() -> Result<()> {
  99. let lnd_client = init_lnd_client().await?;
  100. let seed = Mnemonic::generate(12)?.to_seed_normalized("");
  101. let wallet = Wallet::new(
  102. &get_mint_url(),
  103. CurrencyUnit::Sat,
  104. Arc::new(WalletMemoryDatabase::default()),
  105. &seed,
  106. None,
  107. )?;
  108. let mint_quote = wallet.mint_quote(100.into(), None).await?;
  109. lnd_client.pay_invoice(mint_quote.request).await?;
  110. let mint_amount = wallet
  111. .mint(&mint_quote.id, SplitTarget::default(), None)
  112. .await?;
  113. assert_eq!(mint_amount, 100.into());
  114. let invoice = lnd_client.create_invoice(10).await?;
  115. let melt_quote = wallet.melt_quote(invoice.clone(), None).await?;
  116. let melt = wallet.melt(&melt_quote.id).await.unwrap();
  117. let melt_two = wallet.melt_quote(invoice, None).await?;
  118. let melt_two = wallet.melt(&melt_two.id).await;
  119. match melt_two {
  120. Err(err) => match err {
  121. cdk::Error::RequestAlreadyPaid => (),
  122. _ => {
  123. bail!("Wrong invoice already paid");
  124. }
  125. },
  126. Ok(_) => {
  127. bail!("Should not have allowed second payment");
  128. }
  129. }
  130. let balance = wallet.total_balance().await?;
  131. assert_eq!(balance, (Amount::from(100) - melt.fee_paid - melt.amount));
  132. Ok(())
  133. }
  134. #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
  135. async fn test_internal_payment() -> Result<()> {
  136. let lnd_client = init_lnd_client().await?;
  137. let seed = Mnemonic::generate(12)?.to_seed_normalized("");
  138. let wallet = Wallet::new(
  139. &get_mint_url(),
  140. CurrencyUnit::Sat,
  141. Arc::new(WalletMemoryDatabase::default()),
  142. &seed,
  143. None,
  144. )?;
  145. let mint_quote = wallet.mint_quote(100.into(), None).await?;
  146. lnd_client.pay_invoice(mint_quote.request).await?;
  147. let _mint_amount = wallet
  148. .mint(&mint_quote.id, SplitTarget::default(), None)
  149. .await?;
  150. assert!(wallet.total_balance().await? == 100.into());
  151. let seed = Mnemonic::generate(12)?.to_seed_normalized("");
  152. let wallet_2 = Wallet::new(
  153. &get_mint_url(),
  154. CurrencyUnit::Sat,
  155. Arc::new(WalletMemoryDatabase::default()),
  156. &seed,
  157. None,
  158. )?;
  159. let mint_quote = wallet_2.mint_quote(10.into(), None).await?;
  160. let melt = wallet.melt_quote(mint_quote.request.clone(), None).await?;
  161. assert_eq!(melt.amount, 10.into());
  162. let _melted = wallet.melt(&melt.id).await.unwrap();
  163. let _wallet_2_mint = wallet_2
  164. .mint(&mint_quote.id, SplitTarget::default(), None)
  165. .await
  166. .unwrap();
  167. let cln_client = init_cln_client().await?;
  168. let payment_hash = Bolt11Invoice::from_str(&mint_quote.request)?;
  169. let check_paid = cln_client
  170. .check_incoming_invoice(payment_hash.payment_hash().to_string())
  171. .await?;
  172. match check_paid {
  173. InvoiceStatus::Unpaid => (),
  174. _ => {
  175. bail!("Invoice has incorrect status: {:?}", check_paid);
  176. }
  177. }
  178. let wallet_2_balance = wallet_2.total_balance().await?;
  179. assert!(wallet_2_balance == 10.into());
  180. let wallet_1_balance = wallet.total_balance().await?;
  181. assert!(wallet_1_balance == 90.into());
  182. Ok(())
  183. }