Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Add new Key type; allowing public keys to be named keys #1700

Open
wants to merge 82 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
82 commits
Select commit Hold shift + click to select a range
d33a9fc
feat: Add new `Key` type; allowing public keys to be named keys
willemneal Nov 5, 2024
2aec046
fix: lookup first before parsing raw key
willemneal Nov 12, 2024
52bd5e8
fix: docs
willemneal Nov 14, 2024
8f96343
fix: clippy
willemneal Nov 14, 2024
0cc57fe
feat: update tx new commands to use Address type
willemneal Nov 26, 2024
31cde54
fix: rebase issue
willemneal Nov 29, 2024
e5f0e5d
Merge branch 'main' into feat/muxed_keys
willemneal Nov 30, 2024
a86a98f
Update cmd/soroban-cli/src/config/key.rs
willemneal Dec 2, 2024
a64aae5
Merge branch 'main' into feat/muxed_keys
willemneal Dec 2, 2024
b36db49
fix: PR review
willemneal Dec 2, 2024
8c06970
feat: initial work into system keychain
willemneal Nov 5, 2024
883cb8c
chore: clean up
willemneal Nov 5, 2024
eb70734
Add KeyName struct in address
elizabethengelman Dec 2, 2024
d4e2500
Add Secret::Keychain
elizabethengelman Nov 14, 2024
3ee20b8
keys generate: allow for generating keys that are stored in keychain
elizabethengelman Nov 18, 2024
1a24c00
keys generate: Namespace keychain entry to identity name
elizabethengelman Nov 18, 2024
4a0c900
keys generate: don't allow 'keychain:' as a key name
elizabethengelman Nov 20, 2024
755c318
keys address: use keychain entry in secret to get the pub key
elizabethengelman Nov 20, 2024
dd07bf7
tx sign: allow a keychain identity sign a tx
elizabethengelman Nov 21, 2024
f042d19
Cleanup
elizabethengelman Nov 21, 2024
9cf5916
Use keyring mock for generate tests
elizabethengelman Nov 22, 2024
cd515d3
Refactor keyring: add keyring entry as StellarEntry field
elizabethengelman Nov 25, 2024
276558d
Add tests for keyring
elizabethengelman Nov 25, 2024
20c651b
Update config/secret tests
elizabethengelman Nov 25, 2024
66d1bfc
Cleanup
elizabethengelman Nov 25, 2024
8b9763e
Rename keychain arg to secure_store in generate
elizabethengelman Nov 25, 2024
fdefa2a
Rename Secret::Keychain to Secret::SecureStore
elizabethengelman Nov 25, 2024
36559ea
Rename SignerKind::Keychain to SignerKind::SecureStore
elizabethengelman Nov 25, 2024
f98b709
Use print for new fns in generate
elizabethengelman Dec 2, 2024
0f3106b
Return error when trying to get Secure Store secret
elizabethengelman Dec 2, 2024
e120595
Cleanup tests
elizabethengelman Dec 2, 2024
c4d6f99
Merge branch 'main' into feat/os_keychain
elizabethengelman Dec 2, 2024
c7e2cbc
Merge branch 'main' into feat/os_keychain
elizabethengelman Dec 3, 2024
57ba3a4
Install libdbus for rpc-tests and bindings-ts workflows
elizabethengelman Dec 3, 2024
0814e4b
Update generated docs
elizabethengelman Dec 3, 2024
74fa05b
Install libdbus for binaries workflow when target aarch64-unknown-lin…
elizabethengelman Dec 3, 2024
7ff1f6a
Clippy
elizabethengelman Dec 3, 2024
f263d8d
Install libdbus for rust workflow
elizabethengelman Dec 3, 2024
e2b7342
Install libdbus-1-dev in binaries workflow for build step
elizabethengelman Dec 3, 2024
f952c10
Merge branch 'main' into feat/muxed_keys
willemneal Dec 3, 2024
dabbb27
Impl Display for KeyName
elizabethengelman Dec 3, 2024
d9131b4
Use resolve_muxed_account in resolve_secret
elizabethengelman Dec 3, 2024
3ad6b9b
Use resolve_muxed_account to get public key
elizabethengelman Dec 3, 2024
9d02b01
Clippy
elizabethengelman Dec 3, 2024
4bddd24
Merge branch 'main' into feat/os_keychain
elizabethengelman Dec 3, 2024
a72276d
fix: Sign tx hash instead of tx env with keychain
elizabethengelman Dec 4, 2024
ed81494
Merge branch 'main' into feat/muxed_keys
willemneal Dec 5, 2024
9226b1b
fix: fmt
willemneal Dec 5, 2024
3a25564
fix: update to work with `op add`
willemneal Dec 5, 2024
5f55e7c
fix: simplify op::add::args::Args
willemneal Dec 5, 2024
0e7f08f
fix: remove untagged and use rename to keep same serialized output
willemneal Dec 5, 2024
aa8bb16
fix: docs and clippy in tests
willemneal Dec 5, 2024
65284cc
Merge branch 'main' into feat/os_keychain
elizabethengelman Dec 5, 2024
e9f86e7
Merge branch 'main' into feat/os_keychain
elizabethengelman Dec 9, 2024
4288630
Remove unused bin/secret
elizabethengelman Dec 9, 2024
6661b42
Merge branch 'main' into feat/os_keychain
elizabethengelman Dec 12, 2024
b2120d2
Merge branch 'main' into feat/os_keychain
elizabethengelman Dec 13, 2024
64cd37b
Apply suggestions from code review
willemneal Dec 13, 2024
732e7f4
Merge branch 'main' into feat/muxed_keys
willemneal Dec 13, 2024
d4ba93d
Update cmd/soroban-cli/src/config/locator.rs
willemneal Dec 13, 2024
318f579
fix: refactor with renames from PR
willemneal Dec 13, 2024
60c9d49
Merge branch 'main' into feat/muxed_keys
willemneal Dec 16, 2024
24e7b59
Merge branch 'main' into feat/os_keychain
elizabethengelman Dec 17, 2024
cfdb324
Merge branch 'main' into feat/os_keychain
elizabethengelman Dec 18, 2024
8e158a1
Merge branch 'main' into feat/os_keychain
willemneal Dec 19, 2024
d776f2d
Merge remote-tracking branch 'origin/main' into feat/muxed_keys
willemneal Dec 20, 2024
06932dc
Merge remote-tracking branch 'origin/main' into feat/muxed_keys
willemneal Dec 20, 2024
2241352
Merge remote-tracking branch 'origin/main' into feat/os_keychain
willemneal Dec 20, 2024
03b84e8
Merge remote-tracking branch 'origin/main' into feat/os_keychain
elizabethengelman Dec 20, 2024
24269ce
Fix after merging with main
elizabethengelman Dec 20, 2024
e95fa3a
Apply suggestion from code review
elizabethengelman Dec 20, 2024
95d8b32
Limit key name length
elizabethengelman Dec 23, 2024
01c8753
Update public_key to work with secure storage keys
elizabethengelman Dec 23, 2024
6404da6
Merge remote-tracking branch 'origin/main' into feat/os_keychain
willemneal Jan 6, 2025
085317f
fix(address): remove private key function & use unresolved Address
willemneal Jan 6, 2025
aecc5ac
feat: store seedphrase instead of private key
willemneal Jan 7, 2025
63f247e
fix: clean up
willemneal Jan 7, 2025
73eead7
Merge branch 'feat/os_keychain' into feat/muxed_keys
willemneal Jan 7, 2025
c305d9d
Merge branch 'main' into feat/muxed_keys
elizabethengelman Jan 9, 2025
ed797e0
fix: wait to lookup muxed account
willemneal Jan 9, 2025
70240b2
Merge branch 'main' into feat/muxed_keys
willemneal Jan 9, 2025
5391d3d
Merge remote-tracking branch 'origin/main' into feat/muxed_keys
willemneal Jan 14, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1,721 changes: 960 additions & 761 deletions Cargo.lock

