rust/src/librustdoc/text_pass.rs

320 lines
8.5 KiB
Rust
Raw Normal View History

// 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;
2012-09-18 18:48:40 -05:00
use doc::ItemUtils;
use doc;
use fold::Fold;
use fold;
use pass::Pass;
use util::NominalOp;
use core::cell::Cell;
pub fn mk_pass(name: ~str, op: @fn(&str) -> ~str) -> Pass {
let op = Cell::new(op);
Pass {
2013-01-30 15:14:35 -06:00
name: copy name,
f: |srv: astsrv::Srv, doc: doc::Doc| -> doc::Doc {
2013-01-30 20:52:31 -06:00
run(srv, doc, op.take())
2012-02-27 20:07:16 -06:00
}
}
}
type Op = @fn(&str) -> ~str;
2012-08-01 15:35:33 -05:00
#[allow(non_implicitly_copyable_typarams)]
fn run(
2012-09-18 18:48:40 -05:00
_srv: astsrv::Srv,
2013-01-30 21:32:36 -06:00
doc: doc::Doc,
op: Op
2012-09-18 18:48:40 -05:00
) -> doc::Doc {
let op = NominalOp {
2013-02-15 02:37:08 -06:00
op: op
};
let fold = Fold {
2012-02-17 17:59:57 -06:00
fold_item: fold_item,
fold_enum: fold_enum,
fold_trait: fold_trait,
2012-09-04 15:29:32 -05:00
fold_impl: fold_impl,
2013-02-15 02:37:08 -06:00
.. fold::default_any_fold(op)
};
(fold.fold_doc)(&fold, doc)
}
2013-01-30 21:32:36 -06:00
fn maybe_apply_op(op: NominalOp<Op>, s: &Option<~str>) -> Option<~str> {
2013-01-30 20:52:31 -06:00
s.map(|s| (op.op)(*s) )
}
fn fold_item(
fold: &fold::Fold<NominalOp<Op>>,
2013-01-30 21:32:36 -06:00
doc: doc::ItemDoc
) -> doc::ItemDoc {
2012-02-17 17:59:57 -06:00
let doc = fold::default_seq_fold_item(fold, doc);
doc::ItemDoc {
2013-01-30 20:52:31 -06:00
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),
2012-09-04 15:29:32 -05:00
.. doc
}
}
fn apply_to_sections(
2013-02-01 20:05:10 -06:00
op: NominalOp<Op>,
2013-01-30 21:32:36 -06:00
sections: ~[doc::Section]
) -> ~[doc::Section] {
2013-02-16 16:54:34 -06:00
sections.map(|section| doc::Section {
2013-01-30 15:14:35 -06:00
header: (op.op)(copy section.header),
body: (op.op)(copy section.body)
})
2012-03-09 13:47:31 -06:00
}
fn fold_enum(
fold: &fold::Fold<NominalOp<Op>>,
2013-01-30 21:32:36 -06:00
doc: doc::EnumDoc) -> doc::EnumDoc {
2012-02-17 17:59:57 -06:00
let doc = fold::default_seq_fold_enum(fold, doc);
2012-11-19 20:48:46 -06:00
let fold_copy = copy *fold;
2012-02-17 17:59:57 -06:00
doc::EnumDoc {
2013-02-16 16:54:34 -06:00
variants: do doc.variants.map |variant| {
doc::VariantDoc {
2013-01-30 20:52:31 -06:00
desc: maybe_apply_op(copy fold_copy.ctxt, &variant.desc),
2013-01-30 15:14:35 -06:00
.. copy *variant
}
2012-09-04 15:29:32 -05:00
},
.. doc
}
}
fn fold_trait(
fold: &fold::Fold<NominalOp<Op>>,
2013-01-30 21:32:36 -06:00
doc: doc::TraitDoc
) -> doc::TraitDoc {
let doc = fold::default_seq_fold_trait(fold, doc);
2012-02-17 17:59:57 -06:00
doc::TraitDoc {
2013-01-30 20:52:31 -06:00
methods: apply_to_methods(copy fold.ctxt, copy doc.methods),
2012-09-04 15:29:32 -05:00
.. doc
2012-01-31 22:42:06 -06:00
}
}
fn apply_to_methods(
2013-01-30 21:32:36 -06:00
op: NominalOp<Op>,
docs: ~[doc::MethodDoc]
) -> ~[doc::MethodDoc] {
let op = copy op;
do docs.map |doc| {
doc::MethodDoc {
2013-01-30 20:52:31 -06:00
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),
2013-01-30 15:14:35 -06:00
.. copy *doc
}
2012-01-31 22:42:06 -06:00
}
}
fn fold_impl(
fold: &fold::Fold<NominalOp<Op>>,
2013-01-30 21:32:36 -06:00
doc: doc::ImplDoc
) -> doc::ImplDoc {
2012-02-17 17:59:57 -06:00
let doc = fold::default_seq_fold_impl(fold, doc);
doc::ImplDoc {
2013-01-30 20:52:31 -06:00
methods: apply_to_methods(copy fold.ctxt, copy doc.methods),
2012-09-04 15:29:32 -05:00
.. doc
2012-02-02 00:41:41 -06:00
}
}
#[cfg(test)]
mod test {
use core::prelude::*;
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;
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);
2013-06-10 06:03:16 -05:00
(mk_pass(~"", |s| s.trim().to_owned() ).f)(srv.clone(), doc)
}
}
#[test]
fn should_execute_op_on_enum_brief() {
let doc = mk_doc(~"#[doc = \" a \"] enum a { b }");
assert_eq!(doc.cratemod().enums()[0].brief(), Some(~"a"));
}
#[test]
fn should_execute_op_on_enum_desc() {
let doc = mk_doc(~"#[doc = \" a \"] enum a { b }");
assert_eq!(doc.cratemod().enums()[0].desc(), Some(~"a"));
}
#[test]
fn should_execute_op_on_variant_desc() {
let doc = 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 = mk_doc(
~"#[doc = \" a \"] trait i { fn a(); }");
assert_eq!(doc.cratemod().traits()[0].brief(), Some(~"a"));
}
#[test]
fn should_execute_op_on_trait_desc() {
let doc = mk_doc(
~"#[doc = \" a \"] trait i { fn a(); }");
assert_eq!(doc.cratemod().traits()[0].desc(), Some(~"a"));
}
2012-01-31 22:42:06 -06:00
#[test]
fn should_execute_op_on_trait_method_brief() {
let doc = mk_doc(
~"trait i { #[doc = \" a \"] fn a(); }");
assert!(doc.cratemod().traits()[0].methods[0].brief == Some(~"a"));
}
2012-01-31 22:42:06 -06:00
#[test]
fn should_execute_op_on_trait_method_desc() {
let doc = mk_doc(
~"trait i { #[doc = \" a \"] fn a(); }");
assert!(doc.cratemod().traits()[0].methods[0].desc == Some(~"a"));
}
2012-01-31 22:42:06 -06:00
#[test]
fn should_execute_op_on_impl_brief() {
let doc = mk_doc(
~"#[doc = \" a \"] impl int { fn a() { } }");
assert_eq!(doc.cratemod().impls()[0].brief(), Some(~"a"));
}
2012-01-31 22:42:06 -06:00
#[test]
fn should_execute_op_on_impl_desc() {
let doc = mk_doc(
~"#[doc = \" a \"] impl int { fn a() { } }");
assert_eq!(doc.cratemod().impls()[0].desc(), Some(~"a"));
}
2012-02-02 00:41:41 -06:00
#[test]
fn should_execute_op_on_impl_method_brief() {
let doc = mk_doc(
~"impl int { #[doc = \" a \"] fn a() { } }");
assert!(doc.cratemod().impls()[0].methods[0].brief == Some(~"a"));
}
2012-02-02 00:41:41 -06:00
#[test]
fn should_execute_op_on_impl_method_desc() {
let doc = mk_doc(
~"impl int { #[doc = \" a \"] fn a() { } }");
assert!(doc.cratemod().impls()[0].methods[0].desc == Some(~"a"));
}
2012-03-09 13:47:31 -06:00
#[test]
fn should_execute_op_on_type_brief() {
let doc = mk_doc(
~"#[doc = \" a \"] type t = int;");
assert_eq!(doc.cratemod().types()[0].brief(), Some(~"a"));
}
2012-03-09 13:47:31 -06:00
#[test]
fn should_execute_op_on_type_desc() {
let doc = mk_doc(
~"#[doc = \" a \"] type t = int;");
assert_eq!(doc.cratemod().types()[0].desc(), Some(~"a"));
}
2012-03-09 13:47:31 -06:00
#[test]
fn should_execute_on_item_section_headers() {
let doc = mk_doc(
~"#[doc = \"\
# Header \n\
Body\"]\
fn a() { }");
assert!(doc.cratemod().fns()[0].sections()[0].header == ~"Header");
}
2012-03-09 13:47:31 -06:00
#[test]
fn should_execute_on_item_section_bodies() {
let doc = mk_doc(
~"#[doc = \"\
# Header\n\
Body \"]\
fn a() { }");
assert!(doc.cratemod().fns()[0].sections()[0].body == ~"Body");
}
2012-03-09 13:47:31 -06:00
#[test]
fn should_execute_on_trait_method_section_headers() {
let doc = mk_doc(
~"trait i {
#[doc = \"\
# Header \n\
Body\"]\
fn a(); }");
assert!(doc.cratemod().traits()[0].methods[0].sections[0].header
== ~"Header");
}
2012-03-09 13:47:31 -06:00
#[test]
fn should_execute_on_trait_method_section_bodies() {
let doc = 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 = 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 = mk_doc(
~"impl bool {
#[doc = \"\
# Header\n\
Body \"]\
fn a() { } }");
assert!(doc.cratemod().impls()[0].methods[0].sections[0].body ==
~"Body");
2012-01-31 20:32:37 -06:00
}
}