Compare commits
	
		
			No commits in common. "53665cac2e7097d5fb467b8faf37aa947f955f62" and "88a05f23acbd0acde25a2177ed2875a6a7b1b5ac" have entirely different histories.
		
	
	
		
			53665cac2e
			...
			88a05f23ac
		
	
		|  | @ -1381,22 +1381,10 @@ dependencies = [ | |||
|  "cfg-if", | ||||
|  "js-sys", | ||||
|  "libc", | ||||
|  "wasi 0.11.0+wasi-snapshot-preview1", | ||||
|  "wasi", | ||||
|  "wasm-bindgen", | ||||
| ] | ||||
| 
 | ||||
| [[package]] | ||||
| name = "getrandom" | ||||
| version = "0.3.1" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "43a49c392881ce6d5c3b8cb70f98717b7c07aabbdff06687b9030dbfbe2725f8" | ||||
| dependencies = [ | ||||
|  "cfg-if", | ||||
|  "libc", | ||||
|  "wasi 0.13.3+wasi-0.2.2", | ||||
|  "windows-targets", | ||||
| ] | ||||
| 
 | ||||
| [[package]] | ||||
| name = "ghash" | ||||
| version = "0.5.1" | ||||
|  | @ -1820,9 +1808,7 @@ dependencies = [ | |||
|  "openpgp-card-sequoia", | ||||
|  "sequoia-openpgp", | ||||
|  "serde", | ||||
|  "shlex", | ||||
|  "smex", | ||||
|  "tempfile", | ||||
|  "thiserror", | ||||
|  "tokio", | ||||
| ] | ||||
|  | @ -2258,7 +2244,7 @@ dependencies = [ | |||
|  "hermit-abi 0.3.9", | ||||
|  "libc", | ||||
|  "log", | ||||
|  "wasi 0.11.0+wasi-snapshot-preview1", | ||||
|  "wasi", | ||||
|  "windows-sys 0.52.0", | ||||
| ] | ||||
| 
 | ||||
