2018-07-24 01:45:01 +08:00
|
|
|
use clap::{App, ArgMatches, SubCommand};
|
|
|
|
use get_book_dir;
|
|
|
|
use mdbook::config;
|
|
|
|
use mdbook::errors::Result;
|
|
|
|
use mdbook::MDBook;
|
2017-06-26 07:24:33 +08:00
|
|
|
use std::io;
|
|
|
|
use std::io::Write;
|
2018-03-26 22:37:11 +08:00
|
|
|
use std::process::Command;
|
2017-06-26 07:02:32 +08:00
|
|
|
|
2017-06-27 13:59:50 +08:00
|
|
|
// Create clap subcommand arguments
|
|
|
|
pub fn make_subcommand<'a, 'b>() -> App<'a, 'b> {
|
|
|
|
SubCommand::with_name("init")
|
2018-08-03 04:48:22 +08:00
|
|
|
.about("Creates the boilerplate structure and files for a new book")
|
2017-06-27 13:59:50 +08:00
|
|
|
// the {n} denotes a newline which will properly aligned in all help messages
|
2018-12-04 07:11:41 +08:00
|
|
|
.arg_from_usage(
|
|
|
|
"[dir] 'Directory to create the book in{n}\
|
|
|
|
(Defaults to the Current Directory when omitted)'",
|
|
|
|
).arg_from_usage("--theme 'Copies the default theme into your source folder'")
|
2018-08-03 04:48:22 +08:00
|
|
|
.arg_from_usage("--force 'Skips confirmation prompts'")
|
2017-06-27 13:59:50 +08:00
|
|
|
}
|
|
|
|
|
2017-06-26 07:02:32 +08:00
|
|
|
// Init command implementation
|
2017-06-27 13:59:50 +08:00
|
|
|
pub fn execute(args: &ArgMatches) -> Result<()> {
|
2017-06-26 07:02:32 +08:00
|
|
|
let book_dir = get_book_dir(args);
|
2017-11-18 20:41:04 +08:00
|
|
|
let mut builder = MDBook::init(&book_dir);
|
2018-03-14 23:27:56 +08:00
|
|
|
let mut config = config::Config::default();
|
2017-06-26 07:02:32 +08:00
|
|
|
|
|
|
|
// If flag `--theme` is present, copy theme to src
|
|
|
|
if args.is_present("theme") {
|
2018-03-14 23:27:56 +08:00
|
|
|
config.set("output.html.theme", "src/theme")?;
|
2017-06-26 07:02:32 +08:00
|
|
|
// Skip this if `--force` is present
|
|
|
|
if !args.is_present("force") {
|
|
|
|
// Print warning
|
2018-01-07 22:10:48 +08:00
|
|
|
println!();
|
|
|
|
println!(
|
|
|
|
"Copying the default theme to {}",
|
|
|
|
builder.config().book.src.display()
|
|
|
|
);
|
2017-11-18 20:41:04 +08:00
|
|
|
println!("This could potentially overwrite files already present in that directory.");
|
2017-06-26 07:02:32 +08:00
|
|
|
print!("\nAre you sure you want to continue? (y/n) ");
|
|
|
|
|
|
|
|
// Read answer from user and exit if it's not 'yes'
|
2017-11-18 20:41:04 +08:00
|
|
|
if confirm() {
|
|
|
|
builder.copy_theme(true);
|
2017-06-26 07:02:32 +08:00
|
|
|
}
|
2018-03-14 23:27:56 +08:00
|
|
|
} else {
|
|
|
|
builder.copy_theme(true);
|
2017-06-26 07:02:32 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-18 20:41:04 +08:00
|
|
|
println!("\nDo you want a .gitignore to be created? (y/n)");
|
2017-06-26 07:02:32 +08:00
|
|
|
|
2017-11-18 20:41:04 +08:00
|
|
|
if confirm() {
|
|
|
|
builder.create_gitignore(true);
|
2017-06-26 07:02:32 +08:00
|
|
|
}
|
|
|
|
|
2018-03-14 23:27:56 +08:00
|
|
|
config.book.title = request_book_title();
|
|
|
|
|
|
|
|
if let Some(author) = get_author_name() {
|
|
|
|
debug!("Obtained user name from gitconfig: {:?}", author);
|
|
|
|
config.book.authors.push(author);
|
|
|
|
builder.with_config(config);
|
|
|
|
}
|
|
|
|
|
2017-11-18 20:41:04 +08:00
|
|
|
builder.build()?;
|
2017-06-26 07:02:32 +08:00
|
|
|
println!("\nAll done, no errors...");
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2018-03-26 22:37:11 +08:00
|
|
|
/// Obtains author name from git config file by running the `git config` command.
|
2018-03-14 23:27:56 +08:00
|
|
|
fn get_author_name() -> Option<String> {
|
2018-03-26 22:37:11 +08:00
|
|
|
let output = Command::new("git")
|
|
|
|
.args(&["config", "--get", "user.name"])
|
|
|
|
.output()
|
|
|
|
.ok()?;
|
|
|
|
|
|
|
|
if output.status.success() {
|
|
|
|
Some(String::from_utf8_lossy(&output.stdout).trim().to_owned())
|
2018-03-14 23:27:56 +08:00
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Request book title from user and return if provided.
|
|
|
|
fn request_book_title() -> Option<String> {
|
|
|
|
println!("What title would you like to give the book? ");
|
|
|
|
io::stdout().flush().unwrap();
|
|
|
|
let mut resp = String::new();
|
|
|
|
io::stdin().read_line(&mut resp).unwrap();
|
|
|
|
let resp = resp.trim();
|
|
|
|
if resp.is_empty() {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some(resp.into())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Simple function for user confirmation
|
2017-06-26 07:24:33 +08:00
|
|
|
fn confirm() -> bool {
|
|
|
|
io::stdout().flush().unwrap();
|
|
|
|
let mut s = String::new();
|
|
|
|
io::stdin().read_line(&mut s).ok();
|
|
|
|
match &*s.trim() {
|
|
|
|
"Y" | "y" | "yes" | "Yes" => true,
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|