Meta Ad Library MCP

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.

Category
Visit Server

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 set RESIDENTIAL_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 uvwhich 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 have eu_total_reach = null (Meta's design).
  • fetch_reach=True clicks every result in one warm session (~3–4s/ad) — great for ≤50; otherwise list fast and pull reach per‑ad with get_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_ads reads the SSR'd first page (embedded JSON) + AdLibrarySearchPaginationQuery captured 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:

  1. HTTP transport — switch mcp.run() to streamable-http in server.py.
  2. A public HTTPS endpoint with auth (claude.ai connectors expect OAuth).
  3. Residential egress — set RESIDENTIAL_PROXY_URL; there is no free way around the datacenter‑IP rd_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

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.

Official
Featured
TypeScript
Magic Component Platform (MCP)

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.

Official
Featured
Local
TypeScript
Audiense Insights MCP Server

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.

Official
Featured
Local
TypeScript
VeyraX MCP

VeyraX MCP

Single MCP tool to connect all your favorite tools: Gmail, Calendar and 40 more.

Official
Featured
Local
graphlit-mcp-server

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.

Official
Featured
TypeScript
Kagi MCP Server

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.

Official
Featured
Python
E2B

E2B

Using MCP to run code via e2b.

Official
Featured
Neon Database

Neon Database

MCP server for interacting with Neon Management API and databases

Official
Featured
Qdrant Server

Qdrant Server

This repository is an example of how to create a MCP server for Qdrant, a vector search engine.

Official
Featured
Exa Search

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.

Official
Featured