Large diffs are not rendered by default.

141 changes: 123 additions & 18 deletions FULL_HELP_DOCS.md

Large diffs are not rendered by default.

92 changes: 92 additions & 0 deletions cmd/crates/soroban-test/tests/it/integration/tx/operations.rs
Original file line number Diff line number Diff line change
Expand Up @@ -89,6 +89,71 @@ async fn create_account() {
invoke_hello_world(sandbox, &id);
}

#[tokio::test]
async fn create_account_with_alias() {
let sandbox = &TestEnv::new();
sandbox
.new_assert_cmd("keys")
.args(["generate", "--no-fund", "new"])
.assert()
.success();
let test = test_address(sandbox);
let client = sandbox.client();
let test_account = client.get_account(&test).await.unwrap();
println!("test account has a balance of {}", test_account.balance);
let starting_balance = ONE_XLM * 100;
sandbox
.new_assert_cmd("tx")
.args([
"new",
"create-account",
"--destination",
"new",
"--starting-balance",
starting_balance.to_string().as_str(),
])
.assert()
.success();
let test_account_after = client.get_account(&test).await.unwrap();
assert!(test_account_after.balance < test_account.balance);
let id = deploy_contract(sandbox, HELLO_WORLD, DeployKind::Normal, Some("new")).await;
println!("{id}");
invoke_hello_world(sandbox, &id);
}

