Prd To Goal
Decomposes a PRD, issue, or spec into a copy-pasteable `/goal until ... abort-if ...` line. Use when running /goal against a spec, to reduce acceptance criteria to AND-joined boolean assertions.
/ork:prd-to-goal/ork:prd-to-goal — PRD → /goal Decomposition
Converts a PRD / issue / spec into a single copy-pasteable /goal line. The hard part of /goal is not running it — it is writing an until clause that is convergent (terminates), falsifiable (testable boolean), and observable (the agent can actually check it without subjective judgement). This skill makes that decomposition reproducible.
1. When to use
Use it when:
- You have a written PRD, GitHub issue, or spec and want to run
/goalagainst it. - Past
/goalruns drifted, looped, or burned tokens because theuntilclause was vague (until tests pass,until done,until design is good). - You need to justify the abort budget — turns, tokens, no-progress threshold.
Skip it when:
- One-shot bug fix where the failing test is the acceptance criterion. Just run
/goal until pnpm test -- auth.spec.ts passes. - No written PRD exists. Run
/ork:write-prdfirst — vibes do not decompose. - The work is destructive or irreversible (DB migrations, mass file deletes).
/goalretries; you do not want retries onDROP TABLE.
2. Inputs the skill accepts
| Input | How |
|---|---|
| Pasted PRD text | Provide as the argument, or paste into the chat after invoking. |
| GitHub issue | gh issue view <N> --json title,body,labels — the skill reads body. |
| Spec file | Path to a Markdown / text file; the skill Reads it. |
| ADR / design doc | Same as spec file. |
3. The decomposition algorithm
- Extract acceptance criteria. Pull every
MUST,SHOULD,Definition of Done,Acceptance Criteria, and checkbox-style line. If the doc has none, stop and tell the user to run/ork:write-prdfirst — there is nothing to converge on. - Map each criterion to an observable boolean. Each criterion must reduce to a single shell-checkable assertion. Examples of observable state:
test -f path/to/file(file exists)pnpm test -- pattern passes(test command exits 0)gh pr view <N> --json state | jq -r .state == "MERGED"wc -l < src/auth.tsreturns a number within boundpnpm lintexits 0curl -sf $URLreturns 2xx
- Reject non-observable criteria. Drop or rewrite criteria that depend on subjective judgement (
code is clean,design feels right,users are happy). Either find a proxy (lint exits 0,Lighthouse score > 90,NPS survey ID exists) or surface the criterion back to the user as out of scope for/goal. - Compose the
untilclause. AND-join the observable assertions in priority order — the cheapest, most likely-to-fail check first so the loop short-circuits early. Three to five assertions is the sweet spot; more than seven usually means the PRD is two PRDs. - Compose the
abort-ifclause. Pick a turn cap, token cap, and no-progress detector. Sensible defaults:- Turns:
15for a single feature,30for a refactor,5for a bug fix. - Tokens:
100000(1 USD-ish on Sonnet) for a feature,30000for a bug fix. - No-progress:
3turns with no file changes and no new test passing.
- Turns:
4. Output template
The skill emits exactly two lines, ready to paste:
/goal until <assertion_1> AND <assertion_2> AND <assertion_3>
/goal abort-if turns > <N> OR tokens > <T> OR no_progress_for_<K>_turnsNo commentary, no markdown wrapper — the user copies the block straight into Claude Code.
5. Worked examples
Example A — Bug fix PRD
Input (issue body):
Title: Login fails on emails containing "+"
Acceptance Criteria:
- New regression test in tests/auth/test_login.py covers email with "+"
- The new test passes
- All existing auth tests still passOutput:
/goal until test -f tests/auth/test_login.py AND pnpm test -- tests/auth/test_login.py passes AND pnpm test -- tests/auth passes
/goal abort-if turns > 5 OR tokens > 30000 OR no_progress_for_3_turnsRationale: file existence is the cheapest check; the targeted test is the regression gate; the broad auth suite catches collateral damage.
Example B — Feature PRD
Input (PRD excerpt):
Feature: User Avatar Endpoint
MUST:
- New route GET /users/:id/avatar registered
- Returns 200 with { url: string, updatedAt: ISO8601 } for known user
- Returns 404 for unknown user
- Integration test covers both cases
- OpenAPI spec updatedOutput:
/goal until grep -q "users/:id/avatar" src/routes/users.ts AND pnpm test -- tests/integration/users.avatar.spec.ts passes AND grep -q "/users/{id}/avatar" openapi.yaml AND pnpm lint passes
/goal abort-if turns > 15 OR tokens > 100000 OR no_progress_for_3_turnsRationale: route grep catches a handler that was stubbed but never wired; the integration spec encodes both 200 and 404; OpenAPI grep enforces the docs MUST; lint guards against half-typed code shipping.
Example C — Refactor PRD
Input (PRD excerpt):
Refactor: split monolithic src/auth.ts (1842 LOC) into per-strategy files
Definition of Done:
- src/auth.ts under 200 LOC
- New files in src/auth/strategies/*.ts cover oauth, jwt, password
- All auth tests still pass
- Lint clean
- No new files outside src/auth/Output:
/goal until [ $(wc -l < src/auth.ts) -lt 200 ] AND test -f src/auth/strategies/oauth.ts AND test -f src/auth/strategies/jwt.ts AND test -f src/auth/strategies/password.ts AND pnpm test -- tests/auth passes AND pnpm lint passes
/goal abort-if turns > 30 OR tokens > 150000 OR no_progress_for_3_turnsRationale: the LOC bound is the convergent signal — without it, /goal can keep "improving" forever. File-existence checks pin the structural decomposition; tests + lint guard correctness. Higher budget because refactors run longer than bug fixes.
6. Anti-patterns
| Bad | Why it fails | Good |
|---|---|---|
/goal until tests pass | Which tests? Existing or new? On which command? The agent can pick whatever subset is already green and claim done. | /goal until pnpm test -- tests/auth/test_login.py passes AND pnpm test -- tests/auth passes |
/goal until the code looks clean | Not falsifiable. The agent cannot check "looks clean" — it will either declare victory immediately or never. | /goal until pnpm lint passes AND [ $(wc -l < src/auth.ts) -lt 200 ] |
/goal until done | Unbounded. There is no terminating condition; combined with a generous abort-if turns > 50 this is how runs eat 500K tokens overnight. | Pick 3–5 observable assertions. If you cannot, the PRD is not ready — go to /ork:write-prd. |
7. Related skills
ork:write-prd— if the input has no acceptance criteria, run this first.ork:brainstorm— useful when the PRD itself is contested and you want options before writing the goal.ork:audit-full— after/goalexits, run audit-full to confirm the assertions actually held end-to-end (the loop trusts the boolean; audit re-checks the intent).
Portless
Named HTTPS .localhost URLs for local development with portless (v0.13.x). Eliminates port collisions, enables stable URLs for agents, integrates with emulate for API emulation aliases, git worktrees for branch-named subdomains, LAN mode (--lan) for mDNS .local hostnames reachable across devices, Tailscale sharing (--tailscale / --funnel), and OS startup-service install for boot persistence. Use when setting up local dev environments, configuring agent-accessible URLs, running multi-service dev setups, or testing from phones/tablets on the same wifi. Do NOT use for production deployments, CI environments (set PORTLESS=0), or DNS/hosting configuration.
Presentation Builder
Creates zero-dependency, animation-rich HTML presentations from scratch or by converting PowerPoint files. Use when the user wants to build a presentation, convert a PPT/PPTX to web slides, or create a slide deck for a talk, pitch, or tutorial. Generates single self-contained HTML files with inline CSS/JS.
Last updated on