rust/src/librustdoc/markdown_writer.rs

288 lines
7.7 KiB
Rust
Raw Normal View History

// Copyright 2012-2013 The Rust Project Developers. See the COPYRIGHT
// file at the top-level directory of this distribution and at
// http://rust-lang.org/COPYRIGHT.
//
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
// option. This file may not be copied, modified, or distributed
// except according to those terms.
use config;
2012-09-18 16:48:40 -07:00
use doc::ItemUtils;
use doc;
use std::comm::*;
use std::comm;
use std::io;
use std::result;
use std::run;
use std::str;
use std::task;
2013-05-19 11:17:49 -07:00
use extra::future;
2013-07-02 12:47:32 -07:00
#[deriving(Clone)]
2012-11-19 18:00:12 -08:00
pub enum WriteInstr {
2012-09-18 16:48:40 -07:00
Write(~str),
Done
}
pub type Writer = ~fn(v: WriteInstr);
pub type WriterFactory = ~fn(page: doc::Page) -> Writer;
2012-11-19 18:00:12 -08:00
pub trait WriterUtils {
fn put_str(&self, str: ~str);
fn put_line(&self, str: ~str);
fn put_done(&self);
}
impl WriterUtils for Writer {
fn put_str(&self, str: ~str) {
2013-01-30 19:56:17 -08:00
(*self)(Write(str));
}
fn put_line(&self, str: ~str) {
self.put_str(str + "\n");
}
fn put_done(&self) {
2013-01-30 19:56:17 -08:00
(*self)(Done)
}
}
2013-01-30 19:32:36 -08:00
pub fn make_writer_factory(config: config::Config) -> WriterFactory {
2012-08-06 12:34:08 -07:00
match config.output_format {
2012-09-18 16:48:40 -07:00
config::Markdown => {
markdown_writer_factory(config)
2012-03-01 14:21:14 -08:00
}
2012-09-18 16:48:40 -07:00
config::PandocHtml => {
pandoc_writer_factory(config)
2012-03-01 14:21:14 -08:00
}
}
}
2013-01-30 19:32:36 -08:00
fn markdown_writer_factory(config: config::Config) -> WriterFactory {
let result: ~fn(page: doc::Page) -> Writer = |page| {
markdown_writer(&config, page)
};
result
}
2013-01-30 19:32:36 -08:00
fn pandoc_writer_factory(config: config::Config) -> WriterFactory {
let result: ~fn(doc::Page) -> Writer = |page| {
pandoc_writer(&config, page)
};
result
}
fn markdown_writer(
config: &config::Config,
2013-01-30 19:32:36 -08:00
page: doc::Page
2012-09-18 16:48:40 -07:00
) -> Writer {
let filename = make_local_filename(config, page);
2012-06-30 16:19:07 -07:00
do generic_writer |markdown| {
write_file(&filename, markdown);
2012-03-01 14:21:14 -08:00
}
}
fn pandoc_writer(
config: &config::Config,
2013-01-30 19:32:36 -08:00
page: doc::Page
2012-09-18 16:48:40 -07:00
) -> Writer {
2013-03-28 18:39:09 -07:00
assert!(config.pandoc_cmd.is_some());
2013-07-02 12:47:32 -07:00
let pandoc_cmd = (*config.pandoc_cmd.get_ref()).clone();
let filename = make_local_filename(config, page);
2012-03-01 14:21:14 -08:00
let pandoc_args = ~[
~"--standalone",
~"--section-divs",
~"--from=markdown",
~"--to=html",
~"--css=rust.css",
~"--output=" + filename.to_str()
];
2012-03-01 14:21:14 -08:00
2012-06-30 16:19:07 -07:00
do generic_writer |markdown| {
use std::io::WriterUtil;
2012-03-01 14:21:14 -08:00
2012-08-22 17:24:52 -07:00
debug!("pandoc cmd: %s", pandoc_cmd);
2013-06-10 23:25:25 +10:00
debug!("pandoc args: %s", pandoc_args.connect(" "));
2012-03-01 14:21:14 -08:00
let mut proc = run::Process::new(pandoc_cmd, pandoc_args, run::ProcessOptions::new());
2012-03-01 14:21:14 -08:00
proc.input().write_str(markdown);
let output = proc.finish_with_output();
2012-03-01 14:21:14 -08:00
debug!("pandoc result: %i", output.status);
if output.status != 0 {
error!("pandoc-out: %s", str::from_bytes(output.output));
error!("pandoc-err: %s", str::from_bytes(output.error));
fail!("pandoc failed");
2012-03-01 14:21:14 -08:00
}
}
}
fn generic_writer(process: ~fn(markdown: ~str)) -> Writer {
2013-02-01 18:03:32 -08:00
let (po, ch) = stream::<WriteInstr>();
2013-02-15 00:37:08 -08:00
do task::spawn || {
let mut markdown = ~"";
let mut keep_going = true;
while keep_going {
2012-11-28 20:34:40 -08:00
match po.recv() {
Write(s) => markdown.push_str(s),
2012-09-18 16:48:40 -07:00
Done => keep_going = false
}
}
2013-02-15 00:37:08 -08:00
process(markdown);
};
let result: ~fn(instr: WriteInstr) = |instr| ch.send(instr);
result
}
pub fn make_local_filename(
config: &config::Config,
2013-01-30 19:32:36 -08:00
page: doc::Page
) -> Path {
let filename = make_filename(config, page);
config.output_dir.push_rel(&filename)
}
2012-11-19 18:00:12 -08:00
pub fn make_filename(
config: &config::Config,
2013-01-30 19:32:36 -08:00
page: doc::Page
) -> Path {
let filename = {
2012-08-06 12:34:08 -07:00
match page {
2012-09-18 16:48:40 -07:00
doc::CratePage(doc) => {
if config.output_format == config::PandocHtml &&
config.output_style == config::DocPerMod {
~"index"
} else {
assert!(doc.topmod.name_() != ~"");
doc.topmod.name_()
}
}
2012-09-18 16:48:40 -07:00
doc::ItemPage(doc) => {
(doc.path() + &[doc.name_()]).connect("_")
}
}
};
2012-08-06 12:34:08 -07:00
let ext = match config.output_format {
2012-09-18 16:48:40 -07:00
config::Markdown => ~"md",
config::PandocHtml => ~"html"
};
Path(filename).with_filetype(ext)
}
2013-01-30 19:32:36 -08:00
fn write_file(path: &Path, s: ~str) {
use std::io::WriterUtil;
match io::file_writer(path, [io::Create, io::Truncate]) {
2012-08-26 16:54:31 -07:00
result::Ok(writer) => {
writer.write_str(s);
}
result::Err(e) => fail!(e)
}
}
2012-11-19 18:00:12 -08:00
pub fn future_writer_factory(
2013-02-01 18:19:57 -08:00
) -> (WriterFactory, Port<(doc::Page, ~str)>) {
let (markdown_po, markdown_ch) = stream();
let markdown_ch = SharedChan::new(markdown_ch);
let writer_factory: WriterFactory = |page| {
2013-02-02 03:10:12 -08:00
let (writer_po, writer_ch) = comm::stream();
2013-02-01 18:19:57 -08:00
let markdown_ch = markdown_ch.clone();
2013-02-15 00:37:08 -08:00
do task::spawn || {
let (writer, future) = future_writer();
let mut future = future;
2013-02-15 00:37:08 -08:00
writer_ch.send(writer);
let s = future.get();
2013-07-02 12:47:32 -07:00
markdown_ch.send((page.clone(), s));
}
2012-11-28 20:34:40 -08:00
writer_po.recv()
};
2013-02-15 00:37:08 -08:00
(writer_factory, markdown_po)
}
2012-09-18 16:48:40 -07:00
fn future_writer() -> (Writer, future::Future<~str>) {
2013-02-02 03:10:12 -08:00
let (port, chan) = comm::stream();
2013-07-02 12:47:32 -07:00
let writer: ~fn(instr: WriteInstr) = |instr| chan.send(instr.clone());
2013-02-15 00:37:08 -08:00
let future = do future::from_fn || {
let mut res = ~"";
loop {
match port.recv() {
Write(s) => res.push_str(s),
2012-09-18 16:48:40 -07:00
Done => break
}
}
res
};
2013-02-15 00:37:08 -08:00
(writer, future)
}
#[cfg(test)]
mod test {
use astsrv;
use doc;
use extract;
use path_pass;
use config;
use super::make_local_filename;
fn mk_doc(name: ~str, source: ~str) -> doc::Doc {
do astsrv::from_str(source) |srv| {
2013-07-02 12:47:32 -07:00
let doc = extract::from_srv(srv.clone(), name.clone());
let doc = (path_pass::mk_pass().f)(srv.clone(), doc);
doc
}
}
#[test]
fn should_use_markdown_file_name_based_off_crate() {
let config = config::Config {
output_dir: Path("output/dir"),
output_format: config::Markdown,
output_style: config::DocPerCrate,
.. config::default_config(&Path("input/test.rc"))
};
let doc = mk_doc(~"test", ~"");
let page = doc::CratePage(doc.CrateDoc());
let filename = make_local_filename(&config, page);
assert_eq!(filename.to_str(), ~"output/dir/test.md");
}
#[test]
fn should_name_html_crate_file_name_index_html_when_doc_per_mod() {
let config = config::Config {
output_dir: Path("output/dir"),
output_format: config::PandocHtml,
output_style: config::DocPerMod,
.. config::default_config(&Path("input/test.rc"))
};
let doc = mk_doc(~"", ~"");
let page = doc::CratePage(doc.CrateDoc());
let filename = make_local_filename(&config, page);
assert_eq!(filename.to_str(), ~"output/dir/index.html");
}
#[test]
fn should_name_mod_file_names_by_path() {
let config = config::Config {
output_dir: Path("output/dir"),
output_format: config::PandocHtml,
output_style: config::DocPerMod,
.. config::default_config(&Path("input/test.rc"))
};
let doc = mk_doc(~"", ~"mod a { mod b { } }");
// hidden __std_macros module at the start.
2013-07-02 12:47:32 -07:00
let modb = doc.cratemod().mods()[1].mods()[0].clone();
let page = doc::ItemPage(doc::ModTag(modb));
let filename = make_local_filename(&config, page);
assert_eq!(filename, Path("output/dir/a_b.html"));
}
}