@pugi/cli 0.1.0-beta.4 → 0.1.0-beta.40
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/THIRD_PARTY_NOTICES.md +40 -0
- package/assets/pugi-mascot.ansi +15 -25
- package/bin/run.js +33 -1
- package/dist/commands/jobs-watch.js +201 -0
- package/dist/commands/jobs.js +15 -0
- package/dist/commands/smoke.js +133 -0
- package/dist/core/agent-progress/cleanup.js +134 -0
- package/dist/core/agent-progress/schema.js +144 -0
- package/dist/core/agent-progress/writer.js +101 -0
- package/dist/core/artifact-chain/dispatcher.js +148 -0
- package/dist/core/artifact-chain/exporter.js +164 -0
- package/dist/core/artifact-chain/state.js +243 -0
- package/dist/core/artifact-chain/steps.js +169 -0
- package/dist/core/auth/ensure-authenticated.js +129 -0
- package/dist/core/auth/env-provider.js +238 -0
- package/dist/core/auto-update/channels.js +122 -0
- package/dist/core/auto-update/checker.js +241 -0
- package/dist/core/auto-update/state.js +235 -0
- package/dist/core/bare-mode/index.js +107 -0
- package/dist/core/bash-classifier.js +108 -1
- package/dist/core/checkpoint/resumer.js +149 -0
- package/dist/core/checkpoint/rewinder.js +291 -0
- package/dist/core/codegraph/decision-store.js +248 -0
- package/dist/core/codegraph/detect-repo.js +459 -0
- package/dist/core/codegraph/install.js +134 -0
- package/dist/core/codegraph/offer-hook.js +220 -0
- package/dist/core/compact/auto-trigger.js +96 -0
- package/dist/core/compact/buffer-rewriter.js +115 -0
- package/dist/core/compact/summarizer.js +208 -0
- package/dist/core/compact/token-counter.js +108 -0
- package/dist/core/consensus/diff-capture.js +73 -0
- package/dist/core/context/index.js +7 -0
- package/dist/core/context/markdown-traverse.js +255 -0
- package/dist/core/cost/rate-card.js +129 -0
- package/dist/core/cost/tracker.js +221 -0
- package/dist/core/denial-tracking/index.js +8 -0
- package/dist/core/denial-tracking/state.js +264 -0
- package/dist/core/diagnostics/probe-runner.js +93 -0
- package/dist/core/diagnostics/probes/api.js +46 -0
- package/dist/core/diagnostics/probes/auth.js +86 -0
- package/dist/core/diagnostics/probes/bare-mode.js +42 -0
- package/dist/core/diagnostics/probes/cli-version.js +127 -0
- package/dist/core/diagnostics/probes/config.js +72 -0
- package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
- package/dist/core/diagnostics/probes/disk.js +81 -0
- package/dist/core/diagnostics/probes/git.js +65 -0
- package/dist/core/diagnostics/probes/mcp.js +75 -0
- package/dist/core/diagnostics/probes/node.js +59 -0
- package/dist/core/diagnostics/probes/pnpm.js +36 -0
- package/dist/core/diagnostics/probes/pugi-md.js +89 -0
- package/dist/core/diagnostics/probes/session.js +74 -0
- package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
- package/dist/core/diagnostics/probes/workspace.js +63 -0
- package/dist/core/diagnostics/types.js +70 -0
- package/dist/core/dispatch/cache-cleanup.js +197 -0
- package/dist/core/dispatch/cache-handoff.js +295 -0
- package/dist/core/edits/dispatch.js +218 -2
- package/dist/core/edits/journal.js +199 -0
- package/dist/core/edits/layer-d-ast.js +557 -14
- package/dist/core/edits/verify-hook.js +273 -0
- package/dist/core/edits/worktree.js +322 -0
- package/dist/core/engine/anvil-client.js +115 -5
- package/dist/core/engine/budgets.js +98 -0
- package/dist/core/engine/context-prefix.js +155 -0
- package/dist/core/engine/intent.js +260 -0
- package/dist/core/engine/native-pugi.js +860 -211
- package/dist/core/engine/prompts.js +88 -2
- package/dist/core/engine/strip-internal-fields.js +124 -0
- package/dist/core/engine/tool-bridge.js +992 -36
- package/dist/core/feedback/queue.js +177 -0
- package/dist/core/feedback/submitter.js +145 -0
- package/dist/core/file-cache.js +113 -1
- package/dist/core/hooks/events.js +44 -0
- package/dist/core/hooks/index.js +15 -0
- package/dist/core/hooks/registry.js +213 -0
- package/dist/core/hooks/runner.js +236 -0
- package/dist/core/hooks/v2/event-emitter.js +115 -0
- package/dist/core/hooks/v2/executor.js +282 -0
- package/dist/core/hooks/v2/index.js +25 -0
- package/dist/core/hooks/v2/lifecycle.js +104 -0
- package/dist/core/hooks/v2/loader.js +216 -0
- package/dist/core/hooks/v2/matcher.js +125 -0
- package/dist/core/hooks/v2/trust.js +143 -0
- package/dist/core/hooks/v2/types.js +86 -0
- package/dist/core/lsp/cache.js +105 -0
- package/dist/core/lsp/client.js +776 -0
- package/dist/core/lsp/language-detect.js +66 -0
- package/dist/core/lsp/post-edit-diagnostics.js +171 -0
- package/dist/core/mcp/client.js +75 -6
- package/dist/core/mcp/http-server.js +553 -0
- package/dist/core/mcp/orchestrator-tools.js +662 -0
- package/dist/core/mcp/permission.js +190 -0
- package/dist/core/mcp/registry.js +24 -2
- package/dist/core/mcp/server-tools.js +219 -0
- package/dist/core/mcp/server.js +397 -0
- package/dist/core/memory/dual-write.js +416 -0
- package/dist/core/memory/phase1-kinds.js +20 -0
- package/dist/core/memory-sync/queue.js +158 -0
- package/dist/core/onboarding/ensure-initialized.js +133 -0
- package/dist/core/onboarding/marker.js +111 -0
- package/dist/core/onboarding/telemetry-state.js +108 -0
- package/dist/core/output-style/presets.js +176 -0
- package/dist/core/output-style/state.js +185 -0
- package/dist/core/permissions/auto-classifier.js +124 -0
- package/dist/core/permissions/circuit-breaker.js +83 -0
- package/dist/core/permissions/gate.js +278 -0
- package/dist/core/permissions/index.js +20 -0
- package/dist/core/permissions/mode.js +174 -0
- package/dist/core/permissions/state.js +241 -0
- package/dist/core/permissions/tool-class.js +93 -0
- package/dist/core/prd-check/parser.js +215 -0
- package/dist/core/prd-check/reporter.js +127 -0
- package/dist/core/prd-check/session-review.js +557 -0
- package/dist/core/prd-check/verifiers.js +223 -0
- package/dist/core/pugi-md/context-injector.js +76 -0
- package/dist/core/pugi-md/walk-up.js +207 -0
- package/dist/core/release-notes/parser.js +241 -0
- package/dist/core/release-notes/state.js +116 -0
- package/dist/core/repl/history.js +11 -1
- package/dist/core/repl/model-pricing.js +135 -0
- package/dist/core/repl/session.js +1899 -38
- package/dist/core/repl/slash-commands.js +406 -21
- package/dist/core/repl/store/session-store.js +31 -2
- package/dist/core/repl/workspace-context.js +22 -0
- package/dist/core/repo-map/build.js +125 -0
- package/dist/core/repo-map/cache.js +185 -0
- package/dist/core/repo-map/extractor.js +254 -0
- package/dist/core/repo-map/formatter.js +145 -0
- package/dist/core/repo-map/scanner.js +211 -0
- package/dist/core/retry-budget/budget.js +284 -0
- package/dist/core/retry-budget/index.js +5 -0
- package/dist/core/session.js +92 -0
- package/dist/core/settings.js +80 -0
- package/dist/core/share/formatter.js +271 -0
- package/dist/core/share/redactor.js +221 -0
- package/dist/core/share/uploader.js +267 -0
- package/dist/core/skills/defaults.js +457 -0
- package/dist/core/smoke/headless-driver.js +174 -0
- package/dist/core/smoke/orchestrator.js +194 -0
- package/dist/core/smoke/runner.js +238 -0
- package/dist/core/smoke/scenario-parser.js +316 -0
- package/dist/core/subagents/dispatcher-real.js +600 -0
- package/dist/core/subagents/dispatcher.js +113 -24
- package/dist/core/subagents/index.js +18 -5
- package/dist/core/subagents/isolation-matrix.js +213 -0
- package/dist/core/subagents/spawn.js +19 -4
- package/dist/core/telemetry/emitter.js +229 -0
- package/dist/core/telemetry/queue.js +251 -0
- package/dist/core/theme/context.js +91 -0
- package/dist/core/theme/presets.js +228 -0
- package/dist/core/theme/state.js +181 -0
- package/dist/core/todos/invariant.js +10 -0
- package/dist/core/todos/state.js +177 -0
- package/dist/core/transport/version-interceptor.js +166 -0
- package/dist/core/vim/keymap.js +288 -0
- package/dist/core/vim/state.js +92 -0
- package/dist/index.js +28 -0
- package/dist/runtime/bootstrap.js +190 -0
- package/dist/runtime/cli.js +3073 -321
- package/dist/runtime/commands/cancel.js +231 -0
- package/dist/runtime/commands/chain.js +489 -0
- package/dist/runtime/commands/codegraph-status.js +227 -0
- package/dist/runtime/commands/compact.js +297 -0
- package/dist/runtime/commands/cost.js +199 -0
- package/dist/runtime/commands/delegate.js +242 -11
- package/dist/runtime/commands/dispatch.js +126 -0
- package/dist/runtime/commands/doctor.js +390 -0
- package/dist/runtime/commands/feedback.js +184 -0
- package/dist/runtime/commands/hooks.js +184 -0
- package/dist/runtime/commands/lsp.js +368 -0
- package/dist/runtime/commands/mcp.js +879 -0
- package/dist/runtime/commands/memory.js +508 -0
- package/dist/runtime/commands/model.js +237 -0
- package/dist/runtime/commands/onboarding.js +275 -0
- package/dist/runtime/commands/patch.js +128 -0
- package/dist/runtime/commands/permissions.js +112 -0
- package/dist/runtime/commands/plan.js +143 -0
- package/dist/runtime/commands/prd-check.js +285 -0
- package/dist/runtime/commands/redo-blob-store.js +92 -0
- package/dist/runtime/commands/redo.js +361 -0
- package/dist/runtime/commands/release-notes.js +229 -0
- package/dist/runtime/commands/repo-map.js +95 -0
- package/dist/runtime/commands/report.js +299 -0
- package/dist/runtime/commands/resume.js +118 -0
- package/dist/runtime/commands/review-consensus.js +17 -2
- package/dist/runtime/commands/rewind.js +333 -0
- package/dist/runtime/commands/sessions.js +163 -0
- package/dist/runtime/commands/share.js +316 -0
- package/dist/runtime/commands/status.js +186 -0
- package/dist/runtime/commands/stickers.js +82 -0
- package/dist/runtime/commands/style.js +194 -0
- package/dist/runtime/commands/theme.js +196 -0
- package/dist/runtime/commands/undo.js +32 -0
- package/dist/runtime/commands/update.js +289 -0
- package/dist/runtime/commands/vim.js +140 -0
- package/dist/runtime/commands/worktree.js +177 -0
- package/dist/runtime/headless-repl.js +195 -0
- package/dist/runtime/headless.js +543 -0
- package/dist/runtime/load-hooks-or-exit.js +71 -0
- package/dist/runtime/plan-decompose.js +531 -0
- package/dist/runtime/version.js +65 -0
- package/dist/tools/agent-tool.js +229 -0
- package/dist/tools/apply-patch.js +556 -0
- package/dist/tools/ask-user-question.js +213 -0
- package/dist/tools/ask-user.js +115 -0
- package/dist/tools/file-tools.js +85 -14
- package/dist/tools/lsp-tools.js +189 -0
- package/dist/tools/mcp-tool.js +260 -0
- package/dist/tools/multi-edit.js +361 -0
- package/dist/tools/registry.js +46 -0
- package/dist/tools/skill-tool.js +96 -0
- package/dist/tools/tasks.js +208 -0
- package/dist/tools/todo-write.js +184 -0
- package/dist/tools/web-fetch.js +147 -2
- package/dist/tools/web-search.js +458 -0
- package/dist/tui/agent-progress-card.js +111 -0
- package/dist/tui/agent-tree.js +10 -0
- package/dist/tui/ask-modal.js +2 -2
- package/dist/tui/ask-user-question-prompt.js +192 -0
- package/dist/tui/compact-banner.js +81 -0
- package/dist/tui/conversation-pane.js +82 -8
- package/dist/tui/cost-table.js +111 -0
- package/dist/tui/doctor-table.js +46 -0
- package/dist/tui/feedback-prompt.js +156 -0
- package/dist/tui/input-box.js +69 -2
- package/dist/tui/markdown-render.js +4 -4
- package/dist/tui/onboarding-wizard.js +240 -0
- package/dist/tui/permissions-picker.js +86 -0
- package/dist/tui/render.js +35 -0
- package/dist/tui/repl-render.js +303 -13
- package/dist/tui/repl-splash.js +2 -2
- package/dist/tui/repl.js +72 -14
- package/dist/tui/splash.js +1 -1
- package/dist/tui/status-bar.js +94 -16
- package/dist/tui/status-table.js +7 -0
- package/dist/tui/stickers-art.js +136 -0
- package/dist/tui/style-table.js +28 -0
- package/dist/tui/theme-table.js +29 -0
- package/dist/tui/tool-stream-pane.js +52 -3
- package/dist/tui/update-banner.js +20 -2
- package/dist/tui/vim-input.js +267 -0
- package/docs/examples/codegraph.mcp.json +10 -0
- package/package.json +12 -6
- package/test/scenarios/codegen-create-file.scenario.txt +13 -0
- package/test/scenarios/compact-force.scenario.txt +11 -0
- package/test/scenarios/identity.scenario.txt +11 -0
- package/test/scenarios/persona-handoff.scenario.txt +11 -0
- package/test/scenarios/walkback.scenario.txt +12 -0
- package/dist/core/engine/compaction-hook.js +0 -154
|
@@ -0,0 +1,215 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* PRD parser — Pugi α7 Wave 6 (`/prd-check`, 2026-05-27).
|
|
3
|
+
*
|
|
4
|
+
* Reads a markdown PRD file and extracts the acceptance-criteria
|
|
5
|
+
* section into a list of verifiable criteria. The parser is
|
|
6
|
+
* intentionally narrow: it owns ONE function, accepts raw markdown
|
|
7
|
+
* source as a string (no fs I/O), and returns a structured list the
|
|
8
|
+
* verifier module can fan out over.
|
|
9
|
+
*
|
|
10
|
+
* Why two phases (parse → verify) instead of a single pass:
|
|
11
|
+
*
|
|
12
|
+
* - keeps the parser deterministic + fast (pure string in, JSON
|
|
13
|
+
* out — trivial to unit-test without touching the filesystem)
|
|
14
|
+
*
|
|
15
|
+
* - lets the reporter render the criterion list even when every
|
|
16
|
+
* verifier fails, so operators see WHAT failed before WHY
|
|
17
|
+
*
|
|
18
|
+
* - mirrors the L17 doctor split: `probe-runner` runs a set of
|
|
19
|
+
* probe descriptors → identical contract here, just for PRD
|
|
20
|
+
* acceptance items instead of environment probes
|
|
21
|
+
*
|
|
22
|
+
* Heading recognition is tolerant by design: PRD authors use both
|
|
23
|
+
* `## Acceptance Criteria` and `## Success Criteria`, sometimes with
|
|
24
|
+
* a trailing colon, sometimes inside an h3. We accept any h2/h3
|
|
25
|
+
* matching either label (case-insensitive). The first matching
|
|
26
|
+
* section wins; subsequent matches are ignored so a PRD with both
|
|
27
|
+
* sections does not double-count items.
|
|
28
|
+
*
|
|
29
|
+
* Item recognition supports two shapes documented in the wave-6
|
|
30
|
+
* spec:
|
|
31
|
+
*
|
|
32
|
+
* 1. numbered lists `1. <text>` / `1) <text>`
|
|
33
|
+
* 2. markdown checklists `- [ ] <text>` / `- [x] <text>`
|
|
34
|
+
*
|
|
35
|
+
* Either shape may include inline mentions the verifier extracts:
|
|
36
|
+
* file paths (`apps/foo/bar.ts`), test specs (`*.spec.ts`),
|
|
37
|
+
* route declarations (`GET /api/x`), CLI commands (`pugi prd-check`),
|
|
38
|
+
* and doc references (`docs/foo.md`). The parser captures these
|
|
39
|
+
* verbatim into `mentions` so the verifier module can fan checks
|
|
40
|
+
* without re-tokenising the prose.
|
|
41
|
+
*/
|
|
42
|
+
const ACCEPTANCE_HEADING_RE = /^(#{2,3})\s+(acceptance criteria|success criteria|deliverables)\b\s*:?\s*$/i;
|
|
43
|
+
const ANY_HEADING_RE = /^(#{1,6})\s+\S/;
|
|
44
|
+
const TITLE_HEADING_RE = /^#\s+(.+?)\s*$/;
|
|
45
|
+
const NUMBERED_ITEM_RE = /^(\s*)(\d+)[\.)]\s+(.+?)\s*$/;
|
|
46
|
+
const CHECKLIST_ITEM_RE = /^(\s*)-\s+\[([ xX])\]\s+(.+?)\s*$/;
|
|
47
|
+
/**
|
|
48
|
+
* Public entry: parse a markdown PRD source into `ParsedPrd`. Pure
|
|
49
|
+
* function — no filesystem, no logging. The CLI handler is
|
|
50
|
+
* responsible for reading the file and forwarding the contents.
|
|
51
|
+
*/
|
|
52
|
+
export function parsePrd(source) {
|
|
53
|
+
const lines = source.split(/\r?\n/);
|
|
54
|
+
const title = extractTitle(lines);
|
|
55
|
+
const range = findAcceptanceRange(lines);
|
|
56
|
+
if (!range) {
|
|
57
|
+
return { title, hasAcceptanceSection: false, criteria: [] };
|
|
58
|
+
}
|
|
59
|
+
const sectionLines = lines.slice(range.start, range.end);
|
|
60
|
+
const criteria = extractCriteria(sectionLines);
|
|
61
|
+
return { title, hasAcceptanceSection: true, criteria };
|
|
62
|
+
}
|
|
63
|
+
/**
|
|
64
|
+
* Extract verifiable mentions from a criterion text. Exported for
|
|
65
|
+
* the verifier spec so tests can drive mention classification
|
|
66
|
+
* without running the full parser.
|
|
67
|
+
*/
|
|
68
|
+
export function extractMentions(text) {
|
|
69
|
+
const mentions = [];
|
|
70
|
+
const seen = new Set();
|
|
71
|
+
const push = (key, mention) => {
|
|
72
|
+
if (seen.has(key))
|
|
73
|
+
return;
|
|
74
|
+
seen.add(key);
|
|
75
|
+
mentions.push(mention);
|
|
76
|
+
};
|
|
77
|
+
// 1) Routes — `GET /api/path`, `POST /foo`, etc. Recognised
|
|
78
|
+
// BEFORE file paths because the trailing `/` could otherwise
|
|
79
|
+
// be mis-classified.
|
|
80
|
+
const routeRe = /\b(GET|POST|PUT|PATCH|DELETE|HEAD|OPTIONS)\s+(\/[A-Za-z0-9_./:\-]+)/g;
|
|
81
|
+
for (const match of text.matchAll(routeRe)) {
|
|
82
|
+
const method = match[1].toUpperCase();
|
|
83
|
+
const path = match[2];
|
|
84
|
+
push(`route:${method} ${path}`, { kind: 'route', method, path });
|
|
85
|
+
}
|
|
86
|
+
// 2) Backtick-wrapped tokens — most reliable signal. The parser
|
|
87
|
+
// inspects each token and decides whether it is a path, a test
|
|
88
|
+
// spec, a CLI command, or a route.
|
|
89
|
+
const backtickRe = /`([^`\n]+)`/g;
|
|
90
|
+
for (const match of text.matchAll(backtickRe)) {
|
|
91
|
+
const raw = match[1].trim();
|
|
92
|
+
classifyToken(raw, push);
|
|
93
|
+
}
|
|
94
|
+
// 3) Bare paths with at least one slash + an extension. Authors
|
|
95
|
+
// sometimes forget the backticks; we still surface the file
|
|
96
|
+
// so the verifier can attempt the check.
|
|
97
|
+
const barePathRe = /(?<![A-Za-z0-9])((?:[a-zA-Z0-9_.\-]+\/)+[a-zA-Z0-9_.\-]+\.[a-zA-Z0-9]{1,6})/g;
|
|
98
|
+
for (const match of text.matchAll(barePathRe)) {
|
|
99
|
+
const path = match[1];
|
|
100
|
+
classifyPath(path, push);
|
|
101
|
+
}
|
|
102
|
+
return mentions;
|
|
103
|
+
}
|
|
104
|
+
function classifyToken(raw, push) {
|
|
105
|
+
const trimmed = raw.replace(/[,;.]+$/u, '').trim();
|
|
106
|
+
if (trimmed.length === 0)
|
|
107
|
+
return;
|
|
108
|
+
// Route shape inside backticks (`GET /api/x`).
|
|
109
|
+
const routeMatch = trimmed.match(/^(GET|POST|PUT|PATCH|DELETE|HEAD|OPTIONS)\s+(\/[A-Za-z0-9_./:\-]+)$/u);
|
|
110
|
+
if (routeMatch) {
|
|
111
|
+
const method = routeMatch[1].toUpperCase();
|
|
112
|
+
const path = routeMatch[2];
|
|
113
|
+
push(`route:${method} ${path}`, { kind: 'route', method, path });
|
|
114
|
+
return;
|
|
115
|
+
}
|
|
116
|
+
// Pugi command — `pugi <name>` or `/<name>`. Slash form covers
|
|
117
|
+
// REPL slash commands; pugi form covers shell commands.
|
|
118
|
+
const pugiCmdMatch = trimmed.match(/^pugi\s+([a-z][a-z0-9-]*)(?:\s+.*)?$/u);
|
|
119
|
+
if (pugiCmdMatch) {
|
|
120
|
+
const name = pugiCmdMatch[1];
|
|
121
|
+
push(`command:${name}`, { kind: 'command', name });
|
|
122
|
+
return;
|
|
123
|
+
}
|
|
124
|
+
const slashCmdMatch = trimmed.match(/^\/([a-z][a-z0-9-]*)$/u);
|
|
125
|
+
if (slashCmdMatch) {
|
|
126
|
+
const name = slashCmdMatch[1];
|
|
127
|
+
push(`command:${name}`, { kind: 'command', name });
|
|
128
|
+
return;
|
|
129
|
+
}
|
|
130
|
+
// Path shape — must contain at least one `/` AND an extension.
|
|
131
|
+
if (trimmed.includes('/') && /\.[a-zA-Z0-9]{1,6}$/.test(trimmed)) {
|
|
132
|
+
classifyPath(trimmed, push);
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
function classifyPath(path, push) {
|
|
136
|
+
if (/\.spec\.[a-z]+$|\.test\.[a-z]+$/u.test(path)) {
|
|
137
|
+
push(`test:${path}`, { kind: 'test', path });
|
|
138
|
+
return;
|
|
139
|
+
}
|
|
140
|
+
if (/^docs?\//u.test(path) || /\.md$/u.test(path)) {
|
|
141
|
+
push(`doc:${path}`, { kind: 'doc', path });
|
|
142
|
+
return;
|
|
143
|
+
}
|
|
144
|
+
push(`file:${path}`, { kind: 'file', path });
|
|
145
|
+
}
|
|
146
|
+
function extractTitle(lines) {
|
|
147
|
+
for (const line of lines) {
|
|
148
|
+
const match = line.match(TITLE_HEADING_RE);
|
|
149
|
+
if (match) {
|
|
150
|
+
return match[1].trim();
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
return null;
|
|
154
|
+
}
|
|
155
|
+
function findAcceptanceRange(lines) {
|
|
156
|
+
let startIdx = -1;
|
|
157
|
+
let startHeadingLevel = 0;
|
|
158
|
+
for (let i = 0; i < lines.length; i += 1) {
|
|
159
|
+
const line = lines[i];
|
|
160
|
+
const match = line.match(ACCEPTANCE_HEADING_RE);
|
|
161
|
+
if (match) {
|
|
162
|
+
startIdx = i + 1;
|
|
163
|
+
startHeadingLevel = match[1].length;
|
|
164
|
+
break;
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
if (startIdx === -1)
|
|
168
|
+
return null;
|
|
169
|
+
let endIdx = lines.length;
|
|
170
|
+
for (let i = startIdx; i < lines.length; i += 1) {
|
|
171
|
+
const line = lines[i];
|
|
172
|
+
const headingMatch = line.match(ANY_HEADING_RE);
|
|
173
|
+
if (!headingMatch)
|
|
174
|
+
continue;
|
|
175
|
+
const level = headingMatch[1].length;
|
|
176
|
+
if (level <= startHeadingLevel) {
|
|
177
|
+
endIdx = i;
|
|
178
|
+
break;
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
return { start: startIdx, end: endIdx };
|
|
182
|
+
}
|
|
183
|
+
function extractCriteria(sectionLines) {
|
|
184
|
+
const out = [];
|
|
185
|
+
let index = 0;
|
|
186
|
+
for (const line of sectionLines) {
|
|
187
|
+
const checklistMatch = line.match(CHECKLIST_ITEM_RE);
|
|
188
|
+
if (checklistMatch) {
|
|
189
|
+
index += 1;
|
|
190
|
+
const marker = checklistMatch[2].toLowerCase();
|
|
191
|
+
const text = checklistMatch[3];
|
|
192
|
+
out.push({
|
|
193
|
+
index,
|
|
194
|
+
text,
|
|
195
|
+
preChecked: marker === 'x',
|
|
196
|
+
mentions: extractMentions(text),
|
|
197
|
+
});
|
|
198
|
+
continue;
|
|
199
|
+
}
|
|
200
|
+
const numberedMatch = line.match(NUMBERED_ITEM_RE);
|
|
201
|
+
if (numberedMatch) {
|
|
202
|
+
index += 1;
|
|
203
|
+
const text = numberedMatch[3];
|
|
204
|
+
out.push({
|
|
205
|
+
index,
|
|
206
|
+
text,
|
|
207
|
+
preChecked: false,
|
|
208
|
+
mentions: extractMentions(text),
|
|
209
|
+
});
|
|
210
|
+
continue;
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
return out;
|
|
214
|
+
}
|
|
215
|
+
//# sourceMappingURL=parser.js.map
|
|
@@ -0,0 +1,127 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* PRD-check reporter — Pugi α7 Wave 6 (`/prd-check`, 2026-05-27).
|
|
3
|
+
*
|
|
4
|
+
* Turns a list of `VerifiedCriterion` into either a plain-text
|
|
5
|
+
* table (matches the L17 doctor renderer layout for visual
|
|
6
|
+
* consistency) OR a structured JSON envelope for scripted callers.
|
|
7
|
+
*
|
|
8
|
+
* The reporter is intentionally render-only — it does not perform
|
|
9
|
+
* any verification work. The verifier module decides PASS / FAIL /
|
|
10
|
+
* SKIPPED; the reporter only formats those verdicts. This keeps
|
|
11
|
+
* the JSON envelope deterministic + diff-friendly between runs.
|
|
12
|
+
*
|
|
13
|
+
* Exit-code policy:
|
|
14
|
+
*
|
|
15
|
+
* - `healthy` -> every criterion PASS or SKIPPED. exit 0.
|
|
16
|
+
* - `failing` -> at least one FAIL. exit 1.
|
|
17
|
+
* - `unparsed` -> the PRD had no acceptance section. exit 2
|
|
18
|
+
* (operator authored a stub but never filled it
|
|
19
|
+
* in — distinct signal from "criteria don't
|
|
20
|
+
* verify yet" so CI can route differently).
|
|
21
|
+
*/
|
|
22
|
+
/**
|
|
23
|
+
* Build the JSON envelope. Pure transform — no fs, no clock. The
|
|
24
|
+
* CLI handler wraps this in the writeOutput sink.
|
|
25
|
+
*/
|
|
26
|
+
export function buildEnvelope(input) {
|
|
27
|
+
const counts = {
|
|
28
|
+
pass: 0,
|
|
29
|
+
fail: 0,
|
|
30
|
+
skipped: 0,
|
|
31
|
+
};
|
|
32
|
+
for (const v of input.verified) {
|
|
33
|
+
counts[v.status] += 1;
|
|
34
|
+
}
|
|
35
|
+
const overall = computeOverall(input.hasAcceptanceSection, counts);
|
|
36
|
+
return {
|
|
37
|
+
command: 'prd-check',
|
|
38
|
+
prdPath: input.prdPath,
|
|
39
|
+
title: input.title,
|
|
40
|
+
overall,
|
|
41
|
+
counts,
|
|
42
|
+
criteria: input.verified.map((v) => ({
|
|
43
|
+
index: v.criterion.index,
|
|
44
|
+
text: v.criterion.text,
|
|
45
|
+
status: v.status,
|
|
46
|
+
results: v.results.map((r) => ({
|
|
47
|
+
kind: r.mention.kind,
|
|
48
|
+
target: mentionTarget(r.mention),
|
|
49
|
+
status: r.status,
|
|
50
|
+
evidence: r.evidence,
|
|
51
|
+
})),
|
|
52
|
+
})),
|
|
53
|
+
};
|
|
54
|
+
}
|
|
55
|
+
/** Exit code for the resolved overall verdict. */
|
|
56
|
+
export function exitCodeFor(overall) {
|
|
57
|
+
switch (overall) {
|
|
58
|
+
case 'healthy':
|
|
59
|
+
return 0;
|
|
60
|
+
case 'failing':
|
|
61
|
+
return 1;
|
|
62
|
+
case 'unparsed':
|
|
63
|
+
return 2;
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
/**
|
|
67
|
+
* Plain-text renderer. Mirrors the L17 doctor table for visual
|
|
68
|
+
* consistency — 4 columns: # / STATUS / CRITERION / EVIDENCE. The
|
|
69
|
+
* criterion column is truncated to 60 chars so narrow terminals
|
|
70
|
+
* stay readable; the full text lives in the JSON envelope for
|
|
71
|
+
* scripted callers that want every byte.
|
|
72
|
+
*/
|
|
73
|
+
export function renderTable(envelope) {
|
|
74
|
+
const lines = [];
|
|
75
|
+
const titlePart = envelope.title ? ` — ${envelope.title}` : '';
|
|
76
|
+
lines.push(`Pugi PRD-check${titlePart}`);
|
|
77
|
+
lines.push('='.repeat(50));
|
|
78
|
+
lines.push(`Source: ${envelope.prdPath}`);
|
|
79
|
+
lines.push('');
|
|
80
|
+
if (envelope.overall === 'unparsed') {
|
|
81
|
+
lines.push('No acceptance-criteria section found in PRD.');
|
|
82
|
+
lines.push('');
|
|
83
|
+
lines.push('Expected one of:');
|
|
84
|
+
lines.push(' ## Acceptance Criteria');
|
|
85
|
+
lines.push(' ## Success Criteria');
|
|
86
|
+
lines.push(' ## Deliverables');
|
|
87
|
+
return lines.join('\n');
|
|
88
|
+
}
|
|
89
|
+
if (envelope.criteria.length === 0) {
|
|
90
|
+
lines.push('Acceptance section present but contains 0 items.');
|
|
91
|
+
return lines.join('\n');
|
|
92
|
+
}
|
|
93
|
+
for (const c of envelope.criteria) {
|
|
94
|
+
const status = c.status.toUpperCase().padEnd(7, ' ');
|
|
95
|
+
const truncated = c.text.length > 60 ? `${c.text.slice(0, 57)}...` : c.text;
|
|
96
|
+
lines.push(`#${String(c.index).padStart(2, ' ')} ${status} ${truncated}`);
|
|
97
|
+
for (const r of c.results) {
|
|
98
|
+
const subStatus = r.status.toUpperCase().padEnd(7, ' ');
|
|
99
|
+
lines.push(` ${subStatus} ${r.evidence}`);
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
lines.push('');
|
|
103
|
+
const { pass, fail, skipped } = envelope.counts;
|
|
104
|
+
const summary = envelope.overall === 'healthy' ? 'HEALTHY' : envelope.overall === 'failing' ? 'FAILING' : 'UNPARSED';
|
|
105
|
+
lines.push(`${fail} fail · ${pass} pass · ${skipped} skipped. Overall: ${summary}`);
|
|
106
|
+
return lines.join('\n');
|
|
107
|
+
}
|
|
108
|
+
function computeOverall(hasAcceptanceSection, counts) {
|
|
109
|
+
if (!hasAcceptanceSection)
|
|
110
|
+
return 'unparsed';
|
|
111
|
+
if (counts.fail > 0)
|
|
112
|
+
return 'failing';
|
|
113
|
+
return 'healthy';
|
|
114
|
+
}
|
|
115
|
+
function mentionTarget(mention) {
|
|
116
|
+
switch (mention.kind) {
|
|
117
|
+
case 'file':
|
|
118
|
+
case 'test':
|
|
119
|
+
case 'doc':
|
|
120
|
+
return mention.path;
|
|
121
|
+
case 'command':
|
|
122
|
+
return mention.name;
|
|
123
|
+
case 'route':
|
|
124
|
+
return `${mention.method} ${mention.path}`;
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
//# sourceMappingURL=reporter.js.map
|