Skip to main content

Async web crawler with rate limiting, robots.txt support, and broken link tracking

Project description

linktrace

Lightweight async web crawler for link analysis and HTML document processing.

Perfect for: Site structure analysis, link tracking, concurrent page fetching, HTML document transformation.

Not: A replacement for Scrapy. Use this when you need simple, focused crawling with automatic link classification and clean document models.

Key Features

  • โšก Async/await native โ€” Built on asyncio + aiohttp for concurrent requests
  • ๐Ÿ”— Automatic link classification โ€” Distinguishes internal vs external links by domain
  • ๐Ÿ“„ Rich document model โ€” Full HTML source, parsed links, metadata, headers
  • ๐Ÿ”„ Persistent sessions โ€” Connection pooling for 10-100x faster same-domain crawls
  • ๐Ÿ” Retries + backoff โ€” Exponential backoff for transient errors (timeouts, 5xx)
  • โฑ๏ธ Rate limiting โ€” Per-domain rate limiting with asyncio.Lock, no thundering herd
  • ๐Ÿค– robots.txt support โ€” Automatically respect Crawl-delay directives per domain
  • ๐Ÿ” Broken link tracking โ€” Audit 404s and 5xx errors for site structure validation
  • ๐Ÿ’พ Optional caching โ€” Disk-based cache (1-day TTL) for repeat crawls
  • ๐Ÿ” SSL verification โ€” Secure by default, with corporate proxy support
  • ๐Ÿช Automatic cookies โ€” Set-Cookie extraction and sending built-in
  • ๐Ÿ”€ Traversal strategies โ€” BFS (broad) or DFS (deep) crawling
  • ๐Ÿ“Š Multi-format export โ€” JSON, Pandas, Polars, PyArrow for data analysis
  • ๐Ÿ“ Callbacks & streaming โ€” Process results as crawled without memory buildup

Quick Start

import asyncio
from linktrace import Spider

async def main():
    spider = Spider(start_url="https://example.com", max_depth=2)
    documents = await spider.run_async()
    
    for doc in documents:
        print(f"{doc.url}")
        print(f"  Internal links: {len(doc.internal_links)}")
        print(f"  External links: {len(doc.external_links)}")

asyncio.run(main())

Installation

pip install linktrace

Optional export formats:

pip install linktrace[serializers]  # pandas + polars + pyarrow
pip install linktrace[pandas]       # Just pandas

Core Concepts

Spider

High-level orchestrator that crawls multiple pages using BFS (breadth-first) or DFS (depth-first) traversal.

Crawler

Low-level engine that fetches and parses individual documents. Handles retries, caching, SSL, cookies, sessions.

Document

Rich object containing:

  • url โ€” page URL
  • title โ€” HTML title tag
  • source โ€” raw HTML
  • internal_links โ€” links to same domain
  • external_links โ€” links to other domains
  • status_code, response_headers, domain โ€” metadata

See Core Concepts for more.

Configuration

Basic Crawl

spider = Spider(
    start_url="https://example.com",
    max_depth=3,              # How deep to follow links
    traversal_strategy="bfs"  # "bfs" (default) or "dfs"
)
documents = await spider.run_async()

Retries & Timeouts

spider = Spider(
    start_url="https://example.com",
    request_timeout=15,       # Seconds per request (default: 30)
    max_retries=5,            # Retry transient errors (default: 3)
)

Caching

spider = Spider(
    start_url="https://example.com",
    cache_dir=".webcrawler_cache"  # Enable disk caching (default: None/disabled)
)
# 2nd run will be 10-50x faster for same URLs

SSL & Corporate Proxies

# Default: verify SSL with system CA
spider = Spider(start_url="https://example.com")

# Corporate proxy with custom CA bundle
spider = Spider(
    start_url="https://example.com",
    ssl_verify="/path/to/corporate-ca.pem"
)

# Self-signed certs (testing only)
spider = Spider(
    start_url="https://example.com",
    ssl_verify=False  # โš ๏ธ Insecure
)

Cookies are handled automatically โ€” no configuration needed.

