2012-03-16 17:14:37 -05:00
|
|
|
import result::result;
|
2012-02-26 18:32:57 -06:00
|
|
|
import std::getopts;
|
|
|
|
|
2012-02-29 21:06:50 -06:00
|
|
|
export output_format;
|
|
|
|
export output_style;
|
2012-02-26 18:32:57 -06:00
|
|
|
export config;
|
2012-02-29 20:10:40 -06:00
|
|
|
export default_config;
|
2012-02-26 18:32:57 -06:00
|
|
|
export parse_config;
|
2012-02-28 23:27:38 -06:00
|
|
|
export usage;
|
2012-02-26 18:32:57 -06:00
|
|
|
|
|
|
|
#[doc = "The type of document to output"]
|
|
|
|
enum output_format {
|
|
|
|
#[doc = "Markdown"]
|
|
|
|
markdown,
|
|
|
|
#[doc = "HTML, via markdown and pandoc"]
|
|
|
|
pandoc_html
|
|
|
|
}
|
|
|
|
|
|
|
|
#[doc = "How to organize the output"]
|
|
|
|
enum output_style {
|
|
|
|
#[doc = "All in a single document"]
|
|
|
|
doc_per_crate,
|
|
|
|
#[doc = "Each module in its own document"]
|
|
|
|
doc_per_mod
|
|
|
|
}
|
|
|
|
|
|
|
|
#[doc = "The configuration for a rustdoc session"]
|
|
|
|
type config = {
|
|
|
|
input_crate: str,
|
|
|
|
output_dir: str,
|
|
|
|
output_format: output_format,
|
|
|
|
output_style: output_style,
|
|
|
|
pandoc_cmd: option<str>
|
|
|
|
};
|
|
|
|
|
|
|
|
fn opt_output_dir() -> str { "output-dir" }
|
|
|
|
fn opt_output_format() -> str { "output-format" }
|
|
|
|
fn opt_output_style() -> str { "output-style" }
|
|
|
|
fn opt_pandoc_cmd() -> str { "pandoc-cmd" }
|
2012-02-28 23:27:38 -06:00
|
|
|
fn opt_help() -> str { "h" }
|
2012-02-26 18:32:57 -06:00
|
|
|
|
|
|
|
fn opts() -> [(getopts::opt, str)] {
|
|
|
|
[
|
|
|
|
(getopts::optopt(opt_output_dir()),
|
2012-02-28 23:27:38 -06:00
|
|
|
"--output-dir <val> put documents here"),
|
2012-02-26 18:32:57 -06:00
|
|
|
(getopts::optopt(opt_output_format()),
|
2012-02-28 23:27:38 -06:00
|
|
|
"--output-format <val> either 'markdown' or 'html'"),
|
2012-02-26 18:32:57 -06:00
|
|
|
(getopts::optopt(opt_output_style()),
|
2012-02-28 23:27:38 -06:00
|
|
|
"--output-style <val> either 'doc-per-crate' or 'doc-per-mod'"),
|
2012-02-26 18:32:57 -06:00
|
|
|
(getopts::optopt(opt_pandoc_cmd()),
|
2012-02-28 23:27:38 -06:00
|
|
|
"--pandoc-cmd <val> the command for running pandoc"),
|
|
|
|
(getopts::optflag(opt_help()),
|
|
|
|
"-h print help")
|
2012-02-26 18:32:57 -06:00
|
|
|
]
|
|
|
|
}
|
|
|
|
|
2012-02-28 23:27:38 -06:00
|
|
|
fn usage() {
|
2012-03-12 22:04:27 -05:00
|
|
|
import io::println;
|
2012-02-28 23:27:38 -06:00
|
|
|
|
|
|
|
println("Usage: rustdoc [options] <cratefile>\n");
|
|
|
|
println("Options:\n");
|
2012-04-06 13:01:43 -05:00
|
|
|
for opts().each {|opt|
|
2012-02-28 23:27:38 -06:00
|
|
|
println(#fmt(" %s", tuple::second(opt)));
|
|
|
|
}
|
|
|
|
println("");
|
|
|
|
}
|
|
|
|
|
2012-02-26 18:32:57 -06:00
|
|
|
fn default_config(input_crate: str) -> config {
|
|
|
|
{
|
|
|
|
input_crate: input_crate,
|
|
|
|
output_dir: ".",
|
|
|
|
output_format: pandoc_html,
|
|
|
|
output_style: doc_per_mod,
|
|
|
|
pandoc_cmd: none
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-06 16:46:10 -06:00
|
|
|
type program_output = fn~(str, [str]) -> {status: int, out: str, err: str};
|
|
|
|
|
|
|
|
fn mock_program_output(_prog: str, _args: [str]) -> {
|
|
|
|
status: int, out: str, err: str
|
|
|
|
} {
|
|
|
|
{
|
|
|
|
status: 0,
|
|
|
|
out: "",
|
|
|
|
err: ""
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-13 16:39:28 -05:00
|
|
|
fn parse_config(args: [str]) -> result<config, str> {
|
2012-03-12 22:04:27 -05:00
|
|
|
parse_config_(args, run::program_output)
|
2012-03-06 16:46:10 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
fn parse_config_(
|
|
|
|
args: [str],
|
|
|
|
program_output: program_output
|
2012-03-13 16:39:28 -05:00
|
|
|
) -> result<config, str> {
|
2012-02-26 18:32:57 -06:00
|
|
|
let args = vec::tail(args);
|
|
|
|
let opts = tuple::first(vec::unzip(opts()));
|
|
|
|
alt getopts::getopts(args, opts) {
|
|
|
|
result::ok(match) {
|
|
|
|
if vec::len(match.free) == 1u {
|
|
|
|
let input_crate = vec::head(match.free);
|
2012-03-06 16:46:10 -06:00
|
|
|
config_from_opts(input_crate, match, program_output)
|
2012-02-26 18:32:57 -06:00
|
|
|
} else if vec::is_empty(match.free) {
|
|
|
|
result::err("no crates specified")
|
|
|
|
} else {
|
|
|
|
result::err("multiple crates specified")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
result::err(f) {
|
|
|
|
result::err(getopts::fail_str(f))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn config_from_opts(
|
|
|
|
input_crate: str,
|
2012-03-06 16:46:10 -06:00
|
|
|
match: getopts::match,
|
|
|
|
program_output: program_output
|
2012-03-13 16:39:28 -05:00
|
|
|
) -> result<config, str> {
|
2012-02-26 18:32:57 -06:00
|
|
|
|
|
|
|
let config = default_config(input_crate);
|
|
|
|
let result = result::ok(config);
|
|
|
|
let result = result::chain(result) {|config|
|
|
|
|
let output_dir = getopts::opt_maybe_str(match, opt_output_dir());
|
|
|
|
result::ok({
|
2012-04-02 17:34:49 -05:00
|
|
|
output_dir: option::get_or_default(output_dir, config.output_dir)
|
2012-02-26 18:32:57 -06:00
|
|
|
with config
|
|
|
|
})
|
|
|
|
};
|
|
|
|
let result = result::chain(result) {|config|
|
|
|
|
let output_format = getopts::opt_maybe_str(
|
|
|
|
match, opt_output_format());
|
2012-04-06 14:14:09 -05:00
|
|
|
option::map_default(output_format, result::ok(config))
|
2012-04-02 17:34:49 -05:00
|
|
|
{|output_format|
|
2012-02-26 18:32:57 -06:00
|
|
|
result::chain(parse_output_format(output_format)) {|output_format|
|
|
|
|
result::ok({
|
|
|
|
output_format: output_format
|
|
|
|
with config
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
let result = result::chain(result) {|config|
|
|
|
|
let output_style = getopts::opt_maybe_str(match, opt_output_style());
|
2012-04-06 14:14:09 -05:00
|
|
|
option::map_default(output_style, result::ok(config))
|
2012-04-02 17:34:49 -05:00
|
|
|
{|output_style|
|
2012-02-26 18:32:57 -06:00
|
|
|
result::chain(parse_output_style(output_style)) {|output_style|
|
|
|
|
result::ok({
|
|
|
|
output_style: output_style
|
|
|
|
with config
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
let result = result::chain(result) {|config|
|
|
|
|
let pandoc_cmd = getopts::opt_maybe_str(match, opt_pandoc_cmd());
|
2012-03-06 16:46:10 -06:00
|
|
|
let pandoc_cmd = maybe_find_pandoc(
|
|
|
|
config, pandoc_cmd, program_output);
|
2012-02-26 18:32:57 -06:00
|
|
|
result::chain(pandoc_cmd) {|pandoc_cmd|
|
|
|
|
result::ok({
|
|
|
|
pandoc_cmd: pandoc_cmd
|
|
|
|
with config
|
|
|
|
})
|
|
|
|
}
|
|
|
|
};
|
|
|
|
ret result;
|
|
|
|
}
|
|
|
|
|
2012-03-13 16:39:28 -05:00
|
|
|
fn parse_output_format(output_format: str) -> result<output_format, str> {
|
2012-02-26 18:32:57 -06:00
|
|
|
alt output_format {
|
|
|
|
"markdown" { result::ok(markdown) }
|
|
|
|
"html" { result::ok(pandoc_html) }
|
|
|
|
_ { result::err(#fmt("unknown output format '%s'", output_format)) }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-13 16:39:28 -05:00
|
|
|
fn parse_output_style(output_style: str) -> result<output_style, str> {
|
2012-02-26 18:32:57 -06:00
|
|
|
alt output_style {
|
|
|
|
"doc-per-crate" { result::ok(doc_per_crate) }
|
|
|
|
"doc-per-mod" { result::ok(doc_per_mod) }
|
|
|
|
_ { result::err(#fmt("unknown output style '%s'", output_style)) }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn maybe_find_pandoc(
|
2012-03-06 16:46:10 -06:00
|
|
|
config: config,
|
|
|
|
maybe_pandoc_cmd: option<str>,
|
|
|
|
program_output: program_output
|
2012-03-13 16:39:28 -05:00
|
|
|
) -> result<option<str>, str> {
|
2012-03-06 16:46:10 -06:00
|
|
|
if config.output_format != pandoc_html {
|
|
|
|
ret result::ok(maybe_pandoc_cmd);
|
|
|
|
}
|
|
|
|
|
|
|
|
let possible_pandocs = alt maybe_pandoc_cmd {
|
|
|
|
some(pandoc_cmd) { [pandoc_cmd] }
|
|
|
|
none {
|
2012-03-07 21:22:02 -06:00
|
|
|
["pandoc"] + alt os::homedir() {
|
|
|
|
some(dir) {
|
|
|
|
[path::connect(dir, ".cabal/bin/pandoc")]
|
|
|
|
}
|
|
|
|
none { [] }
|
|
|
|
}
|
2012-03-06 16:46:10 -06:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
let pandoc = vec::find(possible_pandocs) {|pandoc|
|
|
|
|
let output = program_output(pandoc, ["--version"]);
|
|
|
|
#debug("testing pandoc cmd %s: %?", pandoc, output);
|
|
|
|
output.status == 0
|
|
|
|
};
|
|
|
|
|
|
|
|
if option::is_some(pandoc) {
|
|
|
|
result::ok(pandoc)
|
|
|
|
} else {
|
|
|
|
result::err("couldn't find pandoc")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_find_pandoc() {
|
|
|
|
let config = {
|
|
|
|
output_format: pandoc_html
|
|
|
|
with default_config("test")
|
|
|
|
};
|
|
|
|
let mock_program_output = fn~(_prog: str, _args: [str]) -> {
|
|
|
|
status: int, out: str, err: str
|
|
|
|
} {
|
|
|
|
{
|
|
|
|
status: 0, out: "pandoc 1.8.2.1", err: ""
|
|
|
|
}
|
|
|
|
};
|
|
|
|
let result = maybe_find_pandoc(config, none, mock_program_output);
|
|
|
|
assert result == result::ok(some("pandoc"));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_error_with_no_pandoc() {
|
|
|
|
let config = {
|
|
|
|
output_format: pandoc_html
|
|
|
|
with default_config("test")
|
|
|
|
};
|
|
|
|
let mock_program_output = fn~(_prog: str, _args: [str]) -> {
|
|
|
|
status: int, out: str, err: str
|
|
|
|
} {
|
|
|
|
{
|
|
|
|
status: 1, out: "", err: ""
|
|
|
|
}
|
|
|
|
};
|
|
|
|
let result = maybe_find_pandoc(config, none, mock_program_output);
|
|
|
|
assert result == result::err("couldn't find pandoc");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
2012-03-13 16:39:28 -05:00
|
|
|
fn parse_config(args: [str]) -> result<config, str> {
|
2012-03-06 16:46:10 -06:00
|
|
|
parse_config_(args, mock_program_output)
|
2012-02-26 18:32:57 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_error_with_no_crates() {
|
2012-03-06 16:46:10 -06:00
|
|
|
let config = test::parse_config(["rustdoc"]);
|
2012-02-26 18:32:57 -06:00
|
|
|
assert result::get_err(config) == "no crates specified";
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_error_with_multiple_crates() {
|
2012-03-06 16:46:10 -06:00
|
|
|
let config = test::parse_config(["rustdoc", "crate1.rc", "crate2.rc"]);
|
2012-02-26 18:32:57 -06:00
|
|
|
assert result::get_err(config) == "multiple crates specified";
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_set_output_dir_to_cwd_if_not_provided() {
|
2012-03-06 16:46:10 -06:00
|
|
|
let config = test::parse_config(["rustdoc", "crate.rc"]);
|
2012-02-26 18:32:57 -06:00
|
|
|
assert result::get(config).output_dir == ".";
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_set_output_dir_if_provided() {
|
2012-03-06 16:46:10 -06:00
|
|
|
let config = test::parse_config([
|
2012-02-26 18:32:57 -06:00
|
|
|
"rustdoc", "crate.rc", "--output-dir", "snuggles"
|
|
|
|
]);
|
|
|
|
assert result::get(config).output_dir == "snuggles";
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_set_output_format_to_pandoc_html_if_not_provided() {
|
2012-03-06 16:46:10 -06:00
|
|
|
let config = test::parse_config(["rustdoc", "crate.rc"]);
|
2012-02-26 18:32:57 -06:00
|
|
|
assert result::get(config).output_format == pandoc_html;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_set_output_format_to_markdown_if_requested() {
|
2012-03-06 16:46:10 -06:00
|
|
|
let config = test::parse_config([
|
2012-02-26 18:32:57 -06:00
|
|
|
"rustdoc", "crate.rc", "--output-format", "markdown"
|
|
|
|
]);
|
|
|
|
assert result::get(config).output_format == markdown;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_set_output_format_to_pandoc_html_if_requested() {
|
2012-03-06 16:46:10 -06:00
|
|
|
let config = test::parse_config([
|
2012-02-26 18:32:57 -06:00
|
|
|
"rustdoc", "crate.rc", "--output-format", "html"
|
|
|
|
]);
|
|
|
|
assert result::get(config).output_format == pandoc_html;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_error_on_bogus_format() {
|
2012-03-06 16:46:10 -06:00
|
|
|
let config = test::parse_config([
|
2012-02-26 18:32:57 -06:00
|
|
|
"rustdoc", "crate.rc", "--output-format", "bogus"
|
|
|
|
]);
|
|
|
|
assert result::get_err(config) == "unknown output format 'bogus'";
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_set_output_style_to_doc_per_mod_by_default() {
|
2012-03-06 16:46:10 -06:00
|
|
|
let config = test::parse_config(["rustdoc", "crate.rc"]);
|
2012-02-26 18:32:57 -06:00
|
|
|
assert result::get(config).output_style == doc_per_mod;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_set_output_style_to_one_doc_if_requested() {
|
2012-03-06 16:46:10 -06:00
|
|
|
let config = test::parse_config([
|
2012-02-26 18:32:57 -06:00
|
|
|
"rustdoc", "crate.rc", "--output-style", "doc-per-crate"
|
|
|
|
]);
|
|
|
|
assert result::get(config).output_style == doc_per_crate;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_set_output_style_to_doc_per_mod_if_requested() {
|
2012-03-06 16:46:10 -06:00
|
|
|
let config = test::parse_config([
|
2012-02-26 18:32:57 -06:00
|
|
|
"rustdoc", "crate.rc", "--output-style", "doc-per-mod"
|
|
|
|
]);
|
|
|
|
assert result::get(config).output_style == doc_per_mod;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_error_on_bogus_output_style() {
|
2012-03-06 16:46:10 -06:00
|
|
|
let config = test::parse_config([
|
2012-02-26 18:32:57 -06:00
|
|
|
"rustdoc", "crate.rc", "--output-style", "bogus"
|
|
|
|
]);
|
|
|
|
assert result::get_err(config) == "unknown output style 'bogus'";
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_set_pandoc_command_if_requested() {
|
2012-03-06 16:46:10 -06:00
|
|
|
let config = test::parse_config([
|
2012-02-26 18:32:57 -06:00
|
|
|
"rustdoc", "crate.rc", "--pandoc-cmd", "panda-bear-doc"
|
|
|
|
]);
|
|
|
|
assert result::get(config).pandoc_cmd == some("panda-bear-doc");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_set_pandoc_command_when_using_pandoc() {
|
2012-03-06 16:46:10 -06:00
|
|
|
let config = test::parse_config(["rustdoc", "crate.rc"]);
|
2012-02-26 18:32:57 -06:00
|
|
|
assert result::get(config).pandoc_cmd == some("pandoc");
|
|
|
|
}
|