gsd-pi 2.69.0 → 2.70.0
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/gsd/bootstrap/system-context.js +6 -2
- package/dist/resources/extensions/gsd/commands-cmux.js +30 -1
- package/dist/resources/extensions/gsd/workflow-mcp.js +53 -6
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +12 -12
- 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 +1 -1
- 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/api/onboarding/route.js +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 +12 -12
- package/dist/web/standalone/.next/server/chunks/6897.js +3 -3
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/pages/404.html +1 -1
- package/dist/web/standalone/.next/server/pages/500.html +1 -1
- package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
- package/package.json +1 -1
- package/packages/daemon/src/orchestrator.ts +9 -84
- package/packages/mcp-server/README.md +25 -3
- package/packages/mcp-server/dist/cli.d.ts +0 -1
- package/packages/mcp-server/dist/cli.d.ts.map +1 -1
- package/packages/mcp-server/dist/cli.js +4 -2
- package/packages/mcp-server/dist/cli.js.map +1 -1
- package/packages/mcp-server/dist/server.d.ts +32 -1
- package/packages/mcp-server/dist/server.d.ts.map +1 -1
- package/packages/mcp-server/dist/server.js +118 -1
- package/packages/mcp-server/dist/server.js.map +1 -1
- package/packages/mcp-server/dist/tool-credentials.d.ts +6 -0
- package/packages/mcp-server/dist/tool-credentials.d.ts.map +1 -0
- package/packages/mcp-server/dist/tool-credentials.js +90 -0
- package/packages/mcp-server/dist/tool-credentials.js.map +1 -0
- package/packages/mcp-server/dist/workflow-tools.d.ts +1 -0
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +274 -2
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/src/cli.ts +5 -3
- package/packages/mcp-server/src/mcp-server.test.ts +85 -1
- package/packages/mcp-server/src/server.ts +188 -1
- package/packages/mcp-server/src/tool-credentials.test.ts +95 -0
- package/packages/mcp-server/src/tool-credentials.ts +97 -0
- package/packages/mcp-server/src/workflow-tools.test.ts +32 -25
- package/packages/mcp-server/src/workflow-tools.ts +365 -2
- package/packages/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic.js +1 -23
- package/packages/pi-ai/dist/providers/anthropic.js.map +1 -1
- package/packages/pi-ai/dist/utils/oauth/index.d.ts +3 -2
- package/packages/pi-ai/dist/utils/oauth/index.d.ts.map +1 -1
- package/packages/pi-ai/dist/utils/oauth/index.js +3 -5
- package/packages/pi-ai/dist/utils/oauth/index.js.map +1 -1
- package/packages/pi-ai/src/providers/anthropic.ts +1 -31
- package/packages/pi-ai/src/utils/oauth/index.ts +3 -5
- package/packages/pi-coding-agent/package.json +1 -1
- package/pkg/package.json +1 -1
- package/src/resources/extensions/gsd/bootstrap/system-context.ts +9 -5
- package/src/resources/extensions/gsd/commands-cmux.ts +32 -1
- package/src/resources/extensions/gsd/tests/cmux.test.ts +67 -1
- package/src/resources/extensions/gsd/tests/mcp-project-config.test.ts +6 -2
- package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +23 -7
- package/src/resources/extensions/gsd/workflow-mcp.ts +59 -5
- package/packages/pi-ai/dist/utils/oauth/anthropic.d.ts +0 -17
- package/packages/pi-ai/dist/utils/oauth/anthropic.d.ts.map +0 -1
- package/packages/pi-ai/dist/utils/oauth/anthropic.js +0 -106
- package/packages/pi-ai/dist/utils/oauth/anthropic.js.map +0 -1
- package/packages/pi-ai/src/utils/oauth/anthropic.ts +0 -140
- /package/dist/web/standalone/.next/static/{DrWdzskk28E5Qz-Wjw1mj → Nl6lg7zP5dNgNBV1107v1}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{DrWdzskk28E5Qz-Wjw1mj → Nl6lg7zP5dNgNBV1107v1}/_ssgManifest.js +0 -0
|
@@ -63,7 +63,7 @@ function getWriteGateModuleCandidates() {
|
|
|
63
63
|
const candidates = [];
|
|
64
64
|
const explicitModule = process.env.GSD_WORKFLOW_WRITE_GATE_MODULE?.trim();
|
|
65
65
|
if (explicitModule) {
|
|
66
|
-
if (/^[a-z]
|
|
66
|
+
if (/^[a-z]{2,}:/i.test(explicitModule) && !explicitModule.startsWith("file:")) {
|
|
67
67
|
throw new Error("GSD_WORKFLOW_WRITE_GATE_MODULE only supports file: URLs or filesystem paths.");
|
|
68
68
|
}
|
|
69
69
|
candidates.push(explicitModule.startsWith("file:") ? explicitModule : toFileUrl(explicitModule));
|
|
@@ -74,11 +74,14 @@ function getWriteGateModuleCandidates() {
|
|
|
74
74
|
function toFileUrl(modulePath) {
|
|
75
75
|
return pathToFileURL(resolve(modulePath)).href;
|
|
76
76
|
}
|
|
77
|
+
async function importLocalModule(relativePath) {
|
|
78
|
+
return import(new URL(relativePath, import.meta.url).href);
|
|
79
|
+
}
|
|
77
80
|
function getWorkflowExecutorModuleCandidates(env = process.env) {
|
|
78
81
|
const candidates = [];
|
|
79
82
|
const explicitModule = env.GSD_WORKFLOW_EXECUTORS_MODULE?.trim();
|
|
80
83
|
if (explicitModule) {
|
|
81
|
-
if (/^[a-z]
|
|
84
|
+
if (/^[a-z]{2,}:/i.test(explicitModule) && !explicitModule.startsWith("file:")) {
|
|
82
85
|
throw new Error("GSD_WORKFLOW_EXECUTORS_MODULE only supports file: URLs or filesystem paths.");
|
|
83
86
|
}
|
|
84
87
|
candidates.push(explicitModule.startsWith("file:") ? explicitModule : toFileUrl(explicitModule));
|
|
@@ -135,6 +138,37 @@ async function getWorkflowWriteGateModule() {
|
|
|
135
138
|
}
|
|
136
139
|
return workflowWriteGatePromise;
|
|
137
140
|
}
|
|
141
|
+
export const WORKFLOW_TOOL_NAMES = [
|
|
142
|
+
"gsd_decision_save",
|
|
143
|
+
"gsd_save_decision",
|
|
144
|
+
"gsd_requirement_update",
|
|
145
|
+
"gsd_update_requirement",
|
|
146
|
+
"gsd_requirement_save",
|
|
147
|
+
"gsd_save_requirement",
|
|
148
|
+
"gsd_milestone_generate_id",
|
|
149
|
+
"gsd_generate_milestone_id",
|
|
150
|
+
"gsd_plan_milestone",
|
|
151
|
+
"gsd_plan_slice",
|
|
152
|
+
"gsd_plan_task",
|
|
153
|
+
"gsd_task_plan",
|
|
154
|
+
"gsd_replan_slice",
|
|
155
|
+
"gsd_slice_replan",
|
|
156
|
+
"gsd_slice_complete",
|
|
157
|
+
"gsd_complete_slice",
|
|
158
|
+
"gsd_skip_slice",
|
|
159
|
+
"gsd_complete_milestone",
|
|
160
|
+
"gsd_milestone_complete",
|
|
161
|
+
"gsd_validate_milestone",
|
|
162
|
+
"gsd_milestone_validate",
|
|
163
|
+
"gsd_reassess_roadmap",
|
|
164
|
+
"gsd_roadmap_reassess",
|
|
165
|
+
"gsd_save_gate_result",
|
|
166
|
+
"gsd_summary_save",
|
|
167
|
+
"gsd_task_complete",
|
|
168
|
+
"gsd_complete_task",
|
|
169
|
+
"gsd_milestone_status",
|
|
170
|
+
"gsd_journal_query",
|
|
171
|
+
];
|
|
138
172
|
async function runSerializedWorkflowOperation(fn) {
|
|
139
173
|
// The shared DB adapter and workflow log base path are process-global, so
|
|
140
174
|
// workflow MCP mutations must not overlap within a single server process.
|
|
@@ -218,6 +252,15 @@ async function handleSaveGateResult(projectDir, args) {
|
|
|
218
252
|
const { projectDir: _projectDir, ...params } = args;
|
|
219
253
|
return runSerializedWorkflowOperation(() => executeSaveGateResult(params, projectDir));
|
|
220
254
|
}
|
|
255
|
+
async function ensureMilestoneDbRow(milestoneId) {
|
|
256
|
+
try {
|
|
257
|
+
const { insertMilestone } = await importLocalModule("../../../src/resources/extensions/gsd/gsd-db.js");
|
|
258
|
+
insertMilestone({ id: milestoneId, status: "queued" });
|
|
259
|
+
}
|
|
260
|
+
catch {
|
|
261
|
+
// Ignore pre-existing rows or transient DB availability issues.
|
|
262
|
+
}
|
|
263
|
+
}
|
|
221
264
|
const projectDirParam = z.string().describe("Absolute path to the project directory within the configured workflow root");
|
|
222
265
|
const planMilestoneParams = {
|
|
223
266
|
projectDir: projectDirParam,
|
|
@@ -413,6 +456,67 @@ const summarySaveParams = {
|
|
|
413
456
|
content: z.string().describe("The full markdown content of the artifact"),
|
|
414
457
|
};
|
|
415
458
|
const summarySaveSchema = z.object(summarySaveParams);
|
|
459
|
+
const decisionSaveParams = {
|
|
460
|
+
projectDir: projectDirParam,
|
|
461
|
+
scope: z.string().describe("Scope of the decision (e.g. architecture, library, observability)"),
|
|
462
|
+
decision: z.string().describe("What is being decided"),
|
|
463
|
+
choice: z.string().describe("The choice made"),
|
|
464
|
+
rationale: z.string().describe("Why this choice was made"),
|
|
465
|
+
revisable: z.string().optional().describe("Whether this can be revisited"),
|
|
466
|
+
when_context: z.string().optional().describe("When/context for the decision"),
|
|
467
|
+
made_by: z.enum(["human", "agent", "collaborative"]).optional().describe("Who made the decision"),
|
|
468
|
+
};
|
|
469
|
+
const decisionSaveSchema = z.object(decisionSaveParams);
|
|
470
|
+
const requirementUpdateParams = {
|
|
471
|
+
projectDir: projectDirParam,
|
|
472
|
+
id: z.string().describe("Requirement ID (e.g. R001)"),
|
|
473
|
+
status: z.string().optional().describe("New status"),
|
|
474
|
+
validation: z.string().optional().describe("Validation criteria or proof"),
|
|
475
|
+
notes: z.string().optional().describe("Additional notes"),
|
|
476
|
+
description: z.string().optional().describe("Updated description"),
|
|
477
|
+
primary_owner: z.string().optional().describe("Primary owning slice"),
|
|
478
|
+
supporting_slices: z.string().optional().describe("Supporting slices"),
|
|
479
|
+
};
|
|
480
|
+
const requirementUpdateSchema = z.object(requirementUpdateParams);
|
|
481
|
+
const requirementSaveParams = {
|
|
482
|
+
projectDir: projectDirParam,
|
|
483
|
+
class: z.string().describe("Requirement class"),
|
|
484
|
+
description: z.string().describe("Short description of the requirement"),
|
|
485
|
+
why: z.string().describe("Why this requirement matters"),
|
|
486
|
+
source: z.string().describe("Origin of the requirement"),
|
|
487
|
+
status: z.string().optional().describe("Requirement status"),
|
|
488
|
+
primary_owner: z.string().optional().describe("Primary owning slice"),
|
|
489
|
+
supporting_slices: z.string().optional().describe("Supporting slices"),
|
|
490
|
+
validation: z.string().optional().describe("Validation criteria"),
|
|
491
|
+
notes: z.string().optional().describe("Additional notes"),
|
|
492
|
+
};
|
|
493
|
+
const requirementSaveSchema = z.object(requirementSaveParams);
|
|
494
|
+
const milestoneGenerateIdParams = {
|
|
495
|
+
projectDir: projectDirParam,
|
|
496
|
+
};
|
|
497
|
+
const milestoneGenerateIdSchema = z.object(milestoneGenerateIdParams);
|
|
498
|
+
const planTaskParams = {
|
|
499
|
+
projectDir: projectDirParam,
|
|
500
|
+
milestoneId: z.string().describe("Milestone ID (e.g. M001)"),
|
|
501
|
+
sliceId: z.string().describe("Slice ID (e.g. S01)"),
|
|
502
|
+
taskId: z.string().describe("Task ID (e.g. T01)"),
|
|
503
|
+
title: z.string().describe("Task title"),
|
|
504
|
+
description: z.string().describe("Task description / steps block"),
|
|
505
|
+
estimate: z.string().describe("Task estimate"),
|
|
506
|
+
files: z.array(z.string()).describe("Files likely touched"),
|
|
507
|
+
verify: z.string().describe("Verification command or block"),
|
|
508
|
+
inputs: z.array(z.string()).describe("Input files or references"),
|
|
509
|
+
expectedOutput: z.array(z.string()).describe("Expected output files or artifacts"),
|
|
510
|
+
observabilityImpact: z.string().optional().describe("Task observability impact"),
|
|
511
|
+
};
|
|
512
|
+
const planTaskSchema = z.object(planTaskParams);
|
|
513
|
+
const skipSliceParams = {
|
|
514
|
+
projectDir: projectDirParam,
|
|
515
|
+
sliceId: z.string().describe("Slice ID (e.g. S02)"),
|
|
516
|
+
milestoneId: z.string().describe("Milestone ID (e.g. M003)"),
|
|
517
|
+
reason: z.string().optional().describe("Reason for skipping this slice"),
|
|
518
|
+
};
|
|
519
|
+
const skipSliceSchema = z.object(skipSliceParams);
|
|
416
520
|
const taskCompleteParams = {
|
|
417
521
|
projectDir: projectDirParam,
|
|
418
522
|
taskId: z.string().describe("Task ID (e.g. T01)"),
|
|
@@ -442,7 +546,112 @@ const milestoneStatusParams = {
|
|
|
442
546
|
milestoneId: z.string().describe("Milestone ID to query (e.g. M001)"),
|
|
443
547
|
};
|
|
444
548
|
const milestoneStatusSchema = z.object(milestoneStatusParams);
|
|
549
|
+
const journalQueryParams = {
|
|
550
|
+
projectDir: projectDirParam,
|
|
551
|
+
flowId: z.string().optional().describe("Filter by flow ID"),
|
|
552
|
+
unitId: z.string().optional().describe("Filter by unit ID"),
|
|
553
|
+
rule: z.string().optional().describe("Filter by rule name"),
|
|
554
|
+
eventType: z.string().optional().describe("Filter by event type"),
|
|
555
|
+
after: z.string().optional().describe("ISO-8601 lower bound (inclusive)"),
|
|
556
|
+
before: z.string().optional().describe("ISO-8601 upper bound (inclusive)"),
|
|
557
|
+
limit: z.number().optional().describe("Maximum entries to return"),
|
|
558
|
+
};
|
|
559
|
+
const journalQuerySchema = z.object(journalQueryParams);
|
|
445
560
|
export function registerWorkflowTools(server) {
|
|
561
|
+
server.tool("gsd_decision_save", "Record a project decision to the GSD database and regenerate DECISIONS.md.", decisionSaveParams, async (args) => {
|
|
562
|
+
const parsed = parseWorkflowArgs(decisionSaveSchema, args);
|
|
563
|
+
const { projectDir, ...params } = parsed;
|
|
564
|
+
await enforceWorkflowWriteGate("gsd_decision_save", projectDir);
|
|
565
|
+
const result = await runSerializedWorkflowOperation(async () => {
|
|
566
|
+
const { saveDecisionToDb } = await importLocalModule("../../../src/resources/extensions/gsd/db-writer.js");
|
|
567
|
+
return saveDecisionToDb(params, projectDir);
|
|
568
|
+
});
|
|
569
|
+
return { content: [{ type: "text", text: `Saved decision ${result.id}` }] };
|
|
570
|
+
});
|
|
571
|
+
server.tool("gsd_save_decision", "Alias for gsd_decision_save. Record a project decision to the GSD database and regenerate DECISIONS.md.", decisionSaveParams, async (args) => {
|
|
572
|
+
const parsed = parseWorkflowArgs(decisionSaveSchema, args);
|
|
573
|
+
const { projectDir, ...params } = parsed;
|
|
574
|
+
await enforceWorkflowWriteGate("gsd_decision_save", projectDir);
|
|
575
|
+
const result = await runSerializedWorkflowOperation(async () => {
|
|
576
|
+
const { saveDecisionToDb } = await importLocalModule("../../../src/resources/extensions/gsd/db-writer.js");
|
|
577
|
+
return saveDecisionToDb(params, projectDir);
|
|
578
|
+
});
|
|
579
|
+
return { content: [{ type: "text", text: `Saved decision ${result.id}` }] };
|
|
580
|
+
});
|
|
581
|
+
server.tool("gsd_requirement_update", "Update an existing requirement in the GSD database and regenerate REQUIREMENTS.md.", requirementUpdateParams, async (args) => {
|
|
582
|
+
const parsed = parseWorkflowArgs(requirementUpdateSchema, args);
|
|
583
|
+
const { projectDir, id, ...updates } = parsed;
|
|
584
|
+
await enforceWorkflowWriteGate("gsd_requirement_update", projectDir);
|
|
585
|
+
await runSerializedWorkflowOperation(async () => {
|
|
586
|
+
const { updateRequirementInDb } = await importLocalModule("../../../src/resources/extensions/gsd/db-writer.js");
|
|
587
|
+
return updateRequirementInDb(id, updates, projectDir);
|
|
588
|
+
});
|
|
589
|
+
return { content: [{ type: "text", text: `Updated requirement ${id}` }] };
|
|
590
|
+
});
|
|
591
|
+
server.tool("gsd_update_requirement", "Alias for gsd_requirement_update. Update an existing requirement in the GSD database and regenerate REQUIREMENTS.md.", requirementUpdateParams, async (args) => {
|
|
592
|
+
const parsed = parseWorkflowArgs(requirementUpdateSchema, args);
|
|
593
|
+
const { projectDir, id, ...updates } = parsed;
|
|
594
|
+
await enforceWorkflowWriteGate("gsd_requirement_update", projectDir);
|
|
595
|
+
await runSerializedWorkflowOperation(async () => {
|
|
596
|
+
const { updateRequirementInDb } = await importLocalModule("../../../src/resources/extensions/gsd/db-writer.js");
|
|
597
|
+
return updateRequirementInDb(id, updates, projectDir);
|
|
598
|
+
});
|
|
599
|
+
return { content: [{ type: "text", text: `Updated requirement ${id}` }] };
|
|
600
|
+
});
|
|
601
|
+
server.tool("gsd_requirement_save", "Record a new requirement to the GSD database and regenerate REQUIREMENTS.md.", requirementSaveParams, async (args) => {
|
|
602
|
+
const parsed = parseWorkflowArgs(requirementSaveSchema, args);
|
|
603
|
+
const { projectDir, ...params } = parsed;
|
|
604
|
+
await enforceWorkflowWriteGate("gsd_requirement_save", projectDir);
|
|
605
|
+
const result = await runSerializedWorkflowOperation(async () => {
|
|
606
|
+
const { saveRequirementToDb } = await importLocalModule("../../../src/resources/extensions/gsd/db-writer.js");
|
|
607
|
+
return saveRequirementToDb(params, projectDir);
|
|
608
|
+
});
|
|
609
|
+
return { content: [{ type: "text", text: `Saved requirement ${result.id}` }] };
|
|
610
|
+
});
|
|
611
|
+
server.tool("gsd_save_requirement", "Alias for gsd_requirement_save. Record a new requirement to the GSD database and regenerate REQUIREMENTS.md.", requirementSaveParams, async (args) => {
|
|
612
|
+
const parsed = parseWorkflowArgs(requirementSaveSchema, args);
|
|
613
|
+
const { projectDir, ...params } = parsed;
|
|
614
|
+
await enforceWorkflowWriteGate("gsd_requirement_save", projectDir);
|
|
615
|
+
const result = await runSerializedWorkflowOperation(async () => {
|
|
616
|
+
const { saveRequirementToDb } = await importLocalModule("../../../src/resources/extensions/gsd/db-writer.js");
|
|
617
|
+
return saveRequirementToDb(params, projectDir);
|
|
618
|
+
});
|
|
619
|
+
return { content: [{ type: "text", text: `Saved requirement ${result.id}` }] };
|
|
620
|
+
});
|
|
621
|
+
server.tool("gsd_milestone_generate_id", "Generate the next milestone ID for a new GSD milestone.", milestoneGenerateIdParams, async (args) => {
|
|
622
|
+
const { projectDir } = parseWorkflowArgs(milestoneGenerateIdSchema, args);
|
|
623
|
+
await enforceWorkflowWriteGate("gsd_milestone_generate_id", projectDir);
|
|
624
|
+
const id = await runSerializedWorkflowOperation(async () => {
|
|
625
|
+
const { claimReservedId, findMilestoneIds, getReservedMilestoneIds, nextMilestoneId, } = await importLocalModule("../../../src/resources/extensions/gsd/milestone-ids.js");
|
|
626
|
+
const reserved = claimReservedId();
|
|
627
|
+
if (reserved) {
|
|
628
|
+
await ensureMilestoneDbRow(reserved);
|
|
629
|
+
return reserved;
|
|
630
|
+
}
|
|
631
|
+
const allIds = [...new Set([...findMilestoneIds(projectDir), ...getReservedMilestoneIds()])];
|
|
632
|
+
const nextId = nextMilestoneId(allIds);
|
|
633
|
+
await ensureMilestoneDbRow(nextId);
|
|
634
|
+
return nextId;
|
|
635
|
+
});
|
|
636
|
+
return { content: [{ type: "text", text: id }] };
|
|
637
|
+
});
|
|
638
|
+
server.tool("gsd_generate_milestone_id", "Alias for gsd_milestone_generate_id. Generate the next milestone ID for a new GSD milestone.", milestoneGenerateIdParams, async (args) => {
|
|
639
|
+
const { projectDir } = parseWorkflowArgs(milestoneGenerateIdSchema, args);
|
|
640
|
+
await enforceWorkflowWriteGate("gsd_milestone_generate_id", projectDir);
|
|
641
|
+
const id = await runSerializedWorkflowOperation(async () => {
|
|
642
|
+
const { claimReservedId, findMilestoneIds, getReservedMilestoneIds, nextMilestoneId, } = await importLocalModule("../../../src/resources/extensions/gsd/milestone-ids.js");
|
|
643
|
+
const reserved = claimReservedId();
|
|
644
|
+
if (reserved) {
|
|
645
|
+
await ensureMilestoneDbRow(reserved);
|
|
646
|
+
return reserved;
|
|
647
|
+
}
|
|
648
|
+
const allIds = [...new Set([...findMilestoneIds(projectDir), ...getReservedMilestoneIds()])];
|
|
649
|
+
const nextId = nextMilestoneId(allIds);
|
|
650
|
+
await ensureMilestoneDbRow(nextId);
|
|
651
|
+
return nextId;
|
|
652
|
+
});
|
|
653
|
+
return { content: [{ type: "text", text: id }] };
|
|
654
|
+
});
|
|
446
655
|
server.tool("gsd_plan_milestone", "Write milestone planning state to the GSD database and render ROADMAP.md from DB.", planMilestoneParams, async (args) => {
|
|
447
656
|
const parsed = parseWorkflowArgs(planMilestoneSchema, args);
|
|
448
657
|
const { projectDir, ...params } = parsed;
|
|
@@ -457,6 +666,36 @@ export function registerWorkflowTools(server) {
|
|
|
457
666
|
const { executePlanSlice } = await getWorkflowToolExecutors();
|
|
458
667
|
return runSerializedWorkflowOperation(() => executePlanSlice(params, projectDir));
|
|
459
668
|
});
|
|
669
|
+
server.tool("gsd_plan_task", "Write task planning state to the GSD database and render tasks/T##-PLAN.md from DB.", planTaskParams, async (args) => {
|
|
670
|
+
const parsed = parseWorkflowArgs(planTaskSchema, args);
|
|
671
|
+
const { projectDir, ...params } = parsed;
|
|
672
|
+
await enforceWorkflowWriteGate("gsd_plan_task", projectDir, params.milestoneId);
|
|
673
|
+
const result = await runSerializedWorkflowOperation(async () => {
|
|
674
|
+
const { handlePlanTask } = await importLocalModule("../../../src/resources/extensions/gsd/tools/plan-task.js");
|
|
675
|
+
return handlePlanTask(params, projectDir);
|
|
676
|
+
});
|
|
677
|
+
if ("error" in result) {
|
|
678
|
+
throw new Error(result.error);
|
|
679
|
+
}
|
|
680
|
+
return {
|
|
681
|
+
content: [{ type: "text", text: `Planned task ${result.taskId} (${result.sliceId}/${result.milestoneId})` }],
|
|
682
|
+
};
|
|
683
|
+
});
|
|
684
|
+
server.tool("gsd_task_plan", "Alias for gsd_plan_task. Write task planning state to the GSD database and render tasks/T##-PLAN.md from DB.", planTaskParams, async (args) => {
|
|
685
|
+
const parsed = parseWorkflowArgs(planTaskSchema, args);
|
|
686
|
+
const { projectDir, ...params } = parsed;
|
|
687
|
+
await enforceWorkflowWriteGate("gsd_plan_task", projectDir, params.milestoneId);
|
|
688
|
+
const result = await runSerializedWorkflowOperation(async () => {
|
|
689
|
+
const { handlePlanTask } = await importLocalModule("../../../src/resources/extensions/gsd/tools/plan-task.js");
|
|
690
|
+
return handlePlanTask(params, projectDir);
|
|
691
|
+
});
|
|
692
|
+
if ("error" in result) {
|
|
693
|
+
throw new Error(result.error);
|
|
694
|
+
}
|
|
695
|
+
return {
|
|
696
|
+
content: [{ type: "text", text: `Planned task ${result.taskId} (${result.sliceId}/${result.milestoneId})` }],
|
|
697
|
+
};
|
|
698
|
+
});
|
|
460
699
|
server.tool("gsd_replan_slice", "Replan a slice after a blocker is discovered, preserving completed tasks and re-rendering PLAN.md + REPLAN.md.", replanSliceParams, async (args) => {
|
|
461
700
|
const parsed = parseWorkflowArgs(replanSliceSchema, args);
|
|
462
701
|
return handleReplanSlice(parsed.projectDir, parsed);
|
|
@@ -473,6 +712,30 @@ export function registerWorkflowTools(server) {
|
|
|
473
712
|
const parsed = parseWorkflowArgs(sliceCompleteSchema, args);
|
|
474
713
|
return handleSliceComplete(parsed.projectDir, parsed);
|
|
475
714
|
});
|
|
715
|
+
server.tool("gsd_skip_slice", "Mark a slice as skipped so auto-mode advances past it without executing.", skipSliceParams, async (args) => {
|
|
716
|
+
const { projectDir, milestoneId, sliceId, reason } = parseWorkflowArgs(skipSliceSchema, args);
|
|
717
|
+
await enforceWorkflowWriteGate("gsd_skip_slice", projectDir, milestoneId);
|
|
718
|
+
await runSerializedWorkflowOperation(async () => {
|
|
719
|
+
const { getSlice, updateSliceStatus } = await importLocalModule("../../../src/resources/extensions/gsd/gsd-db.js");
|
|
720
|
+
const { invalidateStateCache } = await importLocalModule("../../../src/resources/extensions/gsd/state.js");
|
|
721
|
+
const { rebuildState } = await importLocalModule("../../../src/resources/extensions/gsd/doctor.js");
|
|
722
|
+
const slice = getSlice(milestoneId, sliceId);
|
|
723
|
+
if (!slice) {
|
|
724
|
+
throw new Error(`Slice ${sliceId} not found in milestone ${milestoneId}`);
|
|
725
|
+
}
|
|
726
|
+
if (slice.status === "complete" || slice.status === "done") {
|
|
727
|
+
throw new Error(`Slice ${sliceId} is already complete and cannot be skipped`);
|
|
728
|
+
}
|
|
729
|
+
if (slice.status !== "skipped") {
|
|
730
|
+
updateSliceStatus(milestoneId, sliceId, "skipped");
|
|
731
|
+
invalidateStateCache();
|
|
732
|
+
await rebuildState(projectDir);
|
|
733
|
+
}
|
|
734
|
+
});
|
|
735
|
+
return {
|
|
736
|
+
content: [{ type: "text", text: `Skipped slice ${sliceId} (${milestoneId}). Reason: ${reason ?? "User-directed skip"}.` }],
|
|
737
|
+
};
|
|
738
|
+
});
|
|
476
739
|
server.tool("gsd_complete_milestone", "Record a completed milestone to the GSD database and render its SUMMARY.md.", completeMilestoneParams, async (args) => {
|
|
477
740
|
const parsed = parseWorkflowArgs(completeMilestoneSchema, args);
|
|
478
741
|
return handleCompleteMilestone(parsed.projectDir, parsed);
|
|
@@ -528,5 +791,14 @@ export function registerWorkflowTools(server) {
|
|
|
528
791
|
const { executeMilestoneStatus } = await getWorkflowToolExecutors();
|
|
529
792
|
return runSerializedWorkflowOperation(() => executeMilestoneStatus({ milestoneId }, projectDir));
|
|
530
793
|
});
|
|
794
|
+
server.tool("gsd_journal_query", "Query the structured event journal for auto-mode iterations.", journalQueryParams, async (args) => {
|
|
795
|
+
const { projectDir, limit, ...filters } = parseWorkflowArgs(journalQuerySchema, args);
|
|
796
|
+
const { queryJournal } = await importLocalModule("../../../src/resources/extensions/gsd/journal.js");
|
|
797
|
+
const entries = queryJournal(projectDir, filters).slice(0, limit ?? 100);
|
|
798
|
+
if (entries.length === 0) {
|
|
799
|
+
return { content: [{ type: "text", text: "No matching journal entries found." }] };
|
|
800
|
+
}
|
|
801
|
+
return { content: [{ type: "text", text: JSON.stringify(entries, null, 2) }] };
|
|
802
|
+
});
|
|
531
803
|
}
|
|
532
804
|
//# sourceMappingURL=workflow-tools.js.map
|