Polymarket BTC Backtester MCP Server

Polymarket BTC Backtester MCP Server

Enables backtesting of limit-order strategies on Polymarket's BTC 5-minute markets using historical data, with tools to browse markets, get price series, and run simulations.

Category
访问服务器

README

Polymarket BTC Backtester (hosted MCP server)

A public MCP server for backtesting simple limit-order strategies on Polymarket's "BTC Up or Down" 5-minute markets. You paste one URL into Claude as a custom connector and then ask things like:

Backtest buying Up at 40 cents with a 5 cent take-profit across all markets and show me the win rate, the skew breakdown, and the date range tested.

Connector URL (replace with your deployment):

https://<your-app>.onrender.com/mcp

No login, no API key. The server is read-only and serves a static historical dataset, so there is nothing to protect. A small per-IP rate limit keeps it polite.

How it works

 Kaggle datasets (downloaded once, at image build time)
 ┌─────────────────────────────┐  ┌──────────────────────────┐
 │ 100ms ticks + outcomes      │  │ 2s snapshots (metadata)  │
 │ namz8888, ~14k markets      │  │ debayan31415, 1,191 mkts │
 └─────────────┬───────────────┘  └────────────┬─────────────┘
               │   pmbt/ingest.py              │
               │   - in-window quotes only     │
               │   - dedup to quote changes    │
               │   - prices -> integer 1/10c   │
               │   - winners from outcomes     │
               │   - price_to_beat + x-check   │
               ▼                               ▼
         ┌───────────────────────────────────────────┐
         │ data/store.db (SQLite, read-only, ~700MB) │
         │ markets / quotes / meta                   │
         └─────────────────────┬─────────────────────┘
                               │
         ┌─────────────────────┴─────────────────────┐
         │ pmbt/server.py (FastMCP, streamable HTTP) │
         │  /mcp     5 tools                         │
         │  /        landing page                    │
         │  /health  liveness                        │
         └─────────────────────┬─────────────────────┘
                               │ HTTPS (Render)
                               ▼
                  Claude custom connector

The backtest engine (pmbt/engine.py) is pure functions over integer prices. It knows nothing about BTC reference prices or price_to_beat, so display data can never leak into PnL.

Tools

tool what it does
data_coverage date range, market counts, list of missing or partial days
list_markets browse markets with winner, price_to_beat, tradeability
get_price_series bid/ask series for both sides plus BTC reference, resampled
backtest the strategy simulator, see below
market_context one market in full: window, strike, winner, tick coverage

The strategy that gets backtested

For each market in the chosen date range:

  1. Rest a limit buy on the chosen side at your entry price. It fills at the first tick where that side's best ask is at or below the entry price, always at the entry price exactly. One trade per market, full size.
  2. On entry, rest a limit sell at entry plus the take-profit. It fills at the first later tick where the best bid reaches the target, always at the limit price, even when the bid gaps through it.
  3. If the take-profit never fills, the position rides to resolution: $1.00 if the held side won, $0.00 if it lost.

Stated assumptions (also returned by every backtest call):

  • Touch-fill: resting orders fill when the opposing best quote reaches their level. No queue position or book depth model. Empirically reasonable in these very liquid markets, still an approximation.
  • Both legs are maker orders and Polymarket makers pay zero fees here, so the default fee is 0. A taker_fee override exists for simulating market orders.
  • Redemption at $1/$0 is free.
  • Static dataset. Past quotes, not live ones.

The results make the skew of this strategy very visible: win rates above 70% with negative expectancy, because the upside is capped at the take-profit while a forced loss costs the whole entry price.

Resolution rules

"Up" wins when the Chainlink BTC/USD price at window end is at or above the price at window start. Ties resolve Up. The engine takes winners only from the outcomes file of the 100ms dataset, never from the Binance/Coinbase/Kraken reference prices in the data, because those prints differ from Chainlink enough to flip close markets.

Data

Two Kaggle datasets, credit where due:

  • Polymarket BTC 5-Minute 100ms Market Data by namz8888. The backbone: 100ms quotes and resolved outcomes for ~14,000 markets over about 7 weeks (2026-03-05 to 2026-04-25). The tradeable universe is exactly the markets here that have both quotes and an outcome.
  • Polymarket 5 minutes BTC UP Down data by debayan31415. 2s snapshots for 1,191 markets (2026-02-23 to 2026-03-05). Used only for the official price_to_beat and for cross-checking winners. Markets that exist only here are never backtested, since 2s sampling misses price touches.

