gsd-pi 2.62.0-dev.f6ad485 → 2.62.1-dev.1ae2b74
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/dist/resources/extensions/ask-user-questions.js +47 -3
- package/dist/resources/extensions/gsd/auto/loop.js +8 -1
- package/dist/resources/extensions/gsd/auto/phases.js +10 -3
- package/dist/resources/extensions/gsd/auto-post-unit.js +6 -4
- package/dist/resources/extensions/gsd/auto-start.js +11 -6
- package/dist/resources/extensions/gsd/auto-timers.js +8 -2
- package/dist/resources/extensions/gsd/auto-verification.js +14 -3
- package/dist/resources/extensions/gsd/auto-worktree.js +19 -0
- package/dist/resources/extensions/gsd/auto.js +24 -0
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +4 -0
- package/dist/resources/extensions/gsd/bootstrap/tool-call-loop-guard.js +11 -1
- package/dist/resources/extensions/gsd/db-writer.js +64 -28
- package/dist/resources/extensions/gsd/preferences-models.js +74 -0
- package/dist/resources/extensions/gsd/preferences-skills.js +6 -1
- package/dist/resources/extensions/gsd/prompts/guided-discuss-milestone.md +1 -1
- package/dist/resources/extensions/gsd/prompts/guided-discuss-slice.md +1 -1
- package/dist/resources/extensions/gsd/skill-catalog.js +6 -4
- package/dist/resources/extensions/gsd/skill-discovery.js +24 -6
- package/dist/resources/extensions/gsd/skill-health.js +7 -3
- package/dist/resources/extensions/gsd/skill-telemetry.js +5 -2
- package/dist/resources/extensions/gsd/state.js +1 -0
- package/dist/resources/extensions/gsd/tools/complete-slice.js +3 -3
- package/dist/resources/extensions/gsd/workflow-logger.js +13 -8
- package/dist/resources/extensions/gsd/workflow-reconcile.js +3 -1
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +20 -20
- package/dist/web/standalone/.next/build-manifest.json +2 -2
- package/dist/web/standalone/.next/prerender-manifest.json +3 -3
- package/dist/web/standalone/.next/server/app/_global-error.html +2 -2
- package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app-paths-manifest.json +20 -20
- package/dist/web/standalone/.next/server/pages/404.html +1 -1
- package/dist/web/standalone/.next/server/pages/500.html +2 -2
- package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
- package/package.json +1 -1
- package/packages/mcp-server/src/cli.ts +1 -1
- package/packages/mcp-server/src/index.ts +15 -1
- package/packages/mcp-server/src/readers/captures.ts +119 -0
- package/packages/mcp-server/src/readers/doctor-lite.ts +225 -0
- package/packages/mcp-server/src/readers/index.ts +16 -0
- package/packages/mcp-server/src/readers/knowledge.ts +111 -0
- package/packages/mcp-server/src/readers/metrics.ts +118 -0
- package/packages/mcp-server/src/readers/paths.ts +217 -0
- package/packages/mcp-server/src/readers/readers.test.ts +509 -0
- package/packages/mcp-server/src/readers/roadmap.ts +263 -0
- package/packages/mcp-server/src/readers/state.ts +223 -0
- package/packages/mcp-server/src/server.ts +134 -3
- package/packages/pi-ai/dist/utils/repair-tool-json.d.ts +26 -6
- package/packages/pi-ai/dist/utils/repair-tool-json.d.ts.map +1 -1
- package/packages/pi-ai/dist/utils/repair-tool-json.js +67 -9
- package/packages/pi-ai/dist/utils/repair-tool-json.js.map +1 -1
- package/packages/pi-ai/dist/utils/tests/repair-tool-json.test.js +73 -1
- package/packages/pi-ai/dist/utils/tests/repair-tool-json.test.js.map +1 -1
- package/packages/pi-ai/src/utils/repair-tool-json.ts +74 -10
- package/packages/pi-ai/src/utils/tests/repair-tool-json.test.ts +94 -1
- package/packages/pi-coding-agent/dist/core/agent-session-model-switch.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/agent-session-model-switch.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/agent-session-model-switch.test.js +16 -0
- package/packages/pi-coding-agent/dist/core/agent-session-model-switch.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/agent-session.js +4 -0
- package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/retry-handler.d.ts +3 -0
- package/packages/pi-coding-agent/dist/core/retry-handler.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/retry-handler.js +48 -16
- package/packages/pi-coding-agent/dist/core/retry-handler.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/retry-handler.test.js +20 -3
- package/packages/pi-coding-agent/dist/core/retry-handler.test.js.map +1 -1
- package/packages/pi-coding-agent/package.json +1 -1
- package/packages/pi-coding-agent/src/core/agent-session-model-switch.test.ts +21 -0
- package/packages/pi-coding-agent/src/core/agent-session.ts +4 -0
- package/packages/pi-coding-agent/src/core/retry-handler.test.ts +30 -3
- package/packages/pi-coding-agent/src/core/retry-handler.ts +49 -16
- package/pkg/package.json +1 -1
- package/src/resources/extensions/ask-user-questions.ts +60 -4
- package/src/resources/extensions/gsd/auto/loop.ts +8 -1
- package/src/resources/extensions/gsd/auto/phases.ts +8 -6
- package/src/resources/extensions/gsd/auto-post-unit.ts +6 -3
- package/src/resources/extensions/gsd/auto-start.ts +11 -6
- package/src/resources/extensions/gsd/auto-timers.ts +8 -2
- package/src/resources/extensions/gsd/auto-verification.ts +14 -3
- package/src/resources/extensions/gsd/auto-worktree.ts +18 -0
- package/src/resources/extensions/gsd/auto.ts +25 -0
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +4 -0
- package/src/resources/extensions/gsd/bootstrap/tool-call-loop-guard.ts +13 -1
- package/src/resources/extensions/gsd/db-writer.ts +67 -30
- package/src/resources/extensions/gsd/preferences-models.ts +78 -0
- package/src/resources/extensions/gsd/preferences-skills.ts +6 -1
- package/src/resources/extensions/gsd/prompts/guided-discuss-milestone.md +1 -1
- package/src/resources/extensions/gsd/prompts/guided-discuss-slice.md +1 -1
- package/src/resources/extensions/gsd/skill-catalog.ts +6 -3
- package/src/resources/extensions/gsd/skill-discovery.ts +23 -6
- package/src/resources/extensions/gsd/skill-health.ts +7 -3
- package/src/resources/extensions/gsd/skill-telemetry.ts +5 -2
- package/src/resources/extensions/gsd/state.ts +1 -0
- package/src/resources/extensions/gsd/tests/ask-user-questions-dedup.test.ts +120 -0
- package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +22 -2
- package/src/resources/extensions/gsd/tests/auto-wrapup-inflight-guard.test.ts +107 -0
- package/src/resources/extensions/gsd/tests/claude-skill-dirs.test.ts +51 -0
- package/src/resources/extensions/gsd/tests/db-writer.test.ts +41 -0
- package/src/resources/extensions/gsd/tests/model-isolation.test.ts +75 -1
- package/src/resources/extensions/gsd/tests/tool-call-loop-guard.test.ts +17 -4
- package/src/resources/extensions/gsd/tests/workflow-logger.test.ts +17 -41
- package/src/resources/extensions/gsd/tests/worktree-db-respawn-truncation.test.ts +81 -2
- package/src/resources/extensions/gsd/tools/complete-slice.ts +3 -5
- package/src/resources/extensions/gsd/workflow-logger.ts +13 -8
- package/src/resources/extensions/gsd/workflow-reconcile.ts +3 -1
- package/src/resources/extensions/shared/tests/ask-user-freetext.test.ts +6 -1
- /package/dist/web/standalone/.next/static/{fbkSIi4k8fmB8mi0Sq9sF → erQZ_8_1lkclnPJLJnCxG}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{fbkSIi4k8fmB8mi0Sq9sF → erQZ_8_1lkclnPJLJnCxG}/_ssgManifest.js +0 -0
|
@@ -0,0 +1,217 @@
|
|
|
1
|
+
// GSD MCP Server — .gsd/ directory resolution
|
|
2
|
+
// Copyright (c) 2026 Jeremy McSpadden <jeremy@fluxlabs.net>
|
|
3
|
+
|
|
4
|
+
import { existsSync, statSync, readdirSync } from 'node:fs';
|
|
5
|
+
import { join, resolve, dirname, basename } from 'node:path';
|
|
6
|
+
import { execFileSync } from 'node:child_process';
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Resolve the .gsd/ root directory for a project.
|
|
10
|
+
*
|
|
11
|
+
* Probes in order:
|
|
12
|
+
* 1. projectDir/.gsd (fast path)
|
|
13
|
+
* 2. git repo root/.gsd
|
|
14
|
+
* 3. Walk up from projectDir
|
|
15
|
+
* 4. Fallback: projectDir/.gsd (even if missing — for init)
|
|
16
|
+
*/
|
|
17
|
+
export function resolveGsdRoot(projectDir: string): string {
|
|
18
|
+
const resolved = resolve(projectDir);
|
|
19
|
+
|
|
20
|
+
// Fast path: .gsd/ in the given directory
|
|
21
|
+
const direct = join(resolved, '.gsd');
|
|
22
|
+
if (existsSync(direct) && statSync(direct).isDirectory()) {
|
|
23
|
+
return direct;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
// Try git repo root
|
|
27
|
+
try {
|
|
28
|
+
const gitRoot = execFileSync('git', ['rev-parse', '--show-toplevel'], {
|
|
29
|
+
cwd: resolved,
|
|
30
|
+
encoding: 'utf-8',
|
|
31
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
32
|
+
}).trim();
|
|
33
|
+
const gitGsd = join(gitRoot, '.gsd');
|
|
34
|
+
if (existsSync(gitGsd) && statSync(gitGsd).isDirectory()) {
|
|
35
|
+
return gitGsd;
|
|
36
|
+
}
|
|
37
|
+
} catch {
|
|
38
|
+
// Not a git repo or git not available
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
// Walk up from projectDir
|
|
42
|
+
let dir = resolved;
|
|
43
|
+
while (dir !== dirname(dir)) {
|
|
44
|
+
const candidate = join(dir, '.gsd');
|
|
45
|
+
if (existsSync(candidate) && statSync(candidate).isDirectory()) {
|
|
46
|
+
return candidate;
|
|
47
|
+
}
|
|
48
|
+
dir = dirname(dir);
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
// Fallback
|
|
52
|
+
return direct;
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
/** Resolve path to a .gsd/ root file (STATE.md, KNOWLEDGE.md, etc.) */
|
|
56
|
+
export function resolveRootFile(gsdRoot: string, name: string): string {
|
|
57
|
+
return join(gsdRoot, name);
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
/** Resolve path to milestones directory */
|
|
61
|
+
export function milestonesDir(gsdRoot: string): string {
|
|
62
|
+
return join(gsdRoot, 'milestones');
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
/**
|
|
66
|
+
* Find all milestone directory IDs (M001, M002, etc.).
|
|
67
|
+
* Handles both bare (M001/) and descriptor (M001-FLIGHT-SIM/) naming.
|
|
68
|
+
*/
|
|
69
|
+
export function findMilestoneIds(gsdRoot: string): string[] {
|
|
70
|
+
const dir = milestonesDir(gsdRoot);
|
|
71
|
+
if (!existsSync(dir)) return [];
|
|
72
|
+
|
|
73
|
+
const entries = readdirSync(dir, { withFileTypes: true });
|
|
74
|
+
const ids: string[] = [];
|
|
75
|
+
|
|
76
|
+
for (const entry of entries) {
|
|
77
|
+
if (!entry.isDirectory()) continue;
|
|
78
|
+
const match = entry.name.match(/^(M\d+)/);
|
|
79
|
+
if (match) ids.push(match[1]);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
return ids.sort();
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
/**
|
|
86
|
+
* Resolve the actual directory name for a milestone ID.
|
|
87
|
+
* M001 might live in M001/ or M001-SOME-DESCRIPTOR/.
|
|
88
|
+
*/
|
|
89
|
+
export function resolveMilestoneDir(gsdRoot: string, milestoneId: string): string | null {
|
|
90
|
+
const dir = milestonesDir(gsdRoot);
|
|
91
|
+
if (!existsSync(dir)) return null;
|
|
92
|
+
|
|
93
|
+
// Fast path: exact match
|
|
94
|
+
const exact = join(dir, milestoneId);
|
|
95
|
+
if (existsSync(exact) && statSync(exact).isDirectory()) return exact;
|
|
96
|
+
|
|
97
|
+
// Prefix match
|
|
98
|
+
const entries = readdirSync(dir, { withFileTypes: true });
|
|
99
|
+
for (const entry of entries) {
|
|
100
|
+
if (entry.isDirectory() && entry.name.startsWith(milestoneId)) {
|
|
101
|
+
return join(dir, entry.name);
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
return null;
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
/**
|
|
109
|
+
* Resolve a milestone-level file (M001-ROADMAP.md, M001-CONTEXT.md, etc.).
|
|
110
|
+
* Handles various naming conventions.
|
|
111
|
+
*/
|
|
112
|
+
export function resolveMilestoneFile(gsdRoot: string, milestoneId: string, suffix: string): string | null {
|
|
113
|
+
const mDir = resolveMilestoneDir(gsdRoot, milestoneId);
|
|
114
|
+
if (!mDir) return null;
|
|
115
|
+
|
|
116
|
+
const dirName = basename(mDir);
|
|
117
|
+
|
|
118
|
+
// Try: M001-ROADMAP.md, then DIRNAME-ROADMAP.md
|
|
119
|
+
const candidates = [
|
|
120
|
+
join(mDir, `${milestoneId}-${suffix}.md`),
|
|
121
|
+
join(mDir, `${dirName}-${suffix}.md`),
|
|
122
|
+
join(mDir, `${suffix}.md`),
|
|
123
|
+
];
|
|
124
|
+
|
|
125
|
+
for (const c of candidates) {
|
|
126
|
+
if (existsSync(c)) return c;
|
|
127
|
+
}
|
|
128
|
+
return null;
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
/** Find all slice IDs within a milestone (S01, S02, etc.) */
|
|
132
|
+
export function findSliceIds(gsdRoot: string, milestoneId: string): string[] {
|
|
133
|
+
const mDir = resolveMilestoneDir(gsdRoot, milestoneId);
|
|
134
|
+
if (!mDir) return [];
|
|
135
|
+
|
|
136
|
+
const slicesDir = join(mDir, 'slices');
|
|
137
|
+
if (!existsSync(slicesDir)) return [];
|
|
138
|
+
|
|
139
|
+
const entries = readdirSync(slicesDir, { withFileTypes: true });
|
|
140
|
+
const ids: string[] = [];
|
|
141
|
+
|
|
142
|
+
for (const entry of entries) {
|
|
143
|
+
if (!entry.isDirectory()) continue;
|
|
144
|
+
const match = entry.name.match(/^(S\d+)/);
|
|
145
|
+
if (match) ids.push(match[1]);
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
return ids.sort();
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
/** Resolve the actual directory for a slice */
|
|
152
|
+
export function resolveSliceDir(gsdRoot: string, milestoneId: string, sliceId: string): string | null {
|
|
153
|
+
const mDir = resolveMilestoneDir(gsdRoot, milestoneId);
|
|
154
|
+
if (!mDir) return null;
|
|
155
|
+
|
|
156
|
+
const slicesDir = join(mDir, 'slices');
|
|
157
|
+
if (!existsSync(slicesDir)) return null;
|
|
158
|
+
|
|
159
|
+
const exact = join(slicesDir, sliceId);
|
|
160
|
+
if (existsSync(exact) && statSync(exact).isDirectory()) return exact;
|
|
161
|
+
|
|
162
|
+
const entries = readdirSync(slicesDir, { withFileTypes: true });
|
|
163
|
+
for (const entry of entries) {
|
|
164
|
+
if (entry.isDirectory() && entry.name.startsWith(sliceId)) {
|
|
165
|
+
return join(slicesDir, entry.name);
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
return null;
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
/** Resolve a slice-level file (S01-PLAN.md, etc.) */
|
|
172
|
+
export function resolveSliceFile(
|
|
173
|
+
gsdRoot: string, milestoneId: string, sliceId: string, suffix: string,
|
|
174
|
+
): string | null {
|
|
175
|
+
const sDir = resolveSliceDir(gsdRoot, milestoneId, sliceId);
|
|
176
|
+
if (!sDir) return null;
|
|
177
|
+
|
|
178
|
+
const dirName = basename(sDir);
|
|
179
|
+
const candidates = [
|
|
180
|
+
join(sDir, `${sliceId}-${suffix}.md`),
|
|
181
|
+
join(sDir, `${dirName}-${suffix}.md`),
|
|
182
|
+
join(sDir, `${suffix}.md`),
|
|
183
|
+
];
|
|
184
|
+
|
|
185
|
+
for (const c of candidates) {
|
|
186
|
+
if (existsSync(c)) return c;
|
|
187
|
+
}
|
|
188
|
+
return null;
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
/** Find all task files in a slice's tasks/ directory */
|
|
192
|
+
export function findTaskFiles(
|
|
193
|
+
gsdRoot: string, milestoneId: string, sliceId: string,
|
|
194
|
+
): Array<{ id: string; hasPlan: boolean; hasSummary: boolean }> {
|
|
195
|
+
const sDir = resolveSliceDir(gsdRoot, milestoneId, sliceId);
|
|
196
|
+
if (!sDir) return [];
|
|
197
|
+
|
|
198
|
+
const tasksDir = join(sDir, 'tasks');
|
|
199
|
+
if (!existsSync(tasksDir)) return [];
|
|
200
|
+
|
|
201
|
+
const files = readdirSync(tasksDir);
|
|
202
|
+
const taskMap = new Map<string, { hasPlan: boolean; hasSummary: boolean }>();
|
|
203
|
+
|
|
204
|
+
for (const f of files) {
|
|
205
|
+
const match = f.match(/^(T\d+).*-(PLAN|SUMMARY)\.md$/i);
|
|
206
|
+
if (!match) continue;
|
|
207
|
+
const [, id, type] = match;
|
|
208
|
+
const existing = taskMap.get(id) ?? { hasPlan: false, hasSummary: false };
|
|
209
|
+
if (type.toUpperCase() === 'PLAN') existing.hasPlan = true;
|
|
210
|
+
if (type.toUpperCase() === 'SUMMARY') existing.hasSummary = true;
|
|
211
|
+
taskMap.set(id, existing);
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
return Array.from(taskMap.entries())
|
|
215
|
+
.map(([id, info]) => ({ id, ...info }))
|
|
216
|
+
.sort((a, b) => a.id.localeCompare(b.id));
|
|
217
|
+
}
|