skills/pr-review/SKILL.md
This skill processes unresolved GitHub PR review discussions. Activated when the user provides a GitHub PR link (github.com/.../pull/...) or mentions "PR review", "review PR", "process review comments", etc.
npx skillsauth add pitzcarraldo/skills pr-reviewInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
This skill automatically processes unresolved GitHub Pull Request review discussions by analyzing the code context, validating feedback, and taking appropriate actions. It intelligently categorizes feedback as valid, invalid, or already addressed, then creates action plans or posts clarifying responses accordingly.
| Action Type | When Used | Outcome | |-------------|-----------|---------| | Create Todo | Valid feedback needing fixes | Adds task to todo list | | Reply & Resolve | Invalid or misunderstood feedback | Posts clarifying comment, resolves thread | | Confirm & Resolve | Already fixed feedback | Posts confirmation, resolves thread | | Escalate | Ambiguous or complex feedback | Asks user for guidance |
Legitimate issues that require code changes:
Misunderstandings or incorrect observations:
Feedback that has been fixed since review:
Check if gh CLI is available:
gh --version
Expected output:
gh version X.Y.Z (YYYY-MM-DD)If not installed, display installation guide:
GitHub CLI (gh) is not installed.
Installation:
macOS: brew install gh
Ubuntu/Debian: sudo apt install gh
Windows: winget install GitHub.cli
Linux (other): See https://github.com/cli/cli#installation
After installation, authenticate:
gh auth login
Verify authentication:
gh auth status
If not authenticated:
Please authenticate with GitHub:
gh auth login
Follow the prompts to complete authentication.
Extract repository and PR information from argument:
URL format: https://github.com/{owner}/{repo}/pull/{number}
Extraction logic:
url="$ARGUMENTS"
owner=$(echo "$url" | sed -n 's#.*/github.com/\([^/]*\)/.*#\1#p')
repo=$(echo "$url" | sed -n 's#.*/github.com/[^/]*/\([^/]*\)/.*#\1#p')
pr_number=$(echo "$url" | sed -n 's#.*/pull/\([0-9]*\).*#\1#p')
Example:
Input: https://github.com/anthropics/claude-code/pull/123
Output: owner=anthropics, repo=claude-code, pr_number=123
Validation:
If URL invalid:
Error: Invalid GitHub PR URL
Expected format: https://github.com/OWNER/REPO/pull/NUMBER
Example: https://github.com/anthropics/claude-code/pull/123
Query GitHub GraphQL API for review threads:
gh api graphql -f query='
query($owner: String!, $repo: String!, $pr: Int!) {
repository(owner: $owner, name: $repo) {
pullRequest(number: $pr) {
title
author { login }
reviewThreads(first: 100) {
nodes {
id
isResolved
isOutdated
path
line
comments(first: 10) {
nodes {
id
body
author { login }
createdAt
}
}
}
}
}
}
}
' -f owner="$owner" -f repo="$repo" -F pr="$pr_number"
Response structure:
{
"data": {
"repository": {
"pullRequest": {
"reviewThreads": {
"nodes": [
{
"id": "PRRT_...",
"isResolved": false,
"isOutdated": false,
"path": "src/auth.ts",
"line": 45,
"comments": {
"nodes": [
{
"body": "This needs null checking",
"author": {"login": "reviewer"},
"createdAt": "2024-01-01T12:00:00Z"
}
]
}
}
]
}
}
}
}
}
Filter for unresolved threads:
isResolved: falseisOutdated: true threads (may be relevant)If no unresolved discussions:
✓ All review discussions have been resolved
PR: [PR Title]
Status: No pending feedback
For each unresolved thread, perform comprehensive analysis:
Fetch the file at the specified path:
# Read the file with context around the line
Read file_path="$path"
Extract relevant section:
Get the complete PR diff:
gh pr diff "$pr_number" -R "$owner/$repo"
Parse diff for the specific file:
Analyze all comments in the thread:
Determine the current state:
Has code changed since review?
Does current code address the concern?
Is feedback contextually correct?
Identification:
Action:
# Add to todo list using TodoWrite
TodoWrite:
- content: "Fix [issue] in [file]:[line]"
- status: "pending"
Example:
Discussion: "Missing null check for user.profile"
Current code: No null check present
Category: Valid Feedback
Action: Added todo "Add null check for user.profile in auth.ts:45"
Identification:
Action:
# Post clarifying comment
gh api graphql -f query='
mutation($threadId: ID!, $body: String!) {
addPullRequestReviewThreadReply(input: {
pullRequestReviewThreadId: $threadId
body: $body
}) {
comment { id }
}
}
' -f threadId="$thread_id" -f body="[polite clarification]"
# Resolve the thread
gh api graphql -f query='
mutation($threadId: ID!) {
resolveReviewThread(input: {threadId: $threadId}) {
thread { isResolved }
}
}
' -f threadId="$thread_id"
Comment template:
Thank you for the feedback! This is actually already handled:
[Explanation with code reference]
The [specific mechanism] ensures [desired behavior].
Reference: [file]:[line]
Example:
Discussion: "Variable 'token' is unused"
Current code: Variable used on line 67
Category: Invalid Feedback
Reply: "Thank you for checking! The 'token' variable is actually used
on line 67 in the authentication middleware. Here's the usage:
```typescript
if (token) {
return validateToken(token);
}
```"
Action: Resolved thread
Identification:
Action:
# Post confirmation comment
gh api graphql -f query='[same mutation as Type B]' \
-f threadId="$thread_id" \
-f body="This has been addressed in commit [hash]. [Brief explanation]"
# Resolve thread
gh api graphql -f query='[same resolve mutation]' -f threadId="$thread_id"
Comment template:
This has been addressed in commit [short-hash].
Changes made:
[Brief summary of the fix]
Current implementation:
[Code snippet if helpful]
Example:
Discussion: "Add TypeScript types for User interface"
Current code: User interface fully typed
Category: Already Addressed
Reply: "This has been addressed in commit abc123f. Added comprehensive
TypeScript types for the User interface including optional fields."
Action: Resolved thread
Identification:
Action:
# Ask user for guidance using AskUserQuestion
AskUserQuestion:
question: "[Reviewer] suggests [approach]. How would you like to proceed?"
options:
- Implement suggested approach
- Explain current approach to reviewer
- Discuss alternative solution
After processing all discussions, create summary:
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
PR Review Processing Complete
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
PR: [PR Title] (#[number])
Repository: [owner]/[repo]
Summary:
Total discussions: [X]
Valid feedback: [Y] → Added to todo list
Invalid feedback: [Z] → Clarified and resolved
Already addressed: [W] → Confirmed and resolved
Escalated: [N] → Requires user input
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
Action Items Created:
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
1. [file]:[line] - [description]
2. [file]:[line] - [description]
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
Resolved Discussions:
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
✓ [file]:[line] - [reason for resolution]
✓ [file]:[line] - [reason for resolution]
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
수정사항(Valid Feedback)이 있는 경우 모든 discussion이 해결될 때까지 다음 사이클을 반복합니다:
Todo 항목에 따라 코드 수정 후 push:
# 변경사항 커밋
git add .
git commit -m "fix: address PR review feedback"
# Push
git push
PR의 모든 check가 완료될 때까지 대기:
# Check 상태 모니터링 (완료될 때까지 polling)
gh pr checks "$pr_number" -R "$owner/$repo" --watch
Check 상태 확인:
gh pr checks "$pr_number" -R "$owner/$repo"
예상 출력:
Some checks are still pending
0 failing, 1 pending, 0 passing, and 0 skipped checks
build In progress https://github.com/...
Check 완료 대기 로직:
--watch 플래그로 실시간 모니터링Check 완료 후 새로운 unresolved discussion 조회:
# 3단계의 GraphQL 쿼리 재실행
gh api graphql -f query='
query($owner: String!, $repo: String!, $pr: Int!) {
repository(owner: $owner, name: $repo) {
pullRequest(number: $pr) {
reviewThreads(first: 100) {
nodes {
id
isResolved
isOutdated
path
line
comments(first: 10) {
nodes {
id
body
author { login }
createdAt
}
}
}
}
}
}
}
' -f owner="$owner" -f repo="$repo" -F pr="$pr_number"
새 discussion 발견 시:
루프 종료 조건:
최종 완료 메시지:
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
✓ PR Review 처리 완료
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
PR: [PR Title] (#[number])
Repository: [owner]/[repo]
처리 사이클: [N]회
총 처리 discussion: [X]개
CI Check: ✓ 모두 통과
모든 review feedback이 처리되었습니다.
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
┌─────────────────────────────────────────────────────────┐
│ 시작 │
└────────────────────────┬────────────────────────────────┘
▼
┌─────────────────────────────────────────────────────────┐
│ Unresolved discussion 조회 │
└────────────────────────┬────────────────────────────────┘
▼
┌─────────────────────┐
│ discussion 있음? │
└──────────┬──────────┘
│ │
Yes No
│ │
▼ ▼
┌──────────────────────┐ ┌─────────────────────────┐
│ 분석 및 분류 (4단계) │ │ ✓ 처리 완료 │
└──────────┬───────────┘ │ 루프 종료 │
▼ └─────────────────────────┘
┌──────────────────────┐
│ 처리 및 응답 (5단계) │
│ - Valid: 수정 │
│ - Invalid: resolve │
│ - Addressed: resolve │
└──────────┬───────────┘
▼
┌───────────┐
│ 수정 있음? │
└─────┬─────┘
Yes │
▼
┌──────────────────────┐
│ git commit && push │
└──────────┬───────────┘
▼
┌──────────────────────┐
│ CI Check 대기 │
│ gh pr checks --watch │
└──────────┬───────────┘
▼
│ (처음으로 돌아감)
└────────────────────────────────────────┐
│
┌─────────────────────────────────────┘
▼
┌─────────────────────────────────────────────────────────┐
│ Unresolved discussion 재조회 │
└─────────────────────────────────────────────────────────┘
Review comment:
src/auth.ts:45
"This function should handle null user profiles"
Analysis:
# Read file
Read src/auth.ts
# Check lines 35-55
function getProfile(user) {
return user.profile.name; // Line 45 - no null check
}
Outcome:
Category: Valid Feedback
Action: Added todo "Add null check for user.profile in auth.ts:45"
Review comment:
src/api.ts:120
"Variable 'cache' is declared but never used"
Analysis:
# Read file
Read src/api.ts
# Check lines 110-140
const cache = new Map(); // Line 120
function getData(key) {
if (cache.has(key)) { // Line 130 - using cache
return cache.get(key);
}
// ...
}
Outcome:
Category: Invalid Feedback
Reply posted:
"Thank you for reviewing! The 'cache' variable is actually used in the
getData function on lines 130-132 for memoization. Here's the usage:
```typescript
if (cache.has(key)) {
return cache.get(key);
}
The cache improves performance by storing previously fetched data."
Action: Resolved thread
### Example 3: Already Fixed Type Issue
**Review comment:**
src/types.ts:15 "Missing return type annotation"
**Analysis:**
```bash
# Check current code
Read src/types.ts
# Line 15 now has type annotation
function calculate(): number { // Line 15 - type added
return 42;
}
# Check git log
git log --oneline --since="[review date]" src/types.ts
# Shows: "a1b2c3d feat(types): add return type annotations"
Outcome:
Category: Already Addressed
Reply posted:
"This has been addressed in commit a1b2c3d. Added return type annotations
to all exported functions including this one.
Current implementation:
```typescript
function calculate(): number {
return 42;
}
```"
Action: Resolved thread
| Tool | Purpose | Installation | Check Command |
|------|---------|--------------|---------------|
| GitHub CLI | API access | brew install gh | gh --version |
| Git | Local repository | Built-in or package manager | git --version |
| jq | JSON parsing | brew install jq | jq --version |
Error: gh: command not found
brew install gh or platform equivalentError: To get started with GitHub CLI, run: gh auth login
gh auth login and follow promptsError: GraphQL: Could not resolve to a PullRequest
Error: Resource not accessible by personal access token
gh auth refresh -s repoError: API rate limit exceeded
gh api rate_limit)For PRs with many discussions:
# Process in batches of 10
for i in {0..9}; do
# Process discussion $i
# Add delay to avoid rate limiting
sleep 1
done
Before posting, optionally show draft responses to user:
Draft response for [file]:[line]:
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
[Response text]
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
Post this response? [y/n]
Process discussions from specific reviewer:
# Filter threads by comment author
jq '.data.repository.pullRequest.reviewThreads.nodes[] |
select(.comments.nodes[0].author.login == "specific-reviewer")'
development
Review a tech spec document written in the team's Notion template format (Summary, Background, Goals, Non-Goals, Plan, Measuring Impact, Security/Privacy/Risks, Other Considerations, Milestones, Open Questions). Use when the user asks to "테크 스펙 리뷰", "tech spec 리뷰", "스펙 문서 리뷰", "이 스펙 봐줘", or provides a Notion/Google Docs/Markdown tech spec link or file and asks for feedback, critique, or readiness check before sharing with the team.
tools
Draft a daily standup from the current user's Slack, GitHub, Linear, and Google Calendar activity. Use when the user asks to create, prepare, or summarize a standup/status update from connected app activity; to collect a user's work for a date; or to verify whether Slack, GitHub, Linear, and Google Calendar connectors/MCP tools are available before drafting a standup.
development
List and resume previous coding agent sessions by loading their context into the current session. Use when the user wants to (1) see previous sessions for the current directory with summaries, (2) continue work from a previous session by loading its full conversation context, (3) recover work from an interrupted session, or (4) transfer context between sessions. Supports Claude Code and Codex sessions.
development
Pre-push CodeRabbit review-fix loop. Runs local CodeRabbit review, fixes valid issues, and repeats until clean so the branch passes CodeRabbit review on push with no additional comments. Use when the user says "review and fix", "review loop", or wants to ensure the branch is CodeRabbit-clean before pushing.