317 lines
8.0 KiB
Rust
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)
|
|
}
|
|
}
|
|
}
|