Propel Code Review Smoke
Run async diff-based code reviews using the Propel Review API, poll for completion, retrieve structured findings, and send comment feedback. Use when reviewi...
Description
name: propel-code-review description: Run async diff-based code reviews using the Propel Review API, poll for completion, retrieve structured findings, and send comment feedback. Use when reviewing the current branch diff or any Git diff with Propel. metadata: {"clawdbot":{"requires":{"env":["PROPEL_API_KEY"],"bins":["curl","git","jq"]},"primaryEnv":"PROPEL_API_KEY","homepage":"https://www.propelcode.ai/"}}
Propel Review API Skill
Use this guide to interact with the Propel Review API from an AI agent. Always target the production API unless told otherwise.
Purpose
Run async, diff-based code reviews via the production API and retrieve comments.
Quick Permission Check
Before rollout to a new workspace/repo, run this from the target repository root:
plugins/propel-code-review/skills/propel-code-review/scripts/smoke_test_permissions.sh
It validates:
- good token + connected repo
- good token + unconnected repo
- invalid token + connected repo
Pre-flight: Verify API Key
Before making any API call, check whether PROPEL_API_KEY is set:
if [ -n "$PROPEL_API_KEY" ]; then echo "PROPEL_API_KEY is set"; else echo "PROPEL_API_KEY is not set"; fi
If the variable is empty, unset, or you just received a 401/403 from the Review
API, do not attempt any API calls with the current value. Follow these steps
to capture a fresh token — each step is a separate action:
Step 1 — Tell the user and open the browser. Send this message and run the Bash command in the same response (in parallel):
Message to user:
PROPEL_API_KEYis not set. Opening the token creation page: https://app.propelcode.ai/administration/settings?tab=review-api-tokens&token_name=Claude+Code&scopes=reviews:read,reviews:write The name and scopes are pre-filled. Click Create token, copy it, and paste it here.
Bash command:
URL="https://app.propelcode.ai/administration/settings?tab=review-api-tokens&token_name=Claude+Code&scopes=reviews:read,reviews:write"
if command -v xdg-open >/dev/null; then xdg-open "$URL"; else open "$URL"; fi
Step 2 — Wait for the user to paste the token. Do not proceed until the user
pastes a value starting with rev_. If the value doesn't start with rev_, tell
them it doesn't look valid and ask them to try again.
Step 3 — Once you have a valid token, persist it and load it into the session.
Run this in a single Bash call (replace <TOKEN> with the actual token):
case "$SHELL" in */zsh) SHELL_RC="$HOME/.zshrc" ;; */bash) SHELL_RC="$HOME/.bashrc" ;; *) SHELL_RC="" ;; esac; if [ -z "$SHELL_RC" ] && [ -f "$HOME/.zshrc" ]; then SHELL_RC="$HOME/.zshrc"; fi; if [ -z "$SHELL_RC" ] && [ -f "$HOME/.bashrc" ]; then SHELL_RC="$HOME/.bashrc"; fi; if [ -n "$SHELL_RC" ]; then printf '\n# Propel Review API token\nexport PROPEL_API_KEY="%s"\n' "<TOKEN>" >> "$SHELL_RC" && echo "Saved to $SHELL_RC"; else echo "No shell profile found"; fi; export PROPEL_API_KEY="<TOKEN>"
Tell the user where the key was saved (e.g. "Saved to ~/.zshrc").
Step 4 — Continue with the review workflow.
Setup (Manual)
If you prefer to set the token yourself ahead of time:
export PROPEL_API_KEY="rev_..."
The token must be a Review API token (scoped to both reviews:write and reviews:read).
Base URL
https://api.propelcode.ai
Authentication
Use a bearer token in the Authorization header:
Authorization: Bearer $PROPEL_API_KEY
Endpoints (Only These)
Do not assume any other Review APIs exist. Only use the async endpoints below.
Create Review (Async)
POST /v1/reviews
Request body:
{
"diff": "string (required)",
"repository": "string (required)",
"base_commit": "string (required)",
"head_commit_sha": "string (optional)",
"branch": "string (optional)"
}
Constraints:
diffmax size: 1,000,000 bytesrepositorymax length: 255base_commitmax length: 255head_commit_shamax length: 255branchmax length: 255
Notes:
base_commitshould be a commit that exists in the remote repo history (typically the base commit of the branch you are reviewing).repositoryshould be the canonical repo slug (for example,owner/repo) derived from the git remote URL.
Response (202):
{
"review_id": "uuid",
"status": "queued",
"repository": "owner/repo",
"base_commit": "sha",
"created_at": "...",
"updated_at": "..."
}
Get Review Status/Results
GET /v1/reviews/:review_id
Response (200):
{
"review_id": "uuid",
"status": "queued|running|completed|failed",
"poll_after_ms": 3000,
"comments": [
{
"comment_id": "string",
"file_path": "path",
"line": 123,
"message": "...",
"severity": "error|warning|info"
}
],
"error": {
"code": "generation_failed",
"message": "..."
}
}
Post Comment Feedback
POST /v1/reviews/:review_id/comments/feedback
Request body:
{
"comment_id": "string (required)",
"incorporated": true,
"notes": "string (optional)"
}
Response (200):
{
"review_id": "uuid",
"comment_id": "string",
"incorporated": true
}
Scripts Used by This Skill
Use the helper scripts in scripts/ instead of ad-hoc inline curl loops.
Paths below are relative to this skill directory:
scripts/create_review.shwrapsPOST /v1/reviews.scripts/poll_review.shwraps pollingGET /v1/reviews/:review_iduntil terminal status (completedorfailed) or timeout.scripts/post_comment_feedback.shwrapsPOST /v1/reviews/:review_id/comments/feedback.
Approval-Friendly Prefixes (One-Time)
If your client supports prefix-based trust/approval, approve these once before running this skill:
scripts/create_review.shscripts/poll_review.shscripts/post_comment_feedback.shgit diffgit rev-parsegit remote get-urlgh pr viewjq
Workflow (Recommended)
- Resolve the base branch (PR base when available; otherwise remote default branch):
BASE_BRANCH=$(gh pr view --json baseRefName -q '.baseRefName' 2>/dev/null || git remote show origin | sed -n '/HEAD branch/s/.*: //p')
- Compute the base commit (must exist in the remote repo history):
git rev-parse "$BASE_BRANCH"
- Compute the current head commit and branch (for local review dedup context):
git rev-parse HEADgit rev-parse --abbrev-ref HEAD
- Compute the repository slug:
git remote get-url origin | sed -E 's#(git@github.com:|https://github.com/)##; s/\\.git$//'
- Generate the diff:
git diff "$BASE_BRANCH" > /tmp/review_api.diff
- Call
scripts/create_review.shwith diff, base commit, repository slug, head commit, and branch. - Handle create-review failures before polling (script exits non-zero and
prints the API response body):
401/403: token invalid/expired/missing scope. Stop and ask user to refresh token.404: repository is not connected to the Propel workspace (or slug is wrong). Stop and ask user to connect/fix repo slug.400/413: invalid request or diff too large. Stop and show actionable fix.5xx: transient API error. Retry with bounded backoff, then stop and report if still failing.
- Poll with
scripts/poll_review.shuntil status iscompletedorfailed. The script keeps the existing 15 minute timeout budget, and honorspoll_after_msfrom the API when present. - Present comments to the user with file/line context.
- For each comment, determine whether it is valid and applicable to the code.
- If valid, incorporate the change in the codebase. If invalid, do not change the codebase.
- Immediately call
scripts/post_comment_feedback.shfor each comment withcomment_id,incorporatedtrue/false, and briefnotesexplaining the decision. Do not wait for user confirmation.
Example (Production)
BASE_BRANCH=$(gh pr view --json baseRefName -q '.baseRefName' 2>/dev/null || git remote show origin | sed -n '/HEAD branch/s/.*: //p')
BASE_COMMIT=$(git rev-parse "$BASE_BRANCH")
HEAD_COMMIT=$(git rev-parse HEAD)
BRANCH=$(git rev-parse --abbrev-ref HEAD)
REPO_SLUG=$(git remote get-url origin | sed -E 's#(git@github.com:|https://github.com/)##; s/\\.git$//')
git diff "$BASE_BRANCH" > /tmp/review_api.diff
CREATE_RESPONSE=$(
scripts/create_review.sh \
--diff-file /tmp/review_api.diff \
--repo "$REPO_SLUG" \
--base-commit "$BASE_COMMIT" \
--head-commit-sha "$HEAD_COMMIT" \
--branch "$BRANCH"
)
REVIEW_ID=$(echo "$CREATE_RESPONSE" | jq -r '.review_id // empty')
if [ -z "$REVIEW_ID" ]; then
echo "$CREATE_RESPONSE"
exit 1
fi
scripts/poll_review.sh \
--review-id "$REVIEW_ID" \
--max-attempts 30 \
--sleep-seconds 30 \
--output-file /tmp/review_api.result.json
# The poller uses the 30x30s budget above, but will follow API-provided
# poll_after_ms hints when the server returns them.
cat /tmp/review_api.result.json
# Example feedback posts after deciding incorporate true/false per comment.
jq -c '.comments[]?' /tmp/review_api.result.json | while read -r comment; do
COMMENT_ID=$(echo "$comment" | jq -r '.comment_id // empty')
if [ -z "$COMMENT_ID" ]; then
continue
fi
scripts/post_comment_feedback.sh \
--review-id "$REVIEW_ID" \
--comment-id "$COMMENT_ID" \
--incorporated true \
--notes "Applied in this branch."
done
Troubleshooting
401/403— re-run the pre-flight check above. The token may be missing, expired, or missing scopes. Guide the user to generate a new one at: https://app.propelcode.ai/administration/settings?tab=review-api-tokens&token_name=Claude+Code&scopes=reviews:read,reviews:write404 {"error":"Repository not found"}— the repository string does not match a repo connected to the account. Treat this as an access/config problem, not a retryable failure.413— the diff exceeded the 1,000,000 byte limit.
Permission Handling Contract
If review creation returns 401, 403, or 404, do all of the following:
- Stop the review flow immediately (do not poll and do not retry in a loop).
- Report the exact repository slug used.
- Report an actionable next step:
401/403: "refresh token and confirmreviews:read+reviews:writescopes".404: "connect this repository in Propel workspace, or correct owner/repo slug".
- Mark the run as blocked until user intervention.
Notes for Agents
- Do not log or expose tokens in output.
- Always use
https://api.propelcode.aiuntil told otherwise. - Use
scripts/create_review.sh,scripts/poll_review.sh, andscripts/post_comment_feedback.shinstead of inline curl commands. - The agent must decide whether each comment is valid, incorporate fixes when
valid, and report feedback automatically via the feedback endpoint using the
comment_idfrom the review response (no user confirmation required).
Reviews (0)
No reviews yet. Be the first to review!
Comments (0)
No comments yet. Be the first to share your thoughts!