footnote-mcp
Provides scholarly search capabilities for papers on arXiv.
Provides web search capabilities using Brave Search API for independent web indexing.
Provides web search capabilities via DuckDuckGo (scraped) as a fallback search engine.
Provides web search capabilities using Google Custom Search API.
Provides local LLM-based claim entailment verification and semantic reranking using Ollama models.
Provides encyclopedic search capabilities on Wikipedia.
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@footnote-mcpverify claim 'vaccines save lives' from credible sources"
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
An MCP server for source-grounded web research. It searches the web, fetches and extracts pages, pulls structured data out of tables/files/APIs, and — the part that sets it apart — verifies that a claim is actually supported by its source instead of trusting a snippet. 42 tools over stdio MCP, driven by any MCP client (Claude Desktop, Cursor) or by the companion Scholiast research agent.
The design priority is trustworthiness over convenience: search snippets are treated as discovery only, every fetched page is cached with provenance, and claims are checked against the source text before they count. It also degrades gracefully — with no API keys and no config it still works (scraped search + an automatic headless-browser fallback + an offline verification heuristic); keys and env vars only make it better.
Quick start
From source (Python ≥ 3.10):
python3 -m venv .venv && source .venv/bin/activate
pip install -e . # installs the `footnote-mcp` console script + deps
python -m playwright install chromium # the headless browser used by the fetch fallback
footnote-mcp # start the server (speaks MCP over stdio)footnote-mcp now waits for an MCP client on stdio. Point a client at it by dropping this
into its MCP settings (Claude Desktop: claude_desktop_config.json; Cursor: ~/.cursor/mcp.json):
{
"mcpServers": {
"footnote": { "command": "footnote-mcp" }
}
}No API keys are required to start — search falls back to scraping Bing + DuckDuckGo. Add
keys later under "env" (see Search backends). Pass --headed to watch
the browser tier work.
Optional runtime variables are documented in .env.example. Copy it to
.env for local shells, or paste selected variables into your MCP client config:
{
"mcpServers": {
"footnote": {
"command": "footnote-mcp",
"env": {
"TAVILY_API_KEY": "..."
}
}
}
}To run without installing, straight from the source tree:
PYTHONPATH=src python -m footnote_mcpRelated MCP server: rust-research-mcp
Verifying claims — the differentiator
The reason to use this over a plain search tool is evidence_entailment and friends:
they tell a claim a source supports from one it does not. benchmarks/run_benchmark.py
measures that on a labeled set of claim/source pairs (and demos corroborate_claim and
locate_claim_span):
python benchmarks/run_benchmark.py # offline heuristic (deterministic)
python benchmarks/run_benchmark.py --backend ollama # LLM judge (needs ollama)Offline-heuristic result on the labeled set (benchmarks/REPORT.md):
Set | n | Accuracy | Unsupported-claim catch rate | Precision on "supported" |
Data domain (numeric + factual) | 15 | 100% | 100% | 100% |
Overall (incl. semantic) | 18 | 83% | 78% | 80% |
On its design domain — numeric and factual data claims — the offline heuristic never
blesses an unsupported claim and never misses one. Its blind spot is purely-semantic
negation/paraphrase; for those, evidence_entailment with backend="ollama" (a local LLM
judge) closes the gap. Run the --backend ollama line above to score that path on your own
machine.
Tool surface (42 tools)
Tool | Description |
| Keyed provider (Tavily/Brave/Google) when available, else scraped Bing + DuckDuckGo. Snippets are discovery only. |
| Search restricted to a recency window (day/week/month/year). |
| Search, fetch, extract, rerank, and return source context. |
| Fetch one URL, extract text, classify source quality, persist cache metadata. |
| Search arXiv (papers) or Wikipedia (encyclopedic) corpora. |
| Find the closest Wayback Machine snapshot for a dead/changed URL. |
| Fetch a page that needs cookies or custom headers. |
| Breadth-first crawl from a start URL, on-host by default (≤ 50 pages). |
| Generate operator queries ( |
Tool | Description |
| Parse HTML tables into |
| Detect linked CSV/TSV/XLS/XLSX/PDF/JSON/XML files. |
| Download and parse CSV/TSV/XLS/XLSX/PDF/JSON. |
| Fetch direct API/JSON endpoints into parsed JSON. |
| Validate required date coverage (day/week/month). |
| Detect currencies, currency pairs, measurement units. |
| Reject rows with incompatible units or currency pairs. |
| Align series on a key, compute deltas, flag missing keys/outliers. |
| Write consolidated rows to a |
Tool | Description |
| Classify official / aggregator / blog / forum / interactive / blocked / error. |
| Strict claim-vs-source checker: |
| Triangulate a claim across excerpts (corroborated / conflicting / single_source / …). |
| Locate supporting sentence(s) with char offsets and a containment score. |
| Inspect and write persistent source-cache entries. |
| Compact report of queries, URLs, source quality, verification gaps. |
| Check parser, OCR, browser, and cache dependencies. |
When generic parsers fail, synthesize a sandboxed parser:
Tool | Description |
| Propose a task-specific extraction recipe spec. |
| Generate a starter |
| Validate recipe code against a static safety allowlist. |
| Run validated code in a limited subprocess (JSON output only). |
| Save a validated recipe as reusable memory (no server edit). |
| Manage promoted recipes: |
A controlled Chromium session for JS-heavy or interactive pages:
Tool | Description |
| Drive a page via stable element refs. |
| Set a date range, submit, extract visible tables. |
| Save a PNG and optionally OCR text locked inside the image. |
Search backends
web_search (and everything built on it) routes through a provider layer. With an API key
it uses that provider; otherwise it scrapes Bing + DuckDuckGo. Results are normalized to one
shape regardless of backend.
Provider | Env vars | Notes |
Tavily |
| LLM-oriented search API. |
Brave |
| Independent web index. |
| Programmable Search (Custom Search JSON API). | |
Bing + DuckDuckGo | none | Default fallback; scraped, no key. |
auto (default) tries each keyed provider in order Tavily → Brave → Google, then scrapes.
Force one with the provider argument (tavily/brave/google/scrape).
Semantic reranking. Pass semantic: true to web_search to reorder by meaning rather
than keyword overlap: it over-fetches, embeds query and results with a local ollama model,
and sorts by cosine similarity (each result gains semantic_score). Best-effort — if ollama
is unavailable the original order is returned. Model: FOOTNOTE_EMBED_MODEL (default bge-m3).
Fetching & anti-bot ladder
web_read fetches through an escalation ladder (scraper.py):
the cheapest method runs first and escalates only when a result looks blocked or empty. A
block/quality detector decides when to escalate; a per-domain rate limiter, circuit breaker,
and negative cache keep it polite. The tier used and the full attempt trace come back in
fetch_tier / scrape_tiers.
Tier | Method | Enabled by |
1 | HTTP (curl_cffi TLS impersonation) | always |
2 | HTTP through a rotating proxy |
|
3 | Headless Chromium (runs JavaScript) |
|
4 | Chromium through a proxy | proxies + browser |
5 | Hosted scrape API (Firecrawl / ScrapingBee) |
|
With nothing configured it is the plain HTTP path plus an automatic browser fallback for JavaScript-rendered pages.
Env var | Default | Purpose |
|
| Escalate blocked/JS pages to headless Chromium. |
| (none) | Comma-separated proxy URLs; sticky per domain with health tracking. |
| (none) |
|
|
| Per-domain rate limit (token bucket). |
|
| Per-domain circuit breaker. |
|
| Seconds to remember a blocked URL. |
|
| Below this extracted length, a script-heavy page counts as a JS shell. |
Runtime data
~/.footnote-mcp/source_cache/ # persistent page cache (with provenance)
~/.footnote-mcp/research_memory.json # persistent research memoryOverride the cache location with FOOTNOTE_SOURCE_CACHE=/path/to/cache footnote-mcp.
check_date_completeness supports the calendars calendar, business_day, crypto_24_7,
forex_weekday, us_business_day, and ru_business_day (pass explicit holidays for
source-specific ones; the us_/ru_ variants use the optional holidays package).
Other install paths
Docker bundles Chromium and tesseract — nothing else to install:
docker build -t footnote-mcp .
docker run -i --rm footnote-mcp # the client launches this; see MCP config belowPublished images are available from GitHub Container Registry:
docker run -i --rm ghcr.io/kazkozdev/footnote-mcp:1.1.0
docker run -i --rm ghcr.io/kazkozdev/footnote-mcp:latest{
"mcpServers": {
"footnote": {
"command": "docker",
"args": ["run", "-i", "--rm", "ghcr.io/kazkozdev/footnote-mcp:latest"]
}
}
}pipx / uvx (isolated install of the entry point):
pipx install /path/to/footnote-mcp # or: pipx install git+<repo-url>
uvx --from /path/to/footnote-mcp footnote-mcp # ad-hoc, no installOCR. PDF/image OCR uses pytesseract + the system tesseract binary (brew install tesseract on macOS). Local NLI backend for evidence_entailment backend="local_nli":
pip install -r requirements-nli.txt (model via FOOTNOTE_NLI_MODEL). Either way,
startup_health_check reports what is actually available. Runtime dependency ranges
are declared in pyproject.toml and mirrored in requirements.txt.
Tests
pip install -r requirements-dev.txt
python -m pytest -q # offline unit + smoke tests; no network or keys neededtests/test_mcp_smoke.py launches the server over real MCP stdio and exercises the tools
end to end against a local HTTP fixture; the rest are offline unit tests of the parsers,
fetch ladder, search providers, and dispatch. The live search test is opt-in:
RUN_LIVE_WEB_TESTS=1 python -m pytest -m liveCI runs the same suite (.github/workflows/tests.yml).
License
MIT — see LICENSE.
This server cannot be installed
Maintenance
Resources
Unclaimed servers have limited discoverability.
Looking for Admin?
If you are the server author, to access and configure the admin panel.
Latest Blog Posts
- Why MCP Servers Need Execution Sandboxing (And Why Your Current Stack Isn't Enough)By Om-Shree-0709 on .Agentic AiPrompt InjectionWebAssembly
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/KazKozDev/footnote-mcp'
If you have feedback or need assistance with the MCP directory API, please join our Discord server