Callbacks: Process Results in Real-Time

For large crawls, avoid memory buildup by processing documents as they're crawled:

# Stream results to disk
async def save_result(doc):
    with open("results.jsonl", "a") as f:
        f.write(json.dumps({"url": doc.url, "title": doc.title}) + "\n")

spider = Spider(
    start_url="https://example.com",
    on_page_crawled=save_result,
    accumulate_results=False,  # Don't keep in memory
)
await spider.run_async()  # Returns [], file has results

Callback Hooks:

  • on_page_crawled(doc) โ€” Called after each successful crawl. Return value accumulated if accumulate_results=True
  • on_error(url, exc) โ€” Called on crawl failures
  • on_crawl_complete() โ€” Called when crawl finishes (cleanup hook)

Async Callbacks Supported:

async def save_to_db(doc):
    await db.insert(doc.url, doc.title)
    return doc.url

spider = Spider(
    start_url="https://example.com",
    on_page_crawled=save_to_db,       # Async callback
    accumulate_results=True,
)
results = await spider.run_async()  # Returns list of URLs

Return Logic:

  • No callback โ†’ returns all documents (default)
  • Callback + accumulate_results=False โ†’ returns [] (streaming mode)
  • Callback + accumulate_results=True โ†’ returns callback results

Traversal Strategies

BFS (Breadth-First) โ€” Default

# Explores level by level: all depth-1 links, then depth-2, etc.
spider = Spider(start_url="https://example.com", max_depth=3, traversal_strategy="bfs")

DFS (Depth-First)

# Follows single paths all the way down before exploring siblings
spider = Spider(start_url="https://example.com", max_depth=5, traversal_strategy="dfs")

Use DFS for deep hierarchies (documentation sites, nested directories). Use BFS for broad exploration.

Rate Limiting & robots.txt

By default, linktrace automatically respects robots.txt Crawl-delay directives and enforces per-domain rate limiting:

# Automatic robots.txt respect (default)
spider = Spider(
    start_url="https://example.com",
    user_agent="MyBot/1.0",  # Identifies your bot to robots.txt rules
)
await spider.run_async()

Customize rate limiting:

# Enforce explicit delay (ignores robots.txt)
spider = Spider(
    start_url="https://example.com",
    request_delay=1.0,           # 1 second between requests to same domain
    respect_robots_txt=False,    # Don't fetch robots.txt
)

# Concurrent requests to different domains, serialized to same domain
await spider.run_async()

Broken Link Audit

Track 404s and 5xx errors for site maintenance:

spider = Spider(start_url="https://example.com", max_depth=2)
documents = await spider.run_async()

for doc in documents:
    # Broken internal links (fix these first!)
    for broken in doc.broken_internal_links:
        print(f"{doc.url} โ†’ {broken.url} (HTTP {broken.status_code})")
    
    # Broken external links (check if still valid)
    for broken in doc.broken_external_links:
        print(f"External: {broken.url} (HTTP {broken.status_code})")

Stream broken links in real-time:

async def audit_broken(doc):
    broken_count = len(doc.broken_internal_links) + len(doc.broken_external_links)
    if broken_count > 0:
        print(f"{doc.url}: {broken_count} broken links")

spider = Spider(
    start_url="https://example.com",
    on_page_crawled=audit_broken,
    accumulate_results=False,
)
await spider.run_async()

Export Data

from linktrace import Spider, Serializers

spider = Spider(start_url="https://example.com", max_depth=2)
documents = await spider.run_async()

# Export to JSON
serializer = Serializers(documents)
serializer.to_json("crawl.json", include_html=False)

# Export to Pandas (one row per link)
df = serializer.to_pandas()
print(df[["url", "title", "link_url", "link_type"]])

# Export to Polars (faster for large datasets)
df_polars = serializer.to_polars()

# Export to PyArrow (for data pipelines)
table = serializer.to_arrow()

Link Analysis

from collections import Counter

spider = Spider(start_url="https://example.com", max_depth=2)
documents = await spider.run_async()

