Reverse-engineer any website's internal API with natural language
Modern websites load data through hidden AJAX calls:
- Data isn't in the HTML - it's fetched via internal APIs
- You have to manually inspect network requests
- Complex pagination logic buried in JavaScript
- APIs change endpoints and parameters frequently
Strot analyzes websites and discovers their internal API calls for you.
What Strot actually does:
- Analyzes the webpage to understand what data you want
- Captures the AJAX request that fetches that data
- Detects all parameters - pagination (limit/offset, cursors, page numbers) AND dynamic parameters (sorting, filtering, search)
- Generates extraction code to parse the JSON response
- Returns a Source that replicates the website's own API calls with full parameter control
Get the full Strot experience (Web UI + API) instantly:
Using Patchright browser
docker run -d --name browser-server -p 5678:5678 synacktra/patchright-headless-server
STROT_ANTHROPIC_API_KEY=sk-ant-apiXXXXXX \
docker compose -f https://raw.githubusercontent.com/vertexcover-io/strot/refs/heads/main/docker-compose.yml upOr, using Steel browser
STROT_BROWSER_WS_URL=wss://connect.steel.dev?apiKey=ste-XXXXXX STROT_ANTHROPIC_API_KEY=sk-ant-apiXXXXXX \
docker compose -f https://raw.githubusercontent.com/vertexcover-io/strot/refs/heads/main/docker-compose.yml upThen visit:
- ๐ Web Dashboard: http://localhost:3000 - Visual interface for analyzing websites
- ๐ REST API: http://localhost:1337 - Programmatic access
pip install strotOr using uv:
uv pip install strotimport strot
import asyncio
from pydantic import BaseModel
class Review(BaseModel):
title: str | None = None
username: str | None = None
rating: float | None = None
comment: str | None = None
date: str | None = None
async def get_reviews():
# Strot discovers the internal API that loads reviews
source = await strot.analyze(
url="https://www.getcleanpeople.com/product/fresh-clean-laundry-detergent/",
query="Customer reviews with ratings and comments",
output_schema=Review,
code_executor="e2b" # Defaults to "unsafe"
)
# Use the same API call the website uses, with full parameter control
async for reviews in source.generate_data(limit=500, offset=100, sortBy="date desc"):
for review in reviews:
print(f"{review['rating']}โญ by {review['username']}: {review['comment']}")
if __name__ == "__main__":
asyncio.run(get_reviews())import strot
import asyncio
from pydantic import BaseModel
class Product(BaseModel):
name: str | None = None
price: float | None = None
rating: float | None = None
availability: str | None = None
description: str | None = None
async def get_products():
# Strot finds the AJAX endpoint that loads product listings
source = await strot.analyze(
url="https://blinkit.com/cn/fresh-vegetables/cid/1487/1489",
query="Listed products with names and prices.",
output_schema=Product
)
# Set code executor of your choice into source
source.set_code_executor("e2b")
# Access all products with custom filtering and sorting
async for products in source.generate_data(limit=100, offset=0, category="vegetables", sortBy="price"):
for product in products:
print(f"{product['name']}: ${product['price']} ({product['rating']}โญ)")
if __name__ == "__main__":
asyncio.run(get_products())Strot provides an MCP (Model Context Protocol) server that enables MCP clients like Claude to analyze websites and discover their internal APIs directly.
git clone https://github.com/vertexcover-io/strot.git
cd strot && uv sync --group mcpSet environment variables:
export STROT_ANTHROPIC_API_KEY=sk-ant-apiXXXXXX
export STROT_BROWSER_MODE_OR_WS_URL=headless # or 'headed' or ws://browser-urlIf you want to use a hosted browser, you could do the following:
Run a headless server inside docker
docker run -d --name browser-server -p 5678:5678 synacktra/patchright-headless-server
export STROT_BROWSER_MODE_OR_WS_URL=ws://localhost:5678/patchrightOr, connect to a remote browser server:
export STROT_BROWSER_MODE_OR_WS_URL=wss://...$ uv run strotmcp
Usage: strotmcp COMMAND
โญโ Commands โโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโฎ
โ http Start an http server. โ
โ sse Start an sse server. โ
โ stdio Start a stdio server. โ
โ streamable-http Start a streamable http server. โ
โฐโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโฏ
-
stdiotransportclaude mcp add strot-mcp "uv run strotmcp stdio" --transport stdio \ --env STROT_ANTHROPIC_API_KEY=$STROT_ANTHROPIC_API_KEY STROT_BROWSER_MODE_OR_WS_URL=$STROT_BROWSER_MODE_OR_WS_URL
-
sse,http,streamable-httptransportsRun the MCP server with the desired transport in separate terminal:
uv run strotmcp sse --port 8888 # uv run strotmcp http --host 0.0.0.0 # uv run strotmcp streamable-http --host 127.0.0.1 --port 8777
Add the MCP server to Claude:
claude mcp add strot-mcp http://127.0.0.1:8888/sse --transport sse
Test and validate Strot's analysis accuracy across different websites and individual components. The evaluation system supports five input types:
- existing jobs (evaluate completed jobs by ID),
- new jobs (create and evaluate full analysis tasks),
- request detection (test URL and query combinations),
- parameter detection (test individual request parameter analysis - both pagination and dynamic parameters),
- structured extraction (test response parsing with specific schemas).
It compares actual results against expected outcomes, tracking metrics like source URL matching, pagination key detection, dynamic parameter detection, and entity count accuracy. All evaluations and their detailed analysis steps are stored in Airtable.
git clone https://github.com/vertexcover-io/strot.git
cd strot && uv sync --group eval-
Create a new Airtable base:
- Go to Workspaces
- Click on
Createbutton on your desired workspace - Select
Build an app on your own - Copy the Base ID from the URL (e.g.
appXXXXXXXXXXXXXX)
-
Create a Personal Access Token:
- Go to
/create/tokens - Add the following scopes:
data.records:readdata.records:writeschema.bases:readschema.bases:write
- Give access to the base you created in step 1
- Press
Create tokenand copy the token (e.g.patXXXXXXXXXXXXXX)
- Go to
-
Set environment variables:
export STROT_AIRTABLE_BASE_ID=appXXXXXXXXXXXXXX export STROT_AIRTABLE_TOKEN=patXXXXXXXXXXXXXX # set the following if you wish to evaluate separate tasks export STROT_ANTHROPIC_API_KEY=sk-ant-apiXXXXXX
Note: Required tables are automatically created with proper schema when you run evaluations.
$ uv run stroteval
Usage: stroteval [OPTIONS]
Evaluate multiple job-based or task-based inputs from a file or stdin.
โญโ Parameters โโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโฎ
โ --file -f Path to the JSON/JSONL file. If not provided, reads from stdin. โ
โฐโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโฏ
Make sure the API server is running If you're evaluating job-based inputs.
echo '[
{
"job_id": "existing-job-uuid",
"expected_source": "https://api.example.com/reviews",
"expected_pagination_keys": ["cursor", "limit"],
"expected_dynamic_keys": ["sortBy", "category"],
"expected_entity_count": 243
},
{
"site_url": "https://example.com/category/abc",
"query": "Listed products with name and prices",
"expected_source": "https://api.example.com/products",
"expected_pagination_keys": ["limit", "offset"],
"expected_dynamic_keys": ["category", "sortBy"],
"expected_entity_count": 100
},
{
"request": {
"method": "GET",
"url": "https://example.com/api/products",
"type": "ajax",
"queries": {"page": "2", "limit": "50", "sortBy": "price", "category": "electronics"}
},
"expected_pagination_keys": ["page", "limit"],
"expected_dynamic_keys": ["sortBy", "category"]
},
{
"response": {
"value": "",
"request": {
"method": "GET",
"url": "https://example.com/api/reviews",
"type": "ajax"
},
"preprocessor": {"element_selector": ".reviews-container"}
},
"output_schema_file": "review_schema.json",
"expected_entity_count": 10
}
]' | uv run stroteval- ๐ฌ GitHub Discussions - Ask questions
- ๐ Report Issues - Found a bug?
MIT License - Use it however you want!
Made with โค๏ธ by Vertexcover Labs