fm-tui 0.2.3

FM : a file manager inspired by ranger and dired{n}{n}Config files ~/.config/fm/{n}Documentation https://github.com/qkzk/fm{n}
Documentation
/// Different kind of password
#[derive(Debug, Clone, Copy, Eq, PartialEq)]
pub enum PasswordKind {
    SUDO,
    CRYPTSETUP,
}

impl std::fmt::Display for PasswordKind {
    fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
        let asker = match self {
            Self::SUDO => "sudo   ",
            Self::CRYPTSETUP => "device ",
        };
        write!(f, "{asker}")
    }
}

/// What will this password be used for ?
/// ATM only 3 usages are supported:
/// * mounting an ISO file,
/// * opening an mounting an encrypted device.
/// * running a sudo command
#[derive(Debug, Clone, Copy, Eq, PartialEq)]
pub enum PasswordUsage {
    ISO,
    CRYPTSETUP(PasswordKind),
    SUDOCOMMAND,
    DEVICE,
}

type Password = String;

/// Holds passwords allowing to mount or unmount an encrypted drive.
#[derive(Default, Clone)]
pub struct PasswordHolder {
    sudo: Option<Password>,
    cryptsetup: Option<Password>,
}

/// Custom debug format to prevent leaking passwords in logs.
impl std::fmt::Debug for PasswordHolder {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        f.debug_struct("PasswordHodler")
            .field("sudo", &self.hide_option(PasswordKind::SUDO))
            .field("cryptsetup", &self.hide_option(PasswordKind::CRYPTSETUP))
            .finish()
    }
}

impl PasswordHolder {
    const fn hide_option(&self, password_kind: PasswordKind) -> &str {
        match password_kind {
            PasswordKind::SUDO => Self::hide(self.has_sudo()),
            PasswordKind::CRYPTSETUP => Self::hide(self.has_cryptsetup()),
        }
    }

    const fn hide(is_set: bool) -> &'static str {
        if is_set {
            "Some(****)"
        } else {
            "None"
        }
    }

    /// Set the sudo password.
    pub fn set_sudo(&mut self, password: Password) {
        self.sudo = Some(password);
    }

    /// Set the encrypted device passphrase
    pub fn set_cryptsetup(&mut self, passphrase: Password) {
        self.cryptsetup = Some(passphrase);
    }

    /// Reads the cryptsetup password
    #[must_use]
    pub fn cryptsetup(&mut self) -> Option<Password> {
        std::mem::take(&mut self.cryptsetup)
    }

    /// Reads the sudo password
    #[must_use]
    pub fn sudo(&mut self) -> Option<Password> {
        std::mem::take(&mut self.sudo)
    }

    /// True if the sudo password was set
    #[must_use]
    pub const fn has_sudo(&self) -> bool {
        self.sudo.is_some()
    }

    /// True if the encrypted device passphrase was set
    #[must_use]
    pub const fn has_cryptsetup(&self) -> bool {
        self.cryptsetup.is_some()
    }

    /// Reset every known password, dropping them.
    /// It should be called ASAP.
    pub fn reset(&mut self) {
        std::mem::take(&mut self.sudo);
        std::mem::take(&mut self.cryptsetup);
        self.sudo = None;
        self.cryptsetup = None;
    }
}