Meta Ad Library MCP
Searches the Meta (Facebook) Ad Library by keyword and returns ad details including advertiser, ad copy, and EU reach, without requiring an API key.
README
Meta Ad Library MCP
An MCP server that searches the Meta (Facebook) Ad Library by keyword and returns each ad's advertiser, full ad copy, and EU reach (per‑country / age / gender breakdown) — the data the official Ad Library API does not expose for commercial ads.
It works by driving the public Ad Library like a real user (Playwright + your real Chrome) and reading the page's own data — no API key, no login required.
search_ads("dental implants", country="DE", limit=50, fetch_reach=True)
→ 50 advertisers + ad copy + EU reach, ranked
Requirements
- A residential internet connection. Meta
rd_challenge‑403s datacenter IPs (AWS/GCP/VPS) within seconds. On a normal home/office machine it just works; on a server you must setRESIDENTIAL_PROXY_URL(see Hosting). - Python 3.10+
- Google Chrome installed (the server launches your real Chrome for a clean
fingerprint). No Chrome? Set
META_ADS_CHANNEL=""to use Playwright's bundled Chromium.
Install (uv — recommended)
git clone https://github.com/konstantin-tradient/meta-ad-library-mcp.git
cd meta-ad-library-mcp
uv sync # creates an isolated venv from pyproject.toml
uv run playwright install chromium
Quick local check (no MCP client needed):
uv run python smoke.py "dental implants" DE # live: search + reach
uv run --extra dev pytest -q # offline parser test
<details><summary>Without uv (plain pip)</summary>
py -m pip install -r requirements.txt && py -m playwright install chromium # Windows
python3 -m pip install -r requirements.txt && python3 -m playwright install chromium
Then run/register with py -m meta_ads_mcp.server + --env PYTHONPATH=<dir>.
</details>
No-clone install (uvx)
To use it on a machine without cloning, uvx fetches + builds straight from GitHub.
Install Chromium once, then register the connector:
# one-time: download the browser into the shared cache
uvx --from playwright playwright install chromium
# Claude Code — pin a tag so deps don't drift
claude mcp add meta-ads --scope user -- \
uvx --from git+https://github.com/konstantin-tradient/meta-ad-library-mcp.git@v0.1.0 meta-ads-mcp
(Claude Desktop: "command": "/full/path/to/uvx", "args": ["--from", "git+https://github.com/konstantin-tradient/meta-ad-library-mcp.git@v0.1.0", "meta-ads-mcp"].)
Use this for machines that only run the tool. To develop it, clone + uv run
below (so your local edits apply and uv.lock pins deps). If you hit a
"browser not found" error, re-run playwright install chromium.
Connect it to Claude (from a clone)
The MCP command is just uv run --directory <clone> meta-ads-mcp — uv auto-syncs the
venv, so there's nothing to install globally.
Claude Code (CLI)
# --scope user = available in every project. Use the clone's ABSOLUTE path.
claude mcp add meta-ads --scope user -- uv run --directory /abs/path/to/meta-ad-library-mcp meta-ads-mcp
Restart Claude Code, then ask: "search the Meta Ad Library for dental implants in Germany."
Claude Desktop (Mac/Windows app)
Edit claude_desktop_config.json
(Windows: %APPDATA%\Claude\, macOS: ~/Library/Application Support/Claude/) and add
under mcpServers (use the full path to uv — which uv / where uv):
"meta-ads": {
"command": "/full/path/to/uv",
"args": ["run", "--directory", "/abs/path/to/meta-ad-library-mcp", "meta-ads-mcp"]
}
Fully quit and reopen Claude Desktop (tray → Quit). The tools appear in the chat.
claude.ai (web) can't run a local server — it only connects to remote MCP servers (public HTTPS URL + auth). See Hosting.
Tools
| Tool | Returns |
|---|---|
search_ads(keyword, country="ALL", limit=20, fetch_reach=False, min_reach=0, stop_after_below=5) |
{count, ads:[{library_id, page_name, status, body_text, cta, link_url, start_date, end_date, versions, eu_total_reach?, uk_total_reach?, reach_breakdown?}], reach_meta?} |
get_ad_details(keyword, library_id, country="ALL") |
{library_id, eu_total_reach, uk_total_reach, gender_audience, age_audience, location_audience, reach_breakdown:[{country, age_gender:[{age_range, male, female, unknown}]}]} |
session_status() |
{ready, egress_ip, proxy, last_challenge} |
- Use an EU country code (
DE,FR,NL…) to surface ads that carry EU‑reach data. Ads not delivered in the EU haveeu_total_reach = null(Meta's design). fetch_reach=Trueclicks every result in one warm session (~3–4s/ad) — great for ≤50; otherwise list fast and pull reach per‑ad withget_ad_details.
Top-ads-by-reach (min_reach)
Meta exposes no reach or impressions in the list and offers no reach sort —
the only number is eu_total_reach, behind a click per ad. But Meta orders the
list by impressions (descending), which is ~monotonic with reach. So:
search_ads("biosila.bg", country="BG", limit=300, fetch_reach=True, min_reach=1000)
→ only ads with EU reach ≥ 1000, sorted reach-desc
It walks Meta's order, clicking for reach, and stops after stop_after_below
consecutive ads fall below min_reach (the rest, lower in the order, are assumed
below too) — so you get the top‑by‑reach set after ~40 clicks instead of 800+.
reach_meta reports {checked, stopped_early}. Raise stop_after_below for
multi‑advertiser keywords where impressions‑order is noisier vs. reach.
How it works
- rd_challenge is a soft gate — the first request 403s but sets a clearance cookie; the server warms the session (homepage → library landing) and retries, which clears it. (This is what made headless work without a paid API.)
- List = the page's own GraphQL, not DOM scraping.
search_adsreads the SSR'd first page (embedded JSON) +AdLibrarySearchPaginationQuerycaptured on scroll, so advertiser + ad copy are reliable for every ad (incl. image/video), and it paginates to 50+. - Reach comes from
AdLibraryV3AdDetailsQuery, which only fires when you click an ad's "See ad details" on the search page — so reach is gathered by clicking, either in batch (fetch_reach=True) or per‑ad (get_ad_details).
Environment variables
| Var | Default | Purpose |
|---|---|---|
RESIDENTIAL_PROXY_URL |
– | http://user:pass@host:port — route Meta traffic through a residential proxy (required on datacenter hosts) |
META_ADS_CHANNEL |
chrome |
Browser channel; set "" to use Playwright's bundled Chromium |
META_ADS_HEADFUL |
– | 1 to show the browser window (default headless) |
META_ADS_PROFILE |
~/.meta-ads-profile |
Persistent browser profile dir (keeps the warm session) |
Hosting / remote (claude.ai)
Runs locally on a residential IP for free — ideal for interactive use. To reach it from claude.ai (web) or run it 24/7 on a server, you need:
- HTTP transport — switch
mcp.run()tostreamable-httpinserver.py. - A public HTTPS endpoint with auth (claude.ai connectors expect OAuth).
- Residential egress — set
RESIDENTIAL_PROXY_URL; there is no free way around the datacenter‑IPrd_challenge.
Legal
This automates access to a public transparency tool for competitive research. It nonetheless falls under Meta's automated‑access terms — use a disposable/burner account if you host it, pace conservatively, and don't hammer. Provided as‑is.
Recommended Servers
playwright-mcp
A Model Context Protocol server that enables LLMs to interact with web pages through structured accessibility snapshots without requiring vision models or screenshots.
Magic Component Platform (MCP)
An AI-powered tool that generates modern UI components from natural language descriptions, integrating with popular IDEs to streamline UI development workflow.
Audiense Insights MCP Server
Enables interaction with Audiense Insights accounts via the Model Context Protocol, facilitating the extraction and analysis of marketing insights and audience data including demographics, behavior, and influencer engagement.
VeyraX MCP
Single MCP tool to connect all your favorite tools: Gmail, Calendar and 40 more.
graphlit-mcp-server
The Model Context Protocol (MCP) Server enables integration between MCP clients and the Graphlit service. Ingest anything from Slack to Gmail to podcast feeds, in addition to web crawling, into a Graphlit project - and then retrieve relevant contents from the MCP client.
Kagi MCP Server
An MCP server that integrates Kagi search capabilities with Claude AI, enabling Claude to perform real-time web searches when answering questions that require up-to-date information.
E2B
Using MCP to run code via e2b.
Neon Database
MCP server for interacting with Neon Management API and databases
Qdrant Server
This repository is an example of how to create a MCP server for Qdrant, a vector search engine.
Exa Search
A Model Context Protocol (MCP) server lets AI assistants like Claude use the Exa AI Search API for web searches. This setup allows AI models to get real-time web information in a safe and controlled way.