rgit/src/base.rs

127 lines
3.7 KiB
Rust

use std::collections::HashMap;
use std::fs;
use std::io;
use std::path::Path;
#[path = "data.rs"]
mod data;
pub fn write_tree(directory: String) -> String {
let mut entries: Vec<(String, String, String)> = vec![];
let mut name;
let mut type_: String;
let mut oid: String;
let it = fs::read_dir(&directory).unwrap();
for entry in it {
let item = entry.unwrap();
let metadata = item.metadata().unwrap();
name = item.file_name();
let full = format!("{}/{}", directory, name.to_str().unwrap());
if is_ignored(&full) {
continue;
}
if metadata.is_file() {
type_ = "blob".to_owned();
oid = data::hash_object(&fs::read(&full).unwrap(), type_.clone());
println!("{} {}", oid, full);
} else if metadata.is_dir() {
type_ = "tree".to_owned();
oid = write_tree(full);
} else {
panic!("What is this?");
}
entries.push((
name.to_str().unwrap().to_owned().clone(),
oid.clone(),
type_.clone(),
));
}
entries.sort();
let mut tree = String::new();
for entry in entries {
tree.push_str(&format!("{} {} {}\n", entry.2, entry.1, entry.0));
}
return data::hash_object(&tree.into_bytes(), "tree".to_owned());
}
pub fn read_tree(oid: String) {
empty_current_directory(".").unwrap();
for (path, object_id) in get_tree(oid, "./".to_owned()).iter() {
let mut dirs = Path::new(path).ancestors();
dirs.next();
let dir = dirs.next().unwrap().to_str().unwrap();
fs::create_dir_all(dir).expect("Cannot create required dirs");
fs::write(path, data::get_object(object_id.clone(), "".to_owned()))
.expect("Cannot write required object");
}
}
fn is_ignored(path: &String) -> bool {
if path.contains(".rgit") {
true
} else {
false
}
}
fn tree_entries(oid: String) -> Vec<(String, String, String)> {
let mut entries: Vec<(String, String, String)> = vec![];
let tree_data = data::get_object(oid, "tree".to_owned());
for line in tree_data.split_terminator("\n") {
let items: Vec<&str> = line.splitn(3, " ").collect();
entries.push((
items[0].to_owned(), // _type
items[1].to_owned(), // oid
items[2].to_owned(), // name
));
}
return entries;
}
fn get_tree(oid: String, base_path: String) -> HashMap<String, String> {
let mut result = HashMap::new();
for entry in tree_entries(oid) {
// _type, oid, name
assert!(entry.2.find("/").is_none());
assert!(entry.2 != "..");
assert!(entry.2 != ".");
let path = base_path.clone() + entry.2.as_str();
if entry.0 == "blob".to_owned() {
result.insert(path.clone(), entry.1.clone());
} else if entry.0 == "tree".to_owned() {
result.extend(get_tree(entry.1, format!("{}/", path)));
} else {
panic!("Unknown tree entry: {}", entry.0);
}
}
result
}
fn empty_current_directory(dir: &str) -> io::Result<()> {
// Delete current directory, less the ignored directories
for entry in fs::read_dir(dir)? {
let entry = entry?;
let path = entry.path();
if is_ignored(&path.to_str().unwrap().to_owned()) {
continue;
}
if path.is_dir() {
empty_current_directory(path.to_str().unwrap())?;
match fs::remove_dir(&path) {
Ok(()) => (),
_ => println!("Unable to remove dir {}", path.clone().to_str().unwrap()),
};
} else {
fs::remove_file(&path)?
}
}
Ok(())
}