2015-08-01 00:59:05 -04:00
|
|
|
#[macro_use]
|
|
|
|
|
extern crate clap;
|
2016-08-14 14:55:10 +01:00
|
|
|
extern crate env_logger;
|
2017-10-03 17:25:23 +05:45
|
|
|
extern crate log;
|
|
|
|
|
extern crate mdbook;
|
2017-01-01 09:42:47 -08:00
|
|
|
extern crate open;
|
2015-11-09 14:31:00 +01:00
|
|
|
|
2015-07-06 21:12:24 +02:00
|
|
|
use std::env;
|
2017-01-01 09:42:47 -08:00
|
|
|
use std::ffi::OsStr;
|
2015-08-01 00:59:05 -04:00
|
|
|
use std::io::{self, Write};
|
|
|
|
|
use std::path::{Path, PathBuf};
|
2017-10-03 17:25:23 +05:45
|
|
|
use clap::{App, AppSettings, ArgMatches};
|
|
|
|
|
use log::{LogLevelFilter, LogRecord};
|
2017-06-28 23:37:03 +02:00
|
|
|
use env_logger::LogBuilder;
|
2015-07-06 21:12:24 +02:00
|
|
|
|
2017-06-26 00:53:58 +02:00
|
|
|
pub mod build;
|
2017-06-26 01:02:32 +02:00
|
|
|
pub mod init;
|
2017-06-26 01:24:33 +02:00
|
|
|
pub mod test;
|
2017-06-25 21:41:23 +02:00
|
|
|
#[cfg(feature = "serve")]
|
|
|
|
|
pub mod serve;
|
2015-09-27 14:38:37 +02:00
|
|
|
#[cfg(feature = "watch")]
|
2017-06-25 23:44:28 +02:00
|
|
|
pub mod watch;
|
2015-11-09 14:31:00 +01:00
|
|
|
|
2015-07-06 21:12:24 +02:00
|
|
|
const NAME: &'static str = "mdbook";
|
|
|
|
|
|
|
|
|
|
fn main() {
|
2017-06-28 23:37:03 +02:00
|
|
|
init_logger();
|
2016-08-14 14:55:10 +01:00
|
|
|
|
2015-08-01 00:59:05 -04:00
|
|
|
// Create a list of valid arguments and sub-commands
|
2017-06-26 00:31:42 +02:00
|
|
|
let app = App::new(NAME)
|
2017-06-26 01:24:33 +02:00
|
|
|
.about("Create a book in form of a static website from markdown files")
|
|
|
|
|
.author("Mathieu David <mathieudavid@mathieudavid.org>")
|
|
|
|
|
// Get the version from our Cargo.toml using clap's crate_version!() macro
|
|
|
|
|
.version(concat!("v",crate_version!()))
|
|
|
|
|
.setting(AppSettings::SubcommandRequired)
|
2017-10-03 17:25:23 +05:45
|
|
|
.after_help("For more information about a specific command, \
|
|
|
|
|
try `mdbook <command> --help`\n\
|
|
|
|
|
Source code for mdbook available \
|
|
|
|
|
at: https://github.com/azerupi/mdBook")
|
2017-06-26 01:24:33 +02:00
|
|
|
.subcommand(init::make_subcommand())
|
|
|
|
|
.subcommand(build::make_subcommand())
|
|
|
|
|
.subcommand(test::make_subcommand());
|
2017-06-26 00:31:42 +02:00
|
|
|
|
|
|
|
|
#[cfg(feature = "watch")]
|
|
|
|
|
let app = app.subcommand(watch::make_subcommand());
|
|
|
|
|
#[cfg(feature = "serve")]
|
|
|
|
|
let app = app.subcommand(serve::make_subcommand());
|
2015-08-01 00:59:05 -04:00
|
|
|
|
|
|
|
|
// Check which subcomamnd the user ran...
|
2017-06-26 00:31:42 +02:00
|
|
|
let res = match app.get_matches().subcommand() {
|
2017-06-27 07:59:50 +02:00
|
|
|
("init", Some(sub_matches)) => init::execute(sub_matches),
|
|
|
|
|
("build", Some(sub_matches)) => build::execute(sub_matches),
|
2015-09-27 14:38:37 +02:00
|
|
|
#[cfg(feature = "watch")]
|
2017-06-27 07:59:50 +02:00
|
|
|
("watch", Some(sub_matches)) => watch::execute(sub_matches),
|
2016-04-02 04:46:05 +02:00
|
|
|
#[cfg(feature = "serve")]
|
2017-06-27 07:59:50 +02:00
|
|
|
("serve", Some(sub_matches)) => serve::execute(sub_matches),
|
|
|
|
|
("test", Some(sub_matches)) => test::execute(sub_matches),
|
2016-03-17 22:31:28 +01:00
|
|
|
(_, _) => unreachable!(),
|
2015-07-06 21:12:24 +02:00
|
|
|
};
|
|
|
|
|
|
2015-08-01 00:59:05 -04:00
|
|
|
if let Err(e) = res {
|
2015-08-11 16:13:41 +02:00
|
|
|
writeln!(&mut io::stderr(), "An error occured:\n{}", e).ok();
|
2016-08-06 14:54:07 -04:00
|
|
|
::std::process::exit(101);
|
2015-07-06 21:12:24 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-06-28 23:37:03 +02:00
|
|
|
fn init_logger() {
|
|
|
|
|
let format = |record: &LogRecord| {
|
|
|
|
|
let module_path = record.location().module_path();
|
|
|
|
|
format!("{}:{}: {}", record.level(), module_path, record.args())
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let mut builder = LogBuilder::new();
|
|
|
|
|
builder.format(format).filter(None, LogLevelFilter::Info);
|
|
|
|
|
|
|
|
|
|
if let Ok(var) = env::var("RUST_LOG") {
|
2017-10-03 17:25:23 +05:45
|
|
|
builder.parse(&var);
|
2017-06-28 23:37:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
builder.init().unwrap();
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-01 00:59:05 -04:00
|
|
|
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() {
|
2016-03-17 22:31:28 +01:00
|
|
|
env::current_dir().unwrap().join(dir)
|
2015-08-01 00:59:05 -04:00
|
|
|
} else {
|
2016-03-17 22:31:28 +01:00
|
|
|
p.to_path_buf()
|
2015-08-01 00:59:05 -04:00
|
|
|
}
|
2015-07-18 00:04:20 +02:00
|
|
|
} else {
|
2015-08-01 00:59:05 -04:00
|
|
|
env::current_dir().unwrap()
|
2015-07-06 21:12:24 +02:00
|
|
|
}
|
|
|
|
|
}
|
2016-04-02 04:46:05 +02:00
|
|
|
|
2017-01-01 09:42:47 -08:00
|
|
|
fn open<P: AsRef<OsStr>>(path: P) {
|
|
|
|
|
if let Err(e) = open::that(path) {
|
|
|
|
|
println!("Error opening web browser: {}", e);
|
|
|
|
|
}
|
|
|
|
|
}
|