mirror of
https://github.com/aljazceru/cdk.git
synced 2026-01-07 15:05:34 +01:00
The main goal is to add a subscription to CDK Mint updates into the wallet. This feature will be particularly useful for improving the code whenever loops hit the mint server to check status changes. The goal is to add an easy-to-use interface that will hide the fact that we're connecting to WebSocket and subscribing to events. This will also hide the fact that the CDK-mint server may not support WebSocket updates. To be fully backward compatible, the HttpClientMethods traits have a new method, `subscribe,` which will return an object that implements `ActiveSubscription.` In the primary implementation, there is a `SubscriptionClient` that will attempt to connect through WebSocket and will fall to the HTTP-status pull and sleep approach (the current approach), but upper stream code will receive updates as if they come from a stream of updates through WebSocket. This `SubscriptionClient` struct will also manage reconnections to WebSockets (with automatic resubscriptions) and all the low-level stuff, providing an easy-to-use interface and leaving the upper-level code with a nice interface that is hard to misuse. When `ActiveSubscription` is dropped, it will automatically unsubscribe. Fixed bug with Default as described in https://github.com/cashubtc/cdk/pull/473#discussion_r1871032297
233 lines
7.4 KiB
Rust
233 lines
7.4 KiB
Rust
use std::fs;
|
|
use std::path::PathBuf;
|
|
use std::str::FromStr;
|
|
use std::sync::Arc;
|
|
|
|
use anyhow::{bail, Result};
|
|
use bip39::Mnemonic;
|
|
use cdk::cdk_database;
|
|
use cdk::cdk_database::WalletDatabase;
|
|
use cdk::wallet::client::HttpClient;
|
|
use cdk::wallet::{MultiMintWallet, Wallet};
|
|
use cdk_redb::WalletRedbDatabase;
|
|
use cdk_sqlite::WalletSqliteDatabase;
|
|
use clap::{Parser, Subcommand};
|
|
use rand::Rng;
|
|
use tracing::Level;
|
|
use tracing_subscriber::EnvFilter;
|
|
use url::Url;
|
|
|
|
mod sub_commands;
|
|
|
|
const DEFAULT_WORK_DIR: &str = ".cdk-cli";
|
|
|
|
/// Simple CLI application to interact with cashu
|
|
#[derive(Parser)]
|
|
#[command(name = "cashu-tool")]
|
|
#[command(author = "thesimplekid <tsk@thesimplekid.com>")]
|
|
#[command(version = "0.1.0")]
|
|
#[command(author, version, about, long_about = None)]
|
|
struct Cli {
|
|
/// Database engine to use (sqlite/redb)
|
|
#[arg(short, long, default_value = "sqlite")]
|
|
engine: String,
|
|
/// Path to working dir
|
|
#[arg(short, long)]
|
|
work_dir: Option<PathBuf>,
|
|
/// Logging level
|
|
#[arg(short, long, default_value = "error")]
|
|
log_level: Level,
|
|
/// NWS Proxy
|
|
#[arg(short, long)]
|
|
proxy: Option<Url>,
|
|
#[command(subcommand)]
|
|
command: Commands,
|
|
}
|
|
|
|
#[derive(Subcommand)]
|
|
enum Commands {
|
|
/// Decode a token
|
|
DecodeToken(sub_commands::decode_token::DecodeTokenSubCommand),
|
|
/// Balance
|
|
Balance,
|
|
/// Pay bolt11 invoice
|
|
Melt(sub_commands::melt::MeltSubCommand),
|
|
/// Claim pending mint quotes that have been paid
|
|
MintPending,
|
|
/// Receive token
|
|
Receive(sub_commands::receive::ReceiveSubCommand),
|
|
/// Send
|
|
Send(sub_commands::send::SendSubCommand),
|
|
/// Check if wallet balance is spendable
|
|
CheckSpendable,
|
|
/// View mint info
|
|
MintInfo(sub_commands::mint_info::MintInfoSubcommand),
|
|
/// Mint proofs via bolt11
|
|
Mint(sub_commands::mint::MintSubCommand),
|
|
/// Burn Spent tokens
|
|
Burn(sub_commands::burn::BurnSubCommand),
|
|
/// Restore proofs from seed
|
|
Restore(sub_commands::restore::RestoreSubCommand),
|
|
/// Update Mint Url
|
|
UpdateMintUrl(sub_commands::update_mint_url::UpdateMintUrlSubCommand),
|
|
/// Get proofs from mint.
|
|
ListMintProofs,
|
|
/// Decode a payment request
|
|
DecodeRequest(sub_commands::decode_request::DecodePaymentRequestSubCommand),
|
|
/// Pay a payment request
|
|
PayRequest(sub_commands::pay_request::PayRequestSubCommand),
|
|
/// Create Payment request
|
|
CreateRequest(sub_commands::create_request::CreateRequestSubCommand),
|
|
}
|
|
|
|
#[tokio::main]
|
|
async fn main() -> Result<()> {
|
|
let args: Cli = Cli::parse();
|
|
let default_filter = args.log_level;
|
|
|
|
let sqlx_filter = "sqlx=warn,hyper_util=warn,reqwest=warn";
|
|
|
|
let env_filter = EnvFilter::new(format!("{},{}", default_filter, sqlx_filter));
|
|
|
|
// Parse input
|
|
tracing_subscriber::fmt().with_env_filter(env_filter).init();
|
|
|
|
let work_dir = match &args.work_dir {
|
|
Some(work_dir) => work_dir.clone(),
|
|
None => {
|
|
let home_dir = home::home_dir().unwrap();
|
|
home_dir.join(DEFAULT_WORK_DIR)
|
|
}
|
|
};
|
|
|
|
fs::create_dir_all(&work_dir)?;
|
|
|
|
let localstore: Arc<dyn WalletDatabase<Err = cdk_database::Error> + Send + Sync> =
|
|
match args.engine.as_str() {
|
|
"sqlite" => {
|
|
let sql_path = work_dir.join("cdk-cli.sqlite");
|
|
let sql = WalletSqliteDatabase::new(&sql_path).await?;
|
|
|
|
sql.migrate().await;
|
|
|
|
Arc::new(sql)
|
|
}
|
|
"redb" => {
|
|
let redb_path = work_dir.join("cdk-cli.redb");
|
|
|
|
Arc::new(WalletRedbDatabase::new(&redb_path)?)
|
|
}
|
|
_ => bail!("Unknown DB engine"),
|
|
};
|
|
|
|
let seed_path = work_dir.join("seed");
|
|
|
|
let mnemonic = match fs::metadata(seed_path.clone()) {
|
|
Ok(_) => {
|
|
let contents = fs::read_to_string(seed_path.clone())?;
|
|
Mnemonic::from_str(&contents)?
|
|
}
|
|
Err(_e) => {
|
|
let mut rng = rand::thread_rng();
|
|
let random_bytes: [u8; 32] = rng.gen();
|
|
|
|
let mnemonic = Mnemonic::from_entropy(&random_bytes)?;
|
|
tracing::info!("Creating new seed");
|
|
|
|
fs::write(seed_path, mnemonic.to_string())?;
|
|
|
|
mnemonic
|
|
}
|
|
};
|
|
|
|
let mut wallets: Vec<Wallet> = Vec::new();
|
|
|
|
let mints = localstore.get_mints().await?;
|
|
|
|
for (mint_url, _) in mints {
|
|
let mut wallet = Wallet::new(
|
|
&mint_url.to_string(),
|
|
cdk::nuts::CurrencyUnit::Sat,
|
|
localstore.clone(),
|
|
&mnemonic.to_seed_normalized(""),
|
|
None,
|
|
)?;
|
|
if let Some(proxy_url) = args.proxy.as_ref() {
|
|
let http_client = HttpClient::with_proxy(mint_url, proxy_url.clone(), None, true)?;
|
|
wallet.set_client(http_client);
|
|
}
|
|
|
|
wallets.push(wallet);
|
|
}
|
|
|
|
let multi_mint_wallet = MultiMintWallet::new(wallets);
|
|
|
|
match &args.command {
|
|
Commands::DecodeToken(sub_command_args) => {
|
|
sub_commands::decode_token::decode_token(sub_command_args)
|
|
}
|
|
Commands::Balance => sub_commands::balance::balance(&multi_mint_wallet).await,
|
|
Commands::Melt(sub_command_args) => {
|
|
sub_commands::melt::pay(&multi_mint_wallet, sub_command_args).await
|
|
}
|
|
Commands::Receive(sub_command_args) => {
|
|
sub_commands::receive::receive(
|
|
&multi_mint_wallet,
|
|
localstore,
|
|
&mnemonic.to_seed_normalized(""),
|
|
sub_command_args,
|
|
)
|
|
.await
|
|
}
|
|
Commands::Send(sub_command_args) => {
|
|
sub_commands::send::send(&multi_mint_wallet, sub_command_args).await
|
|
}
|
|
Commands::CheckSpendable => {
|
|
sub_commands::check_spent::check_spent(&multi_mint_wallet).await
|
|
}
|
|
Commands::MintInfo(sub_command_args) => {
|
|
sub_commands::mint_info::mint_info(args.proxy, sub_command_args).await
|
|
}
|
|
Commands::Mint(sub_command_args) => {
|
|
sub_commands::mint::mint(
|
|
&multi_mint_wallet,
|
|
&mnemonic.to_seed_normalized(""),
|
|
localstore,
|
|
sub_command_args,
|
|
)
|
|
.await
|
|
}
|
|
Commands::MintPending => {
|
|
sub_commands::pending_mints::mint_pending(&multi_mint_wallet).await
|
|
}
|
|
Commands::Burn(sub_command_args) => {
|
|
sub_commands::burn::burn(&multi_mint_wallet, sub_command_args).await
|
|
}
|
|
Commands::Restore(sub_command_args) => {
|
|
sub_commands::restore::restore(
|
|
&multi_mint_wallet,
|
|
&mnemonic.to_seed_normalized(""),
|
|
localstore,
|
|
sub_command_args,
|
|
)
|
|
.await
|
|
}
|
|
Commands::UpdateMintUrl(sub_command_args) => {
|
|
sub_commands::update_mint_url::update_mint_url(&multi_mint_wallet, sub_command_args)
|
|
.await
|
|
}
|
|
Commands::ListMintProofs => {
|
|
sub_commands::list_mint_proofs::proofs(&multi_mint_wallet).await
|
|
}
|
|
Commands::DecodeRequest(sub_command_args) => {
|
|
sub_commands::decode_request::decode_payment_request(sub_command_args)
|
|
}
|
|
Commands::PayRequest(sub_command_args) => {
|
|
sub_commands::pay_request::pay_request(&multi_mint_wallet, sub_command_args).await
|
|
}
|
|
Commands::CreateRequest(sub_command_args) => {
|
|
sub_commands::create_request::create_request(&multi_mint_wallet, sub_command_args).await
|
|
}
|
|
}
|
|
}
|