Server data from the Official MCP Registry
Human-in-the-loop LinkedIn outreach and a built-in sales CRM for AI agents. Safety-gated, anti-spam.
Human-in-the-loop LinkedIn outreach and a built-in sales CRM for AI agents. Safety-gated, anti-spam.
Remote endpoints: streamable-http: https://app.salesbot.cz/api/mcp
Valid MCP server (1 strong, 1 medium validity signals). No known CVEs in dependencies. Imported from the Official MCP Registry. 1 finding(s) downgraded by scanner intelligence.
Endpoint verified · Open access · 1 issue found
Security scores are indicators to help you make informed decisions, not guarantees. Always review permissions before connecting any MCP server.
This plugin requests these system permissions. Most are normal for its category.
Remote Plugin
No local installation needed. Your AI client connects to the remote endpoint directly.
Add this to your MCP configuration to connect:
{
"mcpServers": {
"cz-salesbot-linkedin-mcp-server": {
"url": "https://app.salesbot.cz/api/mcp"
}
}
}From the project's GitHub README.
What is this?
linkedin-mcp-server-salesbotis a Model Context Protocol (MCP) server for AI‑assisted LinkedIn relationship operations. It lets AI assistants — Claude Desktop, ChatGPT, Cursor — help you research and organize professional contacts, draft deeply personalized messages for your review and approval, sync inbox conversations, enrich profiles, and pull web context — all under your direction. It's built for hyper‑targeted, meaningful outreach (find 5 ideal contacts, read their recent posts, write 5 thoughtful notes), not bulk blasting. Every send is gated by human‑in‑the‑loop approval and enforced, server‑side daily/hourly safety thresholds that keep your LinkedIn account within safe limits.
It runs as a Supabase Edge Function (Deno + Hono + mcp-lite) exposing the MCP Streamable HTTP transport. LinkedIn actions go through a third‑party LinkedIn integration provider; LinkedIn credentials are never stored by the AI.
| Endpoint | https://app.salesbot.cz/api/mcp |
| Transport | MCP Streamable HTTP (POST + SSE) |
| Auth header | x-mcp-api-key: sb_mcp_… (a Supabase JWT in Authorization also works) |
| Tool count | 42 |
| License | MIT |
Add this to your MCP client config. Get the sb_mcp_… key in the Salesbot app under Settings → MCP.
{
"mcpServers": {
"linkedin-automation": {
"url": "https://app.salesbot.cz/api/mcp",
"headers": {
"x-mcp-api-key": "sb_mcp_YOUR_API_KEY",
"Accept": "application/json, text/event-stream"
}
}
}
}
Important: send the key in the
x-mcp-api-keyheader, notAuthorization: Bearer. The Supabase API gateway rejects unknown Bearer tokens before they reach the server.
sb_mcp_…) — long‑lived; generated in the Salesbot app, stored only as a SHA‑256 hash. Send in x-mcp-api-key.Authorization: Bearer.The AI can do it without leaving the chat:
get_linkedin_status — reports whether LinkedIn is connected/active/blocked.connect_linkedin — returns a white‑labeled https://auth.salesbot.cz/… link. The user opens it, completes LinkedIn login, done.Or connect in the app: Settings → LinkedIn → Connect.
Each tool returns text content; errors return { "ok": false, "code": "<CODE>", "error": "<message>" }.
{ "name": "get_linkedin_status", "input": { "profile_id": "uuid (optional)" } }
{ "name": "connect_linkedin", "input": { "profile_id": "uuid (optional)", "reconnect": "boolean (optional)" } }
{ "name": "search_linkedin_people", "input": { "title": "string (required)", "location": "string", "locationId": "string", "network": "['S'|'O']", "limit": "number 1-50" } }
{ "name": "search_google_xray", "input": { "jobTitle": "string (required)", "location": "string", "keywords": "string[]", "excludeWords": "string[]", "limit": "number 1-100" } }
{ "name": "search_linkedin_navigator", "input": { "search_url": "string (required)", "limit": "number 1-100" } }
{ "name": "scrape_website", "input": { "url": "string (required)", "max_chars": "number (default 8000, max 20000)" } }
{ "name": "get_contact_profile", "input": { "contact_id": "uuid (required)" } }
{ "name": "list_contacts", "input": { "list_id": "uuid (required)", "limit": "number", "offset": "number" } }
{ "name": "list_campaigns", "input": { "status": "draft|running|paused|completed|stopped (optional)" } }
{ "name": "create_campaign", "input": { "name": "string (required)", "profile_id": "uuid (required)", "description": "string", "daily_limit": "number", "sender_context": "string", "steps": "[{ action: 'connect'|'message'|'visit', delay_hours, use_ai, ai_prompt, ai_template, send_without_message }] (required)" } }
{ "name": "update_campaign_settings", "input": { "campaign_id": "uuid (required)", "name": "string", "description": "string", "daily_limit": "number", "sender_context": "string", "auto_approve_messages": "boolean", "status": "running|paused|draft|stopped" } }
{ "name": "start_campaign", "input": { "campaign_id": "uuid (required)" } }
{ "name": "stop_campaign", "input": { "campaign_id": "uuid (required)" } }
{ "name": "add_contacts_to_campaign", "input": { "campaign_id": "uuid (required)", "contact_ids": "uuid[] (required)" } }
{ "name": "generate_campaign_message", "input": { "campaign_contact_id": "uuid (required)", "step_id": "uuid (required)", "custom_instructions": "string" } }
{ "name": "list_pending_approvals", "input": { "campaign_id": "uuid", "limit": "number" } }
{ "name": "approve_message", "input": { "campaign_contact_id": "uuid (required)", "edited_messages": "[{step_id, message}]", "skip_gpt_check": "boolean" } }
{ "name": "reject_message", "input": { "campaign_contact_id": "uuid (required)", "reason": "string (required)" } }
{ "name": "send_connection_request", "input": { "linkedin_id": "string (required)", "profile_id": "uuid (required)", "contact_id": "uuid" } }
{ "name": "send_linkedin_message", "input": { "linkedin_id": "string (required)", "message": "string ≤5000 (required)", "profile_id": "uuid (required)" } }
{ "name": "publish_linkedin_post", "input": { "profile_id": "uuid (required)", "text": "string ≤3000 (required)", "external_link": "string", "as_organization": "string", "auto_publish": "boolean" } }
{ "name": "get_daily_limits", "input": { "profile_id": "uuid (optional)" } }
{ "name": "list_inbox_chats", "input": { "profile_id": "uuid (optional)", "limit": "number 1-50", "cursor": "string" } }
{ "name": "get_chat_messages", "input": { "chat_id": "string (required)", "profile_id": "uuid (optional)", "limit": "number 1-50", "cursor": "string" } }
{ "name": "reply_to_chat", "input": { "chat_id": "string (required)", "message": "string ≤5000 (required)", "profile_id": "uuid (optional)" } }
{ "name": "mark_chat_read", "input": { "chat_id": "string (required)", "profile_id": "uuid (optional)" } }
The CRM is a persistent pipeline separate from contacts. A lead enters it when added to a campaign, or when any of these tools first touch it.
{ "name": "set_deal_stage", "input": { "contact_id": "uuid (required)", "stage": "string (required)", "note": "string" } }
{ "name": "log_crm_note", "input": { "contact_id": "uuid (required)", "summary": "string (required)", "pain_points": "string[]", "sentiment": "positive|neutral|negative" } }
{ "name": "create_task", "input": { "title": "string (required)", "contact_id": "uuid", "due_at": "ISO 8601", "details": "string" } }
{ "name": "list_tasks", "input": { "status": "open|done|cancelled|all", "contact_id": "uuid", "limit": "number" } }
{ "name": "complete_task", "input": { "task_id": "uuid (required)", "status": "done|open|cancelled" } }
{ "name": "get_lead_context", "input": { "contact_id": "uuid (required)", "notes_limit": "number" } }
{ "name": "update_contact", "input": { "contact_id": "uuid (required)", "email": "string", "phone": "string", "location": "string", "company": "string", "position": "string", "headline": "string" } }
{ "name": "set_lead_fields", "input": { "contact_id": "uuid (required)", "fields": "object { field_key: value }" } }
{ "name": "export_crm", "input": { "limit": "number (default 5000, max 20000)" } }
Pipeline stages and custom fields are user-configurable.
{ "name": "list_crm_stages", "input": {} }
{ "name": "add_crm_stage", "input": { "label": "string (required)", "color": "hex string" } }
{ "name": "rename_crm_stage", "input": { "key": "string (required)", "label": "string", "color": "hex string" } }
{ "name": "delete_crm_stage", "input": { "key": "string (required)", "reassign_to": "string" } }
{ "name": "list_crm_fields", "input": {} }
{ "name": "add_crm_field", "input": { "label": "string (required)", "type": "text|number|date|url" } }
{ "name": "delete_crm_field", "input": { "key": "string (required)" } }
Request (MCP tools/call):
{ "jsonrpc": "2.0", "id": 1, "method": "tools/call",
"params": { "name": "get_daily_limits", "arguments": {} } }
Success result content (JSON inside the text part):
{ "profile_active": true,
"limits": { "connections": { "used": 0, "limit": 30, "effective_limit": 30 },
"messages": { "used": 0, "limit": 40, "effective_limit": 40 } } }
Error result content:
{ "ok": false, "code": "ACCOUNT_NOT_CONNECTED", "error": "Profile has no connected LinkedIn account." }
| Code | Meaning |
|---|---|
AUTH_MISSING / AUTH_INVALID / AUTH_EXPIRED | missing / wrong / expired key |
SUBSCRIPTION_REQUIRED | trial expired or no active plan |
RATE_LIMITED | too many MCP requests — slow down |
ACCOUNT_NOT_CONNECTED | profile has no connected LinkedIn (call connect_linkedin) |
ACCOUNT_BLOCKED | LinkedIn restricted the account (campaigns auto‑paused) |
PROFILE_INACTIVE / PROFILE_NOT_FOUND / ACCESS_DENIED | profile / ownership |
DAILY_LIMIT_REACHED / HOURLY_LIMIT_REACHED | quota reached |
OUTSIDE_ALLOWED_HOURS | outside the account's sending window |
BLACKLISTED | target company/domain blacklisted |
APPROVAL_REQUIRED | queued for human approval before sending |
SAFETY_BLOCKED | text looks like prompt‑injection / unrequested URL |
REPLY_LIMIT_REACHED | already 2 AI replies in this conversation |
SCRAPE_LIMIT_REACHED | weekly web‑scrape quota reached |
VALIDATION_ERROR / NOT_FOUND / UPSTREAM_ERROR | bad input / not found / upstream failure |
Built-in LinkedIn algorithmic protection and daily safety thresholds. This is a relationship tool, not a mass-mailer — it's designed to send a few highly personalized, human-approved messages, and the server actively prevents bulk abuse:
Which AI clients work? Any MCP Streamable‑HTTP client — Claude Desktop, the Claude API, Cursor, and similar.
Why x-mcp-api-key and not Authorization? The Supabase gateway validates Authorization bearer tokens and rejects unknown ones; the custom header passes through untouched.
Does the AI see my LinkedIn password? No. Authentication happens through a hosted provider flow (white‑labeled at auth.salesbot.cz); the MCP server only uses an account handle.
Can the AI send messages without me? Only if you disable approval. By default outbound actions are queued for human approval.
Is it safe for my LinkedIn account? Daily/hourly limits, ramp‑up, allowed‑hours, randomized delays, and auto‑pause on a detected block are all enforced server‑side.
Runs on the Salesbot Supabase backend. With the Supabase CLI:
supabase functions deploy mcp-server --no-verify-jwt --project-ref <your-project-ref>
Required function secrets: SUPABASE_URL, SUPABASE_SERVICE_ROLE_KEY, SUPABASE_ANON_KEY, the LinkedIn‑provider credentials, CRON_SECRET, APP_URL. The server does its own auth, hence --no-verify-jwt.
MIT — see LICENSE.
Be the first to review this server!
by Toleno · Developer Tools
Toleno Network MCP Server — Manage your Toleno mining account with Claude AI using natural language.
by mcp-marketplace · Developer Tools
Create, build, and publish Python MCP servers to PyPI — conversationally.
by Microsoft · Content & Media
Convert files (PDF, Word, Excel, images, audio) to Markdown for LLM consumption