use rpassword::prompt_password; use rustyline::error::ReadlineError; use rustyline::Editor; use std::{cell::RefCell, rc::Rc}; use crate::lk::LK; use crate::parser::command_parser; use crate::structs::{Command, LKErr, LKOut, HISTORY_FILE}; #[derive(Debug)] pub struct LKRead { pub rl: Editor<()>, pub prompt: String, pub state: Rc>, pub cmd: String, pub read_password: fn(String) -> std::io::Result, } #[derive(Debug)] pub struct LKEval<'a> { pub cmd: Command<'a>, pub state: Rc>, pub read_password: fn(String) -> std::io::Result, } #[derive(Debug, PartialEq)] pub struct LKPrint { pub out: LKOut, pub quit: bool, pub state: Rc>, } impl LKRead { pub fn new(rl: Editor<()>, prompt: String, state: Rc>) -> Self { Self { rl, prompt, state, cmd: "".to_string(), read_password: prompt_password, } } pub fn read(&mut self) -> LKEval { let history_file = HISTORY_FILE.to_str().unwrap(); self.rl.clear_history(); match self.rl.load_history(&history_file) { Ok(_) => (), Err(_) => { self.rl.add_history_entry("ls"); () } } self.cmd = match self.rl.readline(&*self.prompt) { Ok(str) => str, Err(ReadlineError::Eof | ReadlineError::Interrupted) => "quit".to_string(), Err(err) => { return LKEval::new( Command::Error(LKErr::ReadError(err.to_string())), self.state.clone(), self.read_password, ) } }; self.rl.add_history_entry(self.cmd.as_str()); match self.rl.save_history(&history_file) { Ok(_) => (), Err(_) => (), } match command_parser::cmd(self.cmd.as_str()) { Ok(cmd) => LKEval::new(cmd, self.state.clone(), self.read_password), Err(err) => LKEval::new(Command::Error(LKErr::ParseError(err)), self.state.clone(), self.read_password), } } pub fn refresh(&mut self) {} pub fn quit(&mut self) {} } impl<'a> LKEval<'a> { pub fn new(cmd: Command<'a>, state: Rc>, read_password: fn(String) -> std::io::Result) -> Self { Self { cmd, state, read_password, } } pub fn eval(&self) -> LKPrint { let out = LKOut::new(); let mut quit: bool = false; match &self.cmd { Command::Quit => { out.e("Bye!".to_string()); quit = true; } Command::Ls(filter) => self.cmd_ls(&out, filter.to_string()), Command::Add(name) => self.cmd_add(&out, &name), Command::Comment(name, comment) => self.cmd_comment(&out, &name, &comment), Command::Rm(name) => match self.get_password(name) { Some(pwd) => { self.state.borrow_mut().db.remove(&pwd.borrow().name); out.o(format!("removed {}", pwd.borrow().name)); } None => out.e(format!("error: password {} not found", name)), }, Command::Enc(name) => { self.cmd_enc(&out, name); } Command::Gen(num, name) => self.cmd_gen(&out, &num, &name), Command::PasteBuffer(command) => self.cmd_pb(&out, command), Command::Source(script) => self.cmd_source(&out, script), Command::Dump(script) => self.cmd_dump(&out, script), Command::Pass(name) => self.cmd_pass(&out, &name), Command::UnPass(name) => match self.state.borrow_mut().secrets.remove(name) { Some(_) => out.o(format!("Removed saved password for {}", name)), None => out.e(format!("error: saved password for {} not found", name)), }, Command::Correct(name) => self.cmd_correct(&out, name, true, None), Command::Uncorrect(name) => self.cmd_correct(&out, name, false, None), Command::Noop => (), Command::Help => { out.o("HELP".to_string()); } Command::Mv(name, folder) => self.cmd_mv(&out, &name, &folder), Command::Error(error) => match error { LKErr::ParseError(e) => out.e(e.to_string()), LKErr::ReadError(e) => out.e(e.to_string()), LKErr::Error(e) => out.e(format!("error: {}", e.to_string())), }, } LKPrint::new(out, quit, self.state.clone()) } } impl LKPrint { pub fn new(out: LKOut, quit: bool, state: Rc>) -> Self { Self { out, quit, state } } pub fn print(&mut self) -> bool { self.out.print_err(); self.out.print_out(); return !self.quit; } } #[cfg(test)] mod tests { use super::*; use crate::password::Password; use crate::structs::Mode; use chrono::naive::NaiveDate; use std::collections::HashMap; impl<'a> LKEval<'a> { pub fn news(cmd: Command<'a>, state: Rc>) -> Self { Self { cmd, state, read_password: |_| { Err(std::io::Error::new(std::io::ErrorKind::NotConnected, "could not read password")) }, } } } #[test] fn exec_cmds_basic() { let lk = Rc::new(RefCell::new(LK::new())); assert_eq!( LKEval::news(Command::Ls(".".to_string()), lk.clone()).eval(), LKPrint::new(LKOut::from_vecs(vec![], vec![]), false, lk.clone()) ); let pwd1 = Rc::new(RefCell::new(Password { name: "t1".to_string(), prefix: None, length: None, mode: Mode::Regular, seq: 99, date: NaiveDate::from_ymd_opt(2022, 12, 30).unwrap(), comment: Some("comment".to_string()), parent: None, })); assert_eq!(LKEval::news(Command::Add(pwd1.clone()), lk.clone()).eval().state.borrow().db, { let mut db = HashMap::new(); db.insert(pwd1.borrow().name.to_string(), pwd1.clone()); db }); assert_eq!( LKEval::news(Command::Ls(".".to_string()), lk.clone()).eval(), LKPrint::new( LKOut::from_vecs(vec![" 1 t1 R 99 2022-12-30 comment".to_string()], vec![]), false, lk.clone() ) ); assert_eq!( LKEval::news(Command::Quit, lk.clone()).eval(), LKPrint::new(LKOut::from_vecs(vec![], vec!["Bye!".to_string()]), true, lk.clone()) ); let pwd2 = Rc::new(RefCell::new(Password { name: "t2".to_string(), prefix: None, length: None, mode: Mode::Regular, seq: 99, date: NaiveDate::from_ymd_opt(2022, 12, 31).unwrap(), comment: Some("bli blup".to_string()), parent: None, })); assert_eq!(LKEval::news(Command::Add(pwd2.clone()), lk.clone()).eval().state.borrow().db, { let mut db = HashMap::new(); db.insert(pwd1.borrow().name.to_string(), pwd1.clone()); db.insert(pwd2.borrow().name.to_string(), pwd2.clone()); db }); assert_eq!( LKEval::news(Command::Ls(".".to_string()), lk.clone()).eval(), LKPrint::new( LKOut::from_vecs( vec![" 1 t1 R 99 2022-12-30 comment".to_string(), " 2 t2 R 99 2022-12-31 bli blup".to_string()], vec![] ), false, lk.clone() ) ); assert_eq!( LKEval::news(Command::Rm("2".to_string()), lk.clone()).eval(), LKPrint::new(LKOut::from_vecs(vec!["removed t2".to_string()], vec![]), false, lk.clone()) ); assert_eq!( LKEval::news(Command::Ls(".".to_string()), lk.clone()).eval(), LKPrint::new( LKOut::from_vecs(vec![" 1 t1 R 99 2022-12-30 comment".to_string()], vec![]), false, lk.clone() ) ); } #[test] fn read_pwd_test() { let lk = Rc::new(RefCell::new(LK::new())); let t1 = Rc::new(RefCell::new(Password::new( None, "t1".to_string(), None, Mode::Regular, 99, NaiveDate::from_ymd_opt(2022, 12, 30).unwrap(), None, ))); let t2 = Rc::new(RefCell::new(Password::new( None, "t2".to_string(), None, Mode::Regular, 99, NaiveDate::from_ymd_opt(2022, 12, 30).unwrap(), None, ))); let t3 = Rc::new(RefCell::new(Password::new( None, "t3".to_string(), None, Mode::Regular, 99, NaiveDate::from_ymd_opt(2022, 12, 30).unwrap(), None, ))); assert_eq!( LKEval::news(Command::Add(t1.clone()), lk.clone()).eval(), LKPrint::new(LKOut::from_vecs(vec![], vec![]), false, lk.clone()) ); assert_eq!( LKEval::news(Command::Add(t2.clone()), lk.clone()).eval(), LKPrint::new(LKOut::from_vecs(vec![], vec![]), false, lk.clone()) ); assert_eq!( LKEval::news(Command::Add(t3.clone()), lk.clone()).eval(), LKPrint::new(LKOut::from_vecs(vec![], vec![]), false, lk.clone()) ); assert_eq!( LKEval::news(Command::Mv("t3".to_string(), "t2".to_string()), lk.clone()).eval(), LKPrint::new(LKOut::from_vecs(vec![], vec![]), false, lk.clone()) ); assert_eq!( LKEval::news(Command::Mv("t2".to_string(), "t1".to_string()), lk.clone()).eval(), LKPrint::new(LKOut::from_vecs(vec![], vec![]), false, lk.clone()) ); assert_eq!( LKEval::new(Command::Enc("t3".to_string()), lk.clone(), |p| if p == "NULL" { Ok("a".to_string()) } else { Err(std::io::Error::new(std::io::ErrorKind::NotFound, "test")) }) .eval(), LKPrint::new( LKOut::from_vecs(vec![], vec!["error: master for t3 not found".to_string()]), false, lk.clone() ) ); assert_eq!( LKEval::new(Command::Enc("t3".to_string()), lk.clone(), |p| if p == "Master: " { Ok("a".to_string()) } else { Err(std::io::Error::new(std::io::ErrorKind::NotFound, "test")) }) .eval(), LKPrint::new( LKOut::from_vecs( vec!["san bud most noon jaw cash".to_string()], vec![ "warning: password / is not marked as correct".to_string(), "warning: password t1 is not marked as correct".to_string(), "warning: password t2 is not marked as correct".to_string(), "warning: password t3 is not marked as correct".to_string(), ] ), false, lk.clone() ) ); assert_eq!( LKEval::new(Command::Enc("t2".to_string()), lk.clone(), |p| if p == "NULL" { Ok("a".to_string()) } else { Err(std::io::Error::new(std::io::ErrorKind::NotFound, "test")) }) .eval(), LKPrint::new( LKOut::from_vecs( vec!["alga barn wise tim skin mock".to_string()], vec!["warning: password t2 is not marked as correct".to_string()] ), false, lk.clone() ) ); assert_eq!( LKEval::new(Command::Enc("t1".to_string()), lk.clone(), |p| if p == "NULL" { Ok("a".to_string()) } else { Err(std::io::Error::new(std::io::ErrorKind::NotFound, "test")) }) .eval(), LKPrint::new( LKOut::from_vecs( vec!["lime rudy jay my kong tack".to_string()], vec!["warning: password t1 is not marked as correct".to_string()] ), false, lk.clone() ) ); } }