LinkedIn Ads MCP — MCP server for the LinkedIn Marketing API. Read + write campaigns, analytics, and creatives from any MCP client. Maintained by Nuraveda Lab.
Project description
LinkedIn Ads MCP
Model Context Protocol (MCP) server for the LinkedIn Marketing API. Read campaigns, pull analytics, create campaign groups + campaigns, flip statuses — all from any MCP client (Claude Desktop, Cursor, Continue, or your own agent).
There's no official LinkedIn MCP. This fills the gap with a thin, correctness-first wrapper that handles LinkedIn's quirky restli encoding rules so you don't have to.
Maintained by Nuraveda Lab as open-source tooling alongside the Mesh Pilot agent suite. MIT licensed.
⚡ Skip the setup — connect LinkedIn through Mesh Pilot
LinkedIn's Marketing API gate is the real hassle: you apply for the Advertising API product, wait for approval (days, not always granted), run an OAuth dance, and manage refresh-token rotation yourself.
Don't want any of that? Mesh Pilot runs this MCP for you behind an already-approved LinkedIn Marketing app. Connect your LinkedIn account in one click and you're driving your ad accounts from your AI client immediately — no API application, no OAuth setup, no token management.
| Self-host (this repo) | Mesh Pilot (hosted) | |
|---|---|---|
| LinkedIn Marketing API approval | you apply + wait | already approved |
| OAuth + token rotation | you manage | handled for you |
| Setup time | hours–days | one click |
| Cost | free (MIT) | see meshpilot.app |
| Runs in your own infra | ✅ | hosted |
Prefer to run it yourself? Keep reading — the full self-host path is below.
Why this exists
If you've tried calling LinkedIn's /rest/adAnalytics endpoint by hand
you've probably hit walls like:
- Commas in
fields=get URL-encoded by default HTTP clients →400 not present in schema - URN colons inside
accounts=List(urn:li:sponsoredAccount:NNN)need to be%3Abut date-tuple colons must stay literal - Partial updates need
X-RestLi-Method: PARTIAL_UPDATEor they get silently ignored runSchedule.startmust be ≥ now-ish,totalBudget.amountmust be ≥ $100- New campaigns need
politicalIntent(LinkedIn's EU political-ad declaration)
This server has all those rules already encoded.
Install
From source (works today):
git clone https://github.com/Nuraveda-Labs/linkedin-ads-mcp.git
cd linkedin-ads-mcp
uv pip install -e . # or: pip install -e .
A PyPI release under the name
linkedin-ads-mcpis planned. The prior package name on PyPI isglitch-grow-linkedin-ad-mcp(legacy identity).
OAuth setup (self-host path)
-
Create a LinkedIn app at https://www.linkedin.com/developers/apps.
-
On the Products tab, request Advertising API (auto-approved if you have an active Campaign Manager account).
-
Run any OAuth flow that grants the scopes
r_ads,rw_ads,r_ads_reporting— for example:https://www.linkedin.com/oauth/v2/authorization?response_type=code &client_id=$YOUR_CLIENT_ID &redirect_uri=$YOUR_REDIRECT_URI &scope=r_ads%20rw_ads%20r_ads_reporting -
Exchange the code for tokens; save the access + refresh tokens.
-
Copy
.env.exampleto.envand paste them.
Run
# stdio (Claude Desktop, Cursor, Continue, etc.)
linkedin-ads-mcp
# SSE on :8000
linkedin-ads-mcp --transport sse --port 8000
Claude Desktop config
{
"mcpServers": {
"linkedin-ads": {
"command": "linkedin-ads-mcp",
"env": {
"LINKEDIN_CLIENT_ID": "...",
"LINKEDIN_CLIENT_SECRET": "...",
"LINKEDIN_REFRESH_TOKEN": "..."
}
}
}
}
Tools
Read
| Tool | What it does |
|---|---|
list_ad_accounts() |
Every ad account the OAuth user can access |
list_account_users(account_id) |
User → role assignments |
list_campaign_groups(account_id) |
Campaign groups + total budgets |
list_campaigns(account_id) |
All campaigns + structure (no metrics) |
list_creatives(account_id) |
Creative roster |
get_account_analytics(account_id, days=14) |
Account-level totals |
get_campaign_analytics(account_id, days=14) |
Per-campaign metrics, sorted by spend |
Write
| Tool | What it does |
|---|---|
create_campaign_group(account_id, name, total_budget=100, days=30, status="DRAFT") |
Create a group |
create_campaign(account_id, name, campaign_group_urn, daily_budget=10, …) |
Create a campaign (defaults to safe DRAFT TEXT_AD) |
update_campaign_status(account_id, campaign_id, status) |
DRAFT / ACTIVE / PAUSED / ARCHIVED |
update_campaign_group_status(account_id, group_id, status) |
Same set + CANCELED |
All write tools default to DRAFT so nothing goes live by accident.
Promote a group → ACTIVE first, then promote campaigns → PAUSED → ACTIVE
in two explicit steps.
Multi-tenant pattern
LinkedIn has no MCC, but Campaign Manager has equivalent "Manage Access" sharing. To run this MCP across multiple advertisers:
- Each client adds your OAuth user as
CAMPAIGN_MANAGERon their ad account (Campaign Manager → Account Settings → Manage Access). - After they accept,
list_ad_accounts()returns their account. - Pass that
account_idto any tool call. One OAuth dance, N advertiser accounts — same model as the Google Ads MCC pattern.
Status
Read API + write API for groups + campaigns are battle-tested in
production. Sponsored-creative creation (image/video upload via
initializeUpload → bind to /rest/creatives → attach to a campaign)
reuses a proven /rest/documents + /rest/posts upload pattern; porting
it to the sponsored-ad surface is on the roadmap. PRs welcome.
License
MIT — see LICENSE.
About
Built and maintained by Nuraveda Lab, open-sourced as part of the Mesh Pilot growth-tooling suite. Hardened against real LinkedIn Marketing API behavior in production. If you hit a restli encoding edge case we missed, open an issue with the offending URL and we'll codify the fix.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file linkedin_ads_mcp-0.2.0.tar.gz.
File metadata
- Download URL: linkedin_ads_mcp-0.2.0.tar.gz
- Upload date:
- Size: 11.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.11.12 {"installer":{"name":"uv","version":"0.11.12","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"25.10","id":"questing","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
5104c1af7ada62705bfbcdaaf0fa5d1ceef636950055ccb04703c12a8522ef1f
|
|
| MD5 |
2d941480aee54421ff2224abf5662143
|
|
| BLAKE2b-256 |
2339dc505062dd75f5be0d08cb7bbe6ae0c7c5e9942bd65816fae3c27eb578c2
|
File details
Details for the file linkedin_ads_mcp-0.2.0-py3-none-any.whl.
File metadata
- Download URL: linkedin_ads_mcp-0.2.0-py3-none-any.whl
- Upload date:
- Size: 11.5 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.11.12 {"installer":{"name":"uv","version":"0.11.12","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"25.10","id":"questing","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
2d0dba6ae317be41718f5375a00e98a9b65f53307e264b9a815b78556373dab5
|
|
| MD5 |
e66874705071da42985f04753e0991a0
|
|
| BLAKE2b-256 |
2979a4b108756c474101684761217c8e34ae5deb7a37f5d109e06e5004cde54f
|