main.rs 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232
  1. use std::fs;
  2. use std::path::PathBuf;
  3. use std::str::FromStr;
  4. use std::sync::Arc;
  5. use anyhow::{bail, Result};
  6. use bip39::Mnemonic;
  7. use cdk::cdk_database;
  8. use cdk::cdk_database::WalletDatabase;
  9. use cdk::wallet::client::HttpClient;
  10. use cdk::wallet::{MultiMintWallet, Wallet};
  11. use cdk_redb::WalletRedbDatabase;
  12. use cdk_sqlite::WalletSqliteDatabase;
  13. use clap::{Parser, Subcommand};
  14. use rand::Rng;
  15. use tracing::Level;
  16. use tracing_subscriber::EnvFilter;
  17. use url::Url;
  18. mod sub_commands;
  19. const DEFAULT_WORK_DIR: &str = ".cdk-cli";
  20. /// Simple CLI application to interact with cashu
  21. #[derive(Parser)]
  22. #[command(name = "cashu-tool")]
  23. #[command(author = "thesimplekid <tsk@thesimplekid.com>")]
  24. #[command(version = "0.1.0")]
  25. #[command(author, version, about, long_about = None)]
  26. struct Cli {
  27. /// Database engine to use (sqlite/redb)
  28. #[arg(short, long, default_value = "sqlite")]
  29. engine: String,
  30. /// Path to working dir
  31. #[arg(short, long)]
  32. work_dir: Option<PathBuf>,
  33. /// Logging level
  34. #[arg(short, long, default_value = "error")]
  35. log_level: Level,
  36. /// NWS Proxy
  37. #[arg(short, long)]
  38. proxy: Option<Url>,
  39. #[command(subcommand)]
  40. command: Commands,
  41. }
  42. #[derive(Subcommand)]
  43. enum Commands {
  44. /// Decode a token
  45. DecodeToken(sub_commands::decode_token::DecodeTokenSubCommand),
  46. /// Balance
  47. Balance,
  48. /// Pay bolt11 invoice
  49. Melt(sub_commands::melt::MeltSubCommand),
  50. /// Claim pending mint quotes that have been paid
  51. MintPending,
  52. /// Receive token
  53. Receive(sub_commands::receive::ReceiveSubCommand),
  54. /// Send
  55. Send(sub_commands::send::SendSubCommand),
  56. /// Check if wallet balance is spendable
  57. CheckSpendable,
  58. /// View mint info
  59. MintInfo(sub_commands::mint_info::MintInfoSubcommand),
  60. /// Mint proofs via bolt11
  61. Mint(sub_commands::mint::MintSubCommand),
  62. /// Burn Spent tokens
  63. Burn(sub_commands::burn::BurnSubCommand),
  64. /// Restore proofs from seed
  65. Restore(sub_commands::restore::RestoreSubCommand),
  66. /// Update Mint Url
  67. UpdateMintUrl(sub_commands::update_mint_url::UpdateMintUrlSubCommand),
  68. /// Get proofs from mint.
  69. ListMintProofs,
  70. /// Decode a payment request
  71. DecodeRequest(sub_commands::decode_request::DecodePaymentRequestSubCommand),
  72. /// Pay a payment request
  73. PayRequest(sub_commands::pay_request::PayRequestSubCommand),
  74. /// Create Payment request
  75. CreateRequest(sub_commands::create_request::CreateRequestSubCommand),
  76. }
  77. #[tokio::main]
  78. async fn main() -> Result<()> {
  79. let args: Cli = Cli::parse();
  80. let default_filter = args.log_level;
  81. let sqlx_filter = "sqlx=warn,hyper_util=warn,reqwest=warn";
  82. let env_filter = EnvFilter::new(format!("{},{}", default_filter, sqlx_filter));
  83. // Parse input
  84. tracing_subscriber::fmt().with_env_filter(env_filter).init();
  85. let work_dir = match &args.work_dir {
  86. Some(work_dir) => work_dir.clone(),
  87. None => {
  88. let home_dir = home::home_dir().unwrap();
  89. home_dir.join(DEFAULT_WORK_DIR)
  90. }
  91. };
  92. fs::create_dir_all(&work_dir)?;
  93. let localstore: Arc<dyn WalletDatabase<Err = cdk_database::Error> + Send + Sync> =
  94. match args.engine.as_str() {
  95. "sqlite" => {
  96. let sql_path = work_dir.join("cdk-cli.sqlite");
  97. let sql = WalletSqliteDatabase::new(&sql_path).await?;
  98. sql.migrate().await;
  99. Arc::new(sql)
  100. }
  101. "redb" => {
  102. let redb_path = work_dir.join("cdk-cli.redb");
  103. Arc::new(WalletRedbDatabase::new(&redb_path)?)
  104. }
  105. _ => bail!("Unknown DB engine"),
  106. };
  107. let seed_path = work_dir.join("seed");
  108. let mnemonic = match fs::metadata(seed_path.clone()) {
  109. Ok(_) => {
  110. let contents = fs::read_to_string(seed_path.clone())?;
  111. Mnemonic::from_str(&contents)?
  112. }
  113. Err(_e) => {
  114. let mut rng = rand::thread_rng();
  115. let random_bytes: [u8; 32] = rng.gen();
  116. let mnemonic = Mnemonic::from_entropy(&random_bytes)?;
  117. tracing::info!("Creating new seed");
  118. fs::write(seed_path, mnemonic.to_string())?;
  119. mnemonic
  120. }
  121. };
  122. let mut wallets: Vec<Wallet> = Vec::new();
  123. let mints = localstore.get_mints().await?;
  124. for (mint_url, _) in mints {
  125. let mut wallet = Wallet::new(
  126. &mint_url.to_string(),
  127. cdk::nuts::CurrencyUnit::Sat,
  128. localstore.clone(),
  129. &mnemonic.to_seed_normalized(""),
  130. None,
  131. )?;
  132. if let Some(proxy_url) = args.proxy.as_ref() {
  133. let http_client = HttpClient::with_proxy(mint_url, proxy_url.clone(), None, true)?;
  134. wallet.set_client(http_client);
  135. }
  136. wallets.push(wallet);
  137. }
  138. let multi_mint_wallet = MultiMintWallet::new(wallets);
  139. match &args.command {
  140. Commands::DecodeToken(sub_command_args) => {
  141. sub_commands::decode_token::decode_token(sub_command_args)
  142. }
  143. Commands::Balance => sub_commands::balance::balance(&multi_mint_wallet).await,
  144. Commands::Melt(sub_command_args) => {
  145. sub_commands::melt::pay(&multi_mint_wallet, sub_command_args).await
  146. }
  147. Commands::Receive(sub_command_args) => {
  148. sub_commands::receive::receive(
  149. &multi_mint_wallet,
  150. localstore,
  151. &mnemonic.to_seed_normalized(""),
  152. sub_command_args,
  153. )
  154. .await
  155. }
  156. Commands::Send(sub_command_args) => {
  157. sub_commands::send::send(&multi_mint_wallet, sub_command_args).await
  158. }
  159. Commands::CheckSpendable => {
  160. sub_commands::check_spent::check_spent(&multi_mint_wallet).await
  161. }
  162. Commands::MintInfo(sub_command_args) => {
  163. sub_commands::mint_info::mint_info(args.proxy, sub_command_args).await
  164. }
  165. Commands::Mint(sub_command_args) => {
  166. sub_commands::mint::mint(
  167. &multi_mint_wallet,
  168. &mnemonic.to_seed_normalized(""),
  169. localstore,
  170. sub_command_args,
  171. )
  172. .await
  173. }
  174. Commands::MintPending => {
  175. sub_commands::pending_mints::mint_pending(&multi_mint_wallet).await
  176. }
  177. Commands::Burn(sub_command_args) => {
  178. sub_commands::burn::burn(&multi_mint_wallet, sub_command_args).await
  179. }
  180. Commands::Restore(sub_command_args) => {
  181. sub_commands::restore::restore(
  182. &multi_mint_wallet,
  183. &mnemonic.to_seed_normalized(""),
  184. localstore,
  185. sub_command_args,
  186. )
  187. .await
  188. }
  189. Commands::UpdateMintUrl(sub_command_args) => {
  190. sub_commands::update_mint_url::update_mint_url(&multi_mint_wallet, sub_command_args)
  191. .await
  192. }
  193. Commands::ListMintProofs => {
  194. sub_commands::list_mint_proofs::proofs(&multi_mint_wallet).await
  195. }
  196. Commands::DecodeRequest(sub_command_args) => {
  197. sub_commands::decode_request::decode_payment_request(sub_command_args)
  198. }
  199. Commands::PayRequest(sub_command_args) => {
  200. sub_commands::pay_request::pay_request(&multi_mint_wallet, sub_command_args).await
  201. }
  202. Commands::CreateRequest(sub_command_args) => {
  203. sub_commands::create_request::create_request(&multi_mint_wallet, sub_command_args).await
  204. }
  205. }
  206. }