The two datasets currently share zero markets (they cover adjacent date ranges), so cross-validation had nothing to flag and tradeable markets show an approximate price_to_beat taken from the first reference tick. That value is display only. See docs/DATA_SCHEMA.md for the full schema and decisions.

Run it locally

pip install -r requirements.txt

# build the data store (~600MB download, a few minutes)
python -m pmbt.ingest --download --raw data_raw --db data/store.db

# tests
python -m pytest tests/ -q

# serve
PORT=8000 python -m pmbt.server
# landing page: http://localhost:8000/
# MCP endpoint: http://localhost:8000/mcp

Deploy to Render

The Dockerfile downloads the datasets and builds the SQLite store during the image build, so the running container never fetches data. Host disks being ephemeral does not matter; the store ships inside the image.

  1. Push this repo to GitHub.
  2. On render.com: New, then Web Service, then connect the repo. Render picks up render.yaml and the Dockerfile automatically (runtime: Docker, health check on /health). The starter plan is enough; the image is about 1GB because of the baked store.
  3. Wait for the build (the Kaggle download runs inside it, expect ~10 min).
  4. Your connector URL is https://<app-name>.onrender.com/mcp. HTTPS is automatic. Open the root URL to see the landing page with the copy button.

Railway and Fly work the same way: deploy the Dockerfile, expose $PORT, done. The server listens on 0.0.0.0:$PORT and defaults to 8000.

Add it to Claude

  1. Settings, then Connectors, then Add custom connector.
  2. Paste the /mcp URL. No auth fields needed; the connector flow probes the OAuth discovery endpoints, gets clean 404s, and falls back to anonymous.
  3. In a chat, open the plus menu and enable the connector.
  4. Ask for a backtest.

Repo layout

pmbt/
  ingest.py    Kaggle files -> SQLite store
  store.py     store schema + connections
  db.py        read-only query layer
  engine.py    backtest engine (pure, integer prices)
  server.py    FastMCP server, tools, rate limiting
  landing.py   the landing page HTML
scripts/
  discover_schema.py   raw-schema dump (run before trusting any column name)
docs/
  DATA_SCHEMA.md             schemas and ingest decisions
  schema_discovery_output.txt captured discovery output
tests/         engine + store tests (22)
Dockerfile     two-stage build, store baked at build time
render.yaml    Render blueprint

Not financial advice. Built as a portfolio project.

推荐服务器

Baidu Map

Baidu Map

百度地图核心API现已全面兼容MCP协议,是国内首家兼容MCP协议的地图服务商。

官方
精选
JavaScript
Playwright MCP Server

Playwright MCP Server

一个模型上下文协议服务器,它使大型语言模型能够通过结构化的可访问性快照与网页进行交互,而无需视觉模型或屏幕截图。

官方
精选
TypeScript
Magic Component Platform (MCP)

Magic Component Platform (MCP)

一个由人工智能驱动的工具,可以从自然语言描述生成现代化的用户界面组件,并与流行的集成开发环境(IDE)集成,从而简化用户界面开发流程。

官方
精选
本地
TypeScript
Audiense Insights MCP Server

Audiense Insights MCP Server

通过模型上下文协议启用与 Audiense Insights 账户的交互,从而促进营销洞察和受众数据的提取和分析,包括人口统计信息、行为和影响者互动。

官方
精选
本地
TypeScript
VeyraX

VeyraX

一个单一的 MCP 工具,连接你所有喜爱的工具:Gmail、日历以及其他 40 多个工具。

官方
精选
本地
graphlit-mcp-server

graphlit-mcp-server

模型上下文协议 (MCP) 服务器实现了 MCP 客户端与 Graphlit 服务之间的集成。 除了网络爬取之外,还可以将任何内容(从 Slack 到 Gmail 再到播客订阅源)导入到 Graphlit 项目中,然后从 MCP 客户端检索相关内容。

官方
精选
TypeScript
Kagi MCP Server

Kagi MCP Server

一个 MCP 服务器,集成了 Kagi 搜索功能和 Claude AI,使 Claude 能够在回答需要最新信息的问题时执行实时网络搜索。

官方
精选
Python
e2b-mcp-server

e2b-mcp-server

使用 MCP 通过 e2b 运行代码。

官方
精选
Neon MCP Server

Neon MCP Server

用于与 Neon 管理 API 和数据库交互的 MCP 服务器

官方
精选
Exa MCP Server

Exa MCP Server

模型上下文协议(MCP)服务器允许像 Claude 这样的 AI 助手使用 Exa AI 搜索 API 进行网络搜索。这种设置允许 AI 模型以安全和受控的方式获取实时的网络信息。

官方
精选