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};
use crate::types::{Metadata, Table};
use async_trait::async_trait;
use pulldown_cmark::{Event, Options, Parser, Tag, TagEnd};
use regex::Regex;
use std::borrow::Cow;
use std::sync::LazyLock;
type AnnotationEntry = (u8, u32, Option<(String, Option<String>)>);
static JSX_TAG_RE: LazyLock<Regex> =
LazyLock::new(|| Regex::new(r"</?[A-Z][a-zA-Z0-9_.]*(?:\s[^>]*)?>|<[A-Z][a-zA-Z0-9_.]*(?:\s[^>]*)?\s*/>").unwrap());
static JSX_EXPR_LINE_RE: LazyLock<Regex> = LazyLock::new(|| Regex::new(r"^\s*\{.*\}\s*$").unwrap());
static JSX_INLINE_COMMENT_RE: LazyLock<Regex> = LazyLock::new(|| Regex::new(r"\s*\{/\*.*?\*/\}").unwrap());
pub struct MdxExtractor;
impl MdxExtractor {
pub fn new() -> Self {
Self
}
#[cfg(test)]
pub(crate) fn strip_mdx_syntax(content: &str) -> String {
Self::strip_mdx_syntax_collecting(content, None)
}
fn strip_mdx_syntax_collecting(content: &str, mut jsx_blocks: Option<&mut Vec<String>>) -> String {
let mut result = String::with_capacity(content.len());
let mut in_code_fence = false;
let mut skip_block_depth: i32 = 0;
for line in content.lines() {
let trimmed = line.trim();
if trimmed.starts_with("```") {
in_code_fence = !in_code_fence;
result.push_str(line);
result.push('\n');
continue;
}
if in_code_fence {
result.push_str(line);
result.push('\n');
continue;
}
if skip_block_depth > 0 {
skip_block_depth += count_braces(trimmed);
if skip_block_depth <= 0 {
skip_block_depth = 0;
}
continue;
}
if trimmed.starts_with("import ") || trimmed == "import" {
let depth = count_braces(trimmed);
if depth > 0 {
skip_block_depth = depth;
}
continue;
}
if trimmed.starts_with("export ") || trimmed == "export" {
let depth = count_braces(trimmed);
if depth > 0 {
skip_block_depth = depth;
}
continue;
}
if JSX_EXPR_LINE_RE.is_match(trimmed) {
continue;
}
let without_comments = JSX_INLINE_COMMENT_RE.replace_all(line, "");
let processed = JSX_TAG_RE.replace_all(&without_comments, "");
let processed_trimmed = processed.trim();
if processed_trimmed.is_empty() && !trimmed.is_empty() {
if let Some(ref mut blocks) = jsx_blocks {
let tags: Vec<String> = JSX_TAG_RE
.find_iter(&without_comments)
.map(|m| m.as_str().to_string())
.collect();
for tag in tags {
blocks.push(tag);
}
}
continue;
}
result.push_str(&processed);
result.push('\n');
}
result
}
pub fn build_internal_document(
events: &[Event],
yaml: &Option<serde_yaml_ng::Value>,
raw_jsx_blocks: &[String],
) -> InternalDocument {
use crate::types::builder;
use crate::types::document_structure::TextAnnotation;
let mut b = InternalDocumentBuilder::new("mdx");
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);
}
}
for jsx in raw_jsx_blocks {
if !jsx.trim().is_empty() {
b.push_raw_block("jsx", jsx, 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 {
if in_paragraph {
let end = active_text_offset(¶graph_text);
if start < end {
paragraph_annotations.push(builder::link(start, end, &url, title.as_deref()));
}
} 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()));
}
} 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()));
}
}
if !url.is_empty() {
b.push_uri(Uri::hyperlink(&url, title));
}
}
}
}
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 {
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 {
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()
}
}
fn count_braces(line: &str) -> i32 {
let mut depth: i32 = 0;
for ch in line.chars() {
match ch {
'{' => depth += 1,
'}' => depth -= 1,
_ => {}
}
}
depth
}
impl Default for MdxExtractor {
fn default() -> Self {
Self::new()
}
}
impl Plugin for MdxExtractor {
fn name(&self) -> &str {
"mdx-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 MDX files by stripping JSX syntax and processing as Markdown"
}
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 MdxExtractor {
async fn extract_bytes(
&self,
content: &[u8],
mime_type: &str,
config: &ExtractionConfig,
) -> Result<InternalDocument> {
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()
};
let mut jsx_blocks_buf = Some(Vec::new());
let clean_markdown = Self::strip_mdx_syntax_collecting(&remaining_content, jsx_blocks_buf.as_mut());
if metadata.title.is_none()
&& let Some(title) = extract_title_from_content(&clean_markdown)
{
metadata.title = Some(title);
}
let mut options = Options::ENABLE_TABLES;
options |= Options::ENABLE_STRIKETHROUGH | Options::ENABLE_FOOTNOTES;
let parser = Parser::new_ext(&clean_markdown, options);
let events: Vec<Event> = parser.collect();
let mut extracted_images = Vec::new();
let _ = super::markdown_utils::extract_text_from_events(&events, &mut extracted_images);
let raw_jsx = jsx_blocks_buf.unwrap_or_default();
let mut doc = Self::build_internal_document(&events, &yaml, &raw_jsx);
doc.mime_type = Cow::Owned(mime_type.to_string());
doc.metadata = metadata;
if !extracted_images.is_empty() {
for image in extracted_images {
doc.push_image(image);
}
}
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/mdx", "text/x-mdx"]
}
fn priority(&self) -> i32 {
50
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_strip_import_statements() {
let input = "import { Chart } from './Chart'\nimport Alert from './Alert'\n\n# Hello\n";
let result = MdxExtractor::strip_mdx_syntax(input);
assert!(!result.contains("import"));
assert!(result.contains("# Hello"));
}
#[test]
fn test_strip_multiline_import() {
let input = "import {\n Chart,\n Table,\n} from './components'\n\n# Hello\n";
let result = MdxExtractor::strip_mdx_syntax(input);
assert!(!result.contains("import"));
assert!(!result.contains("Chart"));
assert!(result.contains("# Hello"));
}
#[test]
fn test_strip_export_statements() {
let input = "export const meta = { title: 'Hello' }\n\n# Hello\n";
let result = MdxExtractor::strip_mdx_syntax(input);
assert!(!result.contains("export"));
assert!(result.contains("# Hello"));
}
#[test]
fn test_strip_multiline_export() {
let input = "export const meta = {\n title: 'Hello',\n date: '2024-01-01',\n}\n\n# Hello\n";
let result = MdxExtractor::strip_mdx_syntax(input);
assert!(!result.contains("export"));
assert!(!result.contains("title"));
assert!(result.contains("# Hello"));
}
#[test]
fn test_strip_export_default() {
let input = "export default function Layout({ children }) { return children }\n\n# Hello\n";
let result = MdxExtractor::strip_mdx_syntax(input);
assert!(!result.contains("export"));
assert!(result.contains("# Hello"));
}
#[test]
fn test_strip_jsx_component_tags() {
let input = "# Hello\n\n<Alert type=\"warning\">\nBe careful!\n</Alert>\n\nMore text.\n";
let result = MdxExtractor::strip_mdx_syntax(input);
assert!(!result.contains("<Alert"));
assert!(!result.contains("</Alert>"));
assert!(result.contains("Be careful!"));
assert!(result.contains("More text."));
}
#[test]
fn test_strip_self_closing_jsx() {
let input = "# Hello\n\n<Chart data={data} />\n\nSome text.\n";
let result = MdxExtractor::strip_mdx_syntax(input);
assert!(!result.contains("<Chart"));
assert!(result.contains("Some text."));
}
#[test]
fn test_strip_jsx_expression_lines() {
let input = "# Hello\n\n{/* This is a comment */}\n\n{someExpression}\n\nText.\n";
let result = MdxExtractor::strip_mdx_syntax(input);
assert!(!result.contains("comment"));
assert!(!result.contains("someExpression"));
assert!(result.contains("Text."));
}
#[test]
fn test_preserve_code_fences() {
let input =
"# Hello\n\n```jsx\nimport React from 'react'\nconst x = <Component />\nexport default App\n```\n\nText.\n";
let result = MdxExtractor::strip_mdx_syntax(input);
assert!(result.contains("import React from 'react'"));
assert!(result.contains("<Component />"));
assert!(result.contains("export default App"));
assert!(result.contains("Text."));
}
#[test]
fn test_preserve_standard_html_tags() {
let input = "# Hello\n\n<div>Some content</div>\n\n<p>Paragraph</p>\n";
let result = MdxExtractor::strip_mdx_syntax(input);
assert!(result.contains("<div>"));
assert!(result.contains("</div>"));
assert!(result.contains("<p>"));
}
#[test]
fn test_preserve_markdown_content() {
let input = "# Title\n\nThis is **bold** and *italic* text.\n\n- Item 1\n- Item 2\n\n> Blockquote\n";
let result = MdxExtractor::strip_mdx_syntax(input);
assert!(result.contains("# Title"));
assert!(result.contains("**bold**"));
assert!(result.contains("*italic*"));
assert!(result.contains("- Item 1"));
assert!(result.contains("> Blockquote"));
}
#[test]
fn test_strip_complex_mdx() {
let input = r#"import { Chart } from './Chart'
import Alert from './Alert'
export const meta = {
title: 'My Post',
date: '2024-01-01',
}
# My Post
This is a paragraph with **bold** text.
<Alert type="warning">
Be careful with this!
</Alert>
<Chart data={data} />
{/* A comment */}
Some more text.
```javascript
const x = <div>Not JSX</div>
```
Final paragraph.
"#;
let result = MdxExtractor::strip_mdx_syntax(input);
assert!(!result.contains("import"));
assert!(!result.contains("export"));
assert!(!result.contains("<Alert"));
assert!(!result.contains("<Chart"));
assert!(!result.contains("comment"));
assert!(result.contains("# My Post"));
assert!(result.contains("**bold**"));
assert!(result.contains("Be careful with this!"));
assert!(result.contains("Some more text."));
assert!(result.contains("const x = <div>Not JSX</div>"));
assert!(result.contains("Final paragraph."));
}
#[test]
fn test_empty_content() {
let result = MdxExtractor::strip_mdx_syntax("");
assert!(result.is_empty());
}
#[test]
fn test_plugin_interface() {
let extractor = MdxExtractor::new();
assert_eq!(extractor.name(), "mdx-extractor");
assert_eq!(extractor.version(), env!("CARGO_PKG_VERSION"));
assert_eq!(extractor.priority(), 50);
assert_eq!(extractor.supported_mime_types(), &["text/mdx", "text/x-mdx"]);
}
#[tokio::test]
async fn test_extract_mdx_basic() {
let content = b"import Chart from './Chart'\n\n# Hello World\n\nThis is content.\n";
let extractor = MdxExtractor::new();
let result = extractor
.extract_bytes(content, "text/mdx", &ExtractionConfig::default())
.await
.expect("Should extract MDX content");
let result =
crate::extraction::derive::derive_extraction_result(result, true, crate::core::config::OutputFormat::Plain);
assert_eq!(result.mime_type, "text/mdx");
assert!(result.content.contains("Hello World"));
assert!(result.content.contains("This is content"));
assert!(!result.content.contains("import"));
}
#[tokio::test]
async fn test_extract_mdx_with_frontmatter() {
let content = b"---\ntitle: My MDX Post\nauthor: Test Author\ndate: 2024-01-15\n---\n\nimport Alert from './Alert'\n\n# Content\n\nBody text.\n";
let extractor = MdxExtractor::new();
let result = extractor
.extract_bytes(content, "text/mdx", &ExtractionConfig::default())
.await
.expect("Should extract MDX with frontmatter");
let result =
crate::extraction::derive::derive_extraction_result(result, true, crate::core::config::OutputFormat::Plain);
assert_eq!(result.metadata.title.as_deref(), Some("My MDX Post"));
assert_eq!(result.metadata.created_by.as_deref(), Some("Test Author"));
assert!(result.content.contains("Body text"));
assert!(!result.content.contains("import"));
}
#[tokio::test]
async fn test_extract_mdx_with_jsx_components() {
let content = b"# Title\n\n<Alert type=\"warning\">\nImportant message!\n</Alert>\n\nRegular text.\n";
let extractor = MdxExtractor::new();
let result = extractor
.extract_bytes(content, "text/mdx", &ExtractionConfig::default())
.await
.expect("Should extract MDX with JSX components");
let result =
crate::extraction::derive::derive_extraction_result(result, true, crate::core::config::OutputFormat::Plain);
assert!(result.content.contains("Important message"));
assert!(result.content.contains("Regular text"));
}
#[tokio::test]
async fn test_extract_mdx_with_tables() {
let content = b"# Tables\n\n| Header 1 | Header 2 |\n|----------|----------|\n| Cell 1 | Cell 2 |\n";
let extractor = MdxExtractor::new();
let result = extractor
.extract_bytes(content, "text/mdx", &ExtractionConfig::default())
.await
.expect("Should extract MDX 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_eq!(table.cells[0].len(), 2);
}
#[tokio::test]
async fn test_extract_mdx_title_from_heading() {
let content = b"# My Document Title\n\nContent here.\n";
let extractor = MdxExtractor::new();
let result = extractor
.extract_bytes(content, "text/mdx", &ExtractionConfig::default())
.await
.expect("Should extract title from heading");
let result =
crate::extraction::derive::derive_extraction_result(result, true, crate::core::config::OutputFormat::Plain);
assert_eq!(result.metadata.title, Some("My Document Title".to_string()));
}
#[test]
fn test_count_braces_balanced() {
assert_eq!(count_braces("{ a: 1 }"), 0);
}
#[test]
fn test_count_braces_opening() {
assert_eq!(count_braces("const x = {"), 1);
}
#[test]
fn test_count_braces_closing() {
assert_eq!(count_braces("}"), -1);
}
#[test]
fn test_count_braces_nested() {
assert_eq!(count_braces("{ a: { b: 1 }"), 1);
}
#[test]
fn test_count_braces_none() {
assert_eq!(count_braces("no braces here"), 0);
}
fn load_test_doc(relative_path: &str) -> Vec<u8> {
let manifest_dir = env!("CARGO_MANIFEST_DIR");
let path = std::path::Path::new(manifest_dir)
.parent()
.unwrap()
.parent()
.unwrap()
.join("test_documents")
.join(relative_path);
std::fs::read(&path).unwrap_or_else(|e| panic!("Failed to read {}: {}", path.display(), e))
}
#[tokio::test]
async fn test_extract_real_world_getting_started() {
let content = load_test_doc("markdown/mdx_getting_started.mdx");
let extractor = MdxExtractor::new();
let result = extractor
.extract_bytes(&content, "text/mdx", &ExtractionConfig::default())
.await
.expect("Should extract getting-started.mdx");
let result =
crate::extraction::derive::derive_extraction_result(result, true, crate::core::config::OutputFormat::Plain);
assert!(result.content.contains("Getting started"), "Missing main heading");
assert!(
result.content.contains("how to integrate MDX into your project"),
"Missing introductory text"
);
assert!(
result.content.contains("Prerequisites"),
"Missing Prerequisites section"
);
assert!(result.content.contains("Quick start"), "Missing Quick start section");
assert!(result.content.contains("Bundler"), "Missing Bundler section");
assert!(result.content.contains("Security"), "Missing Security section");
assert!(result.content.contains("Integrations"), "Missing Integrations section");
assert!(result.content.contains("React"), "Missing React mention");
assert!(result.content.contains("webpack"), "Missing webpack mention");
assert!(result.content.contains("esbuild"), "Missing esbuild mention");
assert!(
!result.content.contains("import {Note}"),
"import statement not stripped"
);
assert!(
!result.content.contains("export const info"),
"export const info not stripped"
);
assert!(
!result.content.contains("export const navSortSelf"),
"export const navSortSelf not stripped"
);
assert!(!result.content.contains("{/* more */}"), "JSX comment not stripped");
assert!(
result.content.contains("npm install @types/mdx"),
"Code block content should be preserved"
);
assert!(
result.content.len() > 2000,
"Extracted content too short: {} chars",
result.content.len()
);
}
#[tokio::test]
async fn test_extract_real_world_using_mdx() {
let content = load_test_doc("markdown/mdx_using_mdx.mdx");
let extractor = MdxExtractor::new();
let result = extractor
.extract_bytes(&content, "text/mdx", &ExtractionConfig::default())
.await
.expect("Should extract using-mdx.mdx");
let result =
crate::extraction::derive::derive_extraction_result(result, true, crate::core::config::OutputFormat::Plain);
assert!(result.content.contains("Using MDX"), "Missing main heading");
assert!(
result.content.contains("How MDX works"),
"Missing 'How MDX works' section"
);
assert!(result.content.contains("MDX content"), "Missing 'MDX content' section");
assert!(result.content.contains("Props"), "Missing Props section");
assert!(result.content.contains("Components"), "Missing Components section");
assert!(result.content.contains("Layout"), "Missing Layout section");
assert!(result.content.contains("MDX provider"), "Missing MDX provider section");
assert!(!result.content.contains("import {Note}"), "import not stripped");
assert!(!result.content.contains("export const info"), "export not stripped");
assert!(
result.content.len() > 2000,
"Extracted content too short: {} chars",
result.content.len()
);
}
#[tokio::test]
async fn test_extract_real_world_troubleshooting() {
let content = load_test_doc("markdown/mdx_troubleshooting.mdx");
let extractor = MdxExtractor::new();
let result = extractor
.extract_bytes(&content, "text/mdx", &ExtractionConfig::default())
.await
.expect("Should extract troubleshooting-mdx.mdx");
let result =
crate::extraction::derive::derive_extraction_result(result, true, crate::core::config::OutputFormat::Plain);
assert!(result.content.contains("Troubleshooting MDX"), "Missing main heading");
assert!(
result.content.contains("Problems integrating MDX"),
"Missing integrating section"
);
assert!(result.content.contains("ESM"), "Missing ESM section");
assert!(result.content.contains("Problems using MDX"), "Missing using section");
assert!(
result.content.contains("Problems writing MDX"),
"Missing writing section"
);
assert!(!result.content.contains("import {Note}"), "import not stripped");
assert!(!result.content.contains("export const info"), "export not stripped");
assert!(!result.content.contains("{/* lint disable"), "JSX comment not stripped");
assert!(
result.content.contains("Had trouble with something"),
"Content inside <Note> should be preserved"
);
assert!(
result.content.len() > 2000,
"Extracted content too short: {} chars",
result.content.len()
);
}
#[tokio::test]
async fn test_strip_mdx_real_world_multiline_exports() {
let input = r#"import {Note} from '../_component/note.jsx'
export const info = {
author: [
{github: 'wooorm', name: 'Titus Wormer'}
],
modified: new Date('2025-01-27'),
published: new Date('2021-10-05')
}
export const navSortSelf = 2
# Getting started
Content here.
"#;
let result = MdxExtractor::strip_mdx_syntax(input);
assert!(!result.contains("import"), "import not stripped");
assert!(!result.contains("export"), "export not stripped");
assert!(!result.contains("wooorm"), "Nested export content not stripped");
assert!(!result.contains("navSortSelf"), "Single-line export not stripped");
assert!(result.contains("# Getting started"), "Heading should be preserved");
assert!(result.contains("Content here"), "Content should be preserved");
}
#[tokio::test]
async fn test_trimmed_paragraph_with_emoji_mdx() {
let mdx = b" **bold** \xf0\x9f\x8e\x89 text ";
let extractor = MdxExtractor::new();
let result = extractor
.extract_bytes(mdx, "text/mdx", &ExtractionConfig::default())
.await
.expect("Should handle emoji in trimmed MDX 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_mdx() {
let mdx = "# CJK\n\nこれは**太字**テスト".as_bytes();
let extractor = MdxExtractor::new();
let result = extractor
.extract_bytes(mdx, "text/mdx", &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");
}
}