git clone https://github.com/Intense-Visions/harness-engineering
T=$(mktemp -d) && git clone --depth=1 https://github.com/Intense-Visions/harness-engineering "$T" && mkdir -p ~/.claude/skills && cp -r "$T/agents/skills/claude-code/harness-roadmap-pilot" ~/.claude/skills/intense-visions-harness-engineering-harness-roadmap-pilot && rm -rf "$T"
agents/skills/claude-code/harness-roadmap-pilot/SKILL.mdHarness Roadmap Pilot
AI-assisted selection of the next highest-impact unblocked roadmap item. Scores candidates, recommends one, assigns it, and transitions to the appropriate next skill.
When to Use
- When the team or individual needs to pick the next item to work on from the roadmap
- When there are multiple unblocked items and prioritization guidance is needed
- After completing a feature and looking for the next highest-impact work
- NOT when the roadmap does not exist (direct user to harness-roadmap --create)
- NOT when the user already knows what to work on (use harness-brainstorming or harness-autopilot directly)
Process
Iron Law
Never assign or transition without the human confirming the recommendation first.
Present the ranked candidates, the AI reasoning, and the recommended pick. Wait for explicit confirmation before making any changes.
Phase 1: SCAN -- Score Candidates
- Check if
exists.docs/roadmap.md- If missing: error. "No roadmap found at docs/roadmap.md. Run harness-roadmap --create first."
- Parse the roadmap using
fromparseRoadmap
.@harness-engineering/core - Determine the current user:
- Use the
argument if provided--user - Otherwise, attempt to detect from git config:
orgit config user.namegit config user.email - If neither available, proceed without affinity scoring
- Use the
- Call
fromscoreRoadmapCandidates(roadmap, { currentUser })
.@harness-engineering/core - If no candidates: inform the human. "No unblocked planned or backlog items found. All items are either in-progress, done, blocked, or the roadmap is empty."
Present the top 5 candidates:
ROADMAP PILOT -- Candidate Scoring Top candidates (scored by position 50%, dependents 30%, affinity 20%): # Feature Milestone Priority Score Breakdown 1. Feature A MVP Release P0 0.85 pos:0.9 dep:0.8 aff:1.0 2. Feature B MVP Release P1 0.72 pos:0.8 dep:0.6 aff:0.5 3. Feature C Q2 Release -- 0.65 pos:0.7 dep:0.5 aff:0.0 4. Feature D Backlog -- 0.40 pos:0.3 dep:0.4 aff:0.0 5. Feature E Backlog -- 0.35 pos:0.2 dep:0.3 aff:0.0
Phase 2: RECOMMEND -- AI-Assisted Analysis
-
For the top 3 candidates, read their spec files (if they exist):
- Read the spec's Overview and Goals section
- Read the spec's Success Criteria section
- Assess effort and impact from the spec content
-
Provide a recommendation with reasoning:
RECOMMENDATION I recommend Feature A (MVP Release, P0, score: 0.85). Reasoning: - Highest priority (P0) with strong positional signal (first in MVP milestone) - Unblocks 2 downstream features (Feature X, Feature Y) - You completed its blocker "Foundation" -- high context affinity - Spec exists with clear success criteria (12 acceptance tests) - Estimated effort: medium (8 tasks in the plan) Alternative: Feature B (P1, score: 0.72) -- consider if Feature A's scope is too large for the current time window. Proceed with Feature A? (y/n/pick another)
Phase 3: CONFIRM -- Human Decision
- Wait for human confirmation.
- If yes: proceed to Phase 4.
- If pick another: ask which candidate number, then proceed with that pick.
- If no: stop. No changes made.
Phase 4: ASSIGN -- Execute Assignment and Transition
-
Call
with actionmanage_roadmap
to assign the feature:updatemanage_roadmap({ path: "<project-root>", action: "update", feature: "<feature-name>", assignee: "<currentUser>" })- This updates the feature's
field with assignment history trackingAssignee - Automatically triggers external sync (GitHub Issues) if tracker config exists in
harness.config.json - External sync is fire-and-forget — errors are logged but do not block the assignment
- This updates the feature's
-
Determine the transition target:
- If the feature has a
field (non-null): transition tospecharness:autopilot - If the feature has no
: transition tospecharness:brainstorming
- If the feature has a
-
Present the transition to the human via
:emit_interactionemit_interaction({ path: "<project-root>", type: "transition", transition: { completedPhase: "roadmap-pilot", suggestedNext: "<brainstorming|autopilot>", reason: "Feature '<name>' assigned and ready for <brainstorming|execution>", artifacts: ["docs/roadmap.md"], requiresConfirmation: true, summary: "Assigned '<name>' to <user>. <Spec exists -- ready for autopilot|No spec -- needs brainstorming first>.", qualityGate: { checks: [ { "name": "roadmap-parsed", "passed": true }, { "name": "candidate-scored", "passed": true }, { "name": "human-confirmed", "passed": true }, { "name": "assignment-written", "passed": true } ], allPassed: true } } }) -
Run
.harness validate
Harness Integration
/parseRoadmap
-- Parse and writeserializeRoadmap
. Import fromdocs/roadmap.md
.@harness-engineering/core
-- Core scoring algorithm. Import fromscoreRoadmapCandidates
. Takes a@harness-engineering/core
and optionalRoadmap
(currentUser for affinity).PilotScoringOptions
-- Used for assignment. Supportsmanage_roadmap update
field which delegates toassignee
internally, handles history tracking, and automatically triggers external sync (GitHub Issues).assignFeature
-- Used for the skill transition at the end. Transitions toemit_interaction
(no spec) orharness:brainstorming
(spec exists).harness:autopilot
-- Run after assignment is written.harness validate
Success Criteria
- Roadmap is parsed and unblocked planned/backlog items are scored
- Scoring uses two-tier sort: explicit priority first, then weighted score
- AI reads top candidates' specs and provides recommendation with reasoning
- Human confirms before any changes are made
- Assignment updates feature field, appends history records, and syncs externally
- Reassignment produces two history records (unassigned + assigned)
- Transition routes to brainstorming (no spec) or autopilot (spec exists)
passes after all changesharness validate
Rationalizations to Reject
| Rationalization | Reality |
|---|---|
| "The top-scored candidate is obviously correct, so I can assign it without asking the human" | The Iron Law: never assign or transition without the human confirming the recommendation first. |
| "Affinity data is not available so the scoring is degraded -- I should just pick the first planned item" | Proceed without affinity scoring by zeroing out the affinity weight. Position and dependents signals still produce meaningful rankings. |
| "The feature has no spec, but I can skip brainstorming and jump straight to planning since the summary is clear enough" | No spec routes to brainstorming, spec exists routes to autopilot. A one-line roadmap summary is not a spec. |
Examples
Example: Pick Next Item from a Multi-Milestone Roadmap
Context: A roadmap with 3 milestones, 8 features. 2 are in-progress, 1 is done, 2 are blocked, 3 are planned/backlog and unblocked. User is @cwarner who completed "Core Library Design".
Phase 1: SCAN
ROADMAP PILOT -- Candidate Scoring Top candidates: # Feature Milestone Priority Score Breakdown 1. Graph Connector MVP Release P2 0.78 pos:0.8 dep:0.6 aff:1.0 2. Performance Baselines Q3 Hardening -- 0.45 pos:0.5 dep:0.3 aff:0.0 3. Push Notifications Backlog -- 0.30 pos:0.2 dep:0.2 aff:0.5
Phase 2: RECOMMEND
I recommend Graph Connector (MVP Release, P2, score: 0.78). Reasoning: - Only prioritized item among candidates (P2) - You completed its blocker "Core Library Design" -- maximum affinity bonus - Unblocks "API Integration" downstream - Spec exists at docs/changes/graph-connector/proposal.md Proceed? (y/n/pick another)
Human confirms y.
Phase 4: ASSIGN
manage_roadmap update: Graph Connector assignee -> @cwarner History: +1 record (assigned, 2026-04-02) Roadmap updated: docs/roadmap.md External sync: github:harness-eng/harness#43 assigned (automatic) Transitioning to harness:autopilot (spec exists)...
Gates
- No assignment without human confirmation. The CONFIRM phase must complete with explicit approval. Never auto-assign.
- No transition without assignment. The skill must write the assignment before transitioning to the next skill.
- No scoring without a parsed roadmap. If
does not exist or fails to parse, stop with an error.docs/roadmap.md
Escalation
- When no unblocked candidates exist: Inform the human. Suggest reviewing blocked items to see if blockers can be resolved, or adding new features via
.harness-roadmap --add - When affinity data is unavailable: Proceed without affinity scoring (weight falls to 0 for all candidates). Note this in the output.
- When external sync fails: Log the error, complete the local assignment, and note that external sync can be retried with
.harness-roadmap --sync