// mod.rs
// *************************************************************************
// * Copyright (C) 2019-2020 Daniel Mueller (deso@posteo.net) *
// * *
// * This program is free software: you can redistribute it and/or modify *
// * it under the terms of the GNU General Public License as published by *
// * the Free Software Foundation, either version 3 of the License, or *
// * (at your option) any later version. *
// * *
// * This program is distributed in the hope that it will be useful, *
// * but WITHOUT ANY WARRANTY; without even the implied warranty of *
// * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
// * GNU General Public License for more details. *
// * *
// * You should have received a copy of the GNU General Public License *
// * along with this program. If not, see . *
// *************************************************************************
use std::ffi;
use std::fmt;
use nitrokey_test::test as test_device;
mod config;
mod encrypted;
mod hidden;
mod list;
mod lock;
mod otp;
mod pin;
mod pws;
mod reset;
mod run;
mod status;
mod unencrypted;
/// A trait simplifying checking for expected errors.
pub trait UnwrapError {
/// Unwrap an Error::Error variant.
fn unwrap_str_err(self) -> String;
/// Unwrap a Error::CommandError variant.
fn unwrap_cmd_err(self) -> (Option<&'static str>, nitrokey::CommandError);
/// Unwrap a Error::LibraryError variant.
fn unwrap_lib_err(self) -> (Option<&'static str>, nitrokey::LibraryError);
}
impl UnwrapError for crate::Result
where
T: fmt::Debug,
{
fn unwrap_str_err(self) -> String {
match self.unwrap_err() {
crate::Error::ClapError(err) => {
if err.use_stderr() {
err.message
} else {
String::new()
}
}
crate::Error::Error(err) => err,
err => panic!("Unexpected error variant found: {:?}", err),
}
}
fn unwrap_cmd_err(self) -> (Option<&'static str>, nitrokey::CommandError) {
match self.unwrap_err() {
crate::Error::NitrokeyError(ctx, err) => match err {
nitrokey::Error::CommandError(err) => (ctx, err),
err => panic!("Unexpected error variant found: {:?}", err),
},
err => panic!("Unexpected error variant found: {:?}", err),
}
}
fn unwrap_lib_err(self) -> (Option<&'static str>, nitrokey::LibraryError) {
match self.unwrap_err() {
crate::Error::NitrokeyError(ctx, err) => match err {
nitrokey::Error::LibraryError(err) => (ctx, err),
err => panic!("Unexpected error variant found: {:?}", err),
},
err => panic!("Unexpected error variant found: {:?}", err),
}
}
}
struct Nitrocli {
model: Option,
admin_pin: Option,
user_pin: Option,
new_admin_pin: Option,
new_user_pin: Option,
password: Option,
}
impl Nitrocli {
pub fn new() -> Self {
Self {
model: None,
admin_pin: Some(nitrokey::DEFAULT_ADMIN_PIN.into()),
user_pin: Some(nitrokey::DEFAULT_USER_PIN.into()),
new_admin_pin: None,
new_user_pin: None,
password: None,
}
}
pub fn with_model(model: M) -> Self
where
M: Into,
{
Self {
model: Some(model.into()),
admin_pin: Some(nitrokey::DEFAULT_ADMIN_PIN.into()),
user_pin: Some(nitrokey::DEFAULT_USER_PIN.into()),
new_admin_pin: None,
new_user_pin: None,
password: Some("1234567".into()),
}
}
pub fn admin_pin(&mut self, pin: impl Into) {
self.admin_pin = Some(pin.into())
}
pub fn new_admin_pin(&mut self, pin: impl Into) {
self.new_admin_pin = Some(pin.into())
}
pub fn user_pin(&mut self, pin: impl Into) {
self.user_pin = Some(pin.into())
}
pub fn new_user_pin(&mut self, pin: impl Into) {
self.new_user_pin = Some(pin.into())
}
fn model_to_arg(model: nitrokey::Model) -> &'static str {
match model {
nitrokey::Model::Pro => "--model=pro",
nitrokey::Model::Storage => "--model=storage",
}
}
fn do_run(&mut self, args: &[&str], f: F) -> (R, Vec, Vec)
where
F: FnOnce(&mut crate::RunCtx<'_>, Vec) -> R,
{
let args = ["nitrocli"]
.iter()
.cloned()
.chain(self.model.map(Self::model_to_arg))
.chain(args.iter().cloned())
.map(ToOwned::to_owned)
.collect();
let mut stdout = Vec::new();
let mut stderr = Vec::new();
let ctx = &mut crate::RunCtx {
stdout: &mut stdout,
stderr: &mut stderr,
admin_pin: self.admin_pin.clone(),
user_pin: self.user_pin.clone(),
new_admin_pin: self.new_admin_pin.clone(),
new_user_pin: self.new_user_pin.clone(),
password: self.password.clone(),
no_cache: true,
};
(f(ctx, args), stdout, stderr)
}
/// Run `nitrocli`'s `run` function.
pub fn run(&mut self, args: &[&str]) -> (i32, Vec, Vec) {
self.do_run(args, |c, a| crate::run(c, a))
}
/// Run `nitrocli`'s `handle_arguments` function.
pub fn handle(&mut self, args: &[&str]) -> crate::Result {
let (res, out, _) = self.do_run(args, |c, a| crate::handle_arguments(c, a));
res.map(|_| String::from_utf8_lossy(&out).into_owned())
}
pub fn model(&self) -> Option {
self.model
}
}