#[doc = "Generate markdown from a document tree"]; import std::io; import std::io::writer_util; export mk_pass; fn mk_pass( writer: fn~() -> io::writer ) -> pass { ret fn~( _srv: astsrv::srv, doc: doc::cratedoc ) -> doc::cratedoc { write_markdown(doc, writer()); doc }; } type ctxt = { w: io::writer }; fn write_markdown( doc: doc::cratedoc, writer: io::writer ) { let ctxt = { w: writer }; write_crate(ctxt, doc); } enum hlvl { h1 = 1; h2 = 2; h3 = 3; } fn write_header(ctxt: ctxt, lvl: hlvl, title: str) { let hashes = str::from_chars(vec::init_elt('#', lvl as uint)); ctxt.w.write_line(#fmt("%s %s", hashes, title)); ctxt.w.write_line(""); } fn write_crate( ctxt: ctxt, doc: doc::cratedoc ) { write_header(ctxt, h1, #fmt("Crate %s", doc.topmod.name)); write_top_module(ctxt, doc.topmod); } fn write_top_module( ctxt: ctxt, moddoc: doc::moddoc ) { write_mod_contents(ctxt, moddoc); } fn write_mod( ctxt: ctxt, moddoc: doc::moddoc ) { let fullpath = str::connect(moddoc.path + [moddoc.name], "::"); write_header(ctxt, h2, #fmt("Module `%s`", fullpath)); write_mod_contents(ctxt, moddoc); } #[test] fn should_write_full_path_to_mod() { let markdown = test::render("mod a { mod b { mod c { } } }"); assert str::contains(markdown, "## Module `a::b::c`"); } fn write_mod_contents( ctxt: ctxt, doc: doc::moddoc ) { write_brief(ctxt, doc.brief); write_desc(ctxt, doc.desc); for fndoc in *doc.fns { write_fn(ctxt, fndoc); } for moddoc in *doc.mods { write_mod(ctxt, moddoc); } } #[test] fn should_write_crate_brief_description() { let markdown = test::render("#[doc(brief = \"this is the crate\")];"); assert str::contains(markdown, "this is the crate"); } #[test] fn should_write_crate_description() { let markdown = test::render("#[doc = \"this is the crate\"];"); assert str::contains(markdown, "this is the crate"); } fn write_fn( ctxt: ctxt, doc: doc::fndoc ) { write_header(ctxt, h3, #fmt("Function `%s`", doc.name)); write_sig(ctxt, doc.sig); write_brief(ctxt, doc.brief); write_desc(ctxt, doc.desc); write_args(ctxt, doc.args); write_return(ctxt, doc.return); } fn write_sig(ctxt: ctxt, sig: option) { alt sig { some(sig) { ctxt.w.write_line("```"); ctxt.w.write_line(#fmt("%s", sig)); ctxt.w.write_line("```"); ctxt.w.write_line(""); } none { fail "unimplemented" } } } #[test] fn should_write_the_function_signature() { let markdown = test::render("#[doc = \"f\"] fn a() { }"); assert str::contains(markdown, "```\nfn a()\n```"); } #[test] fn should_insert_blank_line_after_fn_signature() { let markdown = test::render("#[doc = \"f\"] fn a() { }"); assert str::contains(markdown, "fn a()\n```\n\n"); } fn write_brief( ctxt: ctxt, brief: option ) { alt brief { some(brief) { ctxt.w.write_line(brief); ctxt.w.write_line(""); } none { } } } fn write_desc( ctxt: ctxt, desc: option ) { alt desc { some(desc) { ctxt.w.write_line(desc); ctxt.w.write_line(""); } none { } } } fn write_args( ctxt: ctxt, args: [doc::argdoc] ) { if vec::is_not_empty(args) { ctxt.w.write_line("Arguments:"); ctxt.w.write_line(""); vec::iter(args) {|arg| write_arg(ctxt, arg) }; ctxt.w.write_line(""); } } fn write_arg(ctxt: ctxt, arg: doc::argdoc) { assert option::is_some(arg.ty); ctxt.w.write_str(#fmt( "* `%s`: `%s`", arg.name, option::get(arg.ty) )); alt arg.desc { some(desc) { ctxt.w.write_str(#fmt(" - %s", desc)); } none { } } ctxt.w.write_line(""); } #[test] fn should_write_argument_list() { let source = "fn a(b: int, c: int) { }"; let markdown = test::render(source); assert str::contains( markdown, "Arguments:\n\ \n\ * `b`: `int`\n\ * `c`: `int`\n\ \n" ); } #[test] fn should_not_write_arguments_if_none() { let source = "fn a() { } fn b() { }"; let markdown = test::render(source); assert !str::contains(markdown, "Arguments"); } #[test] fn should_write_argument_description() { let source = "#[doc(args(a = \"milk\"))] fn f(a: bool) { }"; let markdown = test::render(source); assert str::contains(markdown, "`a`: `bool` - milk"); } fn write_return( ctxt: ctxt, return: option ) { alt return { some(doc) { alt doc.ty { some(ty) { ctxt.w.write_line(#fmt("Returns `%s`", ty)); ctxt.w.write_line(""); alt doc.desc { some(d) { ctxt.w.write_line(d); ctxt.w.write_line(""); } none { } } } none { fail "unimplemented"; } } } none { } } } #[test] fn should_write_return_type_on_new_line() { let markdown = test::render("fn a() -> int { }"); assert str::contains(markdown, "\nReturns `int`"); } #[test] fn should_write_blank_line_between_return_type_and_next_header() { let markdown = test::render( "fn a() -> int { } \ fn b() -> int { }" ); assert str::contains(markdown, "Returns `int`\n\n##"); } #[test] fn should_not_write_return_type_when_there_is_none() { let markdown = test::render("fn a() { }"); assert !str::contains(markdown, "Returns"); } #[test] fn should_write_blank_line_after_return_description() { let markdown = test::render( "#[doc(return = \"blorp\")] fn a() -> int { }" ); assert str::contains(markdown, "blorp\n\n"); } #[cfg(test)] mod test { fn render(source: str) -> str { let srv = astsrv::mk_srv_from_str(source); let doc = extract::from_srv(srv, ""); #debug("doc (extract): %?", doc); let doc = tystr_pass::mk_pass()(srv, doc); #debug("doc (tystr): %?", doc); let doc = path_pass::mk_pass()(srv, doc); #debug("doc (path): %?", doc); let doc = attr_pass::mk_pass()(srv, doc); #debug("doc (attr): %?", doc); let markdown = write_markdown_str(doc); #debug("markdown: %s", markdown); markdown } fn write_markdown_str( doc: doc::cratedoc ) -> str { let buffer = io::mk_mem_buffer(); let writer = io::mem_buffer_writer(buffer); write_markdown(doc, writer); ret io::mem_buffer_str(buffer); } #[test] fn write_markdown_should_write_crate_header() { let srv = astsrv::mk_srv_from_str(""); let doc = extract::from_srv(srv, "belch"); let doc = attr_pass::mk_pass()(srv, doc); let markdown = write_markdown_str(doc); assert str::contains(markdown, "# Crate belch"); } #[test] fn write_markdown_should_write_function_header() { let markdown = render("fn func() { }"); assert str::contains(markdown, "## Function `func`"); } #[test] fn write_markdown_should_write_mod_headers() { let markdown = render("mod moo { }"); assert str::contains(markdown, "## Module `moo`"); } #[test] fn should_leave_blank_line_after_header() { let markdown = render("mod morp { }"); assert str::contains(markdown, "Module `morp`\n\n"); } #[test] fn should_leave_blank_line_between_fn_header_and_sig() { let markdown = render("#[doc(brief = \"brief\")] fn a() { }"); assert str::contains(markdown, "Function `a`\n\n```\nfn a()"); } #[test] fn should_leave_blank_line_after_brief() { let markdown = render("#[doc(brief = \"brief\")] fn a() { }"); assert str::contains(markdown, "brief\n\n"); } #[test] fn should_leave_blank_line_between_brief_and_desc() { let markdown = render( "#[doc(brief = \"brief\", desc = \"desc\")] fn a() { }" ); assert str::contains(markdown, "brief\n\ndesc"); } }