2020-11-28 15:14:50 +03:00
|
|
|
use eyre::{eyre, Result};
|
|
|
|
use gumdrop::Options;
|
2021-01-05 02:01:28 +03:00
|
|
|
use obsidian_export::{ExportError, Exporter, FrontmatterStrategy, WalkOptions};
|
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,
|
2021-01-05 02:01:28 +03:00
|
|
|
|
|
|
|
#[options(
|
|
|
|
no_short,
|
|
|
|
help = "Read ignore patterns from files with this name",
|
|
|
|
default = ".export-ignore"
|
|
|
|
)]
|
|
|
|
ignore_file: String,
|
|
|
|
|
|
|
|
#[options(no_short, help = "Export hidden files", default = "false")]
|
|
|
|
hidden: bool,
|
|
|
|
|
|
|
|
#[options(no_short, help = "Disable git integration", default = "false")]
|
|
|
|
no_git: bool,
|
2021-01-05 17:37:37 +03:00
|
|
|
|
|
|
|
#[options(no_short, help = "Don't process embeds recursively", default = "false")]
|
|
|
|
no_recursive_embeds: bool,
|
2020-11-28 15:14:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
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")),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-12 15:27:04 +03:00
|
|
|
fn main() {
|
2020-11-28 15:14:50 +03:00
|
|
|
let args = Opts::parse_args_default_or_exit();
|
|
|
|
let source = args.source.unwrap();
|
|
|
|
let destination = args.destination.unwrap();
|
|
|
|
|
2021-02-12 15:27:04 +03:00
|
|
|
let walk_options = WalkOptions {
|
|
|
|
ignore_filename: &args.ignore_file,
|
|
|
|
ignore_hidden: !args.hidden,
|
|
|
|
honor_gitignore: !args.no_git,
|
|
|
|
..Default::default()
|
|
|
|
};
|
2021-01-05 02:01:28 +03:00
|
|
|
|
2020-11-28 15:14:50 +03:00
|
|
|
let mut exporter = Exporter::new(source, destination);
|
|
|
|
exporter.frontmatter_strategy(args.frontmatter_strategy);
|
2021-01-05 17:37:37 +03:00
|
|
|
exporter.process_embeds_recursively(!args.no_recursive_embeds);
|
2021-01-05 02:01:28 +03:00
|
|
|
exporter.walk_options(walk_options);
|
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() {
|
2021-01-05 17:37:37 +03:00
|
|
|
eprintln!(" {}-> {}", " ".repeat(idx), path.display());
|
2020-12-21 15:01:08 +03:00
|
|
|
}
|
2021-01-05 17:37:37 +03:00
|
|
|
eprintln!("\nHint: Ensure notes are non-recursive, or specify --no-recursive-embeds to break cycles")
|
2020-12-21 15:01:08 +03:00
|
|
|
}
|
|
|
|
_ => eprintln!("Error: {:?}", eyre!(err)),
|
|
|
|
},
|
|
|
|
_ => eprintln!("Error: {:?}", eyre!(err)),
|
|
|
|
};
|
|
|
|
std::process::exit(1);
|
|
|
|
};
|
2020-11-28 15:14:50 +03:00
|
|
|
}
|