Jira Cloud client for AI agents and automation
Project description
jira-pro
Jira Cloud client for AI agents and automation. JSON in, JSON out.
Built for LLM agents, CI pipelines, and scripts that need to read, create, and
manage Jira issues programmatically. Also works as a human CLI — pipe through
jq for readable output.
Why
- JSON by default — every command outputs structured JSON that agents parse directly
- Friendly field names — write
story_pointsinstead ofcustomfield_10036 - Schema-aware — syncs your instance's field config, resolves names and values automatically
- Templates — define issue defaults once, reuse across agents and workflows
- Shell completion — field names and enum values from your live schema
- Zero-config auth —
jira auth loginhandles OAuth, token refresh is transparent
Install
uv tool install jira-pro
Or from source:
git clone https://github.com/henriquebastos/jira-pro.git
cd jira-pro
uv tool install -e .
Quick Start
# 1. Login (opens browser for Atlassian OAuth)
jira auth login
# 2. Sync your instance's field schema
jira fields sync
jira fields sync --project DEV # sync type schemas for a specific project
# 3. Get an issue
jira issue get DEV-123
# {"key": "DEV-123", "summary": "Fix auth", "status": "To Do", "assignee": null, "priority": "P2: Medium", "type": "Task"}
# 4. Search with JQL
jira search "project = DEV AND sprint in openSprints() AND status != Done"
# [{"key": "DEV-124", "summary": "...", ...}, ...]
# 5. Create an issue
jira issue create --json '{"project": "DEV", "issuetype": "Task", "summary": "New task", "parent": "DEV-100"}'
# {"id": "12345", "key": "DEV-125", ...}
Agent Integration
Every command returns JSON to stdout. Agents can call jira as a subprocess
and parse the output directly. Common agent workflows:
# Read an issue and decide what to do
jira issue get DEV-123
# Search for unfinished work
jira search "assignee = currentUser() AND status != Done"
# Create a task from agent analysis
jira issue create --template backend --summary "Fix race condition in payment handler"
# Update status after completing work
jira issue transition DEV-123 "Done"
jira issue comment DEV-123 "Fixed in commit abc1234"
# Bulk update parent epic
jira bulk edit DEV-1 DEV-2 DEV-3 --set parent=DEV-100
For raw API access (no field resolution), use --raw on reads and
--raw-payload on writes.
Authentication
jira auth login # opens browser, handles OAuth 2.0 (3LO) + PKCE
jira auth status # show login state
jira auth logout # remove stored tokens
Token refresh is automatic and transparent — you never deal with tokens.
Using your own OAuth app
The CLI ships with a default OAuth app. To use your own, register one at developer.atlassian.com:
- Create an OAuth 2.0 (3LO) app
- Callback URL:
http://localhost:8888/callback - Jira API scopes:
read:jira-work,write:jira-work,read:jira-user - Enable sharing under Distribution
# Via flags
jira auth login --client-id YOUR_ID --client-secret YOUR_SECRET
# Or via environment variables
export JIRA_CLIENT_ID=YOUR_ID
export JIRA_CLIENT_SECRET=YOUR_SECRET
jira auth login
Resolution order: flags → env vars → built-in defaults. Credentials are saved on login and used for all subsequent token refreshes.
Multi-Instance
jira --instance mycompany issue get DEV-123
# or
export JIRA_INSTANCE=mycompany
Matches against site name (mycompany → mycompany.atlassian.net).
Single instance is auto-detected.
Commands
Issues
jira issue get DEV-123 # formatted JSON
jira issue get DEV-123 --fields summary,status # specific fields
jira issue get DEV-123 --raw # raw API response
jira issue create --json '{"project": "DEV", "issuetype": "Task", "summary": "Title"}'
jira issue create --template my-template --summary "Title"
jira issue create --raw-payload '{"fields": {...}}' # bypass resolution
jira issue edit DEV-123 --set priority="P1: High" --set story_points=5
jira issue edit DEV-123 --json '{"team": "Backend"}'
jira issue edit DEV-123 --raw-payload '{"fields": {...}}' # bypass resolution
jira issue transition DEV-123 "In Progress"
jira issue assign DEV-123 alice@example.com
jira issue comment DEV-123 "Deployed to staging"
jira issue link DEV-123 DEV-456 --type blocks
Search
jira search "project = DEV AND sprint in openSprints()"
jira search "parent = DEV-100" --fields summary,status,assignee
Bulk
jira bulk edit DEV-1 DEV-2 DEV-3 --set parent=DEV-100
Sprints and Boards
jira sprint current --board 42
jira sprint list --board 42 --state active,future
jira sprint issues 123 --fields summary,status
jira board list --project DEV
jira board backlog 42
Users
jira user me
jira user search "alice"
Field Resolution
The CLI maps friendly field names to Jira's internal field IDs using
the synced schema. You write story_points, the API receives customfield_10036.
Values are also expanded based on field type:
| You write | API receives |
|---|---|
"project": "DEV" |
"project": {"key": "DEV"} |
"issuetype": "Task" |
"issuetype": {"name": "Task"} |
"parent": "DEV-100" |
"parent": {"key": "DEV-100"} |
"priority": "P1: High" |
"priority": {"name": "P1: High"} |
"team": "Backend" |
"customfield_10001": {"value": "Backend"} |
"components": ["API"] |
"components": [{"name": "API"}] |
"story_points": 5 |
"customfield_10036": 5 |
"labels": ["urgent"] |
"labels": ["urgent"] |
Already-structured values (dicts) pass through without double-wrapping.
Use --raw-payload to bypass all resolution.
Schema
jira fields sync # sync fields + discover available projects
jira fields sync --project DEV # also sync type schemas (required fields, allowed values)
jira fields list # all fields
jira fields list --filter story # search by name
jira fields schema --project DEV --type Task # full type schema for agents
Run after login, and again if your Jira admin changes field configuration.
You can sync multiple projects incrementally — each --project adds to the
existing schema without overwriting previously synced projects.
Stored at ~/.config/jira-pro/{cloud_id}/schema.json.
Templates
Partial JSON with friendly field names. Define defaults for repetitive issue creation.
# Create
jira template create backend --json '{
"project": "DEV",
"issuetype": "Task",
"parent": "DEV-100",
"components": ["API"]
}'
# Use
jira issue create --template backend --summary "Fix the thing"
# Manage
jira template list
jira template show backend
jira template delete backend
jira template default backend # set as default
jira template default # show current default
jira template default --clear # remove default
Composition order
Fields merge in layers — later layers override earlier ones (shallow, not deep):
Template → --json → --set/--summary → resolve_fields() → API
Or bypass everything with --raw-payload.
Shell Completion
jira completion install # prints setup for your shell
Requires argcomplete on PATH: uv tool install argcomplete
Completes commands, template names, field names, and enum values from your live schema:
jira issue create --template <TAB> → backend, bug, ...
jira issue edit DEV-123 --set <TAB> → story_points=, priority=, team=, ...
jira issue edit DEV-123 --set priority=<TAB> → P0: Critical, P1: High, ...
Configuration
~/.config/jira-pro/
├── config.json # default instance
└── {cloud_id}/
├── config.json # client_id, cloud_id, site
├── refresh.json # OAuth refresh token
├── schema.json # field registry + type schemas
└── templates/
└── backend.json
Development
git clone https://github.com/henriquebastos/jira-pro.git
cd jira-pro
uv sync
uv run pytest # 148 tests
uv run ruff check src/ tests/
uv run jira --help
Architecture
src/jira/
├── auth.py # OAuth 2.0 (3LO) — login, token refresh, PKCE
├── cache.py # FileCache — file-backed key-value store with expiry
├── cli.py # argparse CLI — parse/cli split, all dispatch
├── client.py # JiraClient + sub-clients (Issue, Search, Sprint, Board, User)
├── completers.py # Shell completion for argcomplete
├── config.py # Instance discovery and resolution
├── formatters.py # Pure response transformers (raw API → clean JSON)
├── schema.py # Field registry, type schemas, field resolution
└── templates.py # Template CRUD, merge logic, build_issue_fields
See conventions.md for code style and workflow guidelines.
License
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file jira_pro-0.1.0.tar.gz.
File metadata
- Download URL: jira_pro-0.1.0.tar.gz
- Upload date:
- Size: 15.9 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
1d06364a8186c59c23081212d12ca81af5b5398f8191d68330b43f30679b51af
|
|
| MD5 |
e05bde77f84926348da2ce33b9f9051b
|
|
| BLAKE2b-256 |
e2d98c2cfb199fe9f976937f501d7634ace070dfb233aa080a2a9758e1e54d23
|
Provenance
The following attestation bundles were made for jira_pro-0.1.0.tar.gz:
Publisher:
release.yml on henriquebastos/jira-pro
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
jira_pro-0.1.0.tar.gz -
Subject digest:
1d06364a8186c59c23081212d12ca81af5b5398f8191d68330b43f30679b51af - Sigstore transparency entry: 1179309886
- Sigstore integration time:
-
Permalink:
henriquebastos/jira-pro@cce0248053ac9340e7aa184bd85c788c6add6b6f -
Branch / Tag:
refs/tags/v0.1.0 - Owner: https://github.com/henriquebastos
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@cce0248053ac9340e7aa184bd85c788c6add6b6f -
Trigger Event:
release
-
Statement type:
File details
Details for the file jira_pro-0.1.0-py3-none-any.whl.
File metadata
- Download URL: jira_pro-0.1.0-py3-none-any.whl
- Upload date:
- Size: 19.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
06e7aecd4d35a2221caa1fb7e205348b2ed793257cb0e264b12d22084847e030
|
|
| MD5 |
fb6038bcd2f6099ab9975769737638ed
|
|
| BLAKE2b-256 |
9c1b293d68d48fcef08b32039682dac7bdebf5b2af7c13ff36f4ce4176e9eec0
|
Provenance
The following attestation bundles were made for jira_pro-0.1.0-py3-none-any.whl:
Publisher:
release.yml on henriquebastos/jira-pro
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
jira_pro-0.1.0-py3-none-any.whl -
Subject digest:
06e7aecd4d35a2221caa1fb7e205348b2ed793257cb0e264b12d22084847e030 - Sigstore transparency entry: 1179309981
- Sigstore integration time:
-
Permalink:
henriquebastos/jira-pro@cce0248053ac9340e7aa184bd85c788c6add6b6f -
Branch / Tag:
refs/tags/v0.1.0 - Owner: https://github.com/henriquebastos
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@cce0248053ac9340e7aa184bd85c788c6add6b6f -
Trigger Event:
release
-
Statement type: