summaryrefslogtreecommitdiff
path: root/src/context.rs
blob: 55f3f97a1e0ec718c983f6ff44841bb11fb244ec (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
use structopt::StructOpt;
use serde::Deserialize;
use std::collections::HashMap;
use std::fs;
use std::io;
use std::path::PathBuf;

#[derive(Debug, StructOpt)]
#[structopt(name = "gallery")]
pub struct Options {
    #[structopt(long, help = "Just a thing")]
    pub check: bool,
    #[structopt(short, long, help = "Config file location")]
    pub config: Option<PathBuf>,
    #[structopt(help = "Picture location")]
    pub load: PathBuf,
    #[structopt(short, long, default_value = "png", help = "Image extension to use for converted files")]
    pub ext: String,
    #[structopt(long, short, default_value = "build", help = "Where to build site")]
    pub builddir: PathBuf,

    #[structopt(long, default_value = "1080", help = "Scaled size for image")]
    pub size_scaled: u32,
    #[structopt(long, default_value = "720", help = "Thumbnail size for image")]
    pub size_thumb: u32,
}

#[derive(Debug)]
pub enum ConfigError {
    Reading(io::Error),
    Parsing(serde_yaml::Error),
    CompilePattern(glob::PatternError),
}

#[derive(Deserialize, Debug)]
pub struct Config {
    pub imageglob: String,
    pub info: HashMap<String, String>,

    #[serde(skip)]
    pub imageglob_compiled: glob::Pattern,
}

#[derive(Debug)]
pub struct Context {
    pub options: Options,
    pub config: Config,
    pub imgdir: PathBuf,
}

impl Context {
    pub fn new_with_args() -> Result<Context, ConfigError> {
        let opts = Options::from_args();
        let config = Config::load_with_options(&opts)?;
        let imgdir = opts.builddir.join("imgs");

        // Create img dir
        if let Err(err) = fs::create_dir(&imgdir) {
            if err.kind() != io::ErrorKind::AlreadyExists {
                return Err(ConfigError::from(err));
            }
        }

        Ok(Context {
            options: opts,
            config: config,
            imgdir: imgdir,
        })
    }

    pub fn get_image_files(&self) -> Result<Vec<PathBuf>, io::Error> {
        let files = fs::read_dir(&self.options.load)?
            // Remove errored entries (SILENTLY)
            .filter(|entry| entry.is_ok())
            .map(|entry| entry.unwrap().path())
            // Filter out directories and files that do not match pattern
            .filter(|path| path.file_name()
                    .map_or(false, |file| self.config.imageglob_compiled.matches(file.to_str().unwrap()))
                    )
            .collect();

        Ok(files)
    }
}

impl Config {
    fn load_from_file(path: &PathBuf) -> Result<Config, ConfigError> {
        let content = fs::read_to_string(path)?;
        let mut config: Config = serde_yaml::from_str(&content)?;

        config.imageglob_compiled = glob::Pattern::new(&config.imageglob)?;

        Ok(config)
    }

    pub fn load_with_options(opts: &Options) -> Result<Config, ConfigError> {
        let config_path = match &opts.config {
            Some(path) => path.to_path_buf(),
            None => opts.load.join("imginfo.yml"),
        };

        Config::load_from_file(&config_path)
    }


}

impl From<io::Error> for ConfigError {
    fn from(error: io::Error) -> Self {
        ConfigError::Reading(error)
    }
}

impl From<serde_yaml::Error> for ConfigError {
    fn from(error: serde_yaml::Error) -> Self {
        ConfigError::Parsing(error)
    }
}

impl From<glob::PatternError> for ConfigError {
    fn from(error: glob::PatternError) -> Self {
        ConfigError::CompilePattern(error)
    }
}