From 8c2566b27e5e78e6717a0f0c2c85a6787db51250 Mon Sep 17 00:00:00 2001 From: grumbach Date: Thu, 1 Feb 2024 16:44:46 +0900 Subject: [PATCH] feat: dag cli cleanup --- sn_cli/src/subcommands/wallet/audit.rs | 15 +- sn_cli/src/subcommands/wallet/helpers.rs | 23 +- sn_cli/src/subcommands/wallet/hot_wallet.rs | 9 +- sn_cli/src/subcommands/wallet/mod.rs | 828 +++----------------- sn_client/src/audit/mod.rs | 72 +- 5 files changed, 110 insertions(+), 837 deletions(-) diff --git a/sn_cli/src/subcommands/wallet/audit.rs b/sn_cli/src/subcommands/wallet/audit.rs index c190365fc1..5d1f1edd62 100644 --- a/sn_cli/src/subcommands/wallet/audit.rs +++ b/sn_cli/src/subcommands/wallet/audit.rs @@ -14,7 +14,7 @@ use sn_transfers::{SpendAddress, GENESIS_CASHNOTE}; const SPEND_DAG_FILENAME: &str = "spend_dag"; -pub(crate) async fn gather_spend_dag(client: &Client, root_dir: &Path) -> Result { +async fn gather_spend_dag(client: &Client, root_dir: &Path) -> Result { let dag_path = root_dir.join(SPEND_DAG_FILENAME); let dag = match SpendDag::load_from_file(&dag_path) { Ok(mut dag) => { @@ -33,3 +33,16 @@ pub(crate) async fn gather_spend_dag(client: &Client, root_dir: &Path) -> Result Ok(dag) } + +pub async fn audit(client: &Client, to_dot: bool, root_dir: &Path) -> Result<()> { + if to_dot { + let dag = gather_spend_dag(client, root_dir).await?; + println!("{}", dag.dump_dot_format()); + } else { + //NB TODO use the above DAG to audit too + println!("Auditing the Currency, note that this might take a very long time..."); + let genesis_addr = SpendAddress::from_unique_pubkey(&GENESIS_CASHNOTE.unique_pubkey()); + client.follow_spend(genesis_addr).await?; + } + Ok(()) +} diff --git a/sn_cli/src/subcommands/wallet/helpers.rs b/sn_cli/src/subcommands/wallet/helpers.rs index 9017eca504..91e5e902e3 100644 --- a/sn_cli/src/subcommands/wallet/helpers.rs +++ b/sn_cli/src/subcommands/wallet/helpers.rs @@ -8,31 +8,10 @@ use color_eyre::{eyre::eyre, Result}; use sn_client::Client; -use sn_transfers::{LocalWallet, SpendAddress, Transfer, UniquePubkey, GENESIS_CASHNOTE}; +use sn_transfers::{LocalWallet, SpendAddress, Transfer, UniquePubkey}; use std::path::Path; use url::Url; -pub async fn audit( - client: &Client, - to_dot: bool, - find_royalties: bool, - root_dir: &Path, -) -> Result<()> { - let genesis_addr = SpendAddress::from_unique_pubkey(&GENESIS_CASHNOTE.unique_pubkey()); - - if to_dot { - let dag = client.build_spend_dag_from(genesis_addr).await?; - println!("{}", dag.dump_dot_format()); - } else { - println!("Auditing the Currency, note that this might take a very long time..."); - client - .follow_spend(genesis_addr, find_royalties, root_dir) - .await?; - } - - Ok(()) -} - pub async fn get_faucet(root_dir: &Path, client: &Client, url: String) -> Result<()> { let wallet = LocalWallet::load_from(root_dir)?; let address_hex = wallet.address().to_hex(); diff --git a/sn_cli/src/subcommands/wallet/hot_wallet.rs b/sn_cli/src/subcommands/wallet/hot_wallet.rs index 5ece5f0935..35eac20406 100644 --- a/sn_cli/src/subcommands/wallet/hot_wallet.rs +++ b/sn_cli/src/subcommands/wallet/hot_wallet.rs @@ -7,7 +7,8 @@ // permissions and limitations relating to use of the SAFE Network Software. use super::{ - helpers::{audit, get_faucet, receive, verify_spend}, + audit::audit, + helpers::{get_faucet, receive, verify_spend}, WalletApiHelper, }; use crate::get_stdin_response; @@ -145,10 +146,6 @@ pub enum WalletCmds { /// EXPERIMENTAL Dump Audit DAG in dot format on stdout #[clap(long, default_value = "false")] dot: bool, - /// EXPERIMENTAL Find and redeem all Network Royalties - /// only works if the wallet has the Network Royalties private key - #[clap(long, default_value = "false")] - royalties: bool, }, } @@ -228,7 +225,7 @@ pub(crate) async fn wallet_cmds( WalletCmds::Send { amount, to } => send(amount, to, client, root_dir, verify_store).await, WalletCmds::Receive { file, transfer } => receive(transfer, file, client, root_dir).await, WalletCmds::GetFaucet { url } => get_faucet(root_dir, client, url.clone()).await, - WalletCmds::Audit { dot, royalties } => audit(client, dot, royalties, root_dir).await, + WalletCmds::Audit { dot } => audit(client, dot, root_dir).await, WalletCmds::Verify { spend_address, genesis, diff --git a/sn_cli/src/subcommands/wallet/mod.rs b/sn_cli/src/subcommands/wallet/mod.rs index ac48d4bd45..e3f9a99897 100644 --- a/sn_cli/src/subcommands/wallet/mod.rs +++ b/sn_cli/src/subcommands/wallet/mod.rs @@ -1,787 +1,139 @@ -// Copyright 2023 MaidSafe.net limited. +// Copyright 2024 MaidSafe.net limited. // // This SAFE Network Software is licensed to you under The General Public License (GPL), version 3. // Unless required by applicable law or agreed to in writing, the SAFE Network Software distributed // under the GPL Licence is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY // KIND, either express or implied. Please review the Licences for the specific language governing // permissions and limitations relating to use of the SAFE Network Software. +pub(crate) mod audit; +pub(crate) mod helpers; +pub(crate) mod hot_wallet; -mod audit; +use sn_transfers::{CashNote, LocalWallet, MainPubkey, NanoTokens, WatchOnlyWallet}; -use audit::gather_spend_dag; +use color_eyre::Result; +use std::{collections::BTreeSet, io::Read, path::Path}; -use crate::get_stdin_response; -use bls::{PublicKey, SecretKey, PK_SIZE}; -use clap::Parser; -use color_eyre::{ - eyre::{bail, eyre}, - Result, -}; -use sn_client::{Client, ClientEvent, Error as ClientError}; -use sn_transfers::{ - CashNoteRedemption, DerivationIndex, Error as TransferError, LocalWallet, MainPubkey, - MainSecretKey, NanoTokens, SignedSpend, SpendAddress, Transfer, UniquePubkey, UnsignedTransfer, - WalletError, WatchOnlyWallet, GENESIS_CASHNOTE, -}; -use std::{ - collections::{BTreeMap, BTreeSet}, - io::Read, - path::{Path, PathBuf}, - str::FromStr, -}; -use url::Url; - -const DEFAULT_RECEIVE_ONLINE_WALLET_DIR: &str = "receive_online"; -const ROYALTY_TRANSFER_NOTIF_TOPIC: &str = "ROYALTY_TRANSFER_NOTIFICATION"; - -// Please do not remove the blank lines in these doc comments. -// They are used for inserting line breaks when the help menu is rendered in the UI. -#[derive(Parser, Debug)] -pub enum WalletCmds { - /// Print the wallet address. - Address, - /// Print the wallet balance. - Balance { - /// Instead of checking CLI local wallet balance, the PeerId of a node can be used - /// to check the balance of its rewards local wallet. Multiple ids can be provided - /// in order to read the balance of multiple nodes at once. - #[clap(long)] - peer_id: Vec, - }, - /// DEPRECATED will be removed in future versions. - /// Prefer using the send and receive commands instead. - /// - /// Deposit CashNotes from the received directory to the local wallet. - /// Or Read a hex encoded CashNote from stdin. - /// - /// The default received directory is platform specific: - /// - Linux: $HOME/.local/share/safe/wallet/cash_notes - /// - macOS: $HOME/Library/Application Support/safe/wallet/cash_notes - /// - Windows: C:\Users\{username}\AppData\Roaming\safe\wallet\cash_notes - /// - /// If you find the default path unwieldy, you can also set the RECEIVED_CASHNOTES_PATH environment - /// variable to a path you would prefer to work with. - #[clap(verbatim_doc_comment)] - Deposit { - /// Read a hex encoded CashNote from stdin. - #[clap(long, default_value = "false")] - stdin: bool, - /// The hex encoded CashNote. - #[clap(long)] - cash_note: Option, - }, - /// Create a hot or watch-only wallet from the given (hex-encoded) key. - Create { - /// Hex-encoded main secret or public key. If the key is a secret key a hot-wallet will be created - /// which can be used to sign and broadcast transfers. Otherwise, if the passed key is a public key, - /// then a watch-only wallet is created. - #[clap(name = "key")] - key: String, - }, - /// Get tokens from a faucet. - GetFaucet { - /// The http url of the faucet to get tokens from. - #[clap(name = "url")] - url: String, - }, - /// Send a transfer. - /// - /// This command will create a new transfer and encrypt it for the recipient. - /// This encrypted transfer can then be shared with the recipient, who can then - /// use the 'receive' command to claim the funds. - Send { - /// The number of SafeNetworkTokens to send. - #[clap(name = "amount")] - amount: String, - /// Hex-encoded public address of the recipient. - #[clap(name = "to")] - to: String, - }, - /// Builds an unsigned transaction to be signed offline. - Transaction { - /// The number of SafeNetworkTokens to transfer. - #[clap(name = "amount")] - amount: String, - /// Hex-encoded public address of the recipient. - #[clap(name = "to")] - to: String, - }, - /// Signs a transaction to be then broadcasted to the network. - Sign { - /// Hex-encoded unsigned transaction. It requires a hot-wallet was created for CLI. - #[clap(name = "tx")] - tx: String, - }, - /// Broadcast a transaction that was signed offline. - /// - /// This command will create and encrypt the transfer for the recipient. - /// This encrypted transfer can then be shared with the recipient, who can then - /// use the 'receive' command to claim the funds. - Broadcast { - /// Hex-encoded signed transaction. - #[clap(name = "signed_tx")] - signed_tx: String, - }, - /// Receive a transfer created by the 'send' or 'broadcast' command. - Receive { - /// Read the encrypted transfer from a file. - #[clap(long, default_value = "false")] - file: bool, - /// Encrypted transfer. - #[clap(name = "transfer")] - transfer: String, - }, - /// Listen for transfer notifications from the network over gossipsub protocol. - /// - /// Transfers will be deposited to a local (watch-only) wallet. - /// - /// Only cash notes owned by the provided public key will be accepted, verified to be valid - /// against the network, and deposited onto a locally stored watch-only wallet. - ReceiveOnline { - /// Hex-encoded main public key - #[clap(name = "pk")] - pk: String, - /// Optional path where to store the wallet - #[clap(name = "path")] - path: Option, - }, - /// Verify a spend on the Network. - Verify { - /// The Network address or hex encoded UniquePubkey of the Spend to verify - #[clap(name = "spend")] - spend_address: String, - /// Verify all the way to Genesis - /// - /// Used for auditing, note that this might take a very long time - /// Analogous to verifying an UTXO through the entire blockchain in Bitcoin - #[clap(long, default_value = "false")] - genesis: bool, - }, - /// Audit the Currency - /// Note that this might take a very long time - /// Analogous to verifying the entire blockchain in Bitcoin - Audit { - /// EXPERIMENTAL Dump Audit DAG in dot format on stdout - #[clap(long, default_value = "false")] - dot: bool, - }, -} - -pub(crate) async fn wallet_cmds_without_client(cmds: &WalletCmds, root_dir: &Path) -> Result<()> { - match cmds { - WalletCmds::Address => address(root_dir), - WalletCmds::Balance { peer_id } => { - if peer_id.is_empty() { - let balance = balance(root_dir)?; - println!("{balance}"); - } else { - let default_node_dir_path = dirs_next::data_dir() - .ok_or_else(|| eyre!("Failed to obtain data directory path"))? - .join("safe") - .join("node"); - - for id in peer_id { - let path = default_node_dir_path.join(id); - let rewards = balance(&path)?; - println!("Node's rewards wallet balance (PeerId: {id}): {rewards}"); - } - } - Ok(()) - } - WalletCmds::Deposit { stdin, cash_note } => deposit(root_dir, *stdin, cash_note.as_deref()), - WalletCmds::Create { key } => { - match SecretKey::from_hex(key) { - Ok(sk) => { - let main_sk = MainSecretKey::new(sk); - // TODO: encrypt wallet file - // check for existing wallet with balance - let existing_balance = match LocalWallet::load_from(root_dir) { - Ok(wallet) => wallet.balance(), - Err(_) => NanoTokens::zero(), - }; - // if about to overwrite an existing balance, confirm operation - if existing_balance > NanoTokens::zero() { - let prompt = format!("Existing wallet has balance of {existing_balance}. Replace with new wallet? [y/N]"); - let response = get_stdin_response(&prompt); - if response.trim() != "y" { - // Do nothing, return ok and prevent any further operations - println!("Exiting without creating new wallet"); - return Ok(()); - } - // remove existing wallet - let new_location = LocalWallet::clear(root_dir)?; - println!("Old wallet stored at {}", new_location.display()); - } - // Create the new wallet with the new key - let main_pubkey = main_sk.main_pubkey(); - let local_wallet = LocalWallet::create_from_key(root_dir, main_sk)?; - let balance = local_wallet.balance(); - println!( - "Wallet created (balance {balance}) for main public key: {main_pubkey:?}." - ); - } - Err(_err) => { - let main_pk = match PublicKey::from_hex(key) { - Ok(pk) => MainPubkey::new(pk), - Err(err) => return Err(eyre!("Failed to parse hex-encoded PK: {err:?}")), - }; - let pk_hex = main_pk.to_hex(); - let folder_name = - format!("pk_{}_{}", &pk_hex[..6], &pk_hex[pk_hex.len() - 6..]); - let wallet_dir = root_dir.join(folder_name); - let main_pubkey = main_pk.public_key(); - let watch_only_wallet = WatchOnlyWallet::load_from(&wallet_dir, main_pk)?; - let balance = watch_only_wallet.balance(); - println!("Watch-only wallet created (balance {balance}) for main public key: {main_pubkey:?}."); - } - }; - Ok(()) - } - WalletCmds::Transaction { amount, to } => build_unsigned_transaction(amount, to, root_dir), - WalletCmds::Sign { tx } => sign_transaction(tx, root_dir), - cmd => Err(eyre!("{cmd:?} requires us to be connected to the Network")), - } -} - -pub(crate) async fn wallet_cmds( - cmds: WalletCmds, - client: &Client, - root_dir: &Path, - verify_store: bool, -) -> Result<()> { - match cmds { - WalletCmds::Send { amount, to } => send(amount, to, client, root_dir, verify_store).await, - WalletCmds::Broadcast { signed_tx } => { - broadcast_signed_spends(signed_tx, client, root_dir, verify_store).await - } - WalletCmds::Receive { file, transfer } => receive(transfer, file, client, root_dir).await, - WalletCmds::GetFaucet { url } => get_faucet(root_dir, client, url.clone()).await, - WalletCmds::ReceiveOnline { pk, path } => { - let wallet_dir = path.unwrap_or(root_dir.join(DEFAULT_RECEIVE_ONLINE_WALLET_DIR)); - listen_notifs_and_deposit(&wallet_dir, client, pk).await - } - WalletCmds::Audit { dot, royalties } => audit(client, dot, royalties, root_dir).await, - WalletCmds::Verify { - spend_address, - genesis, - } => verify(spend_address, genesis, client).await, - cmd => Err(eyre!( - "{cmd:?} has to be processed before connecting to the network" - )), - } -} - -fn parse_pubkey_address(str_addr: &str) -> Result { - let pk_res = UniquePubkey::from_hex(str_addr); - let addr_res = SpendAddress::from_hex(str_addr); - - match (pk_res, addr_res) { - (Ok(pk), _) => Ok(SpendAddress::from_unique_pubkey(&pk)), - (_, Ok(addr)) => Ok(addr), - _ => Err(eyre!("Failed to parse address: {str_addr}")), - } +// TODO: convert this into a Trait part of the wallet APIs. +enum WalletApiHelper { + #[allow(dead_code)] + WatchOnlyWallet(WatchOnlyWallet), + HotWallet(LocalWallet), } -/// Verify a spend on the Network. -/// if genesis is true, verify all the way to Genesis, note that this might take A VERY LONG TIME -async fn verify(spend_address: String, genesis: bool, client: &Client) -> Result<()> { - if genesis { - println!("Verifying spend all the way to Genesis, note that this might take a while..."); - } else { - println!("Verifying spend..."); - } - - let addr = parse_pubkey_address(&spend_address)?; - match client.verify_spend(addr, genesis).await { - Ok(()) => println!("Spend verified to be stored and unique at {addr:?}"), - Err(e) => println!("Failed to verify spend at {addr:?}: {e}"), +impl WalletApiHelper { + #[allow(dead_code)] + pub fn watch_only_from_pk(main_pk: MainPubkey, root_dir: &Path) -> Result { + let wallet = watch_only_wallet_from_pk(main_pk, root_dir)?; + Ok(Self::WatchOnlyWallet(wallet)) } - Ok(()) -} - -async fn audit(client: &Client, to_dot: bool, find_royalties: bool, root_dir: &Path) -> Result<()> { - if to_dot { - let dag = gather_spend_dag(client, root_dir).await?; - println!("{}", dag.dump_dot_format()); - } else { - //NB TODO use the above DAG to audit too - println!("Auditing the Currency, note that this might take a very long time..."); - let genesis_addr = SpendAddress::from_unique_pubkey(&GENESIS_CASHNOTE.unique_pubkey()); - client - .follow_spend(genesis_addr, find_royalties, root_dir) - .await?; + pub fn load_from(root_dir: &Path) -> Result { + let wallet = LocalWallet::load_from(root_dir)?; + Ok(Self::HotWallet(wallet)) } - Ok(()) -} - -fn address(root_dir: &Path) -> Result<()> { - let wallet = LocalWallet::load_from(root_dir)?; - println!("{:?}", wallet.address()); - Ok(()) -} - -fn balance(root_dir: &Path) -> Result { - let wallet = LocalWallet::try_load_from(root_dir)?; - let balance = wallet.balance(); - Ok(balance) -} - -async fn get_faucet(root_dir: &Path, client: &Client, url: String) -> Result<()> { - let wallet = LocalWallet::load_from(root_dir)?; - let address_hex = wallet.address().to_hex(); - let url = if !url.contains("://") { - format!("{}://{}", "http", url) - } else { - url - }; - let req_url = Url::parse(&format!("{url}/{address_hex}"))?; - println!("Requesting token for wallet address: {address_hex}"); - - let response = reqwest::get(req_url).await?; - let is_ok = response.status().is_success(); - let body = response.text().await?; - if is_ok { - receive(body, false, client, root_dir).await?; - println!("Successfully got tokens from faucet."); - } else { - println!("Failed to get tokens from faucet, server responded with: {body:?}"); - } - Ok(()) -} - -fn deposit(root_dir: &Path, read_from_stdin: bool, cash_note: Option<&str>) -> Result<()> { - if read_from_stdin { - return read_cash_note_from_stdin(root_dir); + pub fn address(&self) -> MainPubkey { + match self { + Self::WatchOnlyWallet(w) => w.address(), + Self::HotWallet(w) => w.address(), + } } - if let Some(cash_note_hex) = cash_note { - return deposit_from_cash_note_hex(root_dir, cash_note_hex); + pub fn balance(&self) -> NanoTokens { + match self { + Self::WatchOnlyWallet(w) => w.balance(), + Self::HotWallet(w) => w.balance(), + } } - let mut wallet = LocalWallet::load_from(root_dir)?; - - let previous_balance = wallet.balance(); - - wallet.try_load_cash_notes()?; - - let deposited = - sn_transfers::NanoTokens::from(wallet.balance().as_nano() - previous_balance.as_nano()); - if deposited.is_zero() { - println!("Nothing deposited."); - } else if let Err(err) = wallet.deposit_and_store_to_disk(&vec![]) { - println!("Failed to store deposited ({deposited}) amount: {err:?}"); - } else { - println!("Deposited {deposited}."); + pub fn read_cash_note_from_stdin(&mut self) -> Result<()> { + println!("Please paste your CashNote below:"); + let mut input = String::new(); + std::io::stdin().read_to_string(&mut input)?; + self.deposit_from_cash_note_hex(&input) } - Ok(()) -} + pub fn deposit_from_cash_note_hex(&mut self, input: &str) -> Result<()> { + let cash_note = CashNote::from_hex(input.trim())?; -fn read_cash_note_from_stdin(root_dir: &Path) -> Result<()> { - println!("Please paste your CashNote below:"); - let mut input = String::new(); - std::io::stdin().read_to_string(&mut input)?; - deposit_from_cash_note_hex(root_dir, &input) -} - -fn deposit_from_cash_note_hex(root_dir: &Path, input: &str) -> Result<()> { - let mut wallet = LocalWallet::load_from(root_dir)?; - let cash_note = sn_transfers::CashNote::from_hex(input.trim())?; + let old_balance = self.balance(); + let cash_notes = vec![cash_note.clone()]; - let old_balance = wallet.balance(); - wallet.deposit_and_store_to_disk(&vec![cash_note])?; - let new_balance = wallet.balance(); - println!("Successfully stored cash_note to wallet dir. \nOld balance: {old_balance}\nNew balance: {new_balance}"); - - Ok(()) -} - -async fn send( - amount: String, - to: String, - client: &Client, - root_dir: &Path, - verify_store: bool, -) -> Result<()> { - let from = LocalWallet::load_from(root_dir)?; - let amount = match NanoTokens::from_str(&amount) { - Ok(amount) => amount, - Err(err) => { - println!("The amount cannot be parsed. Nothing sent."); - return Err(err.into()); - } - }; - let to = match MainPubkey::from_hex(to) { - Ok(to) => to, - Err(err) => { - println!("Error while parsing the recipient's 'to' key: {err:?}"); - return Err(err.into()); - } - }; + let spent_unique_pubkeys: BTreeSet<_> = cash_note + .src_tx + .inputs + .iter() + .map(|input| input.unique_pubkey()) + .collect(); - let cash_note = match sn_client::send(from, amount, to, client, verify_store).await { - Ok(cash_note) => { - let wallet = LocalWallet::load_from(root_dir)?; - println!("Sent {amount:?} to {to:?}"); - println!("New wallet balance is {}.", wallet.balance()); - cash_note - } - Err(err) => { - match err { - ClientError::AmountIsZero => { - println!("Zero amount passed in. Nothing sent."); - } - ClientError::Transfers(WalletError::Transfer(TransferError::NotEnoughBalance( - available, - required, - ))) => { - println!("Could not send due to low balance.\nBalance: {available:?}\nRequired: {required:?}"); - } - _ => { - println!("Failed to send {amount:?} to {to:?} due to {err:?}."); - } + match self { + Self::WatchOnlyWallet(w) => { + w.mark_notes_as_spent(spent_unique_pubkeys); + w.deposit_and_store_to_disk(&cash_notes)? } - return Err(err.into()); - } - }; - - let transfer = Transfer::transfer_from_cash_note(&cash_note)?.to_hex()?; - println!("The encrypted transfer has been successfully created."); - println!("Please share this to the recipient:\n\n{transfer}\n"); - println!("The recipient can then use the 'receive' command to claim the funds."); - - Ok(()) -} - -fn build_unsigned_transaction(amount: &str, to: &str, root_dir: &Path) -> Result<()> { - let mut wallet = LocalWallet::load_from(root_dir)?; - let amount = match NanoTokens::from_str(amount) { - Ok(amount) => amount, - Err(err) => { - println!("The amount cannot be parsed. Nothing sent."); - return Err(err.into()); - } - }; - let to = match MainPubkey::from_hex(to) { - Ok(to) => to, - Err(err) => { - println!("Error while parsing the recipient's 'to' key: {err:?}"); - return Err(err.into()); - } - }; - - let unsigned_transfer = wallet.build_unsigned_transaction(vec![(amount, to)], None)?; - - println!( - "The unsigned transaction has been successfully created:\n\n{}\n", - hex::encode(rmp_serde::to_vec(&unsigned_transfer)?) - ); - println!("Please copy the above text, sign it offline with 'wallet sign' cmd, and then use the signed transaction to broadcast it with 'wallet broadcast' cmd."); - - Ok(()) -} - -fn sign_transaction(tx: &str, root_dir: &Path) -> Result<()> { - let wallet = LocalWallet::load_from(root_dir)?; - let unsigned_transfer: UnsignedTransfer = rmp_serde::from_slice(&hex::decode(tx)?)?; - - println!("The unsigned transaction has been successfully decoded:"); - let mut spent_tx = None; - for (i, (spend, _)) in unsigned_transfer.spends.iter().enumerate() { - println!("\nSpending input #{i}:"); - println!("\tKey: {}", spend.unique_pubkey.to_hex()); - println!("\tAmount: {}", spend.token); - if let Some(ref tx) = spent_tx { - if tx != &spend.spent_tx { - bail!("Transaction seems corrupted, not all Spends (inputs) refer to the same transaction"); + Self::HotWallet(w) => { + w.mark_notes_as_spent(spent_unique_pubkeys); + w.deposit_and_store_to_disk(&cash_notes)? } - } else { - spent_tx = Some(spend.spent_tx.clone()); - } - } - - if let Some(ref tx) = spent_tx { - for (i, output) in tx.outputs.iter().enumerate() { - println!("\nOutput #{i}:"); - println!("\tKey: {}", output.unique_pubkey.to_hex()); - println!("\tAmount: {}", output.amount); } - } else { - bail!("Transaction is corrupted, no transaction information found."); - } - - use dialoguer::Confirm; + let new_balance = self.balance(); + println!("Successfully stored cash_note to wallet dir. \nOld balance: {old_balance}\nNew balance: {new_balance}"); - println!("\n** Please make sure the above information is correct before signing it. **\n"); - let confirmation = Confirm::new() - .with_prompt("Do you want to sign the above transaction?") - .interact()?; - - if !confirmation { - println!("Transaction not signed."); - return Ok(()); + Ok(()) } - println!("Signing the transaction with local hot-wallet..."); - let signed_spends = wallet.sign(unsigned_transfer.spends); - - for signed_spend in signed_spends.iter() { - if let Err(err) = signed_spend.verify(signed_spend.spent_tx_hash()) { - bail!("Signature or transaction generated is invalid: {err:?}"); + pub fn deposit(&mut self, read_from_stdin: bool, cash_note: Option<&str>) -> Result<()> { + if read_from_stdin { + return self.read_cash_note_from_stdin(); } - } - println!( - "The transaction has been successfully signed:\n\n{}\n", - hex::encode(rmp_serde::to_vec(&( - &signed_spends, - unsigned_transfer.output_details, - unsigned_transfer.change_id - ))?) - ); - println!( - "Please copy the above text, and broadcast it to the network with 'wallet broadcast' cmd." - ); + if let Some(cash_note_hex) = cash_note { + return self.deposit_from_cash_note_hex(cash_note_hex); + } - Ok(()) -} + let previous_balance = self.balance(); -async fn broadcast_signed_spends( - signed_tx: String, - client: &Client, - root_dir: &Path, - verify_store: bool, -) -> Result<()> { - let wallet = LocalWallet::load_from(root_dir)?; - let (signed_spends, output_details, change_id): ( - BTreeSet, - BTreeMap, - UniquePubkey, - ) = rmp_serde::from_slice(&hex::decode(signed_tx)?)?; + self.try_load_cash_notes()?; - println!("The signed transaction has been successfully decoded:"); - let mut transaction = None; - for (i, signed_spend) in signed_spends.iter().enumerate() { - println!("\nSpending input #{i}:"); - println!("\tKey: {}", signed_spend.unique_pubkey().to_hex()); - println!("\tAmount: {}", signed_spend.token()); - let linked_tx = signed_spend.spent_tx(); - if let Some(ref tx) = transaction { - if tx != &linked_tx { - bail!("Transaction seems corrupted, not all Spends (inputs) refer to the same transaction"); - } + let deposited = + sn_transfers::NanoTokens::from(self.balance().as_nano() - previous_balance.as_nano()); + if deposited.is_zero() { + println!("Nothing deposited."); + } else if let Err(err) = self.deposit_and_store_to_disk(&vec![]) { + println!("Failed to store deposited ({deposited}) amount: {err:?}"); } else { - transaction = Some(linked_tx); + println!("Deposited {deposited}."); } - if let Err(err) = signed_spend.verify(signed_spend.spent_tx_hash()) { - bail!("Transaction is invalid: {err:?}"); - } + Ok(()) } - let tx = if let Some(tx) = transaction { - for (i, output) in tx.outputs.iter().enumerate() { - println!("\nOutput #{i}:"); - println!("\tKey: {}", output.unique_pubkey.to_hex()); - println!("\tAmount: {}", output.amount); + fn deposit_and_store_to_disk(&mut self, cash_notes: &Vec) -> Result<()> { + match self { + Self::WatchOnlyWallet(w) => w.deposit_and_store_to_disk(cash_notes)?, + Self::HotWallet(w) => w.deposit_and_store_to_disk(cash_notes)?, } - tx - } else { - bail!("Transaction is corrupted, no transaction information found."); - }; - - use dialoguer::Confirm; - - println!("\n** Please make sure the above information is correct before broadcasting it. **\n"); - let confirmation = Confirm::new() - .with_prompt("Do you want to broadcast the above transaction?") - .interact()?; - - if !confirmation { - println!("Transaction was not broadcasted."); - return Ok(()); + Ok(()) } - println!("Broadcasting the transaction to the network..."); - let cash_note = sn_client::broadcast_signed_spends( - wallet, - client, - signed_spends, - tx, - change_id, - output_details, - verify_store, - ) - .await?; - - println!("Transaction broadcasted!."); - let wallet = LocalWallet::load_from(root_dir)?; - println!("New wallet balance is {}.", wallet.balance()); - - let transfer = Transfer::transfer_from_cash_note(&cash_note)?.to_hex()?; - println!("The encrypted transfer has been successfully created."); - println!("Please share this to the recipient:\n\n{transfer}\n"); - println!("The recipient can then use the 'receive' command to claim the funds."); - - Ok(()) -} - -async fn receive(transfer: String, is_file: bool, client: &Client, root_dir: &Path) -> Result<()> { - let transfer = if is_file { - std::fs::read_to_string(transfer)?.trim().to_string() - } else { - transfer - }; - - let transfer = match Transfer::from_hex(&transfer) { - Ok(transfer) => transfer, - Err(err) => { - println!("Failed to parse transfer: {err:?}"); - println!("Transfer: \"{transfer}\""); - return Err(err.into()); - } - }; - println!("Successfully parsed transfer. "); - - println!("Verifying transfer with the Network..."); - let mut wallet = LocalWallet::load_from(root_dir)?; - let cashnotes = match client.receive(&transfer, &wallet).await { - Ok(cashnotes) => cashnotes, - Err(err) => { - println!("Failed to verify and redeem transfer: {err:?}"); - return Err(err.into()); + fn try_load_cash_notes(&mut self) -> Result<()> { + match self { + Self::WatchOnlyWallet(w) => w.try_load_cash_notes()?, + Self::HotWallet(w) => w.try_load_cash_notes()?, } - }; - println!("Successfully verified transfer."); - - let old_balance = wallet.balance(); - wallet.deposit_and_store_to_disk(&cashnotes)?; - let new_balance = wallet.balance(); - - println!("Successfully stored cash_note to wallet dir."); - println!("Old balance: {old_balance}"); - println!("New balance: {new_balance}"); - - Ok(()) -} - -async fn listen_notifs_and_deposit(root_dir: &Path, client: &Client, pk_hex: String) -> Result<()> { - let mut wallet = match MainPubkey::from_hex(&pk_hex) { - Ok(main_pk) => { - let folder_name = format!("pk_{}_{}", &pk_hex[..6], &pk_hex[pk_hex.len() - 6..]); - let wallet_dir = root_dir.join(folder_name); - println!("Loading local wallet from: {}", wallet_dir.display()); - WatchOnlyWallet::load_from(&wallet_dir, main_pk)? - } - Err(err) => return Err(eyre!("Failed to parse hex-encoded public key: {err:?}")), - }; - - let main_pk = wallet.address(); - let pk = main_pk.public_key(); - - client.subscribe_to_topic(ROYALTY_TRANSFER_NOTIF_TOPIC.to_string())?; - let mut events_receiver = client.events_channel(); - - println!("Current balance in local wallet: {}", wallet.balance()); - println!("Listening to transfers notifications for {pk:?}... (press Ctrl+C to exit)"); - println!(); - - while let Ok(event) = events_receiver.recv().await { - let cash_notes = match event { - ClientEvent::GossipsubMsg { topic, msg } => { - // we assume it's a notification of a transfer as that's the only topic we've subscribed to - match try_decode_transfer_notif(&msg) { - Err(err) => { - println!("GossipsubMsg received on topic '{topic}' couldn't be decoded as transfer notif: {err:?}"); - continue; - } - Ok((key, _)) if key != pk => continue, - Ok((key, cashnote_redemptions)) => { - println!("New transfer notification received for {key:?}, containing {} CashNoteRedemption/s.", cashnote_redemptions.len()); - match client - .verify_cash_notes_redemptions(main_pk, &cashnote_redemptions) - .await - { - Err(err) => { - println!("At least one of the CashNoteRedemptions received is invalid, dropping them: {err:?}"); - continue; - } - Ok(cash_notes) => cash_notes, - } - } - } - } - _other_event => continue, - }; - - cash_notes.iter().for_each(|cn| { - let value = match cn.value() { - Ok(value) => value.to_string(), - Err(err) => { - println!("Failed to obtain cash note value: {err}"); - "unknown".to_string() - } - }; - println!( - "CashNote received with {:?}, value: {value}", - cn.unique_pubkey(), - ); - }); - - match wallet.deposit_and_store_to_disk(&cash_notes) { - Ok(()) => {} - Err(err @ WalletError::Io(_)) => { - println!("ERROR: Failed to deposit the received cash notes: {err}"); - println!(); - println!("WARNING: we'll try to reload/recreate the local wallet now, but if it was corrupted there could have been lost funds."); - println!(); - wallet.reload_from_disk_or_recreate()?; - wallet.deposit_and_store_to_disk(&cash_notes)?; - } - Err(other_err) => return Err(other_err.into()), - } - - println!( - "New balance after depositing received CashNote/s: {}", - wallet.balance() - ); - println!(); + Ok(()) } - - Ok(()) } -fn try_decode_transfer_notif(msg: &[u8]) -> Result<(PublicKey, Vec)> { - let mut key_bytes = [0u8; PK_SIZE]; - key_bytes.copy_from_slice( - msg.get(0..PK_SIZE) - .ok_or_else(|| eyre!("msg doesn't have enough bytes"))?, +fn watch_only_wallet_from_pk(main_pk: MainPubkey, root_dir: &Path) -> Result { + let pk_hex = main_pk.to_hex(); + let folder_name = format!("pk_{}_{}", &pk_hex[..6], &pk_hex[pk_hex.len() - 6..]); + let wallet_dir = root_dir.join(folder_name); + println!( + "Loading watch-only local wallet from: {}", + wallet_dir.display() ); - let key = PublicKey::from_bytes(key_bytes)?; - let cashnote_redemptions: Vec = rmp_serde::from_slice(&msg[PK_SIZE..])?; - Ok((key, cashnote_redemptions)) -} - -#[cfg(test)] -mod tests { - use super::*; - use sn_transfers::SpendAddress; - - #[test] - fn test_parse_pubkey_address() -> eyre::Result<()> { - let public_key = SecretKey::random().public_key(); - let unique_pk = UniquePubkey::new(public_key); - let spend_address = SpendAddress::from_unique_pubkey(&unique_pk); - let addr_hex = spend_address.to_hex(); - let unique_pk_hex = unique_pk.to_hex(); - - let addr = parse_pubkey_address(&addr_hex)?; - assert_eq!(addr, spend_address); - - let addr2 = parse_pubkey_address(&unique_pk_hex)?; - assert_eq!(addr2, spend_address); - Ok(()) - } + let wallet = WatchOnlyWallet::load_from(&wallet_dir, main_pk)?; + Ok(wallet) } diff --git a/sn_client/src/audit/mod.rs b/sn_client/src/audit/mod.rs index a50ad6319c..d09d2b815c 100644 --- a/sn_client/src/audit/mod.rs +++ b/sn_client/src/audit/mod.rs @@ -18,11 +18,8 @@ use super::{ use futures::future::join_all; use sn_networking::target_arch::Instant; -use sn_transfers::{ - CashNoteRedemption, SignedSpend, SpendAddress, Transfer, WalletError, WalletResult, - NETWORK_ROYALTIES_PK, -}; -use std::{collections::BTreeSet, iter::Iterator, path::Path}; +use sn_transfers::{SignedSpend, SpendAddress, WalletError, WalletResult}; +use std::{collections::BTreeSet, iter::Iterator}; impl Client { /// Verify that a spend is valid on the network. @@ -159,8 +156,6 @@ impl Client { pub async fn follow_spend( &self, spend_addr: SpendAddress, - find_royalties: bool, - root_dir: &Path, ) -> WalletResult> { let first_spend = self .get_spend_from_network(spend_addr) @@ -208,10 +203,6 @@ impl Client { .map(|s| SpendAddress::from_unique_pubkey(&s.spend.unique_pubkey)), ); - // look for royalties - self.redeem_royalties(find_royalties, &spends, root_dir) - .await; - // add new descendant spends to next gen next_gen_tx.extend(spends.into_iter().map(|s| s.spend.spent_tx)); } @@ -240,65 +231,6 @@ impl Client { println!("Finished auditing! Through {gen} generations, found {n} UTXOs and verified {tx} Transactions in {elapsed:?}"); Ok(all_utxos) } - - /// This function serves as a proof of concept of royalties collection - async fn redeem_royalties( - &self, - find_royalties: bool, - spends: &Vec, - root_dir: &Path, - ) { - if !find_royalties { - return; - } - - // Turn those royalties into a Transfer and redeems them - // This involves encrypting/decrypting the Transfer, which is a waste - // This involves re-verifying, which we don't need as we're already auditing - // This prints out a Transfer for each royalty, which is not ideal but keeps the transfers reasonnably small - // This might print out duplicates as it doens't keep track of what's coming, but that's ok as the cli will know what to do with them - // It is sub-optimial, but it's a working proof of concept that will need to be refined. - // If we decide to adopt this, we will need to turn this indentation space ship into a proper piece of optimized code. - let mut count = 0; - let royalties_key = *NETWORK_ROYALTIES_PK; - let mut wallet = - sn_transfers::LocalWallet::load_from(root_dir).expect("Failed to load wallet"); - for spend in spends { - for derivation_idx in spend.spend.network_royalties.iter() { - count += 1; - let spend_addr = SpendAddress::from_unique_pubkey(&spend.spend.unique_pubkey); - let royalties = vec![CashNoteRedemption::new(*derivation_idx, spend_addr)]; - match Transfer::create(royalties, royalties_key) { - Ok(transfer) => { - let unique_key = royalties_key.new_unique_pubkey(derivation_idx); - println!("Identified royalties token: {unique_key:?}"); - match self.receive(&transfer, &wallet).await { - Ok(cn) => { - println!( - "Successfully received royalties CashNotes, depositing..." - ); - let old_balance = wallet.balance(); - if let Err(e) = wallet.deposit_and_store_to_disk(&cn) { - println!("Failed to store redeemed royalties CashNotes: {e}"); - } else { - let new_balance = wallet.balance(); - println!("Successfully deposited royalties CashNotes, new balance: {new_balance} (was {old_balance})"); - } - } - Err(e) => { - println!("Failed to redeem royalties CashNotes: {e}"); - } - } - } - Err(e) => { - println!("Error creating royalties transfer: {e}"); - } - } - } - } - - println!("Found {count:?} royalties"); - } } fn split_utxos_and_spends(