2020-11-28 15:14:50 +03:00
|
|
|
use eyre::{eyre, Result};
|
|
|
|
use gumdrop::Options;
|
2020-12-21 15:01:08 +03:00
|
|
|
use obsidian_export::{ExportError, Exporter, FrontmatterStrategy};
|
2020-11-28 15:14:50 +03:00
|
|
|
use std::path::PathBuf;
|
|
|
|
|
|
|
|
#[derive(Debug, Options)]
|
|
|
|
struct Opts {
|
|
|
|
#[options(help = "Display program help")]
|
|
|
|
help: bool,
|
|
|
|
|
|
|
|
#[options(help = "Source file containing reference", free, required)]
|
|
|
|
source: Option<PathBuf>,
|
|
|
|
|
|
|
|
#[options(help = "Destination file being linked to", free, required)]
|
|
|
|
destination: Option<PathBuf>,
|
|
|
|
|
|
|
|
#[options(
|
|
|
|
help = "Frontmatter strategy (one of: always, never, auto)",
|
|
|
|
no_short,
|
|
|
|
long = "frontmatter",
|
|
|
|
parse(try_from_str = "frontmatter_strategy_from_str"),
|
|
|
|
default = "auto"
|
|
|
|
)]
|
|
|
|
frontmatter_strategy: FrontmatterStrategy,
|
|
|
|
}
|
|
|
|
|
|
|
|
fn frontmatter_strategy_from_str(input: &str) -> Result<FrontmatterStrategy> {
|
|
|
|
match input {
|
|
|
|
"auto" => Ok(FrontmatterStrategy::Auto),
|
|
|
|
"always" => Ok(FrontmatterStrategy::Always),
|
|
|
|
"never" => Ok(FrontmatterStrategy::Never),
|
|
|
|
_ => Err(eyre!("must be one of: always, never, auto")),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn main() -> Result<()> {
|
|
|
|
let args = Opts::parse_args_default_or_exit();
|
|
|
|
let source = args.source.unwrap();
|
|
|
|
let destination = args.destination.unwrap();
|
|
|
|
|
|
|
|
let mut exporter = Exporter::new(source, destination);
|
|
|
|
exporter.frontmatter_strategy(args.frontmatter_strategy);
|
|
|
|
// TODO: Pass in configurable walk_options here: exporter.walk_options(..);
|
|
|
|
// TODO: This should allow settings for ignore_hidden and honor_gitignore.
|
2020-12-21 15:01:08 +03:00
|
|
|
|
|
|
|
if let Err(err) = exporter.run() {
|
|
|
|
match err {
|
|
|
|
ExportError::FileExportError {
|
|
|
|
ref path,
|
|
|
|
ref source,
|
|
|
|
} => match &**source {
|
|
|
|
// An arguably better way of enhancing error reports would be to construct a custom
|
|
|
|
// `eyre::EyreHandler`, but that would require a fair amount of boilerplate and
|
|
|
|
// reimplementation of basic reporting.
|
|
|
|
ExportError::RecursionLimitExceeded { file_tree } => {
|
|
|
|
eprintln!(
|
|
|
|
"Error: {:?}",
|
|
|
|
eyre!(
|
|
|
|
"'{}' exceeds the maximum nesting limit of embeds",
|
|
|
|
path.display()
|
|
|
|
)
|
|
|
|
);
|
|
|
|
eprintln!("\nFile tree:");
|
|
|
|
for (idx, path) in file_tree.iter().enumerate() {
|
|
|
|
eprintln!("{}-> {}", " ".repeat(idx), path.display());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => eprintln!("Error: {:?}", eyre!(err)),
|
|
|
|
},
|
|
|
|
_ => eprintln!("Error: {:?}", eyre!(err)),
|
|
|
|
};
|
|
|
|
std::process::exit(1);
|
|
|
|
};
|
2020-11-28 15:14:50 +03:00
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|