ffizer 2.7.0

ffizer is a files and folders initializer / generator. It creates or updates any kind (or part) of project from template(s)
Documentation
use crate::path_pattern::PathPattern;
use crate::Result;
use std::path::Path;
use std::path::PathBuf;
use walkdir::WalkDir;

pub const FILEEXT_HANDLEBARS: &str = ".ffizer.hbs";
pub const FILEEXT_RAW: &str = ".ffizer.raw";

#[derive(Debug, Clone, PartialEq, Eq, Default, Hash, Ord, PartialOrd)]
pub struct ChildPath {
    pub relative: PathBuf,
    pub base: PathBuf,
}

impl ChildPath {
    pub fn new<P1, P2>(base: P1, relative: P2) -> ChildPath
    where
        P1: AsRef<Path>,
        P2: AsRef<Path>,
    {
        ChildPath {
            relative: PathBuf::from(relative.as_ref()),
            base: PathBuf::from(base.as_ref()),
        }
    }
}

impl<'a> From<&'a ChildPath> for PathBuf {
    fn from(v: &ChildPath) -> Self {
        v.base.join(&v.relative)
    }
}

pub fn is_ffizer_handlebars(path: &Path) -> bool {
    path.file_name()
        .and_then(|s| s.to_str())
        .map(|str| str.contains(FILEEXT_HANDLEBARS))
        .unwrap_or(false)
}

pub fn remove_special_suffix(path: &Path) -> Result<PathBuf> {
    match path.file_name().and_then(|s| s.to_str()) {
        None => Ok(path.to_path_buf()),
        Some(v) => {
            let file_name = remove_special_suffix_on_filename(v);
            Ok(path.with_file_name(file_name))
        }
    }
}

fn remove_special_suffix_on_filename(v: &str) -> String {
    if v.contains(FILEEXT_RAW) {
        v.replacen(FILEEXT_RAW, "", 1)
    } else if v.contains(FILEEXT_HANDLEBARS) {
        v.replacen(FILEEXT_HANDLEBARS, "", 1)
    } else {
        v.to_owned()
    }
}

pub fn add_suffix<P>(path: P, suffix: &str) -> Result<PathBuf>
where
    P: AsRef<Path>,
{
    Ok(PathBuf::from(format!(
        "{}{}",
        path.as_ref().to_string_lossy(),
        suffix
    )))
}

pub fn find_childpaths<P>(base: P, ignores: &[PathPattern]) -> Vec<ChildPath>
where
    P: AsRef<Path>,
{
    let base = base.as_ref();
    WalkDir::new(base)
        .follow_links(false)
        .into_iter()
        .filter_entry(|e| {
            e.path()
                .strip_prefix(base)
                .expect("scanned child path to be under base")
                .to_str()
                .map(|s| !ignores.iter().any(|f| f.is_match(s)))
                // .map(|s| true)
                .unwrap_or(true)
        })
        .filter_map(|e| e.ok())
        .map(|entry| ChildPath {
            base: base.to_path_buf(),
            relative: entry
                .into_path()
                .strip_prefix(base)
                .expect("scanned child path to be under base")
                .to_path_buf(),
        })
        .collect::<Vec<_>>()
}

#[cfg(test)]
mod tests {
    use super::*;
    use spectral::prelude::*;

    #[test]
    fn test_is_ffizer_handlebars() {
        assert_that!(is_ffizer_handlebars(&PathBuf::from("foo.hbs"))).is_false();
        assert_that!(is_ffizer_handlebars(&PathBuf::from("foo.ffizer.hbs/bar"))).is_false();
        assert_that!(is_ffizer_handlebars(&PathBuf::from("foo_ffizer.hbs"))).is_false();
        assert_that!(is_ffizer_handlebars(&PathBuf::from("fooffizer.hbs"))).is_false();

        assert_that!(is_ffizer_handlebars(&PathBuf::from("foo.ffizer.hbs"))).is_true();
        assert_that!(is_ffizer_handlebars(&PathBuf::from("bar/foo.ffizer.hbs"))).is_true();
    }

    #[test]
    fn test_remove_special_suffix_on_filename() {
        for (input, expected) in vec![
            ("foo.hbs", "foo.hbs"),
            ("foo.json.ffizer.hbs", "foo.json"),
            ("foo.ffizer.hbs.json", "foo.json"),
            ("foo.json.ffizer.raw", "foo.json"),
            ("foo.ffizer.raw.json", "foo.json"),
            ("foo.json.ffizer.raw", "foo.json"),
            ("foo.ffizer.raw.ffizer.raw.json", "foo.ffizer.raw.json"),
            ("foo.ffizer.raw.ffizer.hbs.json", "foo.ffizer.hbs.json"),
            ("foo.json.ffizer.raw.ffizer.hbs", "foo.json.ffizer.hbs"),
        ] {
            assert_that!(remove_special_suffix_on_filename(input))
                .is_equal_to(expected.to_string());
        }
    }

    #[test]
    fn test_add_suffix() -> Result<(), Box<dyn std::error::Error>> {
        assert_that!(add_suffix(&PathBuf::from("foo.ext1"), "")?)
            .is_equal_to(&PathBuf::from("foo.ext1"));
        assert_that!(add_suffix(&PathBuf::from("foo.ext1"), ".REMOTE")?)
            .is_equal_to(&PathBuf::from("foo.ext1.REMOTE"));
        Ok(())
    }
}