The web-scraping library agents deserve.
Selector-cheap. LLM-resilient. Replay-safe. Self-hosted.
Quickstart | Architecture | Features | Why Scrapo | CLI | MCP
Scrapo is a Python library that fuses four worlds the rest of the market keeps separate:
|
AI-native ingestion |
Agentic browsing |
Production crawling |
Managed access |
Plus a feature nobody else ships: deterministic replay of every fetch, so extraction drift is auditable.
Not a developer? layman.md explains what Scrapo does, and what it cannot do, in plain English.
| 5-tier router | Hybrid extractor | Model pinning |
|---|---|---|
| Auto-escalates HTTP, browser, stealth, agent on real failure signals only | Selector-cheap by default; falls back to LLM and self-heals | Strict mode refuses unpinned LLM extraction, so extraction cannot silently drift |
| Provenance | Deterministic replay | Safe by default |
|---|---|---|
| Every chunk carries URL, selector path, byte range, heading trail | Re-extract from archived HTML 6 months later; diff fields between any two runs | SSRF guard, opt-in robots gate, regex+Luhn PII, geo allow/deny, append-only audit log |
+---------------------------------------------+
| scrapo.scrape(...) |
+-------------------+-------------------------+
|
+----------------------------+----------------------------+
| | |
(1) Tier Router (2) Extractor (3) Document Shaper
T0 HTTP (retries) Selector cache (host key) HTML to Markdown
T1 HTTP+session -> if validation fails Heading-aware chunks
T2 Browser LLM fallback (budgeted) Per-chunk provenance
T3 Browser+stealth -> self-heal / evict Cross-crawl dedup
T4 Agent selector cache writeback
| | |
+---------+------------------+---------+-------------------+
| |
(4) Replay store (5) Policy gate
SQLite (WAL) + gzip HTML SSRF / robots / PII / geo / audit
| |
+-------------+--------------+
|
(6) Agent surface
MCP server + tool schemas
scrapo/
├── access/ # (1) 5-tier router + pooled browser + request interception + agent driver + action cache + proxy adapters & rotating pool
├── extract/ # (2) hybrid selector + LLM (scalar & list fields), model pinning, cost-aware budget
├── shape/ # (3) markdown + heading chunker + content-type dispatch (HTML / JSON / feed / PDF / text)
├── replay/ # (4) SQLite metadata + pluggable snapshot store (local or S3) + field-level diff
├── policy/ # (5) robots, PII (flag or redact), geo, append-only audit
├── crawl/ # persistent SQLite queue + async scheduler + sitemap discovery + rel=next pagination
├── agent/ # (6) MCP server + tool schemas
├── results.py # typed ScrapeResult / CrawlResult / ExtractionView
├── watch.py # watch(url) -> Watch.refresh() -> ChangeSet (change tracking)
├── security.py # SSRF guard for fetch targets
├── _db.py # tuned SQLite connections (WAL, busy timeout)
├── logging.py # structlog setup for the CLI / MCP server
├── api.py # public scrape / extract / crawl / crawl_stream
├── web.py # local browser UI (scrapo serve)
└── cli.py # Typer CLI
pip install scrapo
pip install "scrapo[browser,anthropic,mcp]"
playwright install chromiumimport asyncio, scrapo
async def main():
res = await scrapo.scrape("https://example.com/") # res is a typed ScrapeResult
print(res.markdown)
print("run_id:", res.run_id)
# res["markdown"] / res.get("status") still work too (back-compat with the 0.1 dict)
asyncio.run(main())import asyncio, scrapo
from pydantic import BaseModel
class Offer(BaseModel):
name: str
price: str
class Listing(BaseModel):
page_title: str
offers: list[Offer] = [] # array fields become repeated-element extraction
async def main():
res = await scrapo.scrape("https://example.com/shop", schema=Listing)
print(res.extraction.data) # {'page_title': '...', 'offers': [{'name': ..., 'price': ...}, ...]}
print(res.extraction.method) # 'llm' on the first run, 'selector' after
print(res.cost_usd) # 0.0 once selectors are cached
asyncio.run(main())First call uses the LLM and caches the selectors it learns (keyed by host + schema; for
list[Model]fields it caches a container selector plus per-subfield selectors). Every subsequent call against that host + schema uses cached selectors and zero LLM tokens. When the layout drifts, validation fails, Scrapo falls back to the LLM, re-derives selectors, and self-heals; a cache entry that keeps failing is evicted automatically.
await scrapo.crawl(
seeds=["https://docs.python.org/3/"],
max_depth=2,
same_host_only=True,
)scrapo list # recent runs
scrapo replay <run_id> # re-extract from archived HTML, no network
scrapo diff <run_a> <run_b> # field-level diffCost-aware tier router
| Tier | What it does | When |
|---|---|---|
T0 HTTP |
httpx plain GET, with bounded retry/backoff on 429/5xx and transport errors |
static HTML, JSON endpoints |
T1 HTTP_SESSIONED |
+ browser-like headers/cookies | soft anti-bot |
T2 BROWSER |
Playwright headless | JS-rendered pages, SPA shells |
T3 BROWSER_STEALTH |
+ stealth + residential proxy | hard anti-bot |
T4 AGENT |
LLM-driven multi-step browser via a pluggable AgentDriver (a reference LLMAgentDriver ships in; SCRAPO_AGENT_DRIVER=llm), with action caching so a repeated goal replays without the LLM |
logins, captchas, flows |
Escalation triggers: Cloudflare/Akamai/PerimeterX/DataDome/Distil fingerprints, HTTP 403 / 429 / 503, empty body, missing required schema fields, and unrendered single-page-app shells (lots of script, almost no rendered text). Budget(max_tier=..., max_llm_calls=..., max_cost_usd=...) caps how far it goes. The browser tiers block images/fonts/media/css by default and capture JSON XHR/fetch responses onto the result. The Tier-4 driver records the action sequence it used to reach a goal on a host (agent_actions.sqlite) and replays it on later runs with zero LLM tokens, self-healing back to the model only when a recorded step no longer applies (SCRAPO_AGENT_ACTION_CACHE=0 to disable).
Content-type aware: HTML, JSON, feeds, PDFs
A URL is not always an HTML page, so scrape() dispatches on Content-Type (with a little body sniffing):
| Content | What you get | result.kind |
|---|---|---|
text/html |
the normal selectolax + markdown + chunk pipeline | html |
application/json / ld+json |
pretty-printed JSON as markdown; parsed object on result.data |
json |
| RSS / Atom | a markdown list of entries; parsed items on result.data |
feed |
application/pdf |
extracted text (requires pip install "scrapo[pdf]") |
pdf |
text/plain |
the body verbatim | text |
Hybrid selector + LLM extractor (scalar and list fields)
cache hit + validates -> return (method=selector, llm_calls=0, cost_usd=0)
miss / fail / over budget -> LLM with schema -> validate -> verify + persist selectors -> return (method=llm)
repeated cache failures -> evict the stale entry, re-derive next run
The LLM is asked to return both the JSON payload and CSS selectors per field. A scalar field gets a string selector; a list[Model] field gets {"__list__": "<repeating element>", "<subfield>": "<selector relative to it>", ...}, which Scrapo applies as tree.css(container) then per-subfield extraction inside each match. Returned selectors are verified against the live HTML before being cached, so a hallucinated selector never poisons the cache. The cache is keyed by host (not registered domain), so blog.example.com and shop.example.com never collide.
Model pinning (Zyte-style, but built in)
from scrapo.extract.pinning import PinnedModel
pin = PinnedModel.make(
provider="anthropic",
model_id="claude-opus-4-7",
prompt_template="(your prompt template)",
)
await scrapo.scrape(url, schema=Product, pin=pin, strict_pin=True)strict_pin=True makes the extractor refuse to run if the configured LLM does not match the pin. Silent model drift cannot happen in production.
Per-chunk provenance
Every chunk Scrapo emits carries:
{
"url": "https://example.com/page",
"selector_path": "markdown://Features/Pricing",
"byte_start": 8421,
"byte_end": 9842,
"heading_trail": ["Features", "Pricing"],
"chunk_hash": "ab12cd34...",
}You can trace any LLM citation back to a specific section of a specific URL.
Deterministic replay and diff
Every fetch persists raw HTML, headers, screenshots, and the typed extraction:
scrapo replay 9f3e1c... # re-extract from archived HTML, no network
scrapo diff 9f3e1c... abc123... # field-level diff, with notes when model/schema changedSample diff output:
diff 9f3e1c... vs abc123...
HTML changed
! model changed: anthropic:claude-opus-4-7 -> anthropic:claude-sonnet-4-6 (extraction may drift)
field changes:
- price: '$42' -> '$45'
- in_stock: True -> False
scrapo scrape <url> --diff-last prints that diff against the previous run of the same URL in one step.
Watch a URL for changes (cheap re-scrapes)
Re-scraping a URL the HTTP tier fetched before sends a conditional GET (If-None-Match / If-Modified-Since). A 304 Not Modified is rebuilt from the archived snapshot: no body transfer, no LLM call (the selector cache makes re-extraction free), and no duplicate snapshot is written. scrape() / crawl() get this automatically; Config(conditional_requests=False) (or SCRAPO_CONDITIONAL_REQUESTS=0) turns it off.
watch() builds the change-tracking loop on top of that:
import scrapo
w = await scrapo.watch("https://example.com/pricing", schema=Pricing)
# ... later, or on a schedule of your choosing ...
change = await w.refresh()
if change.not_modified:
print("unchanged (304)")
elif change.changed:
print(change.summary()) # field-level diff vs. the previous run
for d in change.field_changes:
print(d) # e.g. price: '$42' -> '$45'Watch is in-process: the run history (and the diff) live in the replay store. Persisting a list of watches with a built-in scheduler is a hosted-service concern and is intentionally left out.
Streaming crawl
async for page in scrapo.crawl_stream(["https://blog.example.com/"], schema=Post):
save(page) # process pages as they complete, not all at the endBreaking out of the loop early stops the crawl and tears the shared browser down. crawl() remains the buffered convenience (returns aggregate stats + an on_page callback).
Resilience and safety
- SSRF guard. Every fetch target is checked before a request goes out; loopback, link-local (including
169.254.169.254), private RFC 1918 / ULA ranges, and well-known local hostnames are refused. IP literals are parsed withinet_aton-style semantics, so obfuscated encodings (decimal2130706433, hex0x7f000001, short-form127.1, dotted-octal0177.0.0.1) of internal addresses are caught too. Tier-4 agentgotoactions chosen by the LLM go through the same guard. Setallow_private_hosts=True(orSCRAPO_ALLOW_PRIVATE_HOSTS=1) for internal scraping. Crawl link discovery applies the same filter and skips obvious binary URLs. - Bounded HTTP retries. Transient
429 / 5xxand transport errors are retried with exponential backoff and jitter before the router escalates to a heavier tier (SCRAPO_HTTP_RETRIES, default2). - Rotating proxy pool with health checks. Hand Scrapo a list of proxy URLs (
Config(proxy_urls=[...])orSCRAPO_PROXY_URLS="http://a,http://b") and the router round-robins across them. Every fetch's outcome is fed back: an HTTP 4xx auth/rate-limit code or an anti-bot fingerprint parks that proxy forproxy_cooldown_seconds(it's an IP-level block); a transient 5xx / network error counts towardmax_failures; a clean fetch resets the streak.ProxyPoolimplements theProxyAdapterprotocol, so it composes with the vendor adapters; passupstream=<adapter>to fall back to a managed gateway when every static endpoint is parked. Credentials are stripped from proxy URLs before they're logged. - Concurrency-safe storage. All SQLite stores (replay, selector cache, crawl queue, agent action cache) open in WAL mode with a busy timeout, and the per-store init step is guarded by an
asyncio.Lockso concurrent first callers can't race on schema setup. Crawl workers honoring per-hostcrawl-delayno longer block workers for other hosts. - Pluggable snapshot storage. Replay metadata stays in SQLite, but the page bodies go through a
SnapshotStore: local files by default (atomic write-then-rename, so a crash mid-write can't leave a partial snapshot recorded as complete), or S3 withsnapshot_backend="s3://bucket/prefix"(pip install "scrapo[s3]"). A corrupt archive is detected on read and falls back to a fresh fetch instead of raising. - Browser reuse and lighter pages. A
TierRouterlaunches one headless Chromium lazily and reuses it across fetches (proxy applied per context); the browser tiers also block images/fonts/media/css and surface JSON XHR responses.TierRouter.aclose()tears it down;scrape()andcrawl()handle that for you. - Cost accounting. LLM cost is computed per call, recorded on the run, and enforceable via
Budget(max_llm_calls=..., max_cost_usd=...). - PII handling. Flag PII in the audit log (
SCRAPO_PII_FILTER=1), or redact it from the stored snapshot, markdown, and chunks (SCRAPO_REDACT_SNAPSHOTS=1). - Local UI hardening.
scrapo servebinds127.0.0.1by default, validates theHostheader against an allowlist (anti DNS-rebinding), serializes scrapes, and warns loudly if you bind a public interface.
Built-in compliance layer
- Optional
robots.txtparser with per-origin caching (off by default; setSCRAPO_RESPECT_ROBOTS=1) - Regex PII classifier (email, phone, SSN, credit card with Luhn, IPv4, IBAN, passport), flag or redact
- Geo policy with EU-only preset (
GeoPolicy.eu_only()) plus custom allow/deny lists - Append-only JSONL audit log of every scrape, block, geo violation, PII detection
You are responsible for complying with each site's terms of use and applicable law; these are tools, not a guarantee.
BYO proxy adapters
from scrapo.access.adapters.brightdata import BrightDataAdapter
import scrapo
adapter = BrightDataAdapter() # reads BRIGHTDATA_USERNAME / BRIGHTDATA_PASSWORD
await scrapo.scrape("https://hard-target.com/", proxy_adapter=adapter)Built-in: brightdata, oxylabs, scrapfly, zyte. Implement the ProxyAdapter protocol for anything else:
from scrapo.access.adapters.base import ProxyConfig
class MyAdapter:
name = "my-vendor"
async def get_proxy(self, geo=None):
return ProxyConfig(url="http://user:pass@my-proxy:8080", region=geo)Got your own list of proxies instead of a managed gateway? Use the built-in rotating pool: it round-robins, tracks per-endpoint health, and parks one that starts getting blocked:
from scrapo.access import ProxyPool
pool = ProxyPool(["http://u:p@a:8080", "http://u:p@b:8080"]) # or ProxyPool.from_env() / Config(proxy_urls=[...])
await scrapo.scrape("https://hard-target.com/", proxy_adapter=pool)
pool.stats() # per-endpoint successes / failures / cooldownPass upstream=BrightDataAdapter() to ProxyPool to fall back to a managed gateway when every endpoint in the pool is cooling down.
BYO LLM adapters
| Provider | Adapter | Install | Default model |
|---|---|---|---|
| Anthropic Claude | anthropic (default) |
pip install "scrapo[anthropic]" |
claude-opus-4-7 |
| OpenAI | openai |
pip install "scrapo[openai]" |
gpt-4o-mini |
| Google Gemini | gemini |
pip install "scrapo[gemini]" |
gemini-2.5-flash |
| Mock (offline) | mock (tests) |
always available | n/a |
Anthropic adapter uses prompt caching on the schema block, so repeated extractions against the same Pydantic schema are cheap.
scrapo scrape https://example.com/
scrapo scrape https://example.com/ --max-tier 3 --screenshot --out-md page.md
scrapo crawl https://docs.python.org/3/ --max-depth 2 --max-pages 100
scrapo list --limit 10
scrapo replay <run_id>
scrapo diff <run_a> <run_b>
scrapo audit # tail the append-only audit log
scrapo adapters # list registered proxy adapters
scrapo serve # local browser UI at http://127.0.0.1:8787
scrapo mcp # run the MCP server over stdioScrapo ships an MCP server exposing five tools to any MCP-compatible client (Claude Code, Claude Desktop, Cursor, and others):
scrapo_scrape scrapo_crawl scrapo_replay scrapo_diff scrapo_list_runs
pip install "scrapo[mcp]"
scrapo mcpAdd to your client config:
{
"mcpServers": {
"scrapo": {
"command": "scrapo",
"args": ["mcp"]
}
}
}The SSRF guard is on by default, which matters here: an MCP client driven by an LLM that just read a page cannot be talked into fetching your internal services.
Every default is overridable via env var:
| Variable | Default | Notes |
|---|---|---|
SCRAPO_DATA_DIR |
platform user-data dir | SQLite + snapshots + audit log |
SCRAPO_USER_AGENT |
scrapo/0.1 |
UA for HTTP and robots |
SCRAPO_TIMEOUT |
30 |
request timeout (s) |
SCRAPO_CONCURRENCY |
8 |
crawl concurrency |
SCRAPO_HTTP_RETRIES |
2 |
retries on 429/5xx/transport errors |
SCRAPO_CONDITIONAL_REQUESTS |
1 |
0 to disable conditional GET / 304-archive reuse on re-scrapes |
SCRAPO_RESPECT_ROBOTS |
0 |
1 to enable the robots gate |
SCRAPO_PII_FILTER |
0 |
1 to flag PII in the audit log |
SCRAPO_REDACT_SNAPSHOTS |
0 |
1 to redact PII from stored snapshots/markdown/chunks |
SCRAPO_ALLOW_PRIVATE_HOSTS |
0 |
1 to allow fetching private/loopback addresses |
SCRAPO_SNAPSHOT_BACKEND |
local |
local or s3://bucket/prefix |
SCRAPO_BROWSER_BLOCK_RESOURCES |
1 |
0 to let the browser tier load images/fonts/media/css |
SCRAPO_BROWSER_CAPTURE_XHR |
1 |
0 to skip capturing JSON XHR/fetch responses |
SCRAPO_AGENT_DRIVER |
unset | llm to enable the built-in Tier-4 agent driver |
SCRAPO_AGENT_ACTION_CACHE |
1 |
0 to disable recording/replaying Tier-4 agent action sequences |
SCRAPO_PROXY_ADAPTER |
unset | default registered adapter name |
SCRAPO_PROXY_URLS |
unset | comma-separated proxy URLs for the rotating pool (used when no adapter is set) |
SCRAPO_PROXY_COOLDOWN |
120 |
seconds a parked proxy stays out of rotation |
SCRAPO_LLM_ADAPTER |
anthropic |
default LLM provider |
SCRAPO_LLM_MODEL |
claude-opus-4-7 |
default model id |
SCRAPO_GEO |
unset | default proxy region |
SCRAPO_LOG_LEVEL |
INFO |
log level for the CLI / MCP server |
SCRAPO_LOG_FORMAT |
console |
console or json |
ANTHROPIC_API_KEY |
unset | for the Claude adapter |
OPENAI_API_KEY |
unset | for the OpenAI adapter |
GEMINI_API_KEY |
unset | for the Gemini adapter |
pip install -e ".[dev]"
pytest -q
ruff check .
mypy scrapo/The suite is fully offline; no test hits the network or a paid LLM. It covers signals (including SPA-shell detection), SSRF, the HTTP retry path, conditional GET / 304-archive reuse, watch() change tracking and crawl_stream, shape, extract (cache eviction, budget, cost), replay (and the schema migration), policy, dedup, queue, router, proxy adapters and the rotating pool (rotation, cooldown, hard vs. soft failures, upstream fallback, the tier feedback loop), the agent driver and its action cache (record / replay / self-heal / eviction with fake page + scripted LLM), the local web UI, config, and end-to-end scrape with monkeypatched fetchers.
Issues and PRs welcome. See CONTRIBUTING.md for the dev setup and house rules, and SECURITY.md for reporting vulnerabilities.
Alpha. The public API (scrape, extract, crawl, crawl_stream, watch) is stable, as are tier escalation, model pinning, replay schema, typed results, list extraction, content-type routing, the pluggable snapshot store, the rotating proxy pool, conditional requests, and the MCP tool surface. The reference Tier-4 agent driver (with action caching: record the steps to a goal, replay them token-free, self-heal back to the LLM when a step breaks) and the in-browser request interception are functional but lightly exercised (a real browser is needed to validate them end to end). The library roadmap is otherwise complete; the only thing intentionally left out is a hosted control plane (a scheduler that runs and persists a list of watches, sends alerts, and gives you a web console). That would be a separate deployable service rather than part of the library.
See CHANGELOG.md for release notes.
MIT (c) Scrapo contributors
⭐ Star the repo if Scrapo saved you a week of glue code.