use clap::{Arg, ArgMatches, Command};
use mdbook_bibtex::BibtexPreprocessor;
use mdbook_preprocessor::Preprocessor;
use mdbook_preprocessor::errors::Result;
use semver::{Version, VersionReq};
use std::io;
use std::process;
use tracing::error;
use tracing::warn;
fn make_app() -> Command {
Command::new("mdbook-bibtex")
.about("A mdbook preprocessor adding BibTeX support")
.subcommand(
Command::new("supports")
.arg(Arg::new("renderer").required(true))
.about("Check whether a renderer is supported by this preprocessor"),
)
}
fn init_logger() {
let filter = tracing_subscriber::EnvFilter::builder()
.with_env_var("MDBOOK_LOG")
.with_default_directive(tracing_subscriber::filter::LevelFilter::INFO.into())
.from_env_lossy();
let log_env = std::env::var("MDBOOK_LOG");
let silence_unless_specified = |filter: tracing_subscriber::EnvFilter, target| {
if !log_env
.as_ref()
.is_ok_and(|s| s.split(',').any(|directive| directive.starts_with(target)))
{
filter.add_directive(format!("{target}=warn").parse().unwrap())
} else {
filter
}
};
let filter = silence_unless_specified(filter, "handlebars");
let filter = silence_unless_specified(filter, "html5ever");
let with_target = log_env.is_ok();
tracing_subscriber::fmt()
.without_time()
.with_ansi(std::io::IsTerminal::is_terminal(&std::io::stderr()))
.with_writer(std::io::stderr)
.with_env_filter(filter)
.with_target(with_target)
.init();
}
fn main() {
init_logger();
let matches = make_app().get_matches();
let preprocessor = BibtexPreprocessor;
if let Some(sub_args) = matches.subcommand_matches("supports") {
handle_supports(&preprocessor, sub_args);
} else if let Err(e) = handle_preprocessing(&preprocessor) {
error!("{e:?}");
process::exit(1);
}
}
fn handle_preprocessing(pre: &dyn Preprocessor) -> Result<()> {
let (ctx, book) = mdbook_preprocessor::parse_input(io::stdin())?;
let book_version = Version::parse(&ctx.mdbook_version)?;
let version_req = VersionReq::parse(mdbook_preprocessor::MDBOOK_VERSION)?;
if !version_req.matches(&book_version) {
warn!(
"Warning: The {} plugin was built against version {} of mdbook, \
but we're being called from version {}",
pre.name(),
mdbook_preprocessor::MDBOOK_VERSION,
ctx.mdbook_version
);
}
let processed_book = pre.run(&ctx, book)?;
serde_json::to_writer(io::stdout(), &processed_book)?;
Ok(())
}
fn handle_supports(pre: &dyn Preprocessor, sub_args: &ArgMatches) -> ! {
let renderer = sub_args
.get_one::<String>("renderer")
.expect("Required argument");
let supported = pre.supports_renderer(renderer).unwrap();
if supported {
process::exit(0);
} else {
process::exit(1);
}
}