Ultra Jira MCP Server/CLI
A token-efficient Jira MCP server and CLI for Claude, ChatGPT, Cursor, Cline, and any other Model Context Protocol (MCP) client. Gives AI agents full access to Jira Cloud via the REST API v3 and Agile API 1.0 — without burning your context window on Jira's verbose JSON.
Built for the way real teams use AI agents with Jira: triaging tickets, planning sprints, summarizing comment threads, drafting status updates, running JQL searches, moving issues through workflows — not just for engineers writing code. If your agent talks to Jira, this server keeps it fast and cheap.
Keywords: Jira MCP server, Jira Cloud MCP, Claude Jira integration, ChatGPT Jira, Cursor Jira, Cline Jira, Atlassian MCP, AI agent for Jira, JQL agent, sprint management AI, ticket triage AI.
Why this over other Jira MCP servers
Most Jira MCP servers wrap the REST API one endpoint per tool and pass raw responses straight to the agent. That works on toy tickets and falls over the moment you point it at a real backlog. Here's what ultra-jira-mcp does differently:
- ~17× smaller per-call responses. A real "investigate this ticket with its comments" call drops from 270 KB (~67k tokens) of raw Jira JSON to 15 KB (~3,900 tokens) — same description, comments, subtasks, and links, just without Jira's
selfURLs, icon URLs, nested schema metadata, and ADF rich-text ASTs. See docs/BENCHMARK.md for the full table. - Up to 99× smaller tool-list cost (the bytes paid into your context window on every conversation, before the agent has done anything). 16 consolidated tools instead of 80+ thin ones; optional
code-apimode collapses that to a single tool that hands the agent a shell binary. - Full responses are never lost — they're written to a session-scoped temp dir and referenced by path. The agent reads the trimmed summary by default and
cats the full payload only when it actually needs the detail. - ADF flattening built in. Atlassian Document Format trees become plain text on the way out, so a 500-byte comment doesn't arrive as a 4 KB nested AST.
- Works as a standalone CLI too. The same
jira-clibinary runs without any MCP server in the picture — drop it in a script, a CI job, or a non-MCP agent framework. - Tool surface you can prune.
JIRA_ENABLED_CATEGORIESandJIRA_DISABLED_ACTIONSlet you whitelist only the operations your agent should see (and block destructive ones likeissue.deleteat the dispatch layer, not just hide them from the schema). - Covers REST v3 and Agile API 1.0. Issues, comments, worklogs, attachments, users, projects, boards, sprints, epics, filters, links, watchers/votes, fields, groups. Not just CRUD on issues.
- Ships a Claude Code skill. One command (
jira-cli install-skill) drops a SKILL.md under~/.claude/skills/jira/so any future Claude Code session discovers the CLI on demand whenever the user mentions Jira.
If you're building anything beyond a demo — an agent that lives in a Slack channel triaging incoming tickets, a sprint-planning assistant that reads three boards, a release-notes drafter that scans a JQL filter — the per-call savings dominate every conversation longer than a handful of calls.
Use cases
This isn't only for coding agents. Anywhere an LLM talks to Jira:
- Ticket triage and routing — agent reads new issues, classifies, assigns, sets priority.
- Sprint planning and retros — pull the backlog, summarize blockers, draft sprint goals.
- Status reporting — JQL → trimmed list → human-readable update for Slack, email, or a Confluence page.
- Comment thread summarization — long discussion threads collapse to the decisions and open questions.
- Workflow automation — agent moves issues through transitions, assigns reviewers, links related work.
- Incident and bug investigation — agent pulls a rich ticket with all comments, attachments, and linked issues in one trimmed response.
- Cross-tool workflows — pair with Confluence, GitHub, Slack, or Bitbucket MCP servers to drive end-to-end flows.
- Coding agents — yes, this too. Claude Code, Cursor, Cline, and friends can pick up tickets, implement them, and update status without flooding context.
Installation
npx ultra-jira-mcp
Or globally:
npm install -g ultra-jira-mcp
Configuration
Set these environment variables on the server process. With Claude Desktop or Claude Code that means the env block on the MCP server config.
| Variable | Description | Required |
|---|---|---|
JIRA_HOST |
Jira instance URL (e.g. https://yourcompany.atlassian.net) |
Yes |
JIRA_EMAIL |
Atlassian account email | Yes |
JIRA_API_TOKEN |
API token from Atlassian Account Settings | Yes |
JIRA_CLOUD_ID |
Cloud ID for scoped (ATATT/ATSTT) tokens; auto-fetched if omitted | No |
JIRA_TOOL_MODE |
"classic" (default — 16 consolidated tools) or "code-api" (one tool, agent drives via the bundled jira-cli shell binary) |
No |
JIRA_ENABLED_CATEGORIES |
Comma-separated category whitelist. Empty = all 16 categories enabled. | No |
JIRA_DISABLED_ACTIONS |
Comma-separated category.action blacklist. Enforced at the dispatch layer in both modes. |
No |
Claude Desktop / Claude Code setup
Add to ~/Library/Application Support/Claude/claude_desktop_config.json (macOS) or your ~/.claude.json:
{
"mcpServers": {
"jira": {
"command": "npx",
"args": ["-y", "ultra-jira-mcp"],
"env": {
"JIRA_HOST": "https://yourcompany.atlassian.net",
"JIRA_EMAIL": "[email protected]",
"JIRA_API_TOKEN": "your-api-token"
}
}
}
}
Tool surface
Default mode is classic: 16 consolidated MCP tools, each taking an action argument plus action-specific args. For example jira_issue covers get, create, update, delete, bulkCreate, listTransitions, transition, assign, changelog — discriminated by action: "...".
The 16 tools and their actions
| Tool | Actions |
|---|---|
jira_issue |
get, create, update, delete, bulkCreate, listTransitions, transition, assign, changelog |
jira_search |
issues, jqlAutocompleteData, jqlSuggestions |
jira_comment |
list, add, update, delete |
jira_user |
myself, search, get, assignable, bulkGet |
jira_project |
list, get, create, update, delete, listComponents, createComponent, listVersions, createVersion, updateVersion, statuses |
jira_board |
list, get, create, delete, configuration, issues, backlog, epics |
jira_sprint |
listForBoard, get, create, update, delete, issues, moveIssues, moveIssuesToBacklog |
jira_epic |
get, issues, moveIn, removeFromCurrent |
jira_worklog |
list, add, update, delete |
jira_attachment |
get, delete, meta |
jira_filter |
list, get, create, update, delete, listFavourite |
jira_link |
create, get, delete, types |
jira_watcher |
list, add, remove, listVotes, addVote, removeVote |
jira_field |
list, issueTypes, priorities, statuses, resolutions, createMeta |
jira_group |
search, members, myPermissions |
jira_server |
info |
Every action returns a trimmed summary (e.g. IssueSummary with key, status, assignee, recent comments, attachment list) plus the full untrimmed body written to disk under ${TMPDIR}/jira-mcp/${session}/. Agents read the full response only when they need the detail.
Tool filtering
Two env vars cut the tool-list cost paid every conversation:
"env": {
"JIRA_ENABLED_CATEGORIES": "issue,search,comment",
"JIRA_DISABLED_ACTIONS": "issue.delete,project.delete"
}
JIRA_ENABLED_CATEGORIES— whitelist of consolidated tool categories (the part afterjira_). Tools outside the whitelist drop from the listing.JIRA_DISABLED_ACTIONS—category.actionpairs (manifest operation names likeissue.delete,permissions.mine,vote.add). Disabled actions are stripped from each tool'soneOfschema and rejected at dispatch time, so they're blocked even in code-api mode.
Concrete numbers from a recent benchmark run on a real Jira instance:
| filter | tool-list bytes | ~tokens | factor |
|---|---|---|---|
| none (16 tools) | 30.6KB | ~7,800 | 1× |
| 3 categories | 6.3KB | ~1,600 | 5× |
| 3 cats + 5 disabled actions | 4.6KB | ~1,200 | 6.6× |
| code-api mode (1 tool) | 0.4KB | ~100 | 76× |
For the full v1-vs-v2 picture (per-call cost, three scenarios, ratios) see docs/BENCHMARK.md.
code-api mode (recommended for shell-capable agents)
If your agent can run shell commands (Claude Code, etc.), set JIRA_TOOL_MODE=code-api for a ~76× smaller tool-list footprint paid on every conversation. Per-call cost is essentially the same as classic, so the savings are pure win once your agent is making more than a few Jira calls.
Set JIRA_TOOL_MODE=code-api to expose a single MCP tool, jira_code_api. Calling it returns the path to the bundled jira-cli binary plus the JIRA_MCP_SOCKET address. The agent then drives Jira from a shell:
JIRA_MCP_SOCKET=/tmp/jira-mcp/${session}/ipc.sock \
node <cli-path> issue.get --issueIdOrKey=PROJ-1
# stdout: trimmed summary as JSON, then a final `ref: /path` line
# pointing at the full response on disk (`cat` it for detail).
Discovery: node <cli-path> --help lists every operation; node <cli-path> <op> --help lists its flags.
Stays as classic by default because tool-only MCP clients (no shell access) can't drive jira-cli. See docs/MIGRATION.md for the full flow.
Standalone CLI (no MCP server)
The same jira-cli binary works without an MCP server. Set JIRA_HOST / JIRA_EMAIL / JIRA_API_TOKEN in your shell (or a .env.local in the cwd) and invoke it directly:
export JIRA_HOST=https://yourcompany.atlassian.net
export [email protected]
export JIRA_API_TOKEN=...
npx -y -p github:scottlepp/ultra-jira-mcp#codeapi jira-cli issue.get --issueIdOrKey=PROJ-1
The CLI auto-selects between bridge mode (JIRA_MCP_SOCKET set, talks to a running server) and direct mode (no socket, builds a JiraClient in-process). Same trim + ref output either way; credentials never leave the CLI's process.env, so they don't enter the agent's context.
For Claude Code users, install the skill once so the agent discovers the CLI on its own:
npx -y -p github:scottlepp/ultra-jira-mcp#codeapi jira-cli install-skill
This writes ~/.claude/skills/jira/SKILL.md. The skill loads on demand whenever the user mentions Jira and teaches the agent the canonical invocation, common operations, and --help discovery. Re-run with --force to update; --print dumps the rendered SKILL.md to stdout without writing.
Resources
The server also exposes Jira data via MCP resources:
jira://projects— list of accessible projectsjira://project/{key}— project detailsjira://issue/{key}— issue detailsjira://boards— all boardsjira://board/{id}— board detailsjira://sprint/{id}— sprint detailsjira://myself— current user
Development
npm install
npm run build # tsc → build/
npm test # vitest, ~400 unit tests, no live Jira
npm run benchmark # measures tool-list + per-call bytes against live Jira
npm run inspector # @modelcontextprotocol/inspector against build/index.js
The benchmark requires .env.local with the regular JIRA_* vars plus JIRA_BENCH_TICKET_RICH and JIRA_BENCH_TICKET_SIMPLE keys. To include v1 in the comparison, set up a sibling worktree once: git worktree add ../ultra-jira-mcp-v1 v1.0.0 && (cd ../ultra-jira-mcp-v1 && npm install && npm run build). See docs/BENCHMARK.md for the latest numbers.
License
MIT