#[tokio::test]
async fn payment_with_alias() {
let sandbox = &TestEnv::new();
let client = sandbox.client();
let (test, test1) = setup_accounts(sandbox);
let test_account = client.get_account(&test).await.unwrap();
println!("test account has a balance of {}", test_account.balance);

let before = client.get_account(&test).await.unwrap();
let test1_account_entry_before = client.get_account(&test1).await.unwrap();

sandbox
.new_assert_cmd("tx")
.args([
"new",
"payment",
"--destination",
"test1",
"--amount",
ONE_XLM.to_string().as_str(),
])
.assert()
.success();
let test1_account_entry = client.get_account(&test1).await.unwrap();
assert_eq!(
ONE_XLM,
test1_account_entry.balance - test1_account_entry_before.balance,
"Should have One XLM more"
);
let after = client.get_account(&test).await.unwrap();
assert_eq!(before.balance - 10_000_100, after.balance);
}

#[tokio::test]
async fn payment() {
let sandbox = &TestEnv::new();
Expand Down Expand Up @@ -172,6 +237,33 @@ async fn account_merge() {
assert_eq!(before.balance + before1.balance - fee, after.balance);
}

#[tokio::test]
async fn account_merge_with_alias() {
let sandbox = &TestEnv::new();
let client = sandbox.client();
let (test, test1) = setup_accounts(sandbox);
let before = client.get_account(&test).await.unwrap();
let before1 = client.get_account(&test1).await.unwrap();
let fee = 100;
sandbox
.new_assert_cmd("tx")
.args([
"new",
"account-merge",
"--source",
"test1",
"--account",
"test",
"--fee",
fee.to_string().as_str(),
])
.assert()
.success();
let after = client.get_account(&test).await.unwrap();
assert!(client.get_account(&test1).await.is_err());
assert_eq!(before.balance + before1.balance - fee, after.balance);
}

#[tokio::test]
async fn set_trustline_flags() {
let sandbox = &TestEnv::new();
Expand Down
1 change: 1 addition & 0 deletions cmd/soroban-cli/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -128,6 +128,7 @@ wasm-gen = "0.1.4"
zeroize = "1.8.1"
keyring = { version = "3", features = ["apple-native", "windows-native", "sync-secret-service"] }
whoami = "1.5.2"
serde_with = "3.11.0"


[build-dependencies]
Expand Down
8 changes: 7 additions & 1 deletion cmd/soroban-cli/src/commands/contract/arg_parsing.rs
Original file line number Diff line number Diff line change
Expand Up @@ -269,5 +269,11 @@ fn resolve_address(addr_or_alias: &str, config: &config::Args) -> Result<String,
}

fn resolve_signer(addr_or_alias: &str, config: &config::Args) -> Option<SigningKey> {
config.locator.key(addr_or_alias).ok()?.key_pair(None).ok()
config
.locator
.read_key(addr_or_alias)
.ok()?
.private_key(None)
.ok()
.map(|pk| SigningKey::from_bytes(&pk.0))
}
17 changes: 13 additions & 4 deletions cmd/soroban-cli/src/commands/keys/add.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,15 +2,16 @@ use clap::command;

use crate::{
commands::global,
config::{address::KeyName, locator, secret},
config::{address::KeyName, key, locator, secret},
print::Print,
};

#[derive(thiserror::Error, Debug)]
pub enum Error {
#[error(transparent)]
Secret(#[from] secret::Error),

#[error(transparent)]
Key(#[from] key::Error),
#[error(transparent)]
Config(#[from] locator::Error),
}
Expand All @@ -26,13 +27,21 @@ pub struct Cmd {

#[command(flatten)]
pub config_locator: locator::Args,

/// Add a public key, ed25519, or muxed account, e.g. G1.., M2..
#[arg(long, conflicts_with = "seed_phrase", conflicts_with = "secret_key")]
pub public_key: Option<String>,
}

impl Cmd {
pub fn run(&self, global_args: &global::Args) -> Result<(), Error> {
let key = if let Some(key) = self.public_key.as_ref() {
key.parse()?
} else {
self.secrets.read_secret()?.into()
};
let print = Print::new(global_args.quiet);
let secret = self.secrets.read_secret()?;
let path = self.config_locator.write_identity(&self.name, &secret)?;
let path = self.config_locator.write_key(&self.name, &key)?;
print.checkln(format!("Key saved with alias {:?} in {path:?}", self.name));
Ok(())
}
Expand Down
9 changes: 5 additions & 4 deletions cmd/soroban-cli/src/commands/keys/generate.rs
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ use super::super::config::{
locator, network,
secret::{self, Secret},
};

use crate::{
commands::global,
config::address::KeyName,
Expand Down Expand Up @@ -178,7 +179,7 @@ impl Cmd {

#[cfg(test)]
mod tests {
use crate::config::{address::KeyName, secret::Secret};
use crate::config::{address::KeyName, key::Key, secret::Secret};
use keyring::{mock, set_default_credential_builder};

fn set_up_test() -> (super::locator::Args, super::Cmd) {
Expand Down Expand Up @@ -220,7 +221,7 @@ mod tests {
let result = cmd.run(&global_args).await;
assert!(result.is_ok());
let identity = test_locator.read_identity("test_name").unwrap();
assert!(matches!(identity, Secret::SeedPhrase { .. }));
assert!(matches!(identity, Key::Secret(Secret::SeedPhrase { .. })));
}

#[tokio::test]
Expand All @@ -232,7 +233,7 @@ mod tests {
let result = cmd.run(&global_args).await;
assert!(result.is_ok());
let identity = test_locator.read_identity("test_name").unwrap();
assert!(matches!(identity, Secret::SecretKey { .. }));
assert!(matches!(identity, Key::Secret(Secret::SecretKey { .. })));
}

#[tokio::test]
Expand All @@ -245,6 +246,6 @@ mod tests {
let result = cmd.run(&global_args).await;
assert!(result.is_ok());
let identity = test_locator.read_identity("test_name").unwrap();
assert!(matches!(identity, Secret::SecureStore { .. }));
assert!(matches!(identity, Key::Secret(Secret::SecureStore { .. })));
}
}
7 changes: 2 additions & 5 deletions cmd/soroban-cli/src/commands/keys/secret.rs
Original file line number Diff line number Diff line change
@@ -1,17 +1,14 @@
use clap::arg;

use crate::config::{locator, secret};
use crate::config::{key, locator};

#[derive(thiserror::Error, Debug)]
pub enum Error {
#[error(transparent)]
Config(#[from] locator::Error),

#[error(transparent)]
Secret(#[from] secret::Error),

#[error(transparent)]
StrKey(#[from] stellar_strkey::DecodeError),
Key(#[from] key::Error),
}

#[derive(Debug, clap::Parser, Clone)]
Expand Down
44 changes: 42 additions & 2 deletions cmd/soroban-cli/src/commands/tx/args.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,10 @@
use crate::{
commands::{global, txn_result::TxnEnvelopeResult},
config::{self, data, network, secret},
config::{
self,
address::{self, UnresolvedMuxedAccount},
data, network, secret,
},
fee,
rpc::{self, Client, GetTransactionResponse},
tx::builder::{self, TxExt},
Expand Down Expand Up @@ -32,6 +36,10 @@ pub enum Error {
Data(#[from] data::Error),
#[error(transparent)]
Xdr(#[from] xdr::Error),
#[error(transparent)]
Address(#[from] address::Error),
#[error(transparent)]
TxXdr(#[from] super::xdr::Error),
}

impl Args {
Expand Down Expand Up @@ -64,7 +72,7 @@ impl Args {
op: impl Into<xdr::OperationBody>,
global_args: &global::Args,
) -> Result<TxnEnvelopeResult<GetTransactionResponse>, Error> {
let tx = self.tx(op.into()).await?;
let tx = self.tx(op).await?;
self.handle_tx(tx, global_args).await
}
pub async fn handle_and_print(
Expand Down Expand Up @@ -104,4 +112,36 @@ impl Args {
pub fn source_account(&self) -> Result<xdr::MuxedAccount, Error> {
Ok(self.config.source_account()?)
}

pub fn resolve_muxed_address(
&self,
address: &UnresolvedMuxedAccount,
) -> Result<xdr::MuxedAccount, Error> {
Ok(address.resolve_muxed_account(&self.config.locator, self.config.hd_path)?)
}

pub fn resolve_account_id(
&self,
address: &UnresolvedMuxedAccount,
) -> Result<xdr::AccountId, Error> {
Ok(address
.resolve_muxed_account(&self.config.locator, self.config.hd_path)?
.account_id())
}

pub fn add_op(
&self,
op_body: impl Into<xdr::OperationBody>,
tx_env: xdr::TransactionEnvelope,
op_source: Option<&address::UnresolvedMuxedAccount>,
) -> Result<xdr::TransactionEnvelope, Error> {
let source_account = op_source
.map(|a| self.resolve_muxed_address(a))
.transpose()?;
let op = xdr::Operation {
source_account,
body: op_body.into(),
};
Ok(super::xdr::add_op(tx_env, op)?)
}
}
2 changes: 2 additions & 0 deletions cmd/soroban-cli/src/commands/tx/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,8 @@ pub enum Error {
#[error(transparent)]
Sign(#[from] sign::Error),
#[error(transparent)]
Args(#[from] args::Error),
#[error(transparent)]
Simulate(#[from] simulate::Error),
}

Expand Down
13 changes: 8 additions & 5 deletions cmd/soroban-cli/src/commands/tx/new/account_merge.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
use clap::{command, Parser};

use crate::{commands::tx, xdr};
use crate::{commands::tx, config::address, xdr};

#[derive(Parser, Debug, Clone)]
#[group(skip)]
Expand All @@ -15,11 +15,14 @@ pub struct Cmd {
pub struct Args {
/// Muxed Account to merge with, e.g. `GBX...`, 'MBX...'
#[arg(long)]
pub account: xdr::MuxedAccount,
pub account: address::UnresolvedMuxedAccount,
}

impl From<&Args> for xdr::OperationBody {
fn from(cmd: &Args) -> Self {
xdr::OperationBody::AccountMerge(cmd.account.clone())
impl TryFrom<&Cmd> for xdr::OperationBody {
type Error = tx::args::Error;
fn try_from(cmd: &Cmd) -> Result<Self, Self::Error> {
Ok(xdr::OperationBody::AccountMerge(
cmd.tx.resolve_muxed_address(&cmd.op.account)?,
))
}
}
6 changes: 3 additions & 3 deletions cmd/soroban-cli/src/commands/tx/new/bump_sequence.rs
Original file line number Diff line number Diff line change
Expand Up @@ -18,10 +18,10 @@ pub struct Args {
pub bump_to: i64,
}

impl From<&Args> for xdr::OperationBody {
fn from(cmd: &Args) -> Self {
impl From<&Cmd> for xdr::OperationBody {
fn from(cmd: &Cmd) -> Self {
xdr::OperationBody::BumpSequence(xdr::BumpSequenceOp {
bump_to: cmd.bump_to.into(),
bump_to: cmd.op.bump_to.into(),
})
}
}
8 changes: 4 additions & 4 deletions cmd/soroban-cli/src/commands/tx/new/change_trust.rs
Original file line number Diff line number Diff line change
Expand Up @@ -20,16 +20,16 @@ pub struct Args {
pub limit: i64,
}

impl From<&Args> for xdr::OperationBody {
fn from(cmd: &Args) -> Self {
let line = match cmd.line.0.clone() {
impl From<&Cmd> for xdr::OperationBody {
fn from(cmd: &Cmd) -> Self {
let line = match cmd.op.line.0.clone() {
xdr::Asset::CreditAlphanum4(asset) => xdr::ChangeTrustAsset::CreditAlphanum4(asset),
xdr::Asset::CreditAlphanum12(asset) => xdr::ChangeTrustAsset::CreditAlphanum12(asset),
xdr::Asset::Native => xdr::ChangeTrustAsset::Native,
};
xdr::OperationBody::ChangeTrust(xdr::ChangeTrustOp {
line,
limit: cmd.limit,
limit: cmd.op.limit,
})
}
}
Loading
Loading