Skip to main content

Extract structured, LLM-ready content from any website — adaptive engine, no LLMs required

Project description

LLMParser

Turn any website into clean, structured content that language models can actually read.

LLMParser extracts articles, documentation, and blog posts from the web and delivers them as clean Markdown, typed content blocks, and normalised metadata — ready to drop into LLM prompts, RAG pipelines, or knowledge bases. No LLMs needed to run it. No API keys. No databases.

PyPI version Python License: MIT


Why LLMParser?

Raw HTML is 80% noise. Navigation bars, cookie banners, sidebars, ad scripts, and boilerplate drown out the 20% of content that actually matters. Feeding raw HTML to an LLM wastes tokens, degrades answer quality, and blows context windows.

LLMParser solves this:

Problem LLMParser Solution
Raw HTML is noisy Removes nav, footer, ads, cookie overlays, sidebars
JS SPAs return empty HTML 4-phase Playwright rendering with accordion expansion
Metadata scattered across 4 formats Unified: JSON-LD → OG → Twitter Card → <meta>
Dates in 30+ formats Normalised to ISO 8601 via dateparser
Sites change structure Zero site-specific selectors — fully generic extraction
Need LLM-ready chunks Typed blocks: heading / paragraph / code / list / quote / table

What Makes It Different

Adaptive Engine — Reads Any Site, Dynamically

Most scrapers break when a site changes its template or uses a JS framework. LLMParser classifies each page before fetching it and picks the right strategy automatically:

Static HTML  → urllib (fast, no browser)
JS SPA       → Playwright (4-phase rendering, accordion expansion)
AMP version  → fetch lightweight AMP URL directly
Cookie wall  → Playwright clicks through GDPR overlays
Unknown      → mobile User-Agent retry → graceful fallback

The classification result (page type, confidence, detected frameworks, AMP URL) is embedded in the output so you always know how the content was fetched.

Playwright 4-Phase Rendering

For JavaScript-heavy pages, LLMParser doesn't just "wait for the page to load" — it:

  1. Load — HTML and synchronous scripts ready
  2. Network idle — React / Vue / Angular XHR bootstrap completes (up to 12 s)
  3. DOM hydration — waits until body.innerText exceeds 50 words (catches WebSocket/long-poll SPAs)
  4. Accordion expansion — auto-clicks all collapsed [aria-expanded="false"] panels, <details>, Angular Material expansion panels, and Bootstrap collapsibles — then waits for the expanded content to settle

No other open-source scraper does step 4. This is critical for documentation sites, FAQs, and service portals where key content is hidden behind click-to-expand UI.

Best-of-Two Content Extraction

Two extractors run independently on every page:

  • readability-lxml — great for blog posts with a clear main content div
  • trafilatura — better for multi-section service pages

LLMParser picks the winner by word count: trafilatura wins only if it returns ≥ 40% more words (avoiding noisy ties). If both fail, a paragraph-density DOM heuristic finds the richest element and falls back to the full stripped <body> — so you always get something.

Zero Site-Specific Code

There are no custom selectors for Medium, Substack, Dev.to, or any other platform. Every rule is generic HTML semantics — which means LLMParser works on sites that don't exist yet.


Features

  • Adaptive engine — classifies each page (Static HTML, JS SPA, Cookie-walled, Paywalled) and selects the best fetch strategy automatically
  • Generic extraction — readability-lxml + trafilatura best-of-two → DOM heuristic cascade; zero site-specific selectors
  • Playwright 4-phase rendering — load → networkidle → DOM hydration → accordion expansion
  • RSS/Atom feed supportfetch_feed() parses any feed and extracts each linked article
  • Pagination auto-follow — detects <link rel="next"> and traverses paginated archives
  • Incremental resume — skip previously-seen URLs; cross-crawl dedup via index.json
  • Concurrent batch APIfetch_batch() fetches multiple URLs in parallel
  • Polite crawling — robots.txt, auto-throttle, Retry-After header support
  • Structured output — JSON + Markdown per page; index.json + index.csv summaries
  • No LLMs — fully deterministic extraction; no API keys, no embeddings

Quickstart

1. Install

pip install llmparser

# Only needed for JS-rendered sites
playwright install chromium

2. Single URL (Python API)

from llmparser import fetch

article = fetch("https://example.com/blog/post")

# Ready for LLM context
print(article.content_markdown)

# Typed blocks (heading/paragraph/code/list/quote/table/image)
for block in article.content_blocks:
    print(block["type"], block.get("text", ""))

# Rich metadata
print(article.title, article.author, article.published_at)

