Skip to content

Sriram-PR/doc-scraper

Repository files navigation

LLM Documentation Scraper (doc-scraper)

Go Version Go Reference Go Report Card License

A configurable, concurrent, and resumable web crawler written in Go. Specifically designed to scrape technical documentation websites, extract core content, convert it cleanly to Markdown format suitable for ingestion by Large Language Models (LLMs), and save the results locally.

Overview

This project provides a powerful command-line tool to crawl documentation sites based on settings defined in a config.yaml file. It navigates the site structure, extracts content from specified HTML sections using CSS selectors, and converts it into clean Markdown files.

Why Use This Tool?

  • Built for LLM Training & RAG Systems - Creates clean, consistent Markdown optimized for ingestion
  • Preserves Documentation Structure - Maintains the original site hierarchy for context preservation
  • Production-Ready Features - Offers resumable crawls, rate limiting, and graceful error handling
  • High Performance - Uses Go's concurrency model for efficient parallel processing

Goal: Preparing Documentation for LLMs

The main objective of this tool is to automate the often tedious process of gathering and cleaning web-based documentation for use with Large Language Models. By converting structured web content into clean Markdown, it aims to provide a dataset that is:

  • Text-Focused: Prioritizes the textual content extracted via CSS selectors
  • Structured: Maintains the directory hierarchy of the original documentation site, preserving context
  • Cleaned: Converts HTML to Markdown, removing web-specific markup and clutter
  • Locally Accessible: Provides the content as local files for easier processing and pipeline integration

Key Features

Feature Description
Configurable Crawling Uses YAML for global and site-specific settings
Scope Control Limits crawling by domain, path prefix, and disallowed path patterns (regex)
Content Extraction Extracts main content using CSS selectors
HTML-to-Markdown Converts extracted HTML to clean Markdown
Image Handling Opt-in downloading and local rewriting of image links with domain and size filtering (disabled by default; doc-scraper is text-first)
Link Rewriting Rewrites internal links to relative paths for local structure
JSONL Output Optional one-record-per-page JSONL with a trailing crawl-summary record, for RAG ingestion
Concurrency Configurable worker pools and semaphore-based request limits (global and per-host)
Rate Limiting Configurable per-host delays with jitter
Robots.txt & Sitemaps Respects robots.txt and processes discovered sitemaps
State Persistence Uses BadgerDB for state; supports resuming crawls via crawl --resume
Graceful Shutdown Handles SIGINT/SIGTERM with proper cleanup
HTTP Retries Exponential backoff with jitter for transient errors
Observability Structured logging (logrus); optional pprof endpoint (build with -tags pprof)
Modular Code Organized into packages for clarity and maintainability
CLI Utilities Built-in config validate and config list commands for configuration management
MCP Server Mode Expose as Model Context Protocol server for Claude Code/Cursor integration
Auto Content Detection Automatic framework detection (Docusaurus, MkDocs, Sphinx, GitBook, ReadTheDocs) with readability fallback
Parallel Site Crawling Crawl multiple sites concurrently with shared resource management
Watch Mode Scheduled periodic re-crawling with state persistence

Getting Started

Prerequisites

  • Go: Version 1.25 or later
  • Git: For cloning the repository
  • Disk Space: Sufficient for storing crawled content and state database

Installation

Option 1: Direct Installation (Recommended)

Install the latest version directly from GitHub:

go install github.com/Sriram-PR/doc-scraper/cmd/doc-scraper@latest

This installs the doc-scraper binary to your GOPATH/bin directory (usually ~/go/bin or %USERPROFILE%\go\bin). Make sure this directory is in your PATH.

Option 2: Clone and Build

  1. Clone the repository:

    git clone https://github.com/Sriram-PR/doc-scraper.git
    cd doc-scraper
  2. Install Dependencies:

    go mod tidy
  3. Build the Binary:

    make build
    # or: go build -o doc-scraper ./cmd/doc-scraper

    This creates an executable named doc-scraper in the project root.

Quick Start

  1. Create a basic config.yaml file (see Configuration section)

  2. Run the crawler:

    ./doc-scraper crawl -site your_site_key -loglevel info
  3. Find your crawled documentation in the ./crawled_docs/ directory

Configuration (config.yaml)

A config.yaml file is required to run the crawler. Create this file in the project root or specify its path using the -config flag.

Key Settings for LLM Use

