summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--Cargo.lock2
-rw-r--r--Cargo.toml3
-rw-r--r--skaldpress.18
-rw-r--r--src/skaldpress/main.rs89
-rw-r--r--src/skaldpress/metadata_parser.rs2
-rw-r--r--src/skaldpress/parseopts.rs82
6 files changed, 158 insertions, 28 deletions
diff --git a/Cargo.lock b/Cargo.lock
index 39c194f..186cdbc 100644
--- a/Cargo.lock
+++ b/Cargo.lock
@@ -236,7 +236,7 @@ checksum = "0fda2ff0d084019ba4d7c6f371c95d8fd75ce3524c3cb8fb653a3023f6323e64"
[[package]]
name = "skaldpress"
-version = "0.1.0"
+version = "0.1.1"
dependencies = [
"chrono",
"markdown",
diff --git a/Cargo.toml b/Cargo.toml
index 69409c8..3f277a1 100644
--- a/Cargo.toml
+++ b/Cargo.toml
@@ -1,6 +1,6 @@
[package]
name = "skaldpress"
-version = "0.1.0"
+version = "0.1.1"
edition = "2021"
default-run = "skaldpress"
@@ -9,6 +9,7 @@ default = ["time", "markdown", "deadlinks"]
time = ["dep:chrono"]
markdown = ["dep:markdown"]
deadlinks = ["dep:minreq"]
+webring = []
[dependencies]
markdown = { version = "1.0.0-alpha.20", optional = true }
diff --git a/skaldpress.1 b/skaldpress.1
index eec8dd3..5ed2046 100644
--- a/skaldpress.1
+++ b/skaldpress.1
@@ -71,6 +71,14 @@ Specifies the directory where templates are stored, defaults to \fItemplates\fR.
Comma-separated list of files to compile, if not specified, all files in \fB--source\fR
will be compiled.
+.IP "\fB-e, --exclude\fR \fIfilter\fR
+Comma-separated list of files to exclude, if not specified, all files in \fB--source\fR
+will be compiled.
+
+.IP "\fB-m, --metadata\fR \fIkey\fR=\fIvalue\fR
+Metadata to add to compiled file, this can e.g. set a template for all files if you don't want to manually add YAML-blocks.
+Specify multiple times, to set multiple fields.
+
.SH EXAMPLES
To run skaldpress on a simple project, simply call it with no arguments
diff --git a/src/skaldpress/main.rs b/src/skaldpress/main.rs
index 574bc6a..9ab988e 100644
--- a/src/skaldpress/main.rs
+++ b/src/skaldpress/main.rs
@@ -308,25 +308,40 @@ fn compile_file(file_path: &Path, opts: &Opts) -> Result<CompiledFile, Skaldpres
file_path.to_str().unwrap_or("unknown file").to_string(),
)
})?;
- let (map, file_content) = match extract_parse_yaml_metadata(&file_content) {
+ let (mut map, file_content) = match extract_parse_yaml_metadata(&file_content) {
Some((map, file_content)) => (map, file_content),
None => (HashMap::new(), file_content.as_str()),
};
+ map.extend(opts.metadata.clone());
- if let Some(skip_smp) = &map.get("skip_smp") {
- if let YamlValue::Scalar(skip_smp) = skip_smp {
- if skip_smp.to_lowercase() == "true" {
- return Ok(CompiledFile {
- content: file_content.to_string(),
- metadata: map,
- extension: String::from(extension),
- source_path: String::from(file_path.to_str().unwrap_or("")),
- needs_recompilation: false,
- stored_smp_state: HashMap::new(),
- });
+ let mut skip_smp = false;
+ if let Some(_skip_smp) = &map.get("skip_smp") {
+ if let YamlValue::Scalar(_skip_smp) = _skip_smp {
+ if _skip_smp.to_lowercase() == "true" {
+ skip_smp = true;
}
}
}
+ if opts.compilefilter.len() != 0 {
+ let e: Vec<String> = Vec::new();
+ if !file_filtered(
+ &file_path.to_str().unwrap_or("").to_string(),
+ &opts.compilefilter,
+ &e,
+ ) {
+ skip_smp = true;
+ }
+ }
+ if skip_smp {
+ return Ok(CompiledFile {
+ content: file_content.to_string(),
+ metadata: map,
+ extension: String::from(extension),
+ source_path: String::from(file_path.to_str().unwrap_or("")),
+ needs_recompilation: false,
+ stored_smp_state: HashMap::new(),
+ });
+ }
let mut stored_smp_state = None;
if let Some(cfile_i) = cached_file_id_by_path(&file_path.to_str().unwrap_or("").to_string()) {
@@ -481,6 +496,36 @@ fn compile_file_and_write(
Ok(())
}
+fn file_pat_match(file: &String, pat: &str) -> bool {
+ if file == pat {
+ return true;
+ }
+ if pat.starts_with("*") && file.ends_with(pat.trim_start_matches("*")) {
+ return true;
+ }
+ if pat.ends_with("*") && file.starts_with(pat.trim_end_matches("*")) {
+ return true;
+ }
+ false
+}
+
+fn file_filtered(file: &String, filters: &Vec<String>, exclude: &Vec<String>) -> bool {
+ for filter in exclude {
+ if file_pat_match(file, filter) {
+ return true;
+ }
+ }
+ if filters.len() == 0 {
+ return false;
+ }
+ for filter in filters {
+ if file_pat_match(file, filter) {
+ return true;
+ }
+ }
+ true
+}
+
fn compile_files_in_directory(directory: &Path, opts: &Opts) -> Result<(), SkaldpressError> {
for entry in fs::read_dir(directory).map_err(|e| {
SkaldpressError::DirectoryReadError(
@@ -514,11 +559,15 @@ fn compile_files_in_directory(directory: &Path, opts: &Opts) -> Result<(), Skald
}
}
+ //let should_compile = (opts.first_run || needs_recompilation)
+ // && ((opts.filter.len() == 0)
+ // || !file_filtered(&path.as_path().to_str().unwrap_or("").to_string(), &opts.filter));
let should_compile = (opts.first_run || needs_recompilation)
- && ((opts.filter.len() == 0)
- || opts
- .filter
- .contains(&path.as_path().to_str().unwrap_or("").to_string()));
+ && !file_filtered(
+ &path.as_path().to_str().unwrap_or("").to_string(),
+ &opts.filter,
+ &opts.exclude,
+ );
if metadata.is_file() && should_compile {
println!("< Compiling {:#?}", path.as_path());
if let Err(e) = compile_file_and_write(path.as_path(), opts) {
@@ -573,6 +622,14 @@ fn copy_files_in_directory(directory: &Path, opts: &Opts) -> Result<(), Skaldpre
continue;
}
};
+ let e: Vec<String> = Vec::new();
+ if file_filtered(
+ &path.to_str().unwrap_or("").to_string(),
+ &e,
+ &opts.static_exclude,
+ ) {
+ continue;
+ }
let dest_file_path = Path::new(&opts.build_dir).join(real_path);
println!(
"< Copying {:#?} -> {:#?}",
diff --git a/src/skaldpress/metadata_parser.rs b/src/skaldpress/metadata_parser.rs
index 66a96b6..e73c953 100644
--- a/src/skaldpress/metadata_parser.rs
+++ b/src/skaldpress/metadata_parser.rs
@@ -63,7 +63,7 @@ impl Ord for YamlValue {
}
}
-fn str_to_yaml_value(in_str: &str) -> YamlValue {
+pub fn str_to_yaml_value(in_str: &str) -> YamlValue {
let in_str = in_str.trim();
if let Ok(int) = in_str.parse::<i64>() {
return YamlValue::Integer(int);
diff --git a/src/skaldpress/parseopts.rs b/src/skaldpress/parseopts.rs
index 163b387..ec32bfd 100644
--- a/src/skaldpress/parseopts.rs
+++ b/src/skaldpress/parseopts.rs
@@ -1,3 +1,7 @@
+use crate::skaldpress::metadata_parser::str_to_yaml_value;
+use crate::skaldpress::metadata_parser::YamlValue;
+use std::collections::HashMap;
+
/// Struct containing command line options
pub struct Opts {
/// Directory where templates are, will be relative to CWD
@@ -10,6 +14,14 @@ pub struct Opts {
pub build_dir: String,
/// Filters, limit to compiling specified files
pub filter: Vec<String>,
+ /// Excludes, limit to compiling specified files
+ pub exclude: Vec<String>,
+ /// Static excludes, limit to copying specified files
+ pub static_exclude: Vec<String>,
+ /// Filters, limit to compiling specified files
+ pub compilefilter: Vec<String>,
+ /// Metadata that will be added to compiled files, useful for global options
+ pub metadata: HashMap<String, YamlValue>,
pub first_run: bool,
}
@@ -27,6 +39,14 @@ pub struct OptsBuilder {
pub build_dir: String,
/// Filters, limit to compiling specified files
pub filter: Vec<String>,
+ /// Excludes, limit to compiling specified files
+ pub exclude: Vec<String>,
+ /// Static excludes, limit to copying specified files
+ pub static_exclude: Vec<String>,
+ /// Filters, limit to compiling specified files
+ pub compilefilter: Vec<String>,
+ /// Metadata that will be added to compiled files, useful for global options
+ pub metadata: HashMap<String, YamlValue>,
}
impl OptsBuilder {
@@ -37,6 +57,10 @@ impl OptsBuilder {
static_dir: self.static_dir,
build_dir: self.build_dir,
filter: self.filter,
+ exclude: self.exclude,
+ static_exclude: self.static_exclude,
+ compilefilter: self.compilefilter,
+ metadata: self.metadata,
first_run: true,
}
}
@@ -47,15 +71,20 @@ impl OptsBuilder {
/// the program exited.
macro_rules! parseopts_panic {
($progname:expr) => {
+ println!("Skaldpress {}", env!("CARGO_PKG_VERSION"));
println!("Usage: {} [OPTIONS]\n", $progname);
println!("OPTIONS:");
- println!(" -h, --help Show this help text");
- println!(" See also 'man skaldpress'");
- println!(" -i, --input DIR Location of content to compile");
- println!(" -s, --static DIR Location of static content to copy");
- println!(" -t, --templates DIR Location of templates");
- println!(" -o, --out DIR Location to write compiled data, THIS WILL BE DELETED AND RECREATED!");
- println!(" -f, --filter FILES Only run on files, comma separated string");
+ println!(" -h, --help Show this help text");
+ println!(" See also 'man skaldpress'");
+ println!(" -i, --input DIR Location of content to compile");
+ println!(" -s, --static DIR Location of static content to copy");
+ println!(" -t, --templates DIR Location of templates");
+ println!(" -o, --out DIR Location to write compiled data, THIS WILL BE DELETED AND RECREATED!");
+ println!(" -f, --filter FILES Only run on files, comma separated string");
+ println!(" -c, --compilefilter FILES Only compile files, comma separated string, non-matching will still be copied");
+ println!(" -e, --exclude FILES Exclude files, comma separated string");
+ println!(" -x, --xclude FILES Exclude files from static, comma separated string");
+ println!(" -m, --metadata KEY=VAL Extra metadata to set for every compiled file");
std::process::exit(1);
};
}
@@ -106,6 +135,36 @@ pub fn parseopt(opts: &mut OptsBuilder, arg: &str, value: Option<String>, progna
};
opts.filter = filter.split(",").map(|x| x.to_string()).collect();
}
+ "e" | "exclude" => {
+ let Some(exclude) = value else {
+ println!("Missing value for {}\n", arg);
+ parseopts_panic!(progname);
+ };
+ opts.exclude = exclude.split(",").map(|x| x.to_string()).collect();
+ }
+ "x" | "xclude" => {
+ let Some(exclude) = value else {
+ println!("Missing value for {}\n", arg);
+ parseopts_panic!(progname);
+ };
+ opts.static_exclude = exclude.split(",").map(|x| x.to_string()).collect();
+ }
+ "c" | "compilefilter" => {
+ let Some(compilefilter) = value else {
+ println!("Missing value for {}\n", arg);
+ parseopts_panic!(progname);
+ };
+ opts.exclude = compilefilter.split(",").map(|x| x.to_string()).collect();
+ }
+ "m" | "metadata" => {
+ let Some(meta) = value else {
+ println!("Missing value for {}\n", arg);
+ parseopts_panic!(progname);
+ };
+ let parts: Vec<&str> = meta.split("=").collect();
+ opts.metadata
+ .insert(String::from(parts[0]), str_to_yaml_value(parts[1]));
+ }
"help" => {
parseopts_panic!(progname);
}
@@ -128,6 +187,10 @@ pub fn parseopts() -> OptsBuilder {
content_dir: String::from("content/"),
static_dir: String::from("static/"),
filter: Vec::new(),
+ exclude: Vec::new(),
+ static_exclude: Vec::new(),
+ compilefilter: Vec::new(),
+ metadata: HashMap::new(),
};
let mut it = std::env::args();
@@ -149,7 +212,8 @@ pub fn parseopts() -> OptsBuilder {
} else {
arg_name = arg.clone();
match arg_name.as_str() {
- "out" | "output" | "input" | "static" | "templates" | "filter" => {
+ "out" | "output" | "input" | "static" | "templates" | "filter" | "exclude"
+ | "compilefilter" | "xclude" | "metadata" => {
arg_value = it.next();
}
_ => (),
@@ -160,7 +224,7 @@ pub fn parseopts() -> OptsBuilder {
arg.remove(0);
for arg_name in arg.chars() {
match arg_name {
- 'o' | 'i' | 's' | 't' | 'f' => {
+ 'o' | 'i' | 's' | 't' | 'f' | 'e' | 'm' | 'c' | 'x' => {
parseopt(&mut opts, &arg_name.to_string(), it.next(), &progname);
}
_ => {