--- name: prd-to-goal description: "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." argument-hint: "[prd-text | issue#N | path/to/spec.md]" tags: [/goal, planning, prd, automation, cc-2.1.139] version: 0.1.0 author: OrchestKit license: MIT compatibility: "Claude Code 2.1.148+ (uses GA `/goal` loop)." user-invocable: true complexity: low context: inherit persuasion-type: guidance allowed-tools: [Read, Bash, Grep] metadata: category: planning milestone: M140 triggers: keywords: ["prd to goal", "goal from prd", "convert prd to goal", "/goal from issue", "decompose acceptance criteria"] examples: - "turn this PRD into a /goal line" - "what should my /goal until clause be for issue #1792" - "decompose this spec into observable assertions" anti-triggers: [implement, fix, commit, refactor, push] --- # /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 `/goal` against it. - Past `/goal` runs drifted, looped, or burned tokens because the `until` clause 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-prd` first — vibes do not decompose. - The work is destructive or irreversible (DB migrations, mass file deletes). `/goal` retries; you do not want retries on `DROP 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 --json title,body,labels` — the skill reads `body`. | | Spec file | Path to a Markdown / text file; the skill `Read`s it. | | ADR / design doc | Same as spec file. | ## 3. The decomposition algorithm 1. **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-prd` first — there is nothing to converge on. 2. **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 --json state | jq -r .state == "MERGED"` - `wc -l < src/auth.ts` returns a number within bound - `pnpm lint` exits 0 - `curl -sf $URL` returns 2xx 3. **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`. 4. **Compose the `until` clause.** 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. 5. **Compose the `abort-if` clause.** Pick a turn cap, token cap, and no-progress detector. Sensible defaults: - Turns: `15` for a single feature, `30` for a refactor, `5` for a bug fix. - Tokens: `100000` (1 USD-ish on Sonnet) for a feature, `30000` for a bug fix. - No-progress: `3` turns with no file changes and no new test passing. ## 4. Output template The skill emits exactly two lines, ready to paste: ``` /goal until AND AND /goal abort-if turns > OR tokens > OR no_progress_for__turns ``` No 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 pass ``` Output: ``` /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_turns ``` Rationale: 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 updated ``` Output: ``` /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_turns ``` Rationale: 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_turns ``` Rationale: 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 `/goal` exits, run audit-full to confirm the assertions actually held end-to-end (the loop trusts the boolean; audit re-checks the intent).