Complete issue → PR → merge lifecycle with readiness checks
This skill is the canonical Copilot-agent lifecycle for the Squad repository. It covers the full path from picking up a GitHub issue to merging a PR. Where older docs (templates, copilot-instructions, CONTRIBUTING.md) conflict with this skill, this skill takes precedence for Copilot agents.
For advanced scenarios (worktrees, multi-repo coordination), see .copilot/skills/git-workflow/SKILL.md.
✅ THIS SKILL COVERS:
❌ THIS SKILL DOES NOT COVER:
git-workflow skill)git-workflow.squad/skills/release-process).copilot/skills/reviewer-protocol)Read the issue. Understand the acceptance criteria before writing code.
Confirm capability fit. Check your capability profile in .squad/team.md. 🟢 = proceed. 🟡 = proceed but flag in PR. 🔴 = comment on issue and stop.
Branch from dev:
git fetch origin dev
git checkout dev
git rebase origin/dev
git checkout -b squad/{issue-number}-{slug}
squad/{issue-number}-{kebab-case-slug}squad/42-fix-login-validationmainMark in-progress (optional):
gh issue edit {number} --add-label "status:in-progress"
Make your changes. Follow the codebase conventions:
@ts-ignoreBuild:
npm run build
Test:
npm test
Type check:
npm run lint
Stage specific files only:
git add path/to/file1.ts path/to/file2.ts
git add ., git add -A, or git commit -aPre-push safety check:
# Verify file count matches intent (expect ≤10 files for most fixes)
git diff --cached --stat
# Verify NO unintended deletions
git diff --cached --diff-filter=D --name-only
If you see unexpected files or deletions, unstage them: git reset HEAD <file>
Single commit with issue reference:
git commit -m "Brief description of change
Closes #{issue-number}
Co-authored-by: Copilot <[email protected]>"
git rebase -i origin/dev and squash all into onegit reset --soft to squash — it picks up delta from dev and contaminates the commitAdd changeset (if required):
A changeset is required when your PR modifies files under packages/squad-sdk/src/ or packages/squad-cli/src/.
npx changeset add
This prompts for: which packages changed, bump type (patch/minor/major), summary.
Or create manually at .changeset/{descriptive-name}.md:
---
'@bradygaster/squad-cli': patch
---
Brief description of the change
If the changeset creates a second commit, squash it into your main commit.
Push:
git push -u origin squad/{issue-number}-{slug}
Create PR targeting dev:
gh pr create --repo bradygaster/squad --base dev \
--title "fix: brief description (#issue-number)" \
--body "Closes #{issue-number}
## Summary
What this PR does and why.
## Changes
- File-level description of changes
Co-authored-by: Copilot <[email protected]>"
Title conventions:
fix: description (#N)feat: description (#N)docs: description (#N)chore: description (#N)Labels:
fix, feat, docs, or repo-health for typesquad:{agent-name} if working as a squad memberskip-changelog only with reviewer approval (escape hatch)Scope rules by label:
repo-health: Only modify .github/, scripts/, root config, tests, docs. Never modify packages/*/src/fix or feat: May modify product source. Must include changeset when touching packages/*/src/An automated readiness check runs on every push and posts a checklist comment on the PR. All 11 checks must pass before review (check 11 is informational-only).
| What | PR must contain exactly 1 commit |
| Pass | Push a single, squashed commit |
| Fix | git rebase -i origin/dev → squash all commits into one → git push --force-with-lease |
| Gotcha | Never git reset --soft to squash — contaminates the commit with unrelated changes |
| What | PR must not be marked as draft |
| Pass | Create PR as ready, or convert: gh pr ready |
| Fix | gh pr ready {number} or use the GitHub UI |
| Gotcha | The readiness check still runs on drafts but will show ❌ until you mark ready |
| What | PR branch must not be behind dev |
| Pass | Rebase onto latest dev before pushing |
| Fix | git fetch origin dev && git rebase origin/dev && git push --force-with-lease |
| Gotcha | After rebasing, the readiness check re-runs automatically on the new push |
| What | The copilot-pull-request-reviewer bot must post an APPROVED review |
| Pass | Wait — Copilot review is triggered automatically on PR creation/push |
| Fix | If Copilot hasn't reviewed after 5 minutes, push an empty commit to re-trigger: git commit --allow-empty -m "trigger review" && git push then squash before merge |
| Gotcha | Copilot review state is APPROVED, CHANGES_REQUESTED, or COMMENTED. Only APPROVED passes |
| What | PRs that modify packages/squad-sdk/src/ or packages/squad-cli/src/ must include a .changeset/*.md file or a CHANGELOG.md edit |
| Pass | Run npx changeset add and commit the generated file |
| Fix | npx changeset add → select affected package(s) → select bump type → write summary → git add .changeset/ && git commit --amend --no-edit && git push --force-with-lease |
| Gotcha | The skip-changelog label bypasses this check but requires reviewer approval. Non-source changes (docs, config, tests) don't need a changeset |
| What | PR must be cleanly mergeable with the base branch |
| Pass | Keep branch rebased on dev |
| Fix | git fetch origin dev && git rebase origin/dev → resolve conflicts → git push --force-with-lease |
| Gotcha | GitHub may show null mergeability briefly while computing — the check treats this as passing |
| What | Warns if PR includes .squad/ or docs/proposals/ files |
| Pass | Don't include team state or proposal files in product PRs |
| Fix | Remove unintended files: git reset HEAD .squad/ docs/proposals/ then amend your commit |
| Gotcha | This check is informational only — it always passes but flags attention. Including these files is OK if intentional (e.g., updating agent history) |
| What | All review threads opened by copilot-pull-request-reviewer must be resolved |
| Pass | Address each Copilot comment, then click "Resolve conversation" in the GitHub UI |
| Fix | Go to the PR's "Files changed" tab → find unresolved Copilot threads → fix the code or explain why no change is needed → click "Resolve conversation" |
| Gotcha | Outdated threads (on code that's since changed) are automatically skipped. Only active, unresolved threads block |
| What | All CI check runs (excluding the readiness check itself) must be green |
| Pass | Fix any build, test, or lint failures and push |
| Fix | Read the failing check's logs (gh run view {run-id} --log-failed), fix the issue, push |
| Gotcha | Pending/in-progress checks also show ❌. Wait for all checks to complete before evaluating. The readiness check re-runs after CI completes via workflow_run trigger |
| What | PR body or commit message must reference an issue (Closes #N, Fixes #N, Resolves #N, or Part of #N) |
| Pass | Include Closes #N in PR body or commit message |
| Fix | Edit PR body to add Closes #{issue-number}, or amend commit message |
| Gotcha | Case-insensitive matching. Only closing keywords are recognized |
| What | Warns when zero-dependency bootstrap files are modified (always passes) |
| Pass | Always passes — this is informational only |
| Fix | If flagged, verify the changed bootstrap file still has zero external dependencies |
| Gotcha | Protected files are listed in copilot-instructions.md. The check warns but does not block |
Handling Copilot review feedback:
git commit --amend --no-edit && git push --force-with-leaseRebasing when behind dev:
git fetch origin dev
git rebase origin/dev
# Resolve any conflicts
git push --force-with-lease
Re-running CI:
gh run rerun {run-id} --failedIf readiness check is stale:
Merge preconditions:
Who merges:
Post-merge cleanup:
git checkout dev
git pull origin dev
git branch -d squad/{issue-number}-{slug}
git push origin --delete squad/{issue-number}-{slug}
Verify issue auto-close:
Closes #{N}, the issue closes automatically on mergegh issue close {number}# Branch
git fetch origin dev && git checkout dev && git rebase origin/dev
git checkout -b squad/610-fix-broken-link
# Fix
# ... edit docs/some-file.md ...
# Validate
npm run build && npm test
# Commit (no changeset needed — docs only)
git add docs/some-file.md
git diff --cached --stat # verify: 1 file
git commit -m "docs: fix broken link in contributing guide
Closes #610
Co-authored-by: Copilot <[email protected]>"
# Push and PR
git push -u origin squad/610-fix-broken-link
gh pr create --repo bradygaster/squad --base dev \
--title "docs: fix broken link (#610)" \
--body "Closes #610"
# Branch
git fetch origin dev && git checkout dev && git rebase origin/dev
git checkout -b squad/42-add-profile-api
# Implement
# ... edit packages/squad-sdk/src/profile/index.ts ...
# ... edit packages/squad-sdk/src/index.ts (re-export) ...
# Validate
npm run build && npm test && npm run lint
# Changeset (required — touches packages/squad-sdk/src/)
npx changeset add
# Select: @bradygaster/squad-sdk, minor, "Add profile API"
# Stage and commit
git add packages/squad-sdk/src/profile/index.ts packages/squad-sdk/src/index.ts .changeset/
git diff --cached --stat # verify file count
git diff --cached --diff-filter=D --name-only # verify no deletions
git commit -m "feat: add profile API
Closes #42
Co-authored-by: Copilot <[email protected]>"
# Push and PR
git push -u origin squad/42-add-profile-api
gh pr create --repo bradygaster/squad --base dev \
--title "feat: add profile API (#42)" \
--body "Closes #42
## Summary
Adds profile resolution API to the SDK.
## Changes
- New module: packages/squad-sdk/src/profile/
- Re-exported from barrel file"
main (always branch from dev)main with PRs (always target dev)git add . or git add -A (stage specific files only)git commit -a (same risk as broad staging)git reset --soft to squash (contaminates commit with dev delta)dev or main (only force-push your own feature branch).squad/ files in product PRs without intention (scope check warns)After analyzing .github/workflows/squad-ci.yml, three gaps were identified. Gaps 1 and 3 are now implemented (checks 10 and 11). Gap 2 is deferred.
Problem: Nothing verifies that the PR body or commit message references an issue (Closes #N or Part of #N). Orphan PRs are hard to trace.
Recommendation: Add checkIssueLinkage(prBody, commitMessages) to pr-readiness.mjs.
/**
* Check: Issue linkage.
* @param {string} prBody — PR description text
* @param {Array<{ commit: { message: string } }>} commits
* @returns {{ pass: boolean, detail: string }}
*/
export function checkIssueLinkage(prBody, commits) {
const issuePattern = /(closes|fixes|resolves|part of)\s+#\d+/i;
const bodyHasRef = issuePattern.test(prBody || '');
const commitHasRef = (commits || []).some(
(c) => issuePattern.test(c.commit?.message || '')
);
if (bodyHasRef || commitHasRef) {
return { pass: true, detail: 'Issue reference found' };
}
return {
pass: false,
detail: 'No issue reference — add `Closes #N` to PR body or commit message',
};
}
Integration point: After check 2 (draft status), before check 3 (branch freshness). Data is already available from the commits and PR body fetched in the orchestrator.
Problem: The CI status check (check 9) verifies that existing checks are green, but doesn't verify that the expected set of checks actually ran. If a workflow is misconfigured or skipped, the PR could pass with zero CI checks.
Recommendation: Add checkRequiredChecksPresent(checkRuns, files) to pr-readiness.mjs.
/** Minimum required check names that must appear for source PRs. */
export const REQUIRED_CHECKS = ['Squad CI / test'];
/**
* Check: Required CI checks are present.
* @param {Array<{ name: string }>} checkRuns
* @param {Array<{ filename: string }>} files
* @returns {{ pass: boolean, detail: string }}
*/
export function checkRequiredChecksPresent(checkRuns, files) {
const touchesSource = (files || []).some(
(f) => SOURCE_PATTERN.test(f.filename)
);
if (!touchesSource) {
return { pass: true, detail: 'No source changes — required checks not enforced' };
}
const checkNames = new Set((checkRuns || []).map((cr) => cr.name));
const missing = REQUIRED_CHECKS.filter((name) => !checkNames.has(name));
if (missing.length > 0) {
return {
pass: false,
detail: `Required check(s) not found: ${missing.join(', ')}`,
};
}
return { pass: true, detail: 'All required checks present' };
}
Integration point: After check 9 (CI status). Uses the same checkRuns and files data already fetched.
Problem: The repo has zero-dependency bootstrap files that must never import external packages (documented in copilot-instructions.md). The squad-repo-health.yml workflow runs a bootstrap protection check, but the PR readiness comment doesn't surface it — contributors don't see the warning until they check the separate workflow.
Recommendation: Add checkProtectedFiles(files) to pr-readiness.mjs as an informational check (always passes, like scope clean).
/** Bootstrap files that must remain zero-dependency. */
export const PROTECTED_FILES = [
'packages/squad-cli/src/cli/core/detect-squad-dir.ts',
'packages/squad-cli/src/cli/core/errors.ts',
'packages/squad-cli/src/cli/core/gh-cli.ts',
'packages/squad-cli/src/cli/core/output.ts',
'packages/squad-cli/src/cli/core/history-split.ts',
];
/**
* Check: Protected file changes (informational).
* @param {Array<{ filename: string }>} files
* @returns {{ pass: boolean, detail: string }}
*/
export function checkProtectedFiles(files) {
const touched = (files || []).filter(
(f) => PROTECTED_FILES.includes(f.filename)
);
if (touched.length === 0) {
return { pass: true, detail: 'No protected bootstrap files changed' };
}
return {
pass: true,
detail: `⚠️ ${touched.length} protected bootstrap file(s) changed: ${touched.map((f) => f.filename.split('/').pop()).join(', ')} — verify zero-dependency constraint`,
};
}
Integration point: After check 7 (scope clean). Uses the same files data. Informational only — warns but doesn't block.
| Check | Type | Blocks PR? | Status |
|---|---|---|---|
| Issue linkage (Check 10) | Hard gate | Yes | Implemented |
| Required checks present | Hard gate | Yes | Deferred |
| Protected file warning (Check 11) | Informational | No | Implemented |
All three use data that pr-readiness.mjs already fetches — no new API calls needed. Total addition: ~60 lines of check functions + ~10 lines of orchestration wiring.