Compare commits
	
		
			3 Commits
		
	
	
		
			88a05f23ac
			...
			53665cac2e
		
	
	| Author | SHA1 | Date | 
|---|---|---|
|  | 53665cac2e | |
|  | a1c3d52c14 | |
|  | 674e2e93c5 | 
|  | @ -1381,10 +1381,22 @@ dependencies = [ | ||||||
|  "cfg-if", |  "cfg-if", | ||||||
|  "js-sys", |  "js-sys", | ||||||
|  "libc", |  "libc", | ||||||
|  "wasi", |  "wasi 0.11.0+wasi-snapshot-preview1", | ||||||
|  "wasm-bindgen", |  "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]] | [[package]] | ||||||
| name = "ghash" | name = "ghash" | ||||||
| version = "0.5.1" | version = "0.5.1" | ||||||
|  | @ -1808,7 +1820,9 @@ dependencies = [ | ||||||
|  "openpgp-card-sequoia", |  "openpgp-card-sequoia", | ||||||
|  "sequoia-openpgp", |  "sequoia-openpgp", | ||||||
|  "serde", |  "serde", | ||||||
|  |  "shlex", | ||||||
|  "smex", |  "smex", | ||||||
|  |  "tempfile", | ||||||
|  "thiserror", |  "thiserror", | ||||||
|  "tokio", |  "tokio", | ||||||
| ] | ] | ||||||
|  | @ -2244,7 +2258,7 @@ dependencies = [ | ||||||
|  "hermit-abi 0.3.9", |  "hermit-abi 0.3.9", | ||||||
|  "libc", |  "libc", | ||||||
|  "log", |  "log", | ||||||
|  "wasi", |  "wasi 0.11.0+wasi-snapshot-preview1", | ||||||
|  "windows-sys 0.52.0", |  "windows-sys 0.52.0", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
|  | @ -2254,7 +2268,7 @@ version = "7.4.0" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "44e6ff4a94e5d34a1fd5abbd39418074646e2fa51b257198701330f22fcd6936" | checksum = "44e6ff4a94e5d34a1fd5abbd39418074646e2fa51b257198701330f22fcd6936" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "getrandom", |  "getrandom 0.2.15", | ||||||
|  "libc", |  "libc", | ||||||
|  "nettle-sys", |  "nettle-sys", | ||||||
|  "thiserror", |  "thiserror", | ||||||
|  | @ -2820,7 +2834,7 @@ version = "0.6.4" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c" | checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "getrandom", |  "getrandom 0.2.15", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
| [[package]] | [[package]] | ||||||
|  | @ -2838,7 +2852,7 @@ version = "0.4.6" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "ba009ff324d1fc1b900bd1fdb31564febe58a8ccc8a6fdbb93b543d33b13ca43" | checksum = "ba009ff324d1fc1b900bd1fdb31564febe58a8ccc8a6fdbb93b543d33b13ca43" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "getrandom", |  "getrandom 0.2.15", | ||||||
|  "libredox", |  "libredox", | ||||||
|  "thiserror", |  "thiserror", | ||||||
| ] | ] | ||||||
|  | @ -3067,7 +3081,7 @@ dependencies = [ | ||||||
|  "ed25519", |  "ed25519", | ||||||
|  "ed25519-dalek", |  "ed25519-dalek", | ||||||
|  "flate2", |  "flate2", | ||||||
|  "getrandom", |  "getrandom 0.2.15", | ||||||
|  "idea", |  "idea", | ||||||
|  "idna", |  "idna", | ||||||
|  "lalrpop", |  "lalrpop", | ||||||
|  | @ -3355,12 +3369,13 @@ dependencies = [ | ||||||
| 
 | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "tempfile" | name = "tempfile" | ||||||
| version = "3.14.0" | version = "3.17.1" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "28cce251fcbc87fac86a866eeb0d6c2d536fc16d06f184bb61aeae11aa4cee0c" | checksum = "22e5a0acb1f3f55f65cc4a866c361b2fb2a0ff6366785ae6fbb5f85df07ba230" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "cfg-if", |  "cfg-if", | ||||||
|  "fastrand", |  "fastrand", | ||||||
|  |  "getrandom 0.3.1", | ||||||
|  "once_cell", |  "once_cell", | ||||||
|  "rustix", |  "rustix", | ||||||
|  "windows-sys 0.59.0", |  "windows-sys 0.59.0", | ||||||
|  | @ -3726,6 +3741,15 @@ version = "0.11.0+wasi-snapshot-preview1" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" | 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]] | [[package]] | ||||||
| name = "wasm-bindgen" | name = "wasm-bindgen" | ||||||
| version = "0.2.95" | version = "0.2.95" | ||||||
|  | @ -3937,6 +3961,15 @@ version = "0.52.6" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "589f6da84c646204747d1270a2a5661ea66ed1cced2631d546fdfb155959f9ec" | 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]] | [[package]] | ||||||
| name = "write16" | name = "write16" | ||||||
| version = "1.0.0" | version = "1.0.0" | ||||||
|  |  | ||||||
|  | @ -76,6 +76,7 @@ thiserror = "1.0.56" | ||||||
| tokio = "1.35.1" | tokio = "1.35.1" | ||||||
| v4l = "0.14.0" | v4l = "0.14.0" | ||||||
| base64 = "0.22.1" | base64 = "0.22.1" | ||||||
|  | tempfile = "3.17.1" | ||||||
| 
 | 
 | ||||||
| [profile.release] | [profile.release] | ||||||
| debug = true | debug = true | ||||||
|  |  | ||||||
|  | @ -32,7 +32,7 @@ tower = { version = "0.4.13", features = ["tokio", "util"] } | ||||||
| # Personally audited | # Personally audited | ||||||
| thiserror = { workspace = true } | thiserror = { workspace = true } | ||||||
| serde = { workspace = true } | serde = { workspace = true } | ||||||
| tempfile = { version = "3.10.0", default-features = false } | tempfile = { workspace = true } | ||||||
| 
 | 
 | ||||||
| [dev-dependencies] | [dev-dependencies] | ||||||
| hex-literal = { workspace = true } | hex-literal = { workspace = true } | ||||||
|  |  | ||||||
|  | @ -48,3 +48,5 @@ sequoia-openpgp = { workspace = true } | ||||||
| keyforkd-models.workspace = true | keyforkd-models.workspace = true | ||||||
| base64.workspace = true | base64.workspace = true | ||||||
| nix = { version = "0.29.0", default-features = false, features = ["process"] } | nix = { version = "0.29.0", default-features = false, features = ["process"] } | ||||||
|  | shlex = "1.3.0" | ||||||
|  | tempfile.workspace = true | ||||||
|  |  | ||||||
|  | @ -2,20 +2,6 @@ | ||||||
| 
 | 
 | ||||||
