nextsync-rust/src/store/object.rs
2023-06-24 16:52:02 +02:00

144 lines
3.3 KiB
Rust

use std::io::{self, Write};
use std::path::{Path, PathBuf};
use std::fs::{OpenOptions, self};
use crypto::sha1::Sha1;
use crypto::digest::Digest;
use crate::utils::{read, path};
pub mod tree;
pub mod blob;
/// Returns (line, hash, name)
///
/// # Examples
/// Input: /foo/bar
/// Result: ("tree hash(/foo/bar) bar", hash(/foo/bar), bar)
pub fn parse_path(path: &Path, is_blob: bool) -> (String, String, String) {
let file_name = path.file_name().unwrap().to_str().unwrap();
let mut hasher = Sha1::new();
hasher.input_str(path.clone().to_str().unwrap());
let hash = hasher.result_str();
let mut line = String::from(if is_blob { "blob" } else { "tree" });
line.push_str(" ");
line.push_str(&hash);
line.push_str(" ");
line.push_str(file_name);
(line, hash, String::from(file_name))
}
fn hash_obj(obj: &str) -> (String, String) {
let mut hasher = Sha1::new();
hasher.input_str(obj);
let hash = hasher.result_str();
let (dir, res) = hash.split_at(2);
(String::from(dir), String::from(res))
}
fn _object_path(obj: &str) -> PathBuf {
let mut root = match path::objects() {
Some(path) => path,
None => todo!(),
};
let (dir, res) = hash_obj(&obj);
root.push(dir);
root.push(res);
root
}
fn rm_node(path: &Path, node: &str) -> io::Result<()> {
let mut root = match path::objects() {
Some(path) => path,
None => todo!(),
};
let (dir, rest) = hash_obj(path.clone().to_str().unwrap());
root.push(dir);
root.push(rest);
read::rm_line(root, node)?;
Ok(())
}
fn add_node(path: &Path, node: &str) -> io::Result<()> {
let mut root = match path::objects() {
Some(path) => path,
None => todo!(),
};
let (dir, rest) = hash_obj(path.clone().to_str().unwrap());
root.push(dir);
if !root.exists() {
todo!();
}
root.push(rest);
let mut file = OpenOptions::new()
.read(true)
.write(true)
.append(true)
.open(root)?;
writeln!(file, "{}", node)?;
Ok(())
}
fn create_obj(name: String, content: &str) -> io::Result<()> {
let mut root = match path::objects() {
Some(path) => path,
None => todo!(),
};
let c = name.clone();
let (dir, rest) = c.split_at(2);
root.push(dir);
if !root.exists() {
fs::create_dir_all(root.clone())?;
}
root.push(rest);
let mut file = OpenOptions::new()
.create_new(true)
.write(true)
.open(root)?;
writeln!(file, "{}", content)?;
Ok(())
}
pub fn get_timestamp(path_s: String) -> Option<i64> {
let mut obj_p = match path::objects() {
Some(path) => path,
None => todo!(),
};
let (dir, res) = hash_obj(&path_s);
obj_p.push(dir);
obj_p.push(res);
match read::read_lines(obj_p) {
Ok(mut reader) => {
match reader.next() {
Some(Ok(line)) => {
let mut data = line.rsplit(' ');
if data.clone().count() >= 2 {
Some(data.next().unwrap().parse::<i64>().unwrap())
} else {
None
}
},
_ => None,
}
},
Err(err) => {
eprintln!("error reading object: {}", err);
None
},
}
}