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.
Files changed (87) hide show
  1. package/dist/resources/extensions/gsd/bootstrap/system-context.js +6 -2
  2. package/dist/resources/extensions/gsd/commands-cmux.js +30 -1
  3. package/dist/resources/extensions/gsd/workflow-mcp.js +53 -6
  4. package/dist/web/standalone/.next/BUILD_ID +1 -1
  5. package/dist/web/standalone/.next/app-path-routes-manifest.json +12 -12
  6. package/dist/web/standalone/.next/build-manifest.json +2 -2
  7. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  8. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  9. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  10. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  11. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  12. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  13. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  14. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  15. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  16. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  17. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  18. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  19. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  20. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  21. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  22. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  23. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  24. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  25. package/dist/web/standalone/.next/server/app/index.html +1 -1
  26. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  27. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  28. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  29. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  30. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  31. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  32. package/dist/web/standalone/.next/server/app-paths-manifest.json +12 -12
  33. package/dist/web/standalone/.next/server/chunks/6897.js +3 -3
  34. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  35. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  36. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  37. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  38. package/package.json +1 -1
  39. package/packages/daemon/src/orchestrator.ts +9 -84
  40. package/packages/mcp-server/README.md +25 -3
  41. package/packages/mcp-server/dist/cli.d.ts +0 -1
  42. package/packages/mcp-server/dist/cli.d.ts.map +1 -1
  43. package/packages/mcp-server/dist/cli.js +4 -2
  44. package/packages/mcp-server/dist/cli.js.map +1 -1
  45. package/packages/mcp-server/dist/server.d.ts +32 -1
  46. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  47. package/packages/mcp-server/dist/server.js +118 -1
  48. package/packages/mcp-server/dist/server.js.map +1 -1
  49. package/packages/mcp-server/dist/tool-credentials.d.ts +6 -0
  50. package/packages/mcp-server/dist/tool-credentials.d.ts.map +1 -0
  51. package/packages/mcp-server/dist/tool-credentials.js +90 -0
  52. package/packages/mcp-server/dist/tool-credentials.js.map +1 -0
  53. package/packages/mcp-server/dist/workflow-tools.d.ts +1 -0
  54. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  55. package/packages/mcp-server/dist/workflow-tools.js +274 -2
  56. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  57. package/packages/mcp-server/src/cli.ts +5 -3
  58. package/packages/mcp-server/src/mcp-server.test.ts +85 -1
  59. package/packages/mcp-server/src/server.ts +188 -1
  60. package/packages/mcp-server/src/tool-credentials.test.ts +95 -0
  61. package/packages/mcp-server/src/tool-credentials.ts +97 -0
  62. package/packages/mcp-server/src/workflow-tools.test.ts +32 -25
  63. package/packages/mcp-server/src/workflow-tools.ts +365 -2
  64. package/packages/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
  65. package/packages/pi-ai/dist/providers/anthropic.js +1 -23
  66. package/packages/pi-ai/dist/providers/anthropic.js.map +1 -1
  67. package/packages/pi-ai/dist/utils/oauth/index.d.ts +3 -2
  68. package/packages/pi-ai/dist/utils/oauth/index.d.ts.map +1 -1
  69. package/packages/pi-ai/dist/utils/oauth/index.js +3 -5
  70. package/packages/pi-ai/dist/utils/oauth/index.js.map +1 -1
  71. package/packages/pi-ai/src/providers/anthropic.ts +1 -31
  72. package/packages/pi-ai/src/utils/oauth/index.ts +3 -5
  73. package/packages/pi-coding-agent/package.json +1 -1
  74. package/pkg/package.json +1 -1
  75. package/src/resources/extensions/gsd/bootstrap/system-context.ts +9 -5
  76. package/src/resources/extensions/gsd/commands-cmux.ts +32 -1
  77. package/src/resources/extensions/gsd/tests/cmux.test.ts +67 -1
  78. package/src/resources/extensions/gsd/tests/mcp-project-config.test.ts +6 -2
  79. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +23 -7
  80. package/src/resources/extensions/gsd/workflow-mcp.ts +59 -5
  81. package/packages/pi-ai/dist/utils/oauth/anthropic.d.ts +0 -17
  82. package/packages/pi-ai/dist/utils/oauth/anthropic.d.ts.map +0 -1
  83. package/packages/pi-ai/dist/utils/oauth/anthropic.js +0 -106
  84. package/packages/pi-ai/dist/utils/oauth/anthropic.js.map +0 -1
  85. package/packages/pi-ai/src/utils/oauth/anthropic.ts +0 -140
  86. /package/dist/web/standalone/.next/static/{DrWdzskk28E5Qz-Wjw1mj → Nl6lg7zP5dNgNBV1107v1}/_buildManifest.js +0 -0
  87. /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]+:/i.test(explicitModule) && !explicitModule.startsWith("file:")) {
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]+:/i.test(explicitModule) && !explicitModule.startsWith("file:")) {
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