Skip to main content

CLI tool to generate full-page screenshots of static HTML files using Playwright

Project description

docs-html-screenshot

CI GHCR PyPI License: MIT

Screenshot your MkDocs site, feed it to GitHub Copilot, get instant visual feedback.

A CLI tool that captures full-page screenshots of static HTML files (MkDocs, Sphinx, Hugo output). Built for feeding documentation renders to LLMs with vision capabilities—so you can ask "what's wrong with this layout?" and get actionable answers.

What Is This?

The problem: You're writing MkDocs documentation, but you can't easily ask an LLM "does this page look right?" because LLMs need images, not HTML source.

The solution: This tool screenshots every page in your built site. Drop those PNGs into GitHub Copilot (or any vision-capable LLM) and ask:

  • "Are there any rendering issues on this page?"
  • "How could I improve this layout?"
  • "Does the navigation look correct?"

How it works:

  1. Build your static site (mkdocs build)
  2. Run this tool on the output directory
  3. Get PNG screenshots of every page
  4. Feed to LLM for visual analysis

Also useful for: Visual regression testing, design QA, documentation archival.

Who Is This For?

Developers who use MkDocs (or similar) and want to leverage LLM vision to review their documentation renders—without manually screenshotting each page.

Prerequisites

You need one of:

  • Docker (recommended) — zero setup, works anywhere
  • Python 3.10+ with uv or pip

What You DON'T Need

  • Playwright installation knowledge
  • Browser driver management
  • Complex configuration files

Quick Start (Docker — Recommended)

Pre-built images are available on GitHub Container Registry. No docker build required.

# Pull the image (multi-arch: amd64/arm64)
docker pull ghcr.io/pankaj28843/docs-html-screenshot:latest

# Generate screenshots from your static site
docker run --rm --init --ipc=host \
  -v "$PWD/site:/input:ro" \
  -v "$PWD/screenshots:/output" \
  ghcr.io/pankaj28843/docs-html-screenshot:latest \
  --input /input --output /output

Example output (flat filenames for easy drag-and-drop):

WROTE /output/index.html-screenshot.png
WROTE /output/getting-started__index.html-screenshot.png
WROTE /output/api__reference.html-screenshot.png

All screenshots are output as flat files using __ as the path separator—no nested directories. This makes it trivial to select all PNGs and drag them into an LLM chat.

Quick Start (Local Installation)

# Install with uv (recommended)
uv tool install docs-html-screenshot

# Or with pip
pip install docs-html-screenshot

# Install Playwright browsers (one-time)
playwright install chromium --with-deps

# Generate screenshots
docs-html-screenshot --input site --output screenshots

Usage Reference

Usage: docs-html-screenshot [OPTIONS]

Options:
  --input DIRECTORY               Input directory containing HTML files.
                                  [required]
  --output DIRECTORY              Output directory for screenshots.
                                  [required]
  --viewport-width INTEGER        Viewport width in pixels.  [default: 1920]
  --viewport-height INTEGER       Viewport height in pixels.  [default: 1020]
  --device-scale-factor INTEGER   Device scale factor for screenshots.
                                  [default: 2]
  --concurrency INTEGER           Number of concurrent pages.  [default: 10]
  --timeout-ms INTEGER            Navigation timeout in milliseconds.
                                  [default: 30000]
  --headed / --headless           Run browser headed (default headless).
                                  [default: headless]
  --fail-on-http / --allow-http-errors
                                  Fail build on HTTP status >= 400.  [default:
                                  fail-on-http]
  --fail-on-console / --allow-console-errors
                                  Fail build on console.error messages.
                                  [default: fail-on-console]
  --fail-on-pageerror / --allow-pageerror
                                  Fail build on page errors.  [default: fail-
                                  on-pageerror]
  --fail-on-weberror / --allow-weberror
                                  Fail build on web errors.  [default: fail-
                                  on-weberror]
  --help                          Show this message and exit.

Configuration Examples

Scenario Command
High-DPI (4K displays) --viewport-width 2560 --viewport-height 1440 --device-scale-factor 2
Fast CI (lower quality) --device-scale-factor 1 --concurrency 4
Permissive mode --allow-http-errors --allow-console-errors --allow-pageerror
Debug rendering --headed --concurrency 1 --timeout-ms 60000

Error Detection

By default, the tool fails (exit code 1) when any rendering error is detected. This is intentional—catch broken documentation early.

Flag What It Catches
--fail-on-http HTTP responses with status ≥ 400 (missing images, broken links)
--fail-on-console console.error() calls (JavaScript errors logged to console)
--fail-on-pageerror Uncaught JavaScript exceptions
--fail-on-weberror Network failures, CORS errors, failed resource loads

Screenshots are still generated even when errors are detected—this helps with debugging.

Exit Codes

Code Meaning
0 All pages rendered successfully
1 One or more pages had errors (screenshots still generated)

Docker Usage

Using Pre-built Images (Recommended)

Images are automatically built and pushed to GHCR on every push to main. Always use :latest for the most up-to-date version:

# Always use :latest (tracks main branch)
docker pull ghcr.io/pankaj28843/docs-html-screenshot:latest

Available tags:

Tag When Updated
:latest Every push to main branch
:v0.1.0 Semantic version releases
:sha-abc1234 Specific commit SHA

Volume Mounting

docker run --rm --init --ipc=host \
  -v "/path/to/html:/input:ro" \      # Read-only input
  -v "/path/to/output:/output" \       # Writable output
  ghcr.io/pankaj28843/docs-html-screenshot:latest \
  --input /input --output /output

Important Docker flags (per Playwright Docker docs):

  • --init — Proper process cleanup, avoids zombie processes
  • --ipc=host — Prevents Chromium OOM crashes
  • --rm — Clean up container after exit

