use fs::{remove_dir_all, rename};
use std::ffi::OsStr;
use std::fs;
use std::path::Path;
use diagnostic_quick::{QError, QResult};
use diagnostic_quick::error_3rd::WalkDir;
use crate::FlattenFlies;
impl FlattenFlies {
pub fn run<P>(&self, input: P) -> QResult where P: AsRef<Path> {
let path = input.as_ref();
match self.try_run(path) {
Ok(_) => {}
Err(e) => {
Err(e)?
}
}
Ok(())
}
fn try_run(&self, path: &Path) -> QResult {
let target = self.output.as_path();
for entry in WalkDir::new(path) {
let entry = entry?;
if entry.path().is_file() {
let source = entry.path();
let target = target.join(source.file_name().unwrap());
self.safe_rename(source, &target)?;
}
}
for entry in WalkDir::new(path) {
self.debug_remove(entry?.path())?;
}
Ok(())
}
fn safe_rename(&self, source: &Path, target: &Path) -> QResult {
if self.overwrite {
return self.debug_rename(source, target);
}
if !target.exists() {
self.debug_rename(source, target)?;
}
let file_name = match target.file_name().and_then(OsStr::to_str) {
None => {
Err(QError::runtime_error("target file name is not valid"))?
}
Some(s) => {
s
}
};
let mut new_id = 1;
while new_id < 65535 {
let new_name = format!("{} ({})", file_name, new_id);
let new_target = target.with_file_name(new_name);
if !new_target.exists() {
return self.debug_rename(source, &new_target);
}
new_id += 1;
}
Ok(())
}
fn debug_rename(&self, source: &Path, target: &Path) -> QResult {
println!("[mv] {} \n -> {}", source.display(), target.display());
if self.execute {
rename(source, target)?;
}
Ok(())
}
fn debug_remove(&self, path: &Path) -> QResult {
if self.delete_empty && is_empty_directory(path) {
println!("[rm] remove {}", path.display());
if self.execute {
remove_dir_all(path)?;
}
}
Ok(())
}
}
pub fn is_empty_directory(path: &Path) -> bool {
match path.read_dir() {
Ok(mut o) => {
o.next().is_none()
}
Err(e) => {
println!("error: {}", e);
false
}
}
}