Refactor SignerSource to expose DerivationPath to other kinds of signers (#16933)

* One use statement

* Add stdin uri scheme

* Convert parse_signer_source to return Result

* A-Z deps

* Convert Usb data to Locator

* Pull DerivationPath out of Locator

* Wrap SignerSource to share derivation_path

* Review comments

* Check Filepath existence, readability in parse_signer_source
This commit is contained in:
Tyera Eulberg 2021-04-29 01:42:21 -06:00 committed by GitHub
parent d640ac143b
commit d6f30b7537
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
15 changed files with 509 additions and 447 deletions

3
Cargo.lock generated
View File

@ -4107,6 +4107,7 @@ dependencies = [
"rpassword", "rpassword",
"solana-remote-wallet", "solana-remote-wallet",
"solana-sdk", "solana-sdk",
"tempfile",
"thiserror", "thiserror",
"tiny-bip39 0.8.0", "tiny-bip39 0.8.0",
"uriparse", "uriparse",
@ -5102,6 +5103,7 @@ dependencies = [
"num-derive", "num-derive",
"num-traits", "num-traits",
"pbkdf2 0.6.0", "pbkdf2 0.6.0",
"qstring",
"rand 0.7.3", "rand 0.7.3",
"rand_chacha 0.2.2", "rand_chacha 0.2.2",
"rand_core 0.6.2", "rand_core 0.6.2",
@ -5121,6 +5123,7 @@ dependencies = [
"solana-sdk-macro 1.7.0", "solana-sdk-macro 1.7.0",
"thiserror", "thiserror",
"tiny-bip39 0.7.3", "tiny-bip39 0.7.3",
"uriparse",
] ]
[[package]] [[package]]

View File

@ -20,6 +20,9 @@ uriparse = "0.6.3"
url = "2.1.0" url = "2.1.0"
chrono = "0.4" chrono = "0.4"
[dev-dependencies]
tempfile = "3.1.0"
[lib] [lib]
name = "solana_clap_utils" name = "solana_clap_utils"

View File

@ -1,5 +1,7 @@
use crate::{input_validators, ArgConstant}; use {
use clap::Arg; crate::{input_validators, ArgConstant},
clap::Arg,
};
pub const FEE_PAYER_ARG: ArgConstant<'static> = ArgConstant { pub const FEE_PAYER_ARG: ArgConstant<'static> = ArgConstant {
name: "fee_payer", name: "fee_payer",

View File

@ -1,19 +1,21 @@
use crate::keypair::{ use {
crate::keypair::{
keypair_from_seed_phrase, pubkey_from_path, resolve_signer_from_path, signer_from_path, keypair_from_seed_phrase, pubkey_from_path, resolve_signer_from_path, signer_from_path,
ASK_KEYWORD, SKIP_SEED_PHRASE_VALIDATION_ARG, ASK_KEYWORD, SKIP_SEED_PHRASE_VALIDATION_ARG,
}; },
use chrono::DateTime; chrono::DateTime,
use clap::ArgMatches; clap::ArgMatches,
use solana_remote_wallet::remote_wallet::RemoteWalletManager; solana_remote_wallet::remote_wallet::RemoteWalletManager,
use solana_sdk::{ solana_sdk::{
clock::UnixTimestamp, clock::UnixTimestamp,
commitment_config::CommitmentConfig, commitment_config::CommitmentConfig,
genesis_config::ClusterType, genesis_config::ClusterType,
native_token::sol_to_lamports, native_token::sol_to_lamports,
pubkey::Pubkey, pubkey::Pubkey,
signature::{read_keypair_file, Keypair, Signature, Signer}, signature::{read_keypair_file, Keypair, Signature, Signer},
},
std::{str::FromStr, sync::Arc},
}; };
use std::{str::FromStr, sync::Arc};
// Return parsed values from matches at `name` // Return parsed values from matches at `name`
pub fn values_of<T>(matches: &ArgMatches<'_>, name: &str) -> Option<Vec<T>> pub fn values_of<T>(matches: &ArgMatches<'_>, name: &str) -> Option<Vec<T>>

View File

@ -1,13 +1,15 @@
use crate::keypair::{parse_signer_source, SignerSource, ASK_KEYWORD}; use {
use chrono::DateTime; crate::keypair::{parse_signer_source, SignerSourceKind, ASK_KEYWORD},
use solana_sdk::{ chrono::DateTime,
solana_sdk::{
clock::{Epoch, Slot}, clock::{Epoch, Slot},
hash::Hash, hash::Hash,
pubkey::{Pubkey, MAX_SEED_LEN}, pubkey::{Pubkey, MAX_SEED_LEN},
signature::{read_keypair_file, Signature}, signature::{read_keypair_file, Signature},
},
std::fmt::Display,
std::str::FromStr,
}; };
use std::fmt::Display;
use std::str::FromStr;
fn is_parsable_generic<U, T>(string: T) -> Result<(), String> fn is_parsable_generic<U, T>(string: T) -> Result<(), String>
where where
@ -108,8 +110,11 @@ pub fn is_valid_pubkey<T>(string: T) -> Result<(), String>
where where
T: AsRef<str> + Display, T: AsRef<str> + Display,
{ {
match parse_signer_source(string.as_ref()) { match parse_signer_source(string.as_ref())
SignerSource::Filepath(path) => is_keypair(path), .map_err(|err| format!("{}", err))?
.kind
{
SignerSourceKind::Filepath(path) => is_keypair(path),
_ => Ok(()), _ => Ok(()),
} }
} }

View File

@ -1,16 +1,19 @@
use crate::{ use {
crate::{
input_parsers::pubkeys_sigs_of, input_parsers::pubkeys_sigs_of,
offline::{SIGNER_ARG, SIGN_ONLY_ARG}, offline::{SIGNER_ARG, SIGN_ONLY_ARG},
ArgConstant, ArgConstant,
}; },
use bip39::{Language, Mnemonic, Seed}; bip39::{Language, Mnemonic, Seed},
use clap::ArgMatches; clap::ArgMatches,
use rpassword::prompt_password_stderr; rpassword::prompt_password_stderr,
use solana_remote_wallet::{ solana_remote_wallet::{
locator::{Locator as RemoteWalletLocator, LocatorError as RemoteWalletLocatorError},
remote_keypair::generate_remote_keypair, remote_keypair::generate_remote_keypair,
remote_wallet::{maybe_wallet_manager, RemoteWalletError, RemoteWalletManager}, remote_wallet::{maybe_wallet_manager, RemoteWalletError, RemoteWalletManager},
}; },
use solana_sdk::{ solana_sdk::{
derivation_path::{DerivationPath, DerivationPathError},
hash::Hash, hash::Hash,
message::Message, message::Message,
pubkey::Pubkey, pubkey::Pubkey,
@ -18,14 +21,16 @@ use solana_sdk::{
keypair_from_seed, keypair_from_seed_phrase_and_passphrase, read_keypair, keypair_from_seed, keypair_from_seed_phrase_and_passphrase, read_keypair,
read_keypair_file, Keypair, NullSigner, Presigner, Signature, Signer, read_keypair_file, Keypair, NullSigner, Presigner, Signature, Signer,
}, },
}; },
use std::{ std::{
convert::TryFrom, convert::TryFrom,
error, error,
io::{stdin, stdout, Write}, io::{stdin, stdout, Write},
process::exit, process::exit,
str::FromStr, str::FromStr,
sync::Arc, sync::Arc,
},
thiserror::Error,
}; };
pub struct SignOnly { pub struct SignOnly {
@ -132,34 +137,72 @@ impl DefaultSigner {
} }
} }
pub(crate) enum SignerSource { pub(crate) struct SignerSource {
pub kind: SignerSourceKind,
pub derivation_path: Option<DerivationPath>,
}
impl SignerSource {
fn new(kind: SignerSourceKind) -> Self {
Self {
kind,
derivation_path: None,
}
}
}
pub(crate) enum SignerSourceKind {
Ask, Ask,
Filepath(String), Filepath(String),
Usb(String), Usb(RemoteWalletLocator),
Stdin, Stdin,
Pubkey(Pubkey), Pubkey(Pubkey),
} }
pub(crate) fn parse_signer_source<S: AsRef<str>>(source: S) -> SignerSource { #[derive(Debug, Error)]
pub(crate) enum SignerSourceError {
#[error("unrecognized signer source")]
UnrecognizedSource,
#[error(transparent)]
RemoteWalletLocatorError(#[from] RemoteWalletLocatorError),
#[error(transparent)]
DerivationPathError(#[from] DerivationPathError),
#[error(transparent)]
IoError(#[from] std::io::Error),
}
pub(crate) fn parse_signer_source<S: AsRef<str>>(
source: S,
) -> Result<SignerSource, SignerSourceError> {
let source = source.as_ref(); let source = source.as_ref();
match uriparse::URIReference::try_from(source) { match uriparse::URIReference::try_from(source) {
Err(_) => SignerSource::Filepath(source.to_string()), Err(_) => Err(SignerSourceError::UnrecognizedSource),
Ok(uri) => { Ok(uri) => {
if let Some(scheme) = uri.scheme() { if let Some(scheme) = uri.scheme() {
let scheme = scheme.as_str().to_ascii_lowercase(); let scheme = scheme.as_str().to_ascii_lowercase();
match scheme.as_str() { match scheme.as_str() {
"ask" => SignerSource::Ask, "ask" => Ok(SignerSource::new(SignerSourceKind::Ask)),
"file" => SignerSource::Filepath(uri.path().to_string()), "file" => Ok(SignerSource::new(SignerSourceKind::Filepath(
"usb" => SignerSource::Usb(source.to_string()), uri.path().to_string(),
_ => SignerSource::Filepath(source.to_string()), ))),
"stdin" => Ok(SignerSource::new(SignerSourceKind::Stdin)),
"usb" => Ok(SignerSource {
kind: SignerSourceKind::Usb(RemoteWalletLocator::new_from_uri(&uri)?),
derivation_path: DerivationPath::from_uri(&uri)?,
}),
_ => Err(SignerSourceError::UnrecognizedSource),
} }
} else { } else {
match source { match source {
"-" => SignerSource::Stdin, "-" => Ok(SignerSource::new(SignerSourceKind::Stdin)),
ASK_KEYWORD => SignerSource::Ask, ASK_KEYWORD => Ok(SignerSource::new(SignerSourceKind::Ask)),
_ => match Pubkey::from_str(source) { _ => match Pubkey::from_str(source) {
Ok(pubkey) => SignerSource::Pubkey(pubkey), Ok(pubkey) => Ok(SignerSource::new(SignerSourceKind::Pubkey(pubkey))),
Err(_) => SignerSource::Filepath(source.to_string()), Err(_) => std::fs::metadata(source)
.map(|_| {
SignerSource::new(SignerSourceKind::Filepath(source.to_string()))
})
.map_err(|err| err.into()),
}, },
} }
} }
@ -210,8 +253,12 @@ pub fn signer_from_path_with_config(
wallet_manager: &mut Option<Arc<RemoteWalletManager>>, wallet_manager: &mut Option<Arc<RemoteWalletManager>>,
config: &SignerFromPathConfig, config: &SignerFromPathConfig,
) -> Result<Box<dyn Signer>, Box<dyn error::Error>> { ) -> Result<Box<dyn Signer>, Box<dyn error::Error>> {
match parse_signer_source(path) { let SignerSource {
SignerSource::Ask => { kind,
derivation_path,
} = parse_signer_source(path)?;
match kind {
SignerSourceKind::Ask => {
let skip_validation = matches.is_present(SKIP_SEED_PHRASE_VALIDATION_ARG.name); let skip_validation = matches.is_present(SKIP_SEED_PHRASE_VALIDATION_ARG.name);
Ok(Box::new(keypair_from_seed_phrase( Ok(Box::new(keypair_from_seed_phrase(
keypair_name, keypair_name,
@ -219,7 +266,7 @@ pub fn signer_from_path_with_config(
false, false,
)?)) )?))
} }
SignerSource::Filepath(path) => match read_keypair_file(&path) { SignerSourceKind::Filepath(path) => match read_keypair_file(&path) {
Err(e) => Err(std::io::Error::new( Err(e) => Err(std::io::Error::new(
std::io::ErrorKind::Other, std::io::ErrorKind::Other,
format!("could not read keypair file \"{}\". Run \"solana-keygen new\" to create a keypair file: {}", path, e), format!("could not read keypair file \"{}\". Run \"solana-keygen new\" to create a keypair file: {}", path, e),
@ -227,17 +274,18 @@ pub fn signer_from_path_with_config(
.into()), .into()),
Ok(file) => Ok(Box::new(file)), Ok(file) => Ok(Box::new(file)),
}, },
SignerSource::Stdin => { SignerSourceKind::Stdin => {
let mut stdin = std::io::stdin(); let mut stdin = std::io::stdin();
Ok(Box::new(read_keypair(&mut stdin)?)) Ok(Box::new(read_keypair(&mut stdin)?))
} }
SignerSource::Usb(path) => { SignerSourceKind::Usb(locator) => {
if wallet_manager.is_none() { if wallet_manager.is_none() {
*wallet_manager = maybe_wallet_manager()?; *wallet_manager = maybe_wallet_manager()?;
} }
if let Some(wallet_manager) = wallet_manager { if let Some(wallet_manager) = wallet_manager {
Ok(Box::new(generate_remote_keypair( Ok(Box::new(generate_remote_keypair(
path, locator,
derivation_path.unwrap_or_default(),
wallet_manager, wallet_manager,
matches.is_present("confirm_key"), matches.is_present("confirm_key"),
keypair_name, keypair_name,
@ -246,7 +294,7 @@ pub fn signer_from_path_with_config(
Err(RemoteWalletError::NoDeviceFound.into()) Err(RemoteWalletError::NoDeviceFound.into())
} }
} }
SignerSource::Pubkey(pubkey) => { SignerSourceKind::Pubkey(pubkey) => {
let presigner = pubkeys_sigs_of(matches, SIGNER_ARG.name) let presigner = pubkeys_sigs_of(matches, SIGNER_ARG.name)
.as_ref() .as_ref()
.and_then(|presigners| presigner_from_pubkey_sigs(&pubkey, presigners)); .and_then(|presigners| presigner_from_pubkey_sigs(&pubkey, presigners));
@ -271,8 +319,9 @@ pub fn pubkey_from_path(
keypair_name: &str, keypair_name: &str,
wallet_manager: &mut Option<Arc<RemoteWalletManager>>, wallet_manager: &mut Option<Arc<RemoteWalletManager>>,
) -> Result<Pubkey, Box<dyn error::Error>> { ) -> Result<Pubkey, Box<dyn error::Error>> {
match parse_signer_source(path) { let SignerSource { kind, .. } = parse_signer_source(path)?;
SignerSource::Pubkey(pubkey) => Ok(pubkey), match kind {
SignerSourceKind::Pubkey(pubkey) => Ok(pubkey),
_ => Ok(signer_from_path(matches, path, keypair_name, wallet_manager)?.pubkey()), _ => Ok(signer_from_path(matches, path, keypair_name, wallet_manager)?.pubkey()),
} }
} }
@ -283,14 +332,18 @@ pub fn resolve_signer_from_path(
keypair_name: &str, keypair_name: &str,
wallet_manager: &mut Option<Arc<RemoteWalletManager>>, wallet_manager: &mut Option<Arc<RemoteWalletManager>>,
) -> Result<Option<String>, Box<dyn error::Error>> { ) -> Result<Option<String>, Box<dyn error::Error>> {
match parse_signer_source(path) { let SignerSource {
SignerSource::Ask => { kind,
derivation_path,
} = parse_signer_source(path)?;
match kind {
SignerSourceKind::Ask => {
let skip_validation = matches.is_present(SKIP_SEED_PHRASE_VALIDATION_ARG.name); let skip_validation = matches.is_present(SKIP_SEED_PHRASE_VALIDATION_ARG.name);
// This method validates the seed phrase, but returns `None` because there is no path // This method validates the seed phrase, but returns `None` because there is no path
// on disk or to a device // on disk or to a device
keypair_from_seed_phrase(keypair_name, skip_validation, false).map(|_| None) keypair_from_seed_phrase(keypair_name, skip_validation, false).map(|_| None)
} }
SignerSource::Filepath(path) => match read_keypair_file(&path) { SignerSourceKind::Filepath(path) => match read_keypair_file(&path) {
Err(e) => Err(std::io::Error::new( Err(e) => Err(std::io::Error::new(
std::io::ErrorKind::Other, std::io::ErrorKind::Other,
format!("could not read keypair file \"{}\". Run \"solana-keygen new\" to create a keypair file: {}", path, e), format!("could not read keypair file \"{}\". Run \"solana-keygen new\" to create a keypair file: {}", path, e),
@ -298,19 +351,20 @@ pub fn resolve_signer_from_path(
.into()), .into()),
Ok(_) => Ok(Some(path.to_string())), Ok(_) => Ok(Some(path.to_string())),
}, },
SignerSource::Stdin => { SignerSourceKind::Stdin => {
let mut stdin = std::io::stdin(); let mut stdin = std::io::stdin();
// This method validates the keypair from stdin, but returns `None` because there is no // This method validates the keypair from stdin, but returns `None` because there is no
// path on disk or to a device // path on disk or to a device
read_keypair(&mut stdin).map(|_| None) read_keypair(&mut stdin).map(|_| None)
} }
SignerSource::Usb(path) => { SignerSourceKind::Usb(locator) => {
if wallet_manager.is_none() { if wallet_manager.is_none() {
*wallet_manager = maybe_wallet_manager()?; *wallet_manager = maybe_wallet_manager()?;
} }
if let Some(wallet_manager) = wallet_manager { if let Some(wallet_manager) = wallet_manager {
let path = generate_remote_keypair( let path = generate_remote_keypair(
path, locator,
derivation_path.unwrap_or_default(),
wallet_manager, wallet_manager,
matches.is_present("confirm_key"), matches.is_present("confirm_key"),
keypair_name, keypair_name,
@ -414,7 +468,9 @@ fn sanitize_seed_phrase(seed_phrase: &str) -> String {
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::*; use super::*;
use solana_remote_wallet::locator::Manufacturer;
use solana_sdk::system_instruction; use solana_sdk::system_instruction;
use tempfile::NamedTempFile;
#[test] #[test]
fn test_sanitize_seed_phrase() { fn test_sanitize_seed_phrase() {
@ -458,35 +514,108 @@ mod tests {
#[test] #[test]
fn test_parse_signer_source() { fn test_parse_signer_source() {
assert!(matches!(parse_signer_source("-"), SignerSource::Stdin));
assert!(matches!( assert!(matches!(
parse_signer_source(ASK_KEYWORD), parse_signer_source("-").unwrap(),
SignerSource::Ask SignerSource {
kind: SignerSourceKind::Stdin,
derivation_path: None,
}
));
let ask = "stdin:".to_string();
assert!(matches!(
parse_signer_source(&ask).unwrap(),
SignerSource {
kind: SignerSourceKind::Stdin,
derivation_path: None,
}
));
assert!(matches!(
parse_signer_source(ASK_KEYWORD).unwrap(),
SignerSource {
kind: SignerSourceKind::Ask,
derivation_path: None,
}
)); ));
let pubkey = Pubkey::new_unique(); let pubkey = Pubkey::new_unique();
assert!( assert!(
matches!(parse_signer_source(&pubkey.to_string()), SignerSource::Pubkey(p) if p == pubkey) matches!(parse_signer_source(&pubkey.to_string()).unwrap(), SignerSource {
kind: SignerSourceKind::Pubkey(p),
derivation_path: None,
}
if p == pubkey)
); );
let path = "/absolute/path".to_string();
assert!(matches!(parse_signer_source(&path), SignerSource::Filepath(p) if p == path)); // Set up absolute and relative path strs
let path = "relative/path".to_string(); let file0 = NamedTempFile::new().unwrap();
assert!(matches!(parse_signer_source(&path), SignerSource::Filepath(p) if p == path)); let path = file0.path();
assert!(path.is_absolute());
let absolute_path_str = path.to_str().unwrap();
let file1 = NamedTempFile::new_in(std::env::current_dir().unwrap()).unwrap();
let path = file1.path().file_name().unwrap().to_str().unwrap();
let path = std::path::Path::new(path);
assert!(path.is_relative());
let relative_path_str = path.to_str().unwrap();
assert!(
matches!(parse_signer_source(absolute_path_str).unwrap(), SignerSource {
kind: SignerSourceKind::Filepath(p),
derivation_path: None,
} if p == absolute_path_str)
);
assert!(
matches!(parse_signer_source(&relative_path_str).unwrap(), SignerSource {
kind: SignerSourceKind::Filepath(p),
derivation_path: None,
} if p == relative_path_str)
);
let usb = "usb://ledger".to_string(); let usb = "usb://ledger".to_string();
assert!(matches!(parse_signer_source(&usb), SignerSource::Usb(u) if u == usb)); let expected_locator = RemoteWalletLocator {
// Catchall into SignerSource::Filepath manufacturer: Manufacturer::Ledger,
let junk = "sometextthatisnotapubkey".to_string(); pubkey: None,
};
assert!(matches!(parse_signer_source(&usb).unwrap(), SignerSource {
kind: SignerSourceKind::Usb(u),
derivation_path: None,
} if u == expected_locator));
let usb = "usb://ledger?key=0/0".to_string();
let expected_locator = RemoteWalletLocator {
manufacturer: Manufacturer::Ledger,
pubkey: None,
};
let expected_derivation_path = Some(DerivationPath::new_bip44(Some(0), Some(0)));
assert!(matches!(parse_signer_source(&usb).unwrap(), SignerSource {
kind: SignerSourceKind::Usb(u),
derivation_path: d,
} if u == expected_locator && d == expected_derivation_path));
// Catchall into SignerSource::Filepath fails
let junk = "sometextthatisnotapubkeyorfile".to_string();
assert!(Pubkey::from_str(&junk).is_err()); assert!(Pubkey::from_str(&junk).is_err());
assert!(matches!(parse_signer_source(&junk), SignerSource::Filepath(j) if j == junk)); assert!(matches!(
parse_signer_source(&junk),
Err(SignerSourceError::IoError(_))
));
let ask = "ask:".to_string(); let ask = "ask:".to_string();
assert!(matches!(parse_signer_source(&ask), SignerSource::Ask)); assert!(matches!(
let path = "/absolute/path".to_string(); parse_signer_source(&ask).unwrap(),
SignerSource {
kind: SignerSourceKind::Ask,
derivation_path: None,
}
));
assert!( assert!(
matches!(parse_signer_source(&format!("file:{}", path)), SignerSource::Filepath(p) if p == path) matches!(parse_signer_source(&format!("file:{}", absolute_path_str)).unwrap(), SignerSource {
kind: SignerSourceKind::Filepath(p),
derivation_path: None,
} if p == absolute_path_str)
); );
let path = "relative/path".to_string();
assert!( assert!(
matches!(parse_signer_source(&format!("file:{}", path)), SignerSource::Filepath(p) if p == path) matches!(parse_signer_source(&format!("file:{}", relative_path_str)).unwrap(), SignerSource {
kind: SignerSourceKind::Filepath(p),
derivation_path: None,
} if p == relative_path_str)
); );
} }
} }

View File

@ -1,5 +1,4 @@
use crate::ArgConstant; use {crate::ArgConstant, clap::Arg};
use clap::Arg;
pub const MEMO_ARG: ArgConstant<'static> = ArgConstant { pub const MEMO_ARG: ArgConstant<'static> = ArgConstant {
name: "memo", name: "memo",

View File

@ -1,5 +1,7 @@
use crate::{input_validators::*, offline::BLOCKHASH_ARG, ArgConstant}; use {
use clap::{App, Arg}; crate::{input_validators::*, offline::BLOCKHASH_ARG, ArgConstant},
clap::{App, Arg},
};
pub const NONCE_ARG: ArgConstant<'static> = ArgConstant { pub const NONCE_ARG: ArgConstant<'static> = ArgConstant {
name: "nonce", name: "nonce",

View File

@ -1,5 +1,7 @@
use crate::{input_validators::*, ArgConstant}; use {
use clap::{App, Arg}; crate::{input_validators::*, ArgConstant},
clap::{App, Arg},
};
pub const BLOCKHASH_ARG: ArgConstant<'static> = ArgConstant { pub const BLOCKHASH_ARG: ArgConstant<'static> = ArgConstant {
name: "blockhash", name: "blockhash",

View File

@ -3461,6 +3461,7 @@ dependencies = [
"num-derive 0.3.0", "num-derive 0.3.0",
"num-traits", "num-traits",
"pbkdf2 0.6.0", "pbkdf2 0.6.0",
"qstring",
"rand 0.7.3", "rand 0.7.3",
"rand_chacha 0.2.2", "rand_chacha 0.2.2",
"rand_core 0.6.2", "rand_core 0.6.2",
@ -3479,6 +3480,7 @@ dependencies = [
"solana-program 1.7.0", "solana-program 1.7.0",
"solana-sdk-macro 1.7.0", "solana-sdk-macro 1.7.0",
"thiserror", "thiserror",
"uriparse",
] ]
[[package]] [[package]]

View File

@ -1,8 +1,5 @@
use { use {
solana_sdk::{ solana_sdk::pubkey::{ParsePubkeyError, Pubkey},
derivation_path::{DerivationPath, DerivationPathError},
pubkey::{ParsePubkeyError, Pubkey},
},
std::{ std::{
convert::{Infallible, TryFrom, TryInto}, convert::{Infallible, TryFrom, TryInto},
str::FromStr, str::FromStr,
@ -77,8 +74,6 @@ pub enum LocatorError {
#[error(transparent)] #[error(transparent)]
PubkeyError(#[from] ParsePubkeyError), PubkeyError(#[from] ParsePubkeyError),
#[error(transparent)] #[error(transparent)]
DerivationPathError(#[from] DerivationPathError),
#[error(transparent)]
UriReferenceError(#[from] URIReferenceError), UriReferenceError(#[from] URIReferenceError),
#[error("unimplemented scheme")] #[error("unimplemented scheme")]
UnimplementedScheme, UnimplementedScheme,
@ -96,15 +91,12 @@ impl From<Infallible> for LocatorError {
pub struct Locator { pub struct Locator {
pub manufacturer: Manufacturer, pub manufacturer: Manufacturer,
pub pubkey: Option<Pubkey>, pub pubkey: Option<Pubkey>,
pub derivation_path: Option<DerivationPath>,
} }
impl std::fmt::Display for Locator { impl std::fmt::Display for Locator {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result { fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
let maybe_path = self.pubkey.map(|p| p.to_string()); let maybe_path = self.pubkey.map(|p| p.to_string());
let path = maybe_path.as_deref().unwrap_or("/"); let path = maybe_path.as_deref().unwrap_or("/");
let maybe_query = self.derivation_path.as_ref().map(|d| d.get_query());
let maybe_query2 = maybe_query.as_ref().map(|q| &q[1..]);
let mut builder = URIReferenceBuilder::new(); let mut builder = URIReferenceBuilder::new();
builder builder
@ -113,8 +105,6 @@ impl std::fmt::Display for Locator {
.try_authority(Some(self.manufacturer.as_ref())) .try_authority(Some(self.manufacturer.as_ref()))
.unwrap() .unwrap()
.try_path(path) .try_path(path)
.unwrap()
.try_query(maybe_query2)
.unwrap(); .unwrap();
let uri = builder.build().unwrap(); let uri = builder.build().unwrap();
@ -141,28 +131,7 @@ impl Locator {
None None
} }
}); });
let key = if let Some(query) = uri.query() { Self::new_from_parts(host.as_str(), path)
let query_str = query.as_str();
let query = qstring::QString::from(query_str);
if query.len() > 1 {
return Err(DerivationPathError::InvalidDerivationPath(
"invalid query string, extra fields not supported".to_string(),
)
.into());
}
let key = query.get("key");
if key.is_none() {
return Err(DerivationPathError::InvalidDerivationPath(format!(
"invalid query string `{}`, only `key` supported",
query_str,
))
.into());
}
key.map(|v| v.to_string())
} else {
None
};
Self::new_from_parts(host.as_str(), path, key.as_deref())
} }
(Some(_scheme), Some(_host)) => Err(LocatorError::UnimplementedScheme), (Some(_scheme), Some(_host)) => Err(LocatorError::UnimplementedScheme),
(None, Some(_host)) => Err(LocatorError::UnimplementedScheme), (None, Some(_host)) => Err(LocatorError::UnimplementedScheme),
@ -170,18 +139,15 @@ impl Locator {
} }
} }
pub fn new_from_parts<V, VE, P, PE, D, DE>( pub fn new_from_parts<V, VE, P, PE>(
manufacturer: V, manufacturer: V,
pubkey: Option<P>, pubkey: Option<P>,
derivation_path: Option<D>,
) -> Result<Self, LocatorError> ) -> Result<Self, LocatorError>
where where
VE: Into<LocatorError>, VE: Into<LocatorError>,
V: TryInto<Manufacturer, Error = VE>, V: TryInto<Manufacturer, Error = VE>,
PE: Into<LocatorError>, PE: Into<LocatorError>,
P: TryInto<Pubkey, Error = PE>, P: TryInto<Pubkey, Error = PE>,
DE: Into<LocatorError>,
D: TryInto<DerivationPath, Error = DE>,
{ {
let manufacturer = manufacturer.try_into().map_err(|e| e.into())?; let manufacturer = manufacturer.try_into().map_err(|e| e.into())?;
let pubkey = if let Some(pubkey) = pubkey { let pubkey = if let Some(pubkey) = pubkey {
@ -189,15 +155,9 @@ impl Locator {
} else { } else {
None None
}; };
let derivation_path = if let Some(derivation_path) = derivation_path {
Some(derivation_path.try_into().map_err(|e| e.into())?)
} else {
None
};
Ok(Self { Ok(Self {
manufacturer, manufacturer,
pubkey, pubkey,
derivation_path,
}) })
} }
} }
@ -225,76 +185,50 @@ mod tests {
let manufacturer_str = "ledger"; let manufacturer_str = "ledger";
let pubkey = Pubkey::new_unique(); let pubkey = Pubkey::new_unique();
let pubkey_str = pubkey.to_string(); let pubkey_str = pubkey.to_string();
let derivation_path = DerivationPath::new_bip44(Some(0), Some(0));
let derivation_path_str = "0/0";
let expect = Locator { let expect = Locator {
manufacturer, manufacturer,
pubkey: None, pubkey: None,
derivation_path: None,
}; };
assert!(matches!( assert!(matches!(
Locator::new_from_parts(manufacturer, None::<Pubkey>, None::<DerivationPath>), Locator::new_from_parts(manufacturer, None::<Pubkey>),
Ok(e) if e == expect, Ok(e) if e == expect,
)); ));
assert!(matches!( assert!(matches!(
Locator::new_from_parts(manufacturer_str, None::<Pubkey>, None::<DerivationPath>), Locator::new_from_parts(manufacturer_str, None::<Pubkey>),
Ok(e) if e == expect, Ok(e) if e == expect,
)); ));
let expect = Locator { let expect = Locator {
manufacturer, manufacturer,
pubkey: Some(pubkey), pubkey: Some(pubkey),
derivation_path: None,
}; };
assert!(matches!( assert!(matches!(
Locator::new_from_parts(manufacturer, Some(pubkey), None::<DerivationPath>), Locator::new_from_parts(manufacturer, Some(pubkey)),
Ok(e) if e == expect, Ok(e) if e == expect,
)); ));
assert!(matches!( assert!(matches!(
Locator::new_from_parts(manufacturer_str, Some(pubkey_str.as_str()), None::<DerivationPath>), Locator::new_from_parts(manufacturer_str, Some(pubkey_str.as_str())),
Ok(e) if e == expect,
));
let expect = Locator {
manufacturer,
pubkey: None,
derivation_path: Some(derivation_path.clone()),
};
assert!(matches!(
Locator::new_from_parts(manufacturer, None::<Pubkey>, Some(derivation_path)),
Ok(e) if e == expect,
));
assert!(matches!(
Locator::new_from_parts(manufacturer, None::<Pubkey>, Some(derivation_path_str)),
Ok(e) if e == expect, Ok(e) if e == expect,
)); ));
assert!(matches!( assert!(matches!(
Locator::new_from_parts("bad-manufacturer", None::<Pubkey>, None::<DerivationPath>), Locator::new_from_parts("bad-manufacturer", None::<Pubkey>),
Err(LocatorError::ManufacturerError(e)) if e == ManufacturerError, Err(LocatorError::ManufacturerError(e)) if e == ManufacturerError,
)); ));
assert!(matches!( assert!(matches!(
Locator::new_from_parts(manufacturer, Some("bad-pubkey"), None::<DerivationPath>), Locator::new_from_parts(manufacturer, Some("bad-pubkey")),
Err(LocatorError::PubkeyError(e)) if e == ParsePubkeyError::Invalid, Err(LocatorError::PubkeyError(e)) if e == ParsePubkeyError::Invalid,
)); ));
let bad_path = "bad-derivation-path".to_string();
assert!(matches!(
Locator::new_from_parts(manufacturer, None::<Pubkey>, Some(bad_path.as_str())),
Err(LocatorError::DerivationPathError(
DerivationPathError::InvalidDerivationPath(_)
)),
));
} }
#[test] #[test]
fn test_locator_new_from_uri() { fn test_locator_new_from_uri() {
let derivation_path = DerivationPath::new_bip44(Some(0), Some(0));
let manufacturer = Manufacturer::Ledger; let manufacturer = Manufacturer::Ledger;
let pubkey = Pubkey::new_unique(); let pubkey = Pubkey::new_unique();
let pubkey_str = pubkey.to_string(); let pubkey_str = pubkey.to_string();
// usb://ledger/{PUBKEY}?key=0'/0' // usb://ledger/{PUBKEY}?key=0/0
let mut builder = URIReferenceBuilder::new(); let mut builder = URIReferenceBuilder::new();
builder builder
.try_scheme(Some("usb")) .try_scheme(Some("usb"))
@ -309,7 +243,6 @@ mod tests {
let expect = Locator { let expect = Locator {
manufacturer, manufacturer,
pubkey: Some(pubkey), pubkey: Some(pubkey),
derivation_path: Some(derivation_path.clone()),
}; };
assert_eq!(Locator::new_from_uri(&uri), Ok(expect)); assert_eq!(Locator::new_from_uri(&uri), Ok(expect));
@ -326,7 +259,6 @@ mod tests {
let expect = Locator { let expect = Locator {
manufacturer, manufacturer,
pubkey: Some(pubkey), pubkey: Some(pubkey),
derivation_path: None,
}; };
assert_eq!(Locator::new_from_uri(&uri), Ok(expect)); assert_eq!(Locator::new_from_uri(&uri), Ok(expect));
@ -343,7 +275,6 @@ mod tests {
let expect = Locator { let expect = Locator {
manufacturer, manufacturer,
pubkey: None, pubkey: None,
derivation_path: None,
}; };
assert_eq!(Locator::new_from_uri(&uri), Ok(expect)); assert_eq!(Locator::new_from_uri(&uri), Ok(expect));
@ -360,64 +291,6 @@ mod tests {
let expect = Locator { let expect = Locator {
manufacturer, manufacturer,
pubkey: None, pubkey: None,
derivation_path: None,
};
assert_eq!(Locator::new_from_uri(&uri), Ok(expect));
// usb://ledger?key=0/0
let mut builder = URIReferenceBuilder::new();
builder
.try_scheme(Some("usb"))
.unwrap()
.try_authority(Some(Manufacturer::Ledger.as_ref()))
.unwrap()
.try_path("")
.unwrap()
.try_query(Some("key=0/0"))
.unwrap();
let uri = builder.build().unwrap();
let expect = Locator {
manufacturer,
pubkey: None,
derivation_path: Some(derivation_path.clone()),
};
assert_eq!(Locator::new_from_uri(&uri), Ok(expect));
// usb://ledger?key=0'/0'
let mut builder = URIReferenceBuilder::new();
builder
.try_scheme(Some("usb"))
.unwrap()
.try_authority(Some(Manufacturer::Ledger.as_ref()))
.unwrap()
.try_path("")
.unwrap()
.try_query(Some("key=0'/0'"))
.unwrap();
let uri = builder.build().unwrap();
let expect = Locator {
manufacturer,
pubkey: None,
derivation_path: Some(derivation_path.clone()),
};
assert_eq!(Locator::new_from_uri(&uri), Ok(expect));
// usb://ledger/?key=0/0
let mut builder = URIReferenceBuilder::new();
builder
.try_scheme(Some("usb"))
.unwrap()
.try_authority(Some(Manufacturer::Ledger.as_ref()))
.unwrap()
.try_path("/")
.unwrap()
.try_query(Some("key=0/0"))
.unwrap();
let uri = builder.build().unwrap();
let expect = Locator {
manufacturer,
pubkey: None,
derivation_path: Some(derivation_path),
}; };
assert_eq!(Locator::new_from_uri(&uri), Ok(expect)); assert_eq!(Locator::new_from_uri(&uri), Ok(expect));
@ -465,79 +338,10 @@ mod tests {
Locator::new_from_uri(&uri), Locator::new_from_uri(&uri),
Err(LocatorError::PubkeyError(ParsePubkeyError::Invalid)) Err(LocatorError::PubkeyError(ParsePubkeyError::Invalid))
); );
// usb://ledger?bad-key=0/0
let mut builder = URIReferenceBuilder::new();
builder
.try_scheme(Some("usb"))
.unwrap()
.try_authority(Some(Manufacturer::Ledger.as_ref()))
.unwrap()
.try_path("")
.unwrap()
.try_query(Some("bad-key=0/0"))
.unwrap();
let uri = builder.build().unwrap();
assert!(matches!(
Locator::new_from_uri(&uri),
Err(LocatorError::DerivationPathError(_))
));
// usb://ledger?key=bad-value
let mut builder = URIReferenceBuilder::new();
builder
.try_scheme(Some("usb"))
.unwrap()
.try_authority(Some(Manufacturer::Ledger.as_ref()))
.unwrap()
.try_path("")
.unwrap()
.try_query(Some("key=bad-value"))
.unwrap();
let uri = builder.build().unwrap();
assert!(matches!(
Locator::new_from_uri(&uri),
Err(LocatorError::DerivationPathError(_))
));
// usb://ledger?key=
let mut builder = URIReferenceBuilder::new();
builder
.try_scheme(Some("usb"))
.unwrap()
.try_authority(Some(Manufacturer::Ledger.as_ref()))
.unwrap()
.try_path("")
.unwrap()
.try_query(Some("key="))
.unwrap();
let uri = builder.build().unwrap();
assert!(matches!(
Locator::new_from_uri(&uri),
Err(LocatorError::DerivationPathError(_))
));
// usb://ledger?key
let mut builder = URIReferenceBuilder::new();
builder
.try_scheme(Some("usb"))
.unwrap()
.try_authority(Some(Manufacturer::Ledger.as_ref()))
.unwrap()
.try_path("")
.unwrap()
.try_query(Some("key"))
.unwrap();
let uri = builder.build().unwrap();
assert!(matches!(
Locator::new_from_uri(&uri),
Err(LocatorError::DerivationPathError(_))
));
} }
#[test] #[test]
fn test_locator_new_from_path() { fn test_locator_new_from_path() {
let derivation_path = DerivationPath::new_bip44(Some(0), Some(0));
let manufacturer = Manufacturer::Ledger; let manufacturer = Manufacturer::Ledger;
let pubkey = Pubkey::new_unique(); let pubkey = Pubkey::new_unique();
let path = format!("usb://ledger/{}?key=0/0", pubkey); let path = format!("usb://ledger/{}?key=0/0", pubkey);
@ -548,7 +352,6 @@ mod tests {
let expect = Locator { let expect = Locator {
manufacturer, manufacturer,
pubkey: Some(pubkey), pubkey: Some(pubkey),
derivation_path: Some(derivation_path.clone()),
}; };
assert_eq!(Locator::new_from_path(path), Ok(expect)); assert_eq!(Locator::new_from_path(path), Ok(expect));
@ -557,7 +360,6 @@ mod tests {
let expect = Locator { let expect = Locator {
manufacturer, manufacturer,
pubkey: Some(pubkey), pubkey: Some(pubkey),
derivation_path: None,
}; };
assert_eq!(Locator::new_from_path(path), Ok(expect)); assert_eq!(Locator::new_from_path(path), Ok(expect));
@ -566,7 +368,6 @@ mod tests {
let expect = Locator { let expect = Locator {
manufacturer, manufacturer,
pubkey: None, pubkey: None,
derivation_path: None,
}; };
assert_eq!(Locator::new_from_path(path), Ok(expect)); assert_eq!(Locator::new_from_path(path), Ok(expect));
@ -575,25 +376,6 @@ mod tests {
let expect = Locator { let expect = Locator {
manufacturer, manufacturer,
pubkey: None, pubkey: None,
derivation_path: None,
};
assert_eq!(Locator::new_from_path(path), Ok(expect));
// usb://ledger?key=0'/0'
let path = "usb://ledger?key=0'/0'";
let expect = Locator {
manufacturer,
pubkey: None,
derivation_path: Some(derivation_path.clone()),
};
assert_eq!(Locator::new_from_path(path), Ok(expect));
// usb://ledger/?key=0'/0'
let path = "usb://ledger?key=0'/0'";
let expect = Locator {
manufacturer,
pubkey: None,
derivation_path: Some(derivation_path),
}; };
assert_eq!(Locator::new_from_path(path), Ok(expect)); assert_eq!(Locator::new_from_path(path), Ok(expect));
@ -617,40 +399,5 @@ mod tests {
Locator::new_from_path(path), Locator::new_from_path(path),
Err(LocatorError::PubkeyError(ParsePubkeyError::Invalid)) Err(LocatorError::PubkeyError(ParsePubkeyError::Invalid))
); );
// usb://ledger?bad-key=0/0
let path = "usb://ledger?bad-key=0/0";
assert!(matches!(
Locator::new_from_path(path),
Err(LocatorError::DerivationPathError(_))
));
// usb://ledger?bad-key=0'/0'
let path = "usb://ledger?bad-key=0'/0'";
assert!(matches!(
Locator::new_from_path(path),
Err(LocatorError::DerivationPathError(_))
));
// usb://ledger?key=bad-value
let path = format!("usb://ledger/{}?key=bad-value", pubkey);
assert!(matches!(
Locator::new_from_path(path),
Err(LocatorError::DerivationPathError(_))
));
// usb://ledger?key=
let path = format!("usb://ledger/{}?key=", pubkey);
assert!(matches!(
Locator::new_from_path(path),
Err(LocatorError::DerivationPathError(_))
));
// usb://ledger?key
let path = format!("usb://ledger/{}?key", pubkey);
assert!(matches!(
Locator::new_from_path(path),
Err(LocatorError::DerivationPathError(_))
));
} }
} }

View File

@ -1,7 +1,7 @@
use { use {
crate::{ crate::{
ledger::get_ledger_from_info, ledger::get_ledger_from_info,
locator::Manufacturer, locator::{Locator, Manufacturer},
remote_wallet::{ remote_wallet::{
RemoteWallet, RemoteWalletError, RemoteWalletInfo, RemoteWalletManager, RemoteWallet, RemoteWalletError, RemoteWalletInfo, RemoteWalletManager,
RemoteWalletType, RemoteWalletType,
@ -56,12 +56,13 @@ impl Signer for RemoteKeypair {
} }
pub fn generate_remote_keypair( pub fn generate_remote_keypair(
path: String, locator: Locator,
derivation_path: DerivationPath,
wallet_manager: &RemoteWalletManager, wallet_manager: &RemoteWalletManager,
confirm_key: bool, confirm_key: bool,
keypair_name: &str, keypair_name: &str,
) -> Result<RemoteKeypair, RemoteWalletError> { ) -> Result<RemoteKeypair, RemoteWalletError> {
let (remote_wallet_info, derivation_path) = RemoteWalletInfo::parse_path(path)?; let remote_wallet_info = RemoteWalletInfo::parse_locator(locator);
if remote_wallet_info.manufacturer == Manufacturer::Ledger { if remote_wallet_info.manufacturer == Manufacturer::Ledger {
let ledger = get_ledger_from_info(remote_wallet_info, keypair_name, wallet_manager)?; let ledger = get_ledger_from_info(remote_wallet_info, keypair_name, wallet_manager)?;
let path = format!("{}{}", ledger.pretty_path, derivation_path.get_query()); let path = format!("{}{}", ledger.pretty_path, derivation_path.get_query());

View File

@ -253,20 +253,12 @@ pub struct RemoteWalletInfo {
} }
impl RemoteWalletInfo { impl RemoteWalletInfo {
pub fn parse_path(path: String) -> Result<(Self, DerivationPath), RemoteWalletError> { pub fn parse_locator(locator: Locator) -> Self {
let Locator {
manufacturer,
pubkey,
derivation_path,
} = Locator::new_from_path(path)?;
Ok((
RemoteWalletInfo { RemoteWalletInfo {
manufacturer, manufacturer: locator.manufacturer,
pubkey: pubkey.unwrap_or_default(), pubkey: locator.pubkey.unwrap_or_default(),
..RemoteWalletInfo::default() ..RemoteWalletInfo::default()
}, }
derivation_path.unwrap_or_default(),
))
} }
pub fn get_pretty_path(&self) -> String { pub fn get_pretty_path(&self) -> String {
@ -308,10 +300,13 @@ mod tests {
use super::*; use super::*;
#[test] #[test]
fn test_parse_path() { fn test_parse_locator() {
let pubkey = solana_sdk::pubkey::new_rand(); let pubkey = solana_sdk::pubkey::new_rand();
let (wallet_info, derivation_path) = let locator = Locator {
RemoteWalletInfo::parse_path(format!("usb://ledger/{:?}?key=1/2", pubkey)).unwrap(); manufacturer: Manufacturer::Ledger,
pubkey: Some(pubkey),
};
let wallet_info = RemoteWalletInfo::parse_locator(locator);
assert!(wallet_info.matches(&RemoteWalletInfo { assert!(wallet_info.matches(&RemoteWalletInfo {
model: "nano-s".to_string(), model: "nano-s".to_string(),
manufacturer: Manufacturer::Ledger, manufacturer: Manufacturer::Ledger,
@ -320,33 +315,13 @@ mod tests {
pubkey, pubkey,
error: None, error: None,
})); }));
assert_eq!(derivation_path, DerivationPath::new_bip44(Some(1), Some(2)));
let (wallet_info, derivation_path) =
RemoteWalletInfo::parse_path(format!("usb://ledger/{:?}?key=1'/2'", pubkey)).unwrap();
assert!(wallet_info.matches(&RemoteWalletInfo {
model: "nano-s".to_string(),
manufacturer: Manufacturer::Ledger,
serial: "".to_string(),
host_device_path: "/host/device/path".to_string(),
pubkey,
error: None,
}));
assert_eq!(derivation_path, DerivationPath::new_bip44(Some(1), Some(2)));
let (wallet_info, derivation_path) =
RemoteWalletInfo::parse_path(format!("usb://ledger/{:?}?key=1\'/2\'", pubkey)).unwrap();
assert!(wallet_info.matches(&RemoteWalletInfo {
model: "nano-s".to_string(),
manufacturer: Manufacturer::Ledger,
serial: "".to_string(),
host_device_path: "/host/device/path".to_string(),
pubkey,
error: None,
}));
assert_eq!(derivation_path, DerivationPath::new_bip44(Some(1), Some(2)));
// Test that wallet id need not be complete for key derivation to work // Test that pubkey need not be populated
let (wallet_info, derivation_path) = let locator = Locator {
RemoteWalletInfo::parse_path("usb://ledger?key=1".to_string()).unwrap(); manufacturer: Manufacturer::Ledger,
pubkey: None,
};
let wallet_info = RemoteWalletInfo::parse_locator(locator);
assert!(wallet_info.matches(&RemoteWalletInfo { assert!(wallet_info.matches(&RemoteWalletInfo {
model: "nano-s".to_string(), model: "nano-s".to_string(),
manufacturer: Manufacturer::Ledger, manufacturer: Manufacturer::Ledger,
@ -355,33 +330,6 @@ mod tests {
pubkey: Pubkey::default(), pubkey: Pubkey::default(),
error: None, error: None,
})); }));
assert_eq!(derivation_path, DerivationPath::new_bip44(Some(1), None));
let (wallet_info, derivation_path) =
RemoteWalletInfo::parse_path("usb://ledger/?key=1/2".to_string()).unwrap();
assert!(wallet_info.matches(&RemoteWalletInfo {
model: "".to_string(),
manufacturer: Manufacturer::Ledger,
serial: "".to_string(),
host_device_path: "/host/device/path".to_string(),
pubkey: Pubkey::default(),
error: None,
}));
assert_eq!(derivation_path, DerivationPath::new_bip44(Some(1), Some(2)));
// Failure cases
assert!(
RemoteWalletInfo::parse_path("usb://ledger/bad-pubkey?key=1/2".to_string()).is_err()
);
assert!(RemoteWalletInfo::parse_path("usb://?key=1/2".to_string()).is_err());
assert!(RemoteWalletInfo::parse_path("usb:/ledger?key=1/2".to_string()).is_err());
assert!(RemoteWalletInfo::parse_path("ledger?key=1/2".to_string()).is_err());
assert!(RemoteWalletInfo::parse_path("usb://ledger?key=1/2/3".to_string()).is_err());
// Other query strings cause an error
assert!(
RemoteWalletInfo::parse_path("usb://ledger/?key=1/2&test=other".to_string()).is_err()
);
assert!(RemoteWalletInfo::parse_path("usb://ledger/?Key=1/2".to_string()).is_err());
assert!(RemoteWalletInfo::parse_path("usb://ledger/?test=other".to_string()).is_err());
} }
#[test] #[test]

View File

@ -44,36 +44,38 @@ byteorder = { version = "1.3.4", optional = true }
chrono = { version = "0.4", optional = true } chrono = { version = "0.4", optional = true }
curve25519-dalek = { version = "2.1.0", optional = true } curve25519-dalek = { version = "2.1.0", optional = true }
derivation-path = { version = "0.1.3", default-features = false } derivation-path = { version = "0.1.3", default-features = false }
digest = { version = "0.9.0", optional = true }
ed25519-dalek = { version = "=1.0.1", optional = true }
generic-array = { version = "0.14.3", default-features = false, features = ["serde", "more_lengths"], optional = true } generic-array = { version = "0.14.3", default-features = false, features = ["serde", "more_lengths"], optional = true }
hex = "0.4.2" hex = "0.4.2"
hmac = "0.10.1" hmac = "0.10.1"
itertools = "0.9.0" itertools = "0.9.0"
lazy_static = "1.4.0" lazy_static = "1.4.0"
libsecp256k1 = { version = "0.3.5", optional = true }
log = "0.4.11" log = "0.4.11"
memmap2 = { version = "0.1.0", optional = true } memmap2 = { version = "0.1.0", optional = true }
num-derive = "0.3" num-derive = "0.3"
num-traits = "0.2" num-traits = "0.2"
pbkdf2 = { version = "0.6.0", default-features = false } pbkdf2 = { version = "0.6.0", default-features = false }
qstring = "0.7.2"
rand = { version = "0.7.0", optional = true } rand = { version = "0.7.0", optional = true }
rand_chacha = { version = "0.2.2", optional = true } rand_chacha = { version = "0.2.2", optional = true }
rand_core = "0.6.2" rand_core = "0.6.2"
rustversion = "1.0.4"
serde = "1.0.122" serde = "1.0.122"
serde_bytes = "0.11" serde_bytes = "0.11"
serde_derive = "1.0.103" serde_derive = "1.0.103"
serde_json = { version = "1.0.56", optional = true } serde_json = { version = "1.0.56", optional = true }
sha2 = "0.9.2" sha2 = "0.9.2"
thiserror = "1.0" sha3 = { version = "0.9.1", optional = true }
ed25519-dalek = { version = "=1.0.1", optional = true }
solana-crate-features = { path = "../crate-features", version = "=1.7.0", optional = true } solana-crate-features = { path = "../crate-features", version = "=1.7.0", optional = true }
solana-logger = { path = "../logger", version = "=1.7.0", optional = true } solana-logger = { path = "../logger", version = "=1.7.0", optional = true }
solana-frozen-abi = { path = "../frozen-abi", version = "=1.7.0" } solana-frozen-abi = { path = "../frozen-abi", version = "=1.7.0" }
solana-frozen-abi-macro = { path = "../frozen-abi/macro", version = "=1.7.0" } solana-frozen-abi-macro = { path = "../frozen-abi/macro", version = "=1.7.0" }
solana-program = { path = "program", version = "=1.7.0" } solana-program = { path = "program", version = "=1.7.0" }
solana-sdk-macro = { path = "macro", version = "=1.7.0" } solana-sdk-macro = { path = "macro", version = "=1.7.0" }
rustversion = "1.0.4" thiserror = "1.0"
libsecp256k1 = { version = "0.3.5", optional = true } uriparse = "0.6.3"
sha3 = { version = "0.9.1", optional = true }
digest = { version = "0.9.0", optional = true }
[dev-dependencies] [dev-dependencies]
curve25519-dalek = "2.1.0" curve25519-dalek = "2.1.0"

View File

@ -7,6 +7,7 @@ use {
str::FromStr, str::FromStr,
}, },
thiserror::Error, thiserror::Error,
uriparse::URIReference,
}; };
const ACCOUNT_INDEX: usize = 2; const ACCOUNT_INDEX: usize = 2;
@ -109,6 +110,7 @@ impl DerivationPath {
self.0.path() self.0.path()
} }
// Assumes `key` query-string key
pub fn get_query(&self) -> String { pub fn get_query(&self) -> String {
if let Some(account) = &self.account() { if let Some(account) = &self.account() {
if let Some(change) = &self.change() { if let Some(change) = &self.change() {
@ -120,6 +122,34 @@ impl DerivationPath {
"".to_string() "".to_string()
} }
} }
// Only accepts single query string pair of type `key`
pub fn from_uri(uri: &URIReference<'_>) -> Result<Option<Self>, DerivationPathError> {
if let Some(query) = uri.query() {
let query_str = query.as_str();
if query_str.is_empty() {
return Ok(None);
}
let query = qstring::QString::from(query_str);
if query.len() > 1 {
return Err(DerivationPathError::InvalidDerivationPath(
"invalid query string, extra fields not supported".to_string(),
));
}
let key = query.get("key");
if key.is_none() {
return Err(DerivationPathError::InvalidDerivationPath(format!(
"invalid query string `{}`, only `key` supported",
query_str,
)));
}
// Use from_key_str instead of TryInto here to make it a little more explicit that this
// generates a Solana bip44 DerivationPath
key.map(Self::from_key_str).transpose()
} else {
Ok(None)
}
}
} }
impl fmt::Debug for DerivationPath { impl fmt::Debug for DerivationPath {
@ -161,6 +191,7 @@ impl Bip44 for Solana {
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::*; use super::*;
use uriparse::URIReferenceBuilder;
struct TestCoin; struct TestCoin;
impl Bip44 for TestCoin { impl Bip44 for TestCoin {
@ -263,6 +294,190 @@ mod tests {
); );
} }
#[test]
fn test_new_from_uri() {
let derivation_path = DerivationPath::new_bip44(Some(0), Some(0));
// test://path?key=0/0
let mut builder = URIReferenceBuilder::new();
builder
.try_scheme(Some("test"))
.unwrap()
.try_authority(Some("path"))
.unwrap()
.try_path("")
.unwrap()
.try_query(Some("key=0/0"))
.unwrap();
let uri = builder.build().unwrap();
assert_eq!(
DerivationPath::from_uri(&uri).unwrap(),
Some(derivation_path.clone())
);
// test://path?key=0'/0'
let mut builder = URIReferenceBuilder::new();
builder
.try_scheme(Some("test"))
.unwrap()
.try_authority(Some("path"))
.unwrap()
.try_path("")
.unwrap()
.try_query(Some("key=0'/0'"))
.unwrap();
let uri = builder.build().unwrap();
assert_eq!(
DerivationPath::from_uri(&uri).unwrap(),
Some(derivation_path.clone())
);
// test://path?key=0\'/0\'
let mut builder = URIReferenceBuilder::new();
builder
.try_scheme(Some("test"))
.unwrap()
.try_authority(Some("path"))
.unwrap()
.try_path("")
.unwrap()
.try_query(Some("key=0\'/0\'"))
.unwrap();
let uri = builder.build().unwrap();
assert_eq!(
DerivationPath::from_uri(&uri).unwrap(),
Some(derivation_path)
);
// test://path
let mut builder = URIReferenceBuilder::new();
builder
.try_scheme(Some("test"))
.unwrap()
.try_authority(Some("path"))
.unwrap()
.try_path("")
.unwrap();
let uri = builder.build().unwrap();
assert_eq!(DerivationPath::from_uri(&uri).unwrap(), None);
// test://path?
let mut builder = URIReferenceBuilder::new();
builder
.try_scheme(Some("test"))
.unwrap()
.try_authority(Some("path"))
.unwrap()
.try_path("")
.unwrap()
.try_query(Some(""))
.unwrap();
let uri = builder.build().unwrap();
assert_eq!(DerivationPath::from_uri(&uri).unwrap(), None);
// test://path?key=0/0/0
let mut builder = URIReferenceBuilder::new();
builder
.try_scheme(Some("test"))
.unwrap()
.try_authority(Some("path"))
.unwrap()
.try_path("")
.unwrap()
.try_query(Some("key=0/0/0"))
.unwrap();
let uri = builder.build().unwrap();
assert!(matches!(
DerivationPath::from_uri(&uri),
Err(DerivationPathError::InvalidDerivationPath(_))
));
// test://path?key=0/0&bad-key=0/0
let mut builder = URIReferenceBuilder::new();
builder
.try_scheme(Some("test"))
.unwrap()
.try_authority(Some("path"))
.unwrap()
.try_path("")
.unwrap()
.try_query(Some("key=0/0&bad-key=0/0"))
.unwrap();
let uri = builder.build().unwrap();
assert!(matches!(
DerivationPath::from_uri(&uri),
Err(DerivationPathError::InvalidDerivationPath(_))
));
// test://path?bad-key=0/0
let mut builder = URIReferenceBuilder::new();
builder
.try_scheme(Some("test"))
.unwrap()
.try_authority(Some("path"))
.unwrap()
.try_path("")
.unwrap()
.try_query(Some("bad-key=0/0"))
.unwrap();
let uri = builder.build().unwrap();
assert!(matches!(
DerivationPath::from_uri(&uri),
Err(DerivationPathError::InvalidDerivationPath(_))
));
// test://path?key=bad-value
let mut builder = URIReferenceBuilder::new();
builder
.try_scheme(Some("test"))
.unwrap()
.try_authority(Some("path"))
.unwrap()
.try_path("")
.unwrap()
.try_query(Some("key=bad-value"))
.unwrap();
let uri = builder.build().unwrap();
assert!(matches!(
DerivationPath::from_uri(&uri),
Err(DerivationPathError::InvalidDerivationPath(_))
));
// test://path?key=
let mut builder = URIReferenceBuilder::new();
builder
.try_scheme(Some("test"))
.unwrap()
.try_authority(Some("path"))
.unwrap()
.try_path("")
.unwrap()
.try_query(Some("key="))
.unwrap();
let uri = builder.build().unwrap();
assert!(matches!(
DerivationPath::from_uri(&uri),
Err(DerivationPathError::InvalidDerivationPath(_))
));
// test://path?key
let mut builder = URIReferenceBuilder::new();
builder
.try_scheme(Some("test"))
.unwrap()
.try_authority(Some("path"))
.unwrap()
.try_path("")
.unwrap()
.try_query(Some("key"))
.unwrap();
let uri = builder.build().unwrap();
assert!(matches!(
DerivationPath::from_uri(&uri),
Err(DerivationPathError::InvalidDerivationPath(_))
));
}
#[test] #[test]
fn test_get_query() { fn test_get_query() {
let derivation_path = DerivationPath::new_bip44_with_coin(TestCoin, None, None); let derivation_path = DerivationPath::new_bip44_with_coin(TestCoin, None, None);