| use std::{collections::HashMap, str::FromStr}; | 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
 | /// An error that occurred while parsing a base value or its
 | ||||||
| #[derive(Debug, thiserror::Error)] | #[derive(Debug, thiserror::Error)] | ||||||
| pub enum ValueParseError { | pub enum ValueParseError { | ||||||
|  | @ -32,6 +18,62 @@ pub enum ValueParseError { | ||||||
|     BadKeyValue, |     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> | impl<T: std::str::FromStr> FromStr for ValueWithOptions<T> | ||||||
| where | where | ||||||
|     <T as FromStr>::Err: std::error::Error, |     <T as FromStr>::Err: std::error::Error, | ||||||
|  |  | ||||||
|  | @ -1,25 +1,36 @@ | ||||||
| use super::Keyfork; | use super::{Keyfork, create}; | ||||||
| use clap::{Args, Parser, Subcommand, ValueEnum}; | use clap::{Args, Parser, Subcommand, ValueEnum}; | ||||||
|  | use std::{fmt::Display, io::Write, path::PathBuf}; | ||||||
| 
 | 
 | ||||||
| use keyfork_derive_openpgp::{ | use keyfork_derive_openpgp::openpgp::{ | ||||||
|     openpgp::{ |  | ||||||
|     armor::{Kind, Writer}, |     armor::{Kind, Writer}, | ||||||
|     packet::UserID, |     packet::UserID, | ||||||
|     serialize::Marshal, |     serialize::Marshal, | ||||||
|     types::KeyFlags, |     types::KeyFlags, | ||||||
|     }, |     Cert, | ||||||
|     XPrvKey, |  | ||||||
| }; | }; | ||||||
| use keyfork_derive_path_data::paths; | use keyfork_derive_path_data::paths; | ||||||
| use keyfork_derive_util::{ | use keyfork_derive_util::{ | ||||||
|     request::{DerivationAlgorithm, DerivationRequest, DerivationResponse}, |     request::DerivationAlgorithm, DerivationIndex, DerivationPath, ExtendedPrivateKey as XPrv, | ||||||
|     DerivationIndex, DerivationPath, IndexError, |     IndexError, PrivateKey, | ||||||
| }; | }; | ||||||
| use keyforkd_client::Client; | use keyforkd_client::Client; | ||||||
| use keyforkd_models::Request; | 
 | ||||||
|  | type OptWrite = Option<Box<dyn Write>>; | ||||||
| 
 | 
 | ||||||
| type Result<T, E = Box<dyn std::error::Error>> = std::result::Result<T, E>; | 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)] | #[derive(Subcommand, Clone, Debug)] | ||||||
| pub enum DeriveSubcommands { | pub enum DeriveSubcommands { | ||||||
|     /// Derive an OpenPGP Transferable Secret Key (private key). The key is encoded using OpenPGP
 |     /// Derive an OpenPGP Transferable Secret Key (private key). The key is encoded using OpenPGP
 | ||||||
|  | @ -34,14 +45,51 @@ pub enum DeriveSubcommands { | ||||||
|     #[command(name = "openpgp")] |     #[command(name = "openpgp")] | ||||||
|     OpenPGP(OpenPGP), |     OpenPGP(OpenPGP), | ||||||
| 
 | 
 | ||||||
|     /// Derive a bare key for a specific algorithm, in a given format.
 |     /// Derive an Ed25519 key for a specific algorithm, in a given format.
 | ||||||
|     Key(Key), |     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)] | #[derive(Args, Clone, Debug)] | ||||||
| pub struct OpenPGP { | pub struct OpenPGP { | ||||||
|     /// Default User ID for the certificate, using the OpenPGP User ID format.
 |     /// Default User ID for the certificate, using the OpenPGP User ID format.
 | ||||||
|     user_id: String, |     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.
 | /// A format for exporting a key.
 | ||||||
|  | @ -83,6 +131,18 @@ 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)] | #[derive(Args, Clone, Debug)] | ||||||
| pub struct Key { | pub struct Key { | ||||||
|     /// The derivation algorithm to derive a key for.
 |     /// The derivation algorithm to derive a key for.
 | ||||||
|  | @ -98,18 +158,34 @@ pub struct Key { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl DeriveSubcommands { | impl DeriveSubcommands { | ||||||
|     fn handle(&self, account: DerivationIndex) -> Result<()> { |     fn handle(&self, account: DerivationIndex, is_public: bool, writer: OptWrite) -> Result<()> { | ||||||
|         match self { |         match self { | ||||||
|             DeriveSubcommands::OpenPGP(opgp) => opgp.handle(account), |             DeriveSubcommands::OpenPGP(opgp) => { | ||||||
|             DeriveSubcommands::Key(key) => key.handle(account), |                 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) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl OpenPGP { | impl OpenPGP { | ||||||
|     pub fn handle(&self, account: DerivationIndex) -> Result<()> { |     fn cert_from_xprv(&self, xprv: keyfork_derive_openpgp::XPrv) -> Result<Cert> { | ||||||
|         let path = paths::OPENPGP.clone().chain_push(account); |  | ||||||
|         // TODO: should this be customizable?
 |  | ||||||
|         let subkeys = vec![ |         let subkeys = vec![ | ||||||
|             KeyFlags::empty().set_certification(), |             KeyFlags::empty().set_certification(), | ||||||
|             KeyFlags::empty().set_signing(), |             KeyFlags::empty().set_signing(), | ||||||
|  | @ -118,40 +194,100 @@ impl OpenPGP { | ||||||
|                 .set_storage_encryption(), |                 .set_storage_encryption(), | ||||||
|             KeyFlags::empty().set_authentication(), |             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 mut w = Writer::new(std::io::stdout(), Kind::SecretKey)?; |         let userid = UserID::from(&*self.user_id); | ||||||
| 
 |         keyfork_derive_openpgp::derive(xprv, &subkeys, &userid).map_err(Into::into) | ||||||
|         for packet in cert.as_tsk().into_packets() { |     } | ||||||
|             packet.serialize(&mut w)?; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|         w.finalize()?; | 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(()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     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()?; | ||||||
|         Ok(()) |         Ok(()) | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl Key { | impl Deriver for Key { | ||||||
|     pub fn handle(&self, account: DerivationIndex) -> Result<()> { |     // HACK: We're abusing that we use the same key as OpenPGP. Maybe we should use ed25519_dalek.
 | ||||||
|         let mut client = keyforkd_client::Client::discover_socket()?; |     type Prv = keyfork_derive_openpgp::XPrvKey; | ||||||
|         let path = DerivationPath::default() |     const DERIVATION_ALGORITHM: DerivationAlgorithm = DerivationAlgorithm::Ed25519; | ||||||
|             .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()?; |  | ||||||
| 
 | 
 | ||||||
|         let formatted = match self.format { |     fn derivation_path(&self) -> DerivationPath { | ||||||
|             KeyFormat::Hex => smex::encode(derived_key.data), |         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"), | ||||||
|             KeyFormat::Base64 => { |             KeyFormat::Base64 => { | ||||||
|                 use base64::prelude::*; |                 use base64::prelude::*; | ||||||
|                 BASE64_STANDARD.encode(derived_key.data) |                 (BASE64_STANDARD.encode(xprv.private_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(()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     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)?; | ||||||
|  |         } | ||||||
|         Ok(()) |         Ok(()) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | @ -159,7 +295,7 @@ impl Key { | ||||||
| #[derive(Parser, Debug, Clone)] | #[derive(Parser, Debug, Clone)] | ||||||
| pub struct Derive { | pub struct Derive { | ||||||
|     #[command(subcommand)] |     #[command(subcommand)] | ||||||
|     command: DeriveSubcommands, |     pub(crate) command: DeriveSubcommands, | ||||||
| 
 | 
 | ||||||
|     /// Account ID. Required for all derivations.
 |     /// Account ID. Required for all derivations.
 | ||||||
|     ///
 |     ///
 | ||||||
|  | @ -167,12 +303,45 @@ pub struct Derive { | ||||||
|     /// account ID can often come as a hindrance in the future. As such, it is always required. If
 |     /// 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`.
 |     /// the account ID is not relevant, it is assumed to be `0`.
 | ||||||
|     #[arg(long, global = true, default_value = "0")] |     #[arg(long, global = true, default_value = "0")] | ||||||
|     account_id: u32, |     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>, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl Derive { | impl Derive { | ||||||
|     pub fn handle(&self, _k: &Keyfork) -> Result<()> { |     pub fn handle(&self, _k: &Keyfork) -> Result<()> { | ||||||
|         let account = DerivationIndex::new(self.account_id, true)?; |         let account = DerivationIndex::new(self.account_id, true)?; | ||||||
|         self.command.handle(account) |         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)), | ||||||
|  |         ) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,12 +1,17 @@ | ||||||
| use super::provision; | use super::{ | ||||||
| use super::Keyfork; |     create, | ||||||
| use crate::{clap_ext::*, config}; |     derive::{self, Deriver}, | ||||||
|  |     provision, | ||||||
|  |     Keyfork, | ||||||
|  | }; | ||||||
|  | use crate::{clap_ext::*, config, openpgp_card::factory_reset_current_card}; | ||||||
|  | use card_backend_pcsc::PcscBackend; | ||||||
| use clap::{builder::PossibleValue, Parser, Subcommand, ValueEnum}; | use clap::{builder::PossibleValue, Parser, Subcommand, ValueEnum}; | ||||||
| use std::{ | use std::{ | ||||||
|     collections::HashMap, |     collections::HashMap, | ||||||
|     fmt::Display, |     fmt::Display, | ||||||
|     fs::File, |     fs::File, | ||||||
|     io::Write, |     io::{IsTerminal, Write}, | ||||||
|     path::{Path, PathBuf}, |     path::{Path, PathBuf}, | ||||||
|     str::FromStr, |     str::FromStr, | ||||||
| }; | }; | ||||||
|  | @ -15,17 +20,21 @@ use keyfork_derive_openpgp::{ | ||||||
|     openpgp::{ |     openpgp::{ | ||||||
|         self, |         self, | ||||||
|         armor::{Kind, Writer}, |         armor::{Kind, Writer}, | ||||||
|         packet::UserID, |         packet::{UserID, signature::SignatureBuilder}, | ||||||
|         policy::StandardPolicy, |         policy::StandardPolicy, | ||||||
|         serialize::{ |         serialize::{ | ||||||
|             stream::{Encryptor2, LiteralWriter, Message, Recipient}, |             stream::{Encryptor2, LiteralWriter, Message, Recipient}, | ||||||
|             Serialize, |             Serialize, | ||||||
|         }, |         }, | ||||||
|         types::KeyFlags, |         types::{KeyFlags, SignatureType}, | ||||||
|     }, |     }, | ||||||
|     XPrv, |     XPrv, | ||||||
| }; | }; | ||||||
| use keyfork_prompt::default_handler; | use keyfork_derive_util::DerivationIndex; | ||||||
|  | use keyfork_prompt::{ | ||||||
|  |     default_handler, prompt_validated_passphrase, | ||||||
|  |     validators::{SecurePinValidator, Validator}, | ||||||
|  | }; | ||||||
| use keyfork_shard::{openpgp::OpenPGP, Format}; | use keyfork_shard::{openpgp::OpenPGP, Format}; | ||||||
| 
 | 
 | ||||||
| type StringMap = HashMap<String, String>; | type StringMap = HashMap<String, String>; | ||||||
|  | @ -150,6 +159,10 @@ pub enum Error { | ||||||
|     /// An error occurred when interacting iwth a file.
 |     /// An error occurred when interacting iwth a file.
 | ||||||
|     #[error("Error while performing IO operation on: {1}")] |     #[error("Error while performing IO operation on: {1}")] | ||||||
|     IOContext(#[source] std::io::Error, PathBuf), |     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 { | fn context_stub<'a>(path: &'a Path) -> impl Fn(std::io::Error) -> Error + 'a { | ||||||
|  | @ -179,6 +192,19 @@ pub enum MnemonicSubcommands { | ||||||
|         #[arg(long, default_value_t = Default::default())] |         #[arg(long, default_value_t = Default::default())] | ||||||
|         size: SeedSize, |         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.
 |         /// 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
 |         /// When given arguments in the format `--encrypt-to input.asc,output=output.asc`, the
 | ||||||
|  | @ -191,7 +217,7 @@ pub enum MnemonicSubcommands { | ||||||
|         /// Shard the mnemonic to the certificates in the given Shardfile. Requires a decrypt
 |         /// Shard the mnemonic to the certificates in the given Shardfile. Requires a decrypt
 | ||||||
|         /// operation on the Shardfile to access the metadata and certificates.
 |         /// operation on the Shardfile to access the metadata and certificates.
 | ||||||
|         ///
 |         ///
 | ||||||
|         /// When given arguments in the format `--encrypt-to input.asc,output=output.asc`, the
 |         /// When given arguments in the format `--shard-to input.asc,output=output.asc`, the
 | ||||||
|         /// output of the encryption will be written to `output.asc`. Otherwise, the default
 |         /// 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,
 |         /// 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.
 |         /// it will not be overwritten, and the command will exit unsuccessfully.
 | ||||||
|  | @ -216,48 +242,56 @@ pub enum MnemonicSubcommands { | ||||||
| 
 | 
 | ||||||
|         /// Encrypt the mnemonic to an OpenPGP certificate derived from the mnemonic, writing the
 |         /// 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
 |         /// output to the provided path. This command must be run in combination with
 | ||||||
|         /// `--provision openpgp-card` or another relevant provisioner, to ensure the newly
 |         /// `--provision openpgp-card`, `--derive openpgp`, or another OpenPGP key derivation
 | ||||||
|         /// generated mnemonic would be decryptable by some form of provisioned hardware.
 |         /// mechanism, to ensure the generated mnemonic would be decryptable.
 | ||||||
|         ///
 |         ///
 | ||||||
|         /// When given arguments in the format `--encrypt-to-self encrypted.asc,output=cert.asc`,
 |         /// When used in combination with `--derive` or `--provision` with OpenPGP configurations,
 | ||||||
|         /// the output of the OpenPGP certificate will be written to `cert.asc`, while the output
 |         /// the default behavior is to encrypt the mnemonic to all derived and provisioned
 | ||||||
|         /// of the encryption will be written to `encrypted.asc`. Otherwise, the
 |         /// accounts. By default, the account `0` is used.
 | ||||||
|         /// 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)] |         #[arg(long)] | ||||||
|         encrypt_to_self: Option<ValueWithOptions<PathBuf>>, |         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>>, | ||||||
| 
 | 
 | ||||||
|         /// Provision a key derived from the mnemonic to a piece of hardware such as an OpenPGP
 |         /// 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`.
 |         /// smartcard. This argument is required when used with `--encrypt-to-self`.
 | ||||||
|         ///
 |         ///
 | ||||||
|         /// Additional arguments, such as the amount of hardware to provision and the
 |         /// Provisioners may choose to output a public key to the current directory by default, but
 | ||||||
|         /// account to use when deriving, can be specified by using (for example)
 |         /// this functionality may be altered on a by-provisioner basis by providing the `output=`
 | ||||||
|         /// `--provision openpgp-card,count=2,account=1`.
 |         /// option to `--provisioner-config`. Additionally, Keyfork may choose to disable
 | ||||||
|         ///
 |         /// provisioner output if a matching public key has been derived using `--derive`, which
 | ||||||
|         /// Provisioners may output their public key, if necessary. The file path may be chosen
 |         /// may allow for controlling additional metadata that is not relevant to the provisioned
 | ||||||
|         /// based on the provided `output` field, or automatically determined based on the content
 |         /// keys, such as an OpenPGP User ID.
 | ||||||
|         /// 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)] |         #[arg(long)] | ||||||
|         provision: Option<ValueWithOptions<provision::Provisioner>>, |         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, | ||||||
|     }, |     }, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -324,7 +358,7 @@ fn do_encrypt_to( | ||||||
|     literal_message.write_all(b"\n")?; |     literal_message.write_all(b"\n")?; | ||||||
|     literal_message.finalize()?; |     literal_message.finalize()?; | ||||||
| 
 | 
 | ||||||
|     let mut file = File::create_new(&output_file).map_err(context_stub(&output_file))?; |     let mut file = File::create(&output_file).map_err(context_stub(&output_file))?; | ||||||
|     if is_armored { |     if is_armored { | ||||||
|         let mut writer = Writer::new(file, Kind::Message)?; |         let mut writer = Writer::new(file, Kind::Message)?; | ||||||
|         writer.write_all(&output)?; |         writer.write_all(&output)?; | ||||||
|  | @ -339,18 +373,12 @@ fn do_encrypt_to( | ||||||
| fn do_encrypt_to_self( | fn do_encrypt_to_self( | ||||||
|     mnemonic: &keyfork_mnemonic::Mnemonic, |     mnemonic: &keyfork_mnemonic::Mnemonic, | ||||||
|     path: &Path, |     path: &Path, | ||||||
|     options: &StringMap, |     accounts: &[keyfork_derive_util::DerivationIndex], | ||||||
| ) -> Result<(), Box<dyn std::error::Error>> { | ) -> Result<(), Box<dyn std::error::Error>> { | ||||||
|     let account = options |     let mut certs = vec![]; | ||||||
|         .get("account") |  | ||||||
|         .map(|account| u32::from_str(account)) |  | ||||||
|         .transpose()? |  | ||||||
|         .unwrap_or(0); |  | ||||||
|     let account_index = keyfork_derive_util::DerivationIndex::new(account, true)?; |  | ||||||
| 
 | 
 | ||||||
|     let userid = options |     for account in accounts.iter().cloned() { | ||||||
|         .get("userid") |         let userid = UserID::from("Keyfork Temporary Key"); | ||||||
|         .map(|userid| UserID::from(userid.as_str())); |  | ||||||
| 
 | 
 | ||||||
|         let subkeys = [ |         let subkeys = [ | ||||||
|             KeyFlags::empty().set_certification(), |             KeyFlags::empty().set_certification(), | ||||||
|  | @ -365,38 +393,33 @@ fn do_encrypt_to_self( | ||||||
|         let xprv = XPrv::new(seed)?; |         let xprv = XPrv::new(seed)?; | ||||||
|         let derivation_path = keyfork_derive_path_data::paths::OPENPGP |         let derivation_path = keyfork_derive_path_data::paths::OPENPGP | ||||||
|             .clone() |             .clone() | ||||||
|         .chain_push(account_index); |             .chain_push(account); | ||||||
| 
 | 
 | ||||||
|     let cert = keyfork_derive_openpgp::derive( |         let cert = | ||||||
|         xprv.derive_path(&derivation_path)?, |             keyfork_derive_openpgp::derive(xprv.derive_path(&derivation_path)?, &subkeys, &userid)?; | ||||||
|         &subkeys, |  | ||||||
|         &userid.unwrap_or(UserID::from("Keyfork-Generated Key")), |  | ||||||
|     )?; |  | ||||||
| 
 | 
 | ||||||
|     let cert_path = match options.get("output") { |         certs.push(cert); | ||||||
|         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 file = File::create_new(&cert_path).map_err(context_stub(&cert_path))?; |     let mut file = tempfile::NamedTempFile::new()?; | ||||||
|     let mut writer = Writer::new(file, Kind::PublicKey)?; | 
 | ||||||
|  |     let mut writer = Writer::new(&mut file, Kind::PublicKey)?; | ||||||
|  |     for cert in certs { | ||||||
|         cert.serialize(&mut writer)?; |         cert.serialize(&mut writer)?; | ||||||
|  |     } | ||||||
|     writer.finalize()?; |     writer.finalize()?; | ||||||
| 
 | 
 | ||||||
|  |     let temp_path = file.into_temp_path(); | ||||||
|  | 
 | ||||||
|     // a sneaky bit of DRY
 |     // a sneaky bit of DRY
 | ||||||
|     do_encrypt_to( |     do_encrypt_to( | ||||||
|         mnemonic, |         mnemonic, | ||||||
|         &cert_path, |         &temp_path, | ||||||
|         &StringMap::from([(String::from("output"), path.to_string_lossy().to_string())]), |         &StringMap::from([(String::from("output"), path.to_string_lossy().to_string())]), | ||||||
|     )?; |     )?; | ||||||
| 
 | 
 | ||||||
|  |     temp_path.close()?; | ||||||
|  | 
 | ||||||
|     Ok(()) |     Ok(()) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -449,7 +472,7 @@ fn do_shard( | ||||||
|     let mut output = vec![]; |     let mut output = vec![]; | ||||||
|     openpgp.shard_and_encrypt(threshold, max, mnemonic.as_bytes(), &certs[..], &mut output)?; |     openpgp.shard_and_encrypt(threshold, max, mnemonic.as_bytes(), &certs[..], &mut output)?; | ||||||
| 
 | 
 | ||||||
|     let mut file = File::create_new(&output_file).map_err(context_stub(&output_file))?; |     let mut file = File::create(&output_file).map_err(context_stub(&output_file))?; | ||||||
|     if is_armored { |     if is_armored { | ||||||
|         file.write_all(&output)?; |         file.write_all(&output)?; | ||||||
|     } else { |     } else { | ||||||
|  | @ -494,7 +517,7 @@ fn do_shard_to( | ||||||
|         &mut output, |         &mut output, | ||||||
|     )?; |     )?; | ||||||
| 
 | 
 | ||||||
|     let mut file = File::create_new(&output_file).map_err(context_stub(&output_file))?; |     let mut file = File::create(&output_file).map_err(context_stub(&output_file))?; | ||||||
|     if is_armored { |     if is_armored { | ||||||
|         file.write_all(&output)?; |         file.write_all(&output)?; | ||||||
|     } else { |     } else { | ||||||
|  | @ -510,50 +533,271 @@ fn do_shard_to( | ||||||
|     Ok(()) |     Ok(()) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| fn do_provision( | fn derive_key(seed: [u8; 64], index: u8) -> Result<openpgp::Cert, Box<dyn std::error::Error>> { | ||||||
|     mnemonic: &keyfork_mnemonic::Mnemonic, |     let subkeys = vec![ | ||||||
|     provisioner: &provision::Provisioner, |         KeyFlags::empty().set_certification(), | ||||||
|     options: &StringMap, |         KeyFlags::empty().set_signing(), | ||||||
| ) -> Result<(), Box<dyn std::error::Error>> { |         KeyFlags::empty() | ||||||
|     let mut options = options.clone(); |             .set_transport_encryption() | ||||||
|     let account = options |             .set_storage_encryption(), | ||||||
|         .remove("account") |         KeyFlags::empty().set_authentication(), | ||||||
|         .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(|(identifier, _)| identifier) |  | ||||||
|                 .collect()) |  | ||||||
|         })?; |  | ||||||
|     let count = options |  | ||||||
|         .remove("count") |  | ||||||
|         .map(|count| usize::from_str(&count)) |  | ||||||
|         .transpose()? |  | ||||||
|         .unwrap_or(identifier.len()); |  | ||||||
| 
 | 
 | ||||||
|     assert_eq!( |     let subkey = DerivationIndex::new(u32::from(index), true)?; | ||||||
|         count, |     let path = keyfork_derive_path_data::paths::OPENPGP_SHARD.clone().chain_push(subkey); | ||||||
|         identifier.len(), |     let xprv = XPrv::new(seed) | ||||||
|         "amount of identifiers discovered or provided did not match provisioner count" |         .expect("could not construct master key from seed") | ||||||
|     ); |         .derive_path(&path)?; | ||||||
| 
 |     let userid = UserID::from(format!("Keyfork Shard {index}")); | ||||||
|     for (_, identifier) in (0..count).zip(identifier.into_iter()) { |     let cert = keyfork_derive_openpgp::derive(xprv, &subkeys, &userid)?; | ||||||
|         let provisioner_config = config::Provisioner { |     Ok(cert) | ||||||
|             account, |  | ||||||
|             identifier, |  | ||||||
|             metadata: Some(options.clone()), |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         provisioner.provision_with_mnemonic(mnemonic, provisioner_config.clone())?; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | 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>, | ||||||
|  | ) -> 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 | ||||||
|  |             .discover()? | ||||||
|  |             .into_iter() | ||||||
|  |             .map(|(name, _ctx)| name) | ||||||
|  |             .collect(), | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     assert_eq!( | ||||||
|  |         identifiers.len(), | ||||||
|  |         count, | ||||||
|  |         "amount of provisionable devices discovered did not match provisioner count" | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     for identifier in identifiers { | ||||||
|  |         let provisioner_with_identifier = provision::Provision { | ||||||
|  |             identifier: Some(identifier), | ||||||
|  |             ..provision.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)?; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     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(()) |     Ok(()) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -567,25 +811,47 @@ impl MnemonicSubcommands { | ||||||
|             MnemonicSubcommands::Generate { |             MnemonicSubcommands::Generate { | ||||||
|                 source, |                 source, | ||||||
|                 size, |                 size, | ||||||
|  |                 derive, | ||||||
|                 encrypt_to, |                 encrypt_to, | ||||||
|                 shard_to, |                 shard_to, | ||||||
|                 shard, |                 shard, | ||||||
|                 encrypt_to_self, |                 encrypt_to_self, | ||||||
|  |                 shard_to_self, | ||||||
|                 provision, |                 provision, | ||||||
|  |                 provision_count, | ||||||
|  |                 provision_config, | ||||||
|             } => { |             } => { | ||||||
|                 // NOTE: We should never have a case where there's Some() of empty vec, but
 |                 // 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 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 = |                 let mut will_print_mnemonic = | ||||||
|                     encrypt_to.is_none() || encrypt_to.as_ref().is_some_and(|e| e.is_empty()); |                     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() |                 will_print_mnemonic = will_print_mnemonic && shard_to.is_none() | ||||||
|                     || shard_to.as_ref().is_some_and(|s| s.is_empty()); |                     || shard_to.as_ref().is_some_and(|s| s.is_empty()); | ||||||
|                 will_print_mnemonic = will_print_mnemonic && shard.is_none() |                 will_print_mnemonic = will_print_mnemonic && shard.is_none() | ||||||
|                     || shard.as_ref().is_some_and(|s| s.is_empty()); |                     || shard.as_ref().is_some_and(|s| s.is_empty()); | ||||||
|                 will_print_mnemonic = will_print_mnemonic |                 will_print_mnemonic = will_print_mnemonic && shard_to_self.is_none(); | ||||||
|                     && (encrypt_to_self.as_ref().is_none() || provision.as_ref().is_none()); |  | ||||||
| 
 | 
 | ||||||
|                 let mnemonic = source.handle(size)?; |                 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 { |                 if let Some(encrypt_to) = encrypt_to { | ||||||
|                     for entry in encrypt_to { |                     for entry in encrypt_to { | ||||||
|                         do_encrypt_to(&mnemonic, &entry.inner, &entry.values)?; |                         do_encrypt_to(&mnemonic, &entry.inner, &entry.values)?; | ||||||
|  | @ -593,46 +859,70 @@ impl MnemonicSubcommands { | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 if let Some(encrypt_to_self) = encrypt_to_self { |                 if let Some(encrypt_to_self) = encrypt_to_self { | ||||||
|                     let mut values = encrypt_to_self.values.clone(); |                     let mut accounts: std::collections::HashSet<u32> = Default::default(); | ||||||
|                     // If we have a userid from `provision` but not one here, use that one.
 |                     if let Some(provision::Provision { | ||||||
|                     if let Some(provision) = provision { |                         provisioner_name: provision::Provisioner::OpenPGPCard(_), | ||||||
|                         if matches!(&provision.inner, provision::Provisioner::OpenPGPCard(_)) |                         account_id, | ||||||
|                             && !values.contains_key("userid") |                         .. | ||||||
|  |                     }) = provision | ||||||
|                     { |                     { | ||||||
|                             if let Some(userid) = provision.values.get("userid") { |                         accounts.insert(*account_id); | ||||||
|                                 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)?; | ||||||
|                 } |                 } | ||||||
|                     do_encrypt_to_self(&mnemonic, &encrypt_to_self.inner, &values)?; | 
 | ||||||
|  |                 if let Some(shard_to_self) = shard_to_self { | ||||||
|  |                     do_shard_to_self(&mnemonic, &shard_to_self.inner, &shard_to_self.values)?; | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 if let Some(provisioner) = provision { |                 if let Some(provisioner) = provision { | ||||||
|                     // NOTE: If we have encrypt_to_self, we likely also have the certificate
 |                     // determine if we should write to standard output based on whether we have a
 | ||||||
|                     // already generated. Therefore, we can skip generating it in the provisioner.
 |                     // matching pair of provisioner and public derivation output.
 | ||||||
|                     // However, if we don't have encrypt_to_self, we might not have the
 |                     let mut will_output_public_key = true; | ||||||
|                     // certificate, therefore the provisioner - by default - generates the public
 | 
 | ||||||
|                     // key output.
 |                     if let Some(derive) = derive { | ||||||
|                     //
 |                         let matches = match (provisioner, derive) { | ||||||
|                     // 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
 |                                 provision::Provision { | ||||||
|                     // priority, since it can only be manually set by the user.
 |                                     provisioner_name: provision::Provisioner::OpenPGPCard(_), | ||||||
|                     let mut values = provisioner.values.clone(); |                                     account_id: p_id, | ||||||
|                     if let Some(encrypt_to_self) = encrypt_to_self { |                                     .. | ||||||
|                         if !values.contains_key("output") { |                                 }, | ||||||
|  |                                 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") { | ||||||
|                         values.insert(String::from("_skip_cert_output"), String::from("1")); |                         values.insert(String::from("_skip_cert_output"), String::from("1")); | ||||||
|                     } |                     } | ||||||
|                         // If we have a userid from `encrypt_to_self` but not one here, use that
 | 
 | ||||||
|                         // one.
 |                     do_provision(&mnemonic, provisioner, *provision_count, &values)?; | ||||||
|                         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 { |                 if let Some(shard_to) = shard_to { | ||||||
|  |  | ||||||
|  | @ -5,7 +5,11 @@ mod mnemonic; | ||||||
| mod provision; | mod provision; | ||||||
| mod recover; | mod recover; | ||||||
| mod shard; | mod shard; | ||||||
| mod wizard; | 
 | ||||||
|  | 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) | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| /// The Kitchen Sink of Entropy.
 | /// The Kitchen Sink of Entropy.
 | ||||||
| #[derive(Parser, Clone, Debug)] | #[derive(Parser, Clone, Debug)] | ||||||
|  | @ -57,9 +61,6 @@ pub enum KeyforkCommands { | ||||||
|     /// leaked by any individual deriver.
 |     /// leaked by any individual deriver.
 | ||||||
|     Recover(recover::Recover), |     Recover(recover::Recover), | ||||||
| 
 | 
 | ||||||
|     /// Utilities to automatically manage the setup of Keyfork.
 |  | ||||||
|     Wizard(wizard::Wizard), |  | ||||||
| 
 |  | ||||||
|     /// Print an autocompletion file to standard output.
 |     /// Print an autocompletion file to standard output.
 | ||||||
|     ///
 |     ///
 | ||||||
|     /// Keyfork does not manage the installation of completion files. Consult the documentation for
 |     /// Keyfork does not manage the installation of completion files. Consult the documentation for
 | ||||||
|  | @ -90,9 +91,6 @@ impl KeyforkCommands { | ||||||
|             KeyforkCommands::Recover(r) => { |             KeyforkCommands::Recover(r) => { | ||||||
|                 r.handle(keyfork)?; |                 r.handle(keyfork)?; | ||||||
|             } |             } | ||||||
|             KeyforkCommands::Wizard(w) => { |  | ||||||
|                 w.handle(keyfork)?; |  | ||||||
|             } |  | ||||||
|             #[cfg(feature = "completion")] |             #[cfg(feature = "completion")] | ||||||
|             KeyforkCommands::Completion { shell } => { |             KeyforkCommands::Completion { shell } => { | ||||||
|                 let mut command = Keyfork::command(); |                 let mut command = Keyfork::command(); | ||||||
|  |  | ||||||
|  | @ -12,20 +12,27 @@ type Identifier = (String, Option<String>); | ||||||
| #[derive(Debug, Clone)] | #[derive(Debug, Clone)] | ||||||
| pub enum Provisioner { | pub enum Provisioner { | ||||||
|     OpenPGPCard(openpgp::OpenPGPCard), |     OpenPGPCard(openpgp::OpenPGPCard), | ||||||
|  |     Shard(openpgp::Shard), | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl std::fmt::Display for Provisioner { | impl std::fmt::Display for Provisioner { | ||||||
|     fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { |     fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { | ||||||
|         match self { |         f.write_str(self.identifier()) | ||||||
|             Provisioner::OpenPGPCard(_) => f.write_str("openpgp-card"), |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl Provisioner { | 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>> { |     pub fn discover(&self) -> Result<Vec<Identifier>, Box<dyn std::error::Error>> { | ||||||
|         match self { |         match self { | ||||||
|             Provisioner::OpenPGPCard(o) => o.discover(), |             Provisioner::OpenPGPCard(o) => o.discover(), | ||||||
|  |             Provisioner::Shard(s) => s.discover(), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -44,6 +51,16 @@ impl Provisioner { | ||||||
|                 let xprv: XPrv = client.request_xprv(&path)?; |                 let xprv: XPrv = client.request_xprv(&path)?; | ||||||
|                 o.provision(xprv, provisioner) |                 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) | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -62,19 +79,26 @@ impl Provisioner { | ||||||
|                 let xprv = XPrv::new(mnemonic.generate_seed(None))?.derive_path(&path)?; |                 let xprv = XPrv::new(mnemonic.generate_seed(None))?.derive_path(&path)?; | ||||||
|                 o.provision(xprv, provisioner) |                 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 { | impl ValueEnum for Provisioner { | ||||||
|     fn value_variants<'a>() -> &'a [Self] { |     fn value_variants<'a>() -> &'a [Self] { | ||||||
|         &[Self::OpenPGPCard(openpgp::OpenPGPCard)] |         &[Self::OpenPGPCard(openpgp::OpenPGPCard), Self::Shard(openpgp::Shard)] | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn to_possible_value(&self) -> Option<PossibleValue> { |     fn to_possible_value(&self) -> Option<PossibleValue> { | ||||||
|         Some(PossibleValue::new(match self { |         Some(PossibleValue::new(self.identifier())) | ||||||
|             Self::OpenPGPCard(_) => "openpgp-card", |  | ||||||
|         })) |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -124,15 +148,27 @@ pub struct Provision { | ||||||
|     #[command(subcommand)] |     #[command(subcommand)] | ||||||
|     pub subcommand: Option<ProvisionSubcommands>, |     pub subcommand: Option<ProvisionSubcommands>, | ||||||
| 
 | 
 | ||||||
|     provisioner_name: Provisioner, |     pub provisioner_name: Provisioner, | ||||||
| 
 | 
 | ||||||
|     /// Account ID.
 |     /// Account ID.
 | ||||||
|     #[arg(long, required(true))] |     #[arg(long, default_value = "0")] | ||||||
|     account_id: Option<u32>, |     pub account_id: u32, | ||||||
| 
 | 
 | ||||||
|     /// Identifier of the hardware to deploy to, listable by running the `discover` subcommand.
 |     /// Identifier of the hardware to deploy to, listable by running the `discover` subcommand.
 | ||||||
|     #[arg(long, required(true))] |     #[arg(long)] | ||||||
|     identifier: Option<String>, |     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)), | ||||||
|  |         ) | ||||||
|  |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // NOTE: In the future, this impl will be used by `keyfork recover` to reprovision hardware from
 | // NOTE: In the future, this impl will be used by `keyfork recover` to reprovision hardware from
 | ||||||
|  | @ -148,7 +184,7 @@ impl TryFrom<Provision> for config::Provisioner { | ||||||
| 
 | 
 | ||||||
|     fn try_from(value: Provision) -> Result<Self, Self::Error> { |     fn try_from(value: Provision) -> Result<Self, Self::Error> { | ||||||
|         Ok(Self { |         Ok(Self { | ||||||
|             account: value.account_id.ok_or(MissingField("account_id"))?, |             account: value.account_id, | ||||||
|             identifier: value.identifier.ok_or(MissingField("identifier"))?, |             identifier: value.identifier.ok_or(MissingField("identifier"))?, | ||||||
|             metadata: Default::default(), |             metadata: Default::default(), | ||||||
|         }) |         }) | ||||||
|  | @ -171,7 +207,21 @@ impl Provision { | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             None => { |             None => { | ||||||
|                 self.provisioner_name.provision(self.clone().try_into()?)?; |                 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)?; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         Ok(()) |         Ok(()) | ||||||
|  |  | ||||||
|  | @ -1,5 +1,8 @@ | ||||||
| use super::ProvisionExec; | use super::ProvisionExec; | ||||||
| use crate::{config, openpgp_card::factory_reset_current_card}; | use crate::{ | ||||||
|  |     config, | ||||||
|  |     openpgp_card::{factory_reset_current_card, get_new_pins}, | ||||||
|  | }; | ||||||
| 
 | 
 | ||||||
| use card_backend_pcsc::PcscBackend; | use card_backend_pcsc::PcscBackend; | ||||||
| use keyfork_derive_openpgp::{ | use keyfork_derive_openpgp::{ | ||||||
|  | @ -11,24 +14,15 @@ use keyfork_derive_openpgp::{ | ||||||
|     }, |     }, | ||||||
|     XPrv, |     XPrv, | ||||||
| }; | }; | ||||||
| use keyfork_prompt::{ | use keyfork_prompt::default_handler; | ||||||
|     default_handler, prompt_validated_passphrase, |  | ||||||
|     validators::{SecurePinValidator, Validator}, |  | ||||||
| }; |  | ||||||
| use openpgp_card_sequoia::{state::Open, Card}; | use openpgp_card_sequoia::{state::Open, Card}; | ||||||
| use std::path::PathBuf; | use std::path::PathBuf; | ||||||
| 
 | 
 | ||||||
| #[derive(Clone, Debug)] |  | ||||||
| pub struct OpenPGPCard; |  | ||||||
| 
 |  | ||||||
| #[derive(thiserror::Error, Debug)] | #[derive(thiserror::Error, Debug)] | ||||||
| #[error("Provisioner was unable to find a matching smartcard")] | #[error("Provisioner was unable to find a matching smartcard")] | ||||||
| struct NoMatchingSmartcard; | struct NoMatchingSmartcard; | ||||||
| 
 | 
 | ||||||
| impl ProvisionExec for OpenPGPCard { | fn discover_cards() -> Result<Vec<(String, Option<String>)>, Box<dyn std::error::Error>> { | ||||||
|     type PrivateKey = keyfork_derive_openpgp::XPrvKey; |  | ||||||
| 
 |  | ||||||
|     fn discover(&self) -> Result<Vec<(String, Option<String>)>, Box<dyn std::error::Error>> { |  | ||||||
|     let mut idents = vec![]; |     let mut idents = vec![]; | ||||||
|     for backend in PcscBackend::cards(None)? { |     for backend in PcscBackend::cards(None)? { | ||||||
|         let backend = backend?; |         let backend = backend?; | ||||||
|  | @ -42,39 +36,13 @@ impl ProvisionExec for OpenPGPCard { | ||||||
|     Ok(idents) |     Ok(idents) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|     fn derivation_prefix() -> keyfork_derive_util::DerivationPath { | fn provision_card( | ||||||
|         keyfork_derive_path_data::paths::OPENPGP.clone() |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn provision( |  | ||||||
|         &self, |  | ||||||
|         xprv: XPrv, |  | ||||||
|     provisioner: config::Provisioner, |     provisioner: config::Provisioner, | ||||||
|  |     xprv: XPrv, | ||||||
| ) -> Result<(), Box<dyn std::error::Error>> { | ) -> Result<(), Box<dyn std::error::Error>> { | ||||||
|     let mut pm = default_handler()?; |     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 = prompt_validated_passphrase( |     let (user_pin, admin_pin) = get_new_pins(&mut *pm)?; | ||||||
|             &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![ |     let subkeys = vec![ | ||||||
|         KeyFlags::empty().set_certification(), |         KeyFlags::empty().set_certification(), | ||||||
|  | @ -96,11 +64,7 @@ impl ProvisionExec for OpenPGPCard { | ||||||
|         .as_ref() |         .as_ref() | ||||||
|         .is_some_and(|m| m.contains_key("_skip_cert_output")) |         .is_some_and(|m| m.contains_key("_skip_cert_output")) | ||||||
|     { |     { | ||||||
|             let cert_output = match provisioner |         let cert_output = match provisioner.metadata.as_ref().and_then(|m| m.get("output")) { | ||||||
|                 .metadata |  | ||||||
|                 .as_ref() |  | ||||||
|                 .and_then(|m| m.get("output")) |  | ||||||
|             { |  | ||||||
|             Some(cert_output) => PathBuf::from(cert_output), |             Some(cert_output) => PathBuf::from(cert_output), | ||||||
|             None => { |             None => { | ||||||
|                 let path = PathBuf::from(cert.fingerprint().to_string()).with_extension("asc"); |                 let path = PathBuf::from(cert.fingerprint().to_string()).with_extension("asc"); | ||||||
|  | @ -112,7 +76,7 @@ impl ProvisionExec for OpenPGPCard { | ||||||
|             } |             } | ||||||
|         }; |         }; | ||||||
| 
 | 
 | ||||||
|             let cert_output_file = std::fs::File::create_new(cert_output)?; |         let cert_output_file = std::fs::File::create(cert_output)?; | ||||||
|         let mut writer = Writer::new(cert_output_file, Kind::PublicKey)?; |         let mut writer = Writer::new(cert_output_file, Kind::PublicKey)?; | ||||||
|         cert.serialize(&mut writer)?; |         cert.serialize(&mut writer)?; | ||||||
|         writer.finalize()?; |         writer.finalize()?; | ||||||
|  | @ -141,4 +105,49 @@ impl ProvisionExec for OpenPGPCard { | ||||||
| 
 | 
 | ||||||
|     Ok(()) |     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) | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,329 +0,0 @@ | ||||||
| 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}; | use serde::{Deserialize, Serialize}; | ||||||
| 
 | 
 | ||||||
| #[derive(Serialize, Deserialize, Clone)] | #[derive(Serialize, Deserialize, Clone, Debug)] | ||||||
| pub struct Mnemonic { | pub struct Mnemonic { | ||||||
|     pub hash: String, |     pub hash: String, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[derive(Serialize, Deserialize, Clone)] | #[derive(Serialize, Deserialize, Clone, Debug)] | ||||||
| pub struct Provisioner { | pub struct Provisioner { | ||||||
|     pub account: u32, |     pub account: u32, | ||||||
|     pub identifier: String, |     pub identifier: String, | ||||||
|     pub metadata: Option<HashMap<String, String>>, |     pub metadata: Option<HashMap<String, String>>, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[derive(Serialize, Deserialize, Clone)] | #[derive(Serialize, Deserialize, Clone, Debug)] | ||||||
| pub struct Config { | pub struct Config { | ||||||
|     pub mnemonic: Mnemonic, |     pub mnemonic: Mnemonic, | ||||||
|     pub provisioner: Vec<Provisioner>, |     pub provisioner: Vec<Provisioner>, | ||||||
|  |  | ||||||
|  | @ -1,6 +1,68 @@ | ||||||
| use card_backend_pcsc::PcscBackend; | use card_backend_pcsc::PcscBackend; | ||||||
| use openpgp_card_sequoia::{state::Open, types::KeyType, Card, types::TouchPolicy}; | use keyfork_derive_openpgp::openpgp::{policy::Policy, Cert}; | ||||||
| use keyfork_derive_openpgp::openpgp::{Cert, policy::Policy}; | 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)) | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| /// Factory reset the current card so long as it does not match the last-used backend.
 | /// Factory reset the current card so long as it does not match the last-used backend.
 | ||||||
| ///
 | ///
 | ||||||
|  | @ -51,4 +113,3 @@ pub fn factory_reset_current_card( | ||||||
|     transaction.change_admin_pin("12345678", admin_pin)?; |     transaction.change_admin_pin("12345678", admin_pin)?; | ||||||
|     Ok(true) |     Ok(true) | ||||||
| } | } | ||||||
| 
 |  | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue