Instagram MCP Server
Provides tools to scrape Instagram profiles, posts, reels, direct messages, and business insights using a custom browser orchestration engine.
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., "@Instagram MCP Serverfetch Instagram profile for natgeo"
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.
Instagram MCP Server
The Problem: Instagram has no official public API for profile scraping, insights, or messaging. AI assistants (Claude, Cursor, Windsurf) need structured Instagram data — profiles, posts, reels, DMs, business insights — but Instagram's front end is a moving target. Hashed CSS classes change weekly. Rate limits and auth barriers are aggressive. A naive selenium script breaks within days.
What this is: A purpose-built MCP server that treats Instagram's web client as an adversarial data source. It uses a custom browser orchestration engine that survives DOM churn, manages browser sessions across three runtime modes (legacy, CDP, Docker), and serializes tool execution to prevent Instagram's session state from corrupting.
Engineering highlights: innerText-based extraction (zero DOM selector dependence), state-machine driven bootstrap with background retries, sequential tool middleware to prevent concurrent navigation conflicts, and a rich error diagnostic system that maps 15+ Instagram failure modes to actionable messages.
The Problem
Instagram's web app is engineered against automation:
No public API for profiles, reels, DMs, or business insights. Everything must go through the web client.
Hashed CSS classes (
x1n2onr6,x1lliihq, etc.) change with every deploy. Any scraper relying on DOM selectors breaks within weeks.Concurrent page states interfere. If two tools navigate Instagram simultaneously (e.g., "fetch profile" and "send DM"), the session enters an inconsistent state and Instagram forces re-authentication.
Rate limits and auth barriers are invisible — no HTTP status code. They appear as in-page "blocked" overlays, requiring DOM-level detection.
Session persistence across restarts requires cookie extraction from browser-native SQLite stores, which differ by browser (Chrome, Firefox, Brave, Edge, etc.).
Existing solutions (Puppeteer wrappers, static scrapers) fail because they couple extraction logic to ephemeral layout details, or lack the state management needed for multi-tool AI workflows.
Engineering Highlights
1. DOM-Agnostic Extraction Engine
Anti-pattern: Most scrapers use CSS selectors like div.x1n2onr6 > span._aacl. When Instagram rotates these classes, the scraper silently returns empty data.
Solution: The extractor (scraping/extractor.py) relies on innerText and URL navigation, not DOM selectors. Each "section" (posts, reels, followers) maps to exactly one page navigation (USER_SECTIONS dict). Extraction reads visible text, then strips Instagram's chrome (footer links, sidebar noise) using regex markers:
# Noise markers strip Instagram chrome instead of brittle CSS selectors
_NOISE_MARKERS = [
re.compile(r"^About\n+(?:Help|Press|API|Jobs|Terms|Privacy)", re.MULTILINE),
re.compile(r"^© \d{4} Instagram from Meta$", re.MULTILINE),
re.compile(r"^Suggested for you$", re.MULTILINE),
]This approach survives layout changes because Instagram cannot hide text content from the user without breaking its own UX — and innerText reads exactly what a human sees.
Key constraint: "One section = one navigation." Each section triggers exactly one page.goto(). Combining multiple data sources into a single navigation creates coupling that breaks when Instagram reorganized pages.
2. Three-Mode Browser Architecture
The server supports three distinct runtime policies, each solving a different operational constraint:
Mode | Entry | Cookie Source | Use Case |
Legacy | Default | SQLite cookie store detection | Desktop AI clients |
CDP |
| Live Brave browser session | Users who don't want cookie extraction |
Docker |
| Pre-exported portable auth ( | Server/headless deployments |
The CDP bridge (drivers/browser._bridge_runtime_profile()) is the most architecturally interesting: it connects to a running Brave browser via Chrome DevTools Protocol, imports cookies from the persistent source session into an ephemeral runtime profile, and isolates scraping in a separate browser context — zero interference with the user's actual browsing.
The Docker runtime solves the "headless auth" problem: Instagram's anti-bot checks detect missing GPU/display. The solution creates an authenticated profile on the host via --login, then tar archives it for injection into the container. The container never handles login — just cookie replay.
3. Sequential Tool Execution Middleware
Instagram's web app is a single-page application with mutable global state. If two MCP tool calls navigate Instagram simultaneously, the following happens:
Tool A navigates to
instagram.com/natgeo/posts/Tool B navigates to
instagram.com/direct/inbox/Instagram's SPA state corrupts — neither tool gets valid data
Instagram detects the "impossible" navigation pattern and forces re-login
Solution: An asyncio.Lock-based middleware (SequentialToolExecutionMiddleware) serializes all MCP tool calls within the same server process. Each tool waits in a queue, acquires the lock, executes, and releases:
class SequentialToolExecutionMiddleware(Middleware):
async def on_call_tool(self, context, call_next):
async with self._lock: # Only one navigation at a time
return await call_next(context)This is a middleware registered at server creation, not per-tool — zero tool implementation changes.
4. Bootstrap & Authentication State Machine
The bootstrap system (bootstrap.py) manages a complex initialization lifecycle across process restarts:
IDLE → SETUP_IN_PROGRESS → READY
↓ (failed)
FAILED → (background retry) → SETUP_IN_PROGRESSThe auth state machine runs independently:
UNKNOWN → CHECKING → READY
↓ (expired) ↓ (expired)
INVALID → RELOGIN_IN_PROGRESS → READYKey design decisions:
Background-first browser setup:
patchrightChromium downloads in a background task, not at startup. Tools become available immediately; if the browser isn't ready, they raiseBrowserSetupInProgressErrorwith a "retry in a few minutes" message.Cookie bridge, not credential store: No Instagram passwords are stored. Auth state is purely cookie-based, extracted from the host browser's SQLite store.
Auto-relogin on expiry: When Instagram invalidates a session mid-flight, the system detects it via
detect_auth_barrier()and triggers a fresh login flow — no manual intervention.
5. Rich Error Diagnostics
A naive scraper returns HTTP 200 with a "login required" page, and the AI client has no way to understand what happened.
Solution: A centralized raise_tool_error() function maps 15+ Instagram-specific exception types to user-friendly ToolError messages with auto-generated diagnostics:
except AuthenticationError:
raise ToolError(
"Authentication failed. Run with --login to re-authenticate."
) from exception
except RateLimitError:
raise ToolError(
f"Rate limit detected. Wait {exception.suggested_wait_time}s before retrying."
) from exceptionEach diagnostic includes an issue template path — a markdown file in docs/ that provides context-specific troubleshooting. This turns opaque scraping failures into actionable guidance.
Architecture Overview
MCP Client (Claude, Cursor, etc.)
│
┌─────▼──────┐
│ FastMCP │ ← MCP protocol (stdio or streamable-http)
│ Server │
└─────┬──────┘
│
┌─────▼──────────────┐
│ SequentialToolExec │ ← asyncio.Lock middleware
│ Middleware │ serializes all navigations
└─────┬──────────────┘
│
┌─────▼──────────────────┐
│ Tool Registry │ ← 28+ tools across 7 categories
│ (user / posts / search │
│ / insights / messaging │
│ / actions / gemini) │
└─────┬──────────────────┘
│
┌─────▼────────────────────────┐
│ Bootstrap & Auth State │
│ Machine │ ← manages browser lifecycle
└─────┬────────────────────────┘
│
┌─────▼──────────────────────┐
│ Browser Orchestrator │
│ (Legacy / CDP / Docker) │ ← three runtime modes
└─────┬──────────────────────┘
│
┌─────▼──────────────┐
│ innerText │
│ Extraction Engine │ ← zero DOM selector dependence
└────────────────────┘Tech Stack
Technology | Purpose |
Python 3.12+ | Type-safe async runtime with structural pattern matching |
FastMCP 3.x | MCP protocol server — enables stdio and streamable-http transports |
Patchright | Anti-detection Playwright fork — bypasses Instagram's webdriver checks |
asyncio.Lock + Middleware | Serialized tool execution — prevents concurrent navigation corruption |
Gemini 2.0 Flash | Multimodal reel analysis — video-to-text without local Whisper |
Docker | Headless deployment — portable auth via cookie archives |
Ruff / Ty | Strict linting (Ruff) and type checking (Ty, not mypy) |
Quick Start
# Install via uvx (no local install needed)
uvx instagram-scraper-mcpOn first tool call, a login window opens. Log in once; cookies persist across restarts.
See MCP Client Configuration for IDE setup.
Advanced Configurations
Mode | Command |
CDP (Brave) |
|
Docker | |
Gemini Analysis |
|
Debug Mode |
|
Tool Suite
Category | Tools | Capabilities |
Profile & Content |
| Posts, reels, stories, highlights, followers/following lists |
Search & Discovery |
| User, hashtag, and location search |
Messaging & Actions |
| Full DM and engagement suite |
Business Insights |
| Reach, impressions, demographics (Business/Creator accounts only) |
AI Analysis |
| Multimodal reel transcription and analysis via Gemini 2.0 Flash |
Transcription |
| Local Whisper-based SRT subtitle generation |
Potentialities
Headless auth recovery: Currently Docker runtime requires host-side
--login. A self-service web portal for one-time auth token generation would eliminate this friction.Multi-account session management: The sequential middleware prevents intra-session conflicts, but switching between Instagram accounts requires a separate profile. Native account switching would enable parallel multi-account extraction.
Webhook-based rate limit mitigation: Rate limits are currently synchronous (wait N seconds). An async queue with webhook callbacks would allow batch processing without tool timeouts.
GraphQL API fallback: Instagram's internal GraphQL API occasionally surfaces in responses. A hybrid strategy (extraction + API probes) could reduce page navigations for known-stable endpoints.
License & Acknowledgements
Licensed under the Apache 2.0 License.
Built with FastMCP and Patchright.
Use in accordance with Instagram's Terms of Use. Web scraping may violate Instagram's terms. This tool is for personal use only.
Developed by Ishan Parihar — If you find this useful, consider supporting
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/ishan-parihar/instagram-mcp-server'
If you have feedback or need assistance with the MCP directory API, please join our Discord server