When configuring for LLM documentation processing, pay special attention to these settings:

  • sites.<your_site_key>.content_selector: Define precisely to capture only relevant text
  • sites.<your_site_key>.allowed_domain / allowed_path_prefix: Define scope accurately
  • skip_images: Images are not downloaded by default (text-first). Set to false globally or per-site to download and localize images for offline consumption
  • Adjust concurrency/delay settings based on the target site and your resources

Example Configuration

# Global settings (applied if not overridden by site)
default_delay_per_host: 500ms
num_workers: 8
num_image_workers: 8
max_requests: 48
max_requests_per_host: 4
output_base_dir: "./crawled_docs"
state_dir: "./crawler_state"
max_retries: 4
initial_retry_delay: 1s
max_retry_delay: 30s
global_crawl_timeout: 0s
skip_images: true # Default. Set to false to download and localize images
max_image_size_bytes: 10485760 # 10 MiB (applies only when images are downloaded)
enable_jsonl_output: true
jsonl_output_filename: "pages.jsonl"

# HTTP Client Settings
http_client_settings:
  timeout: 45s
  max_idle_conns_per_host: 6

# Site-specific configurations
sites:
  # Key used with -site flag
  pytorch_docs:
    start_urls:
      - "https://pytorch.org/docs/stable/"
    allowed_domain: "pytorch.org"
    allowed_path_prefix: "/docs/stable/"
    content_selector: "article.pytorch-article .body"
    max_depth: 0 # 0 for unlimited depth
    skip_images: false # Opt in to downloading images for this site
    disallowed_path_patterns:
      - "/docs/stable/.*/_modules/.*"
      - "/docs/stable/.*\.html#.*"

  tensorflow_docs:
    start_urls:
      - "https://www.tensorflow.org/guide"
      - "https://www.tensorflow.org/tutorials"
    allowed_domain: "www.tensorflow.org"
    allowed_path_prefix: "/"
    content_selector: ".devsite-article-body"
    max_depth: 0
    delay_per_host: 1s  # Site-specific override
    # Disable JSONL output for this site, overriding global
    enable_jsonl_output: false
    disallowed_path_patterns:
      - "/install/.*"
      - "/js/.*"

Full Configuration Options

Option Type Description Default
default_user_agent String Default User-Agent header for requests "" (Go default)
default_delay_per_host Duration Time to wait between requests to the same host 0s (no delay)
num_workers Integer Number of concurrent crawl workers 4
num_image_workers Integer Number of concurrent image download workers same as num_workers
max_requests Integer Maximum concurrent requests (global) 10
max_requests_per_host Integer Maximum concurrent requests per host 2
output_base_dir String Base directory for crawled content "./crawled_docs"
state_dir String Directory for BadgerDB state data "./crawler_state"
max_retries Integer Maximum retry attempts for HTTP requests 3
initial_retry_delay Duration Initial delay for retry backoff 1s
max_retry_delay Duration Maximum delay for retry backoff 30s
global_crawl_timeout Duration Overall timeout for the entire crawl 0s (no timeout)
per_page_timeout Duration Timeout for processing a single page 0s (no timeout)
skip_images Boolean Whether to skip downloading images. Image downloading is opt-in true (skip)
max_image_size_bytes Integer Maximum allowed image size (applies only when images are downloaded) 0 (unlimited)
max_page_size_bytes Integer Maximum HTML page body size 52428800 (50 MiB)
enable_jsonl_output Boolean Enable JSONL page output (one record per page plus a trailing crawl_meta record) for RAG pipelines false
jsonl_output_filename String Filename for JSONL output "pages.jsonl"
enable_incremental Boolean Enable incremental crawling globally false
http_client_settings Object HTTP client configuration (see below)
sites Map Site-specific configurations (required)

HTTP Client Settings: (Global; cannot be overridden per site. Pool, dialer, and TLS timings are baked into pkg/fetch with sane defaults and are not exposed as config knobs.)

  • timeout: Overall request timeout (default 45s)
  • max_idle_conns_per_host: Idle connections per host (default 2)
  • allow_private_networks: Disables the SSRF guard that blocks dials to loopback / private / link-local / CGNAT / multicast addresses. Default false. Set to true only if you intentionally crawl internal documentation servers reachable via private IPs.

