"Everything's coming up Thrillhouse!"
A GraalVM-native PR review bot, built as a GitHub App with Quarkus. It reviews pull requests using any OpenAI-compatible chat API, so the review is language-agnostic and you can pick the provider that suits you.
See how it compares to CodeRabbit, PR-Agent, and Copilot code review.
- Reviews diffs for correctness, security, regressions, stale comments, and code quality
- Configurable auto-review triggers — skip drafts, gate on labels, or filter by base branch
- Inline code suggestions on review comments that you can apply with one click
- Every finding is tagged
critical,high,medium, orlow - Follow-up reviews track whether earlier findings were addressed or justified
- Conversational replies:
@thrillhousebotit in a PR thread or finding reply and the bot answers in context - A summary comment on the first run, with a risk breakdown
- Operable from the PR with comment commands —
/help,/review,/summary,/resolve,/pause,/resume - Live dashboard (Next.js) with a WebSocket activity feed, cost charts, and token tracking
- OpenTelemetry traces, token histograms, cost counters, and latency metrics
- Reads per-repo instructions from
.github/thrillhousebot.md, falling back to Copilot/Claude/Agents files - Compiles ahead-of-time with GraalVM/Mandrel, so it starts fast and stays small
ThrillhouseBot talks to any endpoint that implements the OpenAI chat-completions
API. Point AI_BASE_URL and AI_MODEL at your provider of choice:
| Provider | AI_BASE_URL |
Example AI_MODEL |
|---|---|---|
| DeepSeek | https://api.deepseek.com/v1 |
deepseek-chat |
| OpenRouter | https://openrouter.ai/api/v1 |
openai/gpt-4o-mini |
| Alibaba Cloud (Model Studio) | https://dashscope-intl.aliyuncs.com/compatible-mode/v1 |
qwen-plus |
| OpenAI | https://api.openai.com/v1 |
gpt-4o-mini |
| Ollama (local) | http://localhost:11434/v1 |
llama3.2 |
The default is DeepSeek, used only because it is inexpensive; nothing in the bot is tied to it.
Drive the bot directly from a PR by commenting one of these. Each also accepts the
mention form, e.g. @Thrillhousebot review.
| Command | What it does | Access |
|---|---|---|
/help |
List the available commands | anyone |
/review |
Run (or re-run) a full review of the PR | write |
/summary |
Post the PR summary, but only if one has not been generated yet (otherwise no-op) | write |
/resolve |
Resolve ThrillhouseBot's outstanding finding threads on the PR | write |
/pause |
Silence the bot on the PR | write |
/resume |
Re-enable the bot on a paused PR | write |
Access — every command except /help requires the commenter to hold write access to
the repository (or to be named in
THRILLHOUSEBOT_REVIEW_MANUAL_TRIGGER_ALLOWED_LOGINS), since reviews spend the operator's
AI budget.
Pause — while a PR is paused, ThrillhouseBot skips automatic reviews on new commits,
ignores /review and /summary, and does not answer @thrillhousebot mentions (it replies
once to say it is paused). /resume lifts the pause. /help and /resolve keep working while
paused.
- Docker & Docker Compose
- An API key for any OpenAI-compatible provider
Follow the GitHub App setup section below (2 minutes). You'll get an App ID, private key, webhook secret, and OAuth client ID/secret.
git clone https://github.com/devops-thiago/ThrillhouseBot.git && cd ThrillhouseBot
cp .env.example .envEdit .env with the credentials from step 1:
| Variable | Value |
|---|---|
GITHUB_APP_ID |
From GitHub App settings → About |
GITHUB_PRIVATE_KEY |
Downloaded when you generated a private key |
GITHUB_WEBHOOK_SECRET |
The webhook secret you set |
GITHUB_CLIENT_ID |
From app settings → Identifying and authorizing users |
GITHUB_CLIENT_SECRET |
From app settings → Identifying and authorizing users |
AI_API_KEY |
Your AI provider's API key |
docker compose up -dThe bot is running on http://localhost:8080. Point your reverse proxy at it and
you're done.
Create a GitHub App before starting the bot; you'll need its credentials for .env.
-
Edit
manifest.jsonin the repo root and replace every<your-host>with your public hostname (no trailing slash). For local dev with Smee.io, use your Smee channel URL host. -
Serve the repo root locally:
java -m jdk.httpserver -p 8081
-
Open http://localhost:8081/install.html and click Create ThrillhouseBot GitHub App. GitHub creates the app from your manifest.
-
On the confirmation page, note the App ID, generate a private key, and create a webhook secret. Copy the Client ID and Client secret from the app's Identifying and authorizing users settings (needed for dashboard login).
-
Install the app on your account or organization, then copy the values into
.env.Alternatively, generate
.envautomatically from the manifest conversion response:gh api --method POST /app-manifests/<code>/conversions \ | java scripts/GenEnv.java --host <your-host>
Once the bot is running,
http://<your-host>:8080/install.htmlauto-detects the URL and builds the manifest dynamically, with no file editing or local server needed.
| Setting | Value |
|---|---|
| Webhook URL | https://<your-host>/api/webhook |
| Webhook Secret | Random string |
| Repository Permissions | Pull Requests: R/W, Checks: R/W, Contents: Read, Issues: R/W, Actions: Read, Commit Statuses: Read |
| Subscribe to Events | Pull Request, Issue comment, Pull request review comment |
| Identifying & authorizing users | Enabled (for dashboard login) |
| Callback URL | https://<your-host>/api/auth/callback |
Configuration is read from environment variables (see .env.example). The AI
variables are the ones you will change per provider:
| Variable | Purpose | Default |
|---|---|---|
AI_API_KEY |
API key for the AI provider | (required) |
AI_BASE_URL |
OpenAI-compatible base URL | https://api.deepseek.com/v1 |
AI_MODEL |
Chat model name | deepseek-chat |
AI_PROVIDER |
Provider label for telemetry (gen_ai.provider.name); derived from AI_BASE_URL when unset |
(derived) |
AI_TIMEOUT |
Per-request timeout | 300s |
GITHUB_APP_ID |
GitHub App ID | (required) |
GITHUB_PRIVATE_KEY |
GitHub App private key (PEM) | (required) |
GITHUB_WEBHOOK_SECRET |
Webhook HMAC secret | (required) |
GITHUB_BOT_LOGINS |
Comma-separated bot account login(s) the bot skips to avoid replying to itself; override when deployed under a different App slug (<app-slug>[bot]) |
thrillhousebot[bot],thrillhouse-bot[bot] |
WEBHOOK_DEDUP_TTL |
Webhook deduplication time-to-live for GitHub redeliveries | 24h |
THRILLHOUSEBOT_REVIEW_MANUAL_TRIGGER_ALLOWED_LOGINS |
Comma-separated allowlist of logins permitted to trigger manual /review without repo access |
(empty) |
MANUAL_TRIGGER_AUTH_TIMEOUT |
Upper bound on the manual-trigger write-access check on the webhook ACK thread; fails closed (denies) if GitHub is slower | 5s |
WEBHOOK_SKIP_DRAFTS |
Skip auto-review while a PR is a draft (reviewed once marked ready / on later pushes) | false |
WEBHOOK_REQUIRED_LABELS |
Comma-separated labels; only auto-review PRs carrying at least one (case-insensitive) | (empty — no gate) |
WEBHOOK_EXCLUDED_LABELS |
Comma-separated labels; skip auto-review of PRs carrying any (wins over required) | (empty) |
WEBHOOK_BASE_BRANCHES |
Comma-separated globs; only auto-review PRs whose base branch matches one (e.g. main,release/*). Globs are gitignore-style: * does not cross /, so use ** to span slashes (** alone matches every branch) |
(empty — all branches) |
WEBHOOK_IGNORED_BASE_BRANCHES |
Comma-separated globs; skip auto-review of PRs whose base branch matches one (wins over allowlist; same */** rule — match nested branches with **, e.g. dependabot/**) |
(empty) |
REVIEW_CONVERSATIONAL_REPLIES_ENABLED |
Answer @thrillhousebot mentions in PR threads (including finding replies) with an AI reply |
true |
REVIEW_LABELS_ENABLED |
Opt in to context-aware PR labels (see PR labels) | false |
REVIEW_LABELS_APPLY |
When labels are enabled, add them to the PR instead of only suggesting them in a comment | false |
REVIEW_LABELS_ALLOW_CREATE |
Allow the bot to create suggested labels that don't exist yet | false |
REVIEW_LABELS_MAX |
Maximum labels applied or suggested per PR | 3 |
GITHUB_CLIENT_ID / GITHUB_CLIENT_SECRET |
OAuth credentials for dashboard login | (required for dashboard) |
DASHBOARD_URL |
Public dashboard URL (OAuth callback base) | http://localhost:8080 |
DATASOURCE_DB_KIND |
h2 or postgresql |
h2 (dev), postgresql (%prod) |
HTTP_CONNECT_TIMEOUT |
Outbound HTTP connect timeout (GitHub API, OAuth) | 10s |
HTTP_REQUEST_TIMEOUT |
Outbound HTTP request timeout (GitHub API, OAuth) | 10s |
WEBSOCKET_KEEPALIVE_MS |
Dashboard WebSocket keepalive interval in ms; 0 or negative disables it (and stale replay-buffer eviction) |
25000 |
The app validates configuration at startup and fails fast if a required value
(GITHUB_APP_ID, GITHUB_PRIVATE_KEY, GITHUB_WEBHOOK_SECRET, AI_API_KEY) is missing or — for
the private key — not a valid PEM RSA key, naming every offending variable in one message instead of
surfacing later on the first webhook or review. Dashboard OAuth (GITHUB_CLIENT_ID /
GITHUB_CLIENT_SECRET) is optional: leave both unset and the dashboard login is simply disabled.
Cost tracking uses per-model pricing keyed by the model name, for example:
thrillhousebot.ai.pricing.deepseek-chat.input-per-1k=0.00014
thrillhousebot.ai.pricing.deepseek-chat.output-per-1k=0.00028If you switch to a different AI_MODEL, add a matching
thrillhousebot.ai.pricing.<model>.* pair so the dashboard can compute cost.
After logging in with GitHub OAuth, the dashboard shows an overview page with summary cards and a live activity feed, plus tabs for cost charts by model, input/output token breakdowns, and a paginated session history with PR links.
Access is restricted: the GitHub App owner always has access, and any other login
must be a collaborator on at least one repository where the app is installed
(under that owner account). Everyone else sees an access-denied screen. The owner
is resolved from the app registration; set thrillhousebot.dashboard.github.account-owner
to pin it explicitly when auto-detection fails.
![]() |
![]() |
![]() |
![]() |
The Overview has summary cards, a recent-activity feed, and a live panel that streams the model's output as a review runs:
Place a .github/thrillhousebot.md file in any repo to customize the review:
## Review Priorities
1. Payment calculations must be exact; flag any floating-point usage
2. All DB queries must use the repository pattern, never raw SQL
## Known Gotchas
- The `price` field in Product is in cents, not dollarsFallback chain: .github/thrillhousebot.md → .github/copilot-instructions.md → CLAUDE.md → AGENTS.md → AGENT.md
ThrillhouseBot can suggest context-aware labels (area, change type, risk) drawn
from the diff. The feature is off by default; turn it on with
REVIEW_LABELS_ENABLED=true.
When enabled, the model is shown the repository's existing labels and picks the
few that best describe the PR — it only ever chooses from labels that already
exist, so it respects whatever label scheme the repo already uses. What happens
next depends on REVIEW_LABELS_APPLY:
false(default): the suggestions are posted as a one-line comment on the first review, leaving the decision to a maintainer.true: the labels are added to the PR automatically.
Set REVIEW_LABELS_ALLOW_CREATE=true to let the bot create a suggested label
that doesn't exist yet (off by default, so it never invents labels), and
REVIEW_LABELS_MAX to cap how many labels it applies or suggests (default 3).
Labelling is best-effort — a failure here never blocks or fails the review.
All telemetry is exported via OTLP:
| Signal | Metric |
|---|---|
| Traces | One span per LLM call with request/response events |
gen_ai.client.token.usage |
Histogram: input/output tokens |
gen_ai.client.operation.duration |
Histogram: latency in seconds |
thrillhouse.ai.cost.total |
Counter: USD cost by model |
Spans and metrics are tagged with gen_ai.provider.name, derived from AI_BASE_URL
(e.g. deepseek, openai, groq, openrouter). Loopback and unrecognized endpoints
report unknown; set AI_PROVIDER to label them (e.g. a local ollama or vllm server,
a proxy, or a self-hosted gateway).
AI review is advisory. The model can be wrong in both directions: it raises false positives and misses real bugs. Treat its findings as suggestions and confirm them yourself before acting.
Pull request diffs are sent to whatever endpoint you configure, so use an HTTPS endpoint with an API key, and read the provider's data-retention policy before sending it private code.
Set AI_API_KEY, GITHUB_PRIVATE_KEY, and the webhook secret through your
environment or a secret manager. Never commit them.
To report a vulnerability, see SECURITY.md.
This is still an early-stage project; the current constraints are:
- GitHub only — no GitLab or Bitbucket integration.
- Large diffs — the model sees at most
thrillhousebot.review.max-diff-linesdiff lines (default 5000). Later files are dropped; the last included file is cut at a hunk boundary with its code fence re-closed (only cut mid-hunk if a single hunk alone exceeds the budget). - Single process — OAuth login sessions and the live WebSocket replay buffer are in-memory (lost on restart). Review history and cost totals persist in PostgreSQL. Multiple replicas are unsupported.
- Dashboard access — GitHub OAuth required. Only the app account owner and
collaborators on installed repos can use the dashboard; no admin UI or guest mode.
If the app owner cannot be resolved from GitHub, the dashboard fails closed (denies all access) until
thrillhousebot.dashboard.github.account-owneris set. - Production database — container and native production builds use PostgreSQL
(
%prod). H2 is for localquarkus:devonly. - OpenAI-compatible APIs only — endpoints must implement the chat-completions API shape LangChain4j expects.
- Cost tracking — needs a
thrillhousebot.ai.pricing.<model>.*entry per model; otherwise cost shows as$0. - Review output caps — at most 50 inline PR comments per review
(
thrillhousebot.review.max-review-comments). Lockfiles,pom.xml, generated paths, andtarget/are skipped by default (thrillhousebot.review.ignored-files). - First GitHub App setup — before the bot is running, app creation still needs a
local static server or
gh api;/install.htmlon the running bot covers later installs. - Self-hosted — no managed offering from this project.
Release images are signed with cosign (keyless, via Sigstore) and carry build provenance attestations, as do the binary tarballs. To check a release before running it:
# Signature
cosign verify \
--certificate-identity-regexp='https://github.com/devops-thiago/ThrillhouseBot.*' \
--certificate-oidc-issuer='https://token.actions.githubusercontent.com' \
ghcr.io/devops-thiago/thrillhousebot:v0.1.0
# Provenance (image)
gh attestation verify oci://ghcr.io/devops-thiago/thrillhousebot:v0.1.0 \
--repo devops-thiago/ThrillhouseBot
# Provenance (a downloaded binary)
gh attestation verify thrillhousebot-v0.1.0-linux-amd64.tar.gz \
--repo devops-thiago/ThrillhouseBotFor local development without Docker, you'll need Java 25+, Node.js 20+ (dashboard),
and a Smee.io channel for webhook forwarding. Use ./mvnw
for Maven (wrapper included).
# Terminal 1: Smee proxy
smee -u https://smee.io/YOUR_CHANNEL -t http://localhost:8080/api/webhook
# Terminal 2: Quarkus dev mode
./mvnw quarkus:devcd frontend
npm install
npm run build
cp -r out/* ../src/main/resources/META-INF/resources/dashboard/./mvnw package -Pnative -DskipTests -Dquarkus.native.container-build=true./mvnw verify
./mvnw spotless:checkSee CONTRIBUTING.md for the full development workflow.
Questions and setup help belong in GitHub Discussions (see the pinned welcome post). Use Issues for bugs and feature requests.
See docs/ARCHITECTURE.md.
| Layer | Choice |
|---|---|
| Framework | Quarkus 3.36 (REST) |
| LLM | LangChain4j 1.10 (OpenAI-compatible API) |
| Frontend | Next.js 16 + React 19 (static export) |
| Database | H2 (dev) / PostgreSQL (prod) + Panache |
| Observability | OpenTelemetry |
| Native | GraalVM / Mandrel |
| Container | UBI9-micro (default) / distroless (-distroless) |
Published to GHCR from the same native binary:
ghcr.io/devops-thiago/thrillhousebot:latest— UBI9-micro (default).ghcr.io/devops-thiago/thrillhousebot:latest-distroless— distroless base (:v0.1.0-distroless, etc.).- Snapshot tags:
:snapshot,:v0.1.0-<sha>-snapshot,:<full-sha>(and-distrolessvariants).
Both flavours are multi-arch (linux/amd64, linux/arm64), signed with cosign, and carry build-provenance attestations (see Verifying a release).
Licensed under the Apache License 2.0 (SPDX: Apache-2.0).






