2012-01-17 17:44:32 -08:00
|
|
|
#[doc(
|
|
|
|
brief = "Attribute parsing",
|
|
|
|
desc =
|
|
|
|
"The attribute parser provides methods for pulling documentation out of \
|
|
|
|
an AST's attributes."
|
|
|
|
)];
|
|
|
|
|
2012-01-16 13:22:25 -08:00
|
|
|
import rustc::syntax::ast;
|
2012-01-16 20:00:46 -08:00
|
|
|
import rustc::front::attr;
|
2012-01-17 19:43:29 +01:00
|
|
|
import core::tuple;
|
2012-01-16 13:22:25 -08:00
|
|
|
|
2012-01-18 18:51:04 -08:00
|
|
|
export crate_attrs, mod_attrs, fn_attrs, arg_attrs;
|
2012-01-18 18:02:56 -08:00
|
|
|
export parse_crate, parse_mod, parse_fn;
|
2012-01-18 14:20:26 -08:00
|
|
|
|
|
|
|
type crate_attrs = {
|
2012-01-23 15:15:57 -08:00
|
|
|
name: option<str>,
|
|
|
|
desc: option<str>
|
2012-01-18 14:20:26 -08:00
|
|
|
};
|
2012-01-15 21:50:55 -08:00
|
|
|
|
2012-01-18 18:02:56 -08:00
|
|
|
type mod_attrs = {
|
|
|
|
brief: option<str>,
|
|
|
|
desc: option<str>
|
|
|
|
};
|
|
|
|
|
2012-01-16 16:27:28 -08:00
|
|
|
type fn_attrs = {
|
|
|
|
brief: option<str>,
|
|
|
|
desc: option<str>,
|
|
|
|
args: [arg_attrs],
|
|
|
|
return: option<str>
|
|
|
|
};
|
|
|
|
|
|
|
|
type arg_attrs = {
|
2012-01-16 00:15:03 -08:00
|
|
|
name: str,
|
2012-01-16 16:27:28 -08:00
|
|
|
desc: str
|
|
|
|
};
|
|
|
|
|
2012-01-16 20:00:46 -08:00
|
|
|
fn doc_meta(
|
|
|
|
attrs: [ast::attribute]
|
|
|
|
) -> option<@ast::meta_item> {
|
2012-01-18 14:06:22 -08:00
|
|
|
|
|
|
|
#[doc =
|
|
|
|
"Given a vec of attributes, extract the meta_items contained in the \
|
|
|
|
doc attribute"];
|
|
|
|
|
2012-01-16 20:00:46 -08:00
|
|
|
let doc_attrs = attr::find_attrs_by_name(attrs, "doc");
|
|
|
|
let doc_metas = attr::attr_metas(doc_attrs);
|
|
|
|
if vec::is_not_empty(doc_metas) {
|
|
|
|
if vec::len(doc_metas) != 1u {
|
|
|
|
#warn("ignoring %u doc attributes", vec::len(doc_metas) - 1u);
|
|
|
|
}
|
|
|
|
some(doc_metas[0])
|
|
|
|
} else {
|
|
|
|
none
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-18 14:20:26 -08:00
|
|
|
fn parse_crate(attrs: [ast::attribute]) -> crate_attrs {
|
|
|
|
let link_metas = attr::find_linkage_metas(attrs);
|
2012-01-23 15:15:57 -08:00
|
|
|
let attr_metas = attr::attr_metas(attrs);
|
2012-01-18 14:20:26 -08:00
|
|
|
|
|
|
|
{
|
2012-01-23 15:15:57 -08:00
|
|
|
name: attr::meta_item_value_from_list(link_metas, "name"),
|
|
|
|
desc: attr::meta_item_value_from_list(attr_metas, "desc")
|
2012-01-18 14:20:26 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[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);
|
|
|
|
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);
|
|
|
|
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);
|
|
|
|
assert attrs.name == none;
|
|
|
|
}
|
|
|
|
|
2012-01-23 15:15:57 -08:00
|
|
|
#[test]
|
|
|
|
fn should_extract_crate_desc() {
|
|
|
|
let source = "#[desc = \"Teddybears\"]";
|
|
|
|
let attrs = test::parse_attributes(source);
|
|
|
|
let attrs = parse_crate(attrs);
|
|
|
|
assert attrs.desc == some("Teddybears");
|
|
|
|
}
|
|
|
|
|
2012-01-18 18:02:56 -08:00
|
|
|
fn parse_mod(attrs: [ast::attribute]) -> mod_attrs {
|
|
|
|
parse_short_doc_or(
|
|
|
|
attrs,
|
|
|
|
{|desc|
|
|
|
|
{
|
|
|
|
brief: none,
|
|
|
|
desc: desc
|
|
|
|
}
|
|
|
|
},
|
|
|
|
parse_mod_long_doc
|
|
|
|
)
|
|
|
|
}
|
2012-01-15 22:07:23 -08:00
|
|
|
|
2012-01-18 18:02:56 -08:00
|
|
|
fn parse_mod_long_doc(
|
|
|
|
_items: [@ast::meta_item],
|
|
|
|
brief: option<str>,
|
|
|
|
desc: option<str>
|
|
|
|
) -> mod_attrs {
|
|
|
|
{
|
|
|
|
brief: brief,
|
|
|
|
desc: desc
|
|
|
|
}
|
|
|
|
}
|
2012-01-16 20:00:46 -08:00
|
|
|
|
2012-01-18 18:02:56 -08:00
|
|
|
#[test]
|
|
|
|
fn parse_mod_should_handle_undocumented_mods() {
|
|
|
|
let source = "";
|
|
|
|
let attrs = test::parse_attributes(source);
|
|
|
|
let attrs = parse_mod(attrs);
|
|
|
|
assert attrs.brief == none;
|
|
|
|
assert attrs.desc == none;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parse_mod_should_parse_simple_doc_attributes() {
|
|
|
|
let source = "#[doc = \"basic\"]";
|
|
|
|
let attrs = test::parse_attributes(source);
|
|
|
|
let attrs = parse_mod(attrs);
|
|
|
|
assert attrs.desc == some("basic");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parse_mod_should_parse_the_brief_description() {
|
|
|
|
let source = "#[doc(brief = \"short\")]";
|
|
|
|
let attrs = test::parse_attributes(source);
|
|
|
|
let attrs = parse_mod(attrs);
|
|
|
|
assert attrs.brief == some("short");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parse_mod_should_parse_the_long_description() {
|
|
|
|
let source = "#[doc(desc = \"description\")]";
|
|
|
|
let attrs = test::parse_attributes(source);
|
|
|
|
let attrs = parse_mod(attrs);
|
|
|
|
assert attrs.desc == some("description");
|
|
|
|
}
|
|
|
|
|
|
|
|
fn parse_short_doc_or<T>(
|
|
|
|
attrs: [ast::attribute],
|
|
|
|
handle_short: fn&(
|
|
|
|
short_desc: option<str>
|
|
|
|
) -> T,
|
|
|
|
parse_long: fn&(
|
|
|
|
doc_items: [@ast::meta_item],
|
|
|
|
brief: option<str>,
|
|
|
|
desc: option<str>
|
|
|
|
) -> T
|
|
|
|
) -> T {
|
|
|
|
alt doc_meta(attrs) {
|
2012-01-16 20:00:46 -08:00
|
|
|
some(meta) {
|
|
|
|
alt attr::get_meta_item_value_str(meta) {
|
2012-01-18 18:02:56 -08:00
|
|
|
some(desc) { handle_short(some(desc)) }
|
2012-01-18 22:37:22 -08:00
|
|
|
none {
|
2012-01-16 20:00:46 -08:00
|
|
|
alt attr::get_meta_item_list(meta) {
|
|
|
|
some(list) {
|
2012-01-18 18:02:56 -08:00
|
|
|
let brief = attr::meta_item_value_from_list(list, "brief");
|
|
|
|
let desc = attr::meta_item_value_from_list(list, "desc");
|
|
|
|
parse_long(list, brief, desc)
|
2012-01-16 20:00:46 -08:00
|
|
|
}
|
2012-01-18 22:37:22 -08:00
|
|
|
none {
|
2012-01-18 18:02:56 -08:00
|
|
|
handle_short(none)
|
2012-01-16 20:00:46 -08:00
|
|
|
}
|
|
|
|
}
|
2012-01-15 22:07:23 -08:00
|
|
|
}
|
|
|
|
}
|
2012-01-16 20:00:46 -08:00
|
|
|
}
|
2012-01-18 22:37:22 -08:00
|
|
|
none {
|
2012-01-18 18:02:56 -08:00
|
|
|
handle_short(none)
|
2012-01-16 20:00:46 -08:00
|
|
|
}
|
2012-01-18 18:02:56 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn parse_fn(
|
|
|
|
attrs: [ast::attribute]
|
|
|
|
) -> fn_attrs {
|
|
|
|
|
|
|
|
parse_short_doc_or(
|
|
|
|
attrs,
|
|
|
|
{|desc|
|
|
|
|
{
|
|
|
|
brief: none,
|
|
|
|
desc: desc,
|
|
|
|
args: [],
|
|
|
|
return: none
|
|
|
|
}
|
|
|
|
},
|
|
|
|
parse_fn_long_doc
|
|
|
|
)
|
2012-01-16 20:00:46 -08:00
|
|
|
}
|
|
|
|
|
2012-01-18 18:02:56 -08:00
|
|
|
fn parse_fn_long_doc(
|
|
|
|
items: [@ast::meta_item],
|
|
|
|
brief: option<str>,
|
|
|
|
desc: option<str>
|
2012-01-16 16:27:28 -08:00
|
|
|
) -> fn_attrs {
|
2012-01-16 21:12:08 -08:00
|
|
|
let return = attr::meta_item_value_from_list(items, "return");
|
2012-01-16 20:00:46 -08:00
|
|
|
|
2012-01-16 21:12:08 -08:00
|
|
|
let args = alt attr::meta_item_list_from_list(items, "args") {
|
2012-01-16 20:00:46 -08:00
|
|
|
some(items) {
|
|
|
|
vec::filter_map(items) {|item|
|
2012-01-16 21:12:08 -08:00
|
|
|
option::map(attr::name_value_str_pair(item)) { |pair|
|
2012-01-16 20:00:46 -08:00
|
|
|
{
|
2012-01-17 19:14:05 +01:00
|
|
|
name: tuple::first(pair),
|
|
|
|
desc: tuple::second(pair)
|
2012-01-15 21:50:55 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-01-16 20:00:46 -08:00
|
|
|
}
|
2012-01-18 22:37:22 -08:00
|
|
|
none { [] }
|
2012-01-16 20:00:46 -08:00
|
|
|
};
|
2012-01-15 21:50:55 -08:00
|
|
|
|
2012-01-16 16:27:28 -08:00
|
|
|
{
|
2012-01-16 16:01:33 -08:00
|
|
|
brief: brief,
|
2012-01-15 21:50:55 -08:00
|
|
|
desc: desc,
|
2012-01-16 20:00:46 -08:00
|
|
|
args: args,
|
2012-01-16 16:27:28 -08:00
|
|
|
return: return
|
|
|
|
}
|
2012-01-15 21:50:55 -08:00
|
|
|
}
|
2012-01-15 22:47:23 -08:00
|
|
|
|
2012-01-18 18:02:56 -08:00
|
|
|
#[test]
|
|
|
|
fn parse_fn_should_handle_undocumented_functions() {
|
|
|
|
let source = "";
|
|
|
|
let attrs = test::parse_attributes(source);
|
|
|
|
let attrs = parse_fn(attrs);
|
|
|
|
assert attrs.brief == none;
|
|
|
|
assert attrs.desc == none;
|
|
|
|
assert attrs.return == none;
|
|
|
|
assert vec::len(attrs.args) == 0u;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parse_fn_should_parse_simple_doc_attributes() {
|
|
|
|
let source = "#[doc = \"basic\"]";
|
|
|
|
let attrs = test::parse_attributes(source);
|
|
|
|
let attrs = parse_fn(attrs);
|
|
|
|
assert attrs.desc == some("basic");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parse_fn_should_parse_the_brief_description() {
|
|
|
|
let source = "#[doc(brief = \"short\")]";
|
|
|
|
let attrs = test::parse_attributes(source);
|
|
|
|
let attrs = parse_fn(attrs);
|
|
|
|
assert attrs.brief == some("short");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parse_fn_should_parse_the_long_description() {
|
|
|
|
let source = "#[doc(desc = \"description\")]";
|
|
|
|
let attrs = test::parse_attributes(source);
|
|
|
|
let attrs = parse_fn(attrs);
|
|
|
|
assert attrs.desc == some("description");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parse_fn_should_parse_the_return_value_description() {
|
|
|
|
let source = "#[doc(return = \"return value\")]";
|
|
|
|
let attrs = test::parse_attributes(source);
|
|
|
|
let attrs = parse_fn(attrs);
|
|
|
|
assert attrs.return == some("return value");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parse_fn_should_parse_the_argument_descriptions() {
|
|
|
|
let source = "#[doc(args(a = \"arg a\", b = \"arg b\"))]";
|
|
|
|
let attrs = test::parse_attributes(source);
|
|
|
|
let attrs = parse_fn(attrs);
|
|
|
|
assert attrs.args[0] == {name: "a", desc: "arg a"};
|
|
|
|
assert attrs.args[1] == {name: "b", desc: "arg b"};
|
|
|
|
}
|
|
|
|
|
2012-01-15 22:47:23 -08:00
|
|
|
#[cfg(test)]
|
2012-01-18 14:20:26 -08:00
|
|
|
mod test {
|
2012-01-15 22:47:23 -08:00
|
|
|
|
|
|
|
fn parse_attributes(source: str) -> [ast::attribute] {
|
|
|
|
import rustc::syntax::parse::parser;
|
2012-01-16 20:00:46 -08:00
|
|
|
// FIXME: Uncommenting this results in rustc bugs
|
|
|
|
//import rustc::syntax::codemap;
|
|
|
|
import rustc::driver::diagnostic;
|
2012-01-15 22:47:23 -08:00
|
|
|
|
2012-01-16 20:00:46 -08:00
|
|
|
let cm = rustc::syntax::codemap::new_codemap();
|
2012-01-15 22:47:23 -08:00
|
|
|
let parse_sess = @{
|
|
|
|
cm: cm,
|
|
|
|
mutable next_id: 0,
|
|
|
|
diagnostic: diagnostic::mk_handler(cm, none)
|
|
|
|
};
|
|
|
|
let parser = parser::new_parser_from_source_str(
|
|
|
|
parse_sess, [], "-", source);
|
|
|
|
|
|
|
|
parser::parse_outer_attributes(parser)
|
|
|
|
}
|
2012-01-17 19:14:05 +01:00
|
|
|
}
|