// 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 or the MIT license // , 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 astsrv; use doc::ItemUtils; use doc; use fold::Fold; use fold; use pass::Pass; use util::NominalOp; use std::cell::Cell; pub fn mk_pass(name: ~str, op: @fn(&str) -> ~str) -> Pass { let op = Cell::new(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, s: &Option<~str>) -> Option<~str> { s.map(|s| (op.op)(*s) ) } fn fold_item( fold: &fold::Fold>, 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, 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>, 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>, 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, 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>, 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 } } #[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; 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| 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")); } #[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")); } #[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")); } #[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")); } #[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")); } #[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")); } #[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")); } #[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")); } #[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")); } #[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"); } #[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"); } #[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"); } #[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"); } }