GitHub Actions Integration

jobs:
  screenshots:
    runs-on: ubuntu-latest
    steps:
      - uses: actions/checkout@v4
      
      - name: Build documentation
        run: mkdocs build  # or your static site generator
      
      - name: Generate screenshots
        run: |
          docker run --rm --init --ipc=host \
            -v "${{ github.workspace }}/site:/input:ro" \
            -v "${{ github.workspace }}/screenshots:/output" \
            ghcr.io/pankaj28843/docs-html-screenshot:latest \
            --input /input --output /output
      
      - name: Upload screenshots
        uses: actions/upload-artifact@v4
        with:
          name: documentation-screenshots
          path: screenshots/

Building Custom Images

Only needed if you want to customize the base image:

docker build -t my-screenshot-tool .

docker run --rm --init --ipc=host \
  -v "$PWD/site:/input:ro" \
  -v "$PWD/screenshots:/output" \
  my-screenshot-tool --input /input --output /output

Design Decisions

Why 1920×1020 viewport?

  • 1920px width: Common desktop width, captures most responsive layouts without horizontal scroll
  • 1020px height: Slightly less than 1080p to account for browser chrome in visual comparisons
  • Full-page screenshots capture content beyond the viewport, so height mainly affects above-the-fold rendering

Why device scale factor 2?

  • Produces retina-quality screenshots (3840×2040 effective resolution)
  • Text remains crisp when zooming into screenshots
  • Modern displays are increasingly high-DPI; scale factor 1 looks dated

Why fail by default on errors?

  • Catch broken documentation early in CI pipelines
  • Missing images, broken links, JS errors are real bugs—don't silently ignore them
  • Screenshots are still generated for debugging, just exit code ≠ 0

Why concurrency = CPU count?

  • Balanced resource usage: Each Chromium page consumes memory
  • 10 concurrent pages is reasonable for most machines
  • Reduce with --concurrency 2-4 if running in memory-constrained containers

Why embedded HTTP server?

  • File protocol (file://) has CORS restrictions that break many sites
  • Embedded server serves files on localhost, matching production behavior
  • Zero configuration—server starts/stops automatically

Why flat output filenames?

  • Easy drag-and-drop: Select all PNGs at once without navigating nested folders
  • LLM-friendly: Drop entire folder into Claude, ChatGPT, or Copilot Vision
  • Path preserved: Original path is encoded in filename (docs/api/index.htmldocs__api__index.html-screenshot.png)

Troubleshooting

Blurry screenshots

Cause: Device scale factor too low.

--device-scale-factor 2  # or higher

Out of memory (OOM) in Docker

Cause: Too many concurrent pages.

--concurrency 2  # Reduce parallelism

Or increase Docker memory limit:

docker run --memory=4g ...

Timeout errors

Cause: Pages take too long to load.

--timeout-ms 60000  # Increase to 60 seconds

Permission denied on output directory

Cause: Docker volume mount permissions.

# Ensure output directory exists and is writable
mkdir -p screenshots
chmod 777 screenshots  # Or use proper user mapping

Blank/white screenshots

Cause: Page requires JavaScript execution time.

The tool waits for load event, but some SPAs need more time. Consider:

  • Ensuring your static site is truly static (pre-rendered HTML)
  • Increasing timeout with --timeout-ms

"Executable doesn't exist" error (local install)

Cause: Playwright browsers not installed.

playwright install chromium --with-deps

Development

# Clone and install
git clone https://github.com/pankaj28843/docs-html-screenshot.git
cd docs-html-screenshot
uv sync --extra dev

# Install Playwright browsers
uv run playwright install chromium --with-deps

# Run tests
uv run pytest tests/ -v

# Format and lint
uv run ruff format . && uv run ruff check --fix .

# Test CLI
uv run docs-html-screenshot --help

License

MIT

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

docs_html_screenshot-0.2.0.tar.gz (38.3 kB view details)

Uploaded Source

Built Distribution

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

docs_html_screenshot-0.2.0-py3-none-any.whl (8.9 kB view details)

Uploaded Python 3

File details

Details for the file docs_html_screenshot-0.2.0.tar.gz.

File metadata

  • Download URL: docs_html_screenshot-0.2.0.tar.gz
  • Upload date:
  • Size: 38.3 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for docs_html_screenshot-0.2.0.tar.gz
Algorithm Hash digest
SHA256 5b1ba961d0ad302a8922334b25d81295eb721b75b7a39fb612ccca7ecf378798
MD5 75375674379053182f7db02999aca635
BLAKE2b-256 74145bc923fce86f770bbf25a9eb1d66015ef29ef36bb2e859dcb5991ba82d53

See more details on using hashes here.

Provenance

The following attestation bundles were made for docs_html_screenshot-0.2.0.tar.gz:

Publisher: ci.yml on pankaj28843/docs-html-screenshot

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file docs_html_screenshot-0.2.0-py3-none-any.whl.

File metadata

File hashes

Hashes for docs_html_screenshot-0.2.0-py3-none-any.whl
Algorithm Hash digest
SHA256 1b6dd024737e284da061d16ce147a2906f41f62af14fe75cf99244cbf71f25d9
MD5 08604dff3dd109ee7893fd7aa7baa9d2
BLAKE2b-256 2c20c0775d9e0abb3eb85311b3e9bb4782e438bfe788d87cefbec6d8277beac3

See more details on using hashes here.

Provenance

The following attestation bundles were made for docs_html_screenshot-0.2.0-py3-none-any.whl:

Publisher: ci.yml on pankaj28843/docs-html-screenshot

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

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