2012-12-03 16:48:01 -08:00
|
|
|
// 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.
|
|
|
|
|
2012-07-04 22:53:12 +01:00
|
|
|
/*!
|
2012-09-20 16:09:46 -07:00
|
|
|
Pulls a brief description out of a long description.
|
|
|
|
|
|
|
|
If the first paragraph of a long description is short enough then it
|
|
|
|
is interpreted as the brief description.
|
|
|
|
*/
|
2012-01-24 17:19:27 -08:00
|
|
|
|
2013-05-17 15:28:44 -07:00
|
|
|
use core::prelude::*;
|
|
|
|
|
2012-12-23 17:41:37 -05:00
|
|
|
use astsrv;
|
2012-09-18 16:48:40 -07:00
|
|
|
use doc::ItemUtils;
|
2012-12-23 17:41:37 -05:00
|
|
|
use doc;
|
2012-12-05 15:06:54 -08:00
|
|
|
use fold::Fold;
|
2012-12-23 17:41:37 -05:00
|
|
|
use fold;
|
2013-01-08 19:37:25 -08:00
|
|
|
use pass::Pass;
|
2012-12-23 17:41:37 -05:00
|
|
|
|
2013-01-10 10:59:58 -08:00
|
|
|
use core::util;
|
2012-07-11 15:00:40 -07:00
|
|
|
|
2012-11-19 18:00:12 -08:00
|
|
|
pub fn mk_pass() -> Pass {
|
2013-01-08 14:00:45 -08:00
|
|
|
Pass {
|
2012-07-13 22:57:48 -07:00
|
|
|
name: ~"desc_to_brief",
|
2012-02-27 18:07:16 -08:00
|
|
|
f: run
|
|
|
|
}
|
2012-01-24 17:19:27 -08:00
|
|
|
}
|
|
|
|
|
2013-01-08 19:37:25 -08:00
|
|
|
pub fn run(
|
2012-09-18 16:48:40 -07:00
|
|
|
_srv: astsrv::Srv,
|
2013-01-30 19:32:36 -08:00
|
|
|
doc: doc::Doc
|
2012-09-18 16:48:40 -07:00
|
|
|
) -> doc::Doc {
|
2012-12-05 15:06:54 -08:00
|
|
|
let fold = Fold {
|
2012-02-17 15:55:30 -08:00
|
|
|
fold_item: fold_item,
|
2012-07-03 16:30:42 -07:00
|
|
|
fold_trait: fold_trait,
|
2012-09-04 13:29:32 -07:00
|
|
|
fold_impl: fold_impl,
|
2012-12-05 15:06:54 -08:00
|
|
|
.. fold::default_any_fold(())
|
|
|
|
};
|
2012-11-29 17:51:16 -08:00
|
|
|
(fold.fold_doc)(&fold, doc)
|
2012-01-24 17:19:27 -08:00
|
|
|
}
|
|
|
|
|
2013-01-30 19:32:36 -08:00
|
|
|
fn fold_item(fold: &fold::Fold<()>, doc: doc::ItemDoc) -> doc::ItemDoc {
|
2012-02-17 15:55:30 -08:00
|
|
|
let doc = fold::default_seq_fold_item(fold, doc);
|
2012-01-24 17:19:27 -08:00
|
|
|
|
2013-01-25 16:57:39 -08:00
|
|
|
doc::ItemDoc {
|
2013-01-30 18:52:31 -08:00
|
|
|
brief: extract(copy doc.desc),
|
2012-09-04 13:29:32 -07:00
|
|
|
.. doc
|
2012-01-26 22:19:47 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-30 19:32:36 -08:00
|
|
|
fn fold_trait(fold: &fold::Fold<()>, doc: doc::TraitDoc) -> doc::TraitDoc {
|
2012-07-03 16:30:42 -07:00
|
|
|
let doc =fold::default_seq_fold_trait(fold, doc);
|
2012-01-30 19:24:06 -08:00
|
|
|
|
2013-01-25 16:57:39 -08:00
|
|
|
doc::TraitDoc {
|
2013-01-31 17:12:29 -08:00
|
|
|
methods: doc.methods.map(|doc| doc::MethodDoc {
|
2013-01-30 18:52:31 -08:00
|
|
|
brief: extract(copy doc.desc),
|
2013-01-30 13:14:35 -08:00
|
|
|
.. copy *doc
|
2012-09-04 13:29:32 -07:00
|
|
|
}),
|
|
|
|
.. doc
|
2012-01-30 19:24:06 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-30 19:32:36 -08:00
|
|
|
fn fold_impl(fold: &fold::Fold<()>, doc: doc::ImplDoc) -> doc::ImplDoc {
|
2012-01-31 20:31:52 -08:00
|
|
|
let doc =fold::default_seq_fold_impl(fold, doc);
|
|
|
|
|
2013-01-25 16:57:39 -08:00
|
|
|
doc::ImplDoc {
|
2013-01-31 17:12:29 -08:00
|
|
|
methods: doc.methods.map(|doc| doc::MethodDoc {
|
2013-01-30 18:52:31 -08:00
|
|
|
brief: extract(copy doc.desc),
|
2013-01-30 13:14:35 -08:00
|
|
|
.. copy *doc
|
2012-09-04 13:29:32 -07:00
|
|
|
}),
|
|
|
|
.. doc
|
2012-01-31 20:31:52 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-16 01:09:55 +10:00
|
|
|
pub fn extract(desc: Option<~str>) -> Option<~str> {
|
2012-09-21 19:37:57 -07:00
|
|
|
if desc.is_none() {
|
2012-08-20 12:23:37 -07:00
|
|
|
return None
|
2012-01-24 17:19:27 -08:00
|
|
|
}
|
|
|
|
|
2013-01-30 13:14:35 -08:00
|
|
|
parse_desc((copy desc).get())
|
2012-01-24 17:19:27 -08:00
|
|
|
}
|
|
|
|
|
2013-01-30 19:32:36 -08:00
|
|
|
fn parse_desc(desc: ~str) -> Option<~str> {
|
2013-03-22 14:00:15 -07:00
|
|
|
static max_brief_len: uint = 120u;
|
2012-01-24 17:19:27 -08:00
|
|
|
|
2013-01-30 13:14:35 -08:00
|
|
|
match first_sentence(copy desc) {
|
2012-08-20 12:23:37 -07:00
|
|
|
Some(first_sentence) => {
|
2012-03-09 17:52:06 -08:00
|
|
|
if str::len(first_sentence) <= max_brief_len {
|
2012-08-20 12:23:37 -07:00
|
|
|
Some(first_sentence)
|
2012-01-24 17:19:27 -08:00
|
|
|
} else {
|
2012-08-20 12:23:37 -07:00
|
|
|
None
|
2012-01-24 17:19:27 -08:00
|
|
|
}
|
2012-03-09 17:52:06 -08:00
|
|
|
}
|
2012-08-20 12:23:37 -07:00
|
|
|
None => None
|
2012-03-09 17:52:06 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-30 19:32:36 -08:00
|
|
|
fn first_sentence(s: ~str) -> Option<~str> {
|
2013-03-03 09:01:28 -08:00
|
|
|
let paras = paragraphs(s);
|
2013-01-24 23:24:57 -05:00
|
|
|
if !paras.is_empty() {
|
2013-03-02 21:49:50 -08:00
|
|
|
let first_para = paras.head();
|
2013-05-21 22:55:07 +09:00
|
|
|
Some(str::replace(first_sentence_(*first_para), "\n", " "))
|
2012-01-24 17:19:27 -08:00
|
|
|
} else {
|
2012-08-20 12:23:37 -07:00
|
|
|
None
|
2012-01-24 17:19:27 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-02 21:49:50 -08:00
|
|
|
fn first_sentence_(s: &str) -> ~str {
|
2012-03-22 08:39:41 -07:00
|
|
|
let mut dotcount = 0;
|
2012-03-16 14:50:15 -07:00
|
|
|
// The index of the character following a single dot. This allows
|
|
|
|
// Things like [0..1) to appear in the brief description
|
2012-06-30 16:19:07 -07:00
|
|
|
let idx = do str::find(s) |ch| {
|
2012-03-16 14:50:15 -07:00
|
|
|
if ch == '.' {
|
|
|
|
dotcount += 1;
|
|
|
|
false
|
|
|
|
} else {
|
|
|
|
if dotcount == 1 {
|
|
|
|
true
|
|
|
|
} else {
|
|
|
|
dotcount = 0;
|
|
|
|
false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2012-08-06 12:34:08 -07:00
|
|
|
match idx {
|
2013-03-02 21:49:50 -08:00
|
|
|
Some(idx) if idx > 2u => {
|
2013-05-10 20:08:56 +09:00
|
|
|
str::to_owned(str::slice(s, 0, idx - 1))
|
2013-03-02 21:49:50 -08:00
|
|
|
}
|
|
|
|
_ => {
|
2013-05-21 22:55:07 +09:00
|
|
|
if str::ends_with(s, ".") {
|
2013-05-10 20:08:56 +09:00
|
|
|
str::to_owned(s)
|
2013-03-02 21:49:50 -08:00
|
|
|
} else {
|
2013-05-10 20:08:56 +09:00
|
|
|
str::to_owned(s)
|
2013-03-02 21:49:50 -08:00
|
|
|
}
|
2012-03-16 14:50:15 -07:00
|
|
|
}
|
|
|
|
}
|
2012-03-09 17:52:06 -08:00
|
|
|
}
|
|
|
|
|
2013-04-16 01:09:55 +10:00
|
|
|
pub fn paragraphs(s: &str) -> ~[~str] {
|
2013-03-26 04:39:10 +01:00
|
|
|
let mut lines = ~[];
|
|
|
|
for str::each_line_any(s) |line| { lines.push(line.to_owned()); }
|
2012-03-22 08:39:41 -07:00
|
|
|
let mut whitespace_lines = 0;
|
2012-07-13 22:57:48 -07:00
|
|
|
let mut accum = ~"";
|
2012-06-30 16:19:07 -07:00
|
|
|
let paras = do vec::foldl(~[], lines) |paras, line| {
|
2012-03-22 08:39:41 -07:00
|
|
|
let mut res = paras;
|
2012-01-24 17:19:27 -08:00
|
|
|
|
2012-09-27 22:20:47 -07:00
|
|
|
if str::is_whitespace(*line) {
|
2012-01-24 17:19:27 -08:00
|
|
|
whitespace_lines += 1;
|
|
|
|
} else {
|
|
|
|
if whitespace_lines > 0 {
|
2013-01-24 23:24:57 -05:00
|
|
|
if !accum.is_empty() {
|
2013-01-10 10:59:58 -08:00
|
|
|
let v = util::replace(&mut accum, ~"");
|
|
|
|
res.push(v);
|
2012-01-24 17:19:27 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
whitespace_lines = 0;
|
|
|
|
|
|
|
|
accum = if str::is_empty(accum) {
|
2013-01-30 13:14:35 -08:00
|
|
|
copy *line
|
2012-01-24 17:19:27 -08:00
|
|
|
} else {
|
2012-09-27 22:20:47 -07:00
|
|
|
accum + ~"\n" + *line
|
2012-01-24 17:19:27 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
res
|
|
|
|
};
|
|
|
|
|
2013-01-24 23:24:57 -05:00
|
|
|
if !accum.is_empty() {
|
2012-06-29 16:26:56 -07:00
|
|
|
paras + ~[accum]
|
2012-01-24 17:19:27 -08:00
|
|
|
} else {
|
|
|
|
paras
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-16 01:09:55 +10:00
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
2013-05-22 19:59:22 -07:00
|
|
|
use core::prelude::*;
|
|
|
|
|
2013-04-16 01:09:55 +10:00
|
|
|
use astsrv;
|
|
|
|
use attr_pass;
|
|
|
|
use super::{extract, paragraphs, run};
|
|
|
|
use doc;
|
|
|
|
use extract;
|
2012-01-24 17:19:27 -08:00
|
|
|
|
2013-04-16 01:09:55 +10:00
|
|
|
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);
|
|
|
|
run(srv.clone(), doc)
|
|
|
|
}
|
|
|
|
}
|
2012-01-24 17:19:27 -08:00
|
|
|
|
2013-04-16 01:09:55 +10:00
|
|
|
#[test]
|
|
|
|
fn should_promote_desc() {
|
|
|
|
let doc = mk_doc(~"#[doc = \"desc\"] mod m { }");
|
2013-05-18 22:02:45 -04:00
|
|
|
assert_eq!(doc.cratemod().mods()[0].brief(), Some(~"desc"));
|
2013-04-16 01:09:55 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_promote_trait_method_desc() {
|
|
|
|
let doc = mk_doc(~"trait i { #[doc = \"desc\"] fn a(); }");
|
|
|
|
assert!(doc.cratemod().traits()[0].methods[0].brief ==
|
|
|
|
Some(~"desc"));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_promote_impl_method_desc() {
|
|
|
|
let doc = mk_doc(
|
|
|
|
~"impl int { #[doc = \"desc\"] fn a() { } }");
|
|
|
|
assert!(doc.cratemod().impls()[0].methods[0].brief == Some(~"desc"));
|
|
|
|
}
|
2012-01-24 17:19:27 -08:00
|
|
|
|
2013-04-16 01:09:55 +10:00
|
|
|
#[test]
|
|
|
|
fn test_paragraphs_1() {
|
|
|
|
let paras = paragraphs(~"1\n\n2");
|
2013-05-18 22:02:45 -04:00
|
|
|
assert_eq!(paras, ~[~"1", ~"2"]);
|
2013-04-16 01:09:55 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_paragraphs_2() {
|
|
|
|
let paras = paragraphs(~"\n\n1\n1\n\n2\n\n");
|
2013-05-18 22:02:45 -04:00
|
|
|
assert_eq!(paras, ~[~"1\n1", ~"2"]);
|
2013-04-16 01:09:55 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_promote_short_descs() {
|
|
|
|
let desc = Some(~"desc");
|
|
|
|
let brief = extract(copy desc);
|
2013-05-18 22:02:45 -04:00
|
|
|
assert_eq!(brief, desc);
|
2013-04-16 01:09:55 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn should_not_promote_long_descs() {
|
|
|
|
let desc = Some(~"Warkworth Castle is a ruined medieval building
|
2012-03-09 17:52:06 -08:00
|
|
|
in the town of the same name in the English county of Northumberland,
|
|
|
|
and the town and castle occupy a loop of the River Coquet, less than a mile
|
2012-01-24 17:19:27 -08:00
|
|
|
from England's north-east coast. When the castle was founded is uncertain,
|
|
|
|
but traditionally its construction has been ascribed to Prince Henry of
|
|
|
|
Scotland in the mid 12th century, although it may have been built by
|
|
|
|
King Henry II of England when he took control of England'snorthern
|
|
|
|
counties.");
|
2013-04-16 01:09:55 +10:00
|
|
|
let brief = extract(desc);
|
2013-05-18 22:02:45 -04:00
|
|
|
assert_eq!(brief, None);
|
2013-04-16 01:09:55 +10:00
|
|
|
}
|
2012-03-09 17:52:06 -08:00
|
|
|
|
2013-04-16 01:09:55 +10:00
|
|
|
#[test]
|
|
|
|
fn should_promote_first_sentence() {
|
|
|
|
let desc = Some(~"Warkworth Castle is a ruined medieval building
|
2012-03-09 17:52:06 -08:00
|
|
|
in the town. of the same name in the English county of Northumberland,
|
|
|
|
and the town and castle occupy a loop of the River Coquet, less than a mile
|
|
|
|
from England's north-east coast. When the castle was founded is uncertain,
|
|
|
|
but traditionally its construction has been ascribed to Prince Henry of
|
|
|
|
Scotland in the mid 12th century, although it may have been built by
|
|
|
|
King Henry II of England when he took control of England'snorthern
|
|
|
|
counties.");
|
2013-04-16 01:09:55 +10:00
|
|
|
let brief = extract(desc);
|
|
|
|
assert!(brief == Some(
|
|
|
|
~"Warkworth Castle is a ruined medieval building in the town"));
|
|
|
|
}
|
2012-03-16 14:50:15 -07:00
|
|
|
|
2013-04-16 01:09:55 +10:00
|
|
|
#[test]
|
|
|
|
fn should_not_consider_double_period_to_end_sentence() {
|
|
|
|
let desc = Some(~"Warkworth..Castle is a ruined medieval building
|
2012-03-16 14:50:15 -07:00
|
|
|
in the town. of the same name in the English county of Northumberland,
|
|
|
|
and the town and castle occupy a loop of the River Coquet, less than a mile
|
|
|
|
from England's north-east coast. When the castle was founded is uncertain,
|
|
|
|
but traditionally its construction has been ascribed to Prince Henry of
|
|
|
|
Scotland in the mid 12th century, although it may have been built by
|
|
|
|
King Henry II of England when he took control of England'snorthern
|
|
|
|
counties.");
|
2013-04-16 01:09:55 +10:00
|
|
|
let brief = extract(desc);
|
|
|
|
assert!(brief == Some(
|
|
|
|
~"Warkworth..Castle is a ruined medieval building in the town"));
|
|
|
|
}
|
2012-03-16 14:50:15 -07:00
|
|
|
|
2013-04-16 01:09:55 +10:00
|
|
|
#[test]
|
|
|
|
fn should_not_consider_triple_period_to_end_sentence() {
|
|
|
|
let desc = Some(~"Warkworth... Castle is a ruined medieval building
|
2012-03-16 14:50:15 -07:00
|
|
|
in the town. of the same name in the English county of Northumberland,
|
|
|
|
and the town and castle occupy a loop of the River Coquet, less than a mile
|
|
|
|
from England's north-east coast. When the castle was founded is uncertain,
|
|
|
|
but traditionally its construction has been ascribed to Prince Henry of
|
|
|
|
Scotland in the mid 12th century, although it may have been built by
|
|
|
|
King Henry II of England when he took control of England'snorthern
|
|
|
|
counties.");
|
2013-04-16 01:09:55 +10:00
|
|
|
let brief = extract(desc);
|
|
|
|
assert!(brief == Some(
|
|
|
|
~"Warkworth... Castle is a ruined medieval building in the town"));
|
|
|
|
}
|
2012-03-16 14:50:15 -07:00
|
|
|
}
|