fake_wallet.rs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379
  1. use std::{sync::Arc, time::Duration};
  2. use anyhow::Result;
  3. use bip39::Mnemonic;
  4. use cdk::{
  5. amount::SplitTarget,
  6. cdk_database::WalletMemoryDatabase,
  7. nuts::{CurrencyUnit, MeltBolt11Request, MeltQuoteState, PreMintSecrets, State},
  8. wallet::{
  9. client::{HttpClient, HttpClientMethods},
  10. Wallet,
  11. },
  12. };
  13. use cdk_fake_wallet::{create_fake_invoice, FakeInvoiceDescription};
  14. use cdk_integration_tests::attempt_to_swap_pending;
  15. use tokio::time::sleep;
  16. const MINT_URL: &str = "http://127.0.0.1:8086";
  17. // If both pay and check return pending input proofs should remain pending
  18. #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
  19. async fn test_fake_tokens_pending() -> Result<()> {
  20. let wallet = Wallet::new(
  21. MINT_URL,
  22. CurrencyUnit::Sat,
  23. Arc::new(WalletMemoryDatabase::default()),
  24. &Mnemonic::generate(12)?.to_seed_normalized(""),
  25. None,
  26. )?;
  27. let mint_quote = wallet.mint_quote(100.into(), None).await?;
  28. sleep(Duration::from_millis(5)).await;
  29. let _mint_amount = wallet
  30. .mint(&mint_quote.id, SplitTarget::default(), None)
  31. .await?;
  32. let fake_description = FakeInvoiceDescription {
  33. pay_invoice_state: MeltQuoteState::Pending,
  34. check_payment_state: MeltQuoteState::Pending,
  35. pay_err: false,
  36. check_err: false,
  37. };
  38. let invoice = create_fake_invoice(1000, serde_json::to_string(&fake_description).unwrap());
  39. let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
  40. let melt = wallet.melt(&melt_quote.id).await;
  41. assert!(melt.is_err());
  42. attempt_to_swap_pending(&wallet).await?;
  43. Ok(())
  44. }
  45. // If the pay error fails and the check returns unknown or failed
  46. // The inputs proofs should be unset as spending
  47. #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
  48. async fn test_fake_melt_payment_fail() -> Result<()> {
  49. let wallet = Wallet::new(
  50. MINT_URL,
  51. CurrencyUnit::Sat,
  52. Arc::new(WalletMemoryDatabase::default()),
  53. &Mnemonic::generate(12)?.to_seed_normalized(""),
  54. None,
  55. )?;
  56. let mint_quote = wallet.mint_quote(100.into(), None).await?;
  57. sleep(Duration::from_millis(5)).await;
  58. let _mint_amount = wallet
  59. .mint(&mint_quote.id, SplitTarget::default(), None)
  60. .await?;
  61. let fake_description = FakeInvoiceDescription {
  62. pay_invoice_state: MeltQuoteState::Unknown,
  63. check_payment_state: MeltQuoteState::Unknown,
  64. pay_err: true,
  65. check_err: false,
  66. };
  67. let invoice = create_fake_invoice(1000, serde_json::to_string(&fake_description).unwrap());
  68. let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
  69. // The melt should error at the payment invoice command
  70. let melt = wallet.melt(&melt_quote.id).await;
  71. assert!(melt.is_err());
  72. let fake_description = FakeInvoiceDescription {
  73. pay_invoice_state: MeltQuoteState::Failed,
  74. check_payment_state: MeltQuoteState::Failed,
  75. pay_err: true,
  76. check_err: false,
  77. };
  78. let invoice = create_fake_invoice(1000, serde_json::to_string(&fake_description).unwrap());
  79. let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
  80. // The melt should error at the payment invoice command
  81. let melt = wallet.melt(&melt_quote.id).await;
  82. assert!(melt.is_err());
  83. // The mint should have unset proofs from pending since payment failed
  84. let all_proof = wallet.get_unspent_proofs().await?;
  85. let states = wallet.check_proofs_spent(all_proof).await?;
  86. for state in states {
  87. assert!(state.state == State::Unspent);
  88. }
  89. let wallet_bal = wallet.total_balance().await?;
  90. assert!(wallet_bal == 100.into());
  91. Ok(())
  92. }
  93. // When both the pay_invoice and check_invoice both fail
  94. // the proofs should remain as pending
  95. #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
  96. async fn test_fake_melt_payment_fail_and_check() -> Result<()> {
  97. let wallet = Wallet::new(
  98. MINT_URL,
  99. CurrencyUnit::Sat,
  100. Arc::new(WalletMemoryDatabase::default()),
  101. &Mnemonic::generate(12)?.to_seed_normalized(""),
  102. None,
  103. )?;
  104. let mint_quote = wallet.mint_quote(100.into(), None).await?;
  105. sleep(Duration::from_millis(5)).await;
  106. let _mint_amount = wallet
  107. .mint(&mint_quote.id, SplitTarget::default(), None)
  108. .await?;
  109. let fake_description = FakeInvoiceDescription {
  110. pay_invoice_state: MeltQuoteState::Unknown,
  111. check_payment_state: MeltQuoteState::Unknown,
  112. pay_err: true,
  113. check_err: true,
  114. };
  115. let invoice = create_fake_invoice(7000, serde_json::to_string(&fake_description).unwrap());
  116. let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
  117. // The melt should error at the payment invoice command
  118. let melt = wallet.melt(&melt_quote.id).await;
  119. assert!(melt.is_err());
  120. let pending = wallet
  121. .localstore
  122. .get_proofs(None, None, Some(vec![State::Pending]), None)
  123. .await?;
  124. assert!(!pending.is_empty());
  125. Ok(())
  126. }
  127. // In the case that the ln backend returns a failed status but does not error
  128. // The mint should do a second check, then remove proofs from pending
  129. #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
  130. async fn test_fake_melt_payment_return_fail_status() -> Result<()> {
  131. let wallet = Wallet::new(
  132. MINT_URL,
  133. CurrencyUnit::Sat,
  134. Arc::new(WalletMemoryDatabase::default()),
  135. &Mnemonic::generate(12)?.to_seed_normalized(""),
  136. None,
  137. )?;
  138. let mint_quote = wallet.mint_quote(100.into(), None).await?;
  139. sleep(Duration::from_millis(5)).await;
  140. let _mint_amount = wallet
  141. .mint(&mint_quote.id, SplitTarget::default(), None)
  142. .await?;
  143. let fake_description = FakeInvoiceDescription {
  144. pay_invoice_state: MeltQuoteState::Failed,
  145. check_payment_state: MeltQuoteState::Failed,
  146. pay_err: false,
  147. check_err: false,
  148. };
  149. let invoice = create_fake_invoice(7000, serde_json::to_string(&fake_description).unwrap());
  150. let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
  151. // The melt should error at the payment invoice command
  152. let melt = wallet.melt(&melt_quote.id).await;
  153. assert!(melt.is_err());
  154. let fake_description = FakeInvoiceDescription {
  155. pay_invoice_state: MeltQuoteState::Unknown,
  156. check_payment_state: MeltQuoteState::Unknown,
  157. pay_err: false,
  158. check_err: false,
  159. };
  160. let invoice = create_fake_invoice(7000, serde_json::to_string(&fake_description).unwrap());
  161. let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
  162. // The melt should error at the payment invoice command
  163. let melt = wallet.melt(&melt_quote.id).await;
  164. assert!(melt.is_err());
  165. let pending = wallet
  166. .localstore
  167. .get_proofs(None, None, Some(vec![State::Pending]), None)
  168. .await?;
  169. assert!(pending.is_empty());
  170. Ok(())
  171. }
  172. // In the case that the ln backend returns a failed status but does not error
  173. // The mint should do a second check, then remove proofs from pending
  174. #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
  175. async fn test_fake_melt_payment_error_unknown() -> Result<()> {
  176. let wallet = Wallet::new(
  177. MINT_URL,
  178. CurrencyUnit::Sat,
  179. Arc::new(WalletMemoryDatabase::default()),
  180. &Mnemonic::generate(12)?.to_seed_normalized(""),
  181. None,
  182. )?;
  183. let mint_quote = wallet.mint_quote(100.into(), None).await?;
  184. sleep(Duration::from_millis(5)).await;
  185. let _mint_amount = wallet
  186. .mint(&mint_quote.id, SplitTarget::default(), None)
  187. .await?;
  188. let fake_description = FakeInvoiceDescription {
  189. pay_invoice_state: MeltQuoteState::Failed,
  190. check_payment_state: MeltQuoteState::Unknown,
  191. pay_err: true,
  192. check_err: false,
  193. };
  194. let invoice = create_fake_invoice(7000, serde_json::to_string(&fake_description).unwrap());
  195. let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
  196. // The melt should error at the payment invoice command
  197. let melt = wallet.melt(&melt_quote.id).await;
  198. assert!(melt.is_err());
  199. let fake_description = FakeInvoiceDescription {
  200. pay_invoice_state: MeltQuoteState::Unknown,
  201. check_payment_state: MeltQuoteState::Unknown,
  202. pay_err: true,
  203. check_err: false,
  204. };
  205. let invoice = create_fake_invoice(7000, serde_json::to_string(&fake_description).unwrap());
  206. let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
  207. // The melt should error at the payment invoice command
  208. let melt = wallet.melt(&melt_quote.id).await;
  209. assert!(melt.is_err());
  210. let pending = wallet
  211. .localstore
  212. .get_proofs(None, None, Some(vec![State::Pending]), None)
  213. .await?;
  214. assert!(pending.is_empty());
  215. Ok(())
  216. }
  217. // In the case that the ln backend returns an err
  218. // The mint should do a second check, that returns paid
  219. // Proofs should remain pending
  220. #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
  221. async fn test_fake_melt_payment_err_paid() -> Result<()> {
  222. let wallet = Wallet::new(
  223. MINT_URL,
  224. CurrencyUnit::Sat,
  225. Arc::new(WalletMemoryDatabase::default()),
  226. &Mnemonic::generate(12)?.to_seed_normalized(""),
  227. None,
  228. )?;
  229. let mint_quote = wallet.mint_quote(100.into(), None).await?;
  230. sleep(Duration::from_millis(5)).await;
  231. let _mint_amount = wallet
  232. .mint(&mint_quote.id, SplitTarget::default(), None)
  233. .await?;
  234. let fake_description = FakeInvoiceDescription {
  235. pay_invoice_state: MeltQuoteState::Failed,
  236. check_payment_state: MeltQuoteState::Paid,
  237. pay_err: true,
  238. check_err: false,
  239. };
  240. let invoice = create_fake_invoice(7000, serde_json::to_string(&fake_description).unwrap());
  241. let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
  242. // The melt should error at the payment invoice command
  243. let melt = wallet.melt(&melt_quote.id).await;
  244. assert!(melt.is_err());
  245. attempt_to_swap_pending(&wallet).await?;
  246. Ok(())
  247. }
  248. #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
  249. async fn test_fake_melt_change_in_quote() -> Result<()> {
  250. let wallet = Wallet::new(
  251. MINT_URL,
  252. CurrencyUnit::Sat,
  253. Arc::new(WalletMemoryDatabase::default()),
  254. &Mnemonic::generate(12)?.to_seed_normalized(""),
  255. None,
  256. )?;
  257. let mint_quote = wallet.mint_quote(100.into(), None).await?;
  258. sleep(Duration::from_millis(5)).await;
  259. let _mint_amount = wallet
  260. .mint(&mint_quote.id, SplitTarget::default(), None)
  261. .await?;
  262. let fake_description = FakeInvoiceDescription::default();
  263. let invoice = create_fake_invoice(9000, serde_json::to_string(&fake_description).unwrap());
  264. let proofs = wallet.get_unspent_proofs().await?;
  265. let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
  266. let keyset = wallet.get_active_mint_keyset().await?;
  267. let premint_secrets = PreMintSecrets::random(keyset.id, 100.into(), &SplitTarget::default())?;
  268. let client = HttpClient::new();
  269. let melt_request = MeltBolt11Request {
  270. quote: melt_quote.id.clone(),
  271. inputs: proofs.clone(),
  272. outputs: Some(premint_secrets.blinded_messages()),
  273. };
  274. let melt_response = client.post_melt(MINT_URL.parse()?, melt_request).await?;
  275. assert!(melt_response.change.is_some());
  276. let check = wallet.melt_quote_status(&melt_quote.id).await?;
  277. let mut melt_change = melt_response.change.unwrap();
  278. melt_change.sort_by(|a, b| a.amount.cmp(&b.amount));
  279. let mut check = check.change.unwrap();
  280. check.sort_by(|a, b| a.amount.cmp(&b.amount));
  281. assert_eq!(melt_change, check);
  282. Ok(())
  283. }