Cost-free Ollama-powered code-review agent for Claude Code
Project description
kagura-code-reviewer
Pro-grade code review on your git diff with zero Anthropic billing. A local (or
cloud) Ollama model fans out multi-angle finders, adversarially majority-vote
verifies each one, dedups, and returns a ranked Markdown/JSON report with a
green / yellow / red verdict you can gate CI on.
Cost-free, Ollama-powered code review for Claude Code. The review "brain" runs on an Ollama model (cloud or local), so reviews consume zero Anthropic billing. A companion slash command (/kagura-code-reviewer) integrates with Kagura Memory: the outer Claude session retrieves past conventions and findings, passes them as context to the CLI, then writes durable knowledge back after the review. The CLI itself is a self-contained tool — it does not call Kagura Memory directly.
How it works
Claude Code (outer session)
│ /kagura-code-reviewer slash command
│ 1. Recalls past findings from Kagura Memory (trust_tier: trusted filter)
│ 2. Writes assembled context to /tmp/kcr-ctx.md
│ 3. Invokes CLI ──────────────────────────────────────────────────────┐
│ │
└── Presents report to user ←── 5. Writes durable knowledge back │
▼
kagura-code-reviewer CLI
│ git diff (base...HEAD)
│ sandboxed repo tools
│ read_file / grep / git
▼
Ollama (local or cloud)
│ agentic review loop
▼
Markdown / JSON report
exit 0 = clean
exit 1 = blocking issues
Install
pip install kagura-code-reviewer
System prerequisites
These are not installed by pip — you must set them up separately:
- Ollama daemon running with at least one model pulled.
Default cloud alias usesqwen3-coder:480b-cloud; default local alias usesqwen2.5-coder:7b.
Pull with:ollama pull qwen2.5-coder:7b claudeCLI — required only for the/kagura-code-reviewerslash-command workflow.
Install via:npm install -g @anthropic-ai/claude-code
Quickstart
# Review current branch vs main (Markdown to stdout)
kagura-code-reviewer --base main
# Write report as JSON to a file
kagura-code-reviewer --base main --format json --out report.json
# Use the local model alias (faster, smaller)
kagura-code-reviewer --local
# Limit review to specific paths
kagura-code-reviewer --base main --paths src/foo.py --paths tests/test_foo.py
Exit codes:
0— no blocking issues (severities INFO / LOW / MEDIUM only)1— one or more HIGH or CRITICAL findings2— git error (bad refs, not a git repo, etc.)
Example output
# Code Review
## [CRITICAL] IndexError when orders is empty (correctness)
- **Where:** `orders.py:14`
- **Why:** latest_order accesses ordered[-1] without verifying the list is
non-empty, causing IndexError when orders is empty.
- **Fix:** Guard the empty case before indexing.
- **Seen by:** correctness-linescan, cross-file, removed-behavior ×5; votes: CONFIRMED 2; conf 1.00
Each finding carries provenance (Seen by: — which finder angles surfaced it),
adversarial verify votes, and a conf score, so you can filter noise with
--min-confidence.
Machine-readable output contract
--format json emits a stable, versioned envelope for downstream automation
(no need to scrape Markdown). The findings key stays top-level for backward
compatibility; schema_version / verdict / summary are additive.
{
"schema_version": 1,
"verdict": "green", // green = clean, yellow = non-blocking only, red = blocking finding
"summary": {
"total": 0,
"blocking": 0, // findings with severity >= HIGH
"by_severity": {}, // {"HIGH": 2, "LOW": 1, ...}
"incomplete": false // true if the review did not finish (a "meta" finding is present)
},
"findings": [ /* per-finding: dimension, severity, file, line, title, rationale,
suggestion, angles, votes, merge_count, confidence */ ]
}
Verdict ↔ exit-code invariant: verdict == "red" iff exit code is 1.
green and yellow both exit 0 (use verdict to distinguish clean from
advisory). summary.incomplete lets an actor tell "review failed to run" apart
from "real blocking findings."
Slash-command / Kagura Memory workflow
Install the shipped slash command into your project's .claude/commands/ directory:
# Copy the shipped slash command into your project so Claude Code can run /kagura-code-reviewer
mkdir -p .claude/commands
cp "$(python -c "import kagura_code_reviewer, pathlib; print(pathlib.Path(kagura_code_reviewer.__file__).parent / 'commands' / 'kagura-code-reviewer.md')")" .claude/commands/
Then inside Claude Code, run:
/kagura-code-reviewer
The command will:
- Retrieve this repository's pinned review policy and past findings from Kagura Memory (filtered to
trust_tier: trustedto guard against prompt injection). - Pass assembled context to the CLI via
--context-file. - Present the report.
- Write durable new conventions and recurring patterns back to memory.
Memory security contract
Memory is grounding, not authority. The CLI receives only a string via
--context-file and cannot re-verify its provenance, so callers are
responsible for these guarantees (defense in depth):
- Recall with
trust_tier: "trusted"— required. It excludes external/connector-ingested memories (Slack/Discord/etc.) that could carry injected instructions (OWASP LLM01/LLM03). The CLI assumes the caller has filtered; it does not (and cannot) check. - Injected memory is untrusted, reference-only data. The CLI fences it in
BEGIN/END UNTRUSTED MEMORY CONTEXTmarkers and the system prompt forbids obeying anything inside them. - Memory has no finding-suppression authority. The verdict is computed from
findings produced via the
submit_findingstool and the adversarial verify pass — never from prose or memory. Context can inform a finding's rationale but cannot remove a finding or change the verdict. - For autonomous use, only let owner-pinned memory influence gating
decisions; do not treat agent-authored
on_recallmemories as trusted for security-relevant gates (avoids a self-poisoning feedback loop).
Configuration
Model aliases and defaults are defined in the shipped config.toml:
default_alias = "review-cloud"
[models.review-cloud]
ollama_model = "qwen3-coder:480b-cloud"
base_url = "http://localhost:11434/v1"
num_ctx = 32768
[models.review-local]
ollama_model = "qwen2.5-coder:7b"
base_url = "http://localhost:11434/v1"
num_ctx = 16384
User override: create ~/.config/kagura-code-reviewer/config.toml (or set KAGURA_CODE_REVIEW_CONFIG to an alternate path). Only keys you set override the defaults; everything else inherits.
To add a custom alias:
default_alias = "my-model"
[models.my-model]
ollama_model = "deepseek-r1:70b"
base_url = "http://localhost:11434/v1"
num_ctx = 65536
Status / scope
v0.1 is synchronous: the CLI blocks until the Ollama review loop completes (up to --timeout seconds per call, --max-iters agent iterations).
The following are not yet implemented and are not claimed above:
--backgroundmode (fire-and-forget async review with a status poll command)- Memory-pattern sharing with
kagura-engineeror other kagura-* tools - MCP server mode
Design spec: docs/superpowers/specs/2026-06-06-kagura-code-review-design.md
Contributing
Issues and PRs are welcome. See CONTRIBUTING.md for the dev setup and workflow, and CODE_OF_CONDUCT.md. To report a security issue, follow SECURITY.md (please do not open a public issue for vulnerabilities).
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 kagura_code_reviewer-0.1.1.tar.gz.
File metadata
- Download URL: kagura_code_reviewer-0.1.1.tar.gz
- Upload date:
- Size: 105.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
067d1a07492c71bd80d8c40f3c2b28ece54789a9b2e5bd673800c4b3eaf40550
|
|
| MD5 |
16c35579496ce0c3ed87a0e826f6ec7f
|
|
| BLAKE2b-256 |
9451961c08beb110b925d9dd9b6d25a3048c94a88175f2de23cb05ce1e0910aa
|
Provenance
The following attestation bundles were made for kagura_code_reviewer-0.1.1.tar.gz:
Publisher:
publish.yml on kagura-ai/kagura-code-reviewer
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
kagura_code_reviewer-0.1.1.tar.gz -
Subject digest:
067d1a07492c71bd80d8c40f3c2b28ece54789a9b2e5bd673800c4b3eaf40550 - Sigstore transparency entry: 1749029230
- Sigstore integration time:
-
Permalink:
kagura-ai/kagura-code-reviewer@e18c8a5e3b3f24b6d76e76e6cdaca9bbf5059d1b -
Branch / Tag:
refs/tags/v0.1.1 - Owner: https://github.com/kagura-ai
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@e18c8a5e3b3f24b6d76e76e6cdaca9bbf5059d1b -
Trigger Event:
push
-
Statement type:
File details
Details for the file kagura_code_reviewer-0.1.1-py3-none-any.whl.
File metadata
- Download URL: kagura_code_reviewer-0.1.1-py3-none-any.whl
- Upload date:
- Size: 37.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
d2e785f7053c036e5938b481be3e87cfd631afd83f9ff0494406e9d9985f3874
|
|
| MD5 |
671ae3a20080d31e40cab29b18bffd7d
|
|
| BLAKE2b-256 |
2884bdd4a8b489be2840bce4b74ae0282549056496096544bc2c47c2cc7d25a4
|
Provenance
The following attestation bundles were made for kagura_code_reviewer-0.1.1-py3-none-any.whl:
Publisher:
publish.yml on kagura-ai/kagura-code-reviewer
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
kagura_code_reviewer-0.1.1-py3-none-any.whl -
Subject digest:
d2e785f7053c036e5938b481be3e87cfd631afd83f9ff0494406e9d9985f3874 - Sigstore transparency entry: 1749029937
- Sigstore integration time:
-
Permalink:
kagura-ai/kagura-code-reviewer@e18c8a5e3b3f24b6d76e76e6cdaca9bbf5059d1b -
Branch / Tag:
refs/tags/v0.1.1 - Owner: https://github.com/kagura-ai
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@e18c8a5e3b3f24b6d76e76e6cdaca9bbf5059d1b -
Trigger Event:
push
-
Statement type: