LinkedIn automation that doesn't get you banned.
CLI Β· MCP Server Β· TypeScript API β one runtime, three surfaces.
Install Β· MCP Setup Β· Features Β· TypeScript API Β· Docs
Scraping with Python scripts that break every Tuesday. Rate-limited by APIs that don't actually exist. Copy-pasting from the web UI like it's 2015. Running Chrome extensions that sell your session cookies to whoever's buying.
Stop.
LinkedIn Buddy is a local-first Playwright runtime that operates LinkedIn the way you would β just faster, safer, and without the RSI. Anti-bot evasion, persistent browser profiles, and a two-phase commit system that previews every single write before it fires. Nothing touches LinkedIn until you say so.
- Three surfaces. One runtime. CLI for operators, MCP server for AI agents, TypeScript API for builders. Same services, same safety guarantees, zero duplication.
- Every write previews first. Two-phase commit. Prepare, inspect, confirm. No accidental DMs. No "oops, wrong connection request."
- The chameleon blends in. Human-like typing, Poisson-distributed pauses, BΓ©zier mouse paths. LinkedIn sees a person, not a bot.
- Your machine. Your data. SQLite state, persistent profiles, structured logs, screenshots. Nothing leaves your laptop. No cloud. No telemetry. No accounts.
- 100+ MCP tools. Claude Desktop, Cursor, Cline β your agent gets full LinkedIn access through structured tool calls, not brittle prompt hacks.
You're a developer. You've done this before.
git clone https://github.com/sigvardt/linkedin-buddy.git
cd linkedin-buddy
npm install
npx playwright install chromium
npm run buildAuthenticate and verify:
npm exec -w @linkedin-buddy/cli -- linkedin login --profile default
npm exec -w @linkedin-buddy/cli -- linkedin status --profile defaultRun your first search:
npm exec -w @linkedin-buddy/cli -- linkedin search "developer relations" --category people --limit 5That's it. You're in.
Tip: The CLI installs three equivalent binaries β
lbud, andlinkedin-buddy. After a global install (once published), drop thenpm execprefix entirely:lbud search "developer relations" --category people --limit 5
Prefer manual encrypted session capture over browser-based login?
npm exec -w @linkedin-buddy/cli -- linkedin auth session --session defaultPaste this into your MCP client config. Done.
{
"mcpServers": {
"linkedin": {
"command": "npm",
"args": ["exec", "-w", "@linkedin-buddy/mcp", "--", "linkedin-mcp"]
}
}
}Works with Claude Desktop, Cursor, Cline, and every MCP-compatible client. 100+ tools. Zero boilerplate. Your AI agent interacts with LinkedIn through structured tool calls β not screen-scraping, not prompt engineering, not prayer.
Tools to start with:
linkedin.session.status Β· linkedin.search Β· linkedin.inbox.list_threads Β· linkedin.feed.list Β· linkedin.jobs.search Β· linkedin.notifications.list Β· linkedin.actions.confirm Β· linkedin.activity_poller.run_once
People, companies, posts, jobs, groups, events. One unified surface, structured results.
npm exec -w @linkedin-buddy/cli -- linkedin search "staff engineer" --category people --limit 5
npm exec -w @linkedin-buddy/cli -- linkedin search "open source ai" --category posts --limit 5
npm exec -w @linkedin-buddy/cli -- linkedin jobs search "product manager" --location Copenhagen --limit 10List threads, read messages, stage replies. Every outbound message goes through two-phase commit β you see exactly what will send before it sends.
npm exec -w @linkedin-buddy/cli -- linkedin inbox list --limit 10
npm exec -w @linkedin-buddy/cli -- linkedin inbox show --thread <thread-url-or-id> --limit 20
npm exec -w @linkedin-buddy/cli -- linkedin inbox prepare-reply --thread <thread-url-or-id> --text "Thanks for reaching out."
npm exec -w @linkedin-buddy/cli -- linkedin actions confirm --token ct_...Browse the feed, view posts, stage comments. The confirm flow means you see exactly what will post before it posts. Create text posts, media posts, and polls through the same two-phase pipeline.
npm exec -w @linkedin-buddy/cli -- linkedin feed list --limit 5
npm exec -w @linkedin-buddy/cli -- linkedin feed view <post-url>
npm exec -w @linkedin-buddy/cli -- linkedin feed comment <post-url> --text "Insightful breakdown. Thanks for sharing."
npm exec -w @linkedin-buddy/cli -- linkedin actions confirm --token ct_...Full job search with location filters and Easy Apply support. Save jobs, manage alerts, track everything locally in SQLite.
npm exec -w @linkedin-buddy/cli -- linkedin jobs search "product manager" --location Copenhagen --limit 10Inspect any profile, browse company pages, manage your notification feed.
npm exec -w @linkedin-buddy/cli -- linkedin profile view me
npm exec -w @linkedin-buddy/cli -- linkedin company view openai
npm exec -w @linkedin-buddy/cli -- linkedin notifications list --limit 20Set up watches on LinkedIn activity. Get notified when things change. Fan out webhooks with HMAC-signed payloads and automatic retry logic.
npm exec -w @linkedin-buddy/cli -- linkedin activity watch add --profile default --kind notifications --interval-seconds 600
npm exec -w @linkedin-buddy/cli -- linkedin activity webhook add --watch <watch-id> --url https://example.com/hooks/linkedin
npm exec -w @linkedin-buddy/cli -- linkedin activity run-once --profile defaultThis isn't a YOLO automation tool. Every outbound action β messages, connection requests, comments, profile edits, posts β goes through two-phase commit:
- Prepare β Action stored in SQLite, preview returned, confirm token generated.
- Review β You (or your AI agent) inspect exactly what will happen.
- Confirm β Token validated, action executed, result recorded.
Tokens expire in 30 minutes. HMAC-SHA256 sealed with entropy. No confirmation, no execution. Period.
LinkedIn's bot detection is aggressive. LinkedIn Buddy doesn't fight it β it disappears.
- Human-like typing with configurable typo rates and correction pauses
- Poisson-distributed delays between actions β not fixed sleeps, real statistical distributions
- BΓ©zier curve mouse paths β smooth, natural movement with overshoot and correction
- Fingerprint hardening β WebGL, canvas, timezone, locale, all consistent per profile
- Four evasion levels β
offΒ·lightΒ·moderate(default) Β·aggressive
Configure via --evasion-level, LINKEDIN_BUDDY_EVASION_LEVEL env var, or runtime options. See docs/evasion.md for the full breakdown.
Skip the CLI. Embed the full runtime in your own apps.
import { createCoreRuntime } from "@linkedin-buddy/core";
const runtime = createCoreRuntime();
try {
const result = await runtime.search.search({
profileName: "default",
category: "people",
query: "developer relations",
limit: 5,
});
console.log(result.results.map((person) => person.name));
} finally {
runtime.close();
}Same services, same two-phase commit safety, same evasion layer. Everything the CLI and MCP server use, exposed as clean TypeScript APIs with full type definitions.
Local-first. No cloud. No third-party servers. Everything runs on your machine.
| Architecture | Workflow |
|---|---|
The runtime wires 25+ services through constructor injection. No circular dependencies.
Infrastructure: DB β Logger β Artifacts β ProfileManager β Auth β RateLimiter β TwoPhaseCommit
β
LinkedIn: Inbox, Feed, Connections, Profile, Search, Jobs, Notifications,
Posts, Publishing, Followups, Groups, Events, CompanyPages,
Members, PrivacySettings, Analytics
β
Activity: Watches β Poller β Webhooks β Scheduler
| Tool | CLI | MCP | Dev API | Confirm-before-write | Best fit |
|---|---|---|---|---|---|
| LinkedIn Buddy | β | β | β | β | Local-first workflows for operators and AI agents |
stickerdaniel/linkedin-mcp-server |
β | β | β | β | MCP-focused LinkedIn scraping and job search |
tigillo/linkedin-cli |
β | β | β | β | Terminal-oriented LinkedIn usage |
alabarga/linkedin-api |
β | β | β | β | Library-style LinkedIn integrations |
See docs/repository-seo.md for keyword targets and the GitHub-search baseline.
| Install and build | MCP quick connect | Confirm before write |
|---|---|---|
| Need | Doc |
|---|---|
| Activity polling and webhooks | docs/activity-webhooks.md |
| Anti-bot evasion profiles | docs/evasion.md |
| E2E and replay testing | docs/e2e-testing.md |
| Live validation and account safety | docs/write-validation.md |
| Selector auditing | docs/selector-audit.md |
| Draft quality evaluation | docs/draft-quality-evaluation.md |
| Brand and social preview assets | docs/brand-guidelines.md |
| README media research | docs/readme-media-research.md |
| Media asset inventory | assets/media/README.md |
| Articles and newsletters | docs/articles-newsletters.md |
| Notifications | docs/notifications.md |
| Rate limiting | docs/rate-limiting.md |
| Jobs, alerts, and Easy Apply | docs/jobs.md |
| SEO targets and metadata | docs/repository-seo.md |
Start with CONTRIBUTING.md. Be decent (CODE_OF_CONDUCT.md). Report security issues through SECURITY.md.
If you change CLI commands, MCP tools, or write flows β update the README and the relevant docs so new users discover the feature.
LinkedIn's API is locked behind partner programs most developers will never access. The web UI is designed for humans clicking buttons one at a time. Every existing tool is either a fragile Python scraper, a Chrome extension with questionable permissions, or a wrapper around endpoints that don't actually exist publicly.
I built LinkedIn Buddy because I needed LinkedIn automation that worked. Not a demo. Not a proof-of-concept that impresses on Twitter and breaks in production. A real tool that handles authentication, evades detection, previews every action before executing, and runs entirely on my own machine.
The two-phase commit system isn't a nice-to-have β it's the thing that lets AI agents use LinkedIn without accidentally messaging your CEO. The evasion layer isn't paranoia β it's the difference between a tool that works for a week and one that works for months.
100% open source. No telemetry. No cloud. No accounts. Your data stays on your machine.
Find it useful? Star the repo. Find a bug? Open an issue. Have a better approach? PRs are welcome.
License: pending repository selection.
Built with Playwright, Commander, better-sqlite3, and the Model Context Protocol SDK.
Release notes in CHANGELOG.md.

