use alloc::vec::Vec;
pub use azul_core::biometric::{BiometricKind, BiometricPrompt, BiometricResult};
#[derive(Debug, Clone, PartialEq)]
pub struct BiometricManager {
pub last_result: Option<BiometricResult>,
pub availability: BiometricKind,
}
impl Default for BiometricManager {
fn default() -> Self {
Self {
last_result: None,
availability: BiometricKind::NotAvailable,
}
}
}
impl BiometricManager {
pub fn new() -> Self {
Self::default()
}
pub fn last_result(&self) -> Option<BiometricResult> {
self.last_result
}
pub fn availability(&self) -> BiometricKind {
self.availability
}
pub fn is_available(&self) -> bool {
self.availability.is_available()
}
pub fn set_availability(&mut self, kind: BiometricKind) -> bool {
let changed = self.availability != kind;
self.availability = kind;
changed
}
pub fn set_last_result(&mut self, result: BiometricResult) -> bool {
let changed = self.last_result != Some(result);
self.last_result = Some(result);
changed
}
pub fn last_was_success(&self) -> bool {
matches!(self.last_result, Some(r) if r.is_success())
}
}
static PENDING_RESULTS: std::sync::Mutex<Vec<BiometricResult>> =
std::sync::Mutex::new(Vec::new());
pub fn push_biometric_result(result: BiometricResult) {
let mut q = PENDING_RESULTS.lock().unwrap_or_else(|e| e.into_inner());
q.push(result);
}
pub fn drain_biometric_results() -> Vec<BiometricResult> {
let mut q = PENDING_RESULTS.lock().unwrap_or_else(|e| e.into_inner());
core::mem::take(&mut *q)
}
static PENDING_REQUESTS: std::sync::Mutex<Vec<BiometricPrompt>> =
std::sync::Mutex::new(Vec::new());
pub fn push_biometric_request(prompt: BiometricPrompt) {
let mut q = PENDING_REQUESTS.lock().unwrap_or_else(|e| e.into_inner());
q.push(prompt);
}
pub fn drain_biometric_requests() -> Vec<BiometricPrompt> {
let mut q = PENDING_REQUESTS.lock().unwrap_or_else(|e| e.into_inner());
core::mem::take(&mut *q)
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn manager_defaults_to_unavailable_and_no_result() {
let mgr = BiometricManager::new();
assert_eq!(mgr.availability(), BiometricKind::NotAvailable);
assert!(!mgr.is_available());
assert_eq!(mgr.last_result(), None);
assert!(!mgr.last_was_success());
}
#[test]
fn set_availability_returns_change_flag() {
let mut mgr = BiometricManager::new();
assert!(mgr.set_availability(BiometricKind::Face));
assert!(mgr.is_available());
assert_eq!(mgr.availability(), BiometricKind::Face);
assert!(!mgr.set_availability(BiometricKind::Face));
assert!(mgr.set_availability(BiometricKind::Fingerprint));
}
#[test]
fn set_last_result_returns_change_flag() {
let mut mgr = BiometricManager::new();
assert!(mgr.set_last_result(BiometricResult::Failed));
assert_eq!(mgr.last_result(), Some(BiometricResult::Failed));
assert!(!mgr.last_was_success());
assert!(!mgr.set_last_result(BiometricResult::Failed));
assert!(mgr.set_last_result(BiometricResult::Authenticated));
assert!(mgr.last_was_success());
}
#[test]
fn passcode_fallback_counts_as_success() {
let mut mgr = BiometricManager::new();
mgr.set_last_result(BiometricResult::FellBackToPasscode);
assert!(mgr.last_was_success());
assert!(BiometricResult::FellBackToPasscode.is_success());
for r in [
BiometricResult::Cancelled,
BiometricResult::Failed,
BiometricResult::Unavailable,
BiometricResult::Error,
] {
assert!(!r.is_success(), "{:?} must not be a success", r);
}
}
#[test]
fn async_results_round_trip_through_manager() {
let _ = drain_biometric_results();
push_biometric_result(BiometricResult::Failed);
push_biometric_result(BiometricResult::Authenticated); let drained = drain_biometric_results();
assert_eq!(drained.len(), 2, "both parked results drain in order");
assert_eq!(drained[0], BiometricResult::Failed);
assert_eq!(drained[1], BiometricResult::Authenticated);
let mut mgr = BiometricManager::new();
for r in &drained {
mgr.set_last_result(*r);
}
assert_eq!(
mgr.last_result(),
Some(BiometricResult::Authenticated),
"the last applied result wins"
);
assert!(mgr.last_was_success());
assert!(drain_biometric_results().is_empty());
}
#[test]
fn requests_round_trip_through_channel() {
let _ = drain_biometric_requests();
push_biometric_request(BiometricPrompt::new("Unlock A".into()));
push_biometric_request(BiometricPrompt::new("Unlock B".into()));
let drained = drain_biometric_requests();
assert_eq!(drained.len(), 2, "both queued requests drain in order");
assert_eq!(drained[0].reason.as_str(), "Unlock A");
assert_eq!(drained[1].reason.as_str(), "Unlock B");
assert!(drain_biometric_requests().is_empty());
}
#[test]
fn biometric_prompt_defaults_and_constructor() {
let d = BiometricPrompt::default();
assert!(!d.allow_device_credential);
assert_eq!(d.reason.as_str(), "");
let p = BiometricPrompt::new("Unlock your vault".into());
assert_eq!(p.reason.as_str(), "Unlock your vault");
assert_eq!(p.cancel_label.as_str(), "");
assert!(!p.allow_device_credential);
}
}