sc_cli/params/
keystore_params.rs1use crate::{error, error::Result};
20use clap::Args;
21use sc_service::config::KeystoreConfig;
22use sp_core::crypto::SecretString;
23use std::{
24 fs,
25 path::{Path, PathBuf},
26};
27
28const DEFAULT_KEYSTORE_CONFIG_PATH: &str = "keystore";
30
31#[derive(Debug, Clone, Args)]
33pub struct KeystoreParams {
34 #[arg(long, value_name = "PATH")]
36 pub keystore_path: Option<PathBuf>,
37
38 #[arg(long, conflicts_with_all = &["password", "password_filename"])]
40 pub password_interactive: bool,
41
42 #[arg(
46 long,
47 value_parser = secret_string_from_str,
48 conflicts_with_all = &["password_interactive", "password_filename"]
49 )]
50 pub password: Option<SecretString>,
51
52 #[arg(
54 long,
55 value_name = "PATH",
56 conflicts_with_all = &["password_interactive", "password"]
57 )]
58 pub password_filename: Option<PathBuf>,
59}
60
61pub fn secret_string_from_str(s: &str) -> std::result::Result<SecretString, String> {
63 std::str::FromStr::from_str(s).map_err(|_| "Could not get SecretString".to_string())
64}
65
66impl KeystoreParams {
67 pub fn keystore_config(&self, config_dir: &Path) -> Result<KeystoreConfig> {
69 let password = if self.password_interactive {
70 Some(SecretString::new(input_keystore_password()?))
71 } else if let Some(ref file) = self.password_filename {
72 let password = fs::read_to_string(file).map_err(|e| format!("{}", e))?;
73 Some(SecretString::new(password))
74 } else {
75 self.password.clone()
76 };
77
78 let path = self
79 .keystore_path
80 .clone()
81 .unwrap_or_else(|| config_dir.join(DEFAULT_KEYSTORE_CONFIG_PATH));
82
83 Ok(KeystoreConfig::Path { path, password })
84 }
85
86 pub fn read_password(&self) -> error::Result<Option<SecretString>> {
88 let (password_interactive, password) = (self.password_interactive, self.password.clone());
89
90 let pass = if password_interactive {
91 let password = rpassword::prompt_password("Key password: ")?;
92 Some(SecretString::new(password))
93 } else {
94 password
95 };
96
97 Ok(pass)
98 }
99}
100
101fn input_keystore_password() -> Result<String> {
102 rpassword::prompt_password("Keystore password: ").map_err(|e| format!("{:?}", e).into())
103}