LinkedIn Ads MCP Server
Builds matched LinkedIn audiences from Salesforce data via SOQL queries, enabling targeted campaigns based on CRM segments.
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., "@LinkedIn Ads MCP ServerCreate a draft text ad for my new service targeting CTOs in the tech industry."
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.
Liam
Liam is an ad manager for LinkedIn (LinkedIn Ad Manager). Create campaigns by talking to Claude (MCP) or from a CLI. Built for go-to-market teams who want to spin up many campaigns from a contact list and a brief, then add creative images themselves. Everything is created as a draft, so nothing spends until you explicitly activate it in Campaign Manager.
Unofficial and not affiliated with or endorsed by LinkedIn.
Performance-insight reporting is on the roadmap. This release covers creation, matched audiences (including building one straight from Salesforce), and conversion selection.
Hierarchy mapping (LinkedIn differs from Google/Meta)
Common term | LinkedIn entity | Holds |
Campaign | Campaign Group | status, total/shared budget |
Ad group | Campaign | targeting, budget, bid, schedule, format |
Ad | Creative | the rendered ad ( |
Audience | DMP Segment | attached to a Campaign's targeting |
Related MCP server: LinkedIn Ads MCP Server
Packages
@liads/core— LinkedIn REST client, OAuth, resource modules, CSV + SHA256 hashing, Salesforce reader.@liads/mcp— MCP server (stdio for local; reused by the hosted app). Primary interface.@liads/cli— theliamCLI over the same core, for scripted batch runs.@liads/web— Next.js app that hosts the MCP over HTTP on Vercel.
Prerequisites (everyone needs their own LinkedIn app)
Create a developer app at https://www.linkedin.com/developers/apps and request the Advertising API product. (The Audiences product, needed for CSV audience upload, is requested separately.)
Add an OAuth redirect URL of
http://localhost:53682/callbackfor local login.
Option A — Run locally (self-host, free)
pnpm install
pnpm -r build
# App credentials (alternatively set LIADS_CLIENT_ID / LIADS_CLIENT_SECRET env vars)
mkdir -p ~/.liads
echo '{ "clientId": "...", "clientSecret": "...", "linkedinVersion": "202605" }' > ~/.liads/config.json
node packages/cli/dist/index.js auth login # opens browser, stores tokens in ~/.liads
node packages/cli/dist/index.js accounts list # verify
node packages/cli/dist/index.js launch --brief examples/brief.jsonRegister the local MCP server with Claude (Desktop / Code):
{
"mcpServers": {
"liam": { "command": "node", "args": ["/abs/path/linkedin-ads/packages/mcp/dist/index.js"] }
}
}Option B — Host the MCP on Vercel (single-tenant, for you)
Get your env values locally:
node packages/cli/dist/index.js auth export.Deploy
apps/webto Vercel (set the project Root Directory toapps/web).In Vercel project settings, add the env vars from
.env.example(LIADS_CLIENT_ID,LIADS_CLIENT_SECRET,LIADS_REFRESH_TOKEN,LIADS_LINKEDIN_VERSION, and a strongMCP_AUTH_TOKEN).Your MCP endpoint is
https://<your-app>.vercel.app/api/mcp.
Connect Claude to the remote server (header carries the secret):
npx mcp-remote https://<your-app>.vercel.app/api/mcp --header "Authorization: Bearer <MCP_AUTH_TOKEN>"Tools (MCP)
Accounts:
list_ad_accountsTargeting:
list_targeting_facets,search_targeting(typeahead a facet for entity URNs),list_facet_entities,estimate_audience(structured spec). Talk in plain language ("VPs of demand gen at SaaS companies in the US") and Liam resolves the facets, estimates reach, then builds the campaign.Audiences:
upload_audience_csv,audience_from_salesforce(SOQL → matched audience),get_audience_statusConversions:
list_conversions(select an existing insight-tag conversion to track).create_campaignandlaunch_from_briefacceptconversionIdsorconversionName, and fall back todefaultConversionNamefrom config.Campaigns:
create_campaign_group,create_campaign,create_text_ad,create_image_adOrchestrator:
launch_from_brief(audience + group + campaign + draft creatives in one call)
Targeting spec
Structured targeting uses short facet names mapped to entity URNs (resolve URNs with
search_targeting). URNs within a facet are ORed; facets are ANDed; excluded facets are ORed.
{ "include": { "locations": ["urn:li:geo:103644278"], "seniorities": ["urn:li:seniority:7"], "titles": ["urn:li:title:26587"] },
"exclude": { "industries": ["urn:li:industry:47"] } }CLI reference
All commands also work via node packages/cli/dist/index.js <cmd>.
liam auth login # OAuth, stores tokens in ~/.liads
liam auth export # print env vars for the hosted (Vercel) server
liam accounts list # list accessible ad accounts
liam targeting search <facet> <query> # typeahead a facet for entity URNs
liam targeting estimate <facet> <urns…> # audience size for one facet's URNs
liam audience upload -n <name> -f <csv> # CSV of emails -> matched-audience segment
liam audience from-salesforce -n <name> -q "<SOQL>" # Salesforce query -> matched audience
liam audience status <segmentId> # matching status + resolved size
liam conversions list # account conversions (pick one to track)
liam launch --brief <brief.json> # audience + group + campaign + draft creatives--account defaults to defaultAccountId from config where applicable.
Configuration
Local config lives in ~/.liads/config.json (mode 0600, never in the repo):
Field | Purpose |
| LinkedIn app credentials (or |
| Pinned API version, e.g. |
| Ad account used when a command/brief omits one |
| Conversion auto-selected for new campaigns when none is given |
| Bearer secret for the hosted MCP endpoint |
Hosted equivalents are the LIADS_* env vars plus MCP_AUTH_TOKEN (see .env.example).
OAuth tokens are stored separately in ~/.liads/credentials.json.
Salesforce integration
Liam reads Salesforce by shelling out to the authenticated sf CLI (sf data query), so it
reuses your existing login and needs no new credentials. audience_from_salesforce (and
liam audience from-salesforce) take a SOQL query that selects an email column and turn the
result into a matched audience, closing the loop from "accounts flagged in Salesforce" to
"LinkedIn targeting." Example:
liam audience from-salesforce -n "Q3 target accounts" \
-q "SELECT Email FROM Contact WHERE Account.Target_List__c = true AND Email != null"Safety
Every campaign/creative is created DRAFT/PAUSED. Activation is a separate, explicit step.
Matched-audience matching takes up to 48h, and a campaign needs ~300 matched members to serve.
LinkedIn does not expose person-level ad-view data; cross-referencing is account/segment-level.
Secrets never enter the repo. Local:
~/.liads. Hosted: Vercel env vars.
License
MIT
This server cannot be installed
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
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/stan-default/liam'
If you have feedback or need assistance with the MCP directory API, please join our Discord server