mirror of
https://github.com/logos-blockchain/lssa.git
synced 2026-06-11 03:29:59 +00:00
* feat: add basic commands for communicating with keycard * initialize changes * reorganization * add script file for easier wallet access * update commands * fixes * fixed load for non continuous run * Updates for signatures with keycard * fix BIP-340 signatures for fixed sized messages * fmt * refactor and add pin support to program facades * fix unit test * fixes * Revert "fixes" This reverts commit 41f34f4ff4145b7abb60fd9bec168ae4b60f23b4. * fixes * fixes * Removed privacy keycard calls * Revert "Removed privacy keycard calls" This reverts commit d70ef505a1f40b87159099761f5fce5a31e3f17b. * Add domain separators * Removed privacy txs for keycard * CI fixes * CI fixes * addressed some comments * fix ci * initialize branch * ci fixes * fix integration test issue and updated keycard firmware * addressed more comments * fixed deny * remove keycard-py * fixed from earlier merge * add hash_message tests * add test * fix deny * CI fixes * fixed integration tests * Update public.rs * update artifacts * privacy command fixes * ci and comments * addressed comments * comment fixes * fixes from merging main * adding support to other programs * expanded support * ci fixes * ci and add private account keys test * some fixes and setup notes * Ci fixes * ci fixes * update key paths to avoid collisions in tests * added separated files for keycard_tests_2.sh * first round of comments * Revert "Merge branch 'main' into marvin/keycard-commands" This reverts commit 3fce53f663a3996938dddf77680854570063ca21, reversing changes made to e7b42a5177641455a8917bd2e29db20afd9690e5. * python comments * addressed comments * compile error fixed * fix artifacts * fix main merge error * adjust signer logic workflow * updating logic * fmt * refactored * clippy fix * minor fix * addressing comments * minor fix * ci fix * addressed deferred comments * clean up * minor cleanup * ci fixes * fmt fix * feat!(wallet): Merged `SigningGroup` with `AccountManager` (#500) * feat: account manager extension * feat(wallet): added unified way of sending public transactions to all facades * fix(wallet): no sign option added * fix(deny): deny fix * fix(wallet): suggestion 1 * fix(wallet): suggestion fix 1 * feat!: Add new path for externally provided seed to the circuit. BREAKING CHANGE: add identity variants to the circuit and change semantics for `Claim::Authorized` for private PDAs * feat(ci): use separate job per each integration tests module * feat(ci): cache rust artifacts * feat(ci): build integration tests binary once and reuse it * fix(wallet): fmt * ci: add bench-regression workflow with criterion-compare for crypto_primitives_bench * fix(wallet): merge postfix * feat!(wallet): SigningGroup merged with AccountManager * fix(ci): deny and artifacts fix * fix(deny): deny fix * fix keycard and lint --------- Co-authored-by: Sergio Chouhy <sergio.chouhy@gmail.com> Co-authored-by: Daniil Polyakov <arjentix@gmail.com> Co-authored-by: Moudy <m.ellaz@hotmail.com> Co-authored-by: Sergio Chouhy <41742639+schouhy@users.noreply.github.com> Co-authored-by: jonesmarvin8 <83104039+jonesmarvin8@users.noreply.github.com> * addressed comments * minor comments * Rebase to main * CI fixes --------- Co-authored-by: Pravdyvy <46261001+Pravdyvy@users.noreply.github.com> Co-authored-by: Sergio Chouhy <sergio.chouhy@gmail.com> Co-authored-by: Daniil Polyakov <arjentix@gmail.com> Co-authored-by: Moudy <m.ellaz@hotmail.com> Co-authored-by: Sergio Chouhy <41742639+schouhy@users.noreply.github.com>
519 lines
20 KiB
Rust
519 lines
20 KiB
Rust
use anyhow::{Context as _, Result};
|
|
use clap::Subcommand;
|
|
use common::transaction::LeeTransaction;
|
|
use lee::AccountId;
|
|
|
|
use crate::{
|
|
AccDecodeData::Decode,
|
|
AccountIdentity, WalletCore,
|
|
account::AccountIdWithPrivacy,
|
|
cli::{CliAccountMention, SubcommandReturnValue, WalletSubcommand},
|
|
program_facades::native_token_transfer::NativeTokenTransfer,
|
|
};
|
|
|
|
/// Represents generic CLI subcommand for a wallet working with native token transfer program.
|
|
#[derive(Subcommand, Debug, Clone)]
|
|
pub enum AuthTransferSubcommand {
|
|
/// Initialize account under authenticated transfer program.
|
|
Init {
|
|
/// Either 32 byte base58 account id string with privacy prefix or a label.
|
|
#[arg(long)]
|
|
account_id: CliAccountMention,
|
|
},
|
|
/// Send native tokens from one account to another with variable privacy.
|
|
///
|
|
/// If receiver is private, then `to` and (`to_npk` , `to_vpk`) is a mutually exclusive
|
|
/// patterns.
|
|
///
|
|
/// First is used for owned accounts, second otherwise.
|
|
Send {
|
|
/// Either 32 byte base58 account id string with privacy prefix or a label.
|
|
#[arg(long)]
|
|
from: CliAccountMention,
|
|
/// Either 32 byte base58 account id string with privacy prefix or a label.
|
|
#[arg(long)]
|
|
to: Option<CliAccountMention>,
|
|
/// `to_npk` - valid 32 byte hex string.
|
|
#[arg(long, conflicts_with = "to_keys")]
|
|
to_npk: Option<String>,
|
|
/// `to_vpk` - valid hex-encoded ML-KEM-768 encapsulation key (1184 bytes).
|
|
#[arg(long, conflicts_with = "to_keys")]
|
|
to_vpk: Option<String>,
|
|
/// Path to a keys file exported by `wallet account show-keys`, containing npk
|
|
/// and vpk on separate lines. Replaces `--to-npk` and `--to-vpk`.
|
|
#[arg(long, conflicts_with_all = ["to_npk", "to_vpk"])]
|
|
to_keys: Option<String>,
|
|
/// Identifier for the recipient's private account (only used when sending to a foreign
|
|
/// private account via `--to-npk`/`--to-vpk` or `--to-keys`).
|
|
#[arg(long)]
|
|
to_identifier: Option<u128>,
|
|
/// amount - amount of balance to move.
|
|
#[arg(long)]
|
|
amount: u128,
|
|
},
|
|
}
|
|
|
|
impl WalletSubcommand for AuthTransferSubcommand {
|
|
async fn handle_subcommand(
|
|
self,
|
|
wallet_core: &mut WalletCore,
|
|
) -> Result<SubcommandReturnValue> {
|
|
match self {
|
|
Self::Init { account_id } => {
|
|
let resolved = account_id.resolve(wallet_core.storage())?;
|
|
match resolved {
|
|
AccountIdWithPrivacy::Public(pub_account_id) => {
|
|
let tx_hash = NativeTokenTransfer(wallet_core)
|
|
.register_account(account_id.into_public_identity(pub_account_id))
|
|
.await?;
|
|
|
|
println!("Transaction hash is {tx_hash}");
|
|
|
|
let transfer_tx = wallet_core.poll_native_token_transfer(tx_hash).await?;
|
|
|
|
println!("Transaction data is {transfer_tx:?}");
|
|
|
|
wallet_core.store_persistent_data()?;
|
|
}
|
|
AccountIdWithPrivacy::Private(account_id) => {
|
|
let (tx_hash, secret) = NativeTokenTransfer(wallet_core)
|
|
.register_account_private(account_id)
|
|
.await?;
|
|
|
|
println!("Transaction hash is {tx_hash}");
|
|
|
|
let transfer_tx = wallet_core.poll_native_token_transfer(tx_hash).await?;
|
|
|
|
if let LeeTransaction::PrivacyPreserving(tx) = transfer_tx {
|
|
let acc_decode_data = vec![Decode(secret, account_id)];
|
|
|
|
wallet_core.decode_insert_privacy_preserving_transaction_results(
|
|
&tx,
|
|
&acc_decode_data,
|
|
)?;
|
|
}
|
|
|
|
wallet_core.store_persistent_data()?;
|
|
}
|
|
}
|
|
|
|
Ok(SubcommandReturnValue::Empty)
|
|
}
|
|
Self::Send {
|
|
from: from_account,
|
|
to: to_account,
|
|
to_npk,
|
|
to_vpk,
|
|
to_keys,
|
|
to_identifier,
|
|
amount,
|
|
} => {
|
|
// Resolve --to-keys into --to-npk / --to-vpk equivalents.
|
|
let (to_npk, to_vpk) = if let Some(path) = to_keys {
|
|
let (npk_bytes, vpk_bytes) = crate::cli::read_keys_file(&path)?;
|
|
(Some(hex::encode(npk_bytes)), Some(hex::encode(vpk_bytes)))
|
|
} else {
|
|
(to_npk, to_vpk)
|
|
};
|
|
|
|
let from = from_account.resolve(wallet_core.storage())?;
|
|
let to = to_account
|
|
.as_ref()
|
|
.map(|m| m.resolve(wallet_core.storage()))
|
|
.transpose()?;
|
|
let underlying_subcommand = match (to, to_npk, to_vpk) {
|
|
(None, None, None) => {
|
|
anyhow::bail!(
|
|
"Provide either account account_id of receiver or their public keys"
|
|
);
|
|
}
|
|
(Some(_), Some(_), Some(_)) => {
|
|
anyhow::bail!(
|
|
"Provide only one variant: either account account_id of receiver or their public keys"
|
|
);
|
|
}
|
|
(_, Some(_), None) | (_, None, Some(_)) => {
|
|
anyhow::bail!("List of public keys is uncomplete");
|
|
}
|
|
(Some(to), None, None) => match (from, to) {
|
|
(AccountIdWithPrivacy::Public(from), AccountIdWithPrivacy::Public(to)) => {
|
|
let to_mention = to_account.expect("matched Some branch");
|
|
NativeTokenTransferProgramSubcommand::Public {
|
|
from: Some(from_account.into_public_identity(from)),
|
|
to: Some(to_mention.into_public_identity(to)),
|
|
amount,
|
|
}
|
|
}
|
|
(
|
|
AccountIdWithPrivacy::Private(from),
|
|
AccountIdWithPrivacy::Private(to),
|
|
) => NativeTokenTransferProgramSubcommand::Private(
|
|
NativeTokenTransferProgramSubcommandPrivate::PrivateOwned {
|
|
from,
|
|
to,
|
|
amount,
|
|
},
|
|
),
|
|
(AccountIdWithPrivacy::Private(from), AccountIdWithPrivacy::Public(to)) => {
|
|
NativeTokenTransferProgramSubcommand::Deshielded { from, to, amount }
|
|
}
|
|
(AccountIdWithPrivacy::Public(from), AccountIdWithPrivacy::Private(to)) => {
|
|
NativeTokenTransferProgramSubcommand::Shielded(
|
|
NativeTokenTransferProgramSubcommandShielded::ShieldedOwned {
|
|
from: Some(from_account.into_public_identity(from)),
|
|
to,
|
|
amount,
|
|
},
|
|
)
|
|
}
|
|
},
|
|
(None, Some(to_npk), Some(to_vpk)) => match from {
|
|
AccountIdWithPrivacy::Private(from) => {
|
|
NativeTokenTransferProgramSubcommand::Private(
|
|
NativeTokenTransferProgramSubcommandPrivate::PrivateForeign {
|
|
from,
|
|
to_npk,
|
|
to_vpk,
|
|
to_identifier,
|
|
amount,
|
|
},
|
|
)
|
|
}
|
|
AccountIdWithPrivacy::Public(from) => {
|
|
NativeTokenTransferProgramSubcommand::Shielded(
|
|
NativeTokenTransferProgramSubcommandShielded::ShieldedForeign {
|
|
from: Some(from_account.into_public_identity(from)),
|
|
to_npk,
|
|
to_vpk,
|
|
to_identifier,
|
|
amount,
|
|
},
|
|
)
|
|
}
|
|
},
|
|
};
|
|
|
|
underlying_subcommand.handle_subcommand(wallet_core).await
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Represents generic CLI subcommand for a wallet working with native token transfer program.
|
|
#[derive(Subcommand, Debug, Clone)]
|
|
pub enum NativeTokenTransferProgramSubcommand {
|
|
/// Send native token transfer from `from` to `to` for `amount`.
|
|
///
|
|
/// Public operation.
|
|
Public {
|
|
#[arg(skip)]
|
|
from: Option<AccountIdentity>,
|
|
#[arg(skip)]
|
|
to: Option<AccountIdentity>,
|
|
/// amount - amount of balance to move.
|
|
#[arg(long)]
|
|
amount: u128,
|
|
},
|
|
/// Private execution.
|
|
#[command(subcommand)]
|
|
Private(NativeTokenTransferProgramSubcommandPrivate),
|
|
/// Send native token transfer from `from` to `to` for `amount`.
|
|
///
|
|
/// Deshielded operation.
|
|
Deshielded {
|
|
/// from - valid 32 byte hex string.
|
|
#[arg(long)]
|
|
from: AccountId,
|
|
/// to - valid 32 byte hex string.
|
|
#[arg(long)]
|
|
to: AccountId,
|
|
/// amount - amount of balance to move.
|
|
#[arg(long)]
|
|
amount: u128,
|
|
},
|
|
/// Shielded execution.
|
|
#[command(subcommand)]
|
|
Shielded(NativeTokenTransferProgramSubcommandShielded),
|
|
}
|
|
|
|
/// Represents generic shielded CLI subcommand for a wallet working with native token transfer
|
|
/// program.
|
|
#[derive(Subcommand, Debug, Clone)]
|
|
pub enum NativeTokenTransferProgramSubcommandShielded {
|
|
/// Send native token transfer from `from` to `to` for `amount`.
|
|
///
|
|
/// Shielded operation.
|
|
ShieldedOwned {
|
|
/// from - valid 32 byte hex string.
|
|
#[arg(skip)]
|
|
from: Option<AccountIdentity>,
|
|
/// to - valid 32 byte hex string.
|
|
#[arg(long)]
|
|
to: AccountId,
|
|
/// amount - amount of balance to move.
|
|
#[arg(long)]
|
|
amount: u128,
|
|
},
|
|
/// Send native token transfer from `from` to `to` for `amount`.
|
|
///
|
|
/// Shielded operation.
|
|
ShieldedForeign {
|
|
#[arg(skip)]
|
|
from: Option<AccountIdentity>,
|
|
/// `to_npk` - valid 32 byte hex string.
|
|
#[arg(long)]
|
|
to_npk: String,
|
|
/// `to_vpk` - valid hex-encoded ML-KEM-768 encapsulation key (1184 bytes).
|
|
#[arg(long)]
|
|
to_vpk: String,
|
|
/// Identifier for the recipient's private account.
|
|
#[arg(long)]
|
|
to_identifier: Option<u128>,
|
|
/// amount - amount of balance to move.
|
|
#[arg(long)]
|
|
amount: u128,
|
|
},
|
|
}
|
|
|
|
/// Represents generic private CLI subcommand for a wallet working with native token transfer
|
|
/// program.
|
|
#[derive(Subcommand, Debug, Clone)]
|
|
pub enum NativeTokenTransferProgramSubcommandPrivate {
|
|
/// Send native token transfer from `from` to `to` for `amount`.
|
|
///
|
|
/// Private operation.
|
|
PrivateOwned {
|
|
/// from - valid 32 byte hex string.
|
|
#[arg(long)]
|
|
from: AccountId,
|
|
/// to - valid 32 byte hex string.
|
|
#[arg(long)]
|
|
to: AccountId,
|
|
/// amount - amount of balance to move.
|
|
#[arg(long)]
|
|
amount: u128,
|
|
},
|
|
/// Send native token transfer from `from` to `to` for `amount`.
|
|
///
|
|
/// Private operation.
|
|
PrivateForeign {
|
|
/// from - valid 32 byte hex string.
|
|
#[arg(long)]
|
|
from: AccountId,
|
|
/// `to_npk` - valid 32 byte hex string.
|
|
#[arg(long)]
|
|
to_npk: String,
|
|
/// `to_vpk` - valid hex-encoded ML-KEM-768 encapsulation key (1184 bytes).
|
|
#[arg(long)]
|
|
to_vpk: String,
|
|
/// Identifier for the recipient's private account.
|
|
#[arg(long)]
|
|
to_identifier: Option<u128>,
|
|
/// amount - amount of balance to move.
|
|
#[arg(long)]
|
|
amount: u128,
|
|
},
|
|
}
|
|
|
|
impl WalletSubcommand for NativeTokenTransferProgramSubcommandPrivate {
|
|
async fn handle_subcommand(
|
|
self,
|
|
wallet_core: &mut WalletCore,
|
|
) -> Result<SubcommandReturnValue> {
|
|
match self {
|
|
Self::PrivateOwned { from, to, amount } => {
|
|
let (tx_hash, [secret_from, secret_to]) = NativeTokenTransfer(wallet_core)
|
|
.send_private_transfer_to_owned_account(from, to, amount)
|
|
.await?;
|
|
|
|
println!("Transaction hash is {tx_hash}");
|
|
|
|
let transfer_tx = wallet_core.poll_native_token_transfer(tx_hash).await?;
|
|
|
|
if let LeeTransaction::PrivacyPreserving(tx) = transfer_tx {
|
|
let acc_decode_data = vec![Decode(secret_from, from), Decode(secret_to, to)];
|
|
|
|
wallet_core.decode_insert_privacy_preserving_transaction_results(
|
|
&tx,
|
|
&acc_decode_data,
|
|
)?;
|
|
}
|
|
|
|
wallet_core.store_persistent_data()?;
|
|
|
|
Ok(SubcommandReturnValue::PrivacyPreservingTransfer { tx_hash })
|
|
}
|
|
Self::PrivateForeign {
|
|
from,
|
|
to_npk,
|
|
to_vpk,
|
|
to_identifier,
|
|
amount,
|
|
} => {
|
|
let to_npk_res = hex::decode(to_npk)?;
|
|
let mut to_npk = [0; 32];
|
|
to_npk.copy_from_slice(&to_npk_res);
|
|
let to_npk = lee_core::NullifierPublicKey(to_npk);
|
|
|
|
let to_vpk_res = hex::decode(&to_vpk)
|
|
.context("wallet::cli::programs::native_token_transfer: to_vpk must be a valid hex string")?;
|
|
let to_vpk = lee_core::encryption::ViewingPublicKey::from_bytes(to_vpk_res)
|
|
.map_err(|e| anyhow::anyhow!("{e}"))?;
|
|
|
|
let (tx_hash, [secret_from, _]) = NativeTokenTransfer(wallet_core)
|
|
.send_private_transfer_to_outer_account(
|
|
from,
|
|
to_npk,
|
|
to_vpk,
|
|
to_identifier.unwrap_or_else(rand::random),
|
|
amount,
|
|
)
|
|
.await?;
|
|
|
|
println!("Transaction hash is {tx_hash}");
|
|
|
|
let transfer_tx = wallet_core.poll_native_token_transfer(tx_hash).await?;
|
|
|
|
if let LeeTransaction::PrivacyPreserving(tx) = transfer_tx {
|
|
let acc_decode_data = vec![Decode(secret_from, from)];
|
|
|
|
wallet_core.decode_insert_privacy_preserving_transaction_results(
|
|
&tx,
|
|
&acc_decode_data,
|
|
)?;
|
|
}
|
|
|
|
wallet_core.store_persistent_data()?;
|
|
|
|
Ok(SubcommandReturnValue::PrivacyPreservingTransfer { tx_hash })
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl WalletSubcommand for NativeTokenTransferProgramSubcommandShielded {
|
|
async fn handle_subcommand(
|
|
self,
|
|
wallet_core: &mut WalletCore,
|
|
) -> Result<SubcommandReturnValue> {
|
|
match self {
|
|
Self::ShieldedOwned { from, to, amount } => {
|
|
let (tx_hash, secret) = NativeTokenTransfer(wallet_core)
|
|
.send_shielded_transfer(
|
|
from.expect("from set during Send dispatch"),
|
|
to,
|
|
amount,
|
|
)
|
|
.await?;
|
|
|
|
println!("Transaction hash is {tx_hash}");
|
|
|
|
let transfer_tx = wallet_core.poll_native_token_transfer(tx_hash).await?;
|
|
|
|
if let LeeTransaction::PrivacyPreserving(tx) = transfer_tx {
|
|
let acc_decode_data = vec![Decode(secret, to)];
|
|
|
|
wallet_core.decode_insert_privacy_preserving_transaction_results(
|
|
&tx,
|
|
&acc_decode_data,
|
|
)?;
|
|
}
|
|
|
|
wallet_core.store_persistent_data()?;
|
|
|
|
Ok(SubcommandReturnValue::PrivacyPreservingTransfer { tx_hash })
|
|
}
|
|
Self::ShieldedForeign {
|
|
from,
|
|
to_npk,
|
|
to_vpk,
|
|
to_identifier,
|
|
amount,
|
|
} => {
|
|
let to_npk_res = hex::decode(to_npk)?;
|
|
let mut to_npk = [0; 32];
|
|
to_npk.copy_from_slice(&to_npk_res);
|
|
let to_npk = lee_core::NullifierPublicKey(to_npk);
|
|
|
|
let to_vpk_res = hex::decode(&to_vpk)
|
|
.context("wallet::cli::programs::native_token_transfer: to_vpk must be a valid hex string")?;
|
|
let to_vpk = lee_core::encryption::ViewingPublicKey::from_bytes(to_vpk_res)
|
|
.map_err(|e| anyhow::anyhow!("{e}"))?;
|
|
|
|
let (tx_hash, _) = NativeTokenTransfer(wallet_core)
|
|
.send_shielded_transfer_to_outer_account(
|
|
from.expect("from set during Send dispatch"),
|
|
to_npk,
|
|
to_vpk,
|
|
to_identifier.unwrap_or_else(rand::random),
|
|
amount,
|
|
)
|
|
.await?;
|
|
|
|
println!("Transaction hash is {tx_hash}");
|
|
|
|
wallet_core.store_persistent_data()?;
|
|
|
|
Ok(SubcommandReturnValue::PrivacyPreservingTransfer { tx_hash })
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl WalletSubcommand for NativeTokenTransferProgramSubcommand {
|
|
async fn handle_subcommand(
|
|
self,
|
|
wallet_core: &mut WalletCore,
|
|
) -> Result<SubcommandReturnValue> {
|
|
match self {
|
|
Self::Private(private_subcommand) => {
|
|
private_subcommand.handle_subcommand(wallet_core).await
|
|
}
|
|
Self::Shielded(shielded_subcommand) => {
|
|
shielded_subcommand.handle_subcommand(wallet_core).await
|
|
}
|
|
Self::Deshielded { from, to, amount } => {
|
|
let (tx_hash, secret) = NativeTokenTransfer(wallet_core)
|
|
.send_deshielded_transfer(from, to, amount)
|
|
.await?;
|
|
|
|
println!("Transaction hash is {tx_hash}");
|
|
|
|
let transfer_tx = wallet_core.poll_native_token_transfer(tx_hash).await?;
|
|
|
|
if let LeeTransaction::PrivacyPreserving(tx) = transfer_tx {
|
|
let acc_decode_data = vec![Decode(secret, from)];
|
|
|
|
wallet_core.decode_insert_privacy_preserving_transaction_results(
|
|
&tx,
|
|
&acc_decode_data,
|
|
)?;
|
|
}
|
|
|
|
wallet_core.store_persistent_data()?;
|
|
|
|
Ok(SubcommandReturnValue::PrivacyPreservingTransfer { tx_hash })
|
|
}
|
|
Self::Public { from, to, amount } => {
|
|
let tx_hash = NativeTokenTransfer(wallet_core)
|
|
.send_public_transfer(
|
|
from.expect("from is set during Send dispatch"),
|
|
to.expect("to is set during Send dispatch"),
|
|
amount,
|
|
)
|
|
.await?;
|
|
|
|
println!("Transaction hash is {tx_hash}");
|
|
|
|
let transfer_tx = wallet_core.poll_native_token_transfer(tx_hash).await?;
|
|
|
|
println!("Transaction data is {transfer_tx:?}");
|
|
|
|
wallet_core.store_persistent_data()?;
|
|
|
|
Ok(SubcommandReturnValue::Empty)
|
|
}
|
|
}
|
|
}
|
|
}
|