rust/src/rustdoc/extract.rs

351 lines
8.4 KiB
Rust
Raw Normal View History

//! Converts the Rust AST to the rustdoc document model
2012-01-17 17:44:32 -08:00
import syntax::ast;
import doc::item_utils;
export from_srv, extract;
fn from_srv(
srv: astsrv::srv,
default_name: ~str
) -> doc::doc {
2012-01-18 14:06:22 -08:00
//! Use the AST service to create a document tree
2012-01-18 14:06:22 -08:00
2012-06-30 16:19:07 -07:00
do astsrv::exec(srv) |ctxt| {
extract(ctxt.ast, default_name)
}
}
fn extract(
crate: @ast::crate,
default_name: ~str
) -> doc::doc {
doc::doc_({
pages: ~[
doc::cratepage({
topmod: top_moddoc_from_crate(crate, default_name),
})
]
})
}
fn top_moddoc_from_crate(
crate: @ast::crate,
default_name: ~str
) -> doc::moddoc {
2012-06-10 00:49:59 -07:00
moddoc_from_mod(mk_itemdoc(ast::crate_node_id, @default_name),
crate.node.module)
}
2012-02-17 15:39:05 -08:00
fn mk_itemdoc(id: ast::node_id, name: ast::ident) -> doc::itemdoc {
{
id: id,
2012-06-10 00:49:59 -07:00
name: *name,
path: ~[],
2012-02-17 15:39:05 -08:00
brief: none,
desc: none,
sections: ~[],
2012-02-24 15:43:57 -08:00
reexport: false
2012-02-17 15:39:05 -08:00
}
}
fn moddoc_from_mod(
itemdoc: doc::itemdoc,
module_: ast::_mod
) -> doc::moddoc {
doc::moddoc_({
item: itemdoc,
items: do vec::filter_map(module_.items) |item| {
let itemdoc = mk_itemdoc(item.id, item.ident);
2012-08-06 12:34:08 -07:00
match item.node {
2012-08-03 19:59:04 -07:00
ast::item_mod(m) => {
some(doc::modtag(
moddoc_from_mod(itemdoc, m)
))
}
2012-08-03 19:59:04 -07:00
ast::item_foreign_mod(nm) => {
some(doc::nmodtag(
nmoddoc_from_mod(itemdoc, nm)
))
}
2012-08-03 19:59:04 -07:00
ast::item_fn(_, _, _) => {
some(doc::fntag(
fndoc_from_fn(itemdoc)
))
}
2012-08-03 19:59:04 -07:00
ast::item_const(_, _) => {
some(doc::consttag(
constdoc_from_const(itemdoc)
))
}
ast::item_enum(enum_definition, _) => {
some(doc::enumtag(
enumdoc_from_enum(itemdoc, enum_definition.variants)
))
}
2012-08-03 19:59:04 -07:00
ast::item_trait(_, _, methods) => {
some(doc::traittag(
traitdoc_from_trait(itemdoc, methods)
))
}
2012-08-03 19:59:04 -07:00
ast::item_impl(_, _, _, methods) => {
some(doc::impltag(
impldoc_from_impl(itemdoc, methods)
))
}
2012-08-03 19:59:04 -07:00
ast::item_ty(_, _) => {
2012-02-01 22:41:41 -08:00
some(doc::tytag(
tydoc_from_ty(itemdoc)
2012-02-01 22:41:41 -08:00
))
}
2012-08-03 19:59:04 -07:00
_ => none
}
},
index: none
})
}
fn nmoddoc_from_mod(
itemdoc: doc::itemdoc,
module_: ast::foreign_mod
) -> doc::nmoddoc {
{
item: itemdoc,
fns: do vec::map(module_.items) |item| {
let itemdoc = mk_itemdoc(item.id, item.ident);
2012-08-06 12:34:08 -07:00
match item.node {
2012-08-03 19:59:04 -07:00
ast::foreign_item_fn(_, _) => {
fndoc_from_fn(itemdoc)
}
}
},
index: none
}
}
fn fndoc_from_fn(itemdoc: doc::itemdoc) -> doc::fndoc {
2012-01-30 13:05:25 -08:00
{
item: itemdoc,
sig: none
}
}
fn constdoc_from_const(itemdoc: doc::itemdoc) -> doc::constdoc {
2012-01-30 13:05:25 -08:00
{
item: itemdoc,
sig: none
}
}
#[test]
fn should_extract_const_name_and_id() {
let doc = test::mk_doc(~"const a: int = 0;");
assert doc.cratemod().consts()[0].id() != 0;
assert doc.cratemod().consts()[0].name() == ~"a";
}
fn enumdoc_from_enum(
itemdoc: doc::itemdoc,
variants: ~[ast::variant]
) -> doc::enumdoc {
2012-01-30 13:05:25 -08:00
{
item: itemdoc,
variants: variantdocs_from_variants(variants)
}
}
fn variantdocs_from_variants(
variants: ~[ast::variant]
) -> ~[doc::variantdoc] {
vec::map(variants, variantdoc_from_variant)
}
fn variantdoc_from_variant(variant: ast::variant) -> doc::variantdoc {
2012-01-30 13:05:25 -08:00
{
2012-06-10 00:49:59 -07:00
name: *variant.node.name,
desc: none,
sig: none
}
}
#[test]
fn should_extract_enums() {
let doc = test::mk_doc(~"enum e { v }");
assert doc.cratemod().enums()[0].id() != 0;
assert doc.cratemod().enums()[0].name() == ~"e";
}
#[test]
fn should_extract_enum_variants() {
let doc = test::mk_doc(~"enum e { v }");
assert doc.cratemod().enums()[0].variants[0].name == ~"v";
}
fn traitdoc_from_trait(
itemdoc: doc::itemdoc,
methods: ~[ast::trait_method]
) -> doc::traitdoc {
{
item: itemdoc,
methods: do vec::map(methods) |method| {
2012-08-06 12:34:08 -07:00
match method {
2012-08-03 19:59:04 -07:00
ast::required(ty_m) => {
{
name: *ty_m.ident,
brief: none,
desc: none,
sections: ~[],
sig: none,
implementation: doc::required,
}
}
2012-08-03 19:59:04 -07:00
ast::provided(m) => {
{
name: *m.ident,
brief: none,
desc: none,
sections: ~[],
sig: none,
implementation: doc::provided,
}
}
}
}
}
}
#[test]
fn should_extract_traits() {
let doc = test::mk_doc(~"trait i { fn f(); }");
assert doc.cratemod().traits()[0].name() == ~"i";
}
#[test]
fn should_extract_trait_methods() {
let doc = test::mk_doc(~"trait i { fn f(); }");
assert doc.cratemod().traits()[0].methods[0].name == ~"f";
}
fn impldoc_from_impl(
itemdoc: doc::itemdoc,
methods: ~[@ast::method]
) -> doc::impldoc {
{
item: itemdoc,
trait_types: ~[],
self_ty: none,
methods: do vec::map(methods) |method| {
{
2012-06-10 00:49:59 -07:00
name: *method.ident,
brief: none,
desc: none,
sections: ~[],
sig: none,
implementation: doc::provided,
}
}
}
}
#[test]
fn should_extract_impl_methods() {
2012-08-08 17:19:06 -07:00
let doc = test::mk_doc(~"impl int { fn f() { } }");
assert doc.cratemod().impls()[0].methods[0].name == ~"f";
}
2012-02-01 22:41:41 -08:00
fn tydoc_from_ty(
itemdoc: doc::itemdoc
2012-02-01 22:41:41 -08:00
) -> doc::tydoc {
{
item: itemdoc,
2012-02-01 22:41:41 -08:00
sig: none
}
}
#[test]
fn should_extract_tys() {
let doc = test::mk_doc(~"type a = int;");
assert doc.cratemod().types()[0].name() == ~"a";
2012-02-01 22:41:41 -08:00
}
#[cfg(test)]
2012-01-31 18:32:37 -08:00
mod test {
fn mk_doc(source: ~str) -> doc::doc {
let ast = parse::from_str(source);
extract(ast, ~"")
2012-01-31 18:32:37 -08:00
}
#[test]
fn extract_empty_crate() {
let doc = mk_doc(~"");
assert vec::is_empty(doc.cratemod().mods());
assert vec::is_empty(doc.cratemod().fns());
}
#[test]
fn extract_mods() {
let doc = mk_doc(~"mod a { mod b { } mod c { } }");
assert doc.cratemod().mods()[0].name() == ~"a";
assert doc.cratemod().mods()[0].mods()[0].name() == ~"b";
assert doc.cratemod().mods()[0].mods()[1].name() == ~"c";
}
#[test]
fn extract_foreign_mods() {
let doc = mk_doc(~"extern mod a { }");
assert doc.cratemod().nmods()[0].name() == ~"a";
}
#[test]
fn extract_fns_from_foreign_mods() {
let doc = mk_doc(~"extern mod a { fn a(); }");
assert doc.cratemod().nmods()[0].fns[0].name() == ~"a";
}
#[test]
fn extract_mods_deep() {
let doc = mk_doc(~"mod a { mod b { mod c { } } }");
assert doc.cratemod().mods()[0].mods()[0].mods()[0].name() == ~"c";
}
2012-01-18 18:35:55 -08:00
#[test]
fn extract_should_set_mod_ast_id() {
let doc = mk_doc(~"mod a { }");
assert doc.cratemod().mods()[0].id() != 0;
2012-01-18 18:35:55 -08:00
}
#[test]
fn extract_fns() {
2012-01-31 18:32:37 -08:00
let doc = mk_doc(
~"fn a() { } \
2012-01-31 18:32:37 -08:00
mod b { fn c() { } }");
assert doc.cratemod().fns()[0].name() == ~"a";
assert doc.cratemod().mods()[0].fns()[0].name() == ~"c";
}
#[test]
fn extract_should_set_fn_ast_id() {
let doc = mk_doc(~"fn a() { }");
assert doc.cratemod().fns()[0].id() != 0;
}
#[test]
fn extract_should_use_default_crate_name() {
let source = ~"";
let ast = parse::from_str(source);
let doc = extract(ast, ~"burp");
assert doc.cratemod().name() == ~"burp";
}
#[test]
fn extract_from_seq_srv() {
let source = ~"";
2012-06-30 16:19:07 -07:00
do astsrv::from_str(source) |srv| {
let doc = from_srv(srv, ~"name");
assert doc.cratemod().name() == ~"name";
}
}
}