A small tool for storing passwords locally with git sync
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

229 lines
6.3 KiB

7 months ago
use crate::encoder;
use encoder::Encoder;
7 months ago
use std::collections::HashSet;
7 months ago
use std::hash::{Hash, Hasher};
7 months ago
use std::env;
7 months ago
use std::fs;
7 months ago
use std::path::Path;
7 months ago
use std::io::{self, Write, BufRead};
7 months ago
use std::fmt;
use std::cmp::{PartialEq, Ordering};
7 months ago
static ENV_MPS_HOME: &str = "MPS_HOME";
static PATH_STORAGE: &str = "storage"; // should be under git
static PATH_DB: &str = "db.mps";
7 months ago
7 months ago
7 months ago
#[derive(Clone)]
pub struct Item {
pub id: String,
pub content: String
}
impl Item {
7 months ago
pub fn from(s: String, c: String) -> Item {
Item { id: s, content: c }
}
// used only to search in HashSet
pub fn from_empty(s: String) -> Item {
Item { id: s, content: String::from("") }
7 months ago
}
}
7 months ago
impl PartialEq for Item {
fn eq(&self, other: &Self) -> bool {
self.id == other.id
}
}
impl Eq for Item {}
impl Hash for Item {
fn hash<H: Hasher>(&self, state: &mut H) {
self.id.hash(state);
}
}
7 months ago
impl fmt::Display for Item {
7 months ago
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
7 months ago
writeln!(f, "{}", self.id)?;
writeln!(f, "---------")?;
writeln!(f, "{}", self.content)
}
}
impl PartialOrd for Item {
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
self.id.partial_cmp(&other.id)
7 months ago
}
}
impl Ord for Item {
fn cmp(&self, other: &Self) -> Ordering {
self.id.cmp(&other.id)
}
}
7 months ago
pub struct Storage {
7 months ago
items: HashSet::<Item>,
encoder: Encoder
7 months ago
}
7 months ago
impl Storage {
7 months ago
pub fn new(passphrase: String) -> Storage {
Storage {
items: HashSet::<Item>::new(),
7 months ago
encoder: Encoder::from(&passphrase)
7 months ago
}
}
fn get_storage_path() -> io::Result<String> {
let mps_home = env::var(ENV_MPS_HOME).map_err(|err| {
io::Error::new(io::ErrorKind::NotFound, format!("{} error: {}", ENV_MPS_HOME, err))
})?;
let result = format!("{}/{}", mps_home, PATH_STORAGE);
Ok(result)
}
7 months ago
fn get_db_path() -> io::Result<String> {
let st = Storage::get_storage_path()?;
let result = format!("{}/{}", st, PATH_DB);
Ok(result)
}
7 months ago
pub fn from_db(passphrase: String) -> io::Result<Storage> {
if !Storage::is_inited()? {
return Err(io::Error::new(
io::ErrorKind::Other,
"Storage is not initialized"
));
}
7 months ago
let encoder = Encoder::from(&passphrase);
7 months ago
let file = fs::File::open(Storage::get_db_path()?)?;
7 months ago
let reader = io::BufReader::new(file);
let mut items = HashSet::<Item>::new();
let mut id: Option<String> = None;
7 months ago
let mut lines = reader.lines();
let passtest = match lines.next() {
Some(line) => line?,
7 months ago
None => return Err(
io::Error::new(io::ErrorKind::InvalidData,
"Bad storage db format: no passphrase in the beginnning"
)),
7 months ago
};
7 months ago
if !encoder.test_encoded_passphrase(passtest)? {
return Err(io::Error::new(io::ErrorKind::InvalidData, "Wrong passphrase"));
}
7 months ago
for line in lines {
7 months ago
match line {
Ok(line) => {
if id.is_none() {
7 months ago
let line = encoder.decrypt(line)?;
id = Some(line);
} else {
7 months ago
let content = encoder.decrypt(line)?;
items.insert(Item::from(id.unwrap(), content));
id = None;
}
7 months ago
},
Err(e) => {
eprintln!("Error reading line, {}", e);
7 months ago
}
}
}
7 months ago
Ok(Storage {
items,
encoder
7 months ago
})
7 months ago
}
7 months ago
pub fn init(passphrase: String) -> io::Result<()> {
// TODO: revrite all logic
// Check if the folder exists and is a directory
//if path.exists() && path.is_dir() {
// Err(io::Error::new(io::ErrorKind::NotFound, "Folder does not exist"))
//}
//fs::File::create(&*STORAGE_PATH)?;
//let st = Storage::new(passphrase);
//st.dump()?;
//println!("Storage db created.");
//println!("Initialization complete.");
//println!("");
//println!("Now it's required to add folder `{}` under git manually.", &*STORAGE_FOLDER);
//println!("Don't worry it's going to be encrypted.");
7 months ago
Ok(())
}
7 months ago
7 months ago
pub fn check_installed() -> io::Result<()> {
7 months ago
let sp = Storage::get_storage_path()?;
let storage_path = Path::new(&sp);
// Check if the folder exists and is a directory
if !storage_path.exists() || !storage_path.is_dir() {
return Err(io::Error::new(
io::ErrorKind::NotFound,
format!("{} does not exist or not a dir", Storage::get_storage_path()?)
));
}
7 months ago
Ok(())
7 months ago
// TODO check git
7 months ago
}
pub fn is_inited() -> io::Result<bool> {
Storage::check_installed()?;
let db = Storage::get_db_path()?;
let db_path = Path::new(&db);
Ok(db_path.exists())
7 months ago
}
7 months ago
pub fn ids(&self) -> Vec<String> {
let mut result = Vec::new();
for item in self.items.iter() {
result.push(item.id.clone());
}
result.sort();
result
}
7 months ago
pub fn contains(&self, id: &String) -> bool {
let item = Item::from_empty(id.clone());
self.items.contains(&item)
}
7 months ago
pub fn get(&self, id: &String) -> &Item {
let item = Item::from_empty(id.clone());
self.items.get(&item).unwrap()
}
7 months ago
pub fn add(&mut self, item: Item) {
self.items.insert(item);
}
7 months ago
pub fn update(&mut self, item: Item) {
self.items.remove(&item);
self.items.insert(item);
}
7 months ago
pub fn dump(&self) -> io::Result<()> {
7 months ago
let mut file = fs::OpenOptions::new()
.write(true)
.append(false)
7 months ago
.open(Storage::get_db_path()?)?;
7 months ago
writeln!(file, "{}", self.encoder.get_encoded_test_passphrase()?)?;
7 months ago
for item in self.items.iter() {
7 months ago
writeln!(file, "{}", self.encoder.encrypt(&item.id)?)?;
7 months ago
let content = self.encoder.encrypt(&item.content)?;
writeln!(file, "{}", content)?;
7 months ago
}
Ok(())
}
7 months ago
}
7 months ago