rust/src/librustdoc/attr_parser.rs

164 lines
4.2 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.
/*!
Attribute parsing
The attribute parser provides methods for pulling documentation out of
an AST's attributes.
*/
2012-01-17 17:44:32 -08:00
use core::str;
use core::tuple;
use core::vec;
2012-09-05 10:41:47 -07:00
use syntax::ast;
use syntax::attr;
use syntax::codemap;
use syntax;
2012-11-19 18:00:12 -08:00
pub type CrateAttrs = {
2012-08-20 12:23:37 -07:00
name: Option<~str>
};
2012-01-25 17:31:50 -08:00
#[cfg(test)]
mod test {
#[legacy_exports];
use syntax::ast;
use syntax;
fn parse_attributes(+source: ~str) -> ~[ast::attribute] {
use syntax::parse;
use syntax::parse::parser;
use syntax::parse::attr::parser_attr;
use syntax::codemap;
use syntax::diagnostic;
2012-01-25 17:31:50 -08:00
2012-08-20 12:23:37 -07:00
let parse_sess = syntax::parse::new_parse_sess(None);
let parser = parse::new_parser_from_source_str(
parse_sess, ~[], ~"-", codemap::FssNone, @source);
2012-01-25 17:31:50 -08:00
2012-05-24 13:44:42 -07:00
parser.parse_outer_attributes()
2012-01-25 17:31:50 -08:00
}
}
fn doc_metas(
2012-11-19 18:48:46 -08:00
+attrs: ~[ast::attribute]
) -> ~[@ast::meta_item] {
let doc_attrs = attr::find_attrs_by_name(attrs, ~"doc");
let doc_metas = do doc_attrs.map |attr| {
attr::attr_meta(attr::desugar_doc_attr(attr))
};
return doc_metas;
2012-01-16 20:00:46 -08:00
}
2012-11-19 18:48:46 -08:00
pub fn parse_crate(+attrs: ~[ast::attribute]) -> CrateAttrs {
let link_metas = attr::find_linkage_metas(attrs);
{
2012-07-18 16:18:02 -07:00
name: attr::last_meta_item_value_str_by_name(link_metas, ~"name")
}
}
#[test]
fn should_extract_crate_name_from_link_attribute() {
let source = ~"#[link(name = \"snuggles\")]";
let attrs = test::parse_attributes(source);
let attrs = parse_crate(attrs);
2012-08-20 12:23:37 -07:00
assert attrs.name == Some(~"snuggles");
}
#[test]
fn should_not_extract_crate_name_if_no_link_attribute() {
let source = ~"";
let attrs = test::parse_attributes(source);
let attrs = parse_crate(attrs);
2012-08-20 12:23:37 -07:00
assert attrs.name == None;
}
#[test]
fn should_not_extract_crate_name_if_no_name_value_in_link_attribute() {
let source = ~"#[link(whatever)]";
let attrs = test::parse_attributes(source);
let attrs = parse_crate(attrs);
2012-08-20 12:23:37 -07:00
assert attrs.name == None;
}
2012-11-19 18:48:46 -08:00
pub fn parse_desc(+attrs: ~[ast::attribute]) -> Option<~str> {
2012-10-22 20:56:00 +01:00
let doc_strs = do doc_metas(attrs).filter_map |meta| {
attr::get_meta_item_value_str(*meta)
};
if doc_strs.is_empty() {
None
} else {
Some(str::connect(doc_strs, "\n"))
2012-03-09 18:12:15 -08:00
}
2012-01-18 18:02:56 -08:00
}
2012-01-18 18:02:56 -08:00
#[test]
2012-03-09 18:12:15 -08:00
fn parse_desc_should_handle_undocumented_mods() {
let source = ~"";
2012-01-18 18:02:56 -08:00
let attrs = test::parse_attributes(source);
2012-03-09 18:12:15 -08:00
let attrs = parse_desc(attrs);
2012-08-20 12:23:37 -07:00
assert attrs == None;
2012-01-18 18:02:56 -08:00
}
#[test]
2012-03-09 18:12:15 -08:00
fn parse_desc_should_parse_simple_doc_attributes() {
let source = ~"#[doc = \"basic\"]";
2012-01-18 18:02:56 -08:00
let attrs = test::parse_attributes(source);
2012-03-09 18:12:15 -08:00
let attrs = parse_desc(attrs);
2012-08-20 12:23:37 -07:00
assert attrs == Some(~"basic");
2012-01-18 18:02:56 -08:00
}
2012-11-19 18:48:46 -08:00
pub fn parse_hidden(+attrs: ~[ast::attribute]) -> bool {
do doc_metas(attrs).find |meta| {
match attr::get_meta_item_list(*meta) {
2012-08-20 12:23:37 -07:00
Some(metas) => {
let hiddens = attr::find_meta_items_by_name(metas, ~"hidden");
2012-03-09 18:12:15 -08:00
vec::is_not_empty(hiddens)
2012-02-17 16:52:27 -08:00
}
2012-08-20 12:23:37 -07:00
None => false
2012-02-17 16:52:27 -08:00
}
}.is_some()
2012-02-17 16:52:27 -08:00
}
2012-01-18 18:02:56 -08:00
2012-03-07 14:49:52 -08:00
#[test]
2012-09-19 18:30:14 -07:00
fn should_parse_hidden_attribute() {
let source = ~"#[doc(hidden)]";
2012-03-07 14:49:52 -08:00
let attrs = test::parse_attributes(source);
assert parse_hidden(attrs) == true;
}
#[test]
fn should_parse_hidden_attribute_with_other_docs() {
let source = ~"#[doc = \"foo\"] #[doc(hidden)] #[doc = \"foo\"]";
let attrs = test::parse_attributes(source);
assert parse_hidden(attrs) == true;
}
2012-03-07 14:49:52 -08:00
#[test]
2012-09-19 18:30:14 -07:00
fn should_not_parse_non_hidden_attribute() {
let source = ~"#[doc = \"\"]";
2012-03-07 14:49:52 -08:00
let attrs = test::parse_attributes(source);
assert parse_hidden(attrs) == false;
}
2012-10-22 20:56:00 +01:00
#[test]
fn should_concatenate_multiple_doc_comments() {
let source = ~"/// foo\n/// bar";
let desc = parse_desc(test::parse_attributes(source));
assert desc == Some(~"foo\nbar");
}