summaryrefslogtreecommitdiffstats
path: root/src/tools/rustbook
diff options
context:
space:
mode:
Diffstat (limited to 'src/tools/rustbook')
-rw-r--r--src/tools/rustbook/Cargo.toml14
-rw-r--r--src/tools/rustbook/src/main.rs102
2 files changed, 116 insertions, 0 deletions
diff --git a/src/tools/rustbook/Cargo.toml b/src/tools/rustbook/Cargo.toml
new file mode 100644
index 000000000..33c051804
--- /dev/null
+++ b/src/tools/rustbook/Cargo.toml
@@ -0,0 +1,14 @@
+[package]
+name = "rustbook"
+version = "0.1.0"
+license = "MIT OR Apache-2.0"
+edition = "2021"
+
+[dependencies]
+clap = "3.1.1"
+env_logger = "0.7.1"
+
+[dependencies.mdbook]
+version = "0.4.21"
+default-features = false
+features = ["search"]
diff --git a/src/tools/rustbook/src/main.rs b/src/tools/rustbook/src/main.rs
new file mode 100644
index 000000000..3c7dc0183
--- /dev/null
+++ b/src/tools/rustbook/src/main.rs
@@ -0,0 +1,102 @@
+use clap::crate_version;
+
+use std::env;
+use std::path::{Path, PathBuf};
+
+use clap::{arg, ArgMatches, Command};
+
+use mdbook::errors::Result as Result3;
+use mdbook::MDBook;
+
+fn main() {
+ let crate_version = format!("v{}", crate_version!());
+ env_logger::Builder::from_env(env_logger::Env::default().default_filter_or("warn")).init();
+ let d_arg = arg!(-d --"dest-dir" <DEST_DIR>
+"The output directory for your book\n(Defaults to ./book when omitted)")
+ .required(false);
+ let dir_arg = arg!([dir]
+"A directory for your book\n(Defaults to Current Directory when omitted)");
+
+ let matches = Command::new("rustbook")
+ .about("Build a book with mdBook")
+ .author("Steve Klabnik <steve@steveklabnik.com>")
+ .version(&*crate_version)
+ .subcommand_required(true)
+ .arg_required_else_help(true)
+ .subcommand(
+ Command::new("build")
+ .about("Build the book from the markdown files")
+ .arg(d_arg)
+ .arg(&dir_arg),
+ )
+ .subcommand(
+ Command::new("test")
+ .about("Tests that a book's Rust code samples compile")
+ .arg(dir_arg),
+ )
+ .get_matches();
+
+ // Check which subcomamnd the user ran...
+ match matches.subcommand() {
+ Some(("build", sub_matches)) => {
+ if let Err(e) = build(sub_matches) {
+ handle_error(e);
+ }
+ }
+ Some(("test", sub_matches)) => {
+ if let Err(e) = test(sub_matches) {
+ handle_error(e);
+ }
+ }
+ _ => unreachable!(),
+ };
+}
+
+// Build command implementation
+pub fn build(args: &ArgMatches) -> Result3<()> {
+ let book_dir = get_book_dir(args);
+ let mut book = load_book(&book_dir)?;
+
+ // Set this to allow us to catch bugs in advance.
+ book.config.build.create_missing = false;
+
+ if let Some(dest_dir) = args.value_of("dest-dir") {
+ book.config.build.build_dir = PathBuf::from(dest_dir);
+ }
+
+ book.build()?;
+
+ Ok(())
+}
+
+fn test(args: &ArgMatches) -> Result3<()> {
+ let book_dir = get_book_dir(args);
+ let mut book = load_book(&book_dir)?;
+ book.test(vec![])
+}
+
+fn get_book_dir(args: &ArgMatches) -> PathBuf {
+ if let Some(dir) = args.value_of("dir") {
+ // Check if path is relative from current dir, or absolute...
+ let p = Path::new(dir);
+ if p.is_relative() { env::current_dir().unwrap().join(dir) } else { p.to_path_buf() }
+ } else {
+ env::current_dir().unwrap()
+ }
+}
+
+fn load_book(book_dir: &Path) -> Result3<MDBook> {
+ let mut book = MDBook::load(book_dir)?;
+ book.config.set("output.html.input-404", "").unwrap();
+ Ok(book)
+}
+
+fn handle_error(error: mdbook::errors::Error) -> ! {
+ eprintln!("Error: {}", error);
+
+ for cause in error.chain().skip(1) {
+ eprintln!("\tCaused By: {}", cause);
+ }
+
+ ::std::process::exit(101);
+}