gsd-pi 2.37.1-dev.d3ace49 → 2.38.0-dev.361f5e3
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/app-paths.js +1 -1
- package/dist/cli.js +9 -0
- package/dist/extension-discovery.d.ts +5 -3
- package/dist/extension-discovery.js +14 -9
- package/dist/extension-registry.js +2 -2
- package/dist/remote-questions-config.js +2 -2
- package/dist/resources/extensions/browser-tools/package.json +3 -1
- package/dist/resources/extensions/cmux/index.js +55 -1
- package/dist/resources/extensions/context7/package.json +1 -1
- package/dist/resources/extensions/env-utils.js +29 -0
- package/dist/resources/extensions/get-secrets-from-user.js +5 -24
- package/dist/resources/extensions/github-sync/cli.js +284 -0
- package/dist/resources/extensions/github-sync/index.js +73 -0
- package/dist/resources/extensions/github-sync/mapping.js +67 -0
- package/dist/resources/extensions/github-sync/sync.js +424 -0
- package/dist/resources/extensions/github-sync/templates.js +118 -0
- package/dist/resources/extensions/github-sync/types.js +7 -0
- package/dist/resources/extensions/google-search/package.json +3 -1
- package/dist/resources/extensions/gsd/auto/session.js +6 -23
- package/dist/resources/extensions/gsd/auto-dispatch.js +7 -8
- package/dist/resources/extensions/gsd/auto-loop.js +149 -170
- package/dist/resources/extensions/gsd/auto-post-unit.js +92 -70
- package/dist/resources/extensions/gsd/auto-prompts.js +7 -31
- package/dist/resources/extensions/gsd/auto-start.js +13 -2
- package/dist/resources/extensions/gsd/auto-worktree-sync.js +13 -5
- package/dist/resources/extensions/gsd/auto.js +143 -96
- package/dist/resources/extensions/gsd/captures.js +9 -1
- package/dist/resources/extensions/gsd/commands-extensions.js +3 -2
- package/dist/resources/extensions/gsd/commands-handlers.js +16 -3
- package/dist/resources/extensions/gsd/commands-prefs-wizard.js +1 -1
- package/dist/resources/extensions/gsd/commands.js +22 -2
- package/dist/resources/extensions/gsd/context-budget.js +2 -10
- package/dist/resources/extensions/gsd/detection.js +1 -2
- package/dist/resources/extensions/gsd/docs/preferences-reference.md +0 -2
- package/dist/resources/extensions/gsd/doctor-checks.js +82 -0
- package/dist/resources/extensions/gsd/doctor-environment.js +78 -0
- package/dist/resources/extensions/gsd/doctor-format.js +15 -0
- package/dist/resources/extensions/gsd/doctor-providers.js +27 -11
- package/dist/resources/extensions/gsd/doctor.js +184 -11
- package/dist/resources/extensions/gsd/export.js +1 -1
- package/dist/resources/extensions/gsd/files.js +2 -2
- package/dist/resources/extensions/gsd/forensics.js +1 -1
- package/dist/resources/extensions/gsd/git-service.js +8 -1
- package/dist/resources/extensions/gsd/index.js +2 -1
- package/dist/resources/extensions/gsd/migrate/parsers.js +1 -1
- package/dist/resources/extensions/gsd/package.json +1 -1
- package/dist/resources/extensions/gsd/preferences-models.js +0 -12
- package/dist/resources/extensions/gsd/preferences-types.js +1 -1
- package/dist/resources/extensions/gsd/preferences-validation.js +59 -11
- package/dist/resources/extensions/gsd/preferences.js +8 -5
- package/dist/resources/extensions/gsd/prompts/discuss.md +11 -14
- package/dist/resources/extensions/gsd/prompts/execute-task.md +2 -2
- package/dist/resources/extensions/gsd/prompts/guided-discuss-milestone.md +11 -12
- package/dist/resources/extensions/gsd/prompts/guided-discuss-slice.md +8 -10
- package/dist/resources/extensions/gsd/prompts/guided-resume-task.md +1 -1
- package/dist/resources/extensions/gsd/prompts/queue.md +4 -8
- package/dist/resources/extensions/gsd/prompts/reactive-execute.md +11 -8
- package/dist/resources/extensions/gsd/prompts/run-uat.md +25 -10
- package/dist/resources/extensions/gsd/prompts/workflow-start.md +2 -2
- package/dist/resources/extensions/gsd/repo-identity.js +21 -4
- package/dist/resources/extensions/gsd/resource-version.js +2 -1
- package/dist/resources/extensions/gsd/state.js +1 -1
- package/dist/resources/extensions/gsd/visualizer-data.js +1 -1
- package/dist/resources/extensions/gsd/worktree.js +35 -16
- package/dist/resources/extensions/remote-questions/status.js +2 -1
- package/dist/resources/extensions/remote-questions/store.js +2 -1
- package/dist/resources/extensions/search-the-web/provider.js +2 -1
- package/dist/resources/extensions/subagent/index.js +12 -3
- package/dist/resources/extensions/subagent/isolation.js +2 -1
- package/dist/resources/extensions/ttsr/rule-loader.js +2 -1
- package/dist/resources/extensions/universal-config/package.json +1 -1
- package/dist/welcome-screen.d.ts +12 -0
- package/dist/welcome-screen.js +53 -0
- package/package.json +1 -1
- package/packages/pi-coding-agent/dist/core/package-manager.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/package-manager.js +8 -4
- package/packages/pi-coding-agent/dist/core/package-manager.js.map +1 -1
- package/packages/pi-coding-agent/package.json +1 -1
- package/packages/pi-coding-agent/src/core/package-manager.ts +8 -4
- package/pkg/package.json +1 -1
- package/src/resources/extensions/cmux/index.ts +57 -1
- package/src/resources/extensions/env-utils.ts +31 -0
- package/src/resources/extensions/get-secrets-from-user.ts +5 -24
- package/src/resources/extensions/github-sync/cli.ts +364 -0
- package/src/resources/extensions/github-sync/index.ts +93 -0
- package/src/resources/extensions/github-sync/mapping.ts +81 -0
- package/src/resources/extensions/github-sync/sync.ts +556 -0
- package/src/resources/extensions/github-sync/templates.ts +183 -0
- package/src/resources/extensions/github-sync/tests/cli.test.ts +20 -0
- package/src/resources/extensions/github-sync/tests/commit-linking.test.ts +39 -0
- package/src/resources/extensions/github-sync/tests/mapping.test.ts +104 -0
- package/src/resources/extensions/github-sync/tests/templates.test.ts +110 -0
- package/src/resources/extensions/github-sync/types.ts +47 -0
- package/src/resources/extensions/gsd/auto/session.ts +7 -25
- package/src/resources/extensions/gsd/auto-dispatch.ts +6 -8
- package/src/resources/extensions/gsd/auto-loop.ts +207 -252
- package/src/resources/extensions/gsd/auto-post-unit.ts +69 -41
- package/src/resources/extensions/gsd/auto-prompts.ts +7 -33
- package/src/resources/extensions/gsd/auto-start.ts +18 -2
- package/src/resources/extensions/gsd/auto-worktree-sync.ts +15 -4
- package/src/resources/extensions/gsd/auto.ts +139 -101
- package/src/resources/extensions/gsd/captures.ts +10 -1
- package/src/resources/extensions/gsd/commands-extensions.ts +4 -2
- package/src/resources/extensions/gsd/commands-handlers.ts +17 -2
- package/src/resources/extensions/gsd/commands-prefs-wizard.ts +1 -1
- package/src/resources/extensions/gsd/commands.ts +24 -2
- package/src/resources/extensions/gsd/context-budget.ts +2 -12
- package/src/resources/extensions/gsd/detection.ts +2 -2
- package/src/resources/extensions/gsd/docs/preferences-reference.md +0 -2
- package/src/resources/extensions/gsd/doctor-checks.ts +75 -0
- package/src/resources/extensions/gsd/doctor-environment.ts +82 -1
- package/src/resources/extensions/gsd/doctor-format.ts +20 -0
- package/src/resources/extensions/gsd/doctor-providers.ts +26 -9
- package/src/resources/extensions/gsd/doctor-types.ts +16 -1
- package/src/resources/extensions/gsd/doctor.ts +177 -13
- package/src/resources/extensions/gsd/export.ts +1 -1
- package/src/resources/extensions/gsd/files.ts +2 -2
- package/src/resources/extensions/gsd/forensics.ts +1 -1
- package/src/resources/extensions/gsd/git-service.ts +13 -1
- package/src/resources/extensions/gsd/index.ts +3 -1
- package/src/resources/extensions/gsd/migrate/parsers.ts +1 -1
- package/src/resources/extensions/gsd/preferences-models.ts +0 -12
- package/src/resources/extensions/gsd/preferences-types.ts +4 -4
- package/src/resources/extensions/gsd/preferences-validation.ts +51 -11
- package/src/resources/extensions/gsd/preferences.ts +8 -5
- package/src/resources/extensions/gsd/prompts/discuss.md +11 -14
- package/src/resources/extensions/gsd/prompts/execute-task.md +2 -2
- package/src/resources/extensions/gsd/prompts/guided-discuss-milestone.md +11 -12
- package/src/resources/extensions/gsd/prompts/guided-discuss-slice.md +8 -10
- package/src/resources/extensions/gsd/prompts/guided-resume-task.md +1 -1
- package/src/resources/extensions/gsd/prompts/queue.md +4 -8
- package/src/resources/extensions/gsd/prompts/reactive-execute.md +11 -8
- package/src/resources/extensions/gsd/prompts/run-uat.md +25 -10
- package/src/resources/extensions/gsd/prompts/workflow-start.md +2 -2
- package/src/resources/extensions/gsd/repo-identity.ts +23 -4
- package/src/resources/extensions/gsd/resource-version.ts +3 -1
- package/src/resources/extensions/gsd/state.ts +1 -1
- package/src/resources/extensions/gsd/tests/agent-end-retry.test.ts +21 -18
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +16 -37
- package/src/resources/extensions/gsd/tests/cmux.test.ts +93 -0
- package/src/resources/extensions/gsd/tests/doctor-enhancements.test.ts +266 -0
- package/src/resources/extensions/gsd/tests/doctor-providers.test.ts +86 -3
- package/src/resources/extensions/gsd/tests/preferences.test.ts +2 -7
- package/src/resources/extensions/gsd/tests/prompt-contracts.test.ts +59 -0
- package/src/resources/extensions/gsd/tests/repo-identity-worktree.test.ts +21 -1
- package/src/resources/extensions/gsd/tests/run-uat.test.ts +11 -3
- package/src/resources/extensions/gsd/tests/worktree.test.ts +47 -0
- package/src/resources/extensions/gsd/types.ts +0 -1
- package/src/resources/extensions/gsd/visualizer-data.ts +1 -1
- package/src/resources/extensions/gsd/worktree.ts +35 -15
- package/src/resources/extensions/remote-questions/status.ts +3 -1
- package/src/resources/extensions/remote-questions/store.ts +3 -1
- package/src/resources/extensions/search-the-web/provider.ts +2 -1
- package/src/resources/extensions/subagent/index.ts +12 -3
- package/src/resources/extensions/subagent/isolation.ts +3 -1
- package/src/resources/extensions/ttsr/rule-loader.ts +3 -1
- package/dist/resources/extensions/gsd/prompt-compressor.js +0 -393
- package/dist/resources/extensions/gsd/semantic-chunker.js +0 -254
- package/dist/resources/extensions/gsd/summary-distiller.js +0 -212
- package/src/resources/extensions/gsd/prompt-compressor.ts +0 -508
- package/src/resources/extensions/gsd/semantic-chunker.ts +0 -336
- package/src/resources/extensions/gsd/summary-distiller.ts +0 -258
- package/src/resources/extensions/gsd/tests/context-compression.test.ts +0 -193
- package/src/resources/extensions/gsd/tests/prompt-compressor.test.ts +0 -529
- package/src/resources/extensions/gsd/tests/semantic-chunker.test.ts +0 -426
- package/src/resources/extensions/gsd/tests/summary-distiller.test.ts +0 -323
- package/src/resources/extensions/gsd/tests/token-optimization-benchmark.test.ts +0 -1272
- package/src/resources/extensions/gsd/tests/token-optimization-prefs.test.ts +0 -164
|
@@ -22,7 +22,6 @@ import { writeUnitRuntimeRecord, clearUnitRuntimeRecord } from "./unit-runtime.j
|
|
|
22
22
|
import { runGSDDoctor, rebuildState, summarizeDoctorIssues } from "./doctor.js";
|
|
23
23
|
import { recordHealthSnapshot, checkHealEscalation } from "./doctor-proactive.js";
|
|
24
24
|
import { syncStateToProjectRoot } from "./auto-worktree-sync.js";
|
|
25
|
-
import { resetRewriteCircuitBreaker } from "./auto-dispatch.js";
|
|
26
25
|
import { isDbAvailable } from "./gsd-db.js";
|
|
27
26
|
import { consumeSignal } from "./session-status-io.js";
|
|
28
27
|
import { checkPostUnitHooks, isRetryPending, consumeRetryTrigger, persistHookState, } from "./post-unit-hooks.js";
|
|
@@ -36,7 +35,7 @@ const STATE_REBUILD_MIN_INTERVAL_MS = 30_000;
|
|
|
36
35
|
*
|
|
37
36
|
* Returns "dispatched" if a signal caused stop/pause, "continue" to proceed.
|
|
38
37
|
*/
|
|
39
|
-
export async function postUnitPreVerification(pctx) {
|
|
38
|
+
export async function postUnitPreVerification(pctx, opts) {
|
|
40
39
|
const { s, ctx, pi, buildSnapshotOpts, stopAuto, pauseAuto } = pctx;
|
|
41
40
|
// ── Parallel worker signal check ──
|
|
42
41
|
const milestoneLock = process.env.GSD_MILESTONE_LOCK;
|
|
@@ -55,8 +54,10 @@ export async function postUnitPreVerification(pctx) {
|
|
|
55
54
|
}
|
|
56
55
|
// Invalidate all caches
|
|
57
56
|
invalidateAllCaches();
|
|
58
|
-
// Small delay to let files settle
|
|
59
|
-
|
|
57
|
+
// Small delay to let files settle (skipped for sidecars where latency matters more)
|
|
58
|
+
if (!opts?.skipSettleDelay) {
|
|
59
|
+
await new Promise(r => setTimeout(r, 100));
|
|
60
|
+
}
|
|
60
61
|
// Auto-commit
|
|
61
62
|
if (s.currentUnit) {
|
|
62
63
|
try {
|
|
@@ -71,16 +72,26 @@ export async function postUnitPreVerification(pctx) {
|
|
|
71
72
|
const summaryContent = await loadFile(summaryPath);
|
|
72
73
|
if (summaryContent) {
|
|
73
74
|
const summary = parseSummary(summaryContent);
|
|
75
|
+
// Look up GitHub issue number for commit linking
|
|
76
|
+
let ghIssueNumber;
|
|
77
|
+
try {
|
|
78
|
+
const { getTaskIssueNumberForCommit } = await import("../github-sync/sync.js");
|
|
79
|
+
ghIssueNumber = getTaskIssueNumberForCommit(s.basePath, mid, sid, tid) ?? undefined;
|
|
80
|
+
}
|
|
81
|
+
catch {
|
|
82
|
+
// GitHub sync not available — skip
|
|
83
|
+
}
|
|
74
84
|
taskContext = {
|
|
75
85
|
taskId: `${sid}/${tid}`,
|
|
76
86
|
taskTitle: summary.title?.replace(/^T\d+:\s*/, "") || tid,
|
|
77
87
|
oneLiner: summary.oneLiner || undefined,
|
|
78
88
|
keyFiles: summary.frontmatter.key_files?.filter(f => !f.includes("{{")) || undefined,
|
|
89
|
+
issueNumber: ghIssueNumber,
|
|
79
90
|
};
|
|
80
91
|
}
|
|
81
92
|
}
|
|
82
|
-
catch {
|
|
83
|
-
|
|
93
|
+
catch (e) {
|
|
94
|
+
debugLog("postUnit", { phase: "task-summary-parse", error: String(e) });
|
|
84
95
|
}
|
|
85
96
|
}
|
|
86
97
|
}
|
|
@@ -90,57 +101,68 @@ export async function postUnitPreVerification(pctx) {
|
|
|
90
101
|
ctx.ui.notify(`Committed: ${commitMsg.split("\n")[0]}`, "info");
|
|
91
102
|
}
|
|
92
103
|
}
|
|
93
|
-
catch {
|
|
94
|
-
|
|
104
|
+
catch (e) {
|
|
105
|
+
debugLog("postUnit", { phase: "auto-commit", error: String(e) });
|
|
95
106
|
}
|
|
96
|
-
//
|
|
107
|
+
// GitHub sync (non-blocking, opt-in)
|
|
97
108
|
try {
|
|
98
|
-
const
|
|
99
|
-
|
|
100
|
-
const sliceTerminalUnits = new Set(["complete-slice", "run-uat"]);
|
|
101
|
-
const effectiveFixLevel = sliceTerminalUnits.has(s.currentUnit.type) ? "all" : "task";
|
|
102
|
-
const report = await runGSDDoctor(s.basePath, { fix: true, scope: doctorScope, fixLevel: effectiveFixLevel });
|
|
103
|
-
if (report.fixesApplied.length > 0) {
|
|
104
|
-
ctx.ui.notify(`Post-hook: applied ${report.fixesApplied.length} fix(es).`, "info");
|
|
105
|
-
}
|
|
106
|
-
// Proactive health tracking
|
|
107
|
-
const summary = summarizeDoctorIssues(report.issues);
|
|
108
|
-
recordHealthSnapshot(summary.errors, summary.warnings, report.fixesApplied.length);
|
|
109
|
-
// Check if we should escalate to LLM-assisted heal
|
|
110
|
-
if (summary.errors > 0) {
|
|
111
|
-
const unresolvedErrors = report.issues
|
|
112
|
-
.filter(i => i.severity === "error" && !i.fixable)
|
|
113
|
-
.map(i => ({ code: i.code, message: i.message, unitId: i.unitId }));
|
|
114
|
-
const escalation = checkHealEscalation(summary.errors, unresolvedErrors);
|
|
115
|
-
if (escalation.shouldEscalate) {
|
|
116
|
-
ctx.ui.notify(`Doctor heal escalation: ${escalation.reason}. Dispatching LLM-assisted heal.`, "warning");
|
|
117
|
-
try {
|
|
118
|
-
const { formatDoctorIssuesForPrompt, formatDoctorReport } = await import("./doctor.js");
|
|
119
|
-
const { dispatchDoctorHeal } = await import("./commands-handlers.js");
|
|
120
|
-
const actionable = report.issues.filter(i => i.severity === "error");
|
|
121
|
-
const reportText = formatDoctorReport(report, { scope: doctorScope, includeWarnings: true });
|
|
122
|
-
const structuredIssues = formatDoctorIssuesForPrompt(actionable);
|
|
123
|
-
dispatchDoctorHeal(pi, doctorScope, reportText, structuredIssues);
|
|
124
|
-
}
|
|
125
|
-
catch {
|
|
126
|
-
// Non-fatal
|
|
127
|
-
}
|
|
128
|
-
}
|
|
129
|
-
}
|
|
109
|
+
const { runGitHubSync } = await import("../github-sync/sync.js");
|
|
110
|
+
await runGitHubSync(s.basePath, s.currentUnit.type, s.currentUnit.id);
|
|
130
111
|
}
|
|
131
|
-
catch {
|
|
132
|
-
|
|
112
|
+
catch (e) {
|
|
113
|
+
debugLog("postUnit", { phase: "github-sync", error: String(e) });
|
|
133
114
|
}
|
|
134
|
-
//
|
|
135
|
-
|
|
136
|
-
if (now - s.lastStateRebuildAt >= STATE_REBUILD_MIN_INTERVAL_MS) {
|
|
115
|
+
// Doctor: fix mechanical bookkeeping (skipped for lightweight sidecars)
|
|
116
|
+
if (!opts?.skipDoctor)
|
|
137
117
|
try {
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
118
|
+
const scopeParts = s.currentUnit.id.split("/").slice(0, 2);
|
|
119
|
+
const doctorScope = scopeParts.join("/");
|
|
120
|
+
const sliceTerminalUnits = new Set(["complete-slice", "run-uat"]);
|
|
121
|
+
const effectiveFixLevel = sliceTerminalUnits.has(s.currentUnit.type) ? "all" : "task";
|
|
122
|
+
const report = await runGSDDoctor(s.basePath, { fix: true, scope: doctorScope, fixLevel: effectiveFixLevel });
|
|
123
|
+
if (report.fixesApplied.length > 0) {
|
|
124
|
+
ctx.ui.notify(`Post-hook: applied ${report.fixesApplied.length} fix(es).`, "info");
|
|
125
|
+
}
|
|
126
|
+
// Proactive health tracking
|
|
127
|
+
const summary = summarizeDoctorIssues(report.issues);
|
|
128
|
+
recordHealthSnapshot(summary.errors, summary.warnings, report.fixesApplied.length);
|
|
129
|
+
// Check if we should escalate to LLM-assisted heal
|
|
130
|
+
if (summary.errors > 0) {
|
|
131
|
+
const unresolvedErrors = report.issues
|
|
132
|
+
.filter(i => i.severity === "error" && !i.fixable)
|
|
133
|
+
.map(i => ({ code: i.code, message: i.message, unitId: i.unitId }));
|
|
134
|
+
const escalation = checkHealEscalation(summary.errors, unresolvedErrors);
|
|
135
|
+
if (escalation.shouldEscalate) {
|
|
136
|
+
ctx.ui.notify(`Doctor heal escalation: ${escalation.reason}. Dispatching LLM-assisted heal.`, "warning");
|
|
137
|
+
try {
|
|
138
|
+
const { formatDoctorIssuesForPrompt, formatDoctorReport } = await import("./doctor.js");
|
|
139
|
+
const { dispatchDoctorHeal } = await import("./commands-handlers.js");
|
|
140
|
+
const actionable = report.issues.filter(i => i.severity === "error");
|
|
141
|
+
const reportText = formatDoctorReport(report, { scope: doctorScope, includeWarnings: true });
|
|
142
|
+
const structuredIssues = formatDoctorIssuesForPrompt(actionable);
|
|
143
|
+
dispatchDoctorHeal(pi, doctorScope, reportText, structuredIssues);
|
|
144
|
+
}
|
|
145
|
+
catch (e) {
|
|
146
|
+
debugLog("postUnit", { phase: "doctor-heal-dispatch", error: String(e) });
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
}
|
|
141
150
|
}
|
|
142
|
-
catch {
|
|
143
|
-
|
|
151
|
+
catch (e) {
|
|
152
|
+
debugLog("postUnit", { phase: "doctor", error: String(e) });
|
|
153
|
+
}
|
|
154
|
+
// Throttled STATE.md rebuild (skipped for lightweight sidecars)
|
|
155
|
+
if (!opts?.skipStateRebuild) {
|
|
156
|
+
const now = Date.now();
|
|
157
|
+
if (now - s.lastStateRebuildAt >= STATE_REBUILD_MIN_INTERVAL_MS) {
|
|
158
|
+
try {
|
|
159
|
+
await rebuildState(s.basePath);
|
|
160
|
+
s.lastStateRebuildAt = now;
|
|
161
|
+
autoCommitCurrentBranch(s.basePath, "state-rebuild", s.currentUnit.id);
|
|
162
|
+
}
|
|
163
|
+
catch (e) {
|
|
164
|
+
debugLog("postUnit", { phase: "state-rebuild", error: String(e) });
|
|
165
|
+
}
|
|
144
166
|
}
|
|
145
167
|
}
|
|
146
168
|
// Prune dead bg-shell processes
|
|
@@ -148,27 +170,27 @@ export async function postUnitPreVerification(pctx) {
|
|
|
148
170
|
const { pruneDeadProcesses } = await import("../bg-shell/process-manager.js");
|
|
149
171
|
pruneDeadProcesses();
|
|
150
172
|
}
|
|
151
|
-
catch {
|
|
152
|
-
|
|
173
|
+
catch (e) {
|
|
174
|
+
debugLog("postUnit", { phase: "prune-bg-shell", error: String(e) });
|
|
153
175
|
}
|
|
154
|
-
// Sync worktree state back to project root
|
|
155
|
-
if (s.originalBasePath && s.originalBasePath !== s.basePath) {
|
|
176
|
+
// Sync worktree state back to project root (skipped for lightweight sidecars)
|
|
177
|
+
if (!opts?.skipWorktreeSync && s.originalBasePath && s.originalBasePath !== s.basePath) {
|
|
156
178
|
try {
|
|
157
179
|
syncStateToProjectRoot(s.basePath, s.originalBasePath, s.currentMilestoneId);
|
|
158
180
|
}
|
|
159
|
-
catch {
|
|
160
|
-
|
|
181
|
+
catch (e) {
|
|
182
|
+
debugLog("postUnit", { phase: "worktree-sync", error: String(e) });
|
|
161
183
|
}
|
|
162
184
|
}
|
|
163
185
|
// Rewrite-docs completion
|
|
164
186
|
if (s.currentUnit.type === "rewrite-docs") {
|
|
165
187
|
try {
|
|
166
188
|
await resolveAllOverrides(s.basePath);
|
|
167
|
-
|
|
189
|
+
s.rewriteAttemptCount = 0;
|
|
168
190
|
ctx.ui.notify("Override(s) resolved — rewrite-docs completed.", "info");
|
|
169
191
|
}
|
|
170
|
-
catch {
|
|
171
|
-
|
|
192
|
+
catch (e) {
|
|
193
|
+
debugLog("postUnit", { phase: "rewrite-docs-resolve", error: String(e) });
|
|
172
194
|
}
|
|
173
195
|
}
|
|
174
196
|
// Reactive state cleanup on slice completion
|
|
@@ -181,8 +203,8 @@ export async function postUnitPreVerification(pctx) {
|
|
|
181
203
|
clearReactiveState(s.basePath, mid, sid);
|
|
182
204
|
}
|
|
183
205
|
}
|
|
184
|
-
catch {
|
|
185
|
-
|
|
206
|
+
catch (e) {
|
|
207
|
+
debugLog("postUnit", { phase: "reactive-state-cleanup", error: String(e) });
|
|
186
208
|
}
|
|
187
209
|
}
|
|
188
210
|
// Post-triage: execute actionable resolutions
|
|
@@ -224,8 +246,8 @@ export async function postUnitPreVerification(pctx) {
|
|
|
224
246
|
invalidateAllCaches();
|
|
225
247
|
}
|
|
226
248
|
}
|
|
227
|
-
catch {
|
|
228
|
-
|
|
249
|
+
catch (e) {
|
|
250
|
+
debugLog("postUnit", { phase: "artifact-verify", error: String(e) });
|
|
229
251
|
}
|
|
230
252
|
}
|
|
231
253
|
else {
|
|
@@ -238,8 +260,8 @@ export async function postUnitPreVerification(pctx) {
|
|
|
238
260
|
});
|
|
239
261
|
clearUnitRuntimeRecord(s.basePath, s.currentUnit.type, s.currentUnit.id);
|
|
240
262
|
}
|
|
241
|
-
catch {
|
|
242
|
-
|
|
263
|
+
catch (e) {
|
|
264
|
+
debugLog("postUnit", { phase: "hook-finalize", error: String(e) });
|
|
243
265
|
}
|
|
244
266
|
}
|
|
245
267
|
}
|
|
@@ -352,8 +374,8 @@ export async function postUnitPostVerification(pctx) {
|
|
|
352
374
|
}
|
|
353
375
|
}
|
|
354
376
|
}
|
|
355
|
-
catch {
|
|
356
|
-
|
|
377
|
+
catch (e) {
|
|
378
|
+
debugLog("postUnit", { phase: "triage-check", error: String(e) });
|
|
357
379
|
}
|
|
358
380
|
}
|
|
359
381
|
// ── Quick-task dispatch ──
|
|
@@ -387,8 +409,8 @@ export async function postUnitPostVerification(pctx) {
|
|
|
387
409
|
ctx.ui.notify(`Executing quick-task: ${capture.id} — "${capture.text}"`, "info");
|
|
388
410
|
return "continue";
|
|
389
411
|
}
|
|
390
|
-
catch {
|
|
391
|
-
|
|
412
|
+
catch (e) {
|
|
413
|
+
debugLog("postUnit", { phase: "quick-task-dispatch", error: String(e) });
|
|
392
414
|
}
|
|
393
415
|
}
|
|
394
416
|
// Step mode → show wizard instead of dispatch
|
|
@@ -12,10 +12,7 @@ import { resolveSkillDiscoveryMode, resolveInlineLevel, loadEffectiveGSDPreferen
|
|
|
12
12
|
import { join } from "node:path";
|
|
13
13
|
import { existsSync } from "node:fs";
|
|
14
14
|
import { computeBudgets, resolveExecutorContextWindow } from "./context-budget.js";
|
|
15
|
-
import { compressToTarget } from "./prompt-compressor.js";
|
|
16
|
-
import { distillSummaries } from "./summary-distiller.js";
|
|
17
15
|
import { formatDecisionsCompact, formatRequirementsCompact } from "./structured-data-formatter.js";
|
|
18
|
-
import { chunkByRelevance, formatChunks } from "./semantic-chunker.js";
|
|
19
16
|
// ─── Executor Constraints ─────────────────────────────────────────────────────
|
|
20
17
|
/**
|
|
21
18
|
* Format executor context constraints for injection into the plan-slice prompt.
|
|
@@ -126,14 +123,10 @@ export async function inlineFileSmart(absPath, relPath, label, query, threshold
|
|
|
126
123
|
if (content.length <= threshold || !query) {
|
|
127
124
|
return `### ${label}\nSource: \`${relPath}\`\n\n${content.trim()}`;
|
|
128
125
|
}
|
|
129
|
-
//
|
|
130
|
-
const
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
return `### ${label}\nSource: \`${relPath}\`\n\n${content.trim()}`;
|
|
134
|
-
}
|
|
135
|
-
const formatted = formatChunks(result, relPath);
|
|
136
|
-
return `### ${label} (${result.omittedChunks} sections omitted for relevance)\nSource: \`${relPath}\`\n\n${formatted}`;
|
|
126
|
+
// For large files, truncate at section boundary
|
|
127
|
+
const { truncateAtSectionBoundary } = await import("./context-budget.js");
|
|
128
|
+
const truncated = truncateAtSectionBoundary(content, threshold).content;
|
|
129
|
+
return `### ${label}\nSource: \`${relPath}\`\n\n${truncated}`;
|
|
137
130
|
}
|
|
138
131
|
/**
|
|
139
132
|
* Load and inline dependency slice summaries (full content, not just paths).
|
|
@@ -165,20 +158,6 @@ export async function inlineDependencySummaries(mid, sid, base, budgetChars) {
|
|
|
165
158
|
}
|
|
166
159
|
const result = sections.join("\n\n");
|
|
167
160
|
if (budgetChars !== undefined && result.length > budgetChars) {
|
|
168
|
-
// For 3+ summaries, try distillation first (preserves more information)
|
|
169
|
-
if (sections.length >= 3) {
|
|
170
|
-
const rawSummaries = sections.map(s => {
|
|
171
|
-
// Extract content after the header line
|
|
172
|
-
const lines = s.split("\n");
|
|
173
|
-
const contentStart = lines.findIndex(l => l.startsWith("Source:"));
|
|
174
|
-
return contentStart >= 0 ? lines.slice(contentStart + 1).join("\n").trim() : s;
|
|
175
|
-
});
|
|
176
|
-
const distilled = distillSummaries(rawSummaries, budgetChars);
|
|
177
|
-
if (distilled.content.length <= budgetChars) {
|
|
178
|
-
return distilled.content;
|
|
179
|
-
}
|
|
180
|
-
}
|
|
181
|
-
// Fall back to section-boundary truncation
|
|
182
161
|
const { truncateAtSectionBoundary } = await import("./context-budget.js");
|
|
183
162
|
return truncateAtSectionBoundary(result, budgetChars).content;
|
|
184
163
|
}
|
|
@@ -777,15 +756,12 @@ export async function buildExecuteTaskPrompt(mid, sid, sTitle, tid, tTitle, base
|
|
|
777
756
|
const contextWindow = resolveExecutorContextWindow(undefined, prefs?.preferences);
|
|
778
757
|
const budgets = computeBudgets(contextWindow);
|
|
779
758
|
const verificationBudget = `~${Math.round(budgets.verificationBudgetChars / 1000)}K chars`;
|
|
780
|
-
//
|
|
781
|
-
// Only compress when compression_strategy is "compress" (budget/balanced profiles).
|
|
759
|
+
// Truncate carry-forward section when it exceeds 40% of inline context budget.
|
|
782
760
|
const carryForwardBudget = Math.floor(budgets.inlineContextBudgetChars * 0.4);
|
|
783
761
|
let finalCarryForward = carryForwardSection;
|
|
784
762
|
if (carryForwardSection.length > carryForwardBudget) {
|
|
785
|
-
const {
|
|
786
|
-
|
|
787
|
-
finalCarryForward = compressToTarget(carryForwardSection, carryForwardBudget).content;
|
|
788
|
-
}
|
|
763
|
+
const { truncateAtSectionBoundary } = await import("./context-budget.js");
|
|
764
|
+
finalCarryForward = truncateAtSectionBoundary(carryForwardSection, carryForwardBudget).content;
|
|
789
765
|
}
|
|
790
766
|
return loadPrompt("execute-task", {
|
|
791
767
|
overridesSection,
|
|
@@ -11,7 +11,7 @@
|
|
|
11
11
|
import { deriveState } from "./state.js";
|
|
12
12
|
import { loadFile, getManifestStatus } from "./files.js";
|
|
13
13
|
import { loadEffectiveGSDPreferences, resolveSkillDiscoveryMode, getIsolationMode, } from "./preferences.js";
|
|
14
|
-
import { ensureGsdSymlink } from "./repo-identity.js";
|
|
14
|
+
import { ensureGsdSymlink, validateProjectId } from "./repo-identity.js";
|
|
15
15
|
import { migrateToExternalState, recoverFailedMigration } from "./migrate-external.js";
|
|
16
16
|
import { collectSecretsFromManifest } from "../get-secrets-from-user.js";
|
|
17
17
|
import { gsdRoot, resolveMilestoneFile } from "./paths.js";
|
|
@@ -63,6 +63,12 @@ export async function bootstrapAutoSession(s, ctx, pi, base, verboseMode, reques
|
|
|
63
63
|
return false;
|
|
64
64
|
}
|
|
65
65
|
try {
|
|
66
|
+
// Validate GSD_PROJECT_ID early so the user gets immediate feedback
|
|
67
|
+
const customProjectId = process.env.GSD_PROJECT_ID;
|
|
68
|
+
if (customProjectId && !validateProjectId(customProjectId)) {
|
|
69
|
+
ctx.ui.notify(`GSD_PROJECT_ID must contain only alphanumeric characters, hyphens, and underscores. Got: "${customProjectId}"`, "error");
|
|
70
|
+
return releaseLockAndReturn();
|
|
71
|
+
}
|
|
66
72
|
// Ensure git repo exists
|
|
67
73
|
if (!nativeIsRepo(base)) {
|
|
68
74
|
const mainBranch = loadEffectiveGSDPreferences()?.preferences?.git?.main_branch || "main";
|
|
@@ -303,11 +309,16 @@ export async function bootstrapAutoSession(s, ctx, pi, base, verboseMode, reques
|
|
|
303
309
|
// ── Auto-worktree setup ──
|
|
304
310
|
s.originalBasePath = base;
|
|
305
311
|
const isUnderGsdWorktrees = (p) => {
|
|
312
|
+
// Direct layout: /.gsd/worktrees/
|
|
306
313
|
const marker = `${pathSep}.gsd${pathSep}worktrees${pathSep}`;
|
|
307
314
|
if (p.includes(marker))
|
|
308
315
|
return true;
|
|
309
316
|
const worktreesSuffix = `${pathSep}.gsd${pathSep}worktrees`;
|
|
310
|
-
|
|
317
|
+
if (p.endsWith(worktreesSuffix))
|
|
318
|
+
return true;
|
|
319
|
+
// Symlink-resolved layout: /.gsd/projects/<hash>/worktrees/
|
|
320
|
+
const symlinkRe = new RegExp(`\\${pathSep}\\.gsd\\${pathSep}projects\\${pathSep}[a-f0-9]+\\${pathSep}worktrees(?:\\${pathSep}|$)`);
|
|
321
|
+
return symlinkRe.test(p);
|
|
311
322
|
};
|
|
312
323
|
if (s.currentMilestoneId &&
|
|
313
324
|
shouldUseWorktreeIsolation() &&
|
|
@@ -13,6 +13,7 @@ import { existsSync, readFileSync, unlinkSync, readdirSync, } from "node:fs";
|
|
|
13
13
|
import { join, sep as pathSep } from "node:path";
|
|
14
14
|
import { homedir } from "node:os";
|
|
15
15
|
import { safeCopy, safeCopyRecursive } from "./safe-fs.js";
|
|
16
|
+
const gsdHome = process.env.GSD_HOME || join(homedir(), ".gsd");
|
|
16
17
|
// ─── Project Root → Worktree Sync ─────────────────────────────────────────
|
|
17
18
|
/**
|
|
18
19
|
* Sync milestone artifacts from project root INTO worktree before deriveState.
|
|
@@ -75,7 +76,7 @@ export function syncStateToProjectRoot(worktreePath, projectRoot, milestoneId) {
|
|
|
75
76
|
* doesn't falsely trigger staleness (#804).
|
|
76
77
|
*/
|
|
77
78
|
export function readResourceVersion() {
|
|
78
|
-
const agentDir = process.env.GSD_CODING_AGENT_DIR || join(
|
|
79
|
+
const agentDir = process.env.GSD_CODING_AGENT_DIR || join(gsdHome, "agent");
|
|
79
80
|
const manifestPath = join(agentDir, "managed-resources.json");
|
|
80
81
|
try {
|
|
81
82
|
const manifest = JSON.parse(readFileSync(manifestPath, "utf-8"));
|
|
@@ -115,10 +116,17 @@ export function checkResourcesStale(versionOnStart) {
|
|
|
115
116
|
* Returns the corrected base path.
|
|
116
117
|
*/
|
|
117
118
|
export function escapeStaleWorktree(base) {
|
|
118
|
-
|
|
119
|
-
const
|
|
120
|
-
|
|
121
|
-
|
|
119
|
+
// Direct layout: /.gsd/worktrees/
|
|
120
|
+
const directMarker = `${pathSep}.gsd${pathSep}worktrees${pathSep}`;
|
|
121
|
+
let idx = base.indexOf(directMarker);
|
|
122
|
+
if (idx === -1) {
|
|
123
|
+
// Symlink-resolved layout: /.gsd/projects/<hash>/worktrees/
|
|
124
|
+
const symlinkRe = new RegExp(`\\${pathSep}\\.gsd\\${pathSep}projects\\${pathSep}[a-f0-9]+\\${pathSep}worktrees\\${pathSep}`);
|
|
125
|
+
const match = base.match(symlinkRe);
|
|
126
|
+
if (!match || match.index === undefined)
|
|
127
|
+
return base;
|
|
128
|
+
idx = match.index;
|
|
129
|
+
}
|
|
122
130
|
// base is inside .gsd/worktrees/<something> — extract the project root
|
|
123
131
|
const projectRoot = base.slice(0, idx);
|
|
124
132
|
try {
|