init_pure_tests.rs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344
  1. use std::collections::{HashMap, HashSet};
  2. use std::fmt::{Debug, Formatter};
  3. use std::path::PathBuf;
  4. use std::str::FromStr;
  5. use std::sync::Arc;
  6. use std::{env, fs};
  7. use anyhow::{anyhow, bail, Result};
  8. use async_trait::async_trait;
  9. use bip39::Mnemonic;
  10. use cdk::amount::SplitTarget;
  11. use cdk::cdk_database::{self, MintDatabase, WalletDatabase};
  12. use cdk::mint::{MintBuilder, MintMeltLimits};
  13. use cdk::nuts::nut00::ProofsMethods;
  14. use cdk::nuts::{
  15. CheckStateRequest, CheckStateResponse, CurrencyUnit, Id, KeySet, KeysetResponse,
  16. MeltBolt11Request, MeltQuoteBolt11Request, MeltQuoteBolt11Response, MintBolt11Request,
  17. MintBolt11Response, MintInfo, MintQuoteBolt11Request, MintQuoteBolt11Response, PaymentMethod,
  18. RestoreRequest, RestoreResponse, SwapRequest, SwapResponse,
  19. };
  20. use cdk::types::{FeeReserve, QuoteTTL};
  21. use cdk::util::unix_time;
  22. use cdk::wallet::{AuthWallet, MintConnector, Wallet, WalletBuilder};
  23. use cdk::{Amount, Error, Mint};
  24. use cdk_fake_wallet::FakeWallet;
  25. use tokio::sync::{Notify, RwLock};
  26. use tracing_subscriber::EnvFilter;
  27. use uuid::Uuid;
  28. use crate::wait_for_mint_to_be_paid;
  29. pub struct DirectMintConnection {
  30. pub mint: Mint,
  31. auth_wallet: Arc<RwLock<Option<AuthWallet>>>,
  32. }
  33. impl DirectMintConnection {
  34. pub fn new(mint: Mint) -> Self {
  35. Self {
  36. mint,
  37. auth_wallet: Arc::new(RwLock::new(None)),
  38. }
  39. }
  40. }
  41. impl Debug for DirectMintConnection {
  42. fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
  43. write!(f, "DirectMintConnection",)
  44. }
  45. }
  46. /// Implements the generic [MintConnector] (i.e. use the interface that expects to communicate
  47. /// to a generic mint, where we don't know that quote ID's are [Uuid]s) for [DirectMintConnection],
  48. /// where we know we're dealing with a mint that uses [Uuid]s for quotes.
  49. /// Convert the requests and responses between the [String] and [Uuid] variants as necessary.
  50. #[async_trait]
  51. impl MintConnector for DirectMintConnection {
  52. async fn get_mint_keys(&self) -> Result<Vec<KeySet>, Error> {
  53. self.mint.pubkeys().await.map(|pks| pks.keysets)
  54. }
  55. async fn get_mint_keyset(&self, keyset_id: Id) -> Result<KeySet, Error> {
  56. self.mint
  57. .keyset(&keyset_id)
  58. .await
  59. .and_then(|res| res.ok_or(Error::UnknownKeySet))
  60. }
  61. async fn get_mint_keysets(&self) -> Result<KeysetResponse, Error> {
  62. self.mint.keysets().await
  63. }
  64. async fn post_mint_quote(
  65. &self,
  66. request: MintQuoteBolt11Request,
  67. ) -> Result<MintQuoteBolt11Response<String>, Error> {
  68. self.mint
  69. .get_mint_bolt11_quote(request)
  70. .await
  71. .map(Into::into)
  72. }
  73. async fn get_mint_quote_status(
  74. &self,
  75. quote_id: &str,
  76. ) -> Result<MintQuoteBolt11Response<String>, Error> {
  77. let quote_id_uuid = Uuid::from_str(quote_id).unwrap();
  78. self.mint
  79. .check_mint_quote(&quote_id_uuid)
  80. .await
  81. .map(Into::into)
  82. }
  83. async fn post_mint(
  84. &self,
  85. request: MintBolt11Request<String>,
  86. ) -> Result<MintBolt11Response, Error> {
  87. let request_uuid = request.try_into().unwrap();
  88. self.mint.process_mint_request(request_uuid).await
  89. }
  90. async fn post_melt_quote(
  91. &self,
  92. request: MeltQuoteBolt11Request,
  93. ) -> Result<MeltQuoteBolt11Response<String>, Error> {
  94. self.mint
  95. .get_melt_bolt11_quote(&request)
  96. .await
  97. .map(Into::into)
  98. }
  99. async fn get_melt_quote_status(
  100. &self,
  101. quote_id: &str,
  102. ) -> Result<MeltQuoteBolt11Response<String>, Error> {
  103. let quote_id_uuid = Uuid::from_str(quote_id).unwrap();
  104. self.mint
  105. .check_melt_quote(&quote_id_uuid)
  106. .await
  107. .map(Into::into)
  108. }
  109. async fn post_melt(
  110. &self,
  111. request: MeltBolt11Request<String>,
  112. ) -> Result<MeltQuoteBolt11Response<String>, Error> {
  113. let request_uuid = request.try_into().unwrap();
  114. self.mint.melt_bolt11(&request_uuid).await.map(Into::into)
  115. }
  116. async fn post_swap(&self, swap_request: SwapRequest) -> Result<SwapResponse, Error> {
  117. self.mint.process_swap_request(swap_request).await
  118. }
  119. async fn get_mint_info(&self) -> Result<MintInfo, Error> {
  120. Ok(self.mint.mint_info().await?.clone().time(unix_time()))
  121. }
  122. async fn post_check_state(
  123. &self,
  124. request: CheckStateRequest,
  125. ) -> Result<CheckStateResponse, Error> {
  126. self.mint.check_state(&request).await
  127. }
  128. async fn post_restore(&self, request: RestoreRequest) -> Result<RestoreResponse, Error> {
  129. self.mint.restore(request).await
  130. }
  131. /// Get the auth wallet for the client
  132. async fn get_auth_wallet(&self) -> Option<AuthWallet> {
  133. self.auth_wallet.read().await.clone()
  134. }
  135. /// Set auth wallet on client
  136. async fn set_auth_wallet(&self, wallet: Option<AuthWallet>) {
  137. let mut auth_wallet = self.auth_wallet.write().await;
  138. *auth_wallet = wallet;
  139. }
  140. }
  141. pub fn setup_tracing() {
  142. let default_filter = "debug";
  143. let sqlx_filter = "sqlx=warn";
  144. let hyper_filter = "hyper=warn";
  145. let env_filter = EnvFilter::new(format!(
  146. "{},{},{}",
  147. default_filter, sqlx_filter, hyper_filter
  148. ));
  149. // Ok if successful, Err if already initialized
  150. // Allows us to setup tracing at the start of several parallel tests
  151. let _ = tracing_subscriber::fmt()
  152. .with_env_filter(env_filter)
  153. .try_init();
  154. }
  155. pub async fn create_and_start_test_mint() -> Result<Mint> {
  156. let mut mint_builder = MintBuilder::new();
  157. // Read environment variable to determine database type
  158. let db_type = env::var("CDK_TEST_DB_TYPE").expect("Database type set");
  159. let localstore: Arc<dyn MintDatabase<cdk_database::Error> + Send + Sync> =
  160. match db_type.to_lowercase().as_str() {
  161. "sqlite" => {
  162. // Create a temporary directory for SQLite database
  163. let temp_dir = create_temp_dir("cdk-test-sqlite-mint")?;
  164. let path = temp_dir.join("mint.db").to_str().unwrap().to_string();
  165. let database = cdk_sqlite::MintSqliteDatabase::new(&path)
  166. .await
  167. .expect("Could not create sqlite db");
  168. Arc::new(database)
  169. }
  170. "redb" => {
  171. // Create a temporary directory for ReDB database
  172. let temp_dir = create_temp_dir("cdk-test-redb-mint")?;
  173. let path = temp_dir.join("mint.redb");
  174. let database = cdk_redb::MintRedbDatabase::new(&path)
  175. .expect("Could not create redb mint database");
  176. Arc::new(database)
  177. }
  178. "memory" => {
  179. let database = cdk_sqlite::mint::memory::empty().await?;
  180. Arc::new(database)
  181. }
  182. _ => {
  183. bail!("Db type not set")
  184. }
  185. };
  186. mint_builder = mint_builder.with_localstore(localstore.clone());
  187. let fee_reserve = FeeReserve {
  188. min_fee_reserve: 1.into(),
  189. percent_fee_reserve: 1.0,
  190. };
  191. let ln_fake_backend = FakeWallet::new(
  192. fee_reserve.clone(),
  193. HashMap::default(),
  194. HashSet::default(),
  195. 0,
  196. );
  197. mint_builder = mint_builder
  198. .add_ln_backend(
  199. CurrencyUnit::Sat,
  200. PaymentMethod::Bolt11,
  201. MintMeltLimits::new(1, 10_000),
  202. Arc::new(ln_fake_backend),
  203. )
  204. .await?;
  205. let mnemonic = Mnemonic::generate(12)?;
  206. mint_builder = mint_builder
  207. .with_name("pure test mint".to_string())
  208. .with_description("pure test mint".to_string())
  209. .with_urls(vec!["https://aaa".to_string()])
  210. .with_seed(mnemonic.to_seed_normalized("").to_vec());
  211. localstore
  212. .set_mint_info(mint_builder.mint_info.clone())
  213. .await?;
  214. let quote_ttl = QuoteTTL::new(10000, 10000);
  215. localstore.set_quote_ttl(quote_ttl).await?;
  216. let mint = mint_builder.build().await?;
  217. let mint_clone = mint.clone();
  218. let shutdown = Arc::new(Notify::new());
  219. tokio::spawn({
  220. let shutdown = Arc::clone(&shutdown);
  221. async move { mint_clone.wait_for_paid_invoices(shutdown).await }
  222. });
  223. Ok(mint)
  224. }
  225. pub async fn create_test_wallet_for_mint(mint: Mint) -> Result<Wallet> {
  226. let connector = DirectMintConnection::new(mint.clone());
  227. let mint_info = mint.mint_info().await?;
  228. let mint_url = mint_info
  229. .urls
  230. .as_ref()
  231. .ok_or(anyhow!("Test mint URLs list is unset"))?
  232. .first()
  233. .ok_or(anyhow!("Test mint has empty URLs list"))?;
  234. let seed = Mnemonic::generate(12)?.to_seed_normalized("");
  235. let unit = CurrencyUnit::Sat;
  236. // Read environment variable to determine database type
  237. let db_type = env::var("CDK_TEST_DB_TYPE").expect("Database type set");
  238. let localstore: Arc<dyn WalletDatabase<Err = cdk_database::Error> + Send + Sync> =
  239. match db_type.to_lowercase().as_str() {
  240. "sqlite" => {
  241. // Create a temporary directory for SQLite database
  242. let temp_dir = create_temp_dir("cdk-test-sqlite-wallet")?;
  243. let path = temp_dir.join("wallet.db").to_str().unwrap().to_string();
  244. let database = cdk_sqlite::WalletSqliteDatabase::new(&path)
  245. .await
  246. .expect("Could not create sqlite db");
  247. Arc::new(database)
  248. }
  249. "redb" => {
  250. // Create a temporary directory for ReDB database
  251. let temp_dir = create_temp_dir("cdk-test-redb-wallet")?;
  252. let path = temp_dir.join("wallet.redb");
  253. let database = cdk_redb::WalletRedbDatabase::new(&path)
  254. .expect("Could not create redb mint database");
  255. Arc::new(database)
  256. }
  257. "memory" => {
  258. let database = cdk_sqlite::wallet::memory::empty().await?;
  259. Arc::new(database)
  260. }
  261. _ => {
  262. bail!("Db type not set")
  263. }
  264. };
  265. let wallet = WalletBuilder::new()
  266. .mint_url(mint_url.parse().unwrap())
  267. .unit(unit)
  268. .localstore(localstore)
  269. .seed(&seed)
  270. .client(connector)
  271. .build()?;
  272. Ok(wallet)
  273. }
  274. /// Creates a mint quote for the given amount and checks its state in a loop. Returns when
  275. /// amount is minted.
  276. /// Creates a temporary directory with a unique name based on the prefix
  277. fn create_temp_dir(prefix: &str) -> Result<PathBuf> {
  278. let temp_dir = env::temp_dir();
  279. let unique_dir = temp_dir.join(format!("{}-{}", prefix, Uuid::new_v4()));
  280. fs::create_dir_all(&unique_dir)?;
  281. Ok(unique_dir)
  282. }
  283. pub async fn fund_wallet(
  284. wallet: Wallet,
  285. amount: u64,
  286. split_target: Option<SplitTarget>,
  287. ) -> Result<Amount> {
  288. let desired_amount = Amount::from(amount);
  289. let quote = wallet.mint_quote(desired_amount, None).await?;
  290. wait_for_mint_to_be_paid(&wallet, &quote.id, 60).await?;
  291. Ok(wallet
  292. .mint(&quote.id, split_target.unwrap_or_default(), None)
  293. .await?
  294. .total_amount()?)
  295. }