rust/src/librustdoc/text_pass.rs
2013-02-17 11:02:23 +10:00

317 lines
8.0 KiB
Rust

// Copyright 2012 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.
//! Generic pass for performing an operation on all descriptions
use core::prelude::*;
use astsrv;
use doc::ItemUtils;
use doc;
use fold::Fold;
use fold;
use pass::Pass;
use util::NominalOp;
use std::par;
use std::cell::Cell;
pub fn mk_pass(name: ~str, op: @fn(&str) -> ~str) -> Pass {
let op = Cell(op);
Pass {
name: copy name,
f: |srv: astsrv::Srv, doc: doc::Doc| -> doc::Doc {
run(srv, doc, op.take())
}
}
}
type Op = @fn(&str) -> ~str;
#[allow(non_implicitly_copyable_typarams)]
fn run(
_srv: astsrv::Srv,
doc: doc::Doc,
op: Op
) -> doc::Doc {
let op = NominalOp {
op: op
};
let fold = Fold {
fold_item: fold_item,
fold_enum: fold_enum,
fold_trait: fold_trait,
fold_impl: fold_impl,
.. fold::default_any_fold(op)
};
(fold.fold_doc)(&fold, doc)
}
fn maybe_apply_op(op: NominalOp<Op>, s: &Option<~str>) -> Option<~str> {
s.map(|s| (op.op)(*s) )
}
fn fold_item(
fold: &fold::Fold<NominalOp<Op>>,
doc: doc::ItemDoc
) -> doc::ItemDoc {
let doc = fold::default_seq_fold_item(fold, doc);
doc::ItemDoc {
brief: maybe_apply_op(copy fold.ctxt, &doc.brief),
desc: maybe_apply_op(copy fold.ctxt, &doc.desc),
sections: apply_to_sections(copy fold.ctxt, copy doc.sections),
.. doc
}
}
fn apply_to_sections(
op: NominalOp<Op>,
sections: ~[doc::Section]
) -> ~[doc::Section] {
sections.map(|section| doc::Section {
header: (op.op)(copy section.header),
body: (op.op)(copy section.body)
})
}
fn fold_enum(
fold: &fold::Fold<NominalOp<Op>>,
doc: doc::EnumDoc) -> doc::EnumDoc {
let doc = fold::default_seq_fold_enum(fold, doc);
let fold_copy = copy *fold;
doc::EnumDoc {
variants: do doc.variants.map |variant| {
doc::VariantDoc {
desc: maybe_apply_op(copy fold_copy.ctxt, &variant.desc),
.. copy *variant
}
},
.. doc
}
}
fn fold_trait(
fold: &fold::Fold<NominalOp<Op>>,
doc: doc::TraitDoc
) -> doc::TraitDoc {
let doc = fold::default_seq_fold_trait(fold, doc);
doc::TraitDoc {
methods: apply_to_methods(copy fold.ctxt, copy doc.methods),
.. doc
}
}
fn apply_to_methods(
op: NominalOp<Op>,
docs: ~[doc::MethodDoc]
) -> ~[doc::MethodDoc] {
let op = copy op;
do docs.map |doc| {
doc::MethodDoc {
brief: maybe_apply_op(copy op, &doc.brief),
desc: maybe_apply_op(copy op, &doc.desc),
sections: apply_to_sections(copy op, copy doc.sections),
.. copy *doc
}
}
}
fn fold_impl(
fold: &fold::Fold<NominalOp<Op>>,
doc: doc::ImplDoc
) -> doc::ImplDoc {
let doc = fold::default_seq_fold_impl(fold, doc);
doc::ImplDoc {
methods: apply_to_methods(copy fold.ctxt, copy doc.methods),
.. doc
}
}
#[test]
fn should_execute_op_on_enum_brief() {
let doc = test::mk_doc(~"#[doc = \" a \"] enum a { b }");
assert doc.cratemod().enums()[0].brief() == Some(~"a");
}
#[test]
fn should_execute_op_on_enum_desc() {
let doc = test::mk_doc(~"#[doc = \" a \"] enum a { b }");
assert doc.cratemod().enums()[0].desc() == Some(~"a");
}
#[test]
fn should_execute_op_on_variant_desc() {
let doc = test::mk_doc(~"enum a { #[doc = \" a \"] b }");
assert doc.cratemod().enums()[0].variants[0].desc == Some(~"a");
}
#[test]
fn should_execute_op_on_trait_brief() {
let doc = test::mk_doc(
~"#[doc = \" a \"] trait i { fn a(); }");
assert doc.cratemod().traits()[0].brief() == Some(~"a");
}
#[test]
fn should_execute_op_on_trait_desc() {
let doc = test::mk_doc(
~"#[doc = \" a \"] trait i { fn a(); }");
assert doc.cratemod().traits()[0].desc() == Some(~"a");
}
#[test]
fn should_execute_op_on_trait_method_brief() {
let doc = test::mk_doc(
~"trait i { #[doc = \" a \"] fn a(); }");
assert doc.cratemod().traits()[0].methods[0].brief == Some(~"a");
}
#[test]
fn should_execute_op_on_trait_method_desc() {
let doc = test::mk_doc(
~"trait i { #[doc = \" a \"] fn a(); }");
assert doc.cratemod().traits()[0].methods[0].desc == Some(~"a");
}
#[test]
fn should_execute_op_on_impl_brief() {
let doc = test::mk_doc(
~"#[doc = \" a \"] impl int { fn a() { } }");
assert doc.cratemod().impls()[0].brief() == Some(~"a");
}
#[test]
fn should_execute_op_on_impl_desc() {
let doc = test::mk_doc(
~"#[doc = \" a \"] impl int { fn a() { } }");
assert doc.cratemod().impls()[0].desc() == Some(~"a");
}
#[test]
fn should_execute_op_on_impl_method_brief() {
let doc = test::mk_doc(
~"impl int { #[doc = \" a \"] fn a() { } }");
assert doc.cratemod().impls()[0].methods[0].brief == Some(~"a");
}
#[test]
fn should_execute_op_on_impl_method_desc() {
let doc = test::mk_doc(
~"impl int { #[doc = \" a \"] fn a() { } }");
assert doc.cratemod().impls()[0].methods[0].desc == Some(~"a");
}
#[test]
fn should_execute_op_on_type_brief() {
let doc = test::mk_doc(
~"#[doc = \" a \"] type t = int;");
assert doc.cratemod().types()[0].brief() == Some(~"a");
}
#[test]
fn should_execute_op_on_type_desc() {
let doc = test::mk_doc(
~"#[doc = \" a \"] type t = int;");
assert doc.cratemod().types()[0].desc() == Some(~"a");
}
#[test]
fn should_execute_on_item_section_headers() {
let doc = test::mk_doc(
~"#[doc = \"\
# Header \n\
Body\"]\
fn a() { }");
assert doc.cratemod().fns()[0].sections()[0].header == ~"Header";
}
#[test]
fn should_execute_on_item_section_bodies() {
let doc = test::mk_doc(
~"#[doc = \"\
# Header\n\
Body \"]\
fn a() { }");
assert doc.cratemod().fns()[0].sections()[0].body == ~"Body";
}
#[test]
fn should_execute_on_trait_method_section_headers() {
let doc = test::mk_doc(
~"trait i {
#[doc = \"\
# Header \n\
Body\"]\
fn a(); }");
assert doc.cratemod().traits()[0].methods[0].sections[0].header
== ~"Header";
}
#[test]
fn should_execute_on_trait_method_section_bodies() {
let doc = test::mk_doc(
~"trait i {
#[doc = \"\
# Header\n\
Body \"]\
fn a(); }");
assert doc.cratemod().traits()[0].methods[0].sections[0].body == ~"Body";
}
#[test]
fn should_execute_on_impl_method_section_headers() {
let doc = test::mk_doc(
~"impl bool {
#[doc = \"\
# Header \n\
Body\"]\
fn a() { } }");
assert doc.cratemod().impls()[0].methods[0].sections[0].header
== ~"Header";
}
#[test]
fn should_execute_on_impl_method_section_bodies() {
let doc = test::mk_doc(
~"impl bool {
#[doc = \"\
# Header\n\
Body \"]\
fn a() { } }");
assert doc.cratemod().impls()[0].methods[0].sections[0].body == ~"Body";
}
#[cfg(test)]
mod test {
use astsrv;
use attr_pass;
use desc_to_brief_pass;
use doc;
use extract;
use sectionalize_pass;
use text_pass::mk_pass;
use core::str;
pub fn mk_doc(source: ~str) -> doc::Doc {
do astsrv::from_str(copy source) |srv| {
let doc = extract::from_srv(srv.clone(), ~"");
let doc = (attr_pass::mk_pass().f)(srv.clone(), doc);
let doc = (desc_to_brief_pass::mk_pass().f)(srv.clone(), doc);
let doc = (sectionalize_pass::mk_pass().f)(srv.clone(), doc);
(mk_pass(~"", |s| str::trim(s) ).f)(srv.clone(), doc)
}
}
}