Site-Specific Configuration Options:

  • start_urls: Array of starting URLs for crawling (Required)
  • allowed_domain: Restrict crawling to this domain (Required)
  • allowed_path_prefix: Further restrict crawling to URLs with this prefix (Required)
  • content_selector: CSS selector for main content extraction, or "auto" for automatic detection (Required)
  • max_depth: Maximum crawl depth from start URLs (0 = unlimited)
  • delay_per_host: Override global delay setting for this site
  • disallowed_path_patterns: Array of regex patterns for URLs to skip
  • link_extraction_selectors: Array of CSS selectors for additional link extraction areas
  • respect_nofollow: Boolean. Whether to respect rel="nofollow" links
  • user_agent: String. Override global user agent for this site
  • skip_images: Override the global image setting for this site. Images are skipped unless this (or the global skip_images) is set to false
  • max_image_size_bytes: Integer. Override global max image size for this site
  • allowed_image_domains: Array of domains from which to download images
  • disallowed_image_domains: Array of domains to block image downloads from
  • enable_jsonl_output: true or false. Override global JSONL output enablement for this site
  • jsonl_output_filename: String. Override global JSONL output filename for this site

Usage

Execute the compiled binary from the project root directory:

./doc-scraper <command> [options]

Commands

Command Description
crawl Start a crawl (add --resume to continue an interrupted one)
config validate Validate configuration file without crawling
config list List available site keys from config
mcp-server Start MCP server for AI tool integration
watch Watch sites and re-crawl on schedule
version Show version information

Command Options

crawl:

Flag Description Default
-config <path> Path to config file config.yaml
-site <key> Site key from config (single site) -
-sites <keys> Comma-separated site keys for parallel crawling -
--all-sites Crawl all configured sites in parallel false
--resume Resume an interrupted crawl from existing state false
-loglevel <level> Log level (debug, info, warn, error, fatal) info
-pprof <addr> pprof server address. Only effective in builds with -tags pprof; default builds log a warning and ignore the flag "" (disabled)
-incremental Enable incremental crawling (skip unchanged pages) false
-full Force full crawl (ignore incremental settings) false

Note: One of -site, -sites, or --all-sites is required.

config validate:

Flag Description Default
-config <path> Path to config file config.yaml
-site <key> Site key to validate (optional, validates all if empty) -
-json Emit a single JSON object instead of human-readable text false

config list:

Flag Description Default
-config <path> Path to config file config.yaml
-json Emit a single JSON object instead of human-readable text false

mcp-server: (stdio transport only; the SSE transport was removed in v2.x)

Flag Description Default
-config <path> Path to config file config.yaml
-loglevel <level> Log level (debug, info, warn, error) info

watch:

Flag Description Default
-config <path> Path to config file config.yaml
-site <key> Site key to watch (single site) -
-sites <keys> Comma-separated site keys to watch -
--all-sites Watch all configured sites false
-interval <duration> Crawl interval (e.g., 1h, 24h, 7d) 24h
-loglevel <level> Log level (debug, info, warn, error) info

Note: One of -site, -sites, or --all-sites is required.

Example Usage Scenarios

Basic Crawl:

./doc-scraper crawl -site tensorflow_docs -loglevel info

Resume a Large Crawl:

./doc-scraper crawl -site pytorch_docs --resume -loglevel info

Validate Configuration:

./doc-scraper config validate -config config.yaml
./doc-scraper config validate -site pytorch_docs  # Validate specific site

List Available Sites:

./doc-scraper config list

High Performance Crawl with Profiling:

./doc-scraper crawl -site small_docs -loglevel warn -pprof localhost:6060

Debug Mode for Troubleshooting:

./doc-scraper crawl -site test_site -loglevel debug

Parallel Crawl of Multiple Sites:

./doc-scraper crawl -sites pytorch_docs,tensorflow_docs,langchain_docs

Crawl All Configured Sites:

./doc-scraper crawl --all-sites

Start MCP Server for Claude Desktop:

./doc-scraper mcp-server -config config.yaml

Output Structure

Crawled content is saved under the output_base_dir defined in the config, organized by domain and preserving the site structure:

<output_base_dir>/
└── <sanitized_allowed_domain>/       # e.g., docs.example.com
    ├── images/                       # Only present if skip_images: false
    │   ├── image1.png
    │   └── image2.jpg
    ├── index.md                      # Markdown for the root path
    ├── images/                       # Only present if skip_images: false
    ├── <jsonl_output_filename>       # If enable_jsonl_output: true
    ├── llms.txt                      # Manifest of pages (auto-generated, when JSONL is enabled)
    ├── llms-full.txt                 # Full content concatenated (auto-generated, when JSONL is enabled)
    ├── topic_one/
    │   ├── index.md
    │   └── subtopic_a.md
    └── topic_two.md

llms.txt and llms-full.txt

