arr-stack-mcp
Provides tools for interacting with a Jellyfin media server, including system info, library search, recent additions, user listing, now-playing sessions, and triggering library scans.
Provides tools for managing movies, including searching existing library, looking up movies on TMDB, checking queue and calendar, identifying missing movies, and adding or deleting movies with a confirm-token flow.
Provides tools for managing TV series, including searching existing library, looking up series on TVDB, checking queue and calendar, identifying missing episodes, and adding or deleting series with a confirm-token flow.
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., "@arr-stack-mcpsearch for the movie 'Inception' in Radarr"
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.
arr-stack-mcp
A Model Context Protocol server that exposes Sonarr, Radarr, Lidarr, and Jellyfin to any MCP client (Claude Desktop, Claude Code, n8n, ibis-bot, custom agents) through a hand-curated tool layer written for LLM consumption.
What this gives an agent
28 curated tools split across four services:
Service | Tools | Highlights |
Sonarr | 8 | series search, TVDB lookup, queue, calendar, missing episodes, add/delete |
Radarr | 8 | movie search, TMDB lookup, queue, calendar, missing, add/delete |
Lidarr | 7 | artist + album search, MusicBrainz lookup, queue, add/delete |
Jellyfin | 5 | library search, recent additions, now-playing sessions, library scan |
Every tool:
Has a description written for an agent's decision-making, not for human docs.
Pins its input and output with Pydantic v2, so the MCP catalogue carries strict JSON Schema.
Returns a compact, self-describing envelope (
ok,query,count,total,items).Translates upstream HTTP errors into structured
ToolErrorenvelopes with self-suggesting hints.Destructive operations require a two-call confirm-token flow.
Related MCP server: YARR Media Stack MCP Server
Quickstart
uvx (one-shot)
uvx arr-stack-mcp serve --transport stdioDocker
docker run --rm -i \
-e SONARR_URL=http://host.docker.internal:8989 \
-e SONARR_API_KEY=$SONARR_API_KEY \
-e RADARR_URL=http://host.docker.internal:7878 \
-e RADARR_API_KEY=$RADARR_API_KEY \
-e LIDARR_URL=http://host.docker.internal:8686 \
-e LIDARR_API_KEY=$LIDARR_API_KEY \
-e JELLYFIN_URL=http://host.docker.internal:8096 \
-e JELLYFIN_API_KEY=$JELLYFIN_API_KEY \
ghcr.io/new-usemame/arr-stack-mcp:0.1.0Local development (uv)
git clone https://github.com/new-usemame/arr-stack-mcp
cd arr-stack-mcp
uv sync
uv run arr-stack-mcp init # writes a starter arr-stack-mcp.yaml
uv run arr-stack-mcp serve # boots stdio MCP serverClaude Desktop config
Add to ~/Library/Application Support/Claude/claude_desktop_config.json on macOS, or %APPDATA%\Claude\claude_desktop_config.json on Windows:
{
"mcpServers": {
"arr-stack": {
"command": "uvx",
"args": ["arr-stack-mcp", "serve"],
"env": {
"SONARR_URL": "http://localhost:8989",
"SONARR_API_KEY": "...",
"RADARR_URL": "http://localhost:7878",
"RADARR_API_KEY": "...",
"LIDARR_URL": "http://localhost:8686",
"LIDARR_API_KEY": "...",
"JELLYFIN_URL": "http://localhost:8096",
"JELLYFIN_API_KEY": "..."
}
}
}
}Config reference
arr-stack-mcp init writes a starter arr-stack-mcp.yaml. Every secret can also be supplied via env var. Inline expansion of ${VAR} and ${VAR:-default} is supported.
log_level: info
transport:
stdio: true
http_enabled: false
http_host: 127.0.0.1
http_port: 8080
policy:
read_only: false
disable_destructive: false
confirm_token_ttl_seconds: 300
services:
sonarr:
enabled: true
url: http://localhost:8989
api_key: ${SONARR_API_KEY}
verify_tls: true
timeout_seconds: 30
radarr:
enabled: true
url: http://localhost:7878
api_key: ${RADARR_API_KEY}
lidarr:
enabled: true
url: http://localhost:8686
api_key: ${LIDARR_API_KEY}
jellyfin:
enabled: true
url: http://localhost:8096
api_key: ${JELLYFIN_API_KEY}
default_user_id: 4f8b6d2e-3a9c-4f1d-9c2a-1b3c5d7e9f01Runtime flags
Flag | Env | Effect |
|
| Skip every tool tagged |
|
| Skip only the |
| — | Plan-and-record mode (v0.2). Write/destructive tools short-circuit before the upstream mutation and log the intended payload to a ring buffer surfaced by |
| — | Default. Use for Claude Desktop and Claude Code. |
| — | HTTP transport for n8n / remote consumers. |
| (env name) | Env var the server reads the bearer token from. Streamable-HTTP on a non-loopback bind refuses to start when this env var is unset. |
| — | Override config path. Defaults to |
Confirm-token persistence (v0.2)
Confirm tokens default to in-memory storage. Set policy.state_db_path in the config to persist tokens across server restarts and across streamable-HTTP request lifetimes:
policy:
state_db_path: ~/.local/state/arr-stack-mcp/state.dbThe path is created idempotently. SQLite WAL mode lets multiple worker processes share the same DB safely.
Hourly per-tool call cap (v0.2)
Opt-in runaway-agent protection. Tools without an entry are uncapped:
policy:
hourly_caps:
sonarr.series_add: 50
radarr.movie_add: 50
sonarr.series_delete: 10Per-process, per-tool, rolling-hour. Failed upstream calls still consume budget so a runaway loop hits the cap on the Nth attempt.
Capability matrix
Tools follow the <service>.<verb_object> convention. Tags drive flag-based gating.
Sonarr
Tool | Tag | Use |
| read | Version + branch + uptime. First call when diagnosing. |
| read | Search the existing library (already-added). |
| read | Search TVDB to discover series to add. |
| read | Per-season breakdown (v0.2). Episode counts, files on disk, missing. |
| read | Active downloads with progress. |
| read | Upcoming + recently-aired episodes. |
| read | Monitored episodes not on disk. |
| write | Add a series by TVDB id. Idempotent. |
| destructive | Two-call confirm-token flow. |
Radarr
Tool | Tag | Use |
| read | Mirrors Sonarr. |
| read | Search the existing library. |
| read | Search TMDB to discover movies to add. |
| read | Active downloads. |
| read | Upcoming + recently-released movies. |
| read | Monitored movies not on disk. |
| write | Add a movie by TMDB id. Idempotent. |
| destructive | Two-call confirm-token flow. |
Lidarr
Tool | Tag | Use |
| read | — |
| read | Search the existing library. |
| read | Search MusicBrainz to discover artists. |
| read | List albums under one artist. |
| read | Active downloads. |
| write | Add an artist by MusicBrainz id. Idempotent. |
| destructive | Two-call confirm-token flow. |
Jellyfin
Tool | Tag | Use |
| read | Version + server name. Public endpoint. |
| read | Search items by name. Pass |
| read | Newest items by date_added. Pass |
| read | Enumerate user accounts (v0.2). Returns |
| read | Currently-active sessions with progress. |
| write | Trigger a library refresh. |
Prowlarr (v0.2)
Tool | Tag | Use |
| read | Version + branch. First call when diagnosing Prowlarr. |
| read | Health-check issues (warnings, errors, wiki links). |
| read | Configured indexers with implementation, protocol, priority, enable. |
| read | Query / grab / failure counts per indexer + average response time. |
| read | Currently-failing indexers with retry-after timestamps. |
| write | Trigger the test-every-indexer probe; returns per-indexer pass/fail. |
| read | Search across configured indexers. Returns title, indexer, size, age, seeders. Does NOT download — use |
stack.* (v0.2, cross-service)
Tool | Tag | Use |
| read | Probe every enabled service's |
| read | Read the ring buffer of would-have-fired mutations recorded under |
| read | Compose a pre-filled GitHub issue URL the user can post upstream. Never auto-submits. |
| read | Fan a query across every enabled arr + Jellyfin library; merged result with |
| read | Aggregate download queue across Sonarr, Radarr, Lidarr. Normalized row shape. |
Confirm-token flow
Destructive tools (*_delete) implement a two-call confirm:
# Call 1: returns plan + token, no side effect
> sonarr.series_delete(sonarr_id=42, delete_files=False)
{
"ok": true,
"needs_confirm": true,
"confirm_token": "abc123",
"summary": "remove 'Foo Bar' (sonarr_id=42) from Sonarr; keep files on disk",
"expires_in_seconds": 300
}
# Call 2: executes
> sonarr.series_delete(sonarr_id=42, delete_files=False, confirm_token="abc123")
{
"ok": true,
"deleted_sonarr_id": 42,
"title": "Foo Bar",
"files_deleted": false,
"msg": "deleted sonarr_id=42"
}Tokens are single-use, time-limited, and bound to the request payload — a token from one tool will not confirm another.
Architecture
Two layers:
┌────────────────────────────────────────────────────────────┐
│ Curated MCP tools (arr_stack_mcp.tools.<service>.tools) │
│ Hand-written, LLM-friendly names, descriptions, schemas. │
├────────────────────────────────────────────────────────────┤
│ Generated thin clients (arr_stack_mcp.generated.<svc>) │
│ openapi-python-client output. Do not hand-edit; regen │
│ via scripts/regen-clients.sh on upstream version bumps. │
├────────────────────────────────────────────────────────────┤
│ httpx + pydantic + structlog + mcp.server.fastmcp │
└────────────────────────────────────────────────────────────┘See docs/ARCHITECTURE.md for the longer walk-through.
Development
uv sync # install runtime + dev deps
uv run pytest -q # fast unit tests
uv run ruff check src/ tests/ # lint
uv run mypy src/ tests/ # type check (strict)
uv run ruff format src/ tests/ # formatIntegration tests require the docker test stack:
scripts/test-stack-up.sh # boots Sonarr/Radarr/Lidarr/Prowlarr/Jellyfin
uv run pytest -m integration # runs once stack is healthy
scripts/test-stack-down.sh --clean # teardownContributing
See CONTRIBUTING.md for the OpenAPI regeneration flow and the verification checklist that gates every release.
License
MIT
This server cannot be installed
Maintenance
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/new-usemame/arr-stack-mcp'
If you have feedback or need assistance with the MCP directory API, please join our Discord server