mdBook/src/book/mod.rs

498 lines
15 KiB
Rust
Raw Normal View History

pub mod bookitem;
pub mod bookconfig;
2016-12-23 16:15:32 +08:00
pub mod bookconfig_test;
pub use self::bookitem::{BookItem, BookItems};
2016-04-27 05:04:27 +08:00
use std::path::{Path, PathBuf};
use std::fs::{self, File};
use std::error::Error;
use std::io;
use std::io::{Read, Write};
2016-04-27 05:04:27 +08:00
use std::io::ErrorKind;
use std::process::Command;
use {theme, parse, utils};
use renderer::{Renderer, HtmlHandlebars};
use config::{BookConfig, HtmlConfig};
use config::tomlconfig::TomlConfig;
2016-04-27 05:04:27 +08:00
pub struct MDBook {
config: BookConfig,
2016-04-27 05:04:27 +08:00
pub content: Vec<BookItem>,
renderer: Box<Renderer>,
livereload: Option<String>,
/// Should `mdbook build` create files referenced from SUMMARY.md if they
/// don't exist
pub create_missing: bool,
pub google_analytics: Option<String>,
2016-04-27 05:04:27 +08:00
}
impl MDBook {
/// Create a new `MDBook` struct with root directory `root`
///
2017-03-30 20:09:14 +08:00
/// # Examples
///
/// ```no_run
/// # extern crate mdbook;
/// # use mdbook::MDBook;
/// # use std::path::Path;
/// # fn main() {
/// let book = MDBook::new("root_dir");
2017-03-30 20:09:14 +08:00
/// # }
/// ```
///
/// In this example, `root_dir` will be the root directory of our book
/// and is specified in function of the current working directory
/// by using a relative path instead of an
/// absolute path.
2017-03-30 20:09:14 +08:00
///
/// Default directory paths:
///
/// - source: `root/src`
/// - output: `root/book`
/// - theme: `root/theme`
2016-04-27 05:04:27 +08:00
///
/// They can both be changed by using [`set_src()`](#method.set_src) and
/// [`set_dest()`](#method.set_dest)
2016-04-27 05:04:27 +08:00
pub fn new(root: &Path) -> MDBook {
if !root.exists() || !root.is_dir() {
2016-08-14 21:40:08 +08:00
warn!("{:?} No directory with that name", root);
2016-04-27 05:04:27 +08:00
}
MDBook {
config: BookConfig::new(root),
2016-04-27 05:04:27 +08:00
content: vec![],
renderer: Box::new(HtmlHandlebars::new()),
2016-05-09 03:51:34 +08:00
2016-04-27 05:04:27 +08:00
livereload: None,
create_missing: true,
google_analytics: None,
2016-04-27 05:04:27 +08:00
}
}
/// Returns a flat depth-first iterator over the elements of the book,
/// it returns an [BookItem enum](bookitem.html):
2016-04-27 05:04:27 +08:00
/// `(section: String, bookitem: &BookItem)`
///
/// ```no_run
/// # extern crate mdbook;
/// # use mdbook::MDBook;
/// # use mdbook::BookItem;
/// # use std::path::Path;
/// # fn main() {
/// # let mut book = MDBook::new(Path::new("mybook"));
/// for item in book.iter() {
/// match item {
/// &BookItem::Chapter(ref section, ref chapter) => {},
/// &BookItem::Affix(ref chapter) => {},
/// &BookItem::Spacer => {},
/// }
/// }
///
/// // would print something like this:
/// // 1. Chapter 1
/// // 1.1 Sub Chapter
/// // 1.2 Sub Chapter
/// // 2. Chapter 2
/// //
/// // etc.
/// # }
/// ```
pub fn iter(&self) -> BookItems {
BookItems {
items: &self.content[..],
current_index: 0,
stack: Vec::new(),
}
}
/// `init()` creates some boilerplate files and directories
/// to get you started with your book.
2016-04-27 05:04:27 +08:00
///
/// ```text
/// book-test/
/// ├── book
/// └── src
/// ├── chapter_1.md
/// └── SUMMARY.md
/// ```
///
/// It uses the paths given as source and output directories
/// and adds a `SUMMARY.md` and a
2016-04-27 05:04:27 +08:00
/// `chapter_1.md` to the source directory.
pub fn init(&mut self) -> Result<(), Box<Error>> {
debug!("[fn]: init");
if !self.config.get_root().exists() {
fs::create_dir_all(&self.config.get_root()).unwrap();
info!("{:?} created", &self.config.get_root());
2016-04-27 05:04:27 +08:00
}
{
if let Some(htmlconfig) = self.config.get_html_config() {
if !htmlconfig.get_destination().exists() {
debug!("[*]: {:?} does not exist, trying to create directory", htmlconfig.get_destination());
fs::create_dir_all(htmlconfig.get_destination())?;
}
2016-04-27 05:04:27 +08:00
}
2016-04-27 05:04:27 +08:00
if !self.config.get_source().exists() {
debug!("[*]: {:?} does not exist, trying to create directory", self.config.get_source());
fs::create_dir_all(self.config.get_source())?;
2016-04-27 05:04:27 +08:00
}
let summary = self.config.get_source().join("SUMMARY.md");
2016-04-27 05:04:27 +08:00
if !summary.exists() {
// Summary does not exist, create it
debug!("[*]: {:?} does not exist, trying to create SUMMARY.md", &summary);
let mut f = File::create(&summary)?;
2016-04-27 05:04:27 +08:00
debug!("[*]: Writing to SUMMARY.md");
writeln!(f, "# Summary")?;
writeln!(f, "")?;
writeln!(f, "- [Chapter 1](./chapter_1.md)")?;
2016-04-27 05:04:27 +08:00
}
}
// parse SUMMARY.md, and create the missing item related file
self.parse_summary()?;
2016-04-27 05:04:27 +08:00
debug!("[*]: constructing paths for missing files");
for item in self.iter() {
debug!("[*]: item: {:?}", item);
let ch = match *item {
2016-04-27 05:04:27 +08:00
BookItem::Spacer => continue,
BookItem::Chapter(_, ref ch) |
BookItem::Affix(ref ch) => ch,
};
if !ch.path.as_os_str().is_empty() {
let path = self.config.get_source().join(&ch.path);
if !path.exists() {
if !self.create_missing {
return Err(format!("'{}' referenced from SUMMARY.md does not exist.", path.to_string_lossy())
.into());
2016-04-27 05:04:27 +08:00
}
debug!("[*]: {:?} does not exist, trying to create file", path);
::std::fs::create_dir_all(path.parent().unwrap())?;
let mut f = File::create(path)?;
// debug!("[*]: Writing to {:?}", path);
writeln!(f, "# {}", ch.name)?;
}
2016-04-27 05:04:27 +08:00
}
}
debug!("[*]: init done");
Ok(())
}
pub fn create_gitignore(&self) {
let gitignore = self.get_gitignore();
// If the HTML renderer is not set, return
if self.config.get_html_config().is_none() { return; }
let destination = self.config.get_html_config()
.expect("The HtmlConfig does exist, checked just before")
.get_destination();
// Check that the gitignore does not extist and that the destination path begins with the root path
// We assume tha if it does begin with the root path it is contained within. This assumption
// will not hold true for paths containing double dots to go back up e.g. `root/../destination`
if !gitignore.exists() && destination.starts_with(self.config.get_root()) {
let relative = destination
.strip_prefix(self.config.get_root())
.expect("Could not strip the root prefix, path is not relative to root")
.to_str()
.expect("Could not convert to &str");
2016-04-27 05:04:27 +08:00
debug!("[*]: {:?} does not exist, trying to create .gitignore", gitignore);
let mut f = File::create(&gitignore).expect("Could not create file.");
debug!("[*]: Writing to .gitignore");
writeln!(f, "{}", relative).expect("Could not write to file.");
}
}
/// The `build()` method is the one where everything happens.
/// First it parses `SUMMARY.md` to construct the book's structure
/// in the form of a `Vec<BookItem>` and then calls `render()`
2016-04-27 05:04:27 +08:00
/// method of the current renderer.
///
/// It is the renderer who generates all the output files.
pub fn build(&mut self) -> Result<(), Box<Error>> {
debug!("[fn]: build");
self.init()?;
2016-04-27 05:04:27 +08:00
// Clean output directory
if let Some(htmlconfig) = self.config.get_html_config() {
utils::fs::remove_dir_content(htmlconfig.get_destination())?;
}
self.renderer.render(&self)?;
2016-04-27 05:04:27 +08:00
Ok(())
}
pub fn get_gitignore(&self) -> PathBuf {
self.config.get_root().join(".gitignore")
2016-04-27 05:04:27 +08:00
}
pub fn copy_theme(&self) -> Result<(), Box<Error>> {
debug!("[fn]: copy_theme");
if let Some(themedir) = self.config.get_html_config().and_then(HtmlConfig::get_theme) {
2016-04-27 05:04:27 +08:00
if !themedir.exists() {
debug!("[*]: {:?} does not exist, trying to create directory", themedir);
fs::create_dir(&themedir)?;
}
2016-04-27 05:04:27 +08:00
// index.hbs
let mut index = File::create(&themedir.join("index.hbs"))?;
index.write_all(theme::INDEX)?;
2016-04-27 05:04:27 +08:00
// book.css
let mut css = File::create(&themedir.join("book.css"))?;
css.write_all(theme::CSS)?;
2016-04-27 05:04:27 +08:00
// favicon.png
let mut favicon = File::create(&themedir.join("favicon.png"))?;
favicon.write_all(theme::FAVICON)?;
2016-04-27 05:04:27 +08:00
// book.js
let mut js = File::create(&themedir.join("book.js"))?;
js.write_all(theme::JS)?;
2016-04-27 05:04:27 +08:00
// highlight.css
let mut highlight_css = File::create(&themedir.join("highlight.css"))?;
highlight_css.write_all(theme::HIGHLIGHT_CSS)?;
2016-04-27 05:04:27 +08:00
// highlight.js
let mut highlight_js = File::create(&themedir.join("highlight.js"))?;
highlight_js.write_all(theme::HIGHLIGHT_JS)?;
}
2016-04-27 05:04:27 +08:00
Ok(())
}
pub fn write_file<P: AsRef<Path>>(&self, filename: P, content: &[u8]) -> Result<(), Box<Error>> {
let path = self.get_destination()
.ok_or(String::from("HtmlConfig not set, could not find a destination"))?
.join(filename);
utils::fs::create_file(&path)
.and_then(|mut file| file.write_all(content))
.map_err(|e| io::Error::new(io::ErrorKind::Other, format!("Could not create {}: {}", path.display(), e)))?;
Ok(())
}
/// Parses the `book.json` file (if it exists) to extract
/// the configuration parameters.
2016-04-27 05:04:27 +08:00
/// The `book.json` file should be in the root directory of the book.
/// The root directory is the one specified when creating a new `MDBook`
pub fn read_config(mut self) -> Result<Self, Box<Error>> {
2016-04-27 05:04:27 +08:00
let toml = self.get_root().join("book.toml");
let json = self.get_root().join("book.json");
2016-04-27 05:04:27 +08:00
if toml.exists() {
let mut file = File::open(toml)?;
let mut content = String::new();
file.read_to_string(&mut content)?;
2016-04-27 05:04:27 +08:00
let parsed_config = TomlConfig::from_toml(&content)?;
self.config.fill_from_tomlconfig(parsed_config);
} else if json.exists() {
unimplemented!();
}
2016-04-27 05:04:27 +08:00
Ok(self)
2016-04-27 05:04:27 +08:00
}
/// You can change the default renderer to another one
/// by using this method. The only requirement
/// is for your renderer to implement the
/// [Renderer trait](../../renderer/renderer/trait.Renderer.html)
2016-04-27 05:04:27 +08:00
///
/// ```no_run
/// extern crate mdbook;
/// use mdbook::MDBook;
/// use mdbook::renderer::HtmlHandlebars;
/// # use std::path::Path;
///
/// fn main() {
/// let mut book = MDBook::new(Path::new("mybook"))
/// .set_renderer(Box::new(HtmlHandlebars::new()));
///
/// // In this example we replace the default renderer
/// // by the default renderer...
/// // Don't forget to put your renderer in a Box
2016-04-27 05:04:27 +08:00
/// }
/// ```
///
/// **note:** Don't forget to put your renderer in a `Box`
/// before passing it to `set_renderer()`
2016-04-27 05:04:27 +08:00
pub fn set_renderer(mut self, renderer: Box<Renderer>) -> Self {
self.renderer = renderer;
self
}
pub fn test(&mut self) -> Result<(), Box<Error>> {
// read in the chapters
self.parse_summary()?;
2016-04-27 05:04:27 +08:00
for item in self.iter() {
if let BookItem::Chapter(_, ref ch) = *item {
if ch.path != PathBuf::new() {
2016-04-27 05:04:27 +08:00
let path = self.get_source().join(&ch.path);
2016-04-27 05:04:27 +08:00
println!("[*]: Testing file: {:?}", path);
2016-04-27 05:04:27 +08:00
let output_result = Command::new("rustdoc").arg(&path).arg("--test").output();
let output = output_result?;
2016-04-27 05:04:27 +08:00
if !output.status.success() {
return Err(Box::new(io::Error::new(ErrorKind::Other,
format!("{}\n{}",
String::from_utf8_lossy(&output.stdout),
String::from_utf8_lossy(&output.stderr)))) as
Box<Error>);
2016-04-27 05:04:27 +08:00
}
}
2016-04-27 05:04:27 +08:00
}
}
Ok(())
}
pub fn get_root(&self) -> &Path {
self.config.get_root()
2016-04-27 05:04:27 +08:00
}
pub fn with_destination<T: Into<PathBuf>>(mut self, destination: T) -> Self {
let root = self.config.get_root().to_owned();
if let Some(htmlconfig) = self.config.get_mut_html_config() {
htmlconfig.set_destination(&root, &destination.into());
} else {
error!("There is no HTML renderer set...");
2016-04-27 05:04:27 +08:00
}
self
}
2016-04-27 05:04:27 +08:00
pub fn get_destination(&self) -> Option<&Path> {
if let Some(htmlconfig) = self.config.get_html_config() {
return Some(htmlconfig.get_destination());
2016-04-27 05:04:27 +08:00
}
None
}
pub fn with_source<T: Into<PathBuf>>(mut self, source: T) -> Self {
self.config.set_source(source);
2016-04-27 05:04:27 +08:00
self
}
pub fn get_source(&self) -> &Path {
self.config.get_source()
2016-04-27 05:04:27 +08:00
}
pub fn with_title<T: Into<String>>(mut self, title: T) -> Self {
self.config.set_title(title);
2016-04-27 05:04:27 +08:00
self
}
pub fn get_title(&self) -> &str {
self.config.get_title()
2016-04-27 05:04:27 +08:00
}
/*
2016-04-27 05:04:27 +08:00
pub fn set_author(mut self, author: &str) -> Self {
self.author = author.to_owned();
self
}
pub fn get_author(&self) -> &str {
&self.author
}
*/
pub fn with_description<T: Into<String>>(mut self, description: T) -> Self {
self.config.set_description(description);
2016-04-27 05:04:27 +08:00
self
}
pub fn get_description(&self) -> &str {
self.config.get_description()
2016-04-27 05:04:27 +08:00
}
pub fn set_livereload(&mut self, livereload: String) -> &mut Self {
self.livereload = Some(livereload);
self
}
pub fn unset_livereload(&mut self) -> &Self {
self.livereload = None;
self
}
pub fn get_livereload(&self) -> Option<&String> {
self.livereload.as_ref()
2016-04-27 05:04:27 +08:00
}
pub fn with_theme_path<T: Into<PathBuf>>(mut self, theme_path: T) -> Self {
let root = self.config.get_root().to_owned();
if let Some(htmlconfig) = self.config.get_mut_html_config() {
htmlconfig.set_theme(&root, &theme_path.into());
} else {
error!("There is no HTML renderer set...");
}
self
}
pub fn get_theme_path(&self) -> Option<&PathBuf> {
if let Some(htmlconfig) = self.config.get_html_config() {
return htmlconfig.get_theme();
}
None
}
2016-04-27 05:04:27 +08:00
// Construct book
fn parse_summary(&mut self) -> Result<(), Box<Error>> {
// When append becomes stable, use self.content.append() ...
self.content = parse::construct_bookitems(&self.get_source().join("SUMMARY.md"))?;
2016-04-27 05:04:27 +08:00
Ok(())
}
}