Event-sourced ticket system with a Jira reconciler — CLI, Python library, and MCP server.
Project description
rebar
A git-native ticket system for coordinating coding agents — and the humans working alongside them.
Point several agents at one repo and they immediately need a shared place to coordinate: to claim work without grabbing the same ticket, record what they discover, and hand off cleanly — while your teammates stay in the loop through Jira. rebar makes the tracker part of the repo itself, so it travels with every clone, needs no database or daemon, and lets many agents and sessions write at once without merge conflicts or lost work.
It's an event-sourced ticket system with a Jira reconciler, exposed three ways:
- CLI — the
rebarcommand - Python library —
import rebar - MCP server —
rebar-mcp(stdio)
Tickets are stored as an append-only event log on a dedicated tickets git
orphan branch (worktree at .tickets-tracker/); state is computed by replaying
events. A level-triggered reconciler bidirectionally syncs tickets with Jira.
This project was extracted from the digital-service-orchestra Claude Code
plugin. The bash + Python engine is wrapped verbatim under src/rebar/_engine/;
the three interfaces are thin layers over it.
Why rebar
If you run coding agents against a repo, you eventually want to run several at once — and the moment you do, they need a shared place to coordinate. Most trackers weren't built for that:
- They're heavy. A daemon to babysit or a local database to keep running, with dependencies thick enough that a routine upgrade can break your work tracking across machines.
- They don't travel with the code. State lives outside the repo, so a fresh clone doesn't come with its tickets.
- They fight your git history. A tracker that writes to your working branch tangles ticket churn into your source-code commits.
- They have no concurrency story. Nothing stops two agents from claiming the same work or clobbering each other's state, and concurrent edits produce merge conflicts you resolve by hand — or lose.
- They buckle at scale. Speed and usability fall off past a few hundred tickets.
rebar's answer is to make the tracker part of the repo. Tickets are an
append-only event log on a dedicated tickets orphan branch (linked in through a
gitignored worktree); current state is a fast, deterministic replay of that log.
That single decision pays off across the board:
- Zero infrastructure, fully portable. No database, no daemon — just git and a lightweight Python install. Clone the repo and the tracker comes with it.
- No commit interference. Ticket events live on their own branch and never touch your source history. Every write auto-commits and auto-pushes, so activity is shared in real time.
- Concurrency by design. Each event gets a globally-unique filename, so
parallel writes merge as a clean union, and the rare conflicting fork resolves
deterministically — every clone converges with no lost data.
claimis an atomic, optimistic-concurrency primitive: agents grab work without stepping on each other. - Built to scale. The event log plus cached replay stays fast as tickets grow.
On top of that foundation, rebar adds what parallel agent work actually needs:
- Bidirectional Jira sync — agents work in rebar, teammates work in Jira, and a level-triggered reconciler keeps the two in step.
- Conflict-aware scheduling — tickets record their file impact, so
next-batchhands parallel agents work that won't collide on the same files. - Scratch space — an invisible per-ticket channel for subagents to pass notes to one another.
- Quality gates — clarity, acceptance-criteria, dispatch-readiness, and repo-wide health checks keep work dispatch-ready.
- Provenance links —
discovered_fromties emergent work back to the ticket that surfaced it. - One store, three interfaces — drive it from the CLI, a Python library, or the MCP server.
Requirements
Runtime (system):
- Python ≥ 3.10
git,bash,jq— required.flockfrom util-linux — recommended for robust write serialization, but not strictly required: it is not onPATHby default on macOS (brew install util-linux), and when no util-linuxflockis found rebar falls back to amkdir-based lock automatically. (A non-util-linuxflocksuch as BusyBox's is ignored in favor of the fallback.)acli(Atlassian CLI) — only for live Jira reconciliation.
Python extras (pip): the base install gives the rebar CLI + import rebar
library; [mcp] adds the rebar-mcp server (pulls mcp>=1.2); [dev] adds the
test/dev deps (pytest, mcp, jsonschema) and is required to run the full
test suite — the interface-parity tests import the MCP server, so they error
(not skip) if the mcp extra is absent. See Install and
Tests.
Install
rebar ships from one Python package — PyPI distribution nava-rebar (the
import package and commands stay rebar / rebar-mcp). Pick the channel that
fits. (System prerequisites in all cases: git, jq, flock, bash,
python3; acli only for live Jira reconciliation.)
Homebrew (CLI)
brew install navapbc/rebar/rebar
# or: brew tap navapbc/rebar && brew install rebar
Installs the rebar CLI (and the rebar library inside the formula's venv). For
the MCP server via Homebrew users, install the [mcp] extra with pipx/uvx below.
PyPI — pipx / pip
pipx install nava-rebar # isolated CLI on PATH: rebar
pip install nava-rebar # library: import rebar
pip install 'nava-rebar[mcp]' # + MCP server: rebar-mcp
MCP server — from the MCP Registry
Listed in the MCP Registry as
io.github.navapbc/rebar. Registry-aware MCP clients can add it by that
name; or register it directly in your client config (zero pre-install via
uvx):
{
"mcpServers": {
"rebar": {
"command": "uvx",
"args": ["--from", "nava-rebar[mcp]", "rebar-mcp"],
"env": { "REBAR_ROOT": "/path/to/your/repo" }
}
}
}
(Already pip/pipx-installed nava-rebar[mcp]? Use "command": "rebar-mcp"
instead.) Server flags: REBAR_MCP_READONLY=1 exposes only read tools;
reconcile is dry-run unless REBAR_MCP_ALLOW_RECONCILE_LIVE=1.
From source
git clone https://github.com/navapbc/rebar && cd rebar
pip install . # library + CLI
pip install '.[mcp]' # + MCP server (FastMCP)
pip install -e '.[dev]' # editable + test deps (pytest, mcp)
Packaging note: the engine (bash dispatcher +
ticket-*.sh+ python helpers) is exec'd as real files, so rebar must be installed unpacked to a real on-disk directory — zipimport / zip-safe installs are unsupported. Standard wheel installs (hatchling builds unpacked) and editable installs satisfy this;engine_dir()asserts it at the first engine call and fails loudly otherwise.
CLI
rebar init # create the tickets branch + worktree
rebar create story "Add login page" # prints the ticket id
rebar list [--status=open] [--has-tag=...] # JSON array
rebar show <id|alias> # compiled ticket state (JSON)
rebar transition <id> <current> <target> # optimistic-concurrency status change
rebar comment <id> "<body>"
rebar link <id1> <id2> <relation> # relation REQUIRED (see relations below)
rebar unlink <source> <target> # remove ONE link for the ordered pair (no relation arg)
rebar deps <id> # dependency graph
rebar ready # tickets with all blockers closed
rebar next-batch <epic-id> # unblocked tickets under an epic's hierarchy
rebar validate # repo-wide tracker health (NO ticket id; whole-store score 1-5)
rebar clarity-check <id> / check-ac <id> / quality-check <id> # per-ticket quality gates
rebar reconcile [--mode dry-run|reconcile-check|live] # Jira sync (default: dry-run)
Run rebar help (or rebar --help / -h) for the subcommand overview, and
rebar <subcommand> --help (or rebar help <subcommand>) for a specific
subcommand's usage — --help prints usage and never executes the command.
Help is only recognized as the first argument after the subcommand, so a
--help/-h/help that appears inside a free-text parameter (title, comment
body, search query, …) is treated as literal text, not a help request.
Repo root is resolved from REBAR_ROOT (or PROJECT_ROOT), falling back to the
git toplevel of the working directory.
validate vs. the per-ticket gates. rebar validate takes no ticket id —
it scans the whole store and prints an overall tracker-health score (1-5, exit
0-4) bucketed into critical / major / minor / warning findings (--json,
--terse, --verbose, --fix). Passing it a ticket id errors. The per-ticket
quality gates are separate commands that each take an <id>: clarity-check,
check-ac, quality-check.
Links. rebar link <id1> <id2> <relation> requires a relation; the six
relations are blocks, depends_on, relates_to, duplicates, supersedes,
discovered_from. rebar unlink <source> <target> takes no relation
argument — it is pair-scoped and removes the most-recently-created link
between that ordered pair, one per call, so to remove multiple links between the
same pair you call unlink repeatedly. Note that blocking links
(blocks/depends_on) may be promoted up the parent hierarchy when created (see
below), so unlink must target the promoted (ancestor) endpoint to remove
such a link.
Hierarchy promotion of blocking links
For blocking dependencies only (blocks, depends_on), rebar promotes the
link endpoints up the parent hierarchy so the dependency sits between tickets at
a comparable level (epic↔epic, story↔story, task/bug↔task/bug). When it does so
it emits a REDIRECT: A→B promoted to … note. Non-blocking relations
(relates_to, duplicates, supersedes, discovered_from) are linked exactly
as given, with no promotion.
The store auto-commits and auto-pushes every write
Every rebar write (create, edit, transition, claim, link, …)
auto-commits its event to the tickets branch and auto-pushes that branch to
origin/tickets whenever an origin remote exists. Local ticket activity is
therefore shared with the remote immediately — including test/scratch tickets,
so be deliberate when working against a repo with a shared tickets remote. The
push is best-effort: with no origin remote nothing is pushed, and a push
failure (e.g. non-fast-forward it cannot auto-merge, or no network) never fails
the write — it leaves the local commit intact and the branch diverged.
rebar fsck reports PUSH_PENDING when the local tickets branch is ahead of
origin/tickets, so unpushed activity is observable. See
docs/concurrency.md for the push/merge-retry algorithm.
Python library
import rebar
rebar.init_repo(repo_root="/path/to/repo")
tid = rebar.create_ticket("story", "Add login page", priority=2)
ticket = rebar.show_ticket(tid) # dict
tickets = rebar.list_tickets(status="open") # list[dict]
try:
rebar.transition(tid, "open", "in_progress")
except rebar.ConcurrencyError:
... # ticket changed since last read
result = rebar.reconcile("dry-run") # Jira sync (non-mutating)
# Native, in-process reads (no subprocess):
from rebar import reduce_all_tickets, reduce_ticket
MCP server
rebar-mcp # stdio transport
Exposes ticket operations as MCP tools. reconcile defaults to dry-run
(live requires REBAR_MCP_ALLOW_RECONCILE_LIVE=1). Set REBAR_MCP_READONLY=1
to expose only the read tools (no write/mutation tools). To register it in an MCP
client (registry name io.github.navapbc/rebar, or a direct uvx config), see
Install → MCP server above.
Maintainers: the registry manifest lives in server.json;
publish/update it with the mcp-publisher CLI (see docs/releasing.md). The
registry verifies PyPI-package ownership via this annotation (kept in this
README, which is the PyPI long description):
mcp-name: io.github.navapbc/rebar
License
Apache-2.0 — see LICENSE.
Configuration
Optional .rebar/config.conf (or .rebar.conf) at the repo root, flat
key=value:
ticket.display_mode=auto # auto | canonical | alias | short
ticket_clarity.threshold=70
verify.require_verdict_for_close=true # opt-in (default: false) — gate story/epic
# close on a PASS verdict hash
Tests
Run the suite from an environment with the [dev] extra installed (a venv is
recommended); the interface-parity tests import the MCP server, so a bare
interpreter without the mcp extra will error rather than skip.
python -m venv .venv && source .venv/bin/activate
pip install -e '.[dev]' # editable + pytest, mcp, jsonschema
pytest # full Python suite
pytest tests/interfaces # interface-parity tier only
bash tests/scripts/test-ticket-create.sh # bash engine tests
The Python suite is sub-divided by concern:
tests/scripts,tests/unit— the engine (reducer, graph, reconciler) and bash scripts.tests/interfaces— proves the library, CLI, and MCP interfaces behave identically over one git-backed store:test_parity.pyruns each operation through all three interfaces (and a cross-interface coherence check: write via one, read via the others);test_surface.pypins the per-interface capability surface (e.g. MCP has noinit; there is noclassify);test_library.py/test_cli.py/test_mcp.pycover per-interface specifics (typed exceptions, exit-code passthrough, read-only/live gates).
Project details
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