3. Full site crawl (CLI)

python -m llmparser --url https://example.com/blog --out ./out

4. View results

out/
├── articles/
│   ├── blog-my-first-post.json    # Full structured data
│   ├── blog-my-first-post.md      # Clean Markdown (LLM-ready)
│   └── ...
├── index.json                     # Summary of all pages (sorted by date)
├── index.csv                      # Same summary as CSV
├── skipped.jsonl                  # Skipped URLs with reasons
└── summary.txt                    # Plain-text crawl report

Python API

Single URL

from llmparser import fetch

article = fetch("https://example.com/blog/post")
print(article.title)
print(article.author)
print(article.word_count)
print(article.content_markdown)   # clean Markdown

data = article.model_dump()       # plain dict, JSON-serialisable

Force Playwright

article = fetch("https://angular-app.io/blog/post", render_js=True)

Batch (concurrent)

from llmparser import fetch_batch

articles = fetch_batch([
    "https://example.com/post/1",
    "https://example.com/post/2",
], max_workers=4, on_error="skip")

for article in articles:
    print(article.title, article.word_count)

RSS/Atom feed

from llmparser import fetch_feed

articles = fetch_feed("https://example.com/feed.xml", max_articles=50)
for article in articles:
    print(article.title, article.published_at)

Low-level

from llmparser import fetch_html, extract

html = fetch_html("https://example.com/blog/post")
article = extract(html, url="https://example.com/blog/post")

CLI Reference

python -m llmparser --url URL [options]

Required:
  --url URL              Site URL to start from

Output:
  --out DIR              Output directory (default: ./out)

Crawl limits:
  --max-pages N          Maximum pages to scrape (default: 500)
  --max-depth N          Maximum BFS crawl depth (default: 10)
  --concurrency N        Concurrent requests (default: 8)

Rendering:
  --render-js {auto,always,never}
                         JavaScript rendering mode (default: auto)

Filtering:
  --ignore-robots        Ignore robots.txt
  --include-regex PAT    Only extract from URLs matching PAT
  --exclude-regex PAT    Skip URLs matching PAT

Domain scope:
  --allow-subdomains     Crawl subdomains of the start URL domain
  --extra-domains DOMS   Comma-separated extra domains to crawl

Resume / caching:
  --resume               Skip URLs already in seen_urls.txt and index.json
  --cache                Enable HTTP response cache for faster re-runs

Display:
  --log-level {DEBUG,INFO,WARNING,ERROR}
  --progress             Show live Rich progress bar

Examples

# Parse a documentation site for LLM ingestion
python -m llmparser --url https://docs.example.com/ --max-pages 200

# JS SPA (React, Angular, Vue) — auto-detected, Playwright used automatically
python -m llmparser --url https://modernsite.io/blog/ --render-js auto

# Resume a previous crawl (skips already-parsed pages)
python -m llmparser --url https://example.com/ --resume

# Parse only blog posts, skip everything else
python -m llmparser --url https://example.com/ --include-regex '/blog/'

# Multi-domain (main site + docs subdomain)
python -m llmparser --url https://example.com/ \
  --allow-subdomains \
  --extra-domains "docs.example.com"

# Fast development iteration with caching
python -m llmparser --url https://example.com/ \
  --log-level DEBUG --max-pages 10 --cache

How It Works

Adaptive Engine

Before fetching, each page is classified:

