meta-ads-mcp-server
v1.5.1
Published
MCP server for Meta (Facebook) Ads API integration
Readme
Meta Ads MCP Server
Disclaimer: This is an unofficial third-party tool and is not associated with, endorsed by, or affiliated with Meta in any way. This project is maintained independently and uses Meta's public APIs in accordance with their Terms of Service. Meta, Facebook, Instagram, and other Meta brand names are trademarks of their respective owners.
Table of Contents
- Features
- Requirements
- Installation
- Obtaining a Meta Access Token
- Authentication
- Enabling Write Tools
- Transport Modes
- Cursor / Claude Desktop Setup
- Remote HTTP Server
- Available Tools
- End-to-End: Create an Ad from Scratch
- Pagination
- Development
- Project Structure
- License
Features
| Category | What it does | |----------|--------------| | Accounts | List ad accounts, get account details | | Campaigns | Get / list / create / update / pause / resume / delete | | Ad Sets | Get / list / batch-fetch / create / update / pause / resume / delete | | Ads | Get / list / create / update / pause / resume / delete | | Creatives | Get / list, create + update, compute image crops | | Media | List ad images, upload images, lookup by hash, get ad previews and videos | | Insights | Performance analytics at account, campaign, ad set, and ad level | | Targeting | Search interests / behaviors / demographics / geo, audience size estimation | | Pages | List Facebook Pages reachable from the token, search by name | | Budget Schedules | Schedule temporary budget bumps over a time window | | Activities | Change history log for ad accounts and ad sets | | Pagination | Utility tool to fetch subsequent pages of results |
All mutation tools (create / update / delete / pause / resume / upload / budget schedule) are off by default and only register when you opt in — see Enabling Write Tools.
Requirements
- Node.js >= 18
- A Meta User Access Token with the right permissions for what you plan to do — see below.
Installation
# From npm
npx meta-ads-mcp-server --access-token YOUR_META_ACCESS_TOKEN
# From source
git clone https://github.com/hashcott/meta-ads-mcp.git
cd meta-ads-mcp
npm install
npm run build
node dist/index.js --access-token YOUR_META_ACCESS_TOKENObtaining a Meta Access Token
This server uses the Meta Marketing API. You need an access token attached to a Meta App that has the right permissions.
Quick option — Graph API Explorer (read-only experiments)
- Open the Graph API Explorer.
- Pick your Meta App from the top-right dropdown (create one at developers.facebook.com/apps if you don't have any — choose type "Business").
- Click Generate Access Token, then under Permissions add at minimum:
ads_read— for all the read tools.ads_management— required for any write tool (create / update / delete / pause / resume / upload / budget schedule).business_management— recommended if you operate via Business Manager.pages_show_list,pages_read_engagement— required for the Pages tools.
- Copy the generated token. This is a short-lived token (~1 hour) — fine for testing.
Production option — long-lived User token
Short-lived tokens from the Explorer expire in about an hour. Exchange yours for a 60-day token:
curl -G "https://graph.facebook.com/v22.0/oauth/access_token" \
--data-urlencode "grant_type=fb_exchange_token" \
--data-urlencode "client_id=YOUR_APP_ID" \
--data-urlencode "client_secret=YOUR_APP_SECRET" \
--data-urlencode "fb_exchange_token=YOUR_SHORT_LIVED_TOKEN"Response contains "access_token": "..." — that token is valid for ~60 days. Refresh it the same way before it expires, or build a full OAuth flow if you need permanent access.
Production option — System User token (recommended for servers)
For unattended production use (no expiry), generate a System User token in Business Manager:
- Go to Business Manager → Business Settings → Users → System Users.
- Create a system user (or use an existing one), assign the relevant ad account, and grant
ads_read/ads_management. - Click Generate New Token → pick your Meta App → select the same permissions → Never for expiration.
System User tokens don't expire and are ideal for backend deployments.
Verifying your token
curl "https://graph.facebook.com/v22.0/me?access_token=YOUR_TOKEN"Should return your user/system-user object. If it returns an error, double-check the permissions and that the token isn't expired.
Authentication
Pass your Meta access token using either method:
CLI argument (recommended for Cursor / Claude Desktop):
node dist/index.js --access-token YOUR_META_ACCESS_TOKENEnvironment variable:
export META_ADS_ACCESS_TOKEN=YOUR_META_ACCESS_TOKEN
node dist/index.jsThe token is held only in memory of the running process — it is never written to disk by this server.
Enabling Write Tools
By default the server registers only the 35 read tools — create / update / delete / pause / resume / upload / budget-schedule tools are not exposed. This is intentional: a mistakenly-issued meta_ads_delete_campaign can permanently remove campaigns and their ads.
To opt in, set:
META_ADS_ENABLE_WRITE_TOOLS=trueAccepted truthy values: true, 1, yes, on (case-insensitive). Anything else (or unset) keeps writes off.
When enabled, the server logs a one-line warning to stderr at startup:
[meta-ads-mcp] WARNING: META_ADS_ENABLE_WRITE_TOOLS is on — create/update/delete/pause/resume tools are EXPOSED. These can permanently delete campaigns/ad sets/ads or change live delivery.Your access token also needs the ads_management permission for the writes to succeed.
Example Cursor / Claude Desktop configuration with writes enabled:
{
"mcpServers": {
"meta-ads": {
"command": "npx",
"args": ["-y", "meta-ads-mcp-server"],
"env": {
"META_ADS_ACCESS_TOKEN": "YOUR_META_ACCESS_TOKEN",
"META_ADS_ENABLE_WRITE_TOOLS": "true"
}
}
}
}Transport Modes
| Mode | Use case | How to enable |
|------|----------|---------------|
| stdio (default) | Cursor, Claude Desktop, local tools | No configuration needed |
| http | Claude.ai remote connectors, multi-client setups | Set TRANSPORT=http |
Cursor / Claude Desktop Setup
Add one of the following to your MCP client configuration file:
Via npx (recommended — no local install required):
{
"mcpServers": {
"meta-ads": {
"command": "npx",
"args": ["-y", "meta-ads-mcp-server", "--access-token", "YOUR_META_ACCESS_TOKEN"]
}
}
}Via local build:
{
"mcpServers": {
"meta-ads": {
"command": "node",
"args": ["/path/to/meta-ads-mcp/dist/index.js", "--access-token", "YOUR_META_ACCESS_TOKEN"]
}
}
}Via environment variable (and opt-in writes):
{
"mcpServers": {
"meta-ads": {
"command": "npx",
"args": ["-y", "meta-ads-mcp-server"],
"env": {
"META_ADS_ACCESS_TOKEN": "YOUR_META_ACCESS_TOKEN",
"META_ADS_ENABLE_WRITE_TOOLS": "true"
}
}
}
}Remote HTTP Server
Run as a persistent HTTP server for use with Claude.ai custom connectors or any remote MCP client.
# Start on default port 3000
TRANSPORT=http META_ADS_ACCESS_TOKEN=YOUR_TOKEN node dist/index.js
# Start on a custom port, writes enabled
TRANSPORT=http \
META_ADS_ACCESS_TOKEN=YOUR_TOKEN \
META_ADS_ENABLE_WRITE_TOOLS=true \
PORT=8080 \
node dist/index.jsEndpoints:
POST /mcp— MCP protocol endpointGET /health— Health check ({"status":"ok"})
Adding to Claude.ai
- Go to Settings → Connectors → Add custom connector
- Enter your server URL:
https://your-domain.com/mcp - Click Add
Local testing with ngrok
# Terminal 1 — start the server
TRANSPORT=http META_ADS_ACCESS_TOKEN=YOUR_TOKEN PORT=8080 node dist/index.js
# Terminal 2 — expose publicly
ngrok http 8080Use the generated HTTPS URL (e.g. https://xxxx.ngrok-free.app/mcp) as your connector URL.
Deploying to cloud platforms
Set the following environment variables on your hosting provider (Railway, Render, Fly.io, etc.):
| Variable | Value |
|----------|-------|
| TRANSPORT | http |
| META_ADS_ACCESS_TOKEN | Your Meta access token |
| META_ADS_ENABLE_WRITE_TOOLS | true to also expose mutating tools (off by default) |
| PORT | Assigned automatically by the platform |
Available Tools
Legend: 🔍 read • ✏️ write (gated by META_ADS_ENABLE_WRITE_TOOLS) • 🛠️ pure utility (no API call).
Accounts
| Tool | Type | Description |
|------|------|-------------|
| meta_ads_list_ad_accounts | 🔍 | List all ad accounts accessible with your token |
| meta_ads_get_ad_account_details | 🔍 | Get detailed information for a specific ad account |
Campaigns
| Tool | Type | Description |
|------|------|-------------|
| meta_ads_get_campaign_by_id | 🔍 | Fetch a specific campaign by its ID |
| meta_ads_get_campaigns_by_adaccount | 🔍 | List campaigns within an ad account, with filters and pagination |
| meta_ads_create_campaign | ✏️ | Create a new ODAX campaign (CBO or ABO) |
| meta_ads_update_campaign | ✏️ | Update name/status/budget/bid; supports CBO → ABO migration via adset_budgets |
| meta_ads_delete_campaign | ✏️ | Permanently delete a campaign and its ad sets/ads |
| meta_ads_pause_campaign | ✏️ | Convenience: set status to PAUSED |
| meta_ads_resume_campaign | ✏️ | Convenience: set status to ACTIVE |
meta_ads_create_campaign inputs:
act_id(string) — Ad account ID, formatact_XXXXXXXXX.name(string) — Campaign name.objective(enum) — ODAX outcome-based objective:OUTCOME_AWARENESS,OUTCOME_TRAFFIC,OUTCOME_ENGAGEMENT,OUTCOME_LEADS,OUTCOME_SALES,OUTCOME_APP_PROMOTION.- Legacy objectives (
BRAND_AWARENESS,LINK_CLICKS,CONVERSIONS,APP_INSTALLS, …) are not accepted by Meta v22+ and will return HTTP 400.
status(defaultPAUSED),special_ad_categories(default[])daily_budget/lifetime_budget(cents) — omit both whenuse_adset_level_budgets=true.bid_strategy—LOWEST_COST_WITHOUT_CAP(default),LOWEST_COST_WITH_BID_CAP,COST_CAP,LOWEST_COST_WITH_MIN_ROAS. Bid-cap strategies requirebid_amounton every child ad set.bid_cap,spend_cap,campaign_budget_optimization,use_adset_level_budgets,ab_test_control_setups,buying_type.
{
"act_id": "act_123456789012345",
"name": "2026 - Spring Sale - Awareness",
"objective": "OUTCOME_AWARENESS",
"special_ad_categories": [],
"status": "PAUSED",
"bid_strategy": "LOWEST_COST_WITHOUT_CAP",
"daily_budget": 10000
}Ad Sets
| Tool | Type | Description |
|------|------|-------------|
| meta_ads_get_adset_by_id | 🔍 | Fetch a single ad set by its ID |
| meta_ads_get_adsets_by_ids | 🔍 | Batch fetch multiple ad sets |
| meta_ads_get_adsets_by_adaccount | 🔍 | List ad sets in an ad account |
| meta_ads_get_adsets_by_campaign | 🔍 | List ad sets within a campaign |
| meta_ads_create_adset | ✏️ | Create a new ad set under a campaign |
| meta_ads_update_adset | ✏️ | Update an ad set's fields (note: frequency_control_specs is immutable after creation) |
| meta_ads_delete_adset | ✏️ | Permanently delete an ad set |
| meta_ads_pause_adset | ✏️ | Set status to PAUSED |
| meta_ads_resume_adset | ✏️ | Set status to ACTIVE |
meta_ads_create_adset highlights:
- Required:
act_id,campaign_id,name,optimization_goal,billing_event. targeting(object) — full targeting spec; remembertargeting_automation.advantage_audiencedefaults to0on Meta v24+ — set it explicitly if you want Advantage+ Audience.bid_amount— required forLOWEST_COST_WITH_BID_CAP/COST_CAP.bid_constraints— required forLOWEST_COST_WITH_MIN_ROAS, e.g.,{"roas_average_floor": 20000}for a 2.0× ROAS floor.dsa_beneficiary/dsa_payor— required for EU-targeted ad sets.promoted_object— required forAPP_INSTALLS.frequency_control_specs— MUST be set at creation; Meta makes it immutable afterward.regional_regulated_categories/regional_regulation_identities— Taiwan / Australia / Singapore / India regulated verticals.
Ads
| Tool | Type | Description |
|------|------|-------------|
| meta_ads_get_ad_by_id | 🔍 | Fetch a single ad by ID |
| meta_ads_get_ads_by_adaccount | 🔍 | List ads in an ad account |
| meta_ads_get_ads_by_campaign | 🔍 | List ads within a campaign |
| meta_ads_get_ads_by_adset | 🔍 | List ads within an ad set |
| meta_ads_create_ad | ✏️ | Create a new ad referencing an existing creative |
| meta_ads_update_ad | ✏️ | Update name / status / bid / tracking specs / creative reference |
| meta_ads_delete_ad | ✏️ | Permanently delete an ad |
| meta_ads_pause_ad | ✏️ | Set status to PAUSED |
| meta_ads_resume_ad | ✏️ | Set status to ACTIVE |
ℹ️ Swapping
creative_idon a FLEX ad can fail witherror_subcode 3858355if the new creative'sasset_feed_specimages don't match itsobject_story_spec. In that case, create a new ad with the new creative and pause the old one (you lose social proof but the ad runs).
Creatives
| Tool | Type | Description |
|------|------|-------------|
| meta_ads_get_ad_creative_by_id | 🔍 | Fetch one creative |
| meta_ads_get_ad_creatives_by_ad_id | 🔍 | List creatives attached to an ad |
| meta_ads_get_adcreatives_by_adaccount | 🔍 | List creatives in an ad account |
| meta_ads_compute_image_crops | 🛠️ | Compute centered crop boxes for the 6 Meta-accepted aspect ratios (no API call) |
| meta_ads_create_ad_creative | ✏️ | Create a creative — 3 simple modes plus full object_story_spec escape hatch |
| meta_ads_update_ad_creative | ✏️ | Update name / asset_feed_spec (Meta restricts most other content updates) |
meta_ads_create_ad_creative — three common modes:
- Promote an existing post: pass only
object_story_idin the form{page_id}_{post_id}. - Single-image link ad:
page_id+image_hash+link_url+message+ optionalheadline,description,call_to_action_type. - Single-video ad:
page_id+video_id+link_url+message+ optionalheadline,call_to_action_type,thumbnail_url.
For advanced layouts (FLEX/DOF, Placement Asset Customization, Dynamic Creative, multi-headline, lead-gen forms, branded content, image crops), pass a fully composed object_story_spec and/or asset_feed_spec — those take precedence over the simple-mode auto-construction.
Media
| Tool | Type | Description |
|------|------|-------------|
| meta_ads_get_ad_images | 🔍 | List image assets in an ad account |
| meta_ads_get_image_by_hash | 🔍 | Single-image lookup by hash (URL + dimensions) |
| meta_ads_get_ad_previews | 🔍 | Generate rendered previews of an ad across placements |
| meta_ads_get_ad_video | 🔍 | Video details (source URL, thumbnails, length) by ad_id or video_id |
| meta_ads_upload_ad_image | ✏️ | Upload an image to an account's ad images library and get back its image_hash |
meta_ads_upload_ad_image accepts exactly one of:
file— a data URL (data:image/png;base64,iVBORw0KG...) or a raw base64 string.image_url— a public URL; the server downloads the bytes and uploads them.
Returns the image_hash you then pass to meta_ads_create_ad_creative.
Insights
| Tool | Type | Description |
|------|------|-------------|
| meta_ads_get_adaccount_insights | 🔍 | Performance metrics at the account level |
| meta_ads_get_campaign_insights | 🔍 | Performance metrics for a specific campaign |
| meta_ads_get_adset_insights | 🔍 | Performance metrics for a specific ad set |
| meta_ads_get_ad_insights | 🔍 | Performance metrics for a specific ad |
All four accept the same option surface: fields, date_preset, time_range, time_ranges, time_increment, level, action_attribution_windows, action_breakdowns, breakdowns, filtering, sort, pagination, and locale.
Time-range precedence: time_ranges > time_range > since/until > date_preset.
Targeting Catalog
| Tool | Type | Description |
|------|------|-------------|
| meta_ads_search_interests | 🔍 | Search Meta's interest catalog by keyword |
| meta_ads_get_interest_suggestions | 🔍 | Get related interests from a seed list |
| meta_ads_search_behaviors | 🔍 | List available behavior targeting options |
| meta_ads_search_demographics | 🔍 | List demographic options (demographics / life_events / industries / income / family_statuses / user_device / user_os) |
| meta_ads_search_geo_locations | 🔍 | Search countries / regions / cities / zips / geo_markets / electoral_districts |
| meta_ads_estimate_audience_size | 🔍 | Estimate reach for a targeting spec via /act_X/delivery_estimate |
Pages
| Tool | Type | Description |
|------|------|-------------|
| meta_ads_get_account_pages | 🔍 | List Facebook Pages reachable from the access token (/me/accounts) |
| meta_ads_search_pages_by_name | 🔍 | Substring filter over the token's pages (client-side — Meta does not expose a server-side name filter) |
The returned page_id values are the ones you pass to meta_ads_create_ad_creative.
Budget Schedules
| Tool | Type | Description |
|------|------|-------------|
| meta_ads_create_budget_schedule | ✏️ | Schedule a temporary budget bump for a campaign over a Unix-timestamp window |
Inputs:
campaign_id(string)budget_value(int, positive)budget_value_type—ABSOLUTE(cents in account currency) orMULTIPLIER(e.g.,2doubles the budget)time_start,time_end(Unix timestamps in seconds) —time_end > time_start
Activities
| Tool | Type | Description |
|------|------|-------------|
| meta_ads_get_activities_by_adaccount | 🔍 | Retrieve the change history log for an ad account |
| meta_ads_get_activities_by_adset | 🔍 | Retrieve the change history log for an ad set |
Pagination tool
| Tool | Type | Description |
|------|------|-------------|
| meta_ads_fetch_pagination_url | 🛠️ | Follow paging.next / paging.previous URLs from any other tool's response |
End-to-End: Create an Ad from Scratch
A typical "build a new ad" workflow uses tools across several categories. With META_ADS_ENABLE_WRITE_TOOLS=true:
1. meta_ads_list_ad_accounts → pick an act_id
2. meta_ads_get_account_pages → pick a page_id
3. meta_ads_search_geo_locations(q="Vietnam") → grab the country/region keys
4. meta_ads_search_interests(q="cooking") → grab interest IDs
5. meta_ads_estimate_audience_size(act_id, targeting) → sanity-check reach
6. meta_ads_upload_ad_image(act_id, image_url) → returns image_hash
7. meta_ads_create_campaign(act_id, ...) → returns campaign_id
8. meta_ads_create_adset(act_id, campaign_id, targeting, ...) → returns adset_id
9. meta_ads_create_ad_creative(act_id, page_id, image_hash, link_url, message, ...) → returns creative_id
10. meta_ads_create_ad(act_id, name, adset_id, creative_id, status="PAUSED") → returns ad_id
11. (Optional) meta_ads_get_ad_previews(ad_id, ...) → render placements before going live
12. meta_ads_resume_ad(ad_id) → flip to ACTIVE when readyAll steps that mutate state default to status: "PAUSED" so nothing goes live until you explicitly call a resume tool.
Pagination
Many list tools return paginated results. When a response contains a paging.next URL, use meta_ads_fetch_pagination_url to retrieve subsequent pages:
1. Call meta_ads_get_campaigns_by_adaccount → receive first page
2. Check if response.paging.next exists
3. Call meta_ads_fetch_pagination_url(url=response.paging.next) → receive next page
4. Repeat until paging.next is absentDevelopment
npm run dev # Watch mode — auto-recompile on change
npm run build # Compile TypeScript to dist/
npm run clean # Remove dist/
npm run clean && npm run build # Full rebuild from scratchQuick smoke test:
# Default (read-only)
META_ADS_ACCESS_TOKEN=dummy node dist/index.js
# → "Meta Ads MCP server running via stdio"
# With writes enabled
META_ADS_ACCESS_TOKEN=dummy META_ADS_ENABLE_WRITE_TOOLS=true node dist/index.js
# → WARNING line + "Meta Ads MCP server running via stdio"Project Structure
meta-ads-mcp/
├── src/
│ ├── index.ts # Entry point, server setup, transport selection, write-tools warning
│ ├── constants.ts # API version, base URLs, isWriteToolsEnabled() flag
│ ├── types.ts # Shared TypeScript interfaces
│ ├── services/
│ │ └── graph-api.ts # HTTP client (GET/POST/DELETE), auth, error handling, param builders
│ ├── schemas/
│ │ ├── common.ts # Shared Zod schemas (pagination, date ranges, filters)
│ │ └── insights.ts # Insights-specific Zod schemas
│ └── tools/
│ ├── accounts.ts # Account tools
│ ├── insights.ts # Insights tools (account/campaign/adset/ad level)
│ ├── campaigns.ts # Campaign read + write/lifecycle tools
│ ├── adsets.ts # Ad set read + write/lifecycle tools
│ ├── ads.ts # Ad read + write/lifecycle tools
│ ├── creatives.ts # Creative read tools, image crops utility, create/update creative
│ ├── media.ts # Image list / upload / hash lookup / video / preview
│ ├── activities.ts # Activity log tools
│ ├── pagination.ts # Pagination utility tool
│ ├── targeting.ts # Interest/behavior/demographic/geo search + audience-size estimate
│ ├── pages.ts # Facebook Pages list and name search
│ └── budget-schedules.ts # Campaign budget schedule create
├── dist/ # Compiled JavaScript output (generated)
├── package.json
└── tsconfig.json