gsd-pi 2.37.1 → 2.38.0-dev.e40f839
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/README.md +1 -1
- 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/onboarding.js +1 -0
- 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/google-search/package.json +3 -1
- package/dist/resources/extensions/gsd/auto-dispatch.js +67 -1
- package/dist/resources/extensions/gsd/auto-loop.js +7 -1
- package/dist/resources/extensions/gsd/auto-post-unit.js +14 -0
- package/dist/resources/extensions/gsd/auto-prompts.js +91 -2
- package/dist/resources/extensions/gsd/auto-recovery.js +37 -1
- package/dist/resources/extensions/gsd/auto-start.js +6 -1
- package/dist/resources/extensions/gsd/auto-worktree-sync.js +13 -5
- 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.js +22 -2
- package/dist/resources/extensions/gsd/detection.js +1 -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 +35 -1
- 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 +43 -2
- package/dist/resources/extensions/gsd/forensics.js +1 -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/observability-validator.js +24 -0
- package/dist/resources/extensions/gsd/package.json +1 -1
- package/dist/resources/extensions/gsd/preferences-types.js +2 -1
- package/dist/resources/extensions/gsd/preferences-validation.js +43 -1
- package/dist/resources/extensions/gsd/preferences.js +4 -3
- package/dist/resources/extensions/gsd/prompts/plan-slice.md +2 -1
- package/dist/resources/extensions/gsd/prompts/reactive-execute.md +41 -0
- package/dist/resources/extensions/gsd/reactive-graph.js +227 -0
- package/dist/resources/extensions/gsd/repo-identity.js +2 -1
- package/dist/resources/extensions/gsd/resource-version.js +2 -1
- package/dist/resources/extensions/gsd/state.js +1 -1
- package/dist/resources/extensions/gsd/templates/task-plan.md +11 -3
- 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 +2 -1
- package/packages/pi-ai/dist/env-api-keys.js +13 -0
- package/packages/pi-ai/dist/env-api-keys.js.map +1 -1
- package/packages/pi-ai/dist/models.generated.d.ts +172 -0
- package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
- package/packages/pi-ai/dist/models.generated.js +172 -0
- package/packages/pi-ai/dist/models.generated.js.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic-shared.d.ts +64 -0
- package/packages/pi-ai/dist/providers/anthropic-shared.d.ts.map +1 -0
- package/packages/pi-ai/dist/providers/anthropic-shared.js +668 -0
- package/packages/pi-ai/dist/providers/anthropic-shared.js.map +1 -0
- package/packages/pi-ai/dist/providers/anthropic-vertex.d.ts +5 -0
- package/packages/pi-ai/dist/providers/anthropic-vertex.d.ts.map +1 -0
- package/packages/pi-ai/dist/providers/anthropic-vertex.js +85 -0
- package/packages/pi-ai/dist/providers/anthropic-vertex.js.map +1 -0
- package/packages/pi-ai/dist/providers/anthropic.d.ts +4 -30
- package/packages/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic.js +47 -764
- package/packages/pi-ai/dist/providers/anthropic.js.map +1 -1
- package/packages/pi-ai/dist/providers/register-builtins.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/register-builtins.js +6 -0
- package/packages/pi-ai/dist/providers/register-builtins.js.map +1 -1
- package/packages/pi-ai/dist/types.d.ts +2 -2
- package/packages/pi-ai/dist/types.d.ts.map +1 -1
- package/packages/pi-ai/dist/types.js.map +1 -1
- package/packages/pi-ai/package.json +1 -0
- package/packages/pi-ai/src/env-api-keys.ts +14 -0
- package/packages/pi-ai/src/models.generated.ts +172 -0
- package/packages/pi-ai/src/providers/anthropic-shared.ts +761 -0
- package/packages/pi-ai/src/providers/anthropic-vertex.ts +130 -0
- package/packages/pi-ai/src/providers/anthropic.ts +76 -868
- package/packages/pi-ai/src/providers/register-builtins.ts +7 -0
- package/packages/pi-ai/src/types.ts +2 -0
- package/packages/pi-coding-agent/dist/core/model-resolver.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-resolver.js +1 -0
- package/packages/pi-coding-agent/dist/core/model-resolver.js.map +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/model-resolver.ts +1 -0
- 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/gsd/auto-dispatch.ts +93 -0
- package/src/resources/extensions/gsd/auto-loop.ts +13 -1
- package/src/resources/extensions/gsd/auto-post-unit.ts +14 -0
- package/src/resources/extensions/gsd/auto-prompts.ts +125 -3
- package/src/resources/extensions/gsd/auto-recovery.ts +42 -0
- package/src/resources/extensions/gsd/auto-start.ts +7 -1
- package/src/resources/extensions/gsd/auto-worktree-sync.ts +15 -4
- 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.ts +24 -2
- package/src/resources/extensions/gsd/detection.ts +2 -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 +38 -1
- 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 +47 -2
- package/src/resources/extensions/gsd/forensics.ts +1 -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/observability-validator.ts +27 -0
- package/src/resources/extensions/gsd/preferences-types.ts +5 -1
- package/src/resources/extensions/gsd/preferences-validation.ts +42 -1
- package/src/resources/extensions/gsd/preferences.ts +5 -3
- package/src/resources/extensions/gsd/prompts/plan-slice.md +2 -1
- package/src/resources/extensions/gsd/prompts/reactive-execute.md +41 -0
- package/src/resources/extensions/gsd/reactive-graph.ts +289 -0
- package/src/resources/extensions/gsd/repo-identity.ts +3 -1
- package/src/resources/extensions/gsd/resource-version.ts +3 -1
- package/src/resources/extensions/gsd/state.ts +1 -1
- package/src/resources/extensions/gsd/templates/task-plan.md +11 -3
- 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 +108 -3
- package/src/resources/extensions/gsd/tests/plan-quality-validator.test.ts +111 -0
- package/src/resources/extensions/gsd/tests/reactive-executor.test.ts +511 -0
- package/src/resources/extensions/gsd/tests/reactive-graph.test.ts +299 -0
- package/src/resources/extensions/gsd/tests/worktree.test.ts +47 -0
- package/src/resources/extensions/gsd/types.ts +43 -0
- 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
|
@@ -0,0 +1,289 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Reactive Task Graph — derives dependency edges from task plan IO signatures.
|
|
3
|
+
*
|
|
4
|
+
* Pure functions that build a DAG from task IO intersections and resolve
|
|
5
|
+
* which tasks are currently ready for parallel dispatch. Used by the
|
|
6
|
+
* reactive-execute dispatch path (ADR-004).
|
|
7
|
+
*
|
|
8
|
+
* Graph derivation and resolution functions are pure (no filesystem access).
|
|
9
|
+
* The `loadSliceTaskIO` loader at the bottom is the only async/IO function.
|
|
10
|
+
*/
|
|
11
|
+
|
|
12
|
+
import type { TaskIO, DerivedTaskNode, ReactiveExecutionState } from "./types.js";
|
|
13
|
+
import { loadFile, parsePlan, parseTaskPlanIO } from "./files.js";
|
|
14
|
+
import { resolveTasksDir, resolveTaskFiles } from "./paths.js";
|
|
15
|
+
import { join } from "node:path";
|
|
16
|
+
import { loadJsonFileOrNull, saveJsonFile } from "./json-persistence.js";
|
|
17
|
+
import { existsSync, unlinkSync } from "node:fs";
|
|
18
|
+
|
|
19
|
+
// ─── Graph Construction ───────────────────────────────────────────────────
|
|
20
|
+
|
|
21
|
+
/**
|
|
22
|
+
* Build a dependency graph from task IO signatures.
|
|
23
|
+
*
|
|
24
|
+
* A task T_b depends on T_a when any of T_b's inputFiles appear in T_a's
|
|
25
|
+
* outputFiles. Self-references are excluded.
|
|
26
|
+
*
|
|
27
|
+
* Tasks are returned in the same order as the input array.
|
|
28
|
+
*/
|
|
29
|
+
export function deriveTaskGraph(tasks: TaskIO[]): DerivedTaskNode[] {
|
|
30
|
+
// Build output → producer lookup
|
|
31
|
+
const outputToProducer = new Map<string, string[]>();
|
|
32
|
+
for (const task of tasks) {
|
|
33
|
+
for (const outFile of task.outputFiles) {
|
|
34
|
+
const existing = outputToProducer.get(outFile);
|
|
35
|
+
if (existing) {
|
|
36
|
+
existing.push(task.id);
|
|
37
|
+
} else {
|
|
38
|
+
outputToProducer.set(outFile, [task.id]);
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
return tasks.map((task) => {
|
|
44
|
+
const deps = new Set<string>();
|
|
45
|
+
for (const inFile of task.inputFiles) {
|
|
46
|
+
const producers = outputToProducer.get(inFile);
|
|
47
|
+
if (producers) {
|
|
48
|
+
for (const pid of producers) {
|
|
49
|
+
if (pid !== task.id) deps.add(pid);
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
return {
|
|
54
|
+
...task,
|
|
55
|
+
dependsOn: [...deps].sort(),
|
|
56
|
+
};
|
|
57
|
+
});
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
// ─── Ready Set Resolution ─────────────────────────────────────────────────
|
|
61
|
+
|
|
62
|
+
/**
|
|
63
|
+
* Return task IDs whose dependencies are all in `completed`.
|
|
64
|
+
* Excludes tasks that are already done or in-flight.
|
|
65
|
+
*/
|
|
66
|
+
export function getReadyTasks(
|
|
67
|
+
graph: DerivedTaskNode[],
|
|
68
|
+
completed: Set<string>,
|
|
69
|
+
inFlight: Set<string>,
|
|
70
|
+
): string[] {
|
|
71
|
+
return graph
|
|
72
|
+
.filter((node) => {
|
|
73
|
+
if (node.done || completed.has(node.id) || inFlight.has(node.id)) return false;
|
|
74
|
+
return node.dependsOn.every((dep) => completed.has(dep));
|
|
75
|
+
})
|
|
76
|
+
.map((node) => node.id);
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
// ─── Conflict-Free Subset Selection ──────────────────────────────────────
|
|
80
|
+
|
|
81
|
+
/**
|
|
82
|
+
* Greedy selection of non-conflicting tasks up to `maxParallel`.
|
|
83
|
+
*
|
|
84
|
+
* Two tasks conflict if they share any outputFile. We also exclude tasks
|
|
85
|
+
* whose outputs overlap with `inFlightOutputs` (files being written by
|
|
86
|
+
* tasks currently in progress).
|
|
87
|
+
*/
|
|
88
|
+
export function chooseNonConflictingSubset(
|
|
89
|
+
readyIds: string[],
|
|
90
|
+
graph: DerivedTaskNode[],
|
|
91
|
+
maxParallel: number,
|
|
92
|
+
inFlightOutputs: Set<string>,
|
|
93
|
+
): string[] {
|
|
94
|
+
const nodeMap = new Map(graph.map((n) => [n.id, n]));
|
|
95
|
+
const claimed = new Set(inFlightOutputs);
|
|
96
|
+
const selected: string[] = [];
|
|
97
|
+
|
|
98
|
+
for (const id of readyIds) {
|
|
99
|
+
if (selected.length >= maxParallel) break;
|
|
100
|
+
const node = nodeMap.get(id);
|
|
101
|
+
if (!node) continue;
|
|
102
|
+
|
|
103
|
+
// Check for output overlap with already-selected or in-flight
|
|
104
|
+
const conflicts = node.outputFiles.some((f) => claimed.has(f));
|
|
105
|
+
if (conflicts) continue;
|
|
106
|
+
|
|
107
|
+
// Claim this task's outputs
|
|
108
|
+
for (const f of node.outputFiles) claimed.add(f);
|
|
109
|
+
selected.push(id);
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
return selected;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
// ─── Graph Quality Checks ─────────────────────────────────────────────────
|
|
116
|
+
|
|
117
|
+
/**
|
|
118
|
+
* Returns true if any incomplete task has 0 inputFiles AND 0 outputFiles.
|
|
119
|
+
*
|
|
120
|
+
* An ambiguous graph means IO annotations are too sparse to derive reliable
|
|
121
|
+
* edges — the dispatcher should fall back to sequential execution.
|
|
122
|
+
*/
|
|
123
|
+
export function isGraphAmbiguous(graph: DerivedTaskNode[]): boolean {
|
|
124
|
+
return graph.some(
|
|
125
|
+
(node) =>
|
|
126
|
+
!node.done &&
|
|
127
|
+
node.inputFiles.length === 0 &&
|
|
128
|
+
node.outputFiles.length === 0,
|
|
129
|
+
);
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
/**
|
|
133
|
+
* Detect deadlock: no tasks are ready and none are in-flight, yet incomplete
|
|
134
|
+
* tasks remain. This indicates a circular dependency or impossible state.
|
|
135
|
+
*/
|
|
136
|
+
export function detectDeadlock(
|
|
137
|
+
graph: DerivedTaskNode[],
|
|
138
|
+
completed: Set<string>,
|
|
139
|
+
inFlight: Set<string>,
|
|
140
|
+
): boolean {
|
|
141
|
+
const incomplete = graph.filter(
|
|
142
|
+
(n) => !n.done && !completed.has(n.id) && !inFlight.has(n.id),
|
|
143
|
+
);
|
|
144
|
+
if (incomplete.length === 0) return false; // all done
|
|
145
|
+
if (inFlight.size > 0) return false; // something is running, wait for it
|
|
146
|
+
|
|
147
|
+
// Nothing in flight, but incomplete tasks remain — check if any are ready
|
|
148
|
+
const ready = getReadyTasks(graph, completed, inFlight);
|
|
149
|
+
return ready.length === 0;
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
// ─── Graph Metrics ────────────────────────────────────────────────────────
|
|
153
|
+
|
|
154
|
+
/** Compute summary metrics for logging. */
|
|
155
|
+
export function graphMetrics(graph: DerivedTaskNode[]): {
|
|
156
|
+
taskCount: number;
|
|
157
|
+
edgeCount: number;
|
|
158
|
+
readySetSize: number;
|
|
159
|
+
ambiguous: boolean;
|
|
160
|
+
} {
|
|
161
|
+
const completed = new Set(graph.filter((n) => n.done).map((n) => n.id));
|
|
162
|
+
const ready = getReadyTasks(graph, completed, new Set());
|
|
163
|
+
const edgeCount = graph.reduce((sum, n) => sum + n.dependsOn.length, 0);
|
|
164
|
+
|
|
165
|
+
return {
|
|
166
|
+
taskCount: graph.length,
|
|
167
|
+
edgeCount,
|
|
168
|
+
readySetSize: ready.length,
|
|
169
|
+
ambiguous: isGraphAmbiguous(graph),
|
|
170
|
+
};
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
// ─── IO Loader (async, filesystem) ────────────────────────────────────────
|
|
174
|
+
|
|
175
|
+
/**
|
|
176
|
+
* Load TaskIO for all tasks in a slice by reading the slice plan (for done
|
|
177
|
+
* status and task IDs) and individual task plan files (for IO sections).
|
|
178
|
+
*
|
|
179
|
+
* Returns [] when the slice plan or tasks directory doesn't exist.
|
|
180
|
+
*/
|
|
181
|
+
export async function loadSliceTaskIO(
|
|
182
|
+
basePath: string,
|
|
183
|
+
mid: string,
|
|
184
|
+
sid: string,
|
|
185
|
+
): Promise<TaskIO[]> {
|
|
186
|
+
const { resolveSliceFile } = await import("./paths.js");
|
|
187
|
+
const slicePlanPath = resolveSliceFile(basePath, mid, sid, "PLAN");
|
|
188
|
+
const planContent = slicePlanPath ? await loadFile(slicePlanPath) : null;
|
|
189
|
+
if (!planContent) return [];
|
|
190
|
+
|
|
191
|
+
const plan = parsePlan(planContent);
|
|
192
|
+
const tDir = resolveTasksDir(basePath, mid, sid);
|
|
193
|
+
if (!tDir) return [];
|
|
194
|
+
|
|
195
|
+
const results: TaskIO[] = [];
|
|
196
|
+
|
|
197
|
+
for (const taskEntry of plan.tasks) {
|
|
198
|
+
const planFiles = resolveTaskFiles(tDir, "PLAN");
|
|
199
|
+
const taskFileName = planFiles.find((f) =>
|
|
200
|
+
f.toUpperCase().startsWith(taskEntry.id.toUpperCase() + "-"),
|
|
201
|
+
);
|
|
202
|
+
if (!taskFileName) {
|
|
203
|
+
// Task plan file missing — include with empty IO (will trigger ambiguous)
|
|
204
|
+
results.push({
|
|
205
|
+
id: taskEntry.id,
|
|
206
|
+
title: taskEntry.title,
|
|
207
|
+
inputFiles: [],
|
|
208
|
+
outputFiles: [],
|
|
209
|
+
done: taskEntry.done,
|
|
210
|
+
});
|
|
211
|
+
continue;
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
const taskContent = await loadFile(join(tDir, taskFileName));
|
|
215
|
+
if (!taskContent) {
|
|
216
|
+
results.push({
|
|
217
|
+
id: taskEntry.id,
|
|
218
|
+
title: taskEntry.title,
|
|
219
|
+
inputFiles: [],
|
|
220
|
+
outputFiles: [],
|
|
221
|
+
done: taskEntry.done,
|
|
222
|
+
});
|
|
223
|
+
continue;
|
|
224
|
+
}
|
|
225
|
+
|
|
226
|
+
const io = parseTaskPlanIO(taskContent);
|
|
227
|
+
results.push({
|
|
228
|
+
id: taskEntry.id,
|
|
229
|
+
title: taskEntry.title,
|
|
230
|
+
inputFiles: io.inputFiles,
|
|
231
|
+
outputFiles: io.outputFiles,
|
|
232
|
+
done: taskEntry.done,
|
|
233
|
+
});
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
return results;
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
// ─── State Persistence ────────────────────────────────────────────────────
|
|
240
|
+
|
|
241
|
+
function reactiveStatePath(basePath: string, mid: string, sid: string): string {
|
|
242
|
+
return join(basePath, ".gsd", "runtime", `${mid}-${sid}-reactive.json`);
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
function isReactiveState(data: unknown): data is ReactiveExecutionState {
|
|
246
|
+
if (!data || typeof data !== "object") return false;
|
|
247
|
+
const d = data as Record<string, unknown>;
|
|
248
|
+
return typeof d.sliceId === "string" && Array.isArray(d.completed) && Array.isArray(d.dispatched);
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
/**
|
|
252
|
+
* Load persisted reactive execution state for a slice.
|
|
253
|
+
* Returns null when no state file exists or the file is invalid.
|
|
254
|
+
*/
|
|
255
|
+
export function loadReactiveState(
|
|
256
|
+
basePath: string,
|
|
257
|
+
mid: string,
|
|
258
|
+
sid: string,
|
|
259
|
+
): ReactiveExecutionState | null {
|
|
260
|
+
return loadJsonFileOrNull(reactiveStatePath(basePath, mid, sid), isReactiveState);
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
/**
|
|
264
|
+
* Save reactive execution state to disk.
|
|
265
|
+
*/
|
|
266
|
+
export function saveReactiveState(
|
|
267
|
+
basePath: string,
|
|
268
|
+
mid: string,
|
|
269
|
+
sid: string,
|
|
270
|
+
state: ReactiveExecutionState,
|
|
271
|
+
): void {
|
|
272
|
+
saveJsonFile(reactiveStatePath(basePath, mid, sid), state);
|
|
273
|
+
}
|
|
274
|
+
|
|
275
|
+
/**
|
|
276
|
+
* Remove the reactive state file when a slice completes.
|
|
277
|
+
*/
|
|
278
|
+
export function clearReactiveState(
|
|
279
|
+
basePath: string,
|
|
280
|
+
mid: string,
|
|
281
|
+
sid: string,
|
|
282
|
+
): void {
|
|
283
|
+
const path = reactiveStatePath(basePath, mid, sid);
|
|
284
|
+
try {
|
|
285
|
+
if (existsSync(path)) unlinkSync(path);
|
|
286
|
+
} catch {
|
|
287
|
+
// Non-fatal
|
|
288
|
+
}
|
|
289
|
+
}
|
|
@@ -12,6 +12,8 @@ import { existsSync, lstatSync, mkdirSync, readFileSync, realpathSync, rmSync, s
|
|
|
12
12
|
import { homedir } from "node:os";
|
|
13
13
|
import { join, resolve } from "node:path";
|
|
14
14
|
|
|
15
|
+
const gsdHome = process.env.GSD_HOME || join(homedir(), ".gsd");
|
|
16
|
+
|
|
15
17
|
// ─── Repo Identity ──────────────────────────────────────────────────────────
|
|
16
18
|
|
|
17
19
|
/**
|
|
@@ -113,7 +115,7 @@ export function repoIdentity(basePath: string): string {
|
|
|
113
115
|
* otherwise `~/.gsd/projects/<hash>`.
|
|
114
116
|
*/
|
|
115
117
|
export function externalGsdRoot(basePath: string): string {
|
|
116
|
-
const base = process.env.GSD_STATE_DIR ||
|
|
118
|
+
const base = process.env.GSD_STATE_DIR || gsdHome;
|
|
117
119
|
return join(base, "projects", repoIdentity(basePath));
|
|
118
120
|
}
|
|
119
121
|
|
|
@@ -11,6 +11,8 @@ import { join } from "node:path";
|
|
|
11
11
|
import { homedir } from "node:os";
|
|
12
12
|
import { resolveProjectRoot } from "./worktree.js";
|
|
13
13
|
|
|
14
|
+
const gsdHome = process.env.GSD_HOME || join(homedir(), ".gsd");
|
|
15
|
+
|
|
14
16
|
// ─── Resource Staleness ───────────────────────────────────────────────────
|
|
15
17
|
|
|
16
18
|
/**
|
|
@@ -23,7 +25,7 @@ function isManifestWithVersion(data: unknown): data is { gsdVersion: string } {
|
|
|
23
25
|
}
|
|
24
26
|
|
|
25
27
|
export function readResourceVersion(): string | null {
|
|
26
|
-
const agentDir = process.env.GSD_CODING_AGENT_DIR || join(
|
|
28
|
+
const agentDir = process.env.GSD_CODING_AGENT_DIR || join(gsdHome, "agent");
|
|
27
29
|
const manifestPath = join(agentDir, "managed-resources.json");
|
|
28
30
|
const manifest = loadJsonFileOrNull(manifestPath, isManifestWithVersion);
|
|
29
31
|
return manifest?.gsdVersion ?? null;
|
|
@@ -31,7 +31,7 @@ import {
|
|
|
31
31
|
gsdRoot,
|
|
32
32
|
} from './paths.js';
|
|
33
33
|
|
|
34
|
-
import { milestoneIdSort, findMilestoneIds } from './
|
|
34
|
+
import { milestoneIdSort, findMilestoneIds } from './milestone-ids.js';
|
|
35
35
|
import { nativeBatchParseGsdFiles, type BatchParsedFile } from './native-parser-bridge.js';
|
|
36
36
|
|
|
37
37
|
import { join, resolve } from 'path';
|
|
@@ -42,11 +42,19 @@ estimated_files: {{estimatedFiles}}
|
|
|
42
42
|
|
|
43
43
|
## Inputs
|
|
44
44
|
|
|
45
|
+
<!-- Every input MUST be a backtick-wrapped file path. These paths are machine-parsed to
|
|
46
|
+
derive task dependencies — vague descriptions without paths break dependency detection.
|
|
47
|
+
For the first task in a slice with no prior task outputs, list the existing source files
|
|
48
|
+
this task reads or modifies. -->
|
|
49
|
+
|
|
45
50
|
- `{{filePath}}` — {{whatThisTaskNeedsFromPriorWork}}
|
|
46
|
-
- {{priorTaskSummaryInsight}}
|
|
47
51
|
|
|
48
52
|
## Expected Output
|
|
49
53
|
|
|
50
|
-
<!--
|
|
54
|
+
<!-- Every output MUST be a backtick-wrapped file path — the specific files this task creates
|
|
55
|
+
or modifies. These paths are machine-parsed to derive task dependencies.
|
|
56
|
+
This task should produce a real increment toward making the slice goal/demo true. A full
|
|
57
|
+
slice plan should not be able to mark every task complete while the claimed slice behavior
|
|
58
|
+
still does not work at the stated proof level. -->
|
|
51
59
|
|
|
52
|
-
- `{{filePath}}` — {{
|
|
60
|
+
- `{{filePath}}` — {{whatThisTaskCreatesOrModifies}}
|
|
@@ -100,6 +100,99 @@ test("buildCmuxStatusLabel and progress prefer deepest active unit", () => {
|
|
|
100
100
|
assert.deepEqual(buildCmuxProgress(state), { value: 0.4, label: "2/5 tasks" });
|
|
101
101
|
});
|
|
102
102
|
|
|
103
|
+
describe("createGridLayout", () => {
|
|
104
|
+
// Create a mock CmuxClient that tracks createSplitFrom calls
|
|
105
|
+
function makeMockClient() {
|
|
106
|
+
let nextId = 1;
|
|
107
|
+
const calls: Array<{ source: string | undefined; direction: string }> = [];
|
|
108
|
+
|
|
109
|
+
const client = {
|
|
110
|
+
calls,
|
|
111
|
+
async createGridLayout(count: number) {
|
|
112
|
+
// Simulate the grid layout logic with a fake client
|
|
113
|
+
if (count <= 0) return [];
|
|
114
|
+
const surfaces: string[] = [];
|
|
115
|
+
|
|
116
|
+
const createSplitFrom = async (source: string | undefined, direction: string) => {
|
|
117
|
+
calls.push({ source, direction });
|
|
118
|
+
return `surface-${nextId++}`;
|
|
119
|
+
};
|
|
120
|
+
|
|
121
|
+
const rightCol = await createSplitFrom("gsd-surface", "right");
|
|
122
|
+
surfaces.push(rightCol);
|
|
123
|
+
if (count === 1) return surfaces;
|
|
124
|
+
|
|
125
|
+
const bottomRight = await createSplitFrom(rightCol, "down");
|
|
126
|
+
surfaces.push(bottomRight);
|
|
127
|
+
if (count === 2) return surfaces;
|
|
128
|
+
|
|
129
|
+
const bottomLeft = await createSplitFrom("gsd-surface", "down");
|
|
130
|
+
surfaces.push(bottomLeft);
|
|
131
|
+
if (count === 3) return surfaces;
|
|
132
|
+
|
|
133
|
+
let lastSurface = bottomRight;
|
|
134
|
+
for (let i = 3; i < count; i++) {
|
|
135
|
+
const next = await createSplitFrom(lastSurface, "down");
|
|
136
|
+
surfaces.push(next);
|
|
137
|
+
lastSurface = next;
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
return surfaces;
|
|
141
|
+
},
|
|
142
|
+
};
|
|
143
|
+
return client;
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
test("1 agent creates single right split", async () => {
|
|
147
|
+
const mock = makeMockClient();
|
|
148
|
+
const surfaces = await mock.createGridLayout(1);
|
|
149
|
+
assert.equal(surfaces.length, 1);
|
|
150
|
+
assert.deepEqual(mock.calls, [
|
|
151
|
+
{ source: "gsd-surface", direction: "right" },
|
|
152
|
+
]);
|
|
153
|
+
});
|
|
154
|
+
|
|
155
|
+
test("2 agents creates right column then splits it down", async () => {
|
|
156
|
+
const mock = makeMockClient();
|
|
157
|
+
const surfaces = await mock.createGridLayout(2);
|
|
158
|
+
assert.equal(surfaces.length, 2);
|
|
159
|
+
assert.deepEqual(mock.calls, [
|
|
160
|
+
{ source: "gsd-surface", direction: "right" },
|
|
161
|
+
{ source: "surface-1", direction: "down" },
|
|
162
|
+
]);
|
|
163
|
+
});
|
|
164
|
+
|
|
165
|
+
test("3 agents creates 2x2 grid (gsd + 3 agent surfaces)", async () => {
|
|
166
|
+
const mock = makeMockClient();
|
|
167
|
+
const surfaces = await mock.createGridLayout(3);
|
|
168
|
+
assert.equal(surfaces.length, 3);
|
|
169
|
+
assert.deepEqual(mock.calls, [
|
|
170
|
+
{ source: "gsd-surface", direction: "right" },
|
|
171
|
+
{ source: "surface-1", direction: "down" },
|
|
172
|
+
{ source: "gsd-surface", direction: "down" },
|
|
173
|
+
]);
|
|
174
|
+
});
|
|
175
|
+
|
|
176
|
+
test("4 agents creates 2x2 grid with extra split", async () => {
|
|
177
|
+
const mock = makeMockClient();
|
|
178
|
+
const surfaces = await mock.createGridLayout(4);
|
|
179
|
+
assert.equal(surfaces.length, 4);
|
|
180
|
+
assert.deepEqual(mock.calls, [
|
|
181
|
+
{ source: "gsd-surface", direction: "right" },
|
|
182
|
+
{ source: "surface-1", direction: "down" },
|
|
183
|
+
{ source: "gsd-surface", direction: "down" },
|
|
184
|
+
{ source: "surface-2", direction: "down" },
|
|
185
|
+
]);
|
|
186
|
+
});
|
|
187
|
+
|
|
188
|
+
test("0 agents returns empty", async () => {
|
|
189
|
+
const mock = makeMockClient();
|
|
190
|
+
const surfaces = await mock.createGridLayout(0);
|
|
191
|
+
assert.equal(surfaces.length, 0);
|
|
192
|
+
assert.equal(mock.calls.length, 0);
|
|
193
|
+
});
|
|
194
|
+
});
|
|
195
|
+
|
|
103
196
|
describe("cmux extension discovery opt-out", () => {
|
|
104
197
|
test("cmux directory has package.json with pi manifest to prevent auto-discovery as extension", () => {
|
|
105
198
|
const cmuxDir = path.resolve(
|