//! Structs for handling YubiKeys. use age_core::{ format::{FileKey, FILE_KEY_BYTES}, primitives::{aead_decrypt, hkdf}, secrecy::ExposeSecret, }; use age_plugin::{identity, Callbacks}; use bech32::{ToBase32, Variant}; use dialoguer::Password; use log::{debug, error, warn}; use std::fmt; use std::io; use std::iter; use std::thread::sleep; use std::time::{Duration, Instant, SystemTime}; use yubikey::{ certificate::Certificate, piv::{decrypt_data, AlgorithmId, RetiredSlotId, SlotId}, reader::{Context, Reader}, Key, MgmKey, PinPolicy, Serial, TouchPolicy, YubiKey, }; use crate::{ error::Error, fl, format::{RecipientLine, STANZA_KEY_LABEL}, p256::{Recipient, TAG_BYTES}, util::{otp_serial_prefix, Metadata}, IDENTITY_PREFIX, }; const ONE_SECOND: Duration = Duration::from_secs(1); const FIFTEEN_SECONDS: Duration = Duration::from_secs(15); pub(crate) fn is_connected(reader: Reader) -> bool { filter_connected(&reader) } pub(crate) fn filter_connected(reader: &Reader) -> bool { match reader.open() { Err(yubikey::Error::PcscError { inner: Some(pcsc::Error::RemovedCard), }) => { warn!( "{}", fl!("warn-yk-not-connected", yubikey_name = reader.name()) ); false } _ => true, } } pub(crate) fn wait_for_readers() -> Result { // Start a 15-second timer waiting for a YubiKey to be inserted (if necessary). let start = SystemTime::now(); loop { let mut readers = Context::open()?; if readers.iter()?.any(is_connected) { break Ok(readers); } match SystemTime::now().duration_since(start) { Ok(end) if end >= FIFTEEN_SECONDS => return Err(Error::TimedOut), _ => sleep(ONE_SECOND), } } } /// Looks for agent processes that might be holding exclusive access to a YubiKey, and /// asks them as nicely as possible to release it. /// /// Returns `true` if any known agent was running and was successfully interrupted (or /// killed if the platform doesn't support interrupts). fn hunt_agents() -> bool { debug!("Sharing violation encountered, looking for agent processes"); use sysinfo::{ProcessExt, ProcessRefreshKind, RefreshKind, Signal, System, SystemExt}; let mut interrupted = false; let sys = System::new_with_specifics(RefreshKind::new().with_processes(ProcessRefreshKind::new())); for process in sys.processes().values() { match process.name() { "scdaemon" | "scdaemon.exe" => { // gpg-agent runs scdaemon to interact with smart cards. The canonical way // to reload it is `gpgconf --reload scdaemon`, which kills and restarts // the process. We emulate that here with SIGINT (which it listens to). if process .kill_with(Signal::Interrupt) .unwrap_or_else(|| process.kill()) { debug!("Stopped scdaemon (PID {})", process.pid()); interrupted = true; } } "yubikey-agent" | "yubikey-agent.exe" => { // yubikey-agent releases all YubiKey locks when it receives a SIGHUP. match process.kill_with(Signal::Hangup) { Some(true) => { debug!("Sent SIGHUP to yubikey-agent (PID {})", process.pid()); interrupted = true; } Some(false) => (), None => debug!( "Found yubikey-agent (PID {}) but platform doesn't support SIGHUP", process.pid(), ), } } _ => (), } } // If we did interrupt an agent, pause briefly to allow it to finish up. if interrupted { sleep(Duration::from_millis(100)); } interrupted } fn open_sesame( op: impl Fn() -> Result, ) -> Result { op().or_else(|e| match e { yubikey::Error::PcscError { inner: Some(pcsc::Error::SharingViolation), } if hunt_agents() => op(), _ => Err(e), }) } /// Opens a connection to this reader, returning a `YubiKey` if successful. /// /// This is equivalent to [`Reader::open`], but additionally handles the presence of /// agents (which can indefinitely hold exclusive access to a YubiKey). pub(crate) fn open_connection(reader: &Reader) -> Result { open_sesame(|| reader.open()) } /// Opens a YubiKey with a specific serial number. /// /// This is equivalent to [`YubiKey::open_by_serial`], but additionally handles the /// presence of agents (which can indefinitely hold exclusive access to a YubiKey). fn open_by_serial(serial: Serial) -> Result { // `YubiKey::open_by_serial` has a bug where it ignores all opening errors, even if // it potentially could have found a matching YubiKey if not for an error, and thus // returns `Error::NotFound` if another agent is holding exclusive access to the // required YubiKey. This gives misleading UX behaviour where age-plugin-yubikey asks // the user to insert a YubiKey they have already inserted. // // For now, we instead implement the correct behaviour manually. Once MSRV has been // raised to 1.60, we can upstream this into the `yubikey` crate. open_sesame(|| { let mut readers = Context::open()?; let mut open_error = None; for reader in readers.iter()? { let yubikey = match reader.open() { Ok(yk) => yk, Err(e) => { // Save the first error we see that indicates we might have been able // to find a matching YubiKey. if open_error.is_none() { if let yubikey::Error::PcscError { inner: Some(pcsc::Error::SharingViolation), } = e { open_error = Some(e); } } continue; } }; if serial == yubikey.serial() { return Ok(yubikey); } } Err(if let Some(e) = open_error { e } else { error!("no YubiKey detected with serial: {}", serial); yubikey::Error::NotFound }) }) } pub(crate) fn open(serial: Option) -> Result { if !Context::open()?.iter()?.any(is_connected) { if let Some(serial) = serial { eprintln!( "{}", fl!("open-yk-with-serial", yubikey_serial = serial.to_string()) ); } else { eprintln!("{}", fl!("open-yk-without-serial")); } } let mut readers = wait_for_readers()?; let mut readers_iter = readers.iter()?.filter(filter_connected); // --serial selects the YubiKey to use. If not provided, and more than one YubiKey is // connected, an error is returned. let yubikey = match (readers_iter.next(), readers_iter.next(), serial) { (None, _, _) => unreachable!(), (Some(reader), None, None) => open_connection(&reader)?, (Some(reader), None, Some(serial)) => { let yubikey = open_connection(&reader)?; if yubikey.serial() != serial { return Err(Error::NoMatchingSerial(serial)); } yubikey } (Some(a), Some(b), Some(serial)) => { let reader = iter::empty() .chain(Some(a)) .chain(Some(b)) .chain(readers_iter) .find(|reader| match open_connection(reader) { Ok(yk) => yk.serial() == serial, _ => false, }) .ok_or(Error::NoMatchingSerial(serial))?; open_connection(&reader)? } (Some(_), Some(_), None) => return Err(Error::MultipleYubiKeys), }; Ok(yubikey) } pub(crate) fn manage(yubikey: &mut YubiKey) -> Result<(), Error> { const DEFAULT_PIN: &str = "123456"; const DEFAULT_PUK: &str = "12345678"; eprintln!(); let pin = Password::new() .with_prompt(fl!( "mgr-enter-pin", yubikey_serial = yubikey.serial().to_string(), default_pin = DEFAULT_PIN, )) .interact()?; yubikey.verify_pin(pin.as_bytes())?; // If the user is using the default PIN, help them to change it. if pin == DEFAULT_PIN { eprintln!(); eprintln!("{}", fl!("mgr-change-default-pin")); eprintln!(); let current_puk = Password::new() .with_prompt(fl!("mgr-enter-current-puk", default_puk = DEFAULT_PUK)) .interact()?; let new_pin = Password::new() .with_prompt(fl!("mgr-choose-new-pin")) .with_confirmation(fl!("mgr-repeat-new-pin"), fl!("mgr-pin-mismatch")) .interact()?; if new_pin.len() > 8 { return Err(Error::InvalidPinLength); } yubikey.change_puk(current_puk.as_bytes(), new_pin.as_bytes())?; yubikey.change_pin(pin.as_bytes(), new_pin.as_bytes())?; } if let Ok(mgm_key) = MgmKey::get_protected(yubikey) { yubikey.authenticate(mgm_key)?; } else { // Try to authenticate with the default management key. yubikey .authenticate(MgmKey::default()) .map_err(|_| Error::CustomManagementKey)?; // Migrate to a PIN-protected management key. let mgm_key = MgmKey::generate(); eprintln!(); eprintln!("{}", fl!("mgr-changing-mgmt-key")); eprint!("... "); mgm_key.set_protected(yubikey).map_err(|e| { eprintln!( "{}", fl!( "mgr-changing-mgmt-key-error", management_key = hex::encode(mgm_key.as_ref()), ) ); e })?; eprintln!("{}", fl!("mgr-changing-mgmt-key-success")); } Ok(()) } /// Returns an iterator of keys that are occupying plugin-compatible slots, along with the /// corresponding recipient if the key is compatible with this plugin. pub(crate) fn list_slots( yubikey: &mut YubiKey, ) -> Result)>, Error> { Ok(Key::list(yubikey)?.into_iter().filter_map(|key| { // We only use the retired slots. match key.slot() { SlotId::Retired(slot) => { // Only P-256 keys are compatible with us. let recipient = Recipient::from_certificate(key.certificate()); Some((key, slot, recipient)) } _ => None, } })) } /// Returns an iterator of keys that are compatible with this plugin. pub(crate) fn list_compatible( yubikey: &mut YubiKey, ) -> Result, Error> { list_slots(yubikey) .map(|iter| iter.filter_map(|(key, slot, res)| res.map(|recipient| (key, slot, recipient)))) } /// A reference to an age key stored in a YubiKey. #[derive(Debug)] pub struct Stub { pub(crate) serial: Serial, pub(crate) slot: RetiredSlotId, pub(crate) tag: [u8; TAG_BYTES], pub(crate) identity_index: usize, } impl fmt::Display for Stub { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.write_str( bech32::encode( IDENTITY_PREFIX, self.to_bytes().to_base32(), Variant::Bech32, ) .expect("HRP is valid") .to_uppercase() .as_str(), ) } } impl PartialEq for Stub { fn eq(&self, other: &Self) -> bool { self.to_bytes().eq(&other.to_bytes()) } } impl Stub { /// Returns a key stub and recipient for this `(Serial, SlotId, PublicKey)` tuple. /// /// Does not check that the `PublicKey` matches the given `(Serial, SlotId)` tuple; /// this is checked at decryption time. pub(crate) fn new(serial: Serial, slot: RetiredSlotId, recipient: &Recipient) -> Self { Stub { serial, slot, tag: recipient.tag(), identity_index: 0, } } pub(crate) fn from_bytes(bytes: &[u8], identity_index: usize) -> Option { if bytes.len() < 9 { return None; } let serial = Serial::from(u32::from_le_bytes(bytes[0..4].try_into().unwrap())); let slot: RetiredSlotId = bytes[4].try_into().ok()?; Some(Stub { serial, slot, tag: bytes[5..9].try_into().unwrap(), identity_index, }) } fn to_bytes(&self) -> Vec { let mut bytes = Vec::with_capacity(9); bytes.extend_from_slice(&self.serial.0.to_le_bytes()); bytes.push(self.slot.into()); bytes.extend_from_slice(&self.tag); bytes } pub(crate) fn matches(&self, line: &RecipientLine) -> bool { self.tag == line.tag } /// Returns: /// - `Ok(Ok(Some(connection)))` if we successfully connected to this YubiKey. /// - `Ok(Ok(None))` if the user told us to skip this YubiKey. /// - `Ok(Err(_))` if we encountered an error while trying to connect to the YubiKey. /// - `Err(_)` on communication errors with the age client. pub(crate) fn connect( &self, callbacks: &mut dyn Callbacks, ) -> io::Result, identity::Error>> { let mut yubikey = match open_by_serial(self.serial) { Ok(yk) => yk, Err(yubikey::Error::NotFound) => { let mut message = fl!("plugin-insert-yk", yubikey_serial = self.serial.to_string()); // If the `confirm` command is available, we loop until either the YubiKey // we want is inserted, or the used explicitly skips. let yubikey = loop { match callbacks.confirm( &message, &fl!("plugin-yk-is-plugged-in"), Some(&fl!("plugin-skip-this-yk")), )? { // `confirm` command is not available. Err(age_core::plugin::Error::Unsupported) => break None, // User told us to skip this key. Ok(false) => return Ok(Ok(None)), // User said they plugged it in; try it. Ok(true) => match open_by_serial(self.serial) { Ok(yubikey) => break Some(yubikey), Err(yubikey::Error::NotFound) => (), Err(_) => { return Ok(Err(identity::Error::Identity { index: self.identity_index, message: fl!( "plugin-err-yk-opening", yubikey_serial = self.serial.to_string(), ), })); } }, // We can't communicate with the user. Err(age_core::plugin::Error::Fail) => { return Ok(Err(identity::Error::Identity { index: self.identity_index, message: fl!( "plugin-err-yk-opening", yubikey_serial = self.serial.to_string(), ), })) } } // We're going to loop around, meaning that the first attempt failed. // Change the message to indicate this to the user. message = fl!( "plugin-insert-yk-retry", yubikey_serial = self.serial.to_string(), ); }; if let Some(yk) = yubikey { yk } else { // `confirm` is not available; fall back to `message` with a timeout. if callbacks.message(&message)?.is_err() { return Ok(Err(identity::Error::Identity { index: self.identity_index, message: fl!( "plugin-err-yk-not-found", yubikey_serial = self.serial.to_string(), ), })); } // Start a 15-second timer waiting for the YubiKey to be inserted let start = SystemTime::now(); loop { match open_by_serial(self.serial) { Ok(yubikey) => break yubikey, Err(yubikey::Error::NotFound) => (), Err(_) => { return Ok(Err(identity::Error::Identity { index: self.identity_index, message: fl!( "plugin-err-yk-opening", yubikey_serial = self.serial.to_string(), ), })); } } match SystemTime::now().duration_since(start) { Ok(end) if end >= FIFTEEN_SECONDS => { return Ok(Err(identity::Error::Identity { index: self.identity_index, message: fl!( "plugin-err-yk-timed-out", yubikey_serial = self.serial.to_string(), ), })) } _ => sleep(ONE_SECOND), } } } } Err(_) => { return Ok(Err(identity::Error::Identity { index: self.identity_index, message: fl!( "plugin-err-yk-opening", yubikey_serial = self.serial.to_string(), ), })) } }; // Read the pubkey from the YubiKey slot and check it still matches. let (cert, pk) = match Certificate::read(&mut yubikey, SlotId::Retired(self.slot)) .ok() .and_then(|cert| { Recipient::from_certificate(&cert) .filter(|pk| pk.tag() == self.tag) .map(|pk| (cert, pk)) }) { Some(pk) => pk, None => { return Ok(Err(identity::Error::Identity { index: self.identity_index, message: fl!("plugin-err-yk-stub-mismatch"), })) } }; Ok(Ok(Some(Connection { yubikey, cert, pk, slot: self.slot, tag: self.tag, identity_index: self.identity_index, cached_metadata: None, last_touch: None, }))) } } pub(crate) struct Connection { yubikey: YubiKey, cert: Certificate, pk: Recipient, slot: RetiredSlotId, tag: [u8; 4], identity_index: usize, cached_metadata: Option, last_touch: Option, } impl Connection { pub(crate) fn recipient(&self) -> &Recipient { &self.pk } pub(crate) fn request_pin_if_necessary( &mut self, callbacks: &mut dyn Callbacks, ) -> io::Result> { // Check if we can skip requesting a PIN. if self.cached_metadata.is_none() { self.cached_metadata = match Metadata::extract(&mut self.yubikey, self.slot, &self.cert, true) { None => { return Ok(Err(identity::Error::Identity { index: self.identity_index, message: fl!("plugin-err-yk-invalid-pin-policy"), })) } metadata => metadata, }; } if let Some(PinPolicy::Never) = self.cached_metadata.as_ref().and_then(|m| m.pin_policy) { return Ok(Ok(())); } // The policy requires a PIN, so request it. // Note that we can't distinguish between PinPolicy::Once and PinPolicy::Always // because this plugin is ephemeral, so we always request the PIN. let enter_pin_msg = fl!( "plugin-enter-pin", yubikey_serial = self.yubikey.serial().to_string(), ); let mut message = enter_pin_msg.clone(); let pin = loop { message = match callbacks.request_secret(&message)? { Ok(pin) => match pin.expose_secret().len() { // A PIN must be between 6 and 8 characters. 6..=8 => break pin, // If the string is 44 bytes and starts with the YubiKey's serial // encoded as 12-byte modhex, the user probably touched the YubiKey // early and "typed" an OTP. 44 if pin .expose_secret() .starts_with(&otp_serial_prefix(self.yubikey.serial())) => { format!("{} {}", fl!("plugin-err-accidental-touch"), enter_pin_msg) } // Otherwise, the PIN is either too short or too long. 0..=5 => format!("{} {}", fl!("plugin-err-pin-too-short"), enter_pin_msg), _ => format!("{} {}", fl!("plugin-err-pin-too-long"), enter_pin_msg), }, Err(_) => { return Ok(Err(identity::Error::Identity { index: self.identity_index, message: fl!( "plugin-err-pin-required", yubikey_serial = self.yubikey.serial().to_string(), ), })) } }; }; if let Err(e) = self.yubikey.verify_pin(pin.expose_secret().as_bytes()) { return Ok(Err(identity::Error::Identity { index: self.identity_index, message: format!("{:?}", Error::YubiKey(e)), })); } Ok(Ok(())) } pub(crate) fn unwrap_file_key(&mut self, line: &RecipientLine) -> Result { assert_eq!(self.tag, line.tag); // Check if the touch policy requires a touch. let needs_touch = match ( self.cached_metadata.as_ref().and_then(|m| m.touch_policy), self.last_touch, ) { (Some(TouchPolicy::Always), _) | (Some(TouchPolicy::Cached), None) => true, (Some(TouchPolicy::Cached), Some(last)) if last.elapsed() >= FIFTEEN_SECONDS => true, _ => false, }; // The YubiKey API for performing scalar multiplication takes the point in its // uncompressed SEC-1 encoding. let shared_secret = match decrypt_data( &mut self.yubikey, line.epk_bytes.decompress().as_bytes(), AlgorithmId::EccP256, SlotId::Retired(self.slot), ) { Ok(res) => res, Err(_) => return Err(()), }; // If we requested a touch and reached here, the user touched the YubiKey. if needs_touch { if let Some(TouchPolicy::Cached) = self.cached_metadata.as_ref().and_then(|m| m.touch_policy) { self.last_touch = Some(Instant::now()); } } let mut salt = vec![]; salt.extend_from_slice(line.epk_bytes.as_bytes()); salt.extend_from_slice(self.pk.to_encoded().as_bytes()); let enc_key = hkdf(&salt, STANZA_KEY_LABEL, shared_secret.as_ref()); // A failure to decrypt is fatal, because we assume that we won't // encounter 32-bit collisions on the key tag embedded in the header. match aead_decrypt(&enc_key, FILE_KEY_BYTES, &line.encrypted_file_key) { Ok(pt) => Ok(TryInto::<[u8; FILE_KEY_BYTES]>::try_into(&pt[..]) .unwrap() .into()), Err(_) => Err(()), } } } #[cfg(test)] mod tests { use yubikey::{piv::RetiredSlotId, Serial}; use super::Stub; #[test] fn stub_round_trip() { let stub = Stub { serial: Serial::from(42), slot: RetiredSlotId::R1, tag: [7; 4], identity_index: 0, }; let encoded = stub.to_bytes(); assert_eq!(Stub::from_bytes(&[], 0), None); assert_eq!(Stub::from_bytes(&encoded, 0), Some(stub)); assert_eq!(Stub::from_bytes(&encoded[..encoded.len() - 1], 0), None); } }