# Count external domains
external_domains = Counter()
for doc in documents:
    for link in doc.external_links:
        domain = link.url.split("/")[2]
        external_domains[domain] += 1

print(external_domains.most_common(10))

See Examples for more patterns.

Notebooks

Interactive examples in notebooks/:

  • crawl_cnn.ipynb โ€” Crawls CNN.com, analyzes link structure, demonstrates all export formats

API Reference

See API Reference for complete method documentation.

Troubleshooting

"SSL: CERTIFICATE_VERIFY_FAILED"

Use ssl_verify=False for self-signed certs (testing only), or ssl_verify="/path/to/ca.pem" for corporate proxies.

"Too many connections"

Reduce concurrency by lowering max_retries or increase timeouts. Default settings are conservative.

"Crawler hits timeout on deep sites"

Try DFS traversal instead of BFS, or increase request_timeout.

See Troubleshooting for more.

Performance

Typical performance (single-domain crawl):

  • First run: ~50-500ms per page (network-bound)
  • Cached run: ~1-10ms per page (2-50x faster)
  • Memory: ~1MB per 100 pages

With persistent sessions + connection pooling, same-domain requests are 10-100x faster than per-request session setup.

Architecture

Spider (orchestrator)
  โ””โ”€ Crawler (persistent session)
      โ”œโ”€ aiohttp (HTTP requests + connection pooling)
      โ”œโ”€ lxml (HTML parsing)
      โ”œโ”€ ResponseCache (optional disk caching)
      โ””โ”€ CookieJar (automatic cookie handling)

Spider manages the crawl queue and traversal. Crawler handles individual document fetching/parsing. All requests share one persistent aiohttp session per Spider instance.

Why linktrace?

vs Scrapy: Lightweight, focused, simpler API for link analysis. Scrapy is better for complex extraction pipelines.

vs requests + BeautifulSoup: Built-in async concurrency, automatic session reuse, retries, caching. Better for crawling multiple pages.

vs Selenium: Pure HTTP crawler (no JS execution). Faster, lighter, but can't handle dynamic sites.

Testing

just test          # Run all tests
just test-cov      # Run with coverage report

All 91 tests pass. 100% of core crawling paths tested (rate limiting, broken link tracking, robots.txt, callbacks).

Contributing

Bug reports and pull requests welcome on GitHub.

License

MIT


Documentation:

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

linktrace-0.1.2.tar.gz (174.5 kB view details)

Uploaded Source

Built Distribution

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

linktrace-0.1.2-py3-none-any.whl (16.5 kB view details)

Uploaded Python 3

File details

Details for the file linktrace-0.1.2.tar.gz.

File metadata

  • Download URL: linktrace-0.1.2.tar.gz
  • Upload date:
  • Size: 174.5 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for linktrace-0.1.2.tar.gz
Algorithm Hash digest
SHA256 b431cb5d6580cb76d873190c6ebe95205f8389a6e9e52fbbdaefafc44fa4d32d
MD5 fce7a66f078b1be66df2a16290e41811
BLAKE2b-256 ac0173d10e174744ae25ba38bcead59ee1034687c8b2ea08eb720d50acb10a35

See more details on using hashes here.

Provenance

The following attestation bundles were made for linktrace-0.1.2.tar.gz:

Publisher: publish.yml on JayBaywatch/webcrawler

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

File details

Details for the file linktrace-0.1.2-py3-none-any.whl.

File metadata

  • Download URL: linktrace-0.1.2-py3-none-any.whl
  • Upload date:
  • Size: 16.5 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for linktrace-0.1.2-py3-none-any.whl
Algorithm Hash digest
SHA256 9eeb80a7e1210562227d6cf28c4ada9cf8164afbb5a61d9ff8151e13328350bc
MD5 1941a3a55ef079d81c88ae7511219a93
BLAKE2b-256 30accec46952efbaee579965a50b217beb662f865b68fa90e4b1c0b93384c722

See more details on using hashes here.

Provenance

The following attestation bundles were made for linktrace-0.1.2-py3-none-any.whl:

Publisher: publish.yml on JayBaywatch/webcrawler

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