Page Type Detection Strategy
Static HTML Body words ≥ 150, no JS framework urllib (fast)
JS SPA Framework root (#__next, #app, etc.) + sparse text; or thin body + external scripts Playwright
AMP available <link rel="amphtml"> Fetch lightweight AMP URL
Cookie-walled GDPR overlay in visible text Playwright
Unknown Low confidence Mobile UA retry → static

Classification signals are stored in article.raw_metadata["_classification"].

Playwright 4-Phase Rendering

For JS-heavy pages:

  1. Load — HTML + synchronous scripts ready
  2. Network idle — React/Vue/Angular XHR bootstrap (up to 12 s)
  3. DOM hydration — wait for >50 words in body.innerText
  4. Accordion expansion — click all [aria-expanded="false"], open <details>, expand Angular Material panels and Bootstrap collapsibles

Content Extraction

  1. readability-lxml + trafilatura run independently
  2. If trafilatura returns ≥ 1.4× more words → it wins (multi-section pages)
  3. DOM heuristic fallback — returns full stripped <body> when no single element dominates

Cookie banners and GDPR overlays are removed before any extractor runs.

Output for LLMs

Each parsed page produces:

  • content_markdown — clean Markdown, ready to paste into an LLM prompt or RAG chunk
  • content_blocks — typed structured blocks (heading / paragraph / code / list / quote / table / image)
  • content_text — plain text for embedding or keyword search
  • Full metadata (title, author, date, tags, language, canonical URL)

Output Schema

out/articles/<slug>.json

{
  "url": "https://example.com/blog/post",
  "canonical_url": "https://example.com/blog/post",
  "title": "Page Title",
  "author": "Jane Smith",
  "published_at": "2024-01-15T10:00:00+00:00",
  "language": "en",
  "tags": ["python", "llm"],
  "summary": "Brief description.",
  "content_markdown": "# Page Title\n\n...",
  "content_blocks": [
    {"type": "heading", "level": 1, "text": "Page Title"},
    {"type": "paragraph", "text": "First paragraph..."},
    {"type": "code", "language": "python", "text": "from llmparser import fetch"}
  ],
  "word_count": 842,
  "reading_time_minutes": 5,
  "extraction_method_used": "readability",
  "fetch_strategy": "static",
  "page_type": "static_html",
  "article_score": 72,
  "scraped_at": "2024-02-24T12:00:00+00:00"
}

See docs/schema.md for the full schema.


Running Tests

pytest tests/ -v
# 121 tests, ~9 s

Project Structure

llmparser/                  # Core package
├── __init__.py             # Public API: fetch, fetch_batch, fetch_feed, fetch_html, extract
├── __main__.py             # CLI (python -m llmparser)
├── items.py                # Pydantic ArticleSchema
├── settings.py             # Scrapy settings
├── middlewares.py          # Rotating UA + Playwright logging
├── pipelines.py            # ContentHashDedup → Validation → ArticleWriter → IndexWriter (+CSV)
└── extractors/
    ├── adaptive.py         # Page-type classifier + strategy selector
    ├── feed.py             # RSS 2.0 / Atom 1.0 parser
    ├── metadata.py         # JSON-LD, OG, Twitter Card, <meta>
    ├── main_content.py     # readability + trafilatura + DOM heuristic
    ├── blocks.py           # HTML → typed content blocks
    ├── markdown.py         # HTML → Markdown
    ├── urlnorm.py          # URL normalization + slug generation
    └── heuristics.py       # Article scoring + JS detection

spiders/
└── blog_spider.py          # Generic domain spider (sitemap + BFS + pagination)

docs/
├── architecture.md
├── schema.md
└── heuristics.md

tests/
├── fixtures/
├── test_adaptive.py        # 22 tests
├── test_extractors.py      # Metadata, scoring, extraction
├── test_query.py           # fetch / fetch_batch / extract API
└── test_urlnorm.py         # URL normalization

Limitations

  • Login-gated content — cannot parse content behind authentication
  • Hard infinite scroll — captures initial render only; true infinite scroll needs custom playwright_page_methods
  • CAPTCHAs — auto-throttle helps but does not bypass hard CAPTCHA gates
  • Single domain by default — use --allow-subdomains / --extra-domains for multi-domain crawls

License

MIT — see LICENSE.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

llmparser-0.1.0.tar.gz (68.4 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

llmparser-0.1.0-py3-none-any.whl (64.6 kB view details)

Uploaded Python 3

File details

Details for the file llmparser-0.1.0.tar.gz.

File metadata

  • Download URL: llmparser-0.1.0.tar.gz
  • Upload date:
  • Size: 68.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.13.5

File hashes

Hashes for llmparser-0.1.0.tar.gz
Algorithm Hash digest
SHA256 d5ce9b0796b1390fa0627a6c3b77c1aeda2618bd8704fc7779a9786a2efa1ed7
MD5 c1cea7d4107662b754a745e1817fa5bd
BLAKE2b-256 4b464cd8e16622fddde7d6165109eaa22a6ac70d5ef3c74e5d53fc24361cb6a6

See more details on using hashes here.

File details

Details for the file llmparser-0.1.0-py3-none-any.whl.

File metadata

  • Download URL: llmparser-0.1.0-py3-none-any.whl
  • Upload date:
  • Size: 64.6 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.13.5

File hashes

Hashes for llmparser-0.1.0-py3-none-any.whl
Algorithm Hash digest
SHA256 0cf5d753e30e5b34ef90225daa2b9e714c8dafe38ae42b2e27d27557c040703b
MD5 ecf7b95556895cea57612d80e18d3698
BLAKE2b-256 7237856fdf4d6695c6b93280bf9b96361e694369dca770ba8ba40e584cf1cc6b

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page