|  | @ -2268,7 +2254,7 @@ version = "7.4.0" | |||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "44e6ff4a94e5d34a1fd5abbd39418074646e2fa51b257198701330f22fcd6936" | ||||
| dependencies = [ | ||||
|  "getrandom 0.2.15", | ||||
|  "getrandom", | ||||
|  "libc", | ||||
|  "nettle-sys", | ||||
|  "thiserror", | ||||
|  | @ -2834,7 +2820,7 @@ version = "0.6.4" | |||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c" | ||||
| dependencies = [ | ||||
|  "getrandom 0.2.15", | ||||
|  "getrandom", | ||||
| ] | ||||
| 
 | ||||
| [[package]] | ||||
|  | @ -2852,7 +2838,7 @@ version = "0.4.6" | |||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "ba009ff324d1fc1b900bd1fdb31564febe58a8ccc8a6fdbb93b543d33b13ca43" | ||||
| dependencies = [ | ||||
|  "getrandom 0.2.15", | ||||
|  "getrandom", | ||||
|  "libredox", | ||||
|  "thiserror", | ||||
| ] | ||||
|  | @ -3081,7 +3067,7 @@ dependencies = [ | |||
|  "ed25519", | ||||
|  "ed25519-dalek", | ||||
|  "flate2", | ||||
|  "getrandom 0.2.15", | ||||
|  "getrandom", | ||||
|  "idea", | ||||
|  "idna", | ||||
|  "lalrpop", | ||||
|  | @ -3369,13 +3355,12 @@ dependencies = [ | |||
| 
 | ||||
| [[package]] | ||||
| name = "tempfile" | ||||
| version = "3.17.1" | ||||
| version = "3.14.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "22e5a0acb1f3f55f65cc4a866c361b2fb2a0ff6366785ae6fbb5f85df07ba230" | ||||
| checksum = "28cce251fcbc87fac86a866eeb0d6c2d536fc16d06f184bb61aeae11aa4cee0c" | ||||
| dependencies = [ | ||||
|  "cfg-if", | ||||
|  "fastrand", | ||||
|  "getrandom 0.3.1", | ||||
|  "once_cell", | ||||
|  "rustix", | ||||
|  "windows-sys 0.59.0", | ||||
|  | @ -3741,15 +3726,6 @@ version = "0.11.0+wasi-snapshot-preview1" | |||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" | ||||
| 
 | ||||
| [[package]] | ||||
| name = "wasi" | ||||
| version = "0.13.3+wasi-0.2.2" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "26816d2e1a4a36a2940b96c5296ce403917633dff8f3440e9b236ed6f6bacad2" | ||||
| dependencies = [ | ||||
|  "wit-bindgen-rt", | ||||
| ] | ||||
| 
 | ||||
| [[package]] | ||||
| name = "wasm-bindgen" | ||||
| version = "0.2.95" | ||||
|  | @ -3961,15 +3937,6 @@ version = "0.52.6" | |||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "589f6da84c646204747d1270a2a5661ea66ed1cced2631d546fdfb155959f9ec" | ||||
| 
 | ||||
| [[package]] | ||||
| name = "wit-bindgen-rt" | ||||
| version = "0.33.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "3268f3d866458b787f390cf61f4bbb563b922d091359f9608842999eaee3943c" | ||||
| dependencies = [ | ||||
|  "bitflags 2.6.0", | ||||
| ] | ||||
| 
 | ||||
| [[package]] | ||||
| name = "write16" | ||||
| version = "1.0.0" | ||||
|  |  | |||
|  | @ -76,7 +76,6 @@ thiserror = "1.0.56" | |||
| tokio = "1.35.1" | ||||
| v4l = "0.14.0" | ||||
| base64 = "0.22.1" | ||||
| tempfile = "3.17.1" | ||||
| 
 | ||||
| [profile.release] | ||||
| debug = true | ||||
|  |  | |||
|  | @ -32,7 +32,7 @@ tower = { version = "0.4.13", features = ["tokio", "util"] } | |||
| # Personally audited | ||||
| thiserror = { workspace = true } | ||||
| serde = { workspace = true } | ||||
| tempfile = { workspace = true } | ||||
| tempfile = { version = "3.10.0", default-features = false } | ||||
| 
 | ||||
| [dev-dependencies] | ||||
| hex-literal = { workspace = true } | ||||
|  |  | |||
|  | @ -48,5 +48,3 @@ sequoia-openpgp = { workspace = true } | |||
| keyforkd-models.workspace = true | ||||
| base64.workspace = true | ||||
| nix = { version = "0.29.0", default-features = false, features = ["process"] } | ||||
| shlex = "1.3.0" | ||||
| tempfile.workspace = true | ||||
|  |  | |||
|  | @ -2,6 +2,20 @@ | |||
| 
 | ||||
| use std::{collections::HashMap, str::FromStr}; | ||||
| 
 | ||||
| /// A helper struct for clap arguments that can contain additional arguments. For example:
 | ||||
| /// `keyfork mnemonic generate --encrypt-to cert.asc,output=encrypted.asc`.
 | ||||
| #[derive(Clone, Debug)] | ||||
| pub struct ValueWithOptions<T: FromStr> | ||||
| where | ||||
|     T::Err: std::error::Error, | ||||
| { | ||||
|     /// A mapping between keys and values.
 | ||||
|     pub values: HashMap<String, String>, | ||||
| 
 | ||||
|     /// The first variable for the argument, such as a [`PathBuf`].
 | ||||
|     pub inner: T, | ||||
| } | ||||
| 
 | ||||
| /// An error that occurred while parsing a base value or its
 | ||||
| #[derive(Debug, thiserror::Error)] | ||||
| pub enum ValueParseError { | ||||
|  | @ -18,62 +32,6 @@ pub enum ValueParseError { | |||
|     BadKeyValue, | ||||
| } | ||||
| 
 | ||||
| /// A helper struct to parse key-value arguments, without any prior argument.
 | ||||
| #[derive(Clone, Debug, Default)] | ||||
| pub struct Options { | ||||
|     /// The values provided.
 | ||||
|     pub values: HashMap<String, String>, | ||||
| } | ||||
| 
 | ||||
| impl std::fmt::Display for Options { | ||||
|     fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { | ||||
|         let mut iter = self.values.iter().peekable(); | ||||
|         while let Some((key, value)) = iter.next() { | ||||
|             write!(f, "{key}={value}")?; | ||||
|             if iter.peek().is_some() { | ||||
|                 write!(f, ",")?; | ||||
|             } | ||||
|         } | ||||
|         Ok(()) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl FromStr for Options { | ||||
|     type Err = ValueParseError; | ||||
| 
 | ||||
|     fn from_str(s: &str) -> Result<Self, Self::Err> { | ||||
|         if s.is_empty() { | ||||
|             return Ok(Default::default()) | ||||
|         } | ||||
|         let values = s | ||||
|             .split(',') | ||||
|             .map(|value| { | ||||
|                 let [k, v] = value | ||||
|                     .splitn(2, '=') | ||||
|                     .collect::<Vec<_>>() | ||||
|                     .try_into() | ||||
|                     .map_err(|_| ValueParseError::BadKeyValue)?; | ||||
|                 Ok((k.to_string(), v.to_string())) | ||||
|             }) | ||||
|             .collect::<Result<HashMap<String, String>, ValueParseError>>()?; | ||||
|         Ok(Self { values }) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /// A helper struct for clap arguments that can contain additional arguments. For example:
 | ||||
| /// `keyfork mnemonic generate --encrypt-to cert.asc,output=encrypted.asc`.
 | ||||
| #[derive(Clone, Debug)] | ||||
| pub struct ValueWithOptions<T: FromStr> | ||||
| where | ||||
|     T::Err: std::error::Error, | ||||
| { | ||||
|     /// A mapping between keys and values.
 | ||||
|     pub values: HashMap<String, String>, | ||||
| 
 | ||||
|     /// The first variable for the argument, such as a [`PathBuf`].
 | ||||
|     pub inner: T, | ||||
| } | ||||
| 
 | ||||
| impl<T: std::str::FromStr> FromStr for ValueWithOptions<T> | ||||
| where | ||||
|     <T as FromStr>::Err: std::error::Error, | ||||
|  |  | |||
|  | @ -1,36 +1,25 @@ | |||
| use super::{Keyfork, create}; | ||||
| use super::Keyfork; | ||||
| use clap::{Args, Parser, Subcommand, ValueEnum}; | ||||
| use std::{fmt::Display, io::Write, path::PathBuf}; | ||||
| 
 | ||||
| use keyfork_derive_openpgp::openpgp::{ | ||||
| use keyfork_derive_openpgp::{ | ||||
|     openpgp::{ | ||||
|         armor::{Kind, Writer}, | ||||
|         packet::UserID, | ||||
|         serialize::Marshal, | ||||
|         types::KeyFlags, | ||||
|     Cert, | ||||
|     }, | ||||
|     XPrvKey, | ||||
| }; | ||||
| use keyfork_derive_path_data::paths; | ||||
| use keyfork_derive_util::{ | ||||
|     request::DerivationAlgorithm, DerivationIndex, DerivationPath, ExtendedPrivateKey as XPrv, | ||||
|     IndexError, PrivateKey, | ||||
|     request::{DerivationAlgorithm, DerivationRequest, DerivationResponse}, | ||||
|     DerivationIndex, DerivationPath, IndexError, | ||||
| }; | ||||
| use keyforkd_client::Client; | ||||
| 
 | ||||
| type OptWrite = Option<Box<dyn Write>>; | ||||
| use keyforkd_models::Request; | ||||
| 
 | ||||
| type Result<T, E = Box<dyn std::error::Error>> = std::result::Result<T, E>; | ||||
| 
 | ||||
| pub trait Deriver { | ||||
|     type Prv: PrivateKey + Clone; | ||||
|     const DERIVATION_ALGORITHM: DerivationAlgorithm; | ||||
| 
 | ||||
|     fn derivation_path(&self) -> DerivationPath; | ||||
| 
 | ||||
|     fn derive_with_xprv(&self, writer: OptWrite, xprv: XPrv<Self::Prv>) -> Result<()>; | ||||
| 
 | ||||
|     fn derive_public_with_xprv(&self, writer: OptWrite, xprv: XPrv<Self::Prv>) -> Result<()>; | ||||
| } | ||||
| 
 | ||||
| #[derive(Subcommand, Clone, Debug)] | ||||
| pub enum DeriveSubcommands { | ||||
|     /// Derive an OpenPGP Transferable Secret Key (private key). The key is encoded using OpenPGP
 | ||||
|  | @ -45,51 +34,14 @@ pub enum DeriveSubcommands { | |||
|     #[command(name = "openpgp")] | ||||
|     OpenPGP(OpenPGP), | ||||
| 
 | ||||
|     /// Derive an Ed25519 key for a specific algorithm, in a given format.
 | ||||
|     /// Derive a bare key for a specific algorithm, in a given format.
 | ||||
|     Key(Key), | ||||
| } | ||||
| 
 | ||||
| /// Derivation path to use when deriving OpenPGP keys.
 | ||||
| #[derive(ValueEnum, Clone, Debug, Default)] | ||||
| pub enum Path { | ||||
|     /// The default derivation path; no additional index is used.
 | ||||
|     #[default] | ||||
|     Default, | ||||
| 
 | ||||
|     /// The Disaster Recovery index.
 | ||||
|     DisasterRecovery, | ||||
| } | ||||
| 
 | ||||
| impl std::fmt::Display for Path { | ||||
|     fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { | ||||
|         f.write_str(self.as_str()) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl Path { | ||||
|     fn as_str(&self) -> &'static str { | ||||
|         match self { | ||||
|             Path::Default => "default", | ||||
|             Path::DisasterRecovery => "disaster-recovery", | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fn derivation_path(&self) -> DerivationPath { | ||||
|         match self { | ||||
|             Self::Default => paths::OPENPGP.clone(), | ||||
|             Self::DisasterRecovery => paths::OPENPGP_DISASTER_RECOVERY.clone(), | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| #[derive(Args, Clone, Debug)] | ||||
| pub struct OpenPGP { | ||||
|     /// Default User ID for the certificate, using the OpenPGP User ID format.
 | ||||
|     user_id: String, | ||||
| 
 | ||||
|     /// Derivation path to use when deriving OpenPGP keys.
 | ||||
|     #[arg(long, required = false, default_value = "default")] | ||||
|     derivation_path: Path, | ||||
| } | ||||
| 
 | ||||
| /// A format for exporting a key.
 | ||||
|  | @ -131,18 +83,6 @@ impl std::str::FromStr for Slug { | |||
|     } | ||||
| } | ||||
| 
 | ||||
| impl Display for Slug { | ||||
|     fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { | ||||
|         match (self.0.inner() & (0b1 << 31)).to_be_bytes().as_slice() { | ||||
|             [0, 0, 0, 0] => Ok(()), | ||||
|             [0, 0, 0, bytes @ ..] | [0, 0, bytes @ ..] | [0, bytes @ ..] | [bytes @ ..] => f | ||||
|                 .write_str( | ||||
|                     std::str::from_utf8(&bytes[..]).expect("slug constructed from non-utf8"), | ||||
|                 ), | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| #[derive(Args, Clone, Debug)] | ||||
| pub struct Key { | ||||
|     /// The derivation algorithm to derive a key for.
 | ||||
|  | @ -158,34 +98,18 @@ pub struct Key { | |||
| } | ||||
| 
 | ||||
| impl DeriveSubcommands { | ||||
|     fn handle(&self, account: DerivationIndex, is_public: bool, writer: OptWrite) -> Result<()> { | ||||
|     fn handle(&self, account: DerivationIndex) -> Result<()> { | ||||
|         match self { | ||||
|             DeriveSubcommands::OpenPGP(opgp) => { | ||||
|                 let path = opgp.derivation_path(); | ||||
|                 let xprv = Client::discover_socket()? | ||||
|                     .request_xprv::<<OpenPGP as Deriver>::Prv>(&path.chain_push(account))?; | ||||
|                 if is_public { | ||||
|                     opgp.derive_public_with_xprv(writer, xprv) | ||||
|                 } else { | ||||
|                     opgp.derive_with_xprv(writer, xprv) | ||||
|                 } | ||||
|             } | ||||
|             DeriveSubcommands::Key(key) => { | ||||
|                 let path = key.derivation_path(); | ||||
|                 let xprv = Client::discover_socket()? | ||||
|                     .request_xprv::<<Key as Deriver>::Prv>(&path.chain_push(account))?; | ||||
|                 if is_public { | ||||
|                     key.derive_public_with_xprv(writer, xprv) | ||||
|                 } else { | ||||
|                     key.derive_with_xprv(writer, xprv) | ||||
|                 } | ||||
|             } | ||||
|             DeriveSubcommands::OpenPGP(opgp) => opgp.handle(account), | ||||
|             DeriveSubcommands::Key(key) => key.handle(account), | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl OpenPGP { | ||||
|     fn cert_from_xprv(&self, xprv: keyfork_derive_openpgp::XPrv) -> Result<Cert> { | ||||
|     pub fn handle(&self, account: DerivationIndex) -> Result<()> { | ||||
|         let path = paths::OPENPGP.clone().chain_push(account); | ||||
|         // TODO: should this be customizable?
 | ||||
|         let subkeys = vec![ | ||||
|             KeyFlags::empty().set_certification(), | ||||
|             KeyFlags::empty().set_signing(), | ||||
|  | @ -194,100 +118,40 @@ impl OpenPGP { | |||
|                 .set_storage_encryption(), | ||||
|             KeyFlags::empty().set_authentication(), | ||||
|         ]; | ||||
|         let xprv = Client::discover_socket()?.request_xprv::<XPrvKey>(&path)?; | ||||
|         let default_userid = UserID::from(self.user_id.as_str()); | ||||
|         let cert = keyfork_derive_openpgp::derive(xprv, &subkeys, &default_userid)?; | ||||
| 
 | ||||
|         let userid = UserID::from(&*self.user_id); | ||||
|         keyfork_derive_openpgp::derive(xprv, &subkeys, &userid).map_err(Into::into) | ||||
|     } | ||||
| } | ||||
|         let mut w = Writer::new(std::io::stdout(), Kind::SecretKey)?; | ||||
| 
 | ||||
| impl Deriver for OpenPGP { | ||||
|     type Prv = keyfork_derive_openpgp::XPrvKey; | ||||
|     const DERIVATION_ALGORITHM: DerivationAlgorithm = DerivationAlgorithm::Ed25519; | ||||
| 
 | ||||
|     fn derivation_path(&self) -> DerivationPath { | ||||
|         self.derivation_path.derivation_path() | ||||
|     } | ||||
| 
 | ||||
|     fn derive_with_xprv(&self, writer: OptWrite, xprv: XPrv<Self::Prv>) -> Result<()> { | ||||
|         let cert = self.cert_from_xprv(xprv)?; | ||||
|         let writer = match writer { | ||||
|             Some(w) => w, | ||||
|             None => { | ||||
|                 let path = PathBuf::from(cert.fingerprint().to_string()).with_extension("asc"); | ||||
|                 let file = create(&path)?; | ||||
|                 Box::new(file) | ||||
|             } | ||||
|         }; | ||||
|         let mut writer = Writer::new(writer, Kind::SecretKey)?; | ||||
|         for packet in cert.as_tsk().into_packets() { | ||||
|             packet.serialize(&mut writer)?; | ||||
|         } | ||||
|         writer.finalize()?; | ||||
|         Ok(()) | ||||
|             packet.serialize(&mut w)?; | ||||
|         } | ||||
| 
 | ||||
|     fn derive_public_with_xprv(&self, writer: OptWrite, xprv: XPrv<Self::Prv>) -> Result<()> { | ||||
|         let cert = self.cert_from_xprv(xprv)?; | ||||
|         let writer = match writer { | ||||
|             Some(w) => w, | ||||
|             None => { | ||||
|                 let path = PathBuf::from(cert.fingerprint().to_string()).with_extension("asc"); | ||||
|                 let file = create(&path)?; | ||||
|                 Box::new(file) | ||||
|             } | ||||
|         }; | ||||
|         let mut writer = Writer::new(writer, Kind::PublicKey)?; | ||||
|         for packet in cert.into_packets2() { | ||||
|             packet.serialize(&mut writer)?; | ||||
|         } | ||||
|         writer.finalize()?; | ||||
|         w.finalize()?; | ||||
|         Ok(()) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl Deriver for Key { | ||||
|     // HACK: We're abusing that we use the same key as OpenPGP. Maybe we should use ed25519_dalek.
 | ||||
|     type Prv = keyfork_derive_openpgp::XPrvKey; | ||||
|     const DERIVATION_ALGORITHM: DerivationAlgorithm = DerivationAlgorithm::Ed25519; | ||||
| impl Key { | ||||
|     pub fn handle(&self, account: DerivationIndex) -> Result<()> { | ||||
|         let mut client = keyforkd_client::Client::discover_socket()?; | ||||
|         let path = DerivationPath::default() | ||||
|             .chain_push(self.slug.0.clone()) | ||||
|             .chain_push(account); | ||||
|         let request = DerivationRequest::new(self.derivation_algorithm.clone(), &path); | ||||
|         let request = Request::Derivation(request); | ||||
|         let derived_key: DerivationResponse = client.request(&request)?.try_into()?; | ||||
| 
 | ||||
|     fn derivation_path(&self) -> DerivationPath { | ||||
|         DerivationPath::default().chain_push(self.slug.0.clone()) | ||||
|     } | ||||
| 
 | ||||
|     fn derive_with_xprv(&self, writer: OptWrite, xprv: XPrv<Self::Prv>) -> Result<()> { | ||||
|         let (formatted, ext) = match self.format { | ||||
|             KeyFormat::Hex => (smex::encode(xprv.private_key().to_bytes()), "hex"), | ||||
|         let formatted = match self.format { | ||||
|             KeyFormat::Hex => smex::encode(derived_key.data), | ||||
|             KeyFormat::Base64 => { | ||||
|                 use base64::prelude::*; | ||||
|                 (BASE64_STANDARD.encode(xprv.private_key().to_bytes()), "b64") | ||||
|                 BASE64_STANDARD.encode(derived_key.data) | ||||
|             } | ||||
|         }; | ||||
|         let filename = | ||||
|             PathBuf::from(smex::encode(xprv.public_key().to_bytes())).with_extension(ext); | ||||
|         if let Some(mut writer) = writer { | ||||
|             writeln!(writer, "{formatted}")?; | ||||
|         } else { | ||||
|             std::fs::write(&filename, formatted)?; | ||||
|         } | ||||
| 
 | ||||
|         Ok(()) | ||||
|     } | ||||
| 
 | ||||
|     fn derive_public_with_xprv(&self, writer: OptWrite, xprv: XPrv<Self::Prv>) -> Result<()> { | ||||
|         let (formatted, ext) = match self.format { | ||||
|             KeyFormat::Hex => (smex::encode(xprv.public_key().to_bytes()), "hex"), | ||||
|             KeyFormat::Base64 => { | ||||
|                 use base64::prelude::*; | ||||
|                 (BASE64_STANDARD.encode(xprv.public_key().to_bytes()), "b64") | ||||
|             } | ||||
|         }; | ||||
|         let filename = | ||||
|             PathBuf::from(smex::encode(xprv.public_key().to_bytes())).with_extension(ext); | ||||
|         if let Some(mut writer) = writer { | ||||
|             writeln!(writer, "{formatted}")?; | ||||
|         } else { | ||||
|             std::fs::write(&filename, formatted)?; | ||||
|         } | ||||
|         eprintln!("{formatted}"); | ||||
|         Ok(()) | ||||
|     } | ||||
| } | ||||
|  | @ -295,7 +159,7 @@ impl Deriver for Key { | |||
| #[derive(Parser, Debug, Clone)] | ||||
| pub struct Derive { | ||||
|     #[command(subcommand)] | ||||
|     pub(crate) command: DeriveSubcommands, | ||||
|     command: DeriveSubcommands, | ||||
| 
 | ||||
|     /// Account ID. Required for all derivations.
 | ||||
|     ///
 | ||||
|  | @ -303,45 +167,12 @@ pub struct Derive { | |||
|     /// account ID can often come as a hindrance in the future. As such, it is always required. If
 | ||||
|     /// the account ID is not relevant, it is assumed to be `0`.
 | ||||
|     #[arg(long, global = true, default_value = "0")] | ||||
|     pub(crate) account_id: u32, | ||||
| 
 | ||||
|     /// Whether derivation should return the public key or a private key.
 | ||||
|     #[arg(long, global = true)] | ||||
|     pub(crate) public: bool, | ||||
| 
 | ||||
|     /// Whether the file should be written to standard output, or to a filename generated by the
 | ||||
|     /// derivation system.
 | ||||
|     #[arg(long, global = true, default_value = "false")] | ||||
|     pub to_stdout: bool, | ||||
| 
 | ||||
|     /// The file to write the derived public key to, if not standard output. If omitted, a filename
 | ||||
|     /// will be generated by the relevant deriver.
 | ||||
|     #[arg(long, global = true, conflicts_with = "to_stdout")] | ||||
|     pub output: Option<PathBuf>, | ||||
|     account_id: u32, | ||||
| } | ||||
| 
 | ||||
| impl Derive { | ||||
|     pub fn handle(&self, _k: &Keyfork) -> Result<()> { | ||||
|         let account = DerivationIndex::new(self.account_id, true)?; | ||||
|         let writer = if let Some(output) = self.output.as_deref() { | ||||
|             Some(Box::new(std::fs::File::create(output)?) as Box<dyn Write>) | ||||
|         } else if self.to_stdout { | ||||
|             Some(Box::new(std::io::stdout()) as Box<dyn Write>) | ||||
|         } else { | ||||
|             None | ||||
|         }; | ||||
|         self.command.handle(account, self.public, writer) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl std::str::FromStr for Derive { | ||||
|     type Err = clap::Error; | ||||
| 
 | ||||
|     fn from_str(s: &str) -> std::result::Result<Self, Self::Err> { | ||||
|         Derive::try_parse_from( | ||||
|             [String::from("derive")] | ||||
|                 .into_iter() | ||||
|                 .chain(shlex::Shlex::new(s)), | ||||
|         ) | ||||
|         self.command.handle(account) | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -1,17 +1,12 @@ | |||
| use super::{ | ||||
|     create, | ||||
|     derive::{self, Deriver}, | ||||
|     provision, | ||||
|     Keyfork, | ||||
| }; | ||||
| use crate::{clap_ext::*, config, openpgp_card::factory_reset_current_card}; | ||||
| use card_backend_pcsc::PcscBackend; | ||||
| use super::provision; | ||||
| use super::Keyfork; | ||||
| use crate::{clap_ext::*, config}; | ||||
| use clap::{builder::PossibleValue, Parser, Subcommand, ValueEnum}; | ||||
| use std::{ | ||||
|     collections::HashMap, | ||||
|     fmt::Display, | ||||
|     fs::File, | ||||
|     io::{IsTerminal, Write}, | ||||
|     io::Write, | ||||
|     path::{Path, PathBuf}, | ||||
|     str::FromStr, | ||||
| }; | ||||
|  | @ -20,21 +15,17 @@ use keyfork_derive_openpgp::{ | |||
|     openpgp::{ | ||||
|         self, | ||||
|         armor::{Kind, Writer}, | ||||
|         packet::{UserID, signature::SignatureBuilder}, | ||||
|         packet::UserID, | ||||
|         policy::StandardPolicy, | ||||
|         serialize::{ | ||||
|             stream::{Encryptor2, LiteralWriter, Message, Recipient}, | ||||
|             Serialize, | ||||
|         }, | ||||
|         types::{KeyFlags, SignatureType}, | ||||
|         types::KeyFlags, | ||||
|     }, | ||||
|     XPrv, | ||||
| }; | ||||
| use keyfork_derive_util::DerivationIndex; | ||||
| use keyfork_prompt::{ | ||||
|     default_handler, prompt_validated_passphrase, | ||||
|     validators::{SecurePinValidator, Validator}, | ||||
| }; | ||||
| use keyfork_prompt::default_handler; | ||||
| use keyfork_shard::{openpgp::OpenPGP, Format}; | ||||
| 
 | ||||
| type StringMap = HashMap<String, String>; | ||||
|  | @ -159,10 +150,6 @@ pub enum Error { | |||
|     /// An error occurred when interacting iwth a file.
 | ||||
|     #[error("Error while performing IO operation on: {1}")] | ||||
|     IOContext(#[source] std::io::Error, PathBuf), | ||||
| 
 | ||||
|     /// A required option was not provided.
 | ||||
|     #[error("The required option {0} was not provided")] | ||||
|     MissingOption(&'static str), | ||||
| } | ||||
| 
 | ||||
| fn context_stub<'a>(path: &'a Path) -> impl Fn(std::io::Error) -> Error + 'a { | ||||
|  | @ -192,19 +179,6 @@ pub enum MnemonicSubcommands { | |||
|         #[arg(long, default_value_t = Default::default())] | ||||
|         size: SeedSize, | ||||
| 
 | ||||
|         /// Derive a key. By default, a private key is derived. Unlike other arguments in this
 | ||||
|         /// file, arguments must be passed using the format similar to the CLI. For example:
 | ||||
|         /// `--derive='openpgp --public "Ryan Heywood <ryan@distrust.co>"'` would be synonymous
 | ||||
|         /// with starting the Keyfork daemon with the provided mnemonic, then running
 | ||||
|         /// `keyfork derive openpgp --public "Ryan Heywood <ryan@distrust.co>"`.
 | ||||
|         ///
 | ||||
|         /// The output of the derived key is written to a filename based on the content of the key;
 | ||||
|         /// for instance, OpenPGP keys are written to a file identifiable by the certificate's
 | ||||
|         /// fingerprint. This behavior can be changed by using the `--to-stdout` or `--output`
 | ||||
|         /// modifiers to the `--derive` command.
 | ||||
|         #[arg(long)] | ||||
|         derive: Option<derive::Derive>, | ||||
| 
 | ||||
|         /// Encrypt the mnemonic to an OpenPGP certificate in the provided path.
 | ||||
|         ///
 | ||||
|         /// When given arguments in the format `--encrypt-to input.asc,output=output.asc`, the
 | ||||
|  | @ -217,7 +191,7 @@ pub enum MnemonicSubcommands { | |||
|         /// Shard the mnemonic to the certificates in the given Shardfile. Requires a decrypt
 | ||||
|         /// operation on the Shardfile to access the metadata and certificates.
 | ||||
|         ///
 | ||||
|         /// When given arguments in the format `--shard-to input.asc,output=output.asc`, the
 | ||||
|         /// When given arguments in the format `--encrypt-to input.asc,output=output.asc`, the
 | ||||
|         /// output of the encryption will be written to `output.asc`. Otherwise, the default
 | ||||
|         /// behavior is to write the output to `input.new.asc`. If the output file already exists,
 | ||||
|         /// it will not be overwritten, and the command will exit unsuccessfully.
 | ||||
|  | @ -242,56 +216,48 @@ pub enum MnemonicSubcommands { | |||
| 
 | ||||
|         /// Encrypt the mnemonic to an OpenPGP certificate derived from the mnemonic, writing the
 | ||||
|         /// output to the provided path. This command must be run in combination with
 | ||||
|         /// `--provision openpgp-card`, `--derive openpgp`, or another OpenPGP key derivation
 | ||||
|         /// mechanism, to ensure the generated mnemonic would be decryptable.
 | ||||
|         /// `--provision openpgp-card` or another relevant provisioner, to ensure the newly
 | ||||
|         /// generated mnemonic would be decryptable by some form of provisioned hardware.
 | ||||
|         ///
 | ||||
|         /// When used in combination with `--derive` or `--provision` with OpenPGP configurations,
 | ||||
|         /// the default behavior is to encrypt the mnemonic to all derived and provisioned
 | ||||
|         /// accounts. By default, the account `0` is used.
 | ||||
|         /// When given arguments in the format `--encrypt-to-self encrypted.asc,output=cert.asc`,
 | ||||
|         /// the output of the OpenPGP certificate will be written to `cert.asc`, while the output
 | ||||
|         /// of the encryption will be written to `encrypted.asc`. Otherwise, the
 | ||||
|         /// default behavior is to write the certificate to a file named after the certificate's
 | ||||
|         /// fingerprint. If either output file already exists, it will not be overwritten, and the
 | ||||
|         /// command will exit unsuccessfully. This functionality must happen regardless if a
 | ||||
|         /// provisioner output is specified, as the certificate is then used to encrypt the
 | ||||
|         /// mnemonic.
 | ||||
|         ///
 | ||||
|         /// Additionally, when given the `account=` option (which must match the `account=` option
 | ||||
|         /// of the relevant provisioner), the given account will be used instead of the default
 | ||||
|         /// account of 0.
 | ||||
|         ///
 | ||||
|         /// Because a new OpenPGP cert needs to be created, a User ID can also be supplied, using
 | ||||
|         /// the option `userid=<your User ID>`. It can contain any characters that are not a comma.
 | ||||
|         /// If any other operation generating an OpenPGP key has a `userid=` field, and this
 | ||||
|         /// operation doesn't, that User ID will be used instead.
 | ||||
|         #[arg(long)] | ||||
|         encrypt_to_self: Option<PathBuf>, | ||||
| 
 | ||||
|         /// Shard the mnemonic to freshly-generated OpenPGP certificates derived from the mnemonic,
 | ||||
|         /// writing the output to the provided path, and provisioning OpenPGP smartcards with the
 | ||||
|         /// new certificates.
 | ||||
|         ///
 | ||||
|         /// The following additional arguments are required:
 | ||||
|         ///
 | ||||
|         /// * threshold, m: the minimum amount of shares required to reconstitute the shard.
 | ||||
|         ///
 | ||||
|         /// * max, n: the maximum amount of shares.
 | ||||
|         ///
 | ||||
|         /// * cards_per_shard: the amount of OpenPGP smartcards to provision per shardholder.
 | ||||
|         ///
 | ||||
|         /// * cert_output: the file to write all generated OpenPGP certificates to; if not
 | ||||
|         /// provided, files will be automatically generated for each certificate.
 | ||||
|         #[arg(long)] | ||||
|         shard_to_self: Option<ValueWithOptions<PathBuf>>, | ||||
|         encrypt_to_self: Option<ValueWithOptions<PathBuf>>, | ||||
| 
 | ||||
|         /// Provision a key derived from the mnemonic to a piece of hardware such as an OpenPGP
 | ||||
|         /// smartcard. This argument is required when used with `--encrypt-to-self`.
 | ||||
|         ///
 | ||||
|         /// Provisioners may choose to output a public key to the current directory by default, but
 | ||||
|         /// this functionality may be altered on a by-provisioner basis by providing the `output=`
 | ||||
|         /// option to `--provisioner-config`. Additionally, Keyfork may choose to disable
 | ||||
|         /// provisioner output if a matching public key has been derived using `--derive`, which
 | ||||
|         /// may allow for controlling additional metadata that is not relevant to the provisioned
 | ||||
|         /// keys, such as an OpenPGP User ID.
 | ||||
|         /// Additional arguments, such as the amount of hardware to provision and the
 | ||||
|         /// account to use when deriving, can be specified by using (for example)
 | ||||
|         /// `--provision openpgp-card,count=2,account=1`.
 | ||||
|         ///
 | ||||
|         /// Provisioners may output their public key, if necessary. The file path may be chosen
 | ||||
|         /// based on the provided `output` field, or automatically determined based on the content
 | ||||
|         /// of the key, such as an OpenPGP fingerprint or a public key hash. If automatically
 | ||||
|         /// generated, the filename will be printed.
 | ||||
|         ///
 | ||||
|         /// If the OpenPGP Card provisioner is selected, because a new OpenPGP cert needs to be
 | ||||
|         /// created, a User ID can also be supplied, using the option `userid=<your User ID>`. It
 | ||||
|         /// can contain any characters that are not a comma.  If any other operation generating an
 | ||||
|         /// OpenPGP key has a `userid=` field, and this operation doesn't, that User ID will be
 | ||||
|         /// used instead.
 | ||||
|         #[arg(long)] | ||||
|         provision: Option<provision::Provision>, | ||||
| 
 | ||||
|         /// The amount of times the provisioner should be run. If provisioning multiple devices at
 | ||||
|         /// once, this number should be specified to the number of devices, and all devices should
 | ||||
|         /// be plugged into the system at the same time.
 | ||||
|         #[arg(long, requires = "provision", default_value = "1")] | ||||
|         provision_count: usize, | ||||
| 
 | ||||
|         /// The configuration to pass to the provisioner. These values are specific to each
 | ||||
|         /// provisioner, and should be provided in a `key=value,key=value` format. Most
 | ||||
|         /// provisioners only expect an `output=` option, to be used in place of the default output
 | ||||
|         /// path, if the provisioner needs to write data to a file, such as an OpenPGP certificate.
 | ||||
|         #[arg(long, requires = "provision", default_value_t = Options::default())] | ||||
|         provision_config: Options, | ||||
|         provision: Option<ValueWithOptions<provision::Provisioner>>, | ||||
|     }, | ||||
| } | ||||
| 
 | ||||
|  | @ -358,7 +324,7 @@ fn do_encrypt_to( | |||
|     literal_message.write_all(b"\n")?; | ||||
|     literal_message.finalize()?; | ||||
| 
 | ||||
|     let mut file = File::create(&output_file).map_err(context_stub(&output_file))?; | ||||
|     let mut file = File::create_new(&output_file).map_err(context_stub(&output_file))?; | ||||
|     if is_armored { | ||||
|         let mut writer = Writer::new(file, Kind::Message)?; | ||||
|         writer.write_all(&output)?; | ||||
|  | @ -373,12 +339,18 @@ fn do_encrypt_to( | |||
| fn do_encrypt_to_self( | ||||
|     mnemonic: &keyfork_mnemonic::Mnemonic, | ||||
|     path: &Path, | ||||
|     accounts: &[keyfork_derive_util::DerivationIndex], | ||||
|     options: &StringMap, | ||||
| ) -> Result<(), Box<dyn std::error::Error>> { | ||||
|     let mut certs = vec![]; | ||||
|     let account = options | ||||
|         .get("account") | ||||
|         .map(|account| u32::from_str(account)) | ||||
|         .transpose()? | ||||
|         .unwrap_or(0); | ||||
|     let account_index = keyfork_derive_util::DerivationIndex::new(account, true)?; | ||||
| 
 | ||||
|     for account in accounts.iter().cloned() { | ||||
|         let userid = UserID::from("Keyfork Temporary Key"); | ||||
|     let userid = options | ||||
|         .get("userid") | ||||
|         .map(|userid| UserID::from(userid.as_str())); | ||||
| 
 | ||||
|     let subkeys = [ | ||||
|         KeyFlags::empty().set_certification(), | ||||
|  | @ -393,33 +365,38 @@ fn do_encrypt_to_self( | |||
|     let xprv = XPrv::new(seed)?; | ||||
|     let derivation_path = keyfork_derive_path_data::paths::OPENPGP | ||||
|         .clone() | ||||
|             .chain_push(account); | ||||
|         .chain_push(account_index); | ||||
| 
 | ||||
|         let cert = | ||||
|             keyfork_derive_openpgp::derive(xprv.derive_path(&derivation_path)?, &subkeys, &userid)?; | ||||
|     let cert = keyfork_derive_openpgp::derive( | ||||
|         xprv.derive_path(&derivation_path)?, | ||||
|         &subkeys, | ||||
|         &userid.unwrap_or(UserID::from("Keyfork-Generated Key")), | ||||
|     )?; | ||||
| 
 | ||||
|         certs.push(cert); | ||||
|     let cert_path = match options.get("output") { | ||||
|         Some(path) => PathBuf::from(path), | ||||
|         None => { | ||||
|             let path = PathBuf::from(cert.fingerprint().to_string()).with_extension("asc"); | ||||
|             eprintln!( | ||||
|                 "Writing OpenPGP certificate to default path: {path}", | ||||
|                 path = path.display() | ||||
|             ); | ||||
|             path | ||||
|         } | ||||
|     }; | ||||
| 
 | ||||
|     let mut file = tempfile::NamedTempFile::new()?; | ||||
| 
 | ||||
|     let mut writer = Writer::new(&mut file, Kind::PublicKey)?; | ||||
|     for cert in certs { | ||||
|     let file = File::create_new(&cert_path).map_err(context_stub(&cert_path))?; | ||||
|     let mut writer = Writer::new(file, Kind::PublicKey)?; | ||||
|     cert.serialize(&mut writer)?; | ||||
|     } | ||||
|     writer.finalize()?; | ||||
| 
 | ||||
|     let temp_path = file.into_temp_path(); | ||||
| 
 | ||||
|     // a sneaky bit of DRY
 | ||||
|     do_encrypt_to( | ||||
|         mnemonic, | ||||
|         &temp_path, | ||||
|         &cert_path, | ||||
|         &StringMap::from([(String::from("output"), path.to_string_lossy().to_string())]), | ||||
|     )?; | ||||
| 
 | ||||
|     temp_path.close()?; | ||||
| 
 | ||||
|     Ok(()) | ||||
| } | ||||
| 
 | ||||
|  | @ -472,7 +449,7 @@ fn do_shard( | |||
|     let mut output = vec![]; | ||||
|     openpgp.shard_and_encrypt(threshold, max, mnemonic.as_bytes(), &certs[..], &mut output)?; | ||||
| 
 | ||||
|     let mut file = File::create(&output_file).map_err(context_stub(&output_file))?; | ||||
|     let mut file = File::create_new(&output_file).map_err(context_stub(&output_file))?; | ||||
|     if is_armored { | ||||
|         file.write_all(&output)?; | ||||
|     } else { | ||||
|  | @ -517,7 +494,7 @@ fn do_shard_to( | |||
|         &mut output, | ||||
|     )?; | ||||
| 
 | ||||
|     let mut file = File::create(&output_file).map_err(context_stub(&output_file))?; | ||||
|     let mut file = File::create_new(&output_file).map_err(context_stub(&output_file))?; | ||||
|     if is_armored { | ||||
|         file.write_all(&output)?; | ||||
|     } else { | ||||
|  | @ -533,274 +510,53 @@ fn do_shard_to( | |||
|     Ok(()) | ||||
| } | ||||
| 
 | ||||
| fn derive_key(seed: [u8; 64], index: u8) -> Result<openpgp::Cert, Box<dyn std::error::Error>> { | ||||
|     let subkeys = vec![ | ||||
|         KeyFlags::empty().set_certification(), | ||||
|         KeyFlags::empty().set_signing(), | ||||
|         KeyFlags::empty() | ||||
|             .set_transport_encryption() | ||||
|             .set_storage_encryption(), | ||||
|         KeyFlags::empty().set_authentication(), | ||||
|     ]; | ||||
| 
 | ||||
|     let subkey = DerivationIndex::new(u32::from(index), true)?; | ||||
|     let path = keyfork_derive_path_data::paths::OPENPGP_SHARD.clone().chain_push(subkey); | ||||
|     let xprv = XPrv::new(seed) | ||||
|         .expect("could not construct master key from seed") | ||||
|         .derive_path(&path)?; | ||||
|     let userid = UserID::from(format!("Keyfork Shard {index}")); | ||||
|     let cert = keyfork_derive_openpgp::derive(xprv, &subkeys, &userid)?; | ||||
|     Ok(cert) | ||||
| } | ||||
| 
 | ||||
| fn cross_sign_certs(certs: &mut [openpgp::Cert]) -> Result<(), Box<dyn std::error::Error>> { | ||||
|     let policy = StandardPolicy::new(); | ||||
| 
 | ||||
|     #[allow(clippy::unnecessary_to_owned)] | ||||
|     for signing_cert in certs.to_vec() { | ||||
|         let mut certify_key = signing_cert | ||||
|             .with_policy(&policy, None)? | ||||
|             .keys() | ||||
|             .unencrypted_secret() | ||||
|             .for_certification() | ||||
|             .next() | ||||
|             .expect("certify key unusable/not found") | ||||
|             .key() | ||||
|             .clone() | ||||
|             .into_keypair()?; | ||||
|         for signable_cert in certs.iter_mut() { | ||||
|             let sb = SignatureBuilder::new(SignatureType::GenericCertification); | ||||
|             let userid = signable_cert | ||||
|                 .userids() | ||||
|                 .next() | ||||
|                 .expect("a signable user ID is necessary to create web of trust"); | ||||
|             let signature = sb.sign_userid_binding( | ||||
|                 &mut certify_key, | ||||
|                 signable_cert.primary_key().key(), | ||||
|                 &userid, | ||||
|             )?; | ||||
|             let changed; | ||||
|             (*signable_cert, changed) = signable_cert.clone().insert_packets2(signature)?; | ||||
|             assert!( | ||||
|                 changed, | ||||
|                 "OpenPGP certificate was unchanged after inserting packets" | ||||
|             ); | ||||
|         } | ||||
|     } | ||||
|     Ok(()) | ||||
| } | ||||
| fn do_shard_to_self( | ||||
|     mnemonic: &keyfork_mnemonic::Mnemonic, | ||||
|     path: &Path, | ||||
|     options: &StringMap, | ||||
| ) -> Result<(), Box<dyn std::error::Error>> { | ||||
|     let seed = mnemonic.generate_seed(None); | ||||
|     let mut pm = default_handler()?; | ||||
|     let mut certs = vec![]; | ||||
|     let mut seen_cards = std::collections::HashSet::new(); | ||||
| 
 | ||||
|     let threshold: u8 = options | ||||
|         .get("threshold") | ||||
|         .or(options.get("m")) | ||||
|         .ok_or(Error::MissingOption("threshold"))? | ||||
|         .parse()?; | ||||
|     let max: u8 = options | ||||
|         .get("max") | ||||
|         .or(options.get("n")) | ||||
|         .ok_or(Error::MissingOption("max"))? | ||||
|         .parse()?; | ||||
|     let cards_per_shard = options | ||||
|         .get("cards_per_shard") | ||||
|         .as_deref() | ||||
|         .map(|cps| u8::from_str(cps)) | ||||
|         .transpose()?; | ||||
| 
 | ||||
|     let pin_validator = SecurePinValidator { | ||||
|         min_length: Some(8), | ||||
|         ..Default::default() | ||||
|     } | ||||
|     .to_fn(); | ||||
| 
 | ||||
|     for index in 0..max { | ||||
|         let cert = derive_key(seed, index)?; | ||||
|         for i in 0..cards_per_shard.unwrap_or(1) { | ||||
|             pm.prompt_message(keyfork_prompt::Message::Text(format!( | ||||
|                 "Please remove all keys and insert key #{} for user #{}", | ||||
|                 (i as u16) + 1, | ||||
|                 (index as u16) + 1, | ||||
|             )))?; | ||||
|             let card_backend = loop { | ||||
|                 if let Some(c) = PcscBackend::cards(None)?.next().transpose()? { | ||||
|                     break c; | ||||
|                 } | ||||
|                 pm.prompt_message(keyfork_prompt::Message::Text( | ||||
|                     "No smart card was found. Please plug in a smart card and press enter" | ||||
|                         .to_string(), | ||||
|                 ))?; | ||||
|             }; | ||||
|             let pin = prompt_validated_passphrase( | ||||
|                 &mut *pm, | ||||
|                 "Please enter the new smartcard PIN: ", | ||||
|                 3, | ||||
|                 &pin_validator, | ||||
|             )?; | ||||
|             factory_reset_current_card( | ||||
|                 &mut |application_identifier| { | ||||
|                     if seen_cards.contains(&application_identifier) { | ||||
|                         // we were given a previously-seen card, error
 | ||||
|                         // we're gonna panic because this is a significant error
 | ||||
|                         panic!("Previously used card {application_identifier} was reused"); | ||||
|                     } else { | ||||
|                         seen_cards.insert(application_identifier); | ||||
|                         true | ||||
|                     } | ||||
|                 }, | ||||
|                 pin.trim(), | ||||
|                 pin.trim(), | ||||
|                 &cert, | ||||
|                 &openpgp::policy::NullPolicy::new(), | ||||
|                 card_backend, | ||||
|             )?; | ||||
|         } | ||||
|         certs.push(cert); | ||||
|     } | ||||
| 
 | ||||
|     cross_sign_certs(&mut certs)?; | ||||
| 
 | ||||
|     let opgp = OpenPGP; | ||||
|     let output = File::create(path)?; | ||||
|     opgp.shard_and_encrypt( | ||||
|         threshold, | ||||
|         certs.len() as u8, | ||||
|         mnemonic.as_bytes(), | ||||
|         &certs[..], | ||||
|         output, | ||||
|     )?; | ||||
| 
 | ||||
|     match options.get("cert_output") { | ||||
|         Some(path) => { | ||||
|             let cert_file = std::fs::File::create(path)?; | ||||
|             let mut writer = Writer::new(cert_file, Kind::PublicKey)?; | ||||
|             for cert in &certs { | ||||
|                 cert.serialize(&mut writer)?; | ||||
|             } | ||||
|             writer.finalize()?; | ||||
|         } | ||||
|         None => { | ||||
|             for cert in &certs { | ||||
|                 let path = PathBuf::from(cert.fingerprint().to_string()).with_extension("asc"); | ||||
|                 let file = create(&path)?; | ||||
|                 let mut writer = Writer::new(file, Kind::PublicKey)?; | ||||
|                 cert.serialize(&mut writer)?; | ||||
|                 writer.finalize()?; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     Ok(()) | ||||
| } | ||||
| 
 | ||||
| fn do_provision( | ||||
|     mnemonic: &keyfork_mnemonic::Mnemonic, | ||||
|     provision: &provision::Provision, | ||||
|     count: usize, | ||||
|     config: &HashMap<String, String>, | ||||
|     provisioner: &provision::Provisioner, | ||||
|     options: &StringMap, | ||||
| ) -> Result<(), Box<dyn std::error::Error>> { | ||||
|     assert!( | ||||
|         provision.subcommand.is_none(), | ||||
|         "provisioner was given a subcommand; this functionality is not supported" | ||||
|     ); | ||||
| 
 | ||||
|     let identifiers = match &provision.identifier { | ||||
|         Some(identifier) => { | ||||
|             vec![identifier.clone()] | ||||
|         } | ||||
|         None => provision | ||||
|             .provisioner_name | ||||
|     let mut options = options.clone(); | ||||
|     let account = options | ||||
|         .remove("account") | ||||
|         .map(|account| u32::from_str(&account)) | ||||
|         .transpose()? | ||||
|         .unwrap_or(0); | ||||
|     let identifier = options | ||||
|         .remove("identifier") | ||||
|         .map(|s| s.split('.').map(String::from).collect::<Vec<_>>()) | ||||
|         .map(Result::<_, Box<dyn std::error::Error>>::Ok) | ||||
|         .unwrap_or_else(|| { | ||||
|             Ok(provisioner | ||||
|                 .discover()? | ||||
|                 .into_iter() | ||||
|             .map(|(name, _ctx)| name) | ||||
|             .collect(), | ||||
|     }; | ||||
|                 .map(|(identifier, _)| identifier) | ||||
|                 .collect()) | ||||
|         })?; | ||||
|     let count = options | ||||
|         .remove("count") | ||||
|         .map(|count| usize::from_str(&count)) | ||||
|         .transpose()? | ||||
|         .unwrap_or(identifier.len()); | ||||
| 
 | ||||
|     assert_eq!( | ||||
|         identifiers.len(), | ||||
|         count, | ||||
|         "amount of provisionable devices discovered did not match provisioner count" | ||||
|         identifier.len(), | ||||
|         "amount of identifiers discovered or provided did not match provisioner count" | ||||
|     ); | ||||
| 
 | ||||
|     for identifier in identifiers { | ||||
|         let provisioner_with_identifier = provision::Provision { | ||||
|             identifier: Some(identifier), | ||||
|             ..provision.clone() | ||||
|     for (_, identifier) in (0..count).zip(identifier.into_iter()) { | ||||
|         let provisioner_config = config::Provisioner { | ||||
|             account, | ||||
|             identifier, | ||||
|             metadata: Some(options.clone()), | ||||
|         }; | ||||
|         let mut provisioner = config::Provisioner::try_from(provisioner_with_identifier)?; | ||||
|         match &mut provisioner.metadata { | ||||
|             Some(metadata) => { | ||||
|                 metadata.extend(config.clone().into_iter()); | ||||
|             } | ||||
|             metadata @ None => { | ||||
|                 *metadata = Some(config.clone()); | ||||
|             } | ||||
|         }; | ||||
|         provision | ||||
|             .provisioner_name | ||||
|             .provision_with_mnemonic(mnemonic, provisioner)?; | ||||
| 
 | ||||
|         provisioner.provision_with_mnemonic(mnemonic, provisioner_config.clone())?; | ||||
|     } | ||||
| 
 | ||||
|     Ok(()) | ||||
| } | ||||
| 
 | ||||
| fn do_derive( | ||||
|     mnemonic: &keyfork_mnemonic::MnemonicBase<keyfork_mnemonic::English>, | ||||
|     deriver: &derive::Derive, | ||||
| ) -> Result<(), Box<dyn std::error::Error>> { | ||||
|     let writer = if let Some(output) = deriver.output.as_deref() { | ||||
|         Some(Box::new(std::fs::File::create(output)?) as Box<dyn Write>) | ||||
|     } else if deriver.to_stdout { | ||||
|         Some(Box::new(std::io::stdout()) as Box<dyn Write>) | ||||
|     } else { | ||||
|         None | ||||
|     }; | ||||
|     match deriver { | ||||
|         derive::Derive { | ||||
|             command: derive::DeriveSubcommands::OpenPGP(opgp), | ||||
|             account_id, | ||||
|             public, | ||||
|             .. | ||||
|         } => { | ||||
|             use keyfork_derive_openpgp::XPrv; | ||||
|             let root_xprv = XPrv::new(mnemonic.generate_seed(None))?; | ||||
|             let account = DerivationIndex::new(*account_id, true)?; | ||||
|             let derived = root_xprv.derive_path(&opgp.derivation_path().chain_push(account))?; | ||||
|             if *public { | ||||
|                 opgp.derive_public_with_xprv(writer, derived)?; | ||||
|             } else { | ||||
|                 opgp.derive_with_xprv(writer, derived)?; | ||||
|             } | ||||
|         } | ||||
|         derive::Derive { | ||||
|             command: derive::DeriveSubcommands::Key(key), | ||||
|             account_id, | ||||
|             public, | ||||
|             .. | ||||
|         } => { | ||||
|             // HACK: We're abusing that we use the same key as OpenPGP. Maybe
 | ||||
|             // we should use ed25519_dalek.
 | ||||
|             use keyfork_derive_openpgp::XPrv; | ||||
|             let root_xprv = XPrv::new(mnemonic.generate_seed(None))?; | ||||
|             let account = DerivationIndex::new(*account_id, true)?; | ||||
|             let derived = root_xprv.derive_path(&key.derivation_path().chain_push(account))?; | ||||
|             if *public { | ||||
|                 key.derive_public_with_xprv(writer, derived)?; | ||||
|             } else { | ||||
|                 key.derive_with_xprv(writer, derived)?; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     Ok(()) | ||||
| } | ||||
| 
 | ||||
| impl MnemonicSubcommands { | ||||
|     pub fn handle( | ||||
|         &self, | ||||
|  | @ -811,47 +567,25 @@ impl MnemonicSubcommands { | |||
|             MnemonicSubcommands::Generate { | ||||
|                 source, | ||||
|                 size, | ||||
|                 derive, | ||||
|                 encrypt_to, | ||||
|                 shard_to, | ||||
|                 shard, | ||||
|                 encrypt_to_self, | ||||
|                 shard_to_self, | ||||
|                 provision, | ||||
|                 provision_count, | ||||
|                 provision_config, | ||||
|             } => { | ||||
|                 // NOTE: We should never have a case where there's Some() of empty vec, but
 | ||||
|                 // we will make sure to check it just in case.
 | ||||
|                 //
 | ||||
|                 // We do not print the mnemonic if we are:
 | ||||
|                 // * Encrypting to an existing, usable key
 | ||||
|                 // * Encrypting to a newly provisioned key
 | ||||
|                 // * Sharding to an existing Shardfile with usable keys
 | ||||
|                 // * Sharding to existing, usable keys
 | ||||
|                 // * Sharding to newly provisioned keys
 | ||||
|                 let mut will_print_mnemonic = | ||||
|                     encrypt_to.is_none() || encrypt_to.as_ref().is_some_and(|e| e.is_empty()); | ||||
|                 will_print_mnemonic = will_print_mnemonic | ||||
|                     && (encrypt_to_self.as_ref().is_none() || provision.as_ref().is_none()); | ||||
|                 will_print_mnemonic = will_print_mnemonic && shard_to.is_none() | ||||
|                     || shard_to.as_ref().is_some_and(|s| s.is_empty()); | ||||
|                 will_print_mnemonic = will_print_mnemonic && shard.is_none() | ||||
|                     || shard.as_ref().is_some_and(|s| s.is_empty()); | ||||
|                 will_print_mnemonic = will_print_mnemonic && shard_to_self.is_none(); | ||||
|                 will_print_mnemonic = will_print_mnemonic | ||||
|                     && (encrypt_to_self.as_ref().is_none() || provision.as_ref().is_none()); | ||||
| 
 | ||||
|                 let mnemonic = source.handle(size)?; | ||||
| 
 | ||||
|                 if let Some(derive) = derive { | ||||
|                     let stdout = std::io::stdout(); | ||||
|                     if will_print_mnemonic && !stdout.is_terminal() { | ||||
|                         eprintln!( | ||||
|                             "Writing plaintext mnemonic and derivation output to standard output" | ||||
|                         ); | ||||
|                     } | ||||
|                     do_derive(&mnemonic, derive)?; | ||||
|                 } | ||||
| 
 | ||||
|                 if let Some(encrypt_to) = encrypt_to { | ||||
|                     for entry in encrypt_to { | ||||
|                         do_encrypt_to(&mnemonic, &entry.inner, &entry.values)?; | ||||
|  | @ -859,70 +593,46 @@ impl MnemonicSubcommands { | |||
|                 } | ||||
| 
 | ||||
|                 if let Some(encrypt_to_self) = encrypt_to_self { | ||||
|                     let mut accounts: std::collections::HashSet<u32> = Default::default(); | ||||
|                     if let Some(provision::Provision { | ||||
|                         provisioner_name: provision::Provisioner::OpenPGPCard(_), | ||||
|                         account_id, | ||||
|                         .. | ||||
|                     }) = provision | ||||
|                     let mut values = encrypt_to_self.values.clone(); | ||||
|                     // If we have a userid from `provision` but not one here, use that one.
 | ||||
|                     if let Some(provision) = provision { | ||||
|                         if matches!(&provision.inner, provision::Provisioner::OpenPGPCard(_)) | ||||
|                             && !values.contains_key("userid") | ||||
|                         { | ||||
|                         accounts.insert(*account_id); | ||||
|                             if let Some(userid) = provision.values.get("userid") { | ||||
|                                 values.insert(String::from("userid"), userid.clone()); | ||||
|                             } | ||||
|                     if let Some(derive::Derive { | ||||
|                         command: derive::DeriveSubcommands::OpenPGP(_), | ||||
|                         account_id, | ||||
|                         .. | ||||
|                     }) = derive | ||||
|                     { | ||||
|                         accounts.insert(*account_id); | ||||
|                         } | ||||
|                     let indices = accounts | ||||
|                         .into_iter() | ||||
|                         .map(|i| DerivationIndex::new(i, true)) | ||||
|                         .collect::<Result<Vec<_>, _>>()?; | ||||
|                     assert!( | ||||
|                         !indices.is_empty(), | ||||
|                         "neither derived nor provisioned accounts were found" | ||||
|                     ); | ||||
|                     do_encrypt_to_self(&mnemonic, &encrypt_to_self, &indices)?; | ||||
|                     } | ||||
| 
 | ||||
|                 if let Some(shard_to_self) = shard_to_self { | ||||
|                     do_shard_to_self(&mnemonic, &shard_to_self.inner, &shard_to_self.values)?; | ||||
|                     do_encrypt_to_self(&mnemonic, &encrypt_to_self.inner, &values)?; | ||||
|                 } | ||||
| 
 | ||||
|                 if let Some(provisioner) = provision { | ||||
|                     // determine if we should write to standard output based on whether we have a
 | ||||
|                     // matching pair of provisioner and public derivation output.
 | ||||
|                     let mut will_output_public_key = true; | ||||
| 
 | ||||
|                     if let Some(derive) = derive { | ||||
|                         let matches = match (provisioner, derive) { | ||||
|                             ( | ||||
|                                 provision::Provision { | ||||
|                                     provisioner_name: provision::Provisioner::OpenPGPCard(_), | ||||
|                                     account_id: p_id, | ||||
|                                     .. | ||||
|                                 }, | ||||
|                                 derive::Derive { | ||||
|                                     command: derive::DeriveSubcommands::OpenPGP(_), | ||||
|                                     account_id: d_id, | ||||
|                                     .. | ||||
|                                 }, | ||||
|                             ) => p_id == d_id, | ||||
|                             _ => false, | ||||
|                         }; | ||||
|                         if matches && derive.public { | ||||
|                             will_output_public_key = false; | ||||
|                         } | ||||
|                     } | ||||
| 
 | ||||
|                     let mut values = provision_config.values.clone(); | ||||
|                     if !will_output_public_key && !values.contains_key("output") { | ||||
|                     // NOTE: If we have encrypt_to_self, we likely also have the certificate
 | ||||
|                     // already generated. Therefore, we can skip generating it in the provisioner.
 | ||||
|                     // However, if we don't have encrypt_to_self, we might not have the
 | ||||
|                     // certificate, therefore the provisioner - by default - generates the public
 | ||||
|                     // key output.
 | ||||
|                     //
 | ||||
|                     // We use the atypical `_skip_cert_output` field here to denote an automatic
 | ||||
|                     // marking to skip the cert output. However, the `output` field will take
 | ||||
|                     // priority, since it can only be manually set by the user.
 | ||||
|                     let mut values = provisioner.values.clone(); | ||||
|                     if let Some(encrypt_to_self) = encrypt_to_self { | ||||
|                         if !values.contains_key("output") { | ||||
|                             values.insert(String::from("_skip_cert_output"), String::from("1")); | ||||
|                         } | ||||
| 
 | ||||
|                     do_provision(&mnemonic, provisioner, *provision_count, &values)?; | ||||
|                         // If we have a userid from `encrypt_to_self` but not one here, use that
 | ||||
|                         // one.
 | ||||
|                         if matches!(&provisioner.inner, provision::Provisioner::OpenPGPCard(_)) | ||||
|                             && !values.contains_key("userid") | ||||
|                         { | ||||
|                             if let Some(userid) = encrypt_to_self.values.get("userid") { | ||||
|                                 values.insert(String::from("userid"), userid.clone()); | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                     do_provision(&mnemonic, &provisioner.inner, &values)?; | ||||
|                 } | ||||
| 
 | ||||
|                 if let Some(shard_to) = shard_to { | ||||
|  |  | |||
|  | @ -5,11 +5,7 @@ mod mnemonic; | |||
| mod provision; | ||||
| mod recover; | ||||
| mod shard; | ||||
| 
 | ||||
| pub fn create(path: &std::path::Path) -> std::io::Result<std::fs::File> { | ||||
|     eprintln!("Writing derived key to: {path}", path=path.display()); | ||||
|     std::fs::File::create(path) | ||||
| } | ||||
| mod wizard; | ||||
| 
 | ||||
| /// The Kitchen Sink of Entropy.
 | ||||
| #[derive(Parser, Clone, Debug)] | ||||
|  | @ -61,6 +57,9 @@ pub enum KeyforkCommands { | |||
|     /// leaked by any individual deriver.
 | ||||
|     Recover(recover::Recover), | ||||
| 
 | ||||
|     /// Utilities to automatically manage the setup of Keyfork.
 | ||||
|     Wizard(wizard::Wizard), | ||||
| 
 | ||||
|     /// Print an autocompletion file to standard output.
 | ||||
|     ///
 | ||||
|     /// Keyfork does not manage the installation of completion files. Consult the documentation for
 | ||||
|  | @ -91,6 +90,9 @@ impl KeyforkCommands { | |||
|             KeyforkCommands::Recover(r) => { | ||||
|                 r.handle(keyfork)?; | ||||
|             } | ||||
|             KeyforkCommands::Wizard(w) => { | ||||
|                 w.handle(keyfork)?; | ||||
|             } | ||||
|             #[cfg(feature = "completion")] | ||||
|             KeyforkCommands::Completion { shell } => { | ||||
|                 let mut command = Keyfork::command(); | ||||
|  |  | |||
|  | @ -12,27 +12,20 @@ type Identifier = (String, Option<String>); | |||
| #[derive(Debug, Clone)] | ||||
| pub enum Provisioner { | ||||
|     OpenPGPCard(openpgp::OpenPGPCard), | ||||
|     Shard(openpgp::Shard), | ||||
| } | ||||
| 
 | ||||
| impl std::fmt::Display for Provisioner { | ||||
|     fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { | ||||
|         f.write_str(self.identifier()) | ||||
|         match self { | ||||
|             Provisioner::OpenPGPCard(_) => f.write_str("openpgp-card"), | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl Provisioner { | ||||
|     pub fn identifier(&self) -> &'static str { | ||||
|         match self { | ||||
|             Provisioner::OpenPGPCard(_) => "openpgp-card", | ||||
|             Provisioner::Shard(_) => "shard", | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     pub fn discover(&self) -> Result<Vec<Identifier>, Box<dyn std::error::Error>> { | ||||
|         match self { | ||||
|             Provisioner::OpenPGPCard(o) => o.discover(), | ||||
|             Provisioner::Shard(s) => s.discover(), | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|  | @ -51,16 +44,6 @@ impl Provisioner { | |||
|                 let xprv: XPrv = client.request_xprv(&path)?; | ||||
|                 o.provision(xprv, provisioner) | ||||
|             } | ||||
|             Provisioner::Shard(s) => { | ||||
|                 type Prv = <openpgp::Shard as ProvisionExec>::PrivateKey; | ||||
|                 type XPrv = ExtendedPrivateKey<Prv>; | ||||
|                 let account_index = DerivationIndex::new(provisioner.account, true)?; | ||||
|                 let path = <openpgp::Shard as ProvisionExec>::derivation_prefix() | ||||
|                     .chain_push(account_index); | ||||
|                 let mut client = keyforkd_client::Client::discover_socket()?; | ||||
|                 let xprv: XPrv = client.request_xprv(&path)?; | ||||
|                 s.provision(xprv, provisioner) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|  | @ -79,26 +62,19 @@ impl Provisioner { | |||
|                 let xprv = XPrv::new(mnemonic.generate_seed(None))?.derive_path(&path)?; | ||||
|                 o.provision(xprv, provisioner) | ||||
|             } | ||||
|             Provisioner::Shard(s) => { | ||||
|                 type Prv = <openpgp::Shard as ProvisionExec>::PrivateKey; | ||||
|                 type XPrv = ExtendedPrivateKey<Prv>; | ||||
|                 let account_index = DerivationIndex::new(provisioner.account, true)?; | ||||
|                 let path = <openpgp::Shard as ProvisionExec>::derivation_prefix() | ||||
|                     .chain_push(account_index); | ||||
|                 let xprv = XPrv::new(mnemonic.generate_seed(None))?.derive_path(&path)?; | ||||
|                 s.provision(xprv, provisioner) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl ValueEnum for Provisioner { | ||||
|     fn value_variants<'a>() -> &'a [Self] { | ||||
|         &[Self::OpenPGPCard(openpgp::OpenPGPCard), Self::Shard(openpgp::Shard)] | ||||
|         &[Self::OpenPGPCard(openpgp::OpenPGPCard)] | ||||
|     } | ||||
| 
 | ||||
|     fn to_possible_value(&self) -> Option<PossibleValue> { | ||||
|         Some(PossibleValue::new(self.identifier())) | ||||
|         Some(PossibleValue::new(match self { | ||||
|             Self::OpenPGPCard(_) => "openpgp-card", | ||||
|         })) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
|  | @ -148,27 +124,15 @@ pub struct Provision { | |||
|     #[command(subcommand)] | ||||
|     pub subcommand: Option<ProvisionSubcommands>, | ||||
| 
 | ||||
|     pub provisioner_name: Provisioner, | ||||
|     provisioner_name: Provisioner, | ||||
| 
 | ||||
|     /// Account ID.
 | ||||
|     #[arg(long, default_value = "0")] | ||||
|     pub account_id: u32, | ||||
|     #[arg(long, required(true))] | ||||
|     account_id: Option<u32>, | ||||
| 
 | ||||
|     /// Identifier of the hardware to deploy to, listable by running the `discover` subcommand.
 | ||||
|     #[arg(long)] | ||||
|     pub identifier: Option<String>, | ||||
| } | ||||
| 
 | ||||
| impl std::str::FromStr for Provision { | ||||
|     type Err = clap::Error; | ||||
| 
 | ||||
|     fn from_str(s: &str) -> std::result::Result<Self, Self::Err> { | ||||
|         Provision::try_parse_from( | ||||
|             [String::from("provision")] | ||||
|                 .into_iter() | ||||
|                 .chain(shlex::Shlex::new(s)), | ||||
|         ) | ||||
|     } | ||||
|     #[arg(long, required(true))] | ||||
|     identifier: Option<String>, | ||||
| } | ||||
| 
 | ||||
| // NOTE: In the future, this impl will be used by `keyfork recover` to reprovision hardware from
 | ||||
|  | @ -184,7 +148,7 @@ impl TryFrom<Provision> for config::Provisioner { | |||
| 
 | ||||
|     fn try_from(value: Provision) -> Result<Self, Self::Error> { | ||||
|         Ok(Self { | ||||
|             account: value.account_id, | ||||
|             account: value.account_id.ok_or(MissingField("account_id"))?, | ||||
|             identifier: value.identifier.ok_or(MissingField("identifier"))?, | ||||
|             metadata: Default::default(), | ||||
|         }) | ||||
|  | @ -207,21 +171,7 @@ impl Provision { | |||
|                 } | ||||
|             } | ||||
|             None => { | ||||
|                 let provisioner_with_identifier = match self.identifier { | ||||
|                     Some(_) => self.clone(), | ||||
|                     None => { | ||||
|                         let identifiers = self.provisioner_name.discover()?; | ||||
|                         let [id] = &identifiers[..] else { | ||||
|                             panic!("invalid amount of identifiers; pass --identifier"); | ||||
|                         }; | ||||
|                         Self { | ||||
|                             identifier: Some(id.0.clone()), | ||||
|                             ..self.clone() | ||||
|                         } | ||||
|                     } | ||||
|                 }; | ||||
|                 let config = config::Provisioner::try_from(provisioner_with_identifier)?; | ||||
|                 self.provisioner_name.provision(config)?; | ||||
|                 self.provisioner_name.provision(self.clone().try_into()?)?; | ||||
|             } | ||||
|         } | ||||
|         Ok(()) | ||||
|  |  | |||
|  | @ -1,8 +1,5 @@ | |||
| use super::ProvisionExec; | ||||
| use crate::{ | ||||
|     config, | ||||
|     openpgp_card::{factory_reset_current_card, get_new_pins}, | ||||
| }; | ||||
| use crate::{config, openpgp_card::factory_reset_current_card}; | ||||
| 
 | ||||
| use card_backend_pcsc::PcscBackend; | ||||
| use keyfork_derive_openpgp::{ | ||||
|  | @ -14,15 +11,24 @@ use keyfork_derive_openpgp::{ | |||
|     }, | ||||
|     XPrv, | ||||
| }; | ||||
| use keyfork_prompt::default_handler; | ||||
| use keyfork_prompt::{ | ||||
|     default_handler, prompt_validated_passphrase, | ||||
|     validators::{SecurePinValidator, Validator}, | ||||
| }; | ||||
| use openpgp_card_sequoia::{state::Open, Card}; | ||||
| use std::path::PathBuf; | ||||
| 
 | ||||
| #[derive(Clone, Debug)] | ||||
| pub struct OpenPGPCard; | ||||
| 
 | ||||
| #[derive(thiserror::Error, Debug)] | ||||
| #[error("Provisioner was unable to find a matching smartcard")] | ||||
| struct NoMatchingSmartcard; | ||||
| 
 | ||||
| fn discover_cards() -> Result<Vec<(String, Option<String>)>, Box<dyn std::error::Error>> { | ||||
| impl ProvisionExec for OpenPGPCard { | ||||
|     type PrivateKey = keyfork_derive_openpgp::XPrvKey; | ||||
| 
 | ||||
|     fn discover(&self) -> Result<Vec<(String, Option<String>)>, Box<dyn std::error::Error>> { | ||||
|         let mut idents = vec![]; | ||||
|         for backend in PcscBackend::cards(None)? { | ||||
|             let backend = backend?; | ||||
|  | @ -36,13 +42,39 @@ fn discover_cards() -> Result<Vec<(String, Option<String>)>, Box<dyn std::error: | |||
|         Ok(idents) | ||||
|     } | ||||
| 
 | ||||
| fn provision_card( | ||||
|     provisioner: config::Provisioner, | ||||
|     fn derivation_prefix() -> keyfork_derive_util::DerivationPath { | ||||
|         keyfork_derive_path_data::paths::OPENPGP.clone() | ||||
|     } | ||||
| 
 | ||||
|     fn provision( | ||||
|         &self, | ||||
|         xprv: XPrv, | ||||
|         provisioner: config::Provisioner, | ||||
|     ) -> Result<(), Box<dyn std::error::Error>> { | ||||
|         let mut pm = default_handler()?; | ||||
|         let user_pin_validator = SecurePinValidator { | ||||
|             min_length: Some(6), | ||||
|             ..Default::default() | ||||
|         } | ||||
|         .to_fn(); | ||||
|         let admin_pin_validator = SecurePinValidator { | ||||
|             min_length: Some(8), | ||||
|             ..Default::default() | ||||
|         } | ||||
|         .to_fn(); | ||||
| 
 | ||||
|     let (user_pin, admin_pin) = get_new_pins(&mut *pm)?; | ||||
|         let user_pin = prompt_validated_passphrase( | ||||
|             &mut *pm, | ||||
|             "Please enter the new smartcard User PIN: ", | ||||
|             3, | ||||
|             &user_pin_validator, | ||||
|         )?; | ||||
|         let admin_pin = prompt_validated_passphrase( | ||||
|             &mut *pm, | ||||
|             "Please enter the new smartcard Admin PIN: ", | ||||
|             3, | ||||
|             &admin_pin_validator, | ||||
|         )?; | ||||
| 
 | ||||
|         let subkeys = vec![ | ||||
|             KeyFlags::empty().set_certification(), | ||||
|  | @ -64,7 +96,11 @@ fn provision_card( | |||
|             .as_ref() | ||||
|             .is_some_and(|m| m.contains_key("_skip_cert_output")) | ||||
|         { | ||||
|         let cert_output = match provisioner.metadata.as_ref().and_then(|m| m.get("output")) { | ||||
|             let cert_output = match provisioner | ||||
|                 .metadata | ||||
|                 .as_ref() | ||||
|                 .and_then(|m| m.get("output")) | ||||
|             { | ||||
|                 Some(cert_output) => PathBuf::from(cert_output), | ||||
|                 None => { | ||||
|                     let path = PathBuf::from(cert.fingerprint().to_string()).with_extension("asc"); | ||||
|  | @ -76,7 +112,7 @@ fn provision_card( | |||
|                 } | ||||
|             }; | ||||
| 
 | ||||
|         let cert_output_file = std::fs::File::create(cert_output)?; | ||||
|             let cert_output_file = std::fs::File::create_new(cert_output)?; | ||||
|             let mut writer = Writer::new(cert_output_file, Kind::PublicKey)?; | ||||
|             cert.serialize(&mut writer)?; | ||||
|             writer.finalize()?; | ||||
|  | @ -105,49 +141,4 @@ fn provision_card( | |||
| 
 | ||||
|         Ok(()) | ||||
|     } | ||||
| 
 | ||||
| #[derive(Clone, Debug)] | ||||
| pub struct OpenPGPCard; | ||||
| 
 | ||||
| impl ProvisionExec for OpenPGPCard { | ||||
|     type PrivateKey = keyfork_derive_openpgp::XPrvKey; | ||||
| 
 | ||||
|     fn discover(&self) -> Result<Vec<(String, Option<String>)>, Box<dyn std::error::Error>> { | ||||
|         discover_cards() | ||||
|     } | ||||
| 
 | ||||
|     fn derivation_prefix() -> keyfork_derive_util::DerivationPath { | ||||
|         keyfork_derive_path_data::paths::OPENPGP.clone() | ||||
|     } | ||||
| 
 | ||||
|     fn provision( | ||||
|         &self, | ||||
|         xprv: XPrv, | ||||
|         provisioner: config::Provisioner, | ||||
|     ) -> Result<(), Box<dyn std::error::Error>> { | ||||
|         provision_card(provisioner, xprv) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| #[derive(Clone, Debug)] | ||||
| pub struct Shard; | ||||
| 
 | ||||
| impl ProvisionExec for Shard { | ||||
|     type PrivateKey = keyfork_derive_openpgp::XPrvKey; | ||||
| 
 | ||||
|     fn discover(&self) -> Result<Vec<(String, Option<String>)>, Box<dyn std::error::Error>> { | ||||
|         discover_cards() | ||||
|     } | ||||
| 
 | ||||
|     fn derivation_prefix() -> keyfork_derive_util::DerivationPath { | ||||
|         keyfork_derive_path_data::paths::OPENPGP_SHARD.clone() | ||||
|     } | ||||
| 
 | ||||
|     fn provision( | ||||
|         &self, | ||||
|         xprv: XPrv, | ||||
|         provisioner: config::Provisioner, | ||||
|     ) -> Result<(), Box<dyn std::error::Error>> { | ||||
|         provision_card(provisioner, xprv) | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -0,0 +1,329 @@ | |||
| use super::Keyfork; | ||||
| use crate::openpgp_card::factory_reset_current_card; | ||||
| use clap::{Args, Parser, Subcommand}; | ||||
| use std::{collections::HashSet, fs::File, io::IsTerminal, path::PathBuf}; | ||||
| 
 | ||||
| use card_backend_pcsc::PcscBackend; | ||||
| 
 | ||||
| use keyfork_derive_openpgp::{ | ||||
|     openpgp::{ | ||||
|         self, | ||||
|         armor::{Kind, Writer}, | ||||
|         packet::{signature::SignatureBuilder, UserID}, | ||||
|         policy::StandardPolicy, | ||||
|         serialize::Marshal, | ||||
|         types::{KeyFlags, SignatureType}, | ||||
|         Cert, | ||||
|     }, | ||||
|     XPrv, | ||||
| }; | ||||
| use keyfork_derive_path_data::paths; | ||||
| use keyfork_derive_util::DerivationIndex; | ||||
| use keyfork_mnemonic::Mnemonic; | ||||
| use keyfork_prompt::{ | ||||
|     default_handler, prompt_validated_passphrase, | ||||
|     validators::{SecurePinValidator, Validator}, | ||||
|     Message, | ||||
| }; | ||||
| 
 | ||||
| use keyfork_shard::{openpgp::OpenPGP, Format}; | ||||
| 
 | ||||
| #[derive(thiserror::Error, Debug)] | ||||
| #[error("Invalid PIN length: {0}")] | ||||
| pub struct PinLength(usize); | ||||
| 
 | ||||
| type Result<T, E = Box<dyn std::error::Error>> = std::result::Result<T, E>; | ||||
| 
 | ||||
| // TODO: refactor to use mnemonic derived seed instead of 256 bit entropy to allow for possible
 | ||||
| // recovery in the future.
 | ||||
| fn derive_key(seed: [u8; 32], index: u8) -> Result<Cert> { | ||||
|     let subkeys = vec![ | ||||
|         KeyFlags::empty().set_certification(), | ||||
|         KeyFlags::empty().set_signing(), | ||||
|         KeyFlags::empty() | ||||
|             .set_transport_encryption() | ||||
|             .set_storage_encryption(), | ||||
|         KeyFlags::empty().set_authentication(), | ||||
|     ]; | ||||
| 
 | ||||
|     let subkey = DerivationIndex::new(u32::from(index), true)?; | ||||
|     let path = paths::OPENPGP_SHARD.clone().chain_push(subkey); | ||||
|     let xprv = XPrv::new(seed) | ||||
|         .expect("could not construct master key from seed") | ||||
|         .derive_path(&path)?; | ||||
|     let userid = UserID::from(format!("Keyfork Shard {index}")); | ||||
|     let cert = keyfork_derive_openpgp::derive(xprv, &subkeys, &userid)?; | ||||
|     Ok(cert) | ||||
| } | ||||
| 
 | ||||
| #[derive(Subcommand, Clone, Debug)] | ||||
| pub enum WizardSubcommands { | ||||
|     GenerateShardSecret(GenerateShardSecret), | ||||
|     BottomsUp(BottomsUp), | ||||
| } | ||||
| 
 | ||||
| /// Create a 256 bit secret and shard the secret to smart cards.
 | ||||
| ///
 | ||||
| /// Smart cards will need to be plugged in periodically during the wizard, where they will be
 | ||||
| /// factory reset and provisioned to `m/pgp'/shrd'/<share index>`. The secret can then be recovered
 | ||||
| /// with `keyfork recover shard` or `keyfork recover remote-shard`. The share file will be printed
 | ||||
| /// to standard output.
 | ||||
| #[derive(Args, Clone, Debug)] | ||||
| pub struct GenerateShardSecret { | ||||
|     /// The minimum amount of keys required to decrypt the secret.
 | ||||
|     #[arg(long)] | ||||
|     threshold: u8, | ||||
| 
 | ||||
|     /// The maximum amount of shards.
 | ||||
|     #[arg(long)] | ||||
|     max: u8, | ||||
| 
 | ||||
|     /// The amount of smart cards to provision per-shard.
 | ||||
|     #[arg(long, default_value = "1")] | ||||
|     keys_per_shard: u8, | ||||
| 
 | ||||
|     /// The file to write the generated shard file to.
 | ||||
|     #[arg(long)] | ||||
|     output: Option<PathBuf>, | ||||
| 
 | ||||
|     /// The file to write generated certificates to.
 | ||||
|     #[arg(long)] | ||||
|     cert_output: Option<PathBuf>, | ||||
| } | ||||
| 
 | ||||
| /// Create a 256 bit secret and shard the secret to previously known OpenPGP certificates,
 | ||||
| /// deriving the default OpenPGP certificate for the secret.
 | ||||
| ///
 | ||||
| /// This command was purpose-built for DEFCON and is not intended to be used normally, as it
 | ||||
| /// implies keys used for sharding have been generated by a custom source.
 | ||||
| #[derive(Args, Clone, Debug)] | ||||
| pub struct BottomsUp { | ||||
|     /// The location of OpenPGP certificates to use when sharding.
 | ||||
|     key_discovery: PathBuf, | ||||
| 
 | ||||
|     /// The minimum amount of keys required to decrypt the secret.
 | ||||
|     #[arg(long)] | ||||
|     threshold: u8, | ||||
| 
 | ||||
|     /// The file to write the generated shard file to.
 | ||||
|     #[arg(long)] | ||||
|     output_shardfile: PathBuf, | ||||
| 
 | ||||
|     /// The file to write the generated OpenPGP certificate to.
 | ||||
|     #[arg(long)] | ||||
|     output_cert: PathBuf, | ||||
| 
 | ||||
|     /// The User ID for the generated OpenPGP certificate.
 | ||||
|     #[arg(long, default_value = "Disaster Recovery")] | ||||
|     user_id: String, | ||||
| } | ||||
| 
 | ||||
| impl WizardSubcommands { | ||||
|     // dispatch
 | ||||
|     fn handle(&self) -> Result<()> { | ||||
|         match self { | ||||
|             WizardSubcommands::GenerateShardSecret(gss) => gss.handle(), | ||||
|             WizardSubcommands::BottomsUp(bu) => bu.handle(), | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| fn cross_sign_certs(certs: &mut [Cert]) -> Result<(), Box<dyn std::error::Error>> { | ||||
|     let policy = StandardPolicy::new(); | ||||
| 
 | ||||
|     #[allow(clippy::unnecessary_to_owned)] | ||||
|     for signing_cert in certs.to_vec() { | ||||
|         let mut certify_key = signing_cert | ||||
|             .with_policy(&policy, None)? | ||||
|             .keys() | ||||
|             .unencrypted_secret() | ||||
|             .for_certification() | ||||
|             .next() | ||||
|             .expect("certify key unusable/not found") | ||||
|             .key() | ||||
|             .clone() | ||||
|             .into_keypair()?; | ||||
|         for signable_cert in certs.iter_mut() { | ||||
|             let sb = SignatureBuilder::new(SignatureType::GenericCertification); | ||||
|             let userid = signable_cert | ||||
|                 .userids() | ||||
|                 .next() | ||||
|                 .expect("a signable user ID is necessary to create web of trust"); | ||||
|             let signature = sb.sign_userid_binding( | ||||
|                 &mut certify_key, | ||||
|                 signable_cert.primary_key().key(), | ||||
|                 &userid, | ||||
|             )?; | ||||
|             let changed; | ||||
|             (*signable_cert, changed) = signable_cert.clone().insert_packets2(signature)?; | ||||
|             assert!( | ||||
|                 changed, | ||||
|                 "OpenPGP certificate was unchanged after inserting packets" | ||||
|             ); | ||||
|         } | ||||
|     } | ||||
|     Ok(()) | ||||
| } | ||||
| 
 | ||||
| impl GenerateShardSecret { | ||||
|     fn handle(&self) -> Result<()> { | ||||
|         let seed = keyfork_entropy::generate_entropy_of_const_size::<{ 256 / 8 }>()?; | ||||
|         let mut pm = default_handler()?; | ||||
|         let mut certs = vec![]; | ||||
|         let mut seen_cards: HashSet<String> = HashSet::new(); | ||||
|         let stdout = std::io::stdout(); | ||||
|         if self.output.is_none() { | ||||
|             assert!( | ||||
|                 !stdout.is_terminal(), | ||||
|                 "not printing shard to terminal, redirect output" | ||||
|             ); | ||||
|         } | ||||
| 
 | ||||
|         let user_pin_validator = SecurePinValidator { | ||||
|             min_length: Some(6), | ||||
|             ..Default::default() | ||||
|         } | ||||
|         .to_fn(); | ||||
|         let admin_pin_validator = SecurePinValidator { | ||||
|             min_length: Some(8), | ||||
|             ..Default::default() | ||||
|         } | ||||
|         .to_fn(); | ||||
| 
 | ||||
|         for index in 0..self.max { | ||||
|             let cert = derive_key(seed, index)?; | ||||
|             for i in 0..self.keys_per_shard { | ||||
|                 pm.prompt_message(Message::Text(format!( | ||||
|                     "Please remove all keys and insert key #{} for user #{}", | ||||
|                     (i as u16) + 1, | ||||
|                     (index as u16) + 1, | ||||
|                 )))?; | ||||
|                 let card_backend = loop { | ||||
|                     if let Some(c) = PcscBackend::cards(None)?.next().transpose()? { | ||||
|                         break c; | ||||
|                     } | ||||
|                     pm.prompt_message(Message::Text( | ||||
|                         "No smart card was found. Please plug in a smart card and press enter" | ||||
|                             .to_string(), | ||||
|                     ))?; | ||||
|                 }; | ||||
|                 let user_pin = prompt_validated_passphrase( | ||||
|                     &mut *pm, | ||||
|                     "Please enter the new smartcard User PIN: ", | ||||
|                     3, | ||||
|                     &user_pin_validator, | ||||
|                 )?; | ||||
|                 let admin_pin = prompt_validated_passphrase( | ||||
|                     &mut *pm, | ||||
|                     "Please enter the new smartcard Admin PIN: ", | ||||
|                     3, | ||||
|                     &admin_pin_validator, | ||||
|                 )?; | ||||
|                 factory_reset_current_card( | ||||
|                     &mut |application_identifier| { | ||||
|                         if seen_cards.contains(&application_identifier) { | ||||
|                             // we were given the same card, error
 | ||||
|                             // we're gonna panic because this is a significant error
 | ||||
|                             panic!("Previously used card {application_identifier} was reused"); | ||||
|                         } else { | ||||
|                             seen_cards.insert(application_identifier); | ||||
|                             true | ||||
|                         } | ||||
|                     }, | ||||
|                     user_pin.trim(), | ||||
|                     admin_pin.trim(), | ||||
|                     &cert, | ||||
|                     &openpgp::policy::NullPolicy::new(), | ||||
|                     card_backend, | ||||
|                 )?; | ||||
|             } | ||||
|             certs.push(cert); | ||||
|         } | ||||
| 
 | ||||
|         cross_sign_certs(&mut certs)?; | ||||
| 
 | ||||
|         let opgp = OpenPGP; | ||||
| 
 | ||||
|         if let Some(output_file) = self.output.as_ref() { | ||||
|             let output = File::create(output_file)?; | ||||
|             opgp.shard_and_encrypt(self.threshold, certs.len() as u8, &seed, &certs[..], output)?; | ||||
|         } else { | ||||
|             opgp.shard_and_encrypt( | ||||
|                 self.threshold, | ||||
|                 certs.len() as u8, | ||||
|                 &seed, | ||||
|                 &certs[..], | ||||
|                 std::io::stdout(), | ||||
|             )?; | ||||
|         } | ||||
| 
 | ||||
|         if let Some(cert_output_file) = self.cert_output.as_ref() { | ||||
|             let output = File::create(cert_output_file)?; | ||||
|             let mut writer = Writer::new(output, Kind::PublicKey)?; | ||||
|             for cert in certs { | ||||
|                 cert.serialize(&mut writer)?; | ||||
|             } | ||||
|             writer.finalize()?; | ||||
|         } | ||||
| 
 | ||||
|         Ok(()) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl BottomsUp { | ||||
|     fn handle(&self) -> Result<()> { | ||||
|         let entropy = keyfork_entropy::generate_entropy_of_const_size::<{ 256 / 8 }>()?; | ||||
|         let mnemonic = Mnemonic::from_array(entropy); | ||||
|         let seed = mnemonic.generate_seed(None); | ||||
| 
 | ||||
|         // TODO: should this allow for customizing the account index from 0? Potential for key reuse
 | ||||
|         // errors.
 | ||||
|         let path = paths::OPENPGP_DISASTER_RECOVERY | ||||
|             .clone() | ||||
|             .chain_push(DerivationIndex::new(0, true)?); | ||||
|         let subkeys = [ | ||||
|             KeyFlags::empty().set_certification(), | ||||
|             KeyFlags::empty().set_signing(), | ||||
|             KeyFlags::empty() | ||||
|                 .set_transport_encryption() | ||||
|                 .set_storage_encryption(), | ||||
|             KeyFlags::empty().set_authentication(), | ||||
|         ]; | ||||
|         let xprv = XPrv::new(seed) | ||||
|             .expect("could not construct master key from seed") | ||||
|             .derive_path(&path)?; | ||||
|         let userid = UserID::from(self.user_id.as_str()); | ||||
| 
 | ||||
|         let cert = keyfork_derive_openpgp::derive(xprv, &subkeys, &userid)?; | ||||
|         let certfile = File::create(&self.output_cert)?; | ||||
|         let mut w = Writer::new(certfile, Kind::PublicKey)?; | ||||
|         cert.serialize(&mut w)?; | ||||
|         w.finalize()?; | ||||
| 
 | ||||
|         let opgp = OpenPGP; | ||||
|         let certs = OpenPGP::discover_certs(&self.key_discovery)?; | ||||
| 
 | ||||
|         let shardfile = File::create(&self.output_shardfile)?; | ||||
|         opgp.shard_and_encrypt( | ||||
|             self.threshold, | ||||
|             certs.len() as u8, | ||||
|             &entropy, | ||||
|             &certs[..], | ||||
|             shardfile, | ||||
|         )?; | ||||
|         Ok(()) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| #[derive(Parser, Debug, Clone)] | ||||
| pub struct Wizard { | ||||
|     #[command(subcommand)] | ||||
|     command: WizardSubcommands, | ||||
| } | ||||
| 
 | ||||
| impl Wizard { | ||||
|     pub fn handle(&self, _k: &Keyfork) -> Result<()> { | ||||
|         self.command.handle()?; | ||||
|         Ok(()) | ||||
|     } | ||||
| } | ||||
|  | @ -2,19 +2,19 @@ use std::collections::HashMap; | |||
| 
 | ||||
| use serde::{Deserialize, Serialize}; | ||||
| 
 | ||||
| #[derive(Serialize, Deserialize, Clone, Debug)] | ||||
| #[derive(Serialize, Deserialize, Clone)] | ||||
| pub struct Mnemonic { | ||||
|     pub hash: String, | ||||
| } | ||||
| 
 | ||||
| #[derive(Serialize, Deserialize, Clone, Debug)] | ||||
| #[derive(Serialize, Deserialize, Clone)] | ||||
| pub struct Provisioner { | ||||
|     pub account: u32, | ||||
|     pub identifier: String, | ||||
|     pub metadata: Option<HashMap<String, String>>, | ||||
| } | ||||
| 
 | ||||
| #[derive(Serialize, Deserialize, Clone, Debug)] | ||||
| #[derive(Serialize, Deserialize, Clone)] | ||||
| pub struct Config { | ||||
|     pub mnemonic: Mnemonic, | ||||
|     pub provisioner: Vec<Provisioner>, | ||||
|  |  | |||
|  | @ -1,68 +1,6 @@ | |||
| use card_backend_pcsc::PcscBackend; | ||||
| use keyfork_derive_openpgp::openpgp::{policy::Policy, Cert}; | ||||
| use keyfork_prompt::{ | ||||
|     prompt_validated_passphrase, | ||||
|     validators::{SecurePinValidator, Validator}, | ||||
|     Message, PromptHandler, | ||||
| }; | ||||
| use openpgp_card_sequoia::{state::Open, types::KeyType, types::TouchPolicy, Card}; | ||||
| 
 | ||||
| pub fn get_new_pins( | ||||
|     pm: &mut dyn PromptHandler, | ||||
| ) -> Result<(String, String), Box<dyn std::error::Error>> { | ||||
|     let user_pin_validator = SecurePinValidator { | ||||
|         min_length: Some(6), | ||||
|         ..Default::default() | ||||
|     } | ||||
|     .to_fn(); | ||||
|     let admin_pin_validator = SecurePinValidator { | ||||
|         min_length: Some(8), | ||||
|         ..Default::default() | ||||
|     } | ||||
|     .to_fn(); | ||||
| 
 | ||||
|     let user_pin = loop { | ||||
|         let user_pin = prompt_validated_passphrase( | ||||
|             &mut *pm, | ||||
|             "Please enter the new smartcard User PIN: ", | ||||
|             3, | ||||
|             &user_pin_validator, | ||||
|         )?; | ||||
|         let validated_user_pin = prompt_validated_passphrase( | ||||
|             &mut *pm, | ||||
|             "Please verify the new smartcard User PIN: ", | ||||
|             3, | ||||
|             &user_pin_validator, | ||||
|         )?; | ||||
|         if user_pin != validated_user_pin { | ||||
|             pm.prompt_message(Message::Text("User PINs did not match. Retrying.".into()))?; | ||||
|         } else { | ||||
|             break user_pin; | ||||
|         } | ||||
|     }; | ||||
| 
 | ||||
|     let admin_pin = loop { | ||||
|         let admin_pin = prompt_validated_passphrase( | ||||
|             &mut *pm, | ||||
|             "Please enter the new smartcard Admin PIN: ", | ||||
|             3, | ||||
|             &admin_pin_validator, | ||||
|         )?; | ||||
|         let validated_admin_pin = prompt_validated_passphrase( | ||||
|             &mut *pm, | ||||
|             "Please verify the new smartcard Admin PIN: ", | ||||
|             3, | ||||
|             &admin_pin_validator, | ||||
|         )?; | ||||
|         if admin_pin != validated_admin_pin { | ||||
|             pm.prompt_message(Message::Text("Admin PINs did not match. Retrying.".into()))?; | ||||
|         } else { | ||||
|             break admin_pin; | ||||
|         } | ||||
|     }; | ||||
| 
 | ||||
|     Ok((user_pin, admin_pin)) | ||||
| } | ||||
| use openpgp_card_sequoia::{state::Open, types::KeyType, Card, types::TouchPolicy}; | ||||
| use keyfork_derive_openpgp::openpgp::{Cert, policy::Policy}; | ||||
| 
 | ||||
| /// Factory reset the current card so long as it does not match the last-used backend.
 | ||||
| ///
 | ||||
|  | @ -113,3 +51,4 @@ pub fn factory_reset_current_card( | |||
|     transaction.change_admin_pin("12345678", admin_pin)?; | ||||
|     Ok(true) | ||||
| } | ||||
| 
 | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue