diff options
Diffstat (limited to 'src/librustdoc/fold.rs')
| -rw-r--r-- | src/librustdoc/fold.rs | 481 |
1 files changed, 88 insertions, 393 deletions
diff --git a/src/librustdoc/fold.rs b/src/librustdoc/fold.rs index 3e74916228f..ae74f4e37c3 100644 --- a/src/librustdoc/fold.rs +++ b/src/librustdoc/fold.rs @@ -1,4 +1,4 @@ -// Copyright 2012 The Rust Project Developers. See the COPYRIGHT +// Copyright 2012-2013 The Rust Project Developers. See the COPYRIGHT // file at the top-level directory of this distribution and at // http://rust-lang.org/COPYRIGHT. // @@ -8,397 +8,92 @@ // option. This file may not be copied, modified, or distributed // except according to those terms. - -use doc; -#[cfg(test)] use extract; -#[cfg(test)] use parse; - -pub struct Fold<T> { - ctxt: T, - fold_doc: FoldDoc<T>, - fold_crate: FoldCrate<T>, - fold_item: FoldItem<T>, - fold_mod: FoldMod<T>, - fold_nmod: FoldNmod<T>, - fold_fn: FoldFn<T>, - fold_static: FoldStatic<T>, - fold_enum: FoldEnum<T>, - fold_trait: FoldTrait<T>, - fold_impl: FoldImpl<T>, - fold_type: FoldType<T>, - fold_struct: FoldStruct<T> -} - -impl<T:Clone> Clone for Fold<T> { - fn clone(&self) -> Fold<T> { - Fold { - ctxt: self.ctxt.clone(), - fold_doc: self.fold_doc, - fold_crate: self.fold_crate, - fold_item: self.fold_item, - fold_mod: self.fold_mod, - fold_nmod: self.fold_nmod, - fold_fn: self.fold_fn, - fold_static: self.fold_static, - fold_enum: self.fold_enum, - fold_trait: self.fold_trait, - fold_impl: self.fold_impl, - fold_type: self.fold_type, - fold_struct: self.fold_struct - } - } -} - -type FoldDoc<T> = @fn(fold: &Fold<T>, doc: doc::Doc) -> doc::Doc; -type FoldCrate<T> = @fn(fold: &Fold<T>, doc: doc::CrateDoc) -> doc::CrateDoc; -type FoldItem<T> = @fn(fold: &Fold<T>, doc: doc::ItemDoc) -> doc::ItemDoc; -type FoldMod<T> = @fn(fold: &Fold<T>, doc: doc::ModDoc) -> doc::ModDoc; -type FoldNmod<T> = @fn(fold: &Fold<T>, doc: doc::NmodDoc) -> doc::NmodDoc; -type FoldFn<T> = @fn(fold: &Fold<T>, doc: doc::FnDoc) -> doc::FnDoc; -type FoldStatic<T> = @fn(fold: &Fold<T>, doc: doc::StaticDoc) -> doc::StaticDoc; -type FoldEnum<T> = @fn(fold: &Fold<T>, doc: doc::EnumDoc) -> doc::EnumDoc; -type FoldTrait<T> = @fn(fold: &Fold<T>, doc: doc::TraitDoc) -> doc::TraitDoc; -type FoldImpl<T> = @fn(fold: &Fold<T>, doc: doc::ImplDoc) -> doc::ImplDoc; -type FoldType<T> = @fn(fold: &Fold<T>, doc: doc::TyDoc) -> doc::TyDoc; -type FoldStruct<T> = @fn(fold: &Fold<T>, - doc: doc::StructDoc) -> doc::StructDoc; - -// This exists because fn types don't infer correctly as record -// initializers, but they do as function arguments -fn mk_fold<T>( - ctxt: T, - fold_doc: FoldDoc<T>, - fold_crate: FoldCrate<T>, - fold_item: FoldItem<T>, - fold_mod: FoldMod<T>, - fold_nmod: FoldNmod<T>, - fold_fn: FoldFn<T>, - fold_static: FoldStatic<T>, - fold_enum: FoldEnum<T>, - fold_trait: FoldTrait<T>, - fold_impl: FoldImpl<T>, - fold_type: FoldType<T>, - fold_struct: FoldStruct<T> -) -> Fold<T> { - Fold { - ctxt: ctxt, - fold_doc: fold_doc, - fold_crate: fold_crate, - fold_item: fold_item, - fold_mod: fold_mod, - fold_nmod: fold_nmod, - fold_fn: fold_fn, - fold_static: fold_static, - fold_enum: fold_enum, - fold_trait: fold_trait, - fold_impl: fold_impl, - fold_type: fold_type, - fold_struct: fold_struct - } -} - -pub fn default_any_fold<T:Clone>(ctxt: T) -> Fold<T> { - mk_fold( - ctxt, - |f, d| default_seq_fold_doc(f, d), - |f, d| default_seq_fold_crate(f, d), - |f, d| default_seq_fold_item(f, d), - |f, d| default_any_fold_mod(f, d), - |f, d| default_any_fold_nmod(f, d), - |f, d| default_seq_fold_fn(f, d), - |f, d| default_seq_fold_static(f, d), - |f, d| default_seq_fold_enum(f, d), - |f, d| default_seq_fold_trait(f, d), - |f, d| default_seq_fold_impl(f, d), - |f, d| default_seq_fold_type(f, d), - |f, d| default_seq_fold_struct(f, d) - ) -} - -pub fn default_seq_fold<T:Clone>(ctxt: T) -> Fold<T> { - mk_fold( - ctxt, - |f, d| default_seq_fold_doc(f, d), - |f, d| default_seq_fold_crate(f, d), - |f, d| default_seq_fold_item(f, d), - |f, d| default_seq_fold_mod(f, d), - |f, d| default_seq_fold_nmod(f, d), - |f, d| default_seq_fold_fn(f, d), - |f, d| default_seq_fold_static(f, d), - |f, d| default_seq_fold_enum(f, d), - |f, d| default_seq_fold_trait(f, d), - |f, d| default_seq_fold_impl(f, d), - |f, d| default_seq_fold_type(f, d), - |f, d| default_seq_fold_struct(f, d) - ) -} - -pub fn default_par_fold<T:Clone>(ctxt: T) -> Fold<T> { - mk_fold( - ctxt, - |f, d| default_seq_fold_doc(f, d), - |f, d| default_seq_fold_crate(f, d), - |f, d| default_seq_fold_item(f, d), - |f, d| default_par_fold_mod(f, d), - |f, d| default_par_fold_nmod(f, d), - |f, d| default_seq_fold_fn(f, d), - |f, d| default_seq_fold_static(f, d), - |f, d| default_seq_fold_enum(f, d), - |f, d| default_seq_fold_trait(f, d), - |f, d| default_seq_fold_impl(f, d), - |f, d| default_seq_fold_type(f, d), - |f, d| default_seq_fold_struct(f, d) - ) -} - -pub fn default_seq_fold_doc<T>(fold: &Fold<T>, doc: doc::Doc) -> doc::Doc { - doc::Doc { - pages: do doc.pages.iter().map |page| { - match (*page).clone() { - doc::CratePage(doc) => { - doc::CratePage((fold.fold_crate)(fold, doc)) - } - doc::ItemPage(doc) => { - doc::ItemPage(fold_ItemTag(fold, doc)) - } - } - }.collect(), - .. doc - } -} - -pub fn default_seq_fold_crate<T>( - fold: &Fold<T>, - doc: doc::CrateDoc -) -> doc::CrateDoc { - doc::CrateDoc { - topmod: (fold.fold_mod)(fold, doc.topmod.clone()) - } -} - -pub fn default_seq_fold_item<T>( - _fold: &Fold<T>, - doc: doc::ItemDoc -) -> doc::ItemDoc { - doc -} - -pub fn default_any_fold_mod<T:Clone>( - fold: &Fold<T>, - doc: doc::ModDoc -) -> doc::ModDoc { - doc::ModDoc { - item: (fold.fold_item)(fold, doc.item.clone()), - items: doc.items.iter().map(|ItemTag| { - fold_ItemTag(fold, (*ItemTag).clone()) - }).collect(), - .. doc - } -} - -pub fn default_seq_fold_mod<T>( - fold: &Fold<T>, - doc: doc::ModDoc -) -> doc::ModDoc { - doc::ModDoc { - item: (fold.fold_item)(fold, doc.item.clone()), - items: doc.items.iter().map(|ItemTag| { - fold_ItemTag(fold, (*ItemTag).clone()) - }).collect(), - .. doc - } -} - -pub fn default_par_fold_mod<T:Clone>( - fold: &Fold<T>, - doc: doc::ModDoc -) -> doc::ModDoc { - doc::ModDoc { - item: (fold.fold_item)(fold, doc.item.clone()), - items: doc.items.iter().map(|ItemTag| { - fold_ItemTag(fold, (*ItemTag).clone()) - }).collect(), - .. doc - } -} - -pub fn default_any_fold_nmod<T:Clone>( - fold: &Fold<T>, - doc: doc::NmodDoc -) -> doc::NmodDoc { - doc::NmodDoc { - item: (fold.fold_item)(fold, doc.item.clone()), - fns: doc.fns.iter().map(|FnDoc| { - (fold.fold_fn)(fold, (*FnDoc).clone()) - }).collect(), - .. doc - } -} - -pub fn default_seq_fold_nmod<T>( - fold: &Fold<T>, - doc: doc::NmodDoc -) -> doc::NmodDoc { - doc::NmodDoc { - item: (fold.fold_item)(fold, doc.item.clone()), - fns: doc.fns.iter().map(|FnDoc| { - (fold.fold_fn)(fold, (*FnDoc).clone()) - }).collect(), - .. doc - } -} - -pub fn default_par_fold_nmod<T:Clone>( - fold: &Fold<T>, - doc: doc::NmodDoc -) -> doc::NmodDoc { - doc::NmodDoc { - item: (fold.fold_item)(fold, doc.item.clone()), - fns: doc.fns.iter().map(|FnDoc| { - (fold.fold_fn)(fold, (*FnDoc).clone()) - }).collect(), - .. doc - } -} - -pub fn fold_ItemTag<T>(fold: &Fold<T>, doc: doc::ItemTag) -> doc::ItemTag { - match doc { - doc::ModTag(ModDoc) => { - doc::ModTag((fold.fold_mod)(fold, ModDoc)) - } - doc::NmodTag(nModDoc) => { - doc::NmodTag((fold.fold_nmod)(fold, nModDoc)) - } - doc::FnTag(FnDoc) => { - doc::FnTag((fold.fold_fn)(fold, FnDoc)) - } - doc::StaticTag(StaticDoc) => { - doc::StaticTag((fold.fold_static)(fold, StaticDoc)) - } - doc::EnumTag(EnumDoc) => { - doc::EnumTag((fold.fold_enum)(fold, EnumDoc)) - } - doc::TraitTag(TraitDoc) => { - doc::TraitTag((fold.fold_trait)(fold, TraitDoc)) - } - doc::ImplTag(ImplDoc) => { - doc::ImplTag((fold.fold_impl)(fold, ImplDoc)) - } - doc::TyTag(TyDoc) => { - doc::TyTag((fold.fold_type)(fold, TyDoc)) - } - doc::StructTag(StructDoc) => { - doc::StructTag((fold.fold_struct)(fold, StructDoc)) - } - } -} - -pub fn default_seq_fold_fn<T>( - fold: &Fold<T>, - doc: doc::FnDoc -) -> doc::FnDoc { - doc::SimpleItemDoc { - item: (fold.fold_item)(fold, doc.item.clone()), - .. doc - } -} - -pub fn default_seq_fold_static<T>( - fold: &Fold<T>, - doc: doc::StaticDoc -) -> doc::StaticDoc { - doc::SimpleItemDoc { - item: (fold.fold_item)(fold, doc.item.clone()), - .. doc - } -} - -pub fn default_seq_fold_enum<T>( - fold: &Fold<T>, - doc: doc::EnumDoc -) -> doc::EnumDoc { - doc::EnumDoc { - item: (fold.fold_item)(fold, doc.item.clone()), - .. doc - } -} - -pub fn default_seq_fold_trait<T>( - fold: &Fold<T>, - doc: doc::TraitDoc -) -> doc::TraitDoc { - doc::TraitDoc { - item: (fold.fold_item)(fold, doc.item.clone()), - .. doc - } -} - -pub fn default_seq_fold_impl<T>( - fold: &Fold<T>, - doc: doc::ImplDoc -) -> doc::ImplDoc { - doc::ImplDoc { - item: (fold.fold_item)(fold, doc.item.clone()), - .. doc - } -} - -pub fn default_seq_fold_type<T>( - fold: &Fold<T>, - doc: doc::TyDoc -) -> doc::TyDoc { - doc::SimpleItemDoc { - item: (fold.fold_item)(fold, doc.item.clone()), - .. doc - } -} - -pub fn default_seq_fold_struct<T>( - fold: &Fold<T>, - doc: doc::StructDoc -) -> doc::StructDoc { - doc::StructDoc { - item: (fold.fold_item)(fold, doc.item.clone()), - .. doc +use std; +use clean::*; +use std::iter::Extendable; + +pub trait DocFolder { + fn fold_item(&mut self, item: Item) -> Option<Item> { + self.fold_item_recur(item) + } + + /// don't override! + fn fold_item_recur(&mut self, item: Item) -> Option<Item> { + use std::util::swap; + let Item { attrs, name, source, visibility, id, inner } = item; + let inner = inner; + let c = |x| self.fold_item(x); + let inner = match inner { + StructItem(i) => { + let mut i = i; + let mut foo = ~[]; swap(&mut foo, &mut i.fields); + i.fields.extend(&mut foo.move_iter().filter_map(|x| self.fold_item(x))); + StructItem(i) + }, + ModuleItem(i) => { + ModuleItem(self.fold_mod(i)) + }, + EnumItem(i) => { + let mut i = i; + let mut foo = ~[]; swap(&mut foo, &mut i.variants); + i.variants.extend(&mut foo.move_iter().filter_map(|x| self.fold_item(x))); + EnumItem(i) + }, + TraitItem(i) => { + fn vtrm<T: DocFolder>(this: &mut T, trm: TraitMethod) -> Option<TraitMethod> { + match trm { + Required(it) => { + match this.fold_item(it) { + Some(x) => return Some(Required(x)), + None => return None, + } + }, + Provided(it) => { + match this.fold_item(it) { + Some(x) => return Some(Provided(x)), + None => return None, + } + }, + } + } + let mut i = i; + let mut foo = ~[]; swap(&mut foo, &mut i.methods); + i.methods.extend(&mut foo.move_iter().filter_map(|x| vtrm(self, x))); + TraitItem(i) + }, + ImplItem(i) => { + let mut i = i; + let mut foo = ~[]; swap(&mut foo, &mut i.methods); + i.methods.extend(&mut foo.move_iter().filter_map(|x| self.fold_item(x))); + ImplItem(i) + }, + VariantItem(i) => { + let i2 = i.clone(); // this clone is small + match i.kind { + StructVariant(j) => { + let mut j = j; + let mut foo = ~[]; swap(&mut foo, &mut j.fields); + j.fields.extend(&mut foo.move_iter().filter_map(c)); + VariantItem(Variant {kind: StructVariant(j), ..i2}) + }, + _ => VariantItem(i2) + } + }, + x => x + }; + + Some(Item { attrs: attrs, name: name, source: source, inner: inner, + visibility: visibility, id: id }) + } + + fn fold_mod(&mut self, m: Module) -> Module { + Module { items: m.items.move_iter().filter_map(|i| self.fold_item(i)).collect() } + } + + fn fold_crate(&mut self, mut c: Crate) -> Crate { + c.module = match std::util::replace(&mut c.module, None) { + Some(module) => self.fold_item(module), None => None + }; + return c; } } - -#[test] -fn default_fold_should_produce_same_doc() { - let source = @"mod a { fn b() { } mod c { fn d() { } } }"; - let ast = parse::from_str(source); - let doc = extract::extract(ast, ~""); - let fld = default_seq_fold(()); - let folded = (fld.fold_doc)(&fld, doc.clone()); - assert_eq!(doc, folded); -} - -#[test] -fn default_fold_should_produce_same_statics() { - let source = @"static a: int = 0;"; - let ast = parse::from_str(source); - let doc = extract::extract(ast, ~""); - let fld = default_seq_fold(()); - let folded = (fld.fold_doc)(&fld, doc.clone()); - assert_eq!(doc, folded); -} - -#[test] -fn default_fold_should_produce_same_enums() { - let source = @"enum a { b }"; - let ast = parse::from_str(source); - let doc = extract::extract(ast, ~""); - let fld = default_seq_fold(()); - let folded = (fld.fold_doc)(&fld, doc.clone()); - assert_eq!(doc, folded); -} - -#[test] -fn default_parallel_fold_should_produce_same_doc() { - let source = @"mod a { fn b() { } mod c { fn d() { } } }"; - let ast = parse::from_str(source); - let doc = extract::extract(ast, ~""); - let fld = default_par_fold(()); - let folded = (fld.fold_doc)(&fld, doc.clone()); - assert_eq!(doc, folded); -} |
