2016-05-29 09:48:37 +00:00
|
|
|
use error::MarkdownErrorKind as MEK;
|
2016-05-23 12:11:32 +00:00
|
|
|
use result::Result;
|
|
|
|
|
2016-05-24 14:30:06 +00:00
|
|
|
use hoedown::renderer::Render;
|
|
|
|
use hoedown::Buffer;
|
|
|
|
use hoedown::Markdown;
|
2016-05-29 09:48:37 +00:00
|
|
|
use url::Url;
|
|
|
|
|
|
|
|
use libimagerror::into::IntoError;
|
2016-05-24 14:30:06 +00:00
|
|
|
|
2016-05-25 11:37:14 +00:00
|
|
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
2016-05-24 14:30:06 +00:00
|
|
|
pub struct Link {
|
|
|
|
pub title: String,
|
|
|
|
pub link: String,
|
|
|
|
}
|
|
|
|
|
2016-05-29 09:48:37 +00:00
|
|
|
impl Link {
|
|
|
|
|
|
|
|
/// Translate a `Link` into a `UrlLink`
|
|
|
|
fn into_urllink(self) -> Result<UrlLink> {
|
|
|
|
Url::parse(&self.link[..])
|
|
|
|
.map(move |link| UrlLink { title: self.title, link: link, })
|
|
|
|
.map_err(Box::new)
|
|
|
|
.map_err(|e| MEK::LinkParsingError.into_error_with_cause(e))
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct UrlLink {
|
|
|
|
pub title: String,
|
|
|
|
pub link: Url,
|
|
|
|
}
|
|
|
|
|
2016-05-24 14:30:06 +00:00
|
|
|
struct LinkExtractor {
|
|
|
|
links: Vec<Link>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl LinkExtractor {
|
|
|
|
|
|
|
|
pub fn new() -> LinkExtractor {
|
|
|
|
LinkExtractor { links: vec![] }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn links(self) -> Vec<Link> {
|
|
|
|
self.links
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Render for LinkExtractor {
|
|
|
|
|
|
|
|
fn link(&mut self,
|
|
|
|
_: &mut Buffer,
|
|
|
|
content: Option<&Buffer>,
|
|
|
|
link: Option<&Buffer>,
|
|
|
|
_: Option<&Buffer>)
|
|
|
|
-> bool
|
|
|
|
{
|
|
|
|
let link = link.and_then(|l| l.to_str().ok()).map(String::from);
|
|
|
|
let content = content.and_then(|l| l.to_str().ok()).map(String::from);
|
|
|
|
|
|
|
|
match (link, content) {
|
|
|
|
(Some(link), Some(content)) => {
|
|
|
|
self.links.push(Link { link: link, title: content });
|
|
|
|
false
|
|
|
|
},
|
|
|
|
|
|
|
|
(_, _) => {
|
|
|
|
false
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn extract_links(buf: &str) -> Vec<Link> {
|
|
|
|
let mut le = LinkExtractor::new();
|
|
|
|
le.render(&Markdown::new(buf));
|
|
|
|
le.links()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
|
|
|
use super::{Link, extract_links};
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_one_link() {
|
|
|
|
let testtext = "Some [example text](http://example.com).";
|
|
|
|
|
|
|
|
let exp = Link {
|
|
|
|
title: String::from("example text"),
|
|
|
|
link: String::from("http://example.com"),
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut links = extract_links(testtext);
|
|
|
|
assert_eq!(1, links.len());
|
|
|
|
assert_eq!(exp, links.pop().unwrap())
|
|
|
|
}
|
2016-05-23 12:11:32 +00:00
|
|
|
|
2016-05-29 09:16:55 +00:00
|
|
|
#[test]
|
|
|
|
fn test_two_similar_links() {
|
|
|
|
let testtext = r#"
|
|
|
|
Some [example text](http://example.com).
|
|
|
|
Some more [example text](http://example.com).
|
|
|
|
"#;
|
|
|
|
|
|
|
|
let exp = Link {
|
|
|
|
title: String::from("example text"),
|
|
|
|
link: String::from("http://example.com"),
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut links = extract_links(&testtext[..]);
|
|
|
|
assert_eq!(2, links.len());
|
|
|
|
assert_eq!(exp, links.pop().unwrap());
|
|
|
|
assert_eq!(exp, links.pop().unwrap());
|
|
|
|
}
|
|
|
|
|
2016-05-29 09:18:29 +00:00
|
|
|
#[test]
|
|
|
|
fn test_two_links() {
|
|
|
|
let testtext = r#"
|
|
|
|
Some [example text](http://example.com).
|
|
|
|
Some more [foo](http://example.com/foo).
|
|
|
|
"#;
|
|
|
|
|
|
|
|
let exp1 = Link {
|
|
|
|
title: String::from("example text"),
|
|
|
|
link: String::from("http://example.com"),
|
|
|
|
};
|
|
|
|
|
|
|
|
let exp2 = Link {
|
|
|
|
title: String::from("foo"),
|
|
|
|
link: String::from("http://example.com/foo"),
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut links = extract_links(&testtext[..]);
|
|
|
|
assert_eq!(2, links.len());
|
|
|
|
assert_eq!(exp2, links.pop().unwrap());
|
|
|
|
assert_eq!(exp1, links.pop().unwrap());
|
|
|
|
}
|
|
|
|
|
2016-05-23 12:11:32 +00:00
|
|
|
}
|
|
|
|
|