lssa/wallet/src/lib.rs

188 lines
5.2 KiB
Rust
Raw Normal View History

2025-08-07 14:07:34 +03:00
use std::sync::Arc;
2024-12-05 13:05:58 +02:00
2025-07-22 15:22:20 +03:00
use common::{
2025-08-07 14:07:34 +03:00
execution_input::PublicNativeTokenSend,
sequencer_client::{json::SendTxResponse, SequencerClient},
transaction::Transaction,
ExecutionFailureKind,
2025-07-22 15:22:20 +03:00
};
2024-12-25 09:50:54 +02:00
2025-08-04 15:09:28 +03:00
use accounts::account_core::{address::AccountAddress, Account};
2025-08-07 14:07:34 +03:00
use anyhow::Result;
2025-04-24 15:51:34 +03:00
use chain_storage::NodeChainStore;
2025-08-04 15:09:28 +03:00
use common::transaction::TransactionBody;
2024-12-03 09:32:35 +02:00
use config::NodeConfig;
2025-08-06 14:56:58 +03:00
use log::info;
2025-08-07 14:07:34 +03:00
use sc_core::proofs_circuits::pedersen_commitment_vec;
2025-08-04 15:09:28 +03:00
use tokio::sync::RwLock;
2024-12-03 09:32:35 +02:00
2025-08-06 14:56:58 +03:00
use clap::{Parser, Subcommand};
2025-08-07 14:07:34 +03:00
use crate::helperfunctions::{fetch_config, produce_account_addr_from_hex};
pub const HOME_DIR_ENV_VAR: &str = "NSSA_WALLET_HOME_DIR";
2024-12-25 09:50:54 +02:00
pub const BLOCK_GEN_DELAY_SECS: u64 = 20;
2025-04-24 15:51:34 +03:00
pub mod chain_storage;
2024-12-03 09:32:35 +02:00
pub mod config;
2025-08-07 14:07:34 +03:00
pub mod helperfunctions;
pub mod requests_structs;
2024-12-22 16:14:52 +02:00
2024-12-03 09:32:35 +02:00
pub struct NodeCore {
2024-12-25 09:50:54 +02:00
pub storage: Arc<RwLock<NodeChainStore>>,
2024-12-03 09:32:35 +02:00
pub node_config: NodeConfig,
2024-12-20 11:02:12 +02:00
pub sequencer_client: Arc<SequencerClient>,
2024-12-05 13:05:58 +02:00
}
impl NodeCore {
pub async fn start_from_config_update_chain(config: NodeConfig) -> Result<Self> {
2025-08-07 14:07:34 +03:00
let client = Arc::new(SequencerClient::new(config.sequencer_addr.clone())?);
2024-12-05 13:05:58 +02:00
2025-08-05 14:59:20 +03:00
let mut storage = NodeChainStore::new(config.clone())?;
2025-07-30 14:01:40 +03:00
for acc in config.clone().initial_accounts {
2025-07-23 15:16:53 +03:00
storage.acc_map.insert(acc.address, acc);
}
2024-12-05 13:05:58 +02:00
2024-12-25 09:50:54 +02:00
let wrapped_storage = Arc::new(RwLock::new(storage));
2024-12-05 13:05:58 +02:00
Ok(Self {
storage: wrapped_storage,
node_config: config.clone(),
2024-12-20 11:02:12 +02:00
sequencer_client: client.clone(),
2024-12-05 13:05:58 +02:00
})
}
2024-12-22 16:14:52 +02:00
2025-05-23 09:04:04 +03:00
pub async fn get_roots(&self) -> [[u8; 32]; 2] {
2025-01-24 09:10:42 +02:00
let storage = self.storage.read().await;
[
2025-02-05 12:24:09 +02:00
storage.utxo_commitments_store.get_root().unwrap_or([0; 32]),
storage.pub_tx_store.get_root().unwrap_or([0; 32]),
2025-01-24 09:10:42 +02:00
]
}
2024-12-25 09:50:54 +02:00
pub async fn create_new_account(&mut self) -> AccountAddress {
let account = Account::new();
account.log();
2024-12-25 09:50:54 +02:00
let addr = account.address;
{
let mut write_guard = self.storage.write().await;
write_guard.acc_map.insert(account.address, account);
}
addr
}
pub async fn send_public_native_token_transfer(
&self,
from: AccountAddress,
2025-07-29 16:09:21 -03:00
nonce: u64,
to: AccountAddress,
2025-07-24 16:05:27 +03:00
balance_to_move: u64,
) -> Result<SendTxResponse, ExecutionFailureKind> {
let tx_roots = self.get_roots().await;
let public_context = {
let read_guard = self.storage.read().await;
read_guard.produce_context(from)
};
let (tweak, secret_r, commitment) = pedersen_commitment_vec(
//Will not panic, as public context is serializable
public_context.produce_u64_list_from_context().unwrap(),
);
let sc_addr = hex::encode([0; 32]);
2025-07-22 15:22:20 +03:00
let tx: TransactionBody =
sc_core::transaction_payloads_tools::create_public_transaction_payload(
serde_json::to_vec(&PublicNativeTokenSend {
from,
2025-07-29 16:09:21 -03:00
nonce,
to,
2025-07-24 16:05:27 +03:00
balance_to_move,
})
.unwrap(),
commitment,
tweak,
secret_r,
sc_addr,
2025-07-24 09:14:38 +03:00
);
tx.log();
2025-07-22 15:22:20 +03:00
{
let read_guard = self.storage.read().await;
let account = read_guard.acc_map.get(&from);
if let Some(account) = account {
let key_to_sign_transaction = account.key_holder.get_pub_account_signing_key();
let signed_transaction = Transaction::new(tx, key_to_sign_transaction);
Ok(self
.sequencer_client
.send_tx(signed_transaction, tx_roots)
.await?)
} else {
Err(ExecutionFailureKind::AmountMismatchError)
}
}
}
2024-12-03 09:32:35 +02:00
}
2025-01-10 03:00:32 +02:00
2025-08-06 14:56:58 +03:00
///Represents CLI command for a wallet
#[derive(Subcommand, Debug, Clone)]
2025-08-07 14:07:34 +03:00
#[clap(about)]
2025-08-06 14:56:58 +03:00
pub enum Command {
SendNativeTokenTransfer {
2025-08-07 14:07:34 +03:00
///from - valid 32 byte hex string
2025-08-06 14:56:58 +03:00
#[arg(long)]
from: String,
2025-08-07 14:07:34 +03:00
///to - valid 32 byte hex string
2025-08-06 14:56:58 +03:00
#[arg(long)]
to: String,
2025-08-07 14:07:34 +03:00
///amount - amount of balance to move
2025-08-06 14:56:58 +03:00
#[arg(long)]
amount: u64,
},
2025-08-08 10:07:10 +03:00
DumpAccountsOnDisc,
2025-08-06 14:56:58 +03:00
}
2025-08-07 14:07:34 +03:00
///To execute commands, env var NSSA_WALLET_HOME_DIR must be set into directory with config
2025-08-06 14:56:58 +03:00
#[derive(Parser, Debug)]
2025-08-07 14:07:34 +03:00
#[clap(version, about)]
2025-08-06 14:56:58 +03:00
pub struct Args {
/// Wallet command
#[command(subcommand)]
pub command: Command,
}
pub async fn execute_subcommand(command: Command) -> Result<()> {
match command {
Command::SendNativeTokenTransfer { from, to, amount } => {
let node_config = fetch_config()?;
let from = produce_account_addr_from_hex(from)?;
let to = produce_account_addr_from_hex(to)?;
let wallet_core = NodeCore::start_from_config_update_chain(node_config).await?;
//ToDo: Nonce management
let res = wallet_core
.send_public_native_token_transfer(from, 0, to, amount)
.await?;
info!("Results of tx send is {res:#?}");
}
2025-08-08 10:07:10 +03:00
Command::DumpAccountsOnDisc => {
info!("Accounts stored at path");
}
2025-08-06 14:56:58 +03:00
}
Ok(())
}