use super::annotation_utils::adjust_annotations_for_trim;
use super::frontmatter_utils::{extract_frontmatter, extract_metadata_from_yaml, extract_title_from_content};
use crate::Result;
use crate::core::config::ExtractionConfig;
use crate::plugins::{DocumentExtractor, Plugin};
use crate::types::internal::InternalDocument;
use crate::types::internal_builder::InternalDocumentBuilder;
use crate::types::uri::{Uri, UriKind, classify_uri};
use crate::types::{Metadata, Table};
use async_trait::async_trait;
use pulldown_cmark::{Event, Options, Parser, Tag, TagEnd};
use std::borrow::Cow;
type AnnotationEntry = (u8, u32, Option<(String, Option<String>)>);
pub struct MarkdownExtractor;
impl MarkdownExtractor {
pub fn new() -> Self {
Self
}
pub fn build_internal_document(events: &[Event], yaml: &Option<serde_yaml_ng::Value>) -> InternalDocument {
use crate::types::builder;
use crate::types::document_structure::TextAnnotation;
let mut b = InternalDocumentBuilder::new("markdown");
if let Some(serde_yaml_ng::Value::Mapping(map)) = yaml {
let entries: Vec<(String, String)> = map
.iter()
.filter_map(|(k, v)| {
let key = k.as_str()?.to_string();
let val = match v {
serde_yaml_ng::Value::String(s) => s.clone(),
other => format!("{other:?}"),
};
Some((key, val))
})
.collect();
if !entries.is_empty() {
b.push_metadata_block(&entries, None);
}
}
let mut paragraph_text = String::new();
let mut paragraph_annotations: Vec<TextAnnotation> = Vec::new();
let mut in_paragraph = false;
let mut heading_text = String::new();
let mut heading_annotations: Vec<TextAnnotation> = Vec::new();
let mut heading_level: u8 = 0;
let mut in_heading = false;
let mut code_text = String::new();
let mut code_lang: Option<String> = None;
let mut in_code_block = false;
let mut table_rows: Vec<Vec<String>> = Vec::new();
let mut current_row: Vec<String> = Vec::new();
let mut current_cell = String::new();
let mut in_table_cell = false;
let mut list_stack: Vec<bool> = Vec::new(); let mut list_item_text = String::new();
let mut list_item_annotations: Vec<TextAnnotation> = Vec::new();
let mut in_list_item = false;
let mut in_image = false;
let mut image_alt = String::new();
let mut image_url: Option<String> = None;
let mut footnote_def_label: Option<String> = None;
let mut footnote_def_text = String::new();
let mut annotation_starts: Vec<AnnotationEntry> = Vec::new();
fn active_text_offset(buf: &str) -> u32 {
buf.len() as u32
}
for event in events {
match event {
Event::Start(Tag::Heading { level, .. }) => {
heading_text.clear();
heading_annotations.clear();
annotation_starts.clear();
heading_level = match *level {
pulldown_cmark::HeadingLevel::H1 => 1,
pulldown_cmark::HeadingLevel::H2 => 2,
pulldown_cmark::HeadingLevel::H3 => 3,
pulldown_cmark::HeadingLevel::H4 => 4,
pulldown_cmark::HeadingLevel::H5 => 5,
pulldown_cmark::HeadingLevel::H6 => 6,
};
in_heading = true;
}
Event::End(TagEnd::Heading(_)) => {
in_heading = false;
let trimmed = heading_text.trim();
if !trimmed.is_empty() {
let annotations = adjust_annotations_for_trim(
std::mem::take(&mut heading_annotations),
&heading_text,
trimmed,
);
let idx = b.push_heading(heading_level, trimmed, None, None);
if !annotations.is_empty() {
b.set_annotations(idx, annotations);
}
}
heading_text.clear();
heading_annotations.clear();
}
Event::Start(Tag::Paragraph) => {
if !in_heading && !in_list_item && footnote_def_label.is_none() {
paragraph_text.clear();
paragraph_annotations.clear();
in_paragraph = true;
}
}
Event::End(TagEnd::Paragraph) => {
if in_paragraph {
in_paragraph = false;
let trimmed = paragraph_text.trim();
if !trimmed.is_empty() {
let annotations = adjust_annotations_for_trim(
std::mem::take(&mut paragraph_annotations),
¶graph_text,
trimmed,
);
b.push_paragraph(trimmed, annotations, None, None);
}
paragraph_text.clear();
paragraph_annotations.clear();
}
}
Event::Start(Tag::Strong) => {
if in_paragraph {
annotation_starts.push((0, active_text_offset(¶graph_text), None));
} else if in_heading {
annotation_starts.push((0, active_text_offset(&heading_text), None));
} else if in_list_item {
annotation_starts.push((0, active_text_offset(&list_item_text), None));
}
}
Event::End(TagEnd::Strong) => {
if let Some(i) = annotation_starts.iter().rposition(|(k, _, _)| *k == 0) {
let (_, start, _) = annotation_starts.remove(i);
if in_paragraph {
let end = active_text_offset(¶graph_text);
if start < end {
paragraph_annotations.push(builder::bold(start, end));
}
} else if in_heading {
let end = active_text_offset(&heading_text);
if start < end {
heading_annotations.push(builder::bold(start, end));
}
} else if in_list_item {
let end = active_text_offset(&list_item_text);
if start < end {
list_item_annotations.push(builder::bold(start, end));
}
}
}
}
Event::Start(Tag::Emphasis) => {
if in_paragraph {
annotation_starts.push((1, active_text_offset(¶graph_text), None));
} else if in_heading {
annotation_starts.push((1, active_text_offset(&heading_text), None));
} else if in_list_item {
annotation_starts.push((1, active_text_offset(&list_item_text), None));
}
}
Event::End(TagEnd::Emphasis) => {
if let Some(i) = annotation_starts.iter().rposition(|(k, _, _)| *k == 1) {
let (_, start, _) = annotation_starts.remove(i);
if in_paragraph {
let end = active_text_offset(¶graph_text);
if start < end {
paragraph_annotations.push(builder::italic(start, end));
}
} else if in_heading {
let end = active_text_offset(&heading_text);
if start < end {
heading_annotations.push(builder::italic(start, end));
}
} else if in_list_item {
let end = active_text_offset(&list_item_text);
if start < end {
list_item_annotations.push(builder::italic(start, end));
}
}
}
}
Event::Start(Tag::Strikethrough) => {
if in_paragraph {
annotation_starts.push((2, active_text_offset(¶graph_text), None));
} else if in_heading {
annotation_starts.push((2, active_text_offset(&heading_text), None));
} else if in_list_item {
annotation_starts.push((2, active_text_offset(&list_item_text), None));
}
}
Event::End(TagEnd::Strikethrough) => {
if let Some(i) = annotation_starts.iter().rposition(|(k, _, _)| *k == 2) {
let (_, start, _) = annotation_starts.remove(i);
if in_paragraph {
let end = active_text_offset(¶graph_text);
if start < end {
paragraph_annotations.push(builder::strikethrough(start, end));
}
} else if in_heading {
let end = active_text_offset(&heading_text);
if start < end {
heading_annotations.push(builder::strikethrough(start, end));
}
} else if in_list_item {
let end = active_text_offset(&list_item_text);
if start < end {
list_item_annotations.push(builder::strikethrough(start, end));
}
}
}
}
Event::Start(Tag::Link { dest_url, title, .. }) => {
let url = dest_url.to_string();
let title_opt = if title.is_empty() {
None
} else {
Some(title.to_string())
};
if in_paragraph {
annotation_starts.push((4, active_text_offset(¶graph_text), Some((url, title_opt))));
} else if in_heading {
annotation_starts.push((4, active_text_offset(&heading_text), Some((url, title_opt))));
} else if in_list_item {
annotation_starts.push((4, active_text_offset(&list_item_text), Some((url, title_opt))));
}
}
Event::End(TagEnd::Link) => {
if let Some(i) = annotation_starts.iter().rposition(|(k, _, _)| *k == 4) {
let (_, start, link_data) = annotation_starts.remove(i);
if let Some((url, title)) = link_data {
let label_text = if in_paragraph {
let end = active_text_offset(¶graph_text);
if start < end {
paragraph_annotations.push(builder::link(start, end, &url, title.as_deref()));
Some(paragraph_text[start as usize..end as usize].to_string())
} else {
None
}
} else if in_heading {
let end = active_text_offset(&heading_text);
if start < end {
heading_annotations.push(builder::link(start, end, &url, title.as_deref()));
Some(heading_text[start as usize..end as usize].to_string())
} else {
None
}
} else if in_list_item {
let end = active_text_offset(&list_item_text);
if start < end {
list_item_annotations.push(builder::link(start, end, &url, title.as_deref()));
Some(list_item_text[start as usize..end as usize].to_string())
} else {
None
}
} else {
None
};
if !url.is_empty() {
let kind = classify_uri(&url);
b.push_uri(Uri {
url,
label: label_text.filter(|s| !s.is_empty()),
page: None,
kind,
});
}
}
}
}
Event::Start(Tag::CodeBlock(pulldown_cmark::CodeBlockKind::Fenced(lang))) => {
code_text.clear();
code_lang = if lang.is_empty() { None } else { Some(lang.to_string()) };
in_code_block = true;
}
Event::Start(Tag::CodeBlock(_)) => {
code_text.clear();
code_lang = None;
in_code_block = true;
}
Event::End(TagEnd::CodeBlock) => {
in_code_block = false;
let trimmed = code_text.trim_end();
if !trimmed.is_empty() {
b.push_code(trimmed, code_lang.as_deref(), None, None);
}
code_text.clear();
code_lang = None;
}
Event::Start(Tag::BlockQuote(_)) => {
b.push_quote_start();
}
Event::End(TagEnd::BlockQuote(_)) => {
b.push_quote_end();
}
Event::Start(Tag::List(start)) => {
let ordered = start.is_some();
b.push_list(ordered);
list_stack.push(ordered);
}
Event::End(TagEnd::List(_)) => {
if list_stack.pop().is_some() {
b.end_list();
}
}
Event::Start(Tag::Item) => {
list_item_text.clear();
list_item_annotations.clear();
annotation_starts.clear();
in_list_item = true;
}
Event::End(TagEnd::Item) => {
in_list_item = false;
let trimmed = list_item_text.trim();
if let Some(ordered) = list_stack.last().copied()
&& !trimmed.is_empty()
{
let annotations = adjust_annotations_for_trim(
std::mem::take(&mut list_item_annotations),
&list_item_text,
trimmed,
);
b.push_list_item(trimmed, ordered, annotations, None, None);
}
list_item_text.clear();
list_item_annotations.clear();
}
Event::Start(Tag::Table(_)) => {
table_rows.clear();
}
Event::End(TagEnd::Table) => {
if !table_rows.is_empty() {
let markdown = super::frontmatter_utils::cells_to_markdown(&table_rows);
let table = Table {
cells: std::mem::take(&mut table_rows),
markdown,
page_number: 1,
bounding_box: None,
};
b.push_table(table, None, None);
}
table_rows.clear();
}
Event::Start(Tag::TableHead | Tag::TableRow) => {
current_row.clear();
}
Event::End(TagEnd::TableHead | TagEnd::TableRow) => {
if !current_row.is_empty() {
table_rows.push(std::mem::take(&mut current_row));
}
}
Event::Start(Tag::TableCell) => {
current_cell.clear();
in_table_cell = true;
}
Event::End(TagEnd::TableCell) => {
in_table_cell = false;
current_row.push(current_cell.trim().to_string());
current_cell.clear();
}
Event::Start(Tag::Image { dest_url, .. }) => {
in_image = true;
image_alt.clear();
image_url = Some(dest_url.to_string());
}
Event::End(TagEnd::Image) => {
in_image = false;
let trimmed = image_alt.trim();
let desc = if trimmed.is_empty() { "" } else { trimmed };
{
use crate::types::document_structure::ContentLayer;
use crate::types::internal::{ElementKind, InternalElement, InternalElementId};
let kind = ElementKind::Image { image_index: u32::MAX };
let id = InternalElementId::generate(kind.discriminant(), desc, None, 0);
b.push_element(InternalElement {
id,
kind,
text: desc.to_string(),
depth: 0,
page: None,
bbox: None,
layer: ContentLayer::Body,
annotations: Vec::new(),
attributes: None,
anchor: None,
ocr_geometry: None,
ocr_confidence: None,
ocr_rotation: None,
});
}
if let Some(url) = image_url.take().filter(|u| !u.is_empty()) {
b.push_uri(Uri {
url,
label: if desc.is_empty() { None } else { Some(desc.to_string()) },
page: None,
kind: UriKind::Image,
});
}
image_alt.clear();
}
Event::Start(Tag::FootnoteDefinition(label)) => {
footnote_def_label = Some(label.to_string());
footnote_def_text.clear();
}
Event::End(TagEnd::FootnoteDefinition) => {
if let Some(label) = footnote_def_label.take() {
let text = footnote_def_text.trim().to_string();
if !text.is_empty() {
b.push_footnote_definition(&text, &label, None);
}
}
footnote_def_text.clear();
}
Event::Code(s) => {
if in_code_block {
code_text.push_str(s);
} else if in_heading {
let start = heading_text.len() as u32;
heading_text.push_str(s);
let end = heading_text.len() as u32;
if start < end {
heading_annotations.push(builder::code(start, end));
}
} else if in_image {
image_alt.push_str(s);
} else if in_table_cell {
current_cell.push_str(s);
} else if in_list_item {
let start = list_item_text.len() as u32;
list_item_text.push_str(s);
let end = list_item_text.len() as u32;
if start < end {
list_item_annotations.push(builder::code(start, end));
}
} else if footnote_def_label.is_some() {
footnote_def_text.push_str(s);
} else if in_paragraph {
let start = paragraph_text.len() as u32;
paragraph_text.push_str(s);
let end = paragraph_text.len() as u32;
if start < end {
paragraph_annotations.push(builder::code(start, end));
}
}
}
Event::Text(s) => {
if in_code_block {
code_text.push_str(s);
} else if in_heading {
heading_text.push_str(s);
} else if in_image {
image_alt.push_str(s);
} else if in_table_cell {
current_cell.push_str(s);
} else if in_list_item {
list_item_text.push_str(s);
} else if footnote_def_label.is_some() {
footnote_def_text.push_str(s);
} else if in_paragraph {
paragraph_text.push_str(s);
}
}
Event::SoftBreak | Event::HardBreak => {
if in_code_block {
code_text.push('\n');
} else if in_heading {
heading_text.push(' ');
} else if in_list_item {
list_item_text.push(' ');
} else if footnote_def_label.is_some() {
footnote_def_text.push(' ');
} else if in_paragraph {
paragraph_text.push(' ');
}
}
Event::FootnoteReference(name) => {
b.push_footnote_ref(name, name, None);
}
Event::Html(s) => {
if footnote_def_label.is_some() {
footnote_def_text.push_str(s);
} else if in_paragraph {
paragraph_text.push_str(s);
}
}
Event::TaskListMarker(checked) => {
if in_list_item {
list_item_text.push_str(if *checked { "[x] " } else { "[ ] " });
}
}
_ => {}
}
}
b.build()
}
}
impl Default for MarkdownExtractor {
fn default() -> Self {
Self::new()
}
}
impl Plugin for MarkdownExtractor {
fn name(&self) -> &str {
"markdown-extractor"
}
fn version(&self) -> String {
env!("CARGO_PKG_VERSION").to_string()
}
fn initialize(&self) -> Result<()> {
Ok(())
}
fn shutdown(&self) -> Result<()> {
Ok(())
}
fn description(&self) -> &str {
"Extracts content from Markdown files with YAML frontmatter and table support"
}
fn author(&self) -> &str {
"Kreuzberg Team"
}
}
#[cfg_attr(not(target_arch = "wasm32"), async_trait)]
#[cfg_attr(target_arch = "wasm32", async_trait(?Send))]
impl DocumentExtractor for MarkdownExtractor {
async fn extract_bytes(
&self,
content: &[u8],
mime_type: &str,
config: &ExtractionConfig,
) -> Result<InternalDocument> {
tracing::debug!(format = "markdown", size_bytes = content.len(), "extraction starting");
let _ = config; let text = String::from_utf8_lossy(content).into_owned();
let (yaml, remaining_content) = extract_frontmatter(&text);
let mut metadata = if let Some(ref yaml_value) = yaml {
extract_metadata_from_yaml(yaml_value)
} else {
Metadata::default()
};
if metadata.title.is_none()
&& let Some(title) = extract_title_from_content(&remaining_content)
{
metadata.title = Some(title);
}
let mut options = Options::ENABLE_TABLES;
options |= Options::ENABLE_STRIKETHROUGH | Options::ENABLE_FOOTNOTES;
let parser = Parser::new_ext(&remaining_content, options);
let events: Vec<Event> = parser.collect();
let mut extracted_images = Vec::new();
let _ = crate::extractors::markdown_utils::extract_text_from_events(&events, &mut extracted_images);
let mut doc = Self::build_internal_document(&events, &yaml);
doc.metadata = metadata;
doc.mime_type = Cow::Owned(mime_type.to_string());
if !extracted_images.is_empty() {
for image in extracted_images {
doc.push_image(image);
}
}
tracing::debug!(
element_count = doc.elements.len(),
format = "markdown",
"extraction complete"
);
Ok(doc)
}
async fn extract_file(
&self,
path: &std::path::Path,
mime_type: &str,
config: &ExtractionConfig,
) -> Result<InternalDocument> {
crate::core::path_resolver::extract_file_with_image_resolution(self, path, mime_type, config).await
}
fn supported_mime_types(&self) -> &[&str] {
&[
"text/markdown",
"text/x-markdown",
"text/x-gfm",
"text/x-commonmark",
"text/x-markdown-extra",
"text/x-multimarkdown",
]
}
fn priority(&self) -> i32 {
50
}
}
#[cfg(test)]
mod tests {
use super::super::frontmatter_utils::{cells_to_markdown, extract_frontmatter, extract_metadata_from_yaml};
use super::*;
use serde_yaml_ng::Value as YamlValue;
#[test]
fn test_can_extract_markdown_mime_types() {
let extractor = MarkdownExtractor::new();
let mime_types = extractor.supported_mime_types();
assert!(mime_types.contains(&"text/markdown"));
assert!(mime_types.contains(&"text/x-markdown"));
assert!(mime_types.contains(&"text/x-gfm"));
assert!(mime_types.contains(&"text/x-commonmark"));
assert!(mime_types.contains(&"text/x-markdown-extra"));
assert!(mime_types.contains(&"text/x-multimarkdown"));
}
#[test]
fn test_extract_simple_markdown() {
let content =
b"# Header\n\nThis is a paragraph with **bold** and *italic* text.\n\n## Subheading\n\nMore content here.";
let text = String::from_utf8_lossy(content).into_owned();
let (yaml, remaining) = extract_frontmatter(&text);
assert!(yaml.is_none());
assert!(!remaining.is_empty());
let parser = Parser::new_ext(&remaining, Options::ENABLE_TABLES);
let events: Vec<Event> = parser.collect();
let extracted = crate::extractors::markdown_utils::extract_text_from_events(&events, &mut Vec::new());
assert!(extracted.contains("Header"));
assert!(extracted.contains("This is a paragraph"));
assert!(extracted.contains("bold"));
assert!(extracted.contains("italic"));
}
#[test]
fn test_extract_frontmatter_metadata() {
let content = b"---\ntitle: My Document\nauthor: John Doe\ndate: 2024-01-15\nkeywords: rust, markdown, extraction\ndescription: A test document\n---\n\n# Content\n\nBody text.";
let text = String::from_utf8_lossy(content).into_owned();
let (yaml_opt, remaining) = extract_frontmatter(&text);
assert!(yaml_opt.is_some());
assert!(remaining.contains("# Content"));
let yaml = yaml_opt.expect("Should extract YAML frontmatter");
let metadata = extract_metadata_from_yaml(&yaml);
assert_eq!(metadata.title.as_deref(), Some("My Document"));
assert_eq!(metadata.created_by.as_deref(), Some("John Doe"));
assert_eq!(metadata.created_at, Some("2024-01-15".to_string()));
assert!(metadata.subject.is_some());
assert!(
metadata
.subject
.as_ref()
.expect("Should have subject description")
.contains("test document")
);
}
#[test]
fn test_extract_frontmatter_metadata_array_keywords() {
let content = b"---\ntitle: Document\nkeywords:\n - rust\n - markdown\n - parsing\n---\n\nContent";
let text = String::from_utf8_lossy(content).into_owned();
let (yaml_opt, _remaining) = extract_frontmatter(&text);
assert!(yaml_opt.is_some());
let yaml = yaml_opt.expect("Should extract YAML frontmatter");
let metadata = extract_metadata_from_yaml(&yaml);
let keywords = metadata
.keywords
.as_ref()
.expect("Should extract keywords from metadata");
assert!(keywords.iter().any(|k| k == "rust"));
assert!(keywords.iter().any(|k| k == "markdown"));
}
#[tokio::test]
async fn test_extract_tables() {
let content = b"# Tables Example\n\n| Header 1 | Header 2 |\n|----------|----------|\n| Cell 1 | Cell 2 |\n| Cell 3 | Cell 4 |";
let extractor = MarkdownExtractor::new();
let result = extractor
.extract_bytes(content, "text/markdown", &ExtractionConfig::default())
.await
.expect("Should extract markdown with tables");
let result =
crate::extraction::derive::derive_extraction_result(result, true, crate::core::config::OutputFormat::Plain);
assert!(!result.tables.is_empty());
let table = &result.tables[0];
assert!(!table.cells.is_empty());
assert_eq!(table.cells[0].len(), 2);
assert!(!table.markdown.is_empty());
}
#[test]
fn test_extract_without_frontmatter() {
let content = b"# Main Title\n\nSome content\n\nMore text";
let text = String::from_utf8_lossy(content).into_owned();
let (yaml, remaining) = extract_frontmatter(&text);
assert!(yaml.is_none());
assert_eq!(remaining, text);
let title = extract_title_from_content(&remaining);
assert_eq!(title, Some("Main Title".to_string()));
}
#[test]
fn test_empty_document() {
let content = b"";
let text = String::from_utf8_lossy(content).into_owned();
let (yaml, remaining) = extract_frontmatter(&text);
assert!(yaml.is_none());
assert!(remaining.is_empty());
let parser = Parser::new_ext(&remaining, Options::ENABLE_TABLES);
let events: Vec<Event> = parser.collect();
let extracted = crate::extractors::markdown_utils::extract_text_from_events(&events, &mut Vec::new());
assert!(extracted.is_empty());
}
#[test]
fn test_whitespace_only_document() {
let content = b" \n\n \n";
let text = String::from_utf8_lossy(content).into_owned();
let (yaml, remaining) = extract_frontmatter(&text);
assert!(yaml.is_none());
let parser = Parser::new_ext(&remaining, Options::ENABLE_TABLES);
let events: Vec<Event> = parser.collect();
let extracted = crate::extractors::markdown_utils::extract_text_from_events(&events, &mut Vec::new());
assert!(extracted.trim().is_empty());
}
#[test]
fn test_unicode_content() {
let content = "# 日本語のタイトル\n\nこれは日本語の内容です。\n\n## Español\n\nEste es un documento en español.\n\n## Русский\n\nЭто русский текст.".as_bytes();
let text = String::from_utf8_lossy(content).into_owned();
let (yaml, remaining) = extract_frontmatter(&text);
assert!(yaml.is_none());
let parser = Parser::new_ext(&remaining, Options::ENABLE_TABLES);
let events: Vec<Event> = parser.collect();
let extracted = crate::extractors::markdown_utils::extract_text_from_events(&events, &mut Vec::new());
assert!(extracted.contains("日本語"));
assert!(extracted.contains("Español"));
assert!(extracted.contains("Русский"));
}
#[tokio::test]
async fn test_full_extraction_with_frontmatter_and_tables() {
let content = b"---\ntitle: Complete Document\nauthor: Test Author\ndate: 2024-01-20\n---\n\n# Document\n\nIntroduction text.\n\n| Name | Value |\n|------|-------|\n| A | 1 |\n| B | 2 |";
let extractor = MarkdownExtractor::new();
let result = extractor
.extract_bytes(content, "text/x-markdown", &ExtractionConfig::default())
.await
.expect("Should extract markdown with frontmatter and tables");
let result =
crate::extraction::derive::derive_extraction_result(result, true, crate::core::config::OutputFormat::Plain);
assert_eq!(result.mime_type, "text/x-markdown");
assert!(result.content.contains("Introduction text"));
assert_eq!(result.metadata.title.as_deref(), Some("Complete Document"));
assert_eq!(result.metadata.created_by.as_deref(), Some("Test Author"));
assert!(!result.tables.is_empty());
}
#[test]
fn test_plugin_interface() {
let extractor = MarkdownExtractor::new();
assert_eq!(extractor.name(), "markdown-extractor");
assert_eq!(extractor.version(), env!("CARGO_PKG_VERSION"));
assert_eq!(extractor.priority(), 50);
assert!(extractor.supported_mime_types().contains(&"text/markdown"));
}
#[test]
fn test_cells_to_markdown() {
let cells = vec![
vec!["Header 1".to_string(), "Header 2".to_string()],
vec!["Data 1".to_string(), "Data 2".to_string()],
vec!["Data 3".to_string(), "Data 4".to_string()],
];
let markdown = cells_to_markdown(&cells);
assert!(markdown.contains("Header 1"));
assert!(markdown.contains("Data 1"));
assert!(markdown.contains("---"));
let lines: Vec<&str> = markdown.lines().collect();
assert!(lines.len() >= 4);
}
#[test]
fn test_extract_markdown_with_links() {
let content = b"# Page\n\nCheck [Google](https://google.com) and [Rust](https://rust-lang.org).";
let text = String::from_utf8_lossy(content).into_owned();
let parser = Parser::new_ext(&text, Options::ENABLE_TABLES);
let events: Vec<Event> = parser.collect();
let extracted = crate::extractors::markdown_utils::extract_text_from_events(&events, &mut Vec::new());
assert!(extracted.contains("Google"));
assert!(extracted.contains("Rust"));
}
#[test]
fn test_extract_markdown_with_code_blocks() {
let content = b"# Code Example\n\n```rust\nfn main() {\n println!(\"Hello\");\n}\n```";
let text = String::from_utf8_lossy(content).into_owned();
let parser = Parser::new_ext(&text, Options::ENABLE_TABLES);
let events: Vec<Event> = parser.collect();
let extracted = crate::extractors::markdown_utils::extract_text_from_events(&events, &mut Vec::new());
assert!(extracted.contains("main"));
assert!(extracted.contains("println"));
}
#[test]
fn test_malformed_frontmatter_fallback() {
let content = b"---\nthis: is: invalid: yaml:\n---\n\nContent here";
let text = String::from_utf8_lossy(content).into_owned();
let (yaml, _remaining) = extract_frontmatter(&text);
let _ = yaml;
}
#[test]
fn test_metadata_extraction_completeness() {
let yaml_str = r#"
title: "Test Document"
author: "Test Author"
date: "2024-01-15"
keywords:
- rust
- markdown
- testing
description: "A test description"
abstract: "Test abstract"
subject: "Test subject"
category: "Documentation"
version: "1.2.3"
language: "en"
tags:
- tag1
- tag2
custom_field: "custom_value"
nested:
organization: "Test Corp"
contact:
email: "test@example.com"
"#;
let yaml: YamlValue = serde_yaml_ng::from_str(yaml_str).expect("Valid YAML");
let metadata = extract_metadata_from_yaml(&yaml);
assert_eq!(metadata.created_at, Some("2024-01-15".to_string()));
assert_eq!(metadata.title.as_deref(), Some("Test Document"));
assert_eq!(metadata.created_by.as_deref(), Some("Test Author"));
let keywords = metadata.keywords.as_ref().expect("Should have keywords");
assert!(keywords.iter().any(|k| k == "rust"));
assert!(keywords.iter().any(|k| k == "markdown"));
assert_eq!(metadata.subject, Some("Test subject".to_string()));
assert_eq!(metadata.abstract_text.as_deref(), Some("Test abstract"));
assert_eq!(metadata.category.as_deref(), Some("Documentation"));
let tags = metadata.tags.as_ref().expect("Should have tags");
assert!(tags.iter().any(|t| t == "tag1"));
assert!(tags.iter().any(|t| t == "tag2"));
assert_eq!(metadata.language.as_deref(), Some("en"));
assert_eq!(metadata.document_version.as_deref(), Some("1.2.3"));
println!("\nSuccessfully extracted all typed metadata fields");
}
#[test]
fn test_decode_data_uri_png() {
let png_b64 =
"iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVR42mP8/5+hHgAHggJ/PchI7wAAAABJRU5ErkJggg==";
let uri = format!("data:image/png;base64,{png_b64}");
let image = crate::extractors::markdown_utils::decode_data_uri_image(&uri, 0);
assert!(image.is_some());
let img = image.unwrap();
assert_eq!(img.format.as_ref(), "png");
assert_eq!(img.image_index, 0);
assert!(!img.data.is_empty());
}
#[test]
fn test_decode_data_uri_jpeg() {
let uri = "data:image/jpeg;base64,/9j/4AAQSkZJRg==";
let image = crate::extractors::markdown_utils::decode_data_uri_image(uri, 3);
assert!(image.is_some());
let img = image.unwrap();
assert_eq!(img.format.as_ref(), "jpeg");
assert_eq!(img.image_index, 3);
}
#[test]
fn test_decode_data_uri_unsupported_format() {
let uri = "data:image/tiff;base64,AAAA";
let image = crate::extractors::markdown_utils::decode_data_uri_image(uri, 0);
assert!(image.is_none());
}
#[test]
fn test_decode_data_uri_non_base64() {
let uri = "data:image/png,raw-data-here";
let image = crate::extractors::markdown_utils::decode_data_uri_image(uri, 0);
assert!(image.is_none());
}
#[test]
fn test_decode_data_uri_invalid_base64() {
let uri = "data:image/png;base64,!!!not-valid-base64!!!";
let image = crate::extractors::markdown_utils::decode_data_uri_image(uri, 0);
assert!(image.is_none());
}
#[test]
fn test_decode_data_uri_not_data_uri() {
let uri = "https://example.com/image.png";
let image = crate::extractors::markdown_utils::decode_data_uri_image(uri, 0);
assert!(image.is_none());
}
#[test]
fn test_extract_text_collects_data_uri_images() {
let png_b64 =
"iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVR42mP8/5+hHgAHggJ/PchI7wAAAABJRU5ErkJggg==";
let md = format!("# Title\n\n\n\nSome text.");
let parser = Parser::new_ext(&md, Options::ENABLE_TABLES);
let events: Vec<Event> = parser.collect();
let mut images = Vec::new();
let text = crate::extractors::markdown_utils::extract_text_from_events(&events, &mut images);
assert_eq!(images.len(), 1);
assert_eq!(images[0].format.as_ref(), "png");
assert!(text.contains("[Image: data:image/png;base64,"));
}
#[test]
fn test_extract_text_skips_http_images() {
let md = "# Title\n\n\n\nSome text.";
let parser = Parser::new_ext(md, Options::ENABLE_TABLES);
let events: Vec<Event> = parser.collect();
let mut images = Vec::new();
let text = crate::extractors::markdown_utils::extract_text_from_events(&events, &mut images);
assert!(images.is_empty());
assert!(text.contains("[Image: https://example.com/photo.jpg]"));
}
#[tokio::test]
async fn test_extract_bytes_with_data_uri_image() {
let png_b64 =
"iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVR42mP8/5+hHgAHggJ/PchI7wAAAABJRU5ErkJggg==";
let md = format!("# Doc\n\n\n\nText.");
let extractor = MarkdownExtractor::new();
let result = extractor
.extract_bytes(md.as_bytes(), "text/markdown", &ExtractionConfig::default())
.await
.expect("Should extract markdown with data URI image");
let result =
crate::extraction::derive::derive_extraction_result(result, true, crate::core::config::OutputFormat::Plain);
assert!(result.images.is_some());
let imgs = result.images.unwrap();
assert_eq!(imgs.len(), 1);
assert_eq!(imgs[0].format.as_ref(), "png");
}
#[tokio::test]
async fn test_extract_bytes_no_images() {
let md = b"# Simple\n\nJust text, no images.";
let extractor = MarkdownExtractor::new();
let result = extractor
.extract_bytes(md, "text/markdown", &ExtractionConfig::default())
.await
.expect("Should extract markdown without images");
let result =
crate::extraction::derive::derive_extraction_result(result, true, crate::core::config::OutputFormat::Plain);
assert!(result.images.is_none());
}
#[tokio::test]
async fn test_trimmed_paragraph_with_emoji() {
let md = b" **bold** \xf0\x9f\x8e\x89 text ";
let extractor = MarkdownExtractor::new();
let result = extractor
.extract_bytes(md, "text/markdown", &ExtractionConfig::default())
.await
.expect("Should handle emoji in trimmed paragraph");
let result =
crate::extraction::derive::derive_extraction_result(result, true, crate::core::config::OutputFormat::Plain);
assert!(result.content.contains("bold"), "Bold text preserved");
assert!(result.content.contains("\u{1F389}"), "Emoji preserved after trim");
}
#[tokio::test]
async fn test_cjk_paragraph_with_formatting() {
let md = "# CJK\n\nこれは**太字**テスト".as_bytes();
let extractor = MarkdownExtractor::new();
let result = extractor
.extract_bytes(md, "text/markdown", &ExtractionConfig::default())
.await
.expect("Should handle CJK with bold formatting");
let result =
crate::extraction::derive::derive_extraction_result(result, true, crate::core::config::OutputFormat::Plain);
assert!(result.content.contains("太字"), "Bold CJK content present");
assert!(result.content.contains("これは"), "Leading CJK preserved");
}
}