When JSONL output is enabled, the crawler also emits llms.txt and llms-full.txt following the llmstxt.org convention. llms.txt is a markdown manifest (H1 + summary blockquote + ## Pages list of every crawled page with title and URL). llms-full.txt concatenates the full markdown content of every page, with section separators. Both files are regenerated on every crawl from the JSONL source of truth, so resumed crawls produce a complete updated manifest.

Output Format

Each generated Markdown file contains:

  • Original page title as level-1 heading
  • Clean content converted from HTML to Markdown
  • Relative links to other pages (when within the allowed domain)
  • Local image references (if images are enabled)
  • A footer with metadata including source URL and crawl timestamp

JSONL Output

When enabled, the crawler writes one JSON object per line to a JSONL file. This format is designed for ingestion into RAG pipelines and downstream indexers.

Enable it:

enable_jsonl_output: true
jsonl_output_filename: "pages.jsonl"  # default

The file mixes two record kinds, distinguished by the record_type field:

  • page records, one per crawled page.
  • A single crawl_meta record appended as the final line, holding the crawl-level summary. A resumed crawl appends a fresh crawl_meta record rather than rewriting the original, so a consumer should treat the last crawl_meta record in the file as authoritative.

page record fields (from PageJSONL):

Field Description
record_type Always "page"
url Final absolute URL of the page
title Page title
content Full markdown content
headings Array of headings extracted from the page
links Array of links found in the content
images Array of image URLs found in the content
content_hash SHA-256 hash of the content (used for incremental crawling)
crawled_at Timestamp of when the page was crawled
depth Crawl depth from the start URL

crawl_meta record fields (from CrawlMetaJSONL):

Field Description
record_type Always "crawl_meta"
site_key Site key from the config
allowed_domain The crawled domain
crawl_started_at Crawl start timestamp
crawl_ended_at Crawl end timestamp
total_pages Number of pages recorded in this crawl

The output file is written to each site's output directory. Both the enable flag and filename can be overridden per site.

Auto Content Detection

When you set content_selector: "auto" for a site, the crawler automatically detects the documentation framework and applies the appropriate content selector.

Supported Frameworks

Framework Detection Method Selectors (with fallbacks)
Docusaurus data-docusaurus attribute, __docusaurus marker article[class*='theme-doc'], .theme-doc-markdown, article.markdown, main article
MkDocs Material data-md-component attribute, .md-content class article.md-content__inner, .md-content article, .md-content
Sphinx searchindex.js, sphinxsidebar class div.document, div.body, article.bd-article, main.bd-main
ReadTheDocs readthedocs scripts, .rst-content class .rst-content, div[role='main'], .document
GitBook gitbook class patterns, markdown-section section.normal.markdown-section, .page-inner section, main[class*='gitbook']

Fallback Behavior

If no known framework is detected, the crawler uses Mozilla's Readability algorithm to automatically extract the main content from the page. This provides reliable content extraction for most documentation sites without manual configuration.

Example Usage

sites:
  pytorch_docs:
    start_urls:
      - "https://pytorch.org/docs/stable/"
    allowed_domain: "pytorch.org"
    allowed_path_prefix: "/docs/stable/"
    content_selector: "auto"  # Auto-detect framework
    max_depth: 0

Parallel Site Crawling

Crawl multiple documentation sites concurrently with shared resource management. The orchestrator coordinates multiple crawlers while respecting global rate limits and semaphores.

Usage

# Crawl specific sites in parallel
./doc-scraper crawl -sites pytorch_docs,tensorflow_docs,langchain_docs

# Crawl all configured sites
./doc-scraper crawl --all-sites

# Resume parallel crawl
./doc-scraper crawl -sites pytorch_docs,tensorflow_docs --resume

Resource Sharing

When running parallel crawls, the following resources are shared across all site crawlers:

  • Global semaphore: Limits total concurrent requests across all sites
  • HTTP client: Shared connection pooling
  • Rate limiter: Respects per-host delays

Each site still maintains its own:

  • BadgerDB store for state persistence
  • Output directory for crawled content
  • Per-host semaphores for domain-specific limiting

Results Summary

After all sites complete, the orchestrator outputs a summary:

===========================================
Parallel crawl completed in 2m30s
Site Results:
  pytorch_docs: SUCCESS - 1500 pages in 1m20s
  tensorflow_docs: SUCCESS - 2000 pages in 2m15s
  langchain_docs: FAILED - 0 pages in 10s
    Error: site 'langchain_docs' not found in configuration
-------------------------------------------
Total: 3 sites (2 success, 1 failed), 3500 pages processed
===========================================

Watch Mode

Watch mode enables scheduled periodic re-crawling of documentation sites. The scheduler tracks the last run time for each site and automatically triggers crawls when the configured interval has elapsed.

Usage

# Watch a single site with 24-hour interval
./doc-scraper watch -site pytorch_docs -interval 24h

# Watch multiple sites
./doc-scraper watch -sites pytorch_docs,tensorflow_docs -interval 12h

# Watch all configured sites weekly
./doc-scraper watch --all-sites -interval 7d

Interval Format

The interval supports standard Go duration format plus day units:

  • 30m - 30 minutes
  • 1h - 1 hour
  • 24h - 24 hours
  • 7d - 7 days
  • 1d12h - 1 day and 12 hours

State Persistence

Watch mode persists state to <state_dir>/watch_state.json, tracking:

  • Last run time for each site
  • Success/failure status
  • Pages processed
  • Error messages (if any)

This allows the scheduler to resume correctly after restarts, only running sites when their interval has elapsed.

Example Output

INFO Starting watch mode for 2 sites with interval 24h0m0s
INFO Watch schedule:
INFO   pytorch_docs: last run 2024-01-15T10:30:00Z (success, 1500 pages), next run 2024-01-16T10:30:00Z
INFO   tensorflow_docs: never run, will run immediately
INFO Running crawl for 1 due sites: [tensorflow_docs]
...
INFO Next crawl: pytorch_docs in 23h45m (at 10:30:00)

Graceful Shutdown

Watch mode handles SIGINT/SIGTERM gracefully, completing any in-progress crawls before exiting.

MCP Server Mode

The crawler can run as a Model Context Protocol (MCP) server, enabling integration with AI assistants like Claude Code and Cursor.

Available MCP Tools

Tool Description
describe_server Orientation manifest: server identity + sites + recent jobs in one call (call this first)
list_sites List all configured sites from config file
get_page Fetch a single URL and return content as markdown
crawl_site Start a background crawl for a site (returns job ID)
get_job_status Check the status of a background crawl job
cancel_crawl Cancel a running or pending crawl job by job ID
list_pages Enumerate crawled pages for a site (paginated, metadata only)

Usage

The MCP server uses the stdio transport, compatible with Claude Desktop, Claude Code, and Cursor.

./doc-scraper mcp-server -config config.yaml

Claude Code Integration

Add to your Claude Code configuration (claude_code_config.json):

{
  "mcpServers": {
    "doc-scraper": {
      "command": "/path/to/doc-scraper",
      "args": ["mcp-server", "-config", "/path/to/config.yaml"]
    }
  }
}

Tool Examples

List available sites:

Tool: list_sites
Result: Returns all configured sites with their domains and crawl status

Fetch a single page:

Tool: get_page
Arguments: { "url": "https://docs.example.com/guide", "content_selector": "article" }
Result: Returns page content as markdown with metadata

Start a background crawl:

Tool: crawl_site
Arguments: { "site_key": "pytorch_docs", "incremental": true }
Result: Returns job ID for tracking progress

Check crawl progress:

Tool: get_job_status
Arguments: { "job_id": "abc-123-def" }
Result: Returns status, pages processed, and completion info

Enumerate crawled pages:

Tool: list_pages
Arguments: { "site_key": "pytorch_docs", "max_results": 50, "offset": 0 }
Result: Returns up to 50 page entries (URL, title, depth, crawled_at, content_length), sorted by URL. Use offset for pagination.

Cancel a running crawl:

Tool: cancel_crawl
Arguments: { "job_id": "abc-123-def" }
Result: Returns cancelled: true/false and the job's current status. Has no effect on jobs already in a terminal state.

Contributing

Contributions are welcome! Please feel free to open an issue to discuss bugs, suggest features, or propose changes.

Pull Request Process:

  1. Fork the repository
  2. Create a feature branch (git checkout -b feature/amazing-feature)
  3. Commit your changes (git commit -m 'Add some amazing feature')
  4. Push to the branch (git push origin feature/amazing-feature)
  5. Open a Pull Request

Please ensure code adheres to Go best practices and includes appropriate documentation.

License

This project is licensed under the Apache-2.0 License.

Acknowledgements

About

Go web crawler to scrape documentation sites and convert content to clean Markdown for LLM ingestion (RAG, training data).

Topics

Resources

License

Stars

Watchers

Forks

Packages

 
 
 

Contributors