use super::elements::Verbatim;
use super::range::Range;
use super::{Document, Session};
use crate::lex::inlines::{InlineNode, ReferenceType};
use std::fmt;
#[derive(Debug, Clone, PartialEq)]
pub struct DocumentLink {
pub range: Range,
pub target: String,
pub link_type: LinkType,
}
impl DocumentLink {
pub fn new(range: Range, target: String, link_type: LinkType) -> Self {
Self {
range,
target,
link_type,
}
}
}
impl fmt::Display for DocumentLink {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(
f,
"{:?} link: {} at {}",
self.link_type, self.target, self.range.start
)
}
}
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum LinkType {
Url,
File,
VerbatimSrc,
}
impl Verbatim {
pub fn src_parameter(&self) -> Option<&str> {
self.closing_data
.parameters
.iter()
.find(|p| p.key == "src")
.map(|p| p.value.as_str())
}
}
impl Session {
pub fn find_all_links(&self) -> Vec<DocumentLink> {
use super::elements::content_item::ContentItem;
use super::traits::AstNode;
let mut links = Vec::new();
if let Some(inlines) = self.title.inlines() {
for inline in inlines {
if let InlineNode::Reference { data, .. } = inline {
match &data.reference_type {
ReferenceType::Url { target } => {
let range = self.header_location().unwrap_or(&self.location).clone();
let link = DocumentLink::new(range, target.clone(), LinkType::Url);
links.push(link);
}
ReferenceType::File { target } => {
let range = self.header_location().unwrap_or(&self.location).clone();
let link = DocumentLink::new(range, target.clone(), LinkType::File);
links.push(link);
}
_ => {}
}
}
}
}
for paragraph in self.iter_paragraphs_recursive() {
for line_item in ¶graph.lines {
if let ContentItem::TextLine(line) = line_item {
if let Some(inlines) = line.content.inlines() {
for inline in inlines {
if let InlineNode::Reference { data, .. } = inline {
match &data.reference_type {
ReferenceType::Url { target } => {
let link = DocumentLink::new(
paragraph.range().clone(),
target.clone(),
LinkType::Url,
);
links.push(link);
}
ReferenceType::File { target } => {
let link = DocumentLink::new(
paragraph.range().clone(),
target.clone(),
LinkType::File,
);
links.push(link);
}
_ => {
}
}
}
}
}
}
}
}
for (item, _depth) in self.iter_all_nodes_with_depth() {
if let ContentItem::VerbatimBlock(verbatim) = item {
if let Some(src) = verbatim.src_parameter() {
let link = DocumentLink::new(
verbatim.range().clone(),
src.to_string(),
LinkType::VerbatimSrc,
);
links.push(link);
}
}
}
links
}
}
impl Document {
pub fn find_all_links(&self) -> Vec<DocumentLink> {
let mut links = Vec::new();
if let Some(title) = &self.title {
if let Some(inlines) = title.content.inlines() {
for inline in inlines {
if let crate::lex::inlines::InlineNode::Reference { data, .. } = inline {
let range = title.location.clone();
match &data.reference_type {
crate::lex::inlines::ReferenceType::Url { target } => {
links.push(DocumentLink::new(range, target.clone(), LinkType::Url));
}
crate::lex::inlines::ReferenceType::File { target } => {
links.push(DocumentLink::new(
range,
target.clone(),
LinkType::File,
));
}
_ => {}
}
}
}
}
}
links.extend(self.root.find_all_links());
links
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::lex::parsing::parse_document;
#[test]
fn test_url_link_extraction() {
let source = "Check out [https://example.com] for more info.\n\n";
let doc = parse_document(source).unwrap();
let links = doc.find_all_links();
assert_eq!(links.len(), 1);
assert_eq!(links[0].link_type, LinkType::Url);
assert_eq!(links[0].target, "https://example.com");
}
#[test]
fn test_file_link_extraction() {
let source = "See [./README.md] for details.\n\n";
let doc = parse_document(source).unwrap();
let links = doc.find_all_links();
assert_eq!(links.len(), 1);
assert_eq!(links[0].link_type, LinkType::File);
assert_eq!(links[0].target, "./README.md");
}
#[test]
fn test_multiple_links() {
let source = "Visit [https://example.com] and check [./docs.md].\n\n";
let doc = parse_document(source).unwrap();
let links = doc.find_all_links();
assert_eq!(links.len(), 2);
assert!(links.iter().any(|l| l.link_type == LinkType::Url));
assert!(links.iter().any(|l| l.link_type == LinkType::File));
}
#[test]
fn test_verbatim_src_parameter() {
let source =
"Sunset Photo:\n As the sun sets over the ocean.\n:: image src=./diagram.png ::\n\n";
let doc = parse_document(source).unwrap();
let links = doc.find_all_links();
let src_links: Vec<_> = links
.iter()
.filter(|l| l.link_type == LinkType::VerbatimSrc)
.collect();
assert_eq!(
src_links.len(),
1,
"Expected 1 verbatim src link, found {}. All links: {:?}",
src_links.len(),
links
);
assert_eq!(src_links[0].target, "./diagram.png");
}
#[test]
fn test_verbatim_src_parameter_method() {
use super::super::elements::{Data, Label, Parameter};
let verbatim = Verbatim::with_subject(
"Test".to_string(),
Data::new(
Label::new("image".to_string()),
vec![Parameter::new("src".to_string(), "./test.png".to_string())],
),
);
assert_eq!(verbatim.src_parameter(), Some("./test.png"));
let verbatim_no_src = Verbatim::with_subject(
"Test".to_string(),
Data::new(Label::new("code".to_string()), vec![]),
);
assert_eq!(verbatim_no_src.src_parameter(), None);
}
#[test]
fn test_no_links() {
let source = "Just plain text with no links.\n\n";
let doc = parse_document(source).unwrap();
let links = doc.find_all_links();
assert_eq!(links.len(), 0);
}
#[test]
fn test_footnote_not_a_link() {
let source = "Text with footnote [42].\n\n";
let doc = parse_document(source).unwrap();
let links = doc.find_all_links();
assert_eq!(links.len(), 0);
}
#[test]
fn test_nested_session_links() {
let source = "Outer Session\n\n Inner session with [https://example.com].\n\n";
let doc = parse_document(source).unwrap();
let links = doc.find_all_links();
assert_eq!(links.len(), 1);
assert_eq!(links[0].target, "https://example.com");
}
}