html_to_markdown_rs/metadata.rs
1//! Metadata extraction for HTML to Markdown conversion.
2//!
3//! This module provides comprehensive, type-safe metadata extraction during HTML-to-Markdown
4//! conversion, enabling content analysis, SEO optimization, and document indexing workflows.
5//! Metadata includes:
6//! - **Document metadata**: Title, description, author, language, canonical URL, Open Graph, Twitter Card
7//! - **Headers**: Heading elements (h1-h6) with hierarchy, IDs, and positions
8//! - **Links**: Hyperlinks with type classification (anchor, internal, external, email, phone)
9//! - **Images**: Image elements with source, alt text, dimensions, and type (data URI, external, etc.)
10//! - **Structured data**: JSON-LD, Microdata, and RDFa blocks
11//!
12//! The implementation follows a single-pass collector pattern for zero-overhead extraction
13//! when metadata features are disabled.
14//!
15//! # Architecture
16//!
17//! Metadata extraction uses the [`MetadataCollector`] pattern (similar to [`InlineImageCollector`]):
18//! - **Single-pass collection**: Metadata is gathered during the primary tree traversal without additional passes
19//! - **Zero overhead when disabled**: Entire module can be compiled out via feature flags
20//! - **Configurable granularity**: Use [`MetadataConfig`] to select which metadata types to extract
21//! - **Type-safe APIs**: All metadata types are enum-based with exhaustive matching
22//! - **Memory-bounded**: Size limits prevent memory exhaustion from adversarial documents
23//! - **Pre-allocated buffers**: Typical documents (32 headers, 64 links, 16 images) handled efficiently
24//!
25//! # Type Overview
26//!
27//! ## Enumerations
28//!
29//! - [`TextDirection`]: Document directionality (LTR, RTL, Auto)
30//! - [`LinkType`]: Link classification (Anchor, Internal, External, Email, Phone, Other)
31//! - [`ImageType`]: Image source type (DataUri, External, Relative, InlineSvg)
32//! - [`StructuredDataType`]: Structured data format (JsonLd, Microdata, RDFa)
33//!
34//! ## Structures
35//!
36//! - [`DocumentMetadata`]: Head-level metadata with maps for Open Graph and Twitter Card
37//! - [`HeaderMetadata`]: Heading element with level (1-6), text, ID, hierarchy depth, and position
38//! - [`LinkMetadata`]: Hyperlink with href, text, title, type, rel attributes, and custom attributes
39//! - [`ImageMetadata`]: Image element with src, alt, title, dimensions, type, and attributes
40//! - [`StructuredData`]: Structured data block with type and raw JSON
41//! - [`MetadataConfig`]: Configuration controlling extraction granularity and size limits
42//! - [`ExtendedMetadata`]: Top-level result containing all extracted metadata
43//!
44//! # Examples
45//!
46//! ## Basic Usage with convert_with_metadata
47//!
48//! ```ignore
49//! use html_to_markdown_rs::{convert_with_metadata, MetadataConfig};
50//!
51//! let html = r#"
52//! <html lang="en">
53//! <head>
54//! <title>My Article</title>
55//! <meta name="description" content="An interesting read">
56//! </head>
57//! <body>
58//! <h1 id="main">Title</h1>
59//! <a href="https://example.com">External Link</a>
60//! <img src="photo.jpg" alt="A photo">
61//! </body>
62//! </html>
63//! "#;
64//!
65//! let config = MetadataConfig::default();
66//! let (markdown, metadata) = convert_with_metadata(html, None, config)?;
67//!
68//! // Access document metadata
69//! assert_eq!(metadata.document.title, Some("My Article".to_string()));
70//! assert_eq!(metadata.document.language, Some("en".to_string()));
71//!
72//! // Access headers
73//! assert_eq!(metadata.headers.len(), 1);
74//! assert_eq!(metadata.headers[0].level, 1);
75//! assert_eq!(metadata.headers[0].id, Some("main".to_string()));
76//!
77//! // Access links
78//! assert_eq!(metadata.links.len(), 1);
79//! assert_eq!(metadata.links[0].link_type, LinkType::External);
80//!
81//! // Access images
82//! assert_eq!(metadata.images.len(), 1);
83//! assert_eq!(metadata.images[0].image_type, ImageType::Relative);
84//! # Ok::<(), html_to_markdown_rs::ConversionError>(())
85//! ```
86//!
87//! ## Selective Extraction
88//!
89//! ```ignore
90//! use html_to_markdown_rs::{convert_with_metadata, MetadataConfig};
91//!
92//! let config = MetadataConfig {
93//! extract_headers: true,
94//! extract_links: true,
95//! extract_images: false, // Skip images
96//! extract_structured_data: false, // Skip structured data
97//! max_structured_data_size: 0,
98//! };
99//!
100//! let (markdown, metadata) = convert_with_metadata(html, None, config)?;
101//! assert_eq!(metadata.images.len(), 0); // Images not extracted
102//! # Ok::<(), html_to_markdown_rs::ConversionError>(())
103//! ```
104//!
105//! ## Analyzing Link Types
106//!
107//! ```ignore
108//! use html_to_markdown_rs::{convert_with_metadata, MetadataConfig};
109//! use html_to_markdown_rs::metadata::LinkType;
110//!
111//! let (_markdown, metadata) = convert_with_metadata(html, None, MetadataConfig::default())?;
112//!
113//! for link in &metadata.links {
114//! match link.link_type {
115//! LinkType::External => println!("External: {}", link.href),
116//! LinkType::Internal => println!("Internal: {}", link.href),
117//! LinkType::Anchor => println!("Anchor: {}", link.href),
118//! LinkType::Email => println!("Email: {}", link.href),
119//! _ => {}
120//! }
121//! }
122//! # Ok::<(), html_to_markdown_rs::ConversionError>(())
123//! ```
124//!
125//! # Serialization
126//!
127//! All types in this module support serialization via `serde` when the `metadata` feature is enabled.
128//! This enables easy export to JSON, YAML, or other formats:
129//!
130//! ```ignore
131//! use html_to_markdown_rs::{convert_with_metadata, MetadataConfig};
132//!
133//! let (_markdown, metadata) = convert_with_metadata(html, None, MetadataConfig::default())?;
134//! let json = serde_json::to_string_pretty(&metadata)?;
135//! println!("{}", json);
136//! # Ok::<(), Box<dyn std::error::Error>>(())
137//! ```
138
139use std::cell::RefCell;
140use std::collections::BTreeMap;
141use std::rc::Rc;
142
143/// Text directionality of document content.
144///
145/// Corresponds to the HTML `dir` attribute and `bdi` element directionality.
146#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
147#[cfg_attr(feature = "metadata", derive(serde::Serialize, serde::Deserialize))]
148pub enum TextDirection {
149 /// Left-to-right text flow (default for Latin scripts)
150 #[cfg_attr(feature = "metadata", serde(rename = "ltr"))]
151 LeftToRight,
152 /// Right-to-left text flow (Hebrew, Arabic, Urdu, etc.)
153 #[cfg_attr(feature = "metadata", serde(rename = "rtl"))]
154 RightToLeft,
155 /// Automatic directionality detection
156 #[cfg_attr(feature = "metadata", serde(rename = "auto"))]
157 Auto,
158}
159
160impl std::fmt::Display for TextDirection {
161 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
162 match self {
163 Self::LeftToRight => write!(f, "ltr"),
164 Self::RightToLeft => write!(f, "rtl"),
165 Self::Auto => write!(f, "auto"),
166 }
167 }
168}
169
170impl TextDirection {
171 /// Parse a text direction from string value.
172 ///
173 /// # Arguments
174 ///
175 /// * `s` - Direction string ("ltr", "rtl", or "auto")
176 ///
177 /// # Returns
178 ///
179 /// `Some(TextDirection)` if valid, `None` otherwise.
180 ///
181 /// # Examples
182 ///
183 /// ```
184 /// # use html_to_markdown_rs::metadata::TextDirection;
185 /// assert_eq!(TextDirection::parse("ltr"), Some(TextDirection::LeftToRight));
186 /// assert_eq!(TextDirection::parse("rtl"), Some(TextDirection::RightToLeft));
187 /// assert_eq!(TextDirection::parse("auto"), Some(TextDirection::Auto));
188 /// assert_eq!(TextDirection::parse("invalid"), None);
189 /// ```
190 pub fn parse(s: &str) -> Option<Self> {
191 if s.eq_ignore_ascii_case("ltr") {
192 return Some(Self::LeftToRight);
193 }
194 if s.eq_ignore_ascii_case("rtl") {
195 return Some(Self::RightToLeft);
196 }
197 if s.eq_ignore_ascii_case("auto") {
198 return Some(Self::Auto);
199 }
200 None
201 }
202}
203
204/// Link classification based on href value and document context.
205///
206/// Used to categorize links during extraction for filtering and analysis.
207#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
208#[cfg_attr(feature = "metadata", derive(serde::Serialize, serde::Deserialize))]
209#[cfg_attr(feature = "metadata", serde(rename_all = "snake_case"))]
210pub enum LinkType {
211 /// Anchor link within same document (href starts with #)
212 Anchor,
213 /// Internal link within same domain
214 Internal,
215 /// External link to different domain
216 External,
217 /// Email link (mailto:)
218 Email,
219 /// Phone link (tel:)
220 Phone,
221 /// Other protocol or unclassifiable
222 Other,
223}
224
225impl std::fmt::Display for LinkType {
226 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
227 match self {
228 Self::Anchor => write!(f, "anchor"),
229 Self::Internal => write!(f, "internal"),
230 Self::External => write!(f, "external"),
231 Self::Email => write!(f, "email"),
232 Self::Phone => write!(f, "phone"),
233 Self::Other => write!(f, "other"),
234 }
235 }
236}
237
238/// Image source classification for proper handling and processing.
239///
240/// Determines whether an image is embedded (data URI), inline SVG, external, or relative.
241#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
242#[cfg_attr(feature = "metadata", derive(serde::Serialize, serde::Deserialize))]
243#[cfg_attr(feature = "metadata", serde(rename_all = "snake_case"))]
244pub enum ImageType {
245 /// Data URI embedded image (base64 or other encoding)
246 DataUri,
247 /// Inline SVG element
248 InlineSvg,
249 /// External image URL (http/https)
250 External,
251 /// Relative image path
252 Relative,
253}
254
255impl std::fmt::Display for ImageType {
256 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
257 match self {
258 Self::DataUri => write!(f, "data_uri"),
259 Self::InlineSvg => write!(f, "inline_svg"),
260 Self::External => write!(f, "external"),
261 Self::Relative => write!(f, "relative"),
262 }
263 }
264}
265
266/// Structured data format type.
267///
268/// Identifies the schema/format used for structured data markup.
269#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
270#[cfg_attr(feature = "metadata", derive(serde::Serialize, serde::Deserialize))]
271#[cfg_attr(feature = "metadata", serde(rename_all = "snake_case"))]
272pub enum StructuredDataType {
273 /// JSON-LD (JSON for Linking Data) script blocks
274 #[cfg_attr(feature = "metadata", serde(rename = "json_ld"))]
275 JsonLd,
276 /// HTML5 Microdata attributes (itemscope, itemtype, itemprop)
277 Microdata,
278 /// RDF in Attributes (RDFa) markup
279 #[cfg_attr(feature = "metadata", serde(rename = "rdfa"))]
280 RDFa,
281}
282
283impl std::fmt::Display for StructuredDataType {
284 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
285 match self {
286 Self::JsonLd => write!(f, "json_ld"),
287 Self::Microdata => write!(f, "microdata"),
288 Self::RDFa => write!(f, "rdfa"),
289 }
290 }
291}
292
293/// Document-level metadata extracted from `<head>` and top-level elements.
294///
295/// Contains all metadata typically used by search engines, social media platforms,
296/// and browsers for document indexing and presentation.
297///
298/// # Examples
299///
300/// ```
301/// # use html_to_markdown_rs::metadata::DocumentMetadata;
302/// let doc = DocumentMetadata {
303/// title: Some("My Article".to_string()),
304/// description: Some("A great article about Rust".to_string()),
305/// keywords: vec!["rust".to_string(), "programming".to_string()],
306/// ..Default::default()
307/// };
308///
309/// assert_eq!(doc.title, Some("My Article".to_string()));
310/// ```
311#[derive(Debug, Clone, Default)]
312#[cfg_attr(feature = "metadata", derive(serde::Serialize, serde::Deserialize))]
313pub struct DocumentMetadata {
314 /// Document title from `<title>` tag
315 pub title: Option<String>,
316
317 /// Document description from `<meta name="description">` tag
318 pub description: Option<String>,
319
320 /// Document keywords from `<meta name="keywords">` tag, split on commas
321 pub keywords: Vec<String>,
322
323 /// Document author from `<meta name="author">` tag
324 pub author: Option<String>,
325
326 /// Canonical URL from `<link rel="canonical">` tag
327 pub canonical_url: Option<String>,
328
329 /// Base URL from `<base href="">` tag for resolving relative URLs
330 pub base_href: Option<String>,
331
332 /// Document language from `lang` attribute
333 pub language: Option<String>,
334
335 /// Document text direction from `dir` attribute
336 pub text_direction: Option<TextDirection>,
337
338 /// Open Graph metadata (og:* properties) for social media
339 /// Keys like "title", "description", "image", "url", etc.
340 pub open_graph: BTreeMap<String, String>,
341
342 /// Twitter Card metadata (twitter:* properties)
343 /// Keys like "card", "site", "creator", "title", "description", "image", etc.
344 pub twitter_card: BTreeMap<String, String>,
345
346 /// Additional meta tags not covered by specific fields
347 /// Keys are meta name/property attributes, values are content
348 pub meta_tags: BTreeMap<String, String>,
349}
350
351/// Header element metadata with hierarchy tracking.
352///
353/// Captures heading elements (h1-h6) with their text content, identifiers,
354/// and position in the document structure.
355///
356/// # Examples
357///
358/// ```
359/// # use html_to_markdown_rs::metadata::HeaderMetadata;
360/// let header = HeaderMetadata {
361/// level: 1,
362/// text: "Main Title".to_string(),
363/// id: Some("main-title".to_string()),
364/// depth: 0,
365/// html_offset: 145,
366/// };
367///
368/// assert_eq!(header.level, 1);
369/// assert!(header.is_valid());
370/// ```
371#[derive(Debug, Clone)]
372#[cfg_attr(feature = "metadata", derive(serde::Serialize, serde::Deserialize))]
373pub struct HeaderMetadata {
374 /// Header level: 1 (h1) through 6 (h6)
375 pub level: u8,
376
377 /// Normalized text content of the header
378 pub text: String,
379
380 /// HTML id attribute if present
381 pub id: Option<String>,
382
383 /// Document tree depth at the header element
384 pub depth: usize,
385
386 /// Byte offset in original HTML document
387 pub html_offset: usize,
388}
389
390impl HeaderMetadata {
391 /// Validate that the header level is within valid range (1-6).
392 ///
393 /// # Returns
394 ///
395 /// `true` if level is 1-6, `false` otherwise.
396 ///
397 /// # Examples
398 ///
399 /// ```
400 /// # use html_to_markdown_rs::metadata::HeaderMetadata;
401 /// let valid = HeaderMetadata {
402 /// level: 3,
403 /// text: "Title".to_string(),
404 /// id: None,
405 /// depth: 2,
406 /// html_offset: 100,
407 /// };
408 /// assert!(valid.is_valid());
409 ///
410 /// let invalid = HeaderMetadata {
411 /// level: 7, // Invalid
412 /// text: "Title".to_string(),
413 /// id: None,
414 /// depth: 2,
415 /// html_offset: 100,
416 /// };
417 /// assert!(!invalid.is_valid());
418 /// ```
419 pub fn is_valid(&self) -> bool {
420 self.level >= 1 && self.level <= 6
421 }
422}
423
424/// Hyperlink metadata with categorization and attributes.
425///
426/// Represents `<a>` elements with parsed href values, text content, and link type classification.
427///
428/// # Examples
429///
430/// ```
431/// # use html_to_markdown_rs::metadata::{LinkMetadata, LinkType};
432/// let link = LinkMetadata {
433/// href: "https://example.com".to_string(),
434/// text: "Example".to_string(),
435/// title: Some("Visit Example".to_string()),
436/// link_type: LinkType::External,
437/// rel: vec!["nofollow".to_string()],
438/// attributes: Default::default(),
439/// };
440///
441/// assert_eq!(link.link_type, LinkType::External);
442/// assert_eq!(link.text, "Example");
443/// ```
444#[derive(Debug, Clone)]
445#[cfg_attr(feature = "metadata", derive(serde::Serialize, serde::Deserialize))]
446pub struct LinkMetadata {
447 /// The href URL value
448 pub href: String,
449
450 /// Link text content (normalized, concatenated if mixed with elements)
451 pub text: String,
452
453 /// Optional title attribute (often shown as tooltip)
454 pub title: Option<String>,
455
456 /// Link type classification
457 pub link_type: LinkType,
458
459 /// Rel attribute values (e.g., "nofollow", "stylesheet", "canonical")
460 pub rel: Vec<String>,
461
462 /// Additional HTML attributes
463 pub attributes: BTreeMap<String, String>,
464}
465
466impl LinkMetadata {
467 /// Classify a link based on href value.
468 ///
469 /// # Arguments
470 ///
471 /// * `href` - The href attribute value
472 ///
473 /// # Returns
474 ///
475 /// Appropriate [`LinkType`] based on protocol and content.
476 ///
477 /// # Examples
478 ///
479 /// ```
480 /// # use html_to_markdown_rs::metadata::{LinkMetadata, LinkType};
481 /// assert_eq!(LinkMetadata::classify_link("#section"), LinkType::Anchor);
482 /// assert_eq!(LinkMetadata::classify_link("mailto:test@example.com"), LinkType::Email);
483 /// assert_eq!(LinkMetadata::classify_link("tel:+1234567890"), LinkType::Phone);
484 /// assert_eq!(LinkMetadata::classify_link("https://example.com"), LinkType::External);
485 /// ```
486 pub fn classify_link(href: &str) -> LinkType {
487 if href.starts_with('#') {
488 LinkType::Anchor
489 } else if href.starts_with("mailto:") {
490 LinkType::Email
491 } else if href.starts_with("tel:") {
492 LinkType::Phone
493 } else if href.starts_with("http://") || href.starts_with("https://") {
494 LinkType::External
495 } else if href.starts_with('/') || href.starts_with("../") || href.starts_with("./") {
496 LinkType::Internal
497 } else {
498 LinkType::Other
499 }
500 }
501}
502
503/// Image metadata with source and dimensions.
504///
505/// Captures `<img>` elements and inline `<svg>` elements with metadata
506/// for image analysis and optimization.
507///
508/// # Examples
509///
510/// ```
511/// # use html_to_markdown_rs::metadata::{ImageMetadata, ImageType};
512/// let img = ImageMetadata {
513/// src: "https://example.com/image.jpg".to_string(),
514/// alt: Some("An example image".to_string()),
515/// title: Some("Example".to_string()),
516/// dimensions: Some((800, 600)),
517/// image_type: ImageType::External,
518/// attributes: Default::default(),
519/// };
520///
521/// assert_eq!(img.image_type, ImageType::External);
522/// ```
523#[derive(Debug, Clone)]
524#[cfg_attr(feature = "metadata", derive(serde::Serialize, serde::Deserialize))]
525pub struct ImageMetadata {
526 /// Image source (URL, data URI, or SVG content identifier)
527 pub src: String,
528
529 /// Alternative text from alt attribute (for accessibility)
530 pub alt: Option<String>,
531
532 /// Title attribute (often shown as tooltip)
533 pub title: Option<String>,
534
535 /// Image dimensions as (width, height) if available
536 pub dimensions: Option<(u32, u32)>,
537
538 /// Image type classification
539 pub image_type: ImageType,
540
541 /// Additional HTML attributes
542 pub attributes: BTreeMap<String, String>,
543}
544
545/// Structured data block (JSON-LD, Microdata, or RDFa).
546///
547/// Represents machine-readable structured data found in the document.
548/// JSON-LD blocks are collected as raw JSON strings for flexibility.
549///
550/// # Examples
551///
552/// ```
553/// # use html_to_markdown_rs::metadata::{StructuredData, StructuredDataType};
554/// let schema = StructuredData {
555/// data_type: StructuredDataType::JsonLd,
556/// raw_json: r#"{"@context":"https://schema.org","@type":"Article"}"#.to_string(),
557/// schema_type: Some("Article".to_string()),
558/// };
559///
560/// assert_eq!(schema.data_type, StructuredDataType::JsonLd);
561/// ```
562#[derive(Debug, Clone)]
563#[cfg_attr(feature = "metadata", derive(serde::Serialize, serde::Deserialize))]
564pub struct StructuredData {
565 /// Type of structured data (JSON-LD, Microdata, RDFa)
566 pub data_type: StructuredDataType,
567
568 /// Raw JSON string (for JSON-LD) or serialized representation
569 pub raw_json: String,
570
571 /// Schema type if detectable (e.g., "Article", "Event", "Product")
572 pub schema_type: Option<String>,
573}
574
575/// Default maximum size for structured data extraction (1 MB)
576pub const DEFAULT_MAX_STRUCTURED_DATA_SIZE: usize = 1_000_000;
577
578/// Configuration for metadata extraction granularity.
579///
580/// Controls which metadata types are extracted and size limits for safety.
581///
582/// # Examples
583///
584/// ```
585/// # use html_to_markdown_rs::metadata::MetadataConfig;
586/// let config = MetadataConfig {
587/// extract_document: true,
588/// extract_headers: true,
589/// extract_links: true,
590/// extract_images: true,
591/// extract_structured_data: true,
592/// max_structured_data_size: 1_000_000,
593/// };
594///
595/// assert!(config.extract_headers);
596/// ```
597#[derive(Debug, Clone)]
598#[cfg_attr(feature = "metadata", derive(serde::Serialize, serde::Deserialize))]
599pub struct MetadataConfig {
600 /// Extract document-level metadata (title, description, author, etc.)
601 pub extract_document: bool,
602
603 /// Extract h1-h6 header elements and their hierarchy
604 pub extract_headers: bool,
605
606 /// Extract anchor (a) elements as links with type classification
607 pub extract_links: bool,
608
609 /// Extract image elements and data URIs
610 pub extract_images: bool,
611
612 /// Extract structured data (JSON-LD, Microdata, RDFa)
613 pub extract_structured_data: bool,
614
615 /// Maximum total size of structured data to collect (bytes)
616 /// Prevents memory exhaustion on malformed or adversarial documents
617 pub max_structured_data_size: usize,
618}
619
620/// Partial update for MetadataConfig.
621#[derive(Debug, Clone, Default)]
622#[cfg_attr(any(feature = "serde", feature = "metadata"), derive(serde::Deserialize))]
623#[cfg_attr(any(feature = "serde", feature = "metadata"), serde(rename_all = "camelCase"))]
624pub struct MetadataConfigUpdate {
625 #[cfg_attr(any(feature = "serde", feature = "metadata"), serde(alias = "extract_document"))]
626 pub extract_document: Option<bool>,
627 #[cfg_attr(any(feature = "serde", feature = "metadata"), serde(alias = "extract_headers"))]
628 pub extract_headers: Option<bool>,
629 #[cfg_attr(any(feature = "serde", feature = "metadata"), serde(alias = "extract_links"))]
630 pub extract_links: Option<bool>,
631 #[cfg_attr(any(feature = "serde", feature = "metadata"), serde(alias = "extract_images"))]
632 pub extract_images: Option<bool>,
633 #[cfg_attr(
634 any(feature = "serde", feature = "metadata"),
635 serde(alias = "extract_structured_data")
636 )]
637 pub extract_structured_data: Option<bool>,
638 #[cfg_attr(
639 any(feature = "serde", feature = "metadata"),
640 serde(alias = "max_structured_data_size")
641 )]
642 pub max_structured_data_size: Option<usize>,
643}
644
645impl Default for MetadataConfig {
646 /// Create default metadata configuration.
647 ///
648 /// Defaults to extracting all metadata types with 1MB limit on structured data.
649 fn default() -> Self {
650 Self {
651 extract_document: true,
652 extract_headers: true,
653 extract_links: true,
654 extract_images: true,
655 extract_structured_data: true,
656 max_structured_data_size: DEFAULT_MAX_STRUCTURED_DATA_SIZE,
657 }
658 }
659}
660
661impl MetadataConfig {
662 pub fn any_enabled(&self) -> bool {
663 self.extract_document
664 || self.extract_headers
665 || self.extract_links
666 || self.extract_images
667 || self.extract_structured_data
668 }
669
670 pub fn apply_update(&mut self, update: MetadataConfigUpdate) {
671 if let Some(extract_document) = update.extract_document {
672 self.extract_document = extract_document;
673 }
674 if let Some(extract_headers) = update.extract_headers {
675 self.extract_headers = extract_headers;
676 }
677 if let Some(extract_links) = update.extract_links {
678 self.extract_links = extract_links;
679 }
680 if let Some(extract_images) = update.extract_images {
681 self.extract_images = extract_images;
682 }
683 if let Some(extract_structured_data) = update.extract_structured_data {
684 self.extract_structured_data = extract_structured_data;
685 }
686 if let Some(max_structured_data_size) = update.max_structured_data_size {
687 self.max_structured_data_size = max_structured_data_size;
688 }
689 }
690
691 pub fn from_update(update: MetadataConfigUpdate) -> Self {
692 let mut config = Self::default();
693 config.apply_update(update);
694 config
695 }
696}
697
698impl From<MetadataConfigUpdate> for MetadataConfig {
699 fn from(update: MetadataConfigUpdate) -> Self {
700 Self::from_update(update)
701 }
702}
703
704/// Comprehensive metadata extraction result from HTML document.
705///
706/// Contains all extracted metadata types in a single structure,
707/// suitable for serialization and transmission across language boundaries.
708///
709/// # Examples
710///
711/// ```
712/// # use html_to_markdown_rs::metadata::ExtendedMetadata;
713/// let metadata = ExtendedMetadata {
714/// document: Default::default(),
715/// headers: Vec::new(),
716/// links: Vec::new(),
717/// images: Vec::new(),
718/// structured_data: Vec::new(),
719/// };
720///
721/// assert!(metadata.headers.is_empty());
722/// ```
723#[derive(Debug, Clone, Default)]
724#[cfg_attr(feature = "metadata", derive(serde::Serialize, serde::Deserialize))]
725pub struct ExtendedMetadata {
726 /// Document-level metadata (title, description, canonical, etc.)
727 pub document: DocumentMetadata,
728
729 /// Extracted header elements with hierarchy
730 pub headers: Vec<HeaderMetadata>,
731
732 /// Extracted hyperlinks with type classification
733 pub links: Vec<LinkMetadata>,
734
735 /// Extracted images with source and dimensions
736 pub images: Vec<ImageMetadata>,
737
738 /// Extracted structured data blocks
739 pub structured_data: Vec<StructuredData>,
740}
741
742/// Internal metadata collector for single-pass extraction.
743///
744/// Follows the [`InlineImageCollector`](crate::inline_images::InlineImageCollector) pattern
745/// for efficient metadata extraction during tree traversal. Maintains state for:
746/// - Document metadata from head elements
747/// - Header hierarchy tracking
748/// - Link accumulation
749/// - Structured data collection
750/// - Language and directionality attributes
751///
752/// # Architecture
753///
754/// The collector is designed to be:
755/// - **Performant**: Pre-allocated collections, minimal cloning
756/// - **Single-pass**: Collects during main tree walk without separate passes
757/// - **Optional**: Zero overhead when disabled via feature flags
758/// - **Type-safe**: Strict separation of collection and result types
759///
760/// # Internal State
761///
762/// - `head_metadata`: Raw metadata pairs from head element
763/// - `headers`: Collected header elements
764/// - `header_stack`: For tracking nesting depth
765/// - `links`: Collected link elements
766/// - `base_href`: Base URL for relative link resolution
767/// - `json_ld`: JSON-LD script block contents
768/// - `lang`: Document language
769/// - `dir`: Document text direction
770#[derive(Debug)]
771#[allow(dead_code)]
772pub(crate) struct MetadataCollector {
773 head_metadata: BTreeMap<String, String>,
774 headers: Vec<HeaderMetadata>,
775 header_stack: Vec<usize>,
776 links: Vec<LinkMetadata>,
777 images: Vec<ImageMetadata>,
778 json_ld: Vec<String>,
779 structured_data_size: usize,
780 config: MetadataConfig,
781 lang: Option<String>,
782 dir: Option<String>,
783}
784
785#[allow(dead_code)]
786impl MetadataCollector {
787 /// Create a new metadata collector with configuration.
788 ///
789 /// Pre-allocates collections based on typical document sizes
790 /// for efficient append operations during traversal.
791 ///
792 /// # Arguments
793 ///
794 /// * `config` - Extraction configuration specifying which types to collect
795 ///
796 /// # Returns
797 ///
798 /// A new collector ready for use during tree traversal.
799 ///
800 /// # Examples
801 ///
802 /// ```ignore
803 /// let config = MetadataConfig::default();
804 /// let collector = MetadataCollector::new(config);
805 /// ```
806 pub(crate) fn new(config: MetadataConfig) -> Self {
807 Self {
808 head_metadata: BTreeMap::new(),
809 headers: Vec::with_capacity(32),
810 header_stack: Vec::with_capacity(6),
811 links: Vec::with_capacity(64),
812 images: Vec::with_capacity(16),
813 json_ld: Vec::with_capacity(4),
814 structured_data_size: 0,
815 config,
816 lang: None,
817 dir: None,
818 }
819 }
820
821 /// Add a header element to the collection.
822 ///
823 /// Validates that level is in range 1-6 and tracks hierarchy via depth.
824 ///
825 /// # Arguments
826 ///
827 /// * `level` - Header level (1-6)
828 /// * `text` - Normalized header text content
829 /// * `id` - Optional HTML id attribute
830 /// * `depth` - Current document nesting depth
831 /// * `html_offset` - Byte offset in original HTML
832 pub(crate) fn add_header(&mut self, level: u8, text: String, id: Option<String>, depth: usize, html_offset: usize) {
833 if !self.config.extract_headers {
834 return;
835 }
836
837 if !(1..=6).contains(&level) {
838 return;
839 }
840
841 let header = HeaderMetadata {
842 level,
843 text,
844 id,
845 depth,
846 html_offset,
847 };
848
849 self.headers.push(header);
850 }
851
852 /// Add a link element to the collection.
853 ///
854 /// Classifies the link based on href value and stores with metadata.
855 ///
856 /// # Arguments
857 ///
858 /// * `href` - The href attribute value
859 /// * `text` - Link text content
860 /// * `title` - Optional title attribute
861 /// * `rel` - Comma/space-separated rel attribute value
862 /// * `attributes` - Additional attributes to capture (e.g., data-* or aria-* values)
863 pub(crate) fn add_link(
864 &mut self,
865 href: String,
866 text: String,
867 title: Option<String>,
868 rel: Option<String>,
869 attributes: BTreeMap<String, String>,
870 ) {
871 if !self.config.extract_links {
872 return;
873 }
874
875 let link_type = LinkMetadata::classify_link(&href);
876
877 let rel_vec = rel
878 .map(|r| r.split_whitespace().map(|s| s.to_string()).collect::<Vec<_>>())
879 .unwrap_or_default();
880
881 let link = LinkMetadata {
882 href,
883 text,
884 title,
885 link_type,
886 rel: rel_vec,
887 attributes,
888 };
889
890 self.links.push(link);
891 }
892
893 /// Add an image element to the collection.
894 ///
895 /// # Arguments
896 ///
897 /// * `src` - Image source (URL or data URI)
898 /// * `alt` - Optional alt text
899 /// * `title` - Optional title attribute
900 /// * `dimensions` - Optional (width, height) tuple
901 pub(crate) fn add_image(
902 &mut self,
903 src: String,
904 alt: Option<String>,
905 title: Option<String>,
906 dimensions: Option<(u32, u32)>,
907 attributes: BTreeMap<String, String>,
908 ) {
909 if !self.config.extract_images {
910 return;
911 }
912
913 let image_type = if src.starts_with("data:") {
914 ImageType::DataUri
915 } else if src.starts_with("http://") || src.starts_with("https://") {
916 ImageType::External
917 } else if src.starts_with('<') && src.contains("svg") {
918 ImageType::InlineSvg
919 } else {
920 ImageType::Relative
921 };
922
923 let image = ImageMetadata {
924 src,
925 alt,
926 title,
927 dimensions,
928 image_type,
929 attributes,
930 };
931
932 self.images.push(image);
933 }
934
935 /// Add a JSON-LD structured data block.
936 ///
937 /// Accumulates JSON content with size validation against configured limits.
938 ///
939 /// # Arguments
940 ///
941 /// * `json_content` - Raw JSON string content
942 pub(crate) fn add_json_ld(&mut self, json_content: String) {
943 if !self.config.extract_structured_data {
944 return;
945 }
946
947 let content_size = json_content.len();
948 if content_size > self.config.max_structured_data_size {
949 return;
950 }
951 if self.structured_data_size + content_size > self.config.max_structured_data_size {
952 return;
953 }
954
955 self.structured_data_size += content_size;
956 self.json_ld.push(json_content);
957 }
958
959 /// Set document head metadata from extracted head section.
960 ///
961 /// Merges metadata pairs from head elements (meta, title, link, etc.)
962 /// into the collector's head metadata store.
963 ///
964 /// # Arguments
965 ///
966 /// * `metadata` - BTreeMap of metadata key-value pairs
967 pub(crate) fn set_head_metadata(&mut self, metadata: BTreeMap<String, String>) {
968 if !self.config.extract_document {
969 return;
970 }
971 self.head_metadata.extend(metadata);
972 }
973
974 /// Set document language attribute.
975 ///
976 /// Usually from `lang` attribute on `<html>` or `<body>` tag.
977 /// Only sets if not already set (first occurrence wins).
978 ///
979 /// # Arguments
980 ///
981 /// * `lang` - Language code (e.g., "en", "es", "fr")
982 pub(crate) fn set_language(&mut self, lang: String) {
983 if !self.config.extract_document {
984 return;
985 }
986 if self.lang.is_none() {
987 self.lang = Some(lang);
988 }
989 }
990
991 /// Set document text direction attribute.
992 ///
993 /// Usually from `dir` attribute on `<html>` or `<body>` tag.
994 /// Only sets if not already set (first occurrence wins).
995 ///
996 /// # Arguments
997 ///
998 /// * `dir` - Direction string ("ltr", "rtl", or "auto")
999 pub(crate) fn set_text_direction(&mut self, dir: String) {
1000 if !self.config.extract_document {
1001 return;
1002 }
1003 if self.dir.is_none() {
1004 self.dir = Some(dir);
1005 }
1006 }
1007
1008 pub(crate) fn wants_document(&self) -> bool {
1009 self.config.extract_document
1010 }
1011
1012 pub(crate) fn wants_headers(&self) -> bool {
1013 self.config.extract_headers
1014 }
1015
1016 pub(crate) fn wants_links(&self) -> bool {
1017 self.config.extract_links
1018 }
1019
1020 pub(crate) fn wants_images(&self) -> bool {
1021 self.config.extract_images
1022 }
1023
1024 pub(crate) fn wants_structured_data(&self) -> bool {
1025 self.config.extract_structured_data
1026 }
1027
1028 /// Extract document metadata from collected head metadata.
1029 ///
1030 /// Parses head metadata into structured document metadata,
1031 /// handling special cases like Open Graph, Twitter Card, keywords, etc.
1032 #[allow(dead_code)]
1033 fn extract_document_metadata(
1034 head_metadata: BTreeMap<String, String>,
1035 lang: Option<String>,
1036 dir: Option<String>,
1037 ) -> DocumentMetadata {
1038 let mut doc = DocumentMetadata::default();
1039
1040 for (raw_key, value) in head_metadata {
1041 let mut key = raw_key.as_str();
1042 let mut replaced_key: Option<String> = None;
1043
1044 if let Some(stripped) = key.strip_prefix("meta-") {
1045 key = stripped;
1046 }
1047
1048 if key.as_bytes().contains(&b':') {
1049 replaced_key = Some(key.replace(':', "-"));
1050 key = replaced_key.as_deref().unwrap_or(key);
1051 }
1052
1053 match key {
1054 "title" => doc.title = Some(value),
1055 "description" => doc.description = Some(value),
1056 "author" => doc.author = Some(value),
1057 "canonical" => doc.canonical_url = Some(value),
1058 "base" | "base-href" => doc.base_href = Some(value),
1059 key if key.starts_with("og-") => {
1060 let og_key = if key.as_bytes().contains(&b'-') {
1061 key.trim_start_matches("og-").replace('-', "_")
1062 } else {
1063 key.trim_start_matches("og-").to_string()
1064 };
1065 doc.open_graph.insert(og_key, value);
1066 }
1067 key if key.starts_with("twitter-") => {
1068 let tw_key = if key.as_bytes().contains(&b'-') {
1069 key.trim_start_matches("twitter-").replace('-', "_")
1070 } else {
1071 key.trim_start_matches("twitter-").to_string()
1072 };
1073 doc.twitter_card.insert(tw_key, value);
1074 }
1075 "keywords" => {
1076 doc.keywords = value
1077 .split(',')
1078 .map(|s| s.trim().to_string())
1079 .filter(|s| !s.is_empty())
1080 .collect();
1081 }
1082 _ => {
1083 let meta_key = if key.as_ptr() == raw_key.as_ptr() && key.len() == raw_key.len() {
1084 raw_key
1085 } else if let Some(replaced) = replaced_key {
1086 replaced
1087 } else {
1088 key.to_string()
1089 };
1090 doc.meta_tags.insert(meta_key, value);
1091 }
1092 }
1093 }
1094
1095 if let Some(lang) = lang {
1096 doc.language = Some(lang);
1097 }
1098
1099 if let Some(dir) = dir {
1100 if let Some(parsed_dir) = TextDirection::parse(&dir) {
1101 doc.text_direction = Some(parsed_dir);
1102 }
1103 }
1104
1105 doc
1106 }
1107
1108 /// Extract structured data blocks into StructuredData items.
1109 #[allow(dead_code)]
1110 fn extract_structured_data(json_ld: Vec<String>) -> Vec<StructuredData> {
1111 let mut result = Vec::with_capacity(json_ld.len());
1112
1113 for json_str in json_ld {
1114 let schema_type = Self::scan_schema_type(&json_str)
1115 .or_else(|| {
1116 if json_str.contains("\"@type\"") {
1117 serde_json::from_str::<serde_json::Value>(&json_str)
1118 .ok()
1119 .and_then(|v| v.get("@type").and_then(|t| t.as_str().map(|s| s.to_string())))
1120 } else {
1121 None
1122 }
1123 })
1124 .or_else(|| {
1125 if !json_str.contains("\"@graph\"") {
1126 return None;
1127 }
1128
1129 let value = serde_json::from_str::<serde_json::Value>(&json_str).ok()?;
1130 let graph = value.get("@graph")?;
1131 let items = graph.as_array()?;
1132 items
1133 .iter()
1134 .find_map(|item| item.get("@type").and_then(|t| t.as_str().map(|s| s.to_string())))
1135 });
1136
1137 result.push(StructuredData {
1138 data_type: StructuredDataType::JsonLd,
1139 raw_json: json_str,
1140 schema_type,
1141 });
1142 }
1143
1144 result
1145 }
1146
1147 fn scan_schema_type(json_str: &str) -> Option<String> {
1148 let needle = "\"@type\"";
1149 let start = json_str.find(needle)? + needle.len();
1150 let bytes = json_str.as_bytes();
1151 let mut i = start;
1152
1153 while i < bytes.len() && bytes[i].is_ascii_whitespace() {
1154 i += 1;
1155 }
1156 if i >= bytes.len() || bytes[i] != b':' {
1157 return None;
1158 }
1159 i += 1;
1160 while i < bytes.len() && bytes[i].is_ascii_whitespace() {
1161 i += 1;
1162 }
1163 if i >= bytes.len() {
1164 return None;
1165 }
1166
1167 if bytes[i] == b'[' {
1168 i += 1;
1169 while i < bytes.len() && bytes[i].is_ascii_whitespace() {
1170 i += 1;
1171 }
1172 if i >= bytes.len() || bytes[i] != b'"' {
1173 return None;
1174 }
1175 } else if bytes[i] != b'"' {
1176 return None;
1177 }
1178
1179 let start_quote = i;
1180 i += 1;
1181 let mut escaped = false;
1182 while i < bytes.len() {
1183 let byte = bytes[i];
1184 if escaped {
1185 escaped = false;
1186 i += 1;
1187 continue;
1188 }
1189 if byte == b'\\' {
1190 escaped = true;
1191 i += 1;
1192 continue;
1193 }
1194 if byte == b'"' {
1195 let end_quote = i;
1196 let slice = &json_str[start_quote..=end_quote];
1197 return serde_json::from_str::<String>(slice).ok();
1198 }
1199 i += 1;
1200 }
1201
1202 None
1203 }
1204
1205 /// Finish collection and return all extracted metadata.
1206 ///
1207 /// Performs final processing, validation, and consolidation of all
1208 /// collected data into the [`ExtendedMetadata`] output structure.
1209 ///
1210 /// # Returns
1211 ///
1212 /// Complete [`ExtendedMetadata`] with all extracted information.
1213 #[allow(dead_code)]
1214 pub(crate) fn finish(self) -> ExtendedMetadata {
1215 let structured_data = Self::extract_structured_data(self.json_ld);
1216 let document = Self::extract_document_metadata(self.head_metadata, self.lang, self.dir);
1217
1218 ExtendedMetadata {
1219 document,
1220 headers: self.headers,
1221 links: self.links,
1222 images: self.images,
1223 structured_data,
1224 }
1225 }
1226
1227 /// Categorize links by type for analysis and filtering.
1228 ///
1229 /// Separates collected links into groups by [`LinkType`].
1230 /// This is an analysis helper method; actual categorization happens during add_link.
1231 ///
1232 /// # Returns
1233 ///
1234 /// BTreeMap with LinkType as key and Vec of matching LinkMetadata as value.
1235 #[allow(dead_code)]
1236 pub(crate) fn categorize_links(&self) -> BTreeMap<String, Vec<&LinkMetadata>> {
1237 let mut categorized: BTreeMap<String, Vec<&LinkMetadata>> = BTreeMap::new();
1238
1239 for link in &self.links {
1240 let category = link.link_type.to_string();
1241 categorized.entry(category).or_default().push(link);
1242 }
1243
1244 categorized
1245 }
1246
1247 /// Count headers by level for structural analysis.
1248 ///
1249 /// Returns count of headers at each level (1-6).
1250 ///
1251 /// # Returns
1252 ///
1253 /// BTreeMap with level as string key and count as value.
1254 #[allow(dead_code)]
1255 pub(crate) fn header_counts(&self) -> BTreeMap<String, usize> {
1256 let mut counts: BTreeMap<String, usize> = BTreeMap::new();
1257
1258 for header in &self.headers {
1259 *counts.entry(header.level.to_string()).or_insert(0) += 1;
1260 }
1261
1262 counts
1263 }
1264}
1265
1266/// Handle to a metadata collector via reference-counted mutable cell.
1267///
1268/// Used internally for sharing collector state across the tree traversal.
1269/// Matches the pattern used for [`InlineImageCollector`](crate::inline_images::InlineImageCollector).
1270///
1271/// # Examples
1272///
1273/// ```ignore
1274/// let collector = MetadataCollector::new(MetadataConfig::default());
1275/// let handle = Rc::new(RefCell::new(collector));
1276///
1277/// // In tree walk, can be passed and borrowed
1278/// handle.borrow_mut().add_header(1, "Title".to_string(), None, 0, 100);
1279///
1280/// let metadata = handle.take().finish();
1281/// ```
1282#[allow(dead_code)]
1283pub(crate) type MetadataCollectorHandle = Rc<RefCell<MetadataCollector>>;
1284
1285#[cfg(test)]
1286mod tests {
1287 use super::*;
1288
1289 #[test]
1290 fn test_text_direction_parse() {
1291 assert_eq!(TextDirection::parse("ltr"), Some(TextDirection::LeftToRight));
1292 assert_eq!(TextDirection::parse("rtl"), Some(TextDirection::RightToLeft));
1293 assert_eq!(TextDirection::parse("auto"), Some(TextDirection::Auto));
1294 assert_eq!(TextDirection::parse("invalid"), None);
1295 assert_eq!(TextDirection::parse("LTR"), Some(TextDirection::LeftToRight));
1296 }
1297
1298 #[test]
1299 fn test_text_direction_display() {
1300 assert_eq!(TextDirection::LeftToRight.to_string(), "ltr");
1301 assert_eq!(TextDirection::RightToLeft.to_string(), "rtl");
1302 assert_eq!(TextDirection::Auto.to_string(), "auto");
1303 }
1304
1305 #[test]
1306 fn test_link_classification() {
1307 assert_eq!(LinkMetadata::classify_link("#section"), LinkType::Anchor);
1308 assert_eq!(LinkMetadata::classify_link("mailto:test@example.com"), LinkType::Email);
1309 assert_eq!(LinkMetadata::classify_link("tel:+1234567890"), LinkType::Phone);
1310 assert_eq!(LinkMetadata::classify_link("https://example.com"), LinkType::External);
1311 assert_eq!(LinkMetadata::classify_link("http://example.com"), LinkType::External);
1312 assert_eq!(LinkMetadata::classify_link("/path/to/page"), LinkType::Internal);
1313 assert_eq!(LinkMetadata::classify_link("../relative"), LinkType::Internal);
1314 assert_eq!(LinkMetadata::classify_link("./same"), LinkType::Internal);
1315 }
1316
1317 #[test]
1318 fn test_header_validation() {
1319 let valid = HeaderMetadata {
1320 level: 3,
1321 text: "Title".to_string(),
1322 id: None,
1323 depth: 2,
1324 html_offset: 100,
1325 };
1326 assert!(valid.is_valid());
1327
1328 let invalid_high = HeaderMetadata {
1329 level: 7,
1330 text: "Title".to_string(),
1331 id: None,
1332 depth: 2,
1333 html_offset: 100,
1334 };
1335 assert!(!invalid_high.is_valid());
1336
1337 let invalid_low = HeaderMetadata {
1338 level: 0,
1339 text: "Title".to_string(),
1340 id: None,
1341 depth: 2,
1342 html_offset: 100,
1343 };
1344 assert!(!invalid_low.is_valid());
1345 }
1346
1347 #[test]
1348 fn test_metadata_collector_new() {
1349 let config = MetadataConfig::default();
1350 let collector = MetadataCollector::new(config);
1351
1352 assert_eq!(collector.headers.capacity(), 32);
1353 assert_eq!(collector.links.capacity(), 64);
1354 assert_eq!(collector.images.capacity(), 16);
1355 assert_eq!(collector.json_ld.capacity(), 4);
1356 }
1357
1358 #[test]
1359 fn test_metadata_collector_add_header() {
1360 let config = MetadataConfig::default();
1361 let mut collector = MetadataCollector::new(config);
1362
1363 collector.add_header(1, "Title".to_string(), Some("title".to_string()), 0, 100);
1364 assert_eq!(collector.headers.len(), 1);
1365
1366 let header = &collector.headers[0];
1367 assert_eq!(header.level, 1);
1368 assert_eq!(header.text, "Title");
1369 assert_eq!(header.id, Some("title".to_string()));
1370
1371 collector.add_header(7, "Invalid".to_string(), None, 0, 200);
1372 assert_eq!(collector.headers.len(), 1);
1373 }
1374
1375 #[test]
1376 fn test_metadata_collector_add_link() {
1377 let config = MetadataConfig::default();
1378 let mut collector = MetadataCollector::new(config);
1379
1380 collector.add_link(
1381 "https://example.com".to_string(),
1382 "Example".to_string(),
1383 Some("Visit".to_string()),
1384 Some("nofollow external".to_string()),
1385 BTreeMap::from([("data-id".to_string(), "example".to_string())]),
1386 );
1387
1388 assert_eq!(collector.links.len(), 1);
1389
1390 let link = &collector.links[0];
1391 assert_eq!(link.href, "https://example.com");
1392 assert_eq!(link.text, "Example");
1393 assert_eq!(link.link_type, LinkType::External);
1394 assert_eq!(link.rel, vec!["nofollow", "external"]);
1395 assert_eq!(link.attributes.get("data-id"), Some(&"example".to_string()));
1396 }
1397
1398 #[test]
1399 fn test_metadata_collector_respects_config() {
1400 let config = MetadataConfig {
1401 extract_document: false,
1402 extract_headers: false,
1403 extract_links: false,
1404 extract_images: false,
1405 extract_structured_data: false,
1406 max_structured_data_size: DEFAULT_MAX_STRUCTURED_DATA_SIZE,
1407 };
1408 let mut collector = MetadataCollector::new(config);
1409
1410 collector.add_header(1, "Title".to_string(), None, 0, 100);
1411 collector.add_link(
1412 "https://example.com".to_string(),
1413 "Link".to_string(),
1414 None,
1415 None,
1416 BTreeMap::new(),
1417 );
1418 collector.add_image(
1419 "https://example.com/img.jpg".to_string(),
1420 None,
1421 None,
1422 None,
1423 BTreeMap::new(),
1424 );
1425 collector.add_json_ld("{}".to_string());
1426
1427 assert!(collector.headers.is_empty());
1428 assert!(collector.links.is_empty());
1429 assert!(collector.images.is_empty());
1430 assert!(collector.json_ld.is_empty());
1431 }
1432
1433 #[test]
1434 fn test_metadata_collector_finish() {
1435 let config = MetadataConfig::default();
1436 let mut collector = MetadataCollector::new(config);
1437
1438 collector.set_language("en".to_string());
1439 collector.add_header(1, "Main Title".to_string(), None, 0, 100);
1440 collector.add_link(
1441 "https://example.com".to_string(),
1442 "Example".to_string(),
1443 None,
1444 None,
1445 BTreeMap::new(),
1446 );
1447
1448 let metadata = collector.finish();
1449
1450 assert_eq!(metadata.document.language, Some("en".to_string()));
1451 assert_eq!(metadata.headers.len(), 1);
1452 assert_eq!(metadata.links.len(), 1);
1453 }
1454
1455 #[test]
1456 fn test_document_metadata_default() {
1457 let doc = DocumentMetadata::default();
1458
1459 assert!(doc.title.is_none());
1460 assert!(doc.description.is_none());
1461 assert!(doc.keywords.is_empty());
1462 assert!(doc.open_graph.is_empty());
1463 assert!(doc.twitter_card.is_empty());
1464 assert!(doc.meta_tags.is_empty());
1465 }
1466
1467 #[test]
1468 fn test_metadata_config_default() {
1469 let config = MetadataConfig::default();
1470
1471 assert!(config.extract_headers);
1472 assert!(config.extract_links);
1473 assert!(config.extract_images);
1474 assert!(config.extract_structured_data);
1475 assert_eq!(config.max_structured_data_size, DEFAULT_MAX_STRUCTURED_DATA_SIZE);
1476 }
1477
1478 #[test]
1479 fn test_image_type_classification() {
1480 let data_uri = ImageMetadata {
1481 src: "...".to_string(),
1482 alt: None,
1483 title: None,
1484 dimensions: None,
1485 image_type: ImageType::DataUri,
1486 attributes: BTreeMap::new(),
1487 };
1488 assert_eq!(data_uri.image_type, ImageType::DataUri);
1489
1490 let external = ImageMetadata {
1491 src: "https://example.com/image.jpg".to_string(),
1492 alt: None,
1493 title: None,
1494 dimensions: None,
1495 image_type: ImageType::External,
1496 attributes: BTreeMap::new(),
1497 };
1498 assert_eq!(external.image_type, ImageType::External);
1499 }
1500
1501 #[test]
1502 fn test_link_type_display() {
1503 assert_eq!(LinkType::Anchor.to_string(), "anchor");
1504 assert_eq!(LinkType::Internal.to_string(), "internal");
1505 assert_eq!(LinkType::External.to_string(), "external");
1506 assert_eq!(LinkType::Email.to_string(), "email");
1507 assert_eq!(LinkType::Phone.to_string(), "phone");
1508 assert_eq!(LinkType::Other.to_string(), "other");
1509 }
1510
1511 #[test]
1512 fn test_structured_data_type_display() {
1513 assert_eq!(StructuredDataType::JsonLd.to_string(), "json_ld");
1514 assert_eq!(StructuredDataType::Microdata.to_string(), "microdata");
1515 assert_eq!(StructuredDataType::RDFa.to_string(), "rdfa");
1516 }
1517
1518 #[test]
1519 fn test_categorize_links() {
1520 let config = MetadataConfig::default();
1521 let mut collector = MetadataCollector::new(config);
1522
1523 collector.add_link("#anchor".to_string(), "Anchor".to_string(), None, None, BTreeMap::new());
1524 collector.add_link(
1525 "https://example.com".to_string(),
1526 "External".to_string(),
1527 None,
1528 None,
1529 BTreeMap::new(),
1530 );
1531 collector.add_link(
1532 "mailto:test@example.com".to_string(),
1533 "Email".to_string(),
1534 None,
1535 None,
1536 BTreeMap::new(),
1537 );
1538
1539 let categorized = collector.categorize_links();
1540
1541 assert_eq!(categorized.get("anchor").map(|v| v.len()), Some(1));
1542 assert_eq!(categorized.get("external").map(|v| v.len()), Some(1));
1543 assert_eq!(categorized.get("email").map(|v| v.len()), Some(1));
1544 }
1545
1546 #[test]
1547 fn test_header_counts() {
1548 let config = MetadataConfig::default();
1549 let mut collector = MetadataCollector::new(config);
1550
1551 collector.add_header(1, "H1".to_string(), None, 0, 100);
1552 collector.add_header(2, "H2".to_string(), None, 1, 200);
1553 collector.add_header(2, "H2b".to_string(), None, 1, 300);
1554 collector.add_header(3, "H3".to_string(), None, 2, 400);
1555
1556 let counts = collector.header_counts();
1557
1558 assert_eq!(counts.get("1").copied(), Some(1));
1559 assert_eq!(counts.get("2").copied(), Some(2));
1560 assert_eq!(counts.get("3").copied(), Some(1));
1561 }
1562}