1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237 |
- use std::sync::Arc;
- use anyhow::{bail, Result};
- use bip39::Mnemonic;
- use cdk::amount::SplitTarget;
- use cdk::nuts::nut00::ProofsMethods;
- use cdk::nuts::{
- CurrencyUnit, MeltBolt11Request, MeltQuoteState, MintBolt11Request, PreMintSecrets, Proofs,
- SecretKey, State, SwapRequest,
- };
- use cdk::wallet::client::{HttpClient, MintConnector};
- use cdk::wallet::Wallet;
- use cdk_fake_wallet::{create_fake_invoice, FakeInvoiceDescription};
- use cdk_integration_tests::{attempt_to_swap_pending, wait_for_mint_to_be_paid};
- use cdk_sqlite::wallet::memory;
- const MINT_URL: &str = "http://127.0.0.1:8086";
- // If both pay and check return pending input proofs should remain pending
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_tokens_pending() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let _mint_amount = wallet
- .mint(&mint_quote.id, SplitTarget::default(), None)
- .await?;
- let fake_description = FakeInvoiceDescription {
- pay_invoice_state: MeltQuoteState::Pending,
- check_payment_state: MeltQuoteState::Pending,
- pay_err: false,
- check_err: false,
- };
- let invoice = create_fake_invoice(1000, serde_json::to_string(&fake_description).unwrap());
- let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
- let melt = wallet.melt(&melt_quote.id).await;
- assert!(melt.is_err());
- attempt_to_swap_pending(&wallet).await?;
- Ok(())
- }
- // If the pay error fails and the check returns unknown or failed
- // The inputs proofs should be unset as spending
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_melt_payment_fail() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let _mint_amount = wallet
- .mint(&mint_quote.id, SplitTarget::default(), None)
- .await?;
- let fake_description = FakeInvoiceDescription {
- pay_invoice_state: MeltQuoteState::Unknown,
- check_payment_state: MeltQuoteState::Unknown,
- pay_err: true,
- check_err: false,
- };
- let invoice = create_fake_invoice(1000, serde_json::to_string(&fake_description).unwrap());
- let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
- // The melt should error at the payment invoice command
- let melt = wallet.melt(&melt_quote.id).await;
- assert!(melt.is_err());
- let fake_description = FakeInvoiceDescription {
- pay_invoice_state: MeltQuoteState::Failed,
- check_payment_state: MeltQuoteState::Failed,
- pay_err: true,
- check_err: false,
- };
- let invoice = create_fake_invoice(1000, serde_json::to_string(&fake_description).unwrap());
- let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
- // The melt should error at the payment invoice command
- let melt = wallet.melt(&melt_quote.id).await;
- assert!(melt.is_err());
- // The mint should have unset proofs from pending since payment failed
- let all_proof = wallet.get_unspent_proofs().await?;
- let states = wallet.check_proofs_spent(all_proof).await?;
- for state in states {
- assert!(state.state == State::Unspent);
- }
- let wallet_bal = wallet.total_balance().await?;
- assert!(wallet_bal == 100.into());
- Ok(())
- }
- // When both the pay_invoice and check_invoice both fail
- // the proofs should remain as pending
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_melt_payment_fail_and_check() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let _mint_amount = wallet
- .mint(&mint_quote.id, SplitTarget::default(), None)
- .await?;
- let fake_description = FakeInvoiceDescription {
- pay_invoice_state: MeltQuoteState::Unknown,
- check_payment_state: MeltQuoteState::Unknown,
- pay_err: true,
- check_err: true,
- };
- let invoice = create_fake_invoice(7000, serde_json::to_string(&fake_description).unwrap());
- let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
- // The melt should error at the payment invoice command
- let melt = wallet.melt(&melt_quote.id).await;
- assert!(melt.is_err());
- let pending = wallet
- .localstore
- .get_proofs(None, None, Some(vec![State::Pending]), None)
- .await?;
- assert!(!pending.is_empty());
- Ok(())
- }
- // In the case that the ln backend returns a failed status but does not error
- // The mint should do a second check, then remove proofs from pending
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_melt_payment_return_fail_status() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let _mint_amount = wallet
- .mint(&mint_quote.id, SplitTarget::default(), None)
- .await?;
- let fake_description = FakeInvoiceDescription {
- pay_invoice_state: MeltQuoteState::Failed,
- check_payment_state: MeltQuoteState::Failed,
- pay_err: false,
- check_err: false,
- };
- let invoice = create_fake_invoice(7000, serde_json::to_string(&fake_description).unwrap());
- let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
- // The melt should error at the payment invoice command
- let melt = wallet.melt(&melt_quote.id).await;
- assert!(melt.is_err());
- let fake_description = FakeInvoiceDescription {
- pay_invoice_state: MeltQuoteState::Unknown,
- check_payment_state: MeltQuoteState::Unknown,
- pay_err: false,
- check_err: false,
- };
- let invoice = create_fake_invoice(7000, serde_json::to_string(&fake_description).unwrap());
- let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
- // The melt should error at the payment invoice command
- let melt = wallet.melt(&melt_quote.id).await;
- assert!(melt.is_err());
- let pending = wallet
- .localstore
- .get_proofs(None, None, Some(vec![State::Pending]), None)
- .await?;
- assert!(pending.is_empty());
- Ok(())
- }
- // In the case that the ln backend returns a failed status but does not error
- // The mint should do a second check, then remove proofs from pending
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_melt_payment_error_unknown() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let _mint_amount = wallet
- .mint(&mint_quote.id, SplitTarget::default(), None)
- .await?;
- let fake_description = FakeInvoiceDescription {
- pay_invoice_state: MeltQuoteState::Failed,
- check_payment_state: MeltQuoteState::Unknown,
- pay_err: true,
- check_err: false,
- };
- let invoice = create_fake_invoice(7000, serde_json::to_string(&fake_description).unwrap());
- let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
- // The melt should error at the payment invoice command
- let melt = wallet.melt(&melt_quote.id).await;
- assert!(melt.is_err());
- let fake_description = FakeInvoiceDescription {
- pay_invoice_state: MeltQuoteState::Unknown,
- check_payment_state: MeltQuoteState::Unknown,
- pay_err: true,
- check_err: false,
- };
- let invoice = create_fake_invoice(7000, serde_json::to_string(&fake_description).unwrap());
- let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
- // The melt should error at the payment invoice command
- let melt = wallet.melt(&melt_quote.id).await;
- assert!(melt.is_err());
- let pending = wallet
- .localstore
- .get_proofs(None, None, Some(vec![State::Pending]), None)
- .await?;
- assert!(pending.is_empty());
- Ok(())
- }
- // In the case that the ln backend returns an err
- // The mint should do a second check, that returns paid
- // Proofs should remain pending
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_melt_payment_err_paid() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let _mint_amount = wallet
- .mint(&mint_quote.id, SplitTarget::default(), None)
- .await?;
- let fake_description = FakeInvoiceDescription {
- pay_invoice_state: MeltQuoteState::Failed,
- check_payment_state: MeltQuoteState::Paid,
- pay_err: true,
- check_err: false,
- };
- let invoice = create_fake_invoice(7000, serde_json::to_string(&fake_description).unwrap());
- let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
- // The melt should error at the payment invoice command
- let melt = wallet.melt(&melt_quote.id).await;
- assert!(melt.is_err());
- attempt_to_swap_pending(&wallet).await?;
- Ok(())
- }
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_melt_change_in_quote() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let _mint_amount = wallet
- .mint(&mint_quote.id, SplitTarget::default(), None)
- .await?;
- let fake_description = FakeInvoiceDescription::default();
- let invoice = create_fake_invoice(9000, serde_json::to_string(&fake_description).unwrap());
- let proofs = wallet.get_unspent_proofs().await?;
- let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
- let keyset = wallet.get_active_mint_keyset().await?;
- let premint_secrets = PreMintSecrets::random(keyset.id, 100.into(), &SplitTarget::default())?;
- let client = HttpClient::new(MINT_URL.parse()?);
- let melt_request = MeltBolt11Request {
- quote: melt_quote.id.clone(),
- inputs: proofs.clone(),
- outputs: Some(premint_secrets.blinded_messages()),
- };
- let melt_response = client.post_melt(melt_request).await?;
- assert!(melt_response.change.is_some());
- let check = wallet.melt_quote_status(&melt_quote.id).await?;
- let mut melt_change = melt_response.change.unwrap();
- melt_change.sort_by(|a, b| a.amount.cmp(&b.amount));
- let mut check = check.change.unwrap();
- check.sort_by(|a, b| a.amount.cmp(&b.amount));
- assert_eq!(melt_change, check);
- Ok(())
- }
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_database_type() -> Result<()> {
- // Get the database type and work dir from environment
- let db_type = std::env::var("MINT_DATABASE").expect("MINT_DATABASE env var should be set");
- let work_dir =
- std::env::var("CDK_MINTD_WORK_DIR").expect("CDK_MINTD_WORK_DIR env var should be set");
- // Check that the correct database file exists
- match db_type.as_str() {
- "REDB" => {
- let db_path = std::path::Path::new(&work_dir).join("cdk-mintd.redb");
- assert!(
- db_path.exists(),
- "Expected redb database file to exist at {:?}",
- db_path
- );
- }
- "SQLITE" => {
- let db_path = std::path::Path::new(&work_dir).join("cdk-mintd.sqlite");
- assert!(
- db_path.exists(),
- "Expected sqlite database file to exist at {:?}",
- db_path
- );
- }
- "MEMORY" => {
- // Memory database has no file to check
- println!("Memory database in use - no file to check");
- }
- _ => bail!("Unknown database type: {}", db_type),
- }
- Ok(())
- }
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_mint_with_witness() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let proofs = wallet
- .mint(&mint_quote.id, SplitTarget::default(), None)
- .await?;
- let mint_amount = proofs.total_amount()?;
- assert!(mint_amount == 100.into());
- Ok(())
- }
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_mint_without_witness() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let active_keyset_id = wallet.get_active_mint_keyset().await?.id;
- let premint_secrets =
- PreMintSecrets::random(active_keyset_id, 100.into(), &SplitTarget::default()).unwrap();
- let request = MintBolt11Request {
- quote: mint_quote.id,
- outputs: premint_secrets.blinded_messages(),
- signature: None,
- };
- let response = http_client.post_mint(request.clone()).await;
- match response {
- Err(cdk::error::Error::SignatureMissingOrInvalid) => Ok(()),
- Err(err) => bail!("Wrong mint response for minting without witness: {}", err),
- Ok(_) => bail!("Minting should not have succeed without a witness"),
- }
- }
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_mint_with_wrong_witness() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let active_keyset_id = wallet.get_active_mint_keyset().await?.id;
- let premint_secrets =
- PreMintSecrets::random(active_keyset_id, 100.into(), &SplitTarget::default()).unwrap();
- let mut request = MintBolt11Request {
- quote: mint_quote.id,
- outputs: premint_secrets.blinded_messages(),
- signature: None,
- };
- let secret_key = SecretKey::generate();
- request.sign(secret_key)?;
- let response = http_client.post_mint(request.clone()).await;
- match response {
- Err(cdk::error::Error::SignatureMissingOrInvalid) => Ok(()),
- Err(err) => bail!("Wrong mint response for minting without witness: {}", err),
- Ok(_) => bail!("Minting should not have succeed without a witness"),
- }
- }
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_mint_inflated() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let active_keyset_id = wallet.get_active_mint_keyset().await?.id;
- let pre_mint = PreMintSecrets::random(active_keyset_id, 500.into(), &SplitTarget::None)?;
- let quote_info = wallet
- .localstore
- .get_mint_quote(&mint_quote.id)
- .await?
- .expect("there is a quote");
- let mut mint_request = MintBolt11Request {
- quote: mint_quote.id,
- outputs: pre_mint.blinded_messages(),
- signature: None,
- };
- if let Some(secret_key) = quote_info.secret_key {
- mint_request.sign(secret_key)?;
- }
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let response = http_client.post_mint(mint_request.clone()).await;
- match response {
- Err(err) => match err {
- cdk::Error::TransactionUnbalanced(_, _, _) => (),
- err => {
- bail!("Wrong mint error returned: {}", err.to_string());
- }
- },
- Ok(_) => {
- bail!("Should not have allowed second payment");
- }
- }
- Ok(())
- }
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_mint_multiple_units() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let active_keyset_id = wallet.get_active_mint_keyset().await?.id;
- let pre_mint = PreMintSecrets::random(active_keyset_id, 50.into(), &SplitTarget::None)?;
- let wallet_usd = Wallet::new(
- MINT_URL,
- CurrencyUnit::Usd,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let active_keyset_id = wallet_usd.get_active_mint_keyset().await?.id;
- let usd_pre_mint = PreMintSecrets::random(active_keyset_id, 50.into(), &SplitTarget::None)?;
- let quote_info = wallet
- .localstore
- .get_mint_quote(&mint_quote.id)
- .await?
- .expect("there is a quote");
- let mut sat_outputs = pre_mint.blinded_messages();
- let mut usd_outputs = usd_pre_mint.blinded_messages();
- sat_outputs.append(&mut usd_outputs);
- let mut mint_request = MintBolt11Request {
- quote: mint_quote.id,
- outputs: sat_outputs,
- signature: None,
- };
- if let Some(secret_key) = quote_info.secret_key {
- mint_request.sign(secret_key)?;
- }
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let response = http_client.post_mint(mint_request.clone()).await;
- match response {
- Err(err) => match err {
- cdk::Error::MultipleUnits => (),
- err => {
- bail!("Wrong mint error returned: {}", err.to_string());
- }
- },
- Ok(_) => {
- bail!("Should not have allowed to mint with multiple units");
- }
- }
- Ok(())
- }
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_mint_multiple_unit_swap() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let proofs = wallet.mint(&mint_quote.id, SplitTarget::None, None).await?;
- let wallet_usd = Wallet::new(
- MINT_URL,
- CurrencyUnit::Usd,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet_usd.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet_usd, &mint_quote.id, 60).await?;
- let usd_proofs = wallet_usd
- .mint(&mint_quote.id, SplitTarget::None, None)
- .await?;
- let active_keyset_id = wallet.get_active_mint_keyset().await?.id;
- {
- let inputs: Proofs = vec![
- proofs.first().expect("There is a proof").clone(),
- usd_proofs.first().expect("There is a proof").clone(),
- ];
- let pre_mint =
- PreMintSecrets::random(active_keyset_id, inputs.total_amount()?, &SplitTarget::None)?;
- let swap_request = SwapRequest {
- inputs,
- outputs: pre_mint.blinded_messages(),
- };
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let response = http_client.post_swap(swap_request.clone()).await;
- match response {
- Err(err) => match err {
- cdk::Error::MultipleUnits => (),
- err => {
- bail!("Wrong mint error returned: {}", err.to_string());
- }
- },
- Ok(_) => {
- bail!("Should not have allowed to mint with multiple units");
- }
- }
- }
- {
- let usd_active_keyset_id = wallet_usd.get_active_mint_keyset().await?.id;
- let inputs: Proofs = proofs.into_iter().take(2).collect();
- let total_inputs = inputs.total_amount()?;
- let half = total_inputs / 2.into();
- let usd_pre_mint = PreMintSecrets::random(usd_active_keyset_id, half, &SplitTarget::None)?;
- let pre_mint =
- PreMintSecrets::random(active_keyset_id, total_inputs - half, &SplitTarget::None)?;
- let mut usd_outputs = usd_pre_mint.blinded_messages();
- let mut sat_outputs = pre_mint.blinded_messages();
- usd_outputs.append(&mut sat_outputs);
- let swap_request = SwapRequest {
- inputs,
- outputs: usd_outputs,
- };
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let response = http_client.post_swap(swap_request.clone()).await;
- match response {
- Err(err) => match err {
- cdk::Error::MultipleUnits => (),
- err => {
- bail!("Wrong mint error returned: {}", err.to_string());
- }
- },
- Ok(_) => {
- bail!("Should not have allowed to mint with multiple units");
- }
- }
- }
- Ok(())
- }
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_mint_multiple_unit_melt() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await.unwrap();
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let proofs = wallet
- .mint(&mint_quote.id, SplitTarget::None, None)
- .await
- .unwrap();
- println!("Minted sat");
- let wallet_usd = Wallet::new(
- MINT_URL,
- CurrencyUnit::Usd,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet_usd.mint_quote(100.into(), None).await.unwrap();
- println!("Minted quote usd");
- wait_for_mint_to_be_paid(&wallet_usd, &mint_quote.id, 60).await?;
- let usd_proofs = wallet_usd
- .mint(&mint_quote.id, SplitTarget::None, None)
- .await
- .unwrap();
- {
- let inputs: Proofs = vec![
- proofs.first().expect("There is a proof").clone(),
- usd_proofs.first().expect("There is a proof").clone(),
- ];
- let input_amount: u64 = inputs.total_amount()?.into();
- let invoice = create_fake_invoice((input_amount - 1) * 1000, "".to_string());
- let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
- let melt_request = MeltBolt11Request {
- quote: melt_quote.id,
- inputs,
- outputs: None,
- };
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let response = http_client.post_melt(melt_request.clone()).await;
- match response {
- Err(err) => match err {
- cdk::Error::MultipleUnits => (),
- err => {
- bail!("Wrong mint error returned: {}", err.to_string());
- }
- },
- Ok(_) => {
- bail!("Should not have allowed to melt with multiple units");
- }
- }
- }
- {
- let inputs: Proofs = vec![proofs.first().expect("There is a proof").clone()];
- let input_amount: u64 = inputs.total_amount()?.into();
- let invoice = create_fake_invoice((input_amount - 1) * 1000, "".to_string());
- let active_keyset_id = wallet.get_active_mint_keyset().await?.id;
- let usd_active_keyset_id = wallet_usd.get_active_mint_keyset().await?.id;
- let usd_pre_mint = PreMintSecrets::random(
- usd_active_keyset_id,
- inputs.total_amount()? + 100.into(),
- &SplitTarget::None,
- )?;
- let pre_mint = PreMintSecrets::random(active_keyset_id, 100.into(), &SplitTarget::None)?;
- let mut usd_outputs = usd_pre_mint.blinded_messages();
- let mut sat_outputs = pre_mint.blinded_messages();
- usd_outputs.append(&mut sat_outputs);
- let quote = wallet.melt_quote(invoice.to_string(), None).await?;
- let melt_request = MeltBolt11Request {
- quote: quote.id,
- inputs,
- outputs: Some(usd_outputs),
- };
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let response = http_client.post_melt(melt_request.clone()).await;
- match response {
- Err(err) => match err {
- cdk::Error::MultipleUnits => (),
- err => {
- bail!("Wrong mint error returned: {}", err.to_string());
- }
- },
- Ok(_) => {
- bail!("Should not have allowed to melt with multiple units");
- }
- }
- }
- Ok(())
- }
- /// Test swap where input unit != output unit
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_mint_input_output_mismatch() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let proofs = wallet.mint(&mint_quote.id, SplitTarget::None, None).await?;
- let wallet_usd = Wallet::new(
- MINT_URL,
- CurrencyUnit::Usd,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let usd_active_keyset_id = wallet_usd.get_active_mint_keyset().await?.id;
- let inputs = proofs;
- let pre_mint = PreMintSecrets::random(
- usd_active_keyset_id,
- inputs.total_amount()?,
- &SplitTarget::None,
- )?;
- let swap_request = SwapRequest {
- inputs,
- outputs: pre_mint.blinded_messages(),
- };
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let response = http_client.post_swap(swap_request.clone()).await;
- match response {
- Err(err) => match err {
- cdk::Error::UnitMismatch => (),
- err => bail!("Wrong error returned: {}", err),
- },
- Ok(_) => {
- bail!("Should not have allowed to mint with multiple units");
- }
- }
- Ok(())
- }
- /// Test swap where input is less the output
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_mint_swap_inflated() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let proofs = wallet.mint(&mint_quote.id, SplitTarget::None, None).await?;
- let active_keyset_id = wallet.get_active_mint_keyset().await?.id;
- let pre_mint = PreMintSecrets::random(active_keyset_id, 101.into(), &SplitTarget::None)?;
- let swap_request = SwapRequest {
- inputs: proofs,
- outputs: pre_mint.blinded_messages(),
- };
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let response = http_client.post_swap(swap_request.clone()).await;
- match response {
- Err(err) => match err {
- cdk::Error::TransactionUnbalanced(_, _, _) => (),
- err => {
- bail!("Wrong mint error returned: {}", err.to_string());
- }
- },
- Ok(_) => {
- bail!("Should not have allowed to mint with multiple units");
- }
- }
- Ok(())
- }
- /// Test swap after failure
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_mint_swap_spend_after_fail() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(WalletMemoryDatabase::default()),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let proofs = wallet.mint(&mint_quote.id, SplitTarget::None, None).await?;
- let active_keyset_id = wallet.get_active_mint_keyset().await?.id;
- let pre_mint = PreMintSecrets::random(active_keyset_id, 100.into(), &SplitTarget::None)?;
- let swap_request = SwapRequest {
- inputs: proofs.clone(),
- outputs: pre_mint.blinded_messages(),
- };
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let response = http_client.post_swap(swap_request.clone()).await;
- assert!(response.is_ok());
- let pre_mint = PreMintSecrets::random(active_keyset_id, 101.into(), &SplitTarget::None)?;
- let swap_request = SwapRequest {
- inputs: proofs.clone(),
- outputs: pre_mint.blinded_messages(),
- };
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let response = http_client.post_swap(swap_request.clone()).await;
- match response {
- Err(err) => match err {
- cdk::Error::TokenAlreadySpent => (),
- err => {
- bail!(
- "Wrong mint error returned expected already spent: {}",
- err.to_string()
- );
- }
- },
- Ok(_) => {
- bail!("Should not have allowed swap with unbalanced");
- }
- }
- let pre_mint = PreMintSecrets::random(active_keyset_id, 100.into(), &SplitTarget::None)?;
- let swap_request = SwapRequest {
- inputs: proofs,
- outputs: pre_mint.blinded_messages(),
- };
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let response = http_client.post_swap(swap_request.clone()).await;
- match response {
- Err(err) => match err {
- cdk::Error::TokenAlreadySpent => (),
- err => {
- bail!("Wrong mint error returned: {}", err.to_string());
- }
- },
- Ok(_) => {
- bail!("Should not have allowed to mint with multiple units");
- }
- }
- Ok(())
- }
- /// Test swap after failure
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_mint_melt_spend_after_fail() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(WalletMemoryDatabase::default()),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let proofs = wallet.mint(&mint_quote.id, SplitTarget::None, None).await?;
- let active_keyset_id = wallet.get_active_mint_keyset().await?.id;
- let pre_mint = PreMintSecrets::random(active_keyset_id, 100.into(), &SplitTarget::None)?;
- let swap_request = SwapRequest {
- inputs: proofs.clone(),
- outputs: pre_mint.blinded_messages(),
- };
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let response = http_client.post_swap(swap_request.clone()).await;
- assert!(response.is_ok());
- let pre_mint = PreMintSecrets::random(active_keyset_id, 101.into(), &SplitTarget::None)?;
- let swap_request = SwapRequest {
- inputs: proofs.clone(),
- outputs: pre_mint.blinded_messages(),
- };
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let response = http_client.post_swap(swap_request.clone()).await;
- match response {
- Err(err) => match err {
- cdk::Error::TokenAlreadySpent => (),
- err => {
- bail!("Wrong mint error returned: {}", err.to_string());
- }
- },
- Ok(_) => {
- bail!("Should not have allowed to mint with multiple units");
- }
- }
- let input_amount: u64 = proofs.total_amount()?.into();
- let invoice = create_fake_invoice((input_amount - 1) * 1000, "".to_string());
- let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
- let melt_request = MeltBolt11Request {
- quote: melt_quote.id,
- inputs: proofs,
- outputs: None,
- };
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let response = http_client.post_melt(melt_request.clone()).await;
- match response {
- Err(err) => match err {
- cdk::Error::TokenAlreadySpent => (),
- err => {
- bail!("Wrong mint error returned: {}", err.to_string());
- }
- },
- Ok(_) => {
- bail!("Should not have allowed to melt with multiple units");
- }
- }
- Ok(())
- }
- /// Test swap where input unit != output unit
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_mint_duplicate_proofs_swap() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let proofs = wallet.mint(&mint_quote.id, SplitTarget::None, None).await?;
- let active_keyset_id = wallet.get_active_mint_keyset().await?.id;
- let inputs = vec![proofs[0].clone(), proofs[0].clone()];
- let pre_mint =
- PreMintSecrets::random(active_keyset_id, inputs.total_amount()?, &SplitTarget::None)?;
- let swap_request = SwapRequest {
- inputs: inputs.clone(),
- outputs: pre_mint.blinded_messages(),
- };
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let response = http_client.post_swap(swap_request.clone()).await;
- match response {
- Err(err) => match err {
- cdk::Error::DuplicateInputs => (),
- err => {
- bail!(
- "Wrong mint error returned, expected duplicate inputs: {}",
- err.to_string()
- );
- }
- },
- Ok(_) => {
- bail!("Should not have allowed duplicate inputs");
- }
- }
- let blinded_message = pre_mint.blinded_messages();
- let outputs = vec![blinded_message[0].clone(), blinded_message[0].clone()];
- let swap_request = SwapRequest { inputs, outputs };
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let response = http_client.post_swap(swap_request.clone()).await;
- match response {
- Err(err) => match err {
- cdk::Error::DuplicateOutputs => (),
- err => {
- bail!(
- "Wrong mint error returned, expected duplicate outputs: {}",
- err.to_string()
- );
- }
- },
- Ok(_) => {
- bail!("Should not have allow duplicate inputs");
- }
- }
- Ok(())
- }
- /// Test duplicate proofs in melt
- #[tokio::test(flavor = "multi_thread", worker_threads = 1)]
- async fn test_fake_mint_duplicate_proofs_melt() -> Result<()> {
- let wallet = Wallet::new(
- MINT_URL,
- CurrencyUnit::Sat,
- Arc::new(memory::empty().await?),
- &Mnemonic::generate(12)?.to_seed_normalized(""),
- None,
- )?;
- let mint_quote = wallet.mint_quote(100.into(), None).await?;
- wait_for_mint_to_be_paid(&wallet, &mint_quote.id, 60).await?;
- let proofs = wallet.mint(&mint_quote.id, SplitTarget::None, None).await?;
- let inputs = vec![proofs[0].clone(), proofs[0].clone()];
- let invoice = create_fake_invoice(7000, "".to_string());
- let melt_quote = wallet.melt_quote(invoice.to_string(), None).await?;
- let melt_request = MeltBolt11Request {
- quote: melt_quote.id,
- inputs,
- outputs: None,
- };
- let http_client = HttpClient::new(MINT_URL.parse()?);
- let response = http_client.post_melt(melt_request.clone()).await;
- match response {
- Err(err) => match err {
- cdk::Error::DuplicateInputs => (),
- err => {
- bail!("Wrong mint error returned: {}", err.to_string());
- }
- },
- Ok(_) => {
- bail!("Should not have allow duplicate inputs");
- }
- }
- Ok(())
- }
|