gsd-pi 2.67.0-dev.a5b1d8f → 2.67.0-dev.fe39184
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 +41 -31
- package/dist/resources/extensions/claude-code-cli/stream-adapter.js +121 -8
- package/dist/resources/extensions/gsd/auto/phases.js +17 -0
- package/dist/resources/extensions/gsd/auto/session.js +6 -0
- package/dist/resources/extensions/gsd/auto-direct-dispatch.js +12 -0
- package/dist/resources/extensions/gsd/auto-start.js +12 -0
- package/dist/resources/extensions/gsd/auto.js +27 -0
- package/dist/resources/extensions/gsd/bootstrap/db-tools.js +11 -435
- package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +1 -4
- package/dist/resources/extensions/gsd/bootstrap/query-tools.js +7 -64
- package/dist/resources/extensions/gsd/bootstrap/write-gate.js +88 -8
- package/dist/resources/extensions/gsd/commands/catalog.js +2 -1
- package/dist/resources/extensions/gsd/commands/handlers/core.js +39 -25
- package/dist/resources/extensions/gsd/commands/index.js +8 -1
- package/dist/resources/extensions/gsd/commands-mcp-status.js +43 -7
- package/dist/resources/extensions/gsd/guided-flow.js +16 -0
- package/dist/resources/extensions/gsd/init-wizard.js +37 -0
- package/dist/resources/extensions/gsd/mcp-project-config.js +83 -0
- package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +508 -0
- package/dist/resources/extensions/gsd/workflow-logger.js +18 -3
- package/dist/resources/extensions/gsd/workflow-mcp.js +261 -0
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +15 -15
- package/dist/web/standalone/.next/build-manifest.json +3 -3
- package/dist/web/standalone/.next/prerender-manifest.json +3 -3
- package/dist/web/standalone/.next/react-loadable-manifest.json +1 -1
- 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/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 +15 -15
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/middleware-react-loadable-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/dist/web/standalone/.next/static/chunks/6502.5dcdcf1e1432e20d.js +9 -0
- package/dist/web/standalone/.next/static/chunks/{webpack-b49b09f97429b5d0.js → webpack-42a66876b763aa26.js} +1 -1
- package/package.json +4 -2
- package/packages/mcp-server/README.md +38 -0
- package/packages/mcp-server/dist/cli.d.ts +9 -0
- package/packages/mcp-server/dist/cli.d.ts.map +1 -0
- package/packages/mcp-server/dist/cli.js +58 -0
- package/packages/mcp-server/dist/cli.js.map +1 -0
- package/packages/mcp-server/dist/index.d.ts +20 -0
- package/packages/mcp-server/dist/index.d.ts.map +1 -0
- package/packages/mcp-server/dist/index.js +14 -0
- package/packages/mcp-server/dist/index.js.map +1 -0
- package/packages/mcp-server/dist/readers/captures.d.ts +25 -0
- package/packages/mcp-server/dist/readers/captures.d.ts.map +1 -0
- package/packages/mcp-server/dist/readers/captures.js +67 -0
- package/packages/mcp-server/dist/readers/captures.js.map +1 -0
- package/packages/mcp-server/dist/readers/doctor-lite.d.ts +20 -0
- package/packages/mcp-server/dist/readers/doctor-lite.d.ts.map +1 -0
- package/packages/mcp-server/dist/readers/doctor-lite.js +173 -0
- package/packages/mcp-server/dist/readers/doctor-lite.js.map +1 -0
- package/packages/mcp-server/dist/readers/index.d.ts +14 -0
- package/packages/mcp-server/dist/readers/index.d.ts.map +1 -0
- package/packages/mcp-server/dist/readers/index.js +10 -0
- package/packages/mcp-server/dist/readers/index.js.map +1 -0
- package/packages/mcp-server/dist/readers/knowledge.d.ts +18 -0
- package/packages/mcp-server/dist/readers/knowledge.d.ts.map +1 -0
- package/packages/mcp-server/dist/readers/knowledge.js +82 -0
- package/packages/mcp-server/dist/readers/knowledge.js.map +1 -0
- package/packages/mcp-server/dist/readers/metrics.d.ts +32 -0
- package/packages/mcp-server/dist/readers/metrics.d.ts.map +1 -0
- package/packages/mcp-server/dist/readers/metrics.js +74 -0
- package/packages/mcp-server/dist/readers/metrics.js.map +1 -0
- package/packages/mcp-server/dist/readers/paths.d.ts +42 -0
- package/packages/mcp-server/dist/readers/paths.d.ts.map +1 -0
- package/packages/mcp-server/dist/readers/paths.js +199 -0
- package/packages/mcp-server/dist/readers/paths.js.map +1 -0
- package/packages/mcp-server/dist/readers/roadmap.d.ts +26 -0
- package/packages/mcp-server/dist/readers/roadmap.d.ts.map +1 -0
- package/packages/mcp-server/dist/readers/roadmap.js +194 -0
- package/packages/mcp-server/dist/readers/roadmap.js.map +1 -0
- package/packages/mcp-server/dist/readers/state.d.ts +43 -0
- package/packages/mcp-server/dist/readers/state.d.ts.map +1 -0
- package/packages/mcp-server/dist/readers/state.js +184 -0
- package/packages/mcp-server/dist/readers/state.js.map +1 -0
- package/packages/mcp-server/dist/server.d.ts +28 -0
- package/packages/mcp-server/dist/server.d.ts.map +1 -0
- package/packages/mcp-server/dist/server.js +319 -0
- package/packages/mcp-server/dist/server.js.map +1 -0
- package/packages/mcp-server/dist/session-manager.d.ts +54 -0
- package/packages/mcp-server/dist/session-manager.d.ts.map +1 -0
- package/packages/mcp-server/dist/session-manager.js +284 -0
- package/packages/mcp-server/dist/session-manager.js.map +1 -0
- package/packages/mcp-server/dist/types.d.ts +61 -0
- package/packages/mcp-server/dist/types.d.ts.map +1 -0
- package/packages/mcp-server/dist/types.js +11 -0
- package/packages/mcp-server/dist/types.js.map +1 -0
- package/packages/mcp-server/dist/workflow-tools.d.ts +9 -0
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -0
- package/packages/mcp-server/dist/workflow-tools.js +532 -0
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -0
- package/packages/mcp-server/src/server.ts +6 -2
- package/packages/mcp-server/src/workflow-tools.test.ts +976 -0
- package/packages/mcp-server/src/workflow-tools.ts +997 -0
- package/packages/mcp-server/tsconfig.json +1 -1
- package/packages/pi-agent-core/dist/agent-loop.js +14 -6
- package/packages/pi-agent-core/dist/agent-loop.js.map +1 -1
- package/packages/pi-agent-core/src/agent-loop.test.ts +53 -0
- package/packages/pi-agent-core/src/agent-loop.ts +20 -6
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js +28 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +17 -12
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +19 -0
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
- package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-execution.test.ts +54 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +18 -12
- package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +21 -0
- package/packages/rpc-client/dist/index.d.ts +10 -0
- package/packages/rpc-client/dist/index.d.ts.map +1 -0
- package/packages/rpc-client/dist/index.js +9 -0
- package/packages/rpc-client/dist/index.js.map +1 -0
- package/packages/rpc-client/dist/jsonl.d.ts +17 -0
- package/packages/rpc-client/dist/jsonl.d.ts.map +1 -0
- package/packages/rpc-client/dist/jsonl.js +54 -0
- package/packages/rpc-client/dist/jsonl.js.map +1 -0
- package/packages/rpc-client/dist/rpc-client.d.ts +259 -0
- package/packages/rpc-client/dist/rpc-client.d.ts.map +1 -0
- package/packages/rpc-client/dist/rpc-client.js +541 -0
- package/packages/rpc-client/dist/rpc-client.js.map +1 -0
- package/packages/rpc-client/dist/rpc-client.test.d.ts +2 -0
- package/packages/rpc-client/dist/rpc-client.test.d.ts.map +1 -0
- package/packages/rpc-client/dist/rpc-client.test.js +477 -0
- package/packages/rpc-client/dist/rpc-client.test.js.map +1 -0
- package/packages/rpc-client/dist/rpc-types.d.ts +566 -0
- package/packages/rpc-client/dist/rpc-types.d.ts.map +1 -0
- package/packages/rpc-client/dist/rpc-types.js +12 -0
- package/packages/rpc-client/dist/rpc-types.js.map +1 -0
- package/scripts/ensure-workspace-builds.cjs +2 -0
- package/scripts/link-workspace-packages.cjs +21 -14
- package/src/resources/extensions/claude-code-cli/stream-adapter.ts +157 -8
- package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +182 -0
- package/src/resources/extensions/gsd/auto/phases.ts +25 -0
- package/src/resources/extensions/gsd/auto/session.ts +6 -0
- package/src/resources/extensions/gsd/auto-direct-dispatch.ts +20 -0
- package/src/resources/extensions/gsd/auto-start.ts +15 -1
- package/src/resources/extensions/gsd/auto.ts +29 -1
- package/src/resources/extensions/gsd/bootstrap/db-tools.ts +22 -435
- package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +1 -5
- package/src/resources/extensions/gsd/bootstrap/query-tools.ts +7 -72
- package/src/resources/extensions/gsd/bootstrap/write-gate.ts +122 -6
- package/src/resources/extensions/gsd/commands/catalog.ts +2 -1
- package/src/resources/extensions/gsd/commands/handlers/core.ts +53 -26
- package/src/resources/extensions/gsd/commands/index.ts +7 -1
- package/src/resources/extensions/gsd/commands-mcp-status.ts +53 -7
- package/src/resources/extensions/gsd/guided-flow.ts +24 -0
- package/src/resources/extensions/gsd/init-wizard.ts +40 -0
- package/src/resources/extensions/gsd/mcp-project-config.ts +128 -0
- package/src/resources/extensions/gsd/tests/auto-project-root-env.test.ts +29 -0
- package/src/resources/extensions/gsd/tests/core-overlay-fallback.test.ts +101 -0
- package/src/resources/extensions/gsd/tests/ensure-db-open.test.ts +66 -0
- package/src/resources/extensions/gsd/tests/mcp-project-config.test.ts +85 -0
- package/src/resources/extensions/gsd/tests/mcp-status.test.ts +15 -0
- package/src/resources/extensions/gsd/tests/workflow-logger.test.ts +16 -0
- package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +500 -0
- package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +625 -0
- package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +629 -0
- package/src/resources/extensions/gsd/workflow-logger.ts +19 -3
- package/src/resources/extensions/gsd/workflow-mcp.ts +320 -0
- package/dist/web/standalone/.next/static/chunks/6502.b804e48b7919f55e.js +0 -9
- /package/dist/web/standalone/.next/static/{NllX5BEOLdTXS9ypf1i3i → gbSATDX4Jt2ufxzUr5nYm}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{NllX5BEOLdTXS9ypf1i3i → gbSATDX4Jt2ufxzUr5nYm}/_ssgManifest.js +0 -0
|
@@ -0,0 +1,319 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* MCP Server — registers GSD orchestration, project-state, and workflow tools.
|
|
3
|
+
*
|
|
4
|
+
* Session tools (6): gsd_execute, gsd_status, gsd_result, gsd_cancel, gsd_query, gsd_resolve_blocker
|
|
5
|
+
* Read-only tools (6): gsd_progress, gsd_roadmap, gsd_history, gsd_doctor, gsd_captures, gsd_knowledge
|
|
6
|
+
* Workflow tools (17): planning, replanning, completion, validation, reassessment, gate result, and milestone status tools
|
|
7
|
+
*
|
|
8
|
+
* Uses dynamic imports for @modelcontextprotocol/sdk because TS Node16
|
|
9
|
+
* cannot resolve the SDK's subpath exports statically (same pattern as
|
|
10
|
+
* src/mcp-server.ts in the main package).
|
|
11
|
+
*/
|
|
12
|
+
import { readFile, readdir, stat } from 'node:fs/promises';
|
|
13
|
+
import { join, resolve } from 'node:path';
|
|
14
|
+
import { z } from 'zod';
|
|
15
|
+
import { readProgress } from './readers/state.js';
|
|
16
|
+
import { readRoadmap } from './readers/roadmap.js';
|
|
17
|
+
import { readHistory } from './readers/metrics.js';
|
|
18
|
+
import { readCaptures } from './readers/captures.js';
|
|
19
|
+
import { readKnowledge } from './readers/knowledge.js';
|
|
20
|
+
import { runDoctorLite } from './readers/doctor-lite.js';
|
|
21
|
+
import { registerWorkflowTools } from './workflow-tools.js';
|
|
22
|
+
// ---------------------------------------------------------------------------
|
|
23
|
+
// Constants
|
|
24
|
+
// ---------------------------------------------------------------------------
|
|
25
|
+
const MCP_PKG = '@modelcontextprotocol/sdk';
|
|
26
|
+
const SERVER_NAME = 'gsd';
|
|
27
|
+
const SERVER_VERSION = '2.53.0';
|
|
28
|
+
// ---------------------------------------------------------------------------
|
|
29
|
+
// Tool result helpers
|
|
30
|
+
// ---------------------------------------------------------------------------
|
|
31
|
+
/** Wrap a JSON-serializable value as MCP tool content. */
|
|
32
|
+
function jsonContent(data) {
|
|
33
|
+
return { content: [{ type: 'text', text: JSON.stringify(data, null, 2) }] };
|
|
34
|
+
}
|
|
35
|
+
/** Return an MCP error response. */
|
|
36
|
+
function errorContent(message) {
|
|
37
|
+
return { isError: true, content: [{ type: 'text', text: message }] };
|
|
38
|
+
}
|
|
39
|
+
// ---------------------------------------------------------------------------
|
|
40
|
+
// gsd_query filesystem reader
|
|
41
|
+
// ---------------------------------------------------------------------------
|
|
42
|
+
async function readProjectState(projectDir, _query) {
|
|
43
|
+
const gsdDir = join(resolve(projectDir), '.gsd');
|
|
44
|
+
const result = { projectDir: resolve(projectDir) };
|
|
45
|
+
// STATE.md — current execution state
|
|
46
|
+
try {
|
|
47
|
+
result.state = await readFile(join(gsdDir, 'STATE.md'), 'utf-8');
|
|
48
|
+
}
|
|
49
|
+
catch {
|
|
50
|
+
result.state = null;
|
|
51
|
+
}
|
|
52
|
+
// PROJECT.md — project description
|
|
53
|
+
try {
|
|
54
|
+
result.project = await readFile(join(gsdDir, 'PROJECT.md'), 'utf-8');
|
|
55
|
+
}
|
|
56
|
+
catch {
|
|
57
|
+
result.project = null;
|
|
58
|
+
}
|
|
59
|
+
// REQUIREMENTS.md — requirement contract
|
|
60
|
+
try {
|
|
61
|
+
result.requirements = await readFile(join(gsdDir, 'REQUIREMENTS.md'), 'utf-8');
|
|
62
|
+
}
|
|
63
|
+
catch {
|
|
64
|
+
result.requirements = null;
|
|
65
|
+
}
|
|
66
|
+
// List milestones with basic metadata
|
|
67
|
+
const milestonesDir = join(gsdDir, 'milestones');
|
|
68
|
+
try {
|
|
69
|
+
const entries = await readdir(milestonesDir, { withFileTypes: true });
|
|
70
|
+
const milestones = [];
|
|
71
|
+
for (const entry of entries) {
|
|
72
|
+
if (!entry.isDirectory())
|
|
73
|
+
continue;
|
|
74
|
+
const mDir = join(milestonesDir, entry.name);
|
|
75
|
+
const hasRoadmap = await fileExists(join(mDir, `${entry.name}-ROADMAP.md`));
|
|
76
|
+
const hasSummary = await fileExists(join(mDir, `${entry.name}-SUMMARY.md`));
|
|
77
|
+
milestones.push({ id: entry.name, hasRoadmap, hasSummary });
|
|
78
|
+
}
|
|
79
|
+
result.milestones = milestones;
|
|
80
|
+
}
|
|
81
|
+
catch {
|
|
82
|
+
result.milestones = [];
|
|
83
|
+
}
|
|
84
|
+
return result;
|
|
85
|
+
}
|
|
86
|
+
async function fileExists(path) {
|
|
87
|
+
try {
|
|
88
|
+
await stat(path);
|
|
89
|
+
return true;
|
|
90
|
+
}
|
|
91
|
+
catch {
|
|
92
|
+
return false;
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
// ---------------------------------------------------------------------------
|
|
96
|
+
// createMcpServer
|
|
97
|
+
// ---------------------------------------------------------------------------
|
|
98
|
+
/**
|
|
99
|
+
* Create and configure an MCP server with session, read-only, and workflow tools.
|
|
100
|
+
*
|
|
101
|
+
* Returns the McpServer instance — call `connect(transport)` to start serving.
|
|
102
|
+
* Uses dynamic imports for the MCP SDK to avoid TS subpath resolution issues.
|
|
103
|
+
*/
|
|
104
|
+
export async function createMcpServer(sessionManager) {
|
|
105
|
+
// Dynamic import — same workaround as src/mcp-server.ts
|
|
106
|
+
const mcpMod = await import(`${MCP_PKG}/server/mcp.js`);
|
|
107
|
+
const McpServer = mcpMod.McpServer;
|
|
108
|
+
const server = new McpServer({ name: SERVER_NAME, version: SERVER_VERSION }, { capabilities: { tools: {} } });
|
|
109
|
+
// -----------------------------------------------------------------------
|
|
110
|
+
// gsd_execute — start a new GSD auto-mode session
|
|
111
|
+
// -----------------------------------------------------------------------
|
|
112
|
+
server.tool('gsd_execute', 'Start a GSD auto-mode session for a project directory. Returns a sessionId for tracking.', {
|
|
113
|
+
projectDir: z.string().describe('Absolute path to the project directory'),
|
|
114
|
+
command: z.string().optional().describe('Command to send (default: "/gsd auto")'),
|
|
115
|
+
model: z.string().optional().describe('Model ID override'),
|
|
116
|
+
bare: z.boolean().optional().describe('Run in bare mode (skip user config)'),
|
|
117
|
+
}, async (args) => {
|
|
118
|
+
const { projectDir, command, model, bare } = args;
|
|
119
|
+
try {
|
|
120
|
+
const sessionId = await sessionManager.startSession(projectDir, { command, model, bare });
|
|
121
|
+
return jsonContent({ sessionId, status: 'started' });
|
|
122
|
+
}
|
|
123
|
+
catch (err) {
|
|
124
|
+
return errorContent(err instanceof Error ? err.message : String(err));
|
|
125
|
+
}
|
|
126
|
+
});
|
|
127
|
+
// -----------------------------------------------------------------------
|
|
128
|
+
// gsd_status — poll session status
|
|
129
|
+
// -----------------------------------------------------------------------
|
|
130
|
+
server.tool('gsd_status', 'Get the current status of a GSD session including progress, recent events, and pending blockers.', {
|
|
131
|
+
sessionId: z.string().describe('Session ID returned from gsd_execute'),
|
|
132
|
+
}, async (args) => {
|
|
133
|
+
const { sessionId } = args;
|
|
134
|
+
try {
|
|
135
|
+
const session = sessionManager.getSession(sessionId);
|
|
136
|
+
if (!session)
|
|
137
|
+
return errorContent(`Session not found: ${sessionId}`);
|
|
138
|
+
const durationMs = Date.now() - session.startTime;
|
|
139
|
+
const toolCallCount = session.events.filter((e) => e.type === 'tool_use' ||
|
|
140
|
+
e.type === 'tool_execution_start').length;
|
|
141
|
+
return jsonContent({
|
|
142
|
+
status: session.status,
|
|
143
|
+
progress: {
|
|
144
|
+
eventCount: session.events.length,
|
|
145
|
+
toolCalls: toolCallCount,
|
|
146
|
+
},
|
|
147
|
+
recentEvents: session.events.slice(-10),
|
|
148
|
+
pendingBlocker: session.pendingBlocker
|
|
149
|
+
? {
|
|
150
|
+
id: session.pendingBlocker.id,
|
|
151
|
+
method: session.pendingBlocker.method,
|
|
152
|
+
message: session.pendingBlocker.message,
|
|
153
|
+
}
|
|
154
|
+
: null,
|
|
155
|
+
cost: session.cost,
|
|
156
|
+
durationMs,
|
|
157
|
+
});
|
|
158
|
+
}
|
|
159
|
+
catch (err) {
|
|
160
|
+
return errorContent(err instanceof Error ? err.message : String(err));
|
|
161
|
+
}
|
|
162
|
+
});
|
|
163
|
+
// -----------------------------------------------------------------------
|
|
164
|
+
// gsd_result — get accumulated session result
|
|
165
|
+
// -----------------------------------------------------------------------
|
|
166
|
+
server.tool('gsd_result', 'Get the result of a GSD session. Returns partial results if the session is still running.', {
|
|
167
|
+
sessionId: z.string().describe('Session ID returned from gsd_execute'),
|
|
168
|
+
}, async (args) => {
|
|
169
|
+
const { sessionId } = args;
|
|
170
|
+
try {
|
|
171
|
+
const result = sessionManager.getResult(sessionId);
|
|
172
|
+
return jsonContent(result);
|
|
173
|
+
}
|
|
174
|
+
catch (err) {
|
|
175
|
+
return errorContent(err instanceof Error ? err.message : String(err));
|
|
176
|
+
}
|
|
177
|
+
});
|
|
178
|
+
// -----------------------------------------------------------------------
|
|
179
|
+
// gsd_cancel — cancel a running session
|
|
180
|
+
// -----------------------------------------------------------------------
|
|
181
|
+
server.tool('gsd_cancel', 'Cancel a running GSD session. Aborts the current operation and stops the process.', {
|
|
182
|
+
sessionId: z.string().describe('Session ID returned from gsd_execute'),
|
|
183
|
+
}, async (args) => {
|
|
184
|
+
const { sessionId } = args;
|
|
185
|
+
try {
|
|
186
|
+
await sessionManager.cancelSession(sessionId);
|
|
187
|
+
return jsonContent({ cancelled: true });
|
|
188
|
+
}
|
|
189
|
+
catch (err) {
|
|
190
|
+
return errorContent(err instanceof Error ? err.message : String(err));
|
|
191
|
+
}
|
|
192
|
+
});
|
|
193
|
+
// -----------------------------------------------------------------------
|
|
194
|
+
// gsd_query — read project state from filesystem (no session needed)
|
|
195
|
+
// -----------------------------------------------------------------------
|
|
196
|
+
server.tool('gsd_query', 'Query GSD project state from the filesystem. Returns STATE.md, PROJECT.md, requirements, and milestone listing. Does not require an active session.', {
|
|
197
|
+
projectDir: z.string().describe('Absolute path to the project directory'),
|
|
198
|
+
query: z.string().describe('What to query (e.g. "status", "milestones", "requirements")'),
|
|
199
|
+
}, async (args) => {
|
|
200
|
+
const { projectDir, query } = args;
|
|
201
|
+
try {
|
|
202
|
+
const state = await readProjectState(projectDir, query);
|
|
203
|
+
return jsonContent(state);
|
|
204
|
+
}
|
|
205
|
+
catch (err) {
|
|
206
|
+
return errorContent(err instanceof Error ? err.message : String(err));
|
|
207
|
+
}
|
|
208
|
+
});
|
|
209
|
+
// -----------------------------------------------------------------------
|
|
210
|
+
// gsd_resolve_blocker — resolve a pending blocker
|
|
211
|
+
// -----------------------------------------------------------------------
|
|
212
|
+
server.tool('gsd_resolve_blocker', 'Resolve a pending blocker in a GSD session by sending a response to the UI request.', {
|
|
213
|
+
sessionId: z.string().describe('Session ID returned from gsd_execute'),
|
|
214
|
+
response: z.string().describe('Response to send for the pending blocker'),
|
|
215
|
+
}, async (args) => {
|
|
216
|
+
const { sessionId, response } = args;
|
|
217
|
+
try {
|
|
218
|
+
await sessionManager.resolveBlocker(sessionId, response);
|
|
219
|
+
return jsonContent({ resolved: true });
|
|
220
|
+
}
|
|
221
|
+
catch (err) {
|
|
222
|
+
return errorContent(err instanceof Error ? err.message : String(err));
|
|
223
|
+
}
|
|
224
|
+
});
|
|
225
|
+
// =======================================================================
|
|
226
|
+
// READ-ONLY TOOLS — no session required, pure filesystem reads
|
|
227
|
+
// =======================================================================
|
|
228
|
+
// -----------------------------------------------------------------------
|
|
229
|
+
// gsd_progress — structured project progress metrics
|
|
230
|
+
// -----------------------------------------------------------------------
|
|
231
|
+
server.tool('gsd_progress', 'Get structured project progress: active milestone/slice/task, phase, completion counts, blockers, and next action. No session required — reads directly from .gsd/ on disk.', {
|
|
232
|
+
projectDir: z.string().describe('Absolute path to the project directory'),
|
|
233
|
+
}, async (args) => {
|
|
234
|
+
const { projectDir } = args;
|
|
235
|
+
try {
|
|
236
|
+
return jsonContent(readProgress(projectDir));
|
|
237
|
+
}
|
|
238
|
+
catch (err) {
|
|
239
|
+
return errorContent(err instanceof Error ? err.message : String(err));
|
|
240
|
+
}
|
|
241
|
+
});
|
|
242
|
+
// -----------------------------------------------------------------------
|
|
243
|
+
// gsd_roadmap — milestone/slice/task structure with status
|
|
244
|
+
// -----------------------------------------------------------------------
|
|
245
|
+
server.tool('gsd_roadmap', 'Get the full project roadmap structure: milestones with their slices, tasks, status, risk, and dependencies. Optionally filter to a single milestone. No session required.', {
|
|
246
|
+
projectDir: z.string().describe('Absolute path to the project directory'),
|
|
247
|
+
milestoneId: z.string().optional().describe('Filter to a specific milestone (e.g. "M001")'),
|
|
248
|
+
}, async (args) => {
|
|
249
|
+
const { projectDir, milestoneId } = args;
|
|
250
|
+
try {
|
|
251
|
+
return jsonContent(readRoadmap(projectDir, milestoneId));
|
|
252
|
+
}
|
|
253
|
+
catch (err) {
|
|
254
|
+
return errorContent(err instanceof Error ? err.message : String(err));
|
|
255
|
+
}
|
|
256
|
+
});
|
|
257
|
+
// -----------------------------------------------------------------------
|
|
258
|
+
// gsd_history — execution history with cost/token metrics
|
|
259
|
+
// -----------------------------------------------------------------------
|
|
260
|
+
server.tool('gsd_history', 'Get execution history with cost, token usage, model, and duration per unit. Returns totals across all units. No session required.', {
|
|
261
|
+
projectDir: z.string().describe('Absolute path to the project directory'),
|
|
262
|
+
limit: z.number().optional().describe('Max entries to return (most recent first). Default: all.'),
|
|
263
|
+
}, async (args) => {
|
|
264
|
+
const { projectDir, limit } = args;
|
|
265
|
+
try {
|
|
266
|
+
return jsonContent(readHistory(projectDir, limit));
|
|
267
|
+
}
|
|
268
|
+
catch (err) {
|
|
269
|
+
return errorContent(err instanceof Error ? err.message : String(err));
|
|
270
|
+
}
|
|
271
|
+
});
|
|
272
|
+
// -----------------------------------------------------------------------
|
|
273
|
+
// gsd_doctor — lightweight structural health check
|
|
274
|
+
// -----------------------------------------------------------------------
|
|
275
|
+
server.tool('gsd_doctor', 'Run a lightweight structural health check on the .gsd/ directory. Checks for missing files, status inconsistencies, and orphaned state. No session required.', {
|
|
276
|
+
projectDir: z.string().describe('Absolute path to the project directory'),
|
|
277
|
+
scope: z.string().optional().describe('Limit checks to a specific milestone (e.g. "M001")'),
|
|
278
|
+
}, async (args) => {
|
|
279
|
+
const { projectDir, scope } = args;
|
|
280
|
+
try {
|
|
281
|
+
return jsonContent(runDoctorLite(projectDir, scope));
|
|
282
|
+
}
|
|
283
|
+
catch (err) {
|
|
284
|
+
return errorContent(err instanceof Error ? err.message : String(err));
|
|
285
|
+
}
|
|
286
|
+
});
|
|
287
|
+
// -----------------------------------------------------------------------
|
|
288
|
+
// gsd_captures — pending captures and ideas
|
|
289
|
+
// -----------------------------------------------------------------------
|
|
290
|
+
server.tool('gsd_captures', 'Get captured ideas and thoughts from CAPTURES.md with triage status. Filter by pending, actionable, or all. No session required.', {
|
|
291
|
+
projectDir: z.string().describe('Absolute path to the project directory'),
|
|
292
|
+
filter: z.enum(['all', 'pending', 'actionable']).optional().describe('Filter captures (default: "all")'),
|
|
293
|
+
}, async (args) => {
|
|
294
|
+
const { projectDir, filter } = args;
|
|
295
|
+
try {
|
|
296
|
+
return jsonContent(readCaptures(projectDir, filter ?? 'all'));
|
|
297
|
+
}
|
|
298
|
+
catch (err) {
|
|
299
|
+
return errorContent(err instanceof Error ? err.message : String(err));
|
|
300
|
+
}
|
|
301
|
+
});
|
|
302
|
+
// -----------------------------------------------------------------------
|
|
303
|
+
// gsd_knowledge — project knowledge base
|
|
304
|
+
// -----------------------------------------------------------------------
|
|
305
|
+
server.tool('gsd_knowledge', 'Get the project knowledge base: rules, patterns, and lessons learned accumulated during development. No session required.', {
|
|
306
|
+
projectDir: z.string().describe('Absolute path to the project directory'),
|
|
307
|
+
}, async (args) => {
|
|
308
|
+
const { projectDir } = args;
|
|
309
|
+
try {
|
|
310
|
+
return jsonContent(readKnowledge(projectDir));
|
|
311
|
+
}
|
|
312
|
+
catch (err) {
|
|
313
|
+
return errorContent(err instanceof Error ? err.message : String(err));
|
|
314
|
+
}
|
|
315
|
+
});
|
|
316
|
+
registerWorkflowTools(server);
|
|
317
|
+
return { server };
|
|
318
|
+
}
|
|
319
|
+
//# sourceMappingURL=server.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"server.js","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;AAEH,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,kBAAkB,CAAC;AAC3D,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAExB,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;AAClD,OAAO,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AACnD,OAAO,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AACnD,OAAO,EAAE,YAAY,EAAE,MAAM,uBAAuB,CAAC;AACrD,OAAO,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AACvD,OAAO,EAAE,aAAa,EAAE,MAAM,0BAA0B,CAAC;AACzD,OAAO,EAAE,qBAAqB,EAAE,MAAM,qBAAqB,CAAC;AAE5D,8EAA8E;AAC9E,YAAY;AACZ,8EAA8E;AAE9E,MAAM,OAAO,GAAG,2BAA2B,CAAC;AAC5C,MAAM,WAAW,GAAG,KAAK,CAAC;AAC1B,MAAM,cAAc,GAAG,QAAQ,CAAC;AAEhC,8EAA8E;AAC9E,sBAAsB;AACtB,8EAA8E;AAE9E,0DAA0D;AAC1D,SAAS,WAAW,CAAC,IAAa;IAChC,OAAO,EAAE,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC;AACvF,CAAC;AAED,oCAAoC;AACpC,SAAS,YAAY,CAAC,OAAe;IACnC,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,OAAO,EAAE,CAAC,EAAE,CAAC;AAChF,CAAC;AAED,8EAA8E;AAC9E,8BAA8B;AAC9B,8EAA8E;AAE9E,KAAK,UAAU,gBAAgB,CAAC,UAAkB,EAAE,MAAc;IAChE,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,EAAE,MAAM,CAAC,CAAC;IACjD,MAAM,MAAM,GAA4B,EAAE,UAAU,EAAE,OAAO,CAAC,UAAU,CAAC,EAAE,CAAC;IAE5E,qCAAqC;IACrC,IAAI,CAAC;QACH,MAAM,CAAC,KAAK,GAAG,MAAM,QAAQ,CAAC,IAAI,CAAC,MAAM,EAAE,UAAU,CAAC,EAAE,OAAO,CAAC,CAAC;IACnE,CAAC;IAAC,MAAM,CAAC;QACP,MAAM,CAAC,KAAK,GAAG,IAAI,CAAC;IACtB,CAAC;IAED,mCAAmC;IACnC,IAAI,CAAC;QACH,MAAM,CAAC,OAAO,GAAG,MAAM,QAAQ,CAAC,IAAI,CAAC,MAAM,EAAE,YAAY,CAAC,EAAE,OAAO,CAAC,CAAC;IACvE,CAAC;IAAC,MAAM,CAAC;QACP,MAAM,CAAC,OAAO,GAAG,IAAI,CAAC;IACxB,CAAC;IAED,yCAAyC;IACzC,IAAI,CAAC;QACH,MAAM,CAAC,YAAY,GAAG,MAAM,QAAQ,CAAC,IAAI,CAAC,MAAM,EAAE,iBAAiB,CAAC,EAAE,OAAO,CAAC,CAAC;IACjF,CAAC;IAAC,MAAM,CAAC;QACP,MAAM,CAAC,YAAY,GAAG,IAAI,CAAC;IAC7B,CAAC;IAED,sCAAsC;IACtC,MAAM,aAAa,GAAG,IAAI,CAAC,MAAM,EAAE,YAAY,CAAC,CAAC;IACjD,IAAI,CAAC;QACH,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,aAAa,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC;QACtE,MAAM,UAAU,GAAoE,EAAE,CAAC;QACvF,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE,CAAC;YAC5B,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE;gBAAE,SAAS;YACnC,MAAM,IAAI,GAAG,IAAI,CAAC,aAAa,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC;YAC7C,MAAM,UAAU,GAAG,MAAM,UAAU,CAAC,IAAI,CAAC,IAAI,EAAE,GAAG,KAAK,CAAC,IAAI,aAAa,CAAC,CAAC,CAAC;YAC5E,MAAM,UAAU,GAAG,MAAM,UAAU,CAAC,IAAI,CAAC,IAAI,EAAE,GAAG,KAAK,CAAC,IAAI,aAAa,CAAC,CAAC,CAAC;YAC5E,UAAU,CAAC,IAAI,CAAC,EAAE,EAAE,EAAE,KAAK,CAAC,IAAI,EAAE,UAAU,EAAE,UAAU,EAAE,CAAC,CAAC;QAC9D,CAAC;QACD,MAAM,CAAC,UAAU,GAAG,UAAU,CAAC;IACjC,CAAC;IAAC,MAAM,CAAC;QACP,MAAM,CAAC,UAAU,GAAG,EAAE,CAAC;IACzB,CAAC;IAED,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,KAAK,UAAU,UAAU,CAAC,IAAY;IACpC,IAAI,CAAC;QACH,MAAM,IAAI,CAAC,IAAI,CAAC,CAAC;QACjB,OAAO,IAAI,CAAC;IACd,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,KAAK,CAAC;IACf,CAAC;AACH,CAAC;AAYD,8EAA8E;AAC9E,kBAAkB;AAClB,8EAA8E;AAE9E;;;;;GAKG;AACH,MAAM,CAAC,KAAK,UAAU,eAAe,CAAC,cAA8B;IAGlE,wDAAwD;IACxD,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,GAAG,OAAO,gBAAgB,CAAC,CAAC;IACxD,MAAM,SAAS,GAAG,MAAM,CAAC,SAAS,CAAC;IAEnC,MAAM,MAAM,GAAsB,IAAI,SAAS,CAC7C,EAAE,IAAI,EAAE,WAAW,EAAE,OAAO,EAAE,cAAc,EAAE,EAC9C,EAAE,YAAY,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE,EAAE,CAChC,CAAC;IAEF,0EAA0E;IAC1E,kDAAkD;IAClD,0EAA0E;IAC1E,MAAM,CAAC,IAAI,CACT,aAAa,EACb,0FAA0F,EAC1F;QACE,UAAU,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,wCAAwC,CAAC;QACzE,OAAO,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,wCAAwC,CAAC;QACjF,KAAK,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,mBAAmB,CAAC;QAC1D,IAAI,EAAE,CAAC,CAAC,OAAO,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,qCAAqC,CAAC;KAC7E,EACD,KAAK,EAAE,IAA6B,EAAE,EAAE;QACtC,MAAM,EAAE,UAAU,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,GAAG,IAE5C,CAAC;QACF,IAAI,CAAC;YACH,MAAM,SAAS,GAAG,MAAM,cAAc,CAAC,YAAY,CAAC,UAAU,EAAE,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;YAC1F,OAAO,WAAW,CAAC,EAAE,SAAS,EAAE,MAAM,EAAE,SAAS,EAAE,CAAC,CAAC;QACvD,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,OAAO,YAAY,CAAC,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;QACxE,CAAC;IACH,CAAC,CACF,CAAC;IAEF,0EAA0E;IAC1E,mCAAmC;IACnC,0EAA0E;IAC1E,MAAM,CAAC,IAAI,CACT,YAAY,EACZ,kGAAkG,EAClG;QACE,SAAS,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,sCAAsC,CAAC;KACvE,EACD,KAAK,EAAE,IAA6B,EAAE,EAAE;QACtC,MAAM,EAAE,SAAS,EAAE,GAAG,IAA6B,CAAC;QACpD,IAAI,CAAC;YACH,MAAM,OAAO,GAAG,cAAc,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC;YACrD,IAAI,CAAC,OAAO;gBAAE,OAAO,YAAY,CAAC,sBAAsB,SAAS,EAAE,CAAC,CAAC;YAErE,MAAM,UAAU,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,OAAO,CAAC,SAAS,CAAC;YAClD,MAAM,aAAa,GAAG,OAAO,CAAC,MAAM,CAAC,MAAM,CACzC,CAAC,CAAC,EAAE,EAAE,CAAE,CAA6B,CAAC,IAAI,KAAK,UAAU;gBACjD,CAA6B,CAAC,IAAI,KAAK,sBAAsB,CACtE,CAAC,MAAM,CAAC;YAET,OAAO,WAAW,CAAC;gBACjB,MAAM,EAAE,OAAO,CAAC,MAAM;gBACtB,QAAQ,EAAE;oBACR,UAAU,EAAE,OAAO,CAAC,MAAM,CAAC,MAAM;oBACjC,SAAS,EAAE,aAAa;iBACzB;gBACD,YAAY,EAAE,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC;gBACvC,cAAc,EAAE,OAAO,CAAC,cAAc;oBACpC,CAAC,CAAC;wBACE,EAAE,EAAE,OAAO,CAAC,cAAc,CAAC,EAAE;wBAC7B,MAAM,EAAE,OAAO,CAAC,cAAc,CAAC,MAAM;wBACrC,OAAO,EAAE,OAAO,CAAC,cAAc,CAAC,OAAO;qBACxC;oBACH,CAAC,CAAC,IAAI;gBACR,IAAI,EAAE,OAAO,CAAC,IAAI;gBAClB,UAAU;aACX,CAAC,CAAC;QACL,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,OAAO,YAAY,CAAC,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;QACxE,CAAC;IACH,CAAC,CACF,CAAC;IAEF,0EAA0E;IAC1E,8CAA8C;IAC9C,0EAA0E;IAC1E,MAAM,CAAC,IAAI,CACT,YAAY,EACZ,2FAA2F,EAC3F;QACE,SAAS,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,sCAAsC,CAAC;KACvE,EACD,KAAK,EAAE,IAA6B,EAAE,EAAE;QACtC,MAAM,EAAE,SAAS,EAAE,GAAG,IAA6B,CAAC;QACpD,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,cAAc,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;YACnD,OAAO,WAAW,CAAC,MAAM,CAAC,CAAC;QAC7B,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,OAAO,YAAY,CAAC,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;QACxE,CAAC;IACH,CAAC,CACF,CAAC;IAEF,0EAA0E;IAC1E,wCAAwC;IACxC,0EAA0E;IAC1E,MAAM,CAAC,IAAI,CACT,YAAY,EACZ,mFAAmF,EACnF;QACE,SAAS,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,sCAAsC,CAAC;KACvE,EACD,KAAK,EAAE,IAA6B,EAAE,EAAE;QACtC,MAAM,EAAE,SAAS,EAAE,GAAG,IAA6B,CAAC;QACpD,IAAI,CAAC;YACH,MAAM,cAAc,CAAC,aAAa,CAAC,SAAS,CAAC,CAAC;YAC9C,OAAO,WAAW,CAAC,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAC1C,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,OAAO,YAAY,CAAC,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;QACxE,CAAC;IACH,CAAC,CACF,CAAC;IAEF,0EAA0E;IAC1E,qEAAqE;IACrE,0EAA0E;IAC1E,MAAM,CAAC,IAAI,CACT,WAAW,EACX,qJAAqJ,EACrJ;QACE,UAAU,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,wCAAwC,CAAC;QACzE,KAAK,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,6DAA6D,CAAC;KAC1F,EACD,KAAK,EAAE,IAA6B,EAAE,EAAE;QACtC,MAAM,EAAE,UAAU,EAAE,KAAK,EAAE,GAAG,IAA6C,CAAC;QAC5E,IAAI,CAAC;YACH,MAAM,KAAK,GAAG,MAAM,gBAAgB,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC;YACxD,OAAO,WAAW,CAAC,KAAK,CAAC,CAAC;QAC5B,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,OAAO,YAAY,CAAC,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;QACxE,CAAC;IACH,CAAC,CACF,CAAC;IAEF,0EAA0E;IAC1E,kDAAkD;IAClD,0EAA0E;IAC1E,MAAM,CAAC,IAAI,CACT,qBAAqB,EACrB,qFAAqF,EACrF;QACE,SAAS,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,sCAAsC,CAAC;QACtE,QAAQ,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,0CAA0C,CAAC;KAC1E,EACD,KAAK,EAAE,IAA6B,EAAE,EAAE;QACtC,MAAM,EAAE,SAAS,EAAE,QAAQ,EAAE,GAAG,IAA+C,CAAC;QAChF,IAAI,CAAC;YACH,MAAM,cAAc,CAAC,cAAc,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC;YACzD,OAAO,WAAW,CAAC,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC;QACzC,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,OAAO,YAAY,CAAC,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;QACxE,CAAC;IACH,CAAC,CACF,CAAC;IAEF,0EAA0E;IAC1E,+DAA+D;IAC/D,0EAA0E;IAE1E,0EAA0E;IAC1E,qDAAqD;IACrD,0EAA0E;IAC1E,MAAM,CAAC,IAAI,CACT,cAAc,EACd,6KAA6K,EAC7K;QACE,UAAU,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,wCAAwC,CAAC;KAC1E,EACD,KAAK,EAAE,IAA6B,EAAE,EAAE;QACtC,MAAM,EAAE,UAAU,EAAE,GAAG,IAA8B,CAAC;QACtD,IAAI,CAAC;YACH,OAAO,WAAW,CAAC,YAAY,CAAC,UAAU,CAAC,CAAC,CAAC;QAC/C,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,OAAO,YAAY,CAAC,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;QACxE,CAAC;IACH,CAAC,CACF,CAAC;IAEF,0EAA0E;IAC1E,2DAA2D;IAC3D,0EAA0E;IAC1E,MAAM,CAAC,IAAI,CACT,aAAa,EACb,4KAA4K,EAC5K;QACE,UAAU,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,wCAAwC,CAAC;QACzE,WAAW,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,8CAA8C,CAAC;KAC5F,EACD,KAAK,EAAE,IAA6B,EAAE,EAAE;QACtC,MAAM,EAAE,UAAU,EAAE,WAAW,EAAE,GAAG,IAAoD,CAAC;QACzF,IAAI,CAAC;YACH,OAAO,WAAW,CAAC,WAAW,CAAC,UAAU,EAAE,WAAW,CAAC,CAAC,CAAC;QAC3D,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,OAAO,YAAY,CAAC,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;QACxE,CAAC;IACH,CAAC,CACF,CAAC;IAEF,0EAA0E;IAC1E,0DAA0D;IAC1D,0EAA0E;IAC1E,MAAM,CAAC,IAAI,CACT,aAAa,EACb,mIAAmI,EACnI;QACE,UAAU,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,wCAAwC,CAAC;QACzE,KAAK,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,0DAA0D,CAAC;KAClG,EACD,KAAK,EAAE,IAA6B,EAAE,EAAE;QACtC,MAAM,EAAE,UAAU,EAAE,KAAK,EAAE,GAAG,IAA8C,CAAC;QAC7E,IAAI,CAAC;YACH,OAAO,WAAW,CAAC,WAAW,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC,CAAC;QACrD,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,OAAO,YAAY,CAAC,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;QACxE,CAAC;IACH,CAAC,CACF,CAAC;IAEF,0EAA0E;IAC1E,mDAAmD;IACnD,0EAA0E;IAC1E,MAAM,CAAC,IAAI,CACT,YAAY,EACZ,8JAA8J,EAC9J;QACE,UAAU,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,wCAAwC,CAAC;QACzE,KAAK,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,oDAAoD,CAAC;KAC5F,EACD,KAAK,EAAE,IAA6B,EAAE,EAAE;QACtC,MAAM,EAAE,UAAU,EAAE,KAAK,EAAE,GAAG,IAA8C,CAAC;QAC7E,IAAI,CAAC;YACH,OAAO,WAAW,CAAC,aAAa,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC,CAAC;QACvD,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,OAAO,YAAY,CAAC,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;QACxE,CAAC;IACH,CAAC,CACF,CAAC;IAEF,0EAA0E;IAC1E,4CAA4C;IAC5C,0EAA0E;IAC1E,MAAM,CAAC,IAAI,CACT,cAAc,EACd,kIAAkI,EAClI;QACE,UAAU,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,wCAAwC,CAAC;QACzE,MAAM,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,KAAK,EAAE,SAAS,EAAE,YAAY,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,kCAAkC,CAAC;KACzG,EACD,KAAK,EAAE,IAA6B,EAAE,EAAE;QACtC,MAAM,EAAE,UAAU,EAAE,MAAM,EAAE,GAAG,IAAyE,CAAC;QACzG,IAAI,CAAC;YACH,OAAO,WAAW,CAAC,YAAY,CAAC,UAAU,EAAE,MAAM,IAAI,KAAK,CAAC,CAAC,CAAC;QAChE,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,OAAO,YAAY,CAAC,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;QACxE,CAAC;IACH,CAAC,CACF,CAAC;IAEF,0EAA0E;IAC1E,yCAAyC;IACzC,0EAA0E;IAC1E,MAAM,CAAC,IAAI,CACT,eAAe,EACf,2HAA2H,EAC3H;QACE,UAAU,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,wCAAwC,CAAC;KAC1E,EACD,KAAK,EAAE,IAA6B,EAAE,EAAE;QACtC,MAAM,EAAE,UAAU,EAAE,GAAG,IAA8B,CAAC;QACtD,IAAI,CAAC;YACH,OAAO,WAAW,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC,CAAC;QAChD,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,OAAO,YAAY,CAAC,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;QACxE,CAAC;IACH,CAAC,CACF,CAAC;IAEF,qBAAqB,CAAC,MAAM,CAAC,CAAC;IAE9B,OAAO,EAAE,MAAM,EAAE,CAAC;AACpB,CAAC","sourcesContent":["/**\n * MCP Server — registers GSD orchestration, project-state, and workflow tools.\n *\n * Session tools (6): gsd_execute, gsd_status, gsd_result, gsd_cancel, gsd_query, gsd_resolve_blocker\n * Read-only tools (6): gsd_progress, gsd_roadmap, gsd_history, gsd_doctor, gsd_captures, gsd_knowledge\n * Workflow tools (17): planning, replanning, completion, validation, reassessment, gate result, and milestone status tools\n *\n * Uses dynamic imports for @modelcontextprotocol/sdk because TS Node16\n * cannot resolve the SDK's subpath exports statically (same pattern as\n * src/mcp-server.ts in the main package).\n */\n\nimport { readFile, readdir, stat } from 'node:fs/promises';\nimport { join, resolve } from 'node:path';\nimport { z } from 'zod';\nimport type { SessionManager } from './session-manager.js';\nimport { readProgress } from './readers/state.js';\nimport { readRoadmap } from './readers/roadmap.js';\nimport { readHistory } from './readers/metrics.js';\nimport { readCaptures } from './readers/captures.js';\nimport { readKnowledge } from './readers/knowledge.js';\nimport { runDoctorLite } from './readers/doctor-lite.js';\nimport { registerWorkflowTools } from './workflow-tools.js';\n\n// ---------------------------------------------------------------------------\n// Constants\n// ---------------------------------------------------------------------------\n\nconst MCP_PKG = '@modelcontextprotocol/sdk';\nconst SERVER_NAME = 'gsd';\nconst SERVER_VERSION = '2.53.0';\n\n// ---------------------------------------------------------------------------\n// Tool result helpers\n// ---------------------------------------------------------------------------\n\n/** Wrap a JSON-serializable value as MCP tool content. */\nfunction jsonContent(data: unknown): { content: Array<{ type: 'text'; text: string }> } {\n return { content: [{ type: 'text' as const, text: JSON.stringify(data, null, 2) }] };\n}\n\n/** Return an MCP error response. */\nfunction errorContent(message: string): { isError: true; content: Array<{ type: 'text'; text: string }> } {\n return { isError: true, content: [{ type: 'text' as const, text: message }] };\n}\n\n// ---------------------------------------------------------------------------\n// gsd_query filesystem reader\n// ---------------------------------------------------------------------------\n\nasync function readProjectState(projectDir: string, _query: string): Promise<Record<string, unknown>> {\n const gsdDir = join(resolve(projectDir), '.gsd');\n const result: Record<string, unknown> = { projectDir: resolve(projectDir) };\n\n // STATE.md — current execution state\n try {\n result.state = await readFile(join(gsdDir, 'STATE.md'), 'utf-8');\n } catch {\n result.state = null;\n }\n\n // PROJECT.md — project description\n try {\n result.project = await readFile(join(gsdDir, 'PROJECT.md'), 'utf-8');\n } catch {\n result.project = null;\n }\n\n // REQUIREMENTS.md — requirement contract\n try {\n result.requirements = await readFile(join(gsdDir, 'REQUIREMENTS.md'), 'utf-8');\n } catch {\n result.requirements = null;\n }\n\n // List milestones with basic metadata\n const milestonesDir = join(gsdDir, 'milestones');\n try {\n const entries = await readdir(milestonesDir, { withFileTypes: true });\n const milestones: Array<{ id: string; hasRoadmap: boolean; hasSummary: boolean }> = [];\n for (const entry of entries) {\n if (!entry.isDirectory()) continue;\n const mDir = join(milestonesDir, entry.name);\n const hasRoadmap = await fileExists(join(mDir, `${entry.name}-ROADMAP.md`));\n const hasSummary = await fileExists(join(mDir, `${entry.name}-SUMMARY.md`));\n milestones.push({ id: entry.name, hasRoadmap, hasSummary });\n }\n result.milestones = milestones;\n } catch {\n result.milestones = [];\n }\n\n return result;\n}\n\nasync function fileExists(path: string): Promise<boolean> {\n try {\n await stat(path);\n return true;\n } catch {\n return false;\n }\n}\n\n// ---------------------------------------------------------------------------\n// MCP Server type — minimal interface for the dynamically-imported McpServer\n// ---------------------------------------------------------------------------\n\ninterface McpServerInstance {\n tool(name: string, description: string, params: Record<string, unknown>, handler: (args: Record<string, unknown>) => Promise<unknown>): unknown;\n connect(transport: unknown): Promise<void>;\n close(): Promise<void>;\n}\n\n// ---------------------------------------------------------------------------\n// createMcpServer\n// ---------------------------------------------------------------------------\n\n/**\n * Create and configure an MCP server with session, read-only, and workflow tools.\n *\n * Returns the McpServer instance — call `connect(transport)` to start serving.\n * Uses dynamic imports for the MCP SDK to avoid TS subpath resolution issues.\n */\nexport async function createMcpServer(sessionManager: SessionManager): Promise<{\n server: McpServerInstance;\n}> {\n // Dynamic import — same workaround as src/mcp-server.ts\n const mcpMod = await import(`${MCP_PKG}/server/mcp.js`);\n const McpServer = mcpMod.McpServer;\n\n const server: McpServerInstance = new McpServer(\n { name: SERVER_NAME, version: SERVER_VERSION },\n { capabilities: { tools: {} } },\n );\n\n // -----------------------------------------------------------------------\n // gsd_execute — start a new GSD auto-mode session\n // -----------------------------------------------------------------------\n server.tool(\n 'gsd_execute',\n 'Start a GSD auto-mode session for a project directory. Returns a sessionId for tracking.',\n {\n projectDir: z.string().describe('Absolute path to the project directory'),\n command: z.string().optional().describe('Command to send (default: \"/gsd auto\")'),\n model: z.string().optional().describe('Model ID override'),\n bare: z.boolean().optional().describe('Run in bare mode (skip user config)'),\n },\n async (args: Record<string, unknown>) => {\n const { projectDir, command, model, bare } = args as {\n projectDir: string; command?: string; model?: string; bare?: boolean;\n };\n try {\n const sessionId = await sessionManager.startSession(projectDir, { command, model, bare });\n return jsonContent({ sessionId, status: 'started' });\n } catch (err) {\n return errorContent(err instanceof Error ? err.message : String(err));\n }\n },\n );\n\n // -----------------------------------------------------------------------\n // gsd_status — poll session status\n // -----------------------------------------------------------------------\n server.tool(\n 'gsd_status',\n 'Get the current status of a GSD session including progress, recent events, and pending blockers.',\n {\n sessionId: z.string().describe('Session ID returned from gsd_execute'),\n },\n async (args: Record<string, unknown>) => {\n const { sessionId } = args as { sessionId: string };\n try {\n const session = sessionManager.getSession(sessionId);\n if (!session) return errorContent(`Session not found: ${sessionId}`);\n\n const durationMs = Date.now() - session.startTime;\n const toolCallCount = session.events.filter(\n (e) => (e as Record<string, unknown>).type === 'tool_use' ||\n (e as Record<string, unknown>).type === 'tool_execution_start'\n ).length;\n\n return jsonContent({\n status: session.status,\n progress: {\n eventCount: session.events.length,\n toolCalls: toolCallCount,\n },\n recentEvents: session.events.slice(-10),\n pendingBlocker: session.pendingBlocker\n ? {\n id: session.pendingBlocker.id,\n method: session.pendingBlocker.method,\n message: session.pendingBlocker.message,\n }\n : null,\n cost: session.cost,\n durationMs,\n });\n } catch (err) {\n return errorContent(err instanceof Error ? err.message : String(err));\n }\n },\n );\n\n // -----------------------------------------------------------------------\n // gsd_result — get accumulated session result\n // -----------------------------------------------------------------------\n server.tool(\n 'gsd_result',\n 'Get the result of a GSD session. Returns partial results if the session is still running.',\n {\n sessionId: z.string().describe('Session ID returned from gsd_execute'),\n },\n async (args: Record<string, unknown>) => {\n const { sessionId } = args as { sessionId: string };\n try {\n const result = sessionManager.getResult(sessionId);\n return jsonContent(result);\n } catch (err) {\n return errorContent(err instanceof Error ? err.message : String(err));\n }\n },\n );\n\n // -----------------------------------------------------------------------\n // gsd_cancel — cancel a running session\n // -----------------------------------------------------------------------\n server.tool(\n 'gsd_cancel',\n 'Cancel a running GSD session. Aborts the current operation and stops the process.',\n {\n sessionId: z.string().describe('Session ID returned from gsd_execute'),\n },\n async (args: Record<string, unknown>) => {\n const { sessionId } = args as { sessionId: string };\n try {\n await sessionManager.cancelSession(sessionId);\n return jsonContent({ cancelled: true });\n } catch (err) {\n return errorContent(err instanceof Error ? err.message : String(err));\n }\n },\n );\n\n // -----------------------------------------------------------------------\n // gsd_query — read project state from filesystem (no session needed)\n // -----------------------------------------------------------------------\n server.tool(\n 'gsd_query',\n 'Query GSD project state from the filesystem. Returns STATE.md, PROJECT.md, requirements, and milestone listing. Does not require an active session.',\n {\n projectDir: z.string().describe('Absolute path to the project directory'),\n query: z.string().describe('What to query (e.g. \"status\", \"milestones\", \"requirements\")'),\n },\n async (args: Record<string, unknown>) => {\n const { projectDir, query } = args as { projectDir: string; query: string };\n try {\n const state = await readProjectState(projectDir, query);\n return jsonContent(state);\n } catch (err) {\n return errorContent(err instanceof Error ? err.message : String(err));\n }\n },\n );\n\n // -----------------------------------------------------------------------\n // gsd_resolve_blocker — resolve a pending blocker\n // -----------------------------------------------------------------------\n server.tool(\n 'gsd_resolve_blocker',\n 'Resolve a pending blocker in a GSD session by sending a response to the UI request.',\n {\n sessionId: z.string().describe('Session ID returned from gsd_execute'),\n response: z.string().describe('Response to send for the pending blocker'),\n },\n async (args: Record<string, unknown>) => {\n const { sessionId, response } = args as { sessionId: string; response: string };\n try {\n await sessionManager.resolveBlocker(sessionId, response);\n return jsonContent({ resolved: true });\n } catch (err) {\n return errorContent(err instanceof Error ? err.message : String(err));\n }\n },\n );\n\n // =======================================================================\n // READ-ONLY TOOLS — no session required, pure filesystem reads\n // =======================================================================\n\n // -----------------------------------------------------------------------\n // gsd_progress — structured project progress metrics\n // -----------------------------------------------------------------------\n server.tool(\n 'gsd_progress',\n 'Get structured project progress: active milestone/slice/task, phase, completion counts, blockers, and next action. No session required — reads directly from .gsd/ on disk.',\n {\n projectDir: z.string().describe('Absolute path to the project directory'),\n },\n async (args: Record<string, unknown>) => {\n const { projectDir } = args as { projectDir: string };\n try {\n return jsonContent(readProgress(projectDir));\n } catch (err) {\n return errorContent(err instanceof Error ? err.message : String(err));\n }\n },\n );\n\n // -----------------------------------------------------------------------\n // gsd_roadmap — milestone/slice/task structure with status\n // -----------------------------------------------------------------------\n server.tool(\n 'gsd_roadmap',\n 'Get the full project roadmap structure: milestones with their slices, tasks, status, risk, and dependencies. Optionally filter to a single milestone. No session required.',\n {\n projectDir: z.string().describe('Absolute path to the project directory'),\n milestoneId: z.string().optional().describe('Filter to a specific milestone (e.g. \"M001\")'),\n },\n async (args: Record<string, unknown>) => {\n const { projectDir, milestoneId } = args as { projectDir: string; milestoneId?: string };\n try {\n return jsonContent(readRoadmap(projectDir, milestoneId));\n } catch (err) {\n return errorContent(err instanceof Error ? err.message : String(err));\n }\n },\n );\n\n // -----------------------------------------------------------------------\n // gsd_history — execution history with cost/token metrics\n // -----------------------------------------------------------------------\n server.tool(\n 'gsd_history',\n 'Get execution history with cost, token usage, model, and duration per unit. Returns totals across all units. No session required.',\n {\n projectDir: z.string().describe('Absolute path to the project directory'),\n limit: z.number().optional().describe('Max entries to return (most recent first). Default: all.'),\n },\n async (args: Record<string, unknown>) => {\n const { projectDir, limit } = args as { projectDir: string; limit?: number };\n try {\n return jsonContent(readHistory(projectDir, limit));\n } catch (err) {\n return errorContent(err instanceof Error ? err.message : String(err));\n }\n },\n );\n\n // -----------------------------------------------------------------------\n // gsd_doctor — lightweight structural health check\n // -----------------------------------------------------------------------\n server.tool(\n 'gsd_doctor',\n 'Run a lightweight structural health check on the .gsd/ directory. Checks for missing files, status inconsistencies, and orphaned state. No session required.',\n {\n projectDir: z.string().describe('Absolute path to the project directory'),\n scope: z.string().optional().describe('Limit checks to a specific milestone (e.g. \"M001\")'),\n },\n async (args: Record<string, unknown>) => {\n const { projectDir, scope } = args as { projectDir: string; scope?: string };\n try {\n return jsonContent(runDoctorLite(projectDir, scope));\n } catch (err) {\n return errorContent(err instanceof Error ? err.message : String(err));\n }\n },\n );\n\n // -----------------------------------------------------------------------\n // gsd_captures — pending captures and ideas\n // -----------------------------------------------------------------------\n server.tool(\n 'gsd_captures',\n 'Get captured ideas and thoughts from CAPTURES.md with triage status. Filter by pending, actionable, or all. No session required.',\n {\n projectDir: z.string().describe('Absolute path to the project directory'),\n filter: z.enum(['all', 'pending', 'actionable']).optional().describe('Filter captures (default: \"all\")'),\n },\n async (args: Record<string, unknown>) => {\n const { projectDir, filter } = args as { projectDir: string; filter?: 'all' | 'pending' | 'actionable' };\n try {\n return jsonContent(readCaptures(projectDir, filter ?? 'all'));\n } catch (err) {\n return errorContent(err instanceof Error ? err.message : String(err));\n }\n },\n );\n\n // -----------------------------------------------------------------------\n // gsd_knowledge — project knowledge base\n // -----------------------------------------------------------------------\n server.tool(\n 'gsd_knowledge',\n 'Get the project knowledge base: rules, patterns, and lessons learned accumulated during development. No session required.',\n {\n projectDir: z.string().describe('Absolute path to the project directory'),\n },\n async (args: Record<string, unknown>) => {\n const { projectDir } = args as { projectDir: string };\n try {\n return jsonContent(readKnowledge(projectDir));\n } catch (err) {\n return errorContent(err instanceof Error ? err.message : String(err));\n }\n },\n );\n\n registerWorkflowTools(server);\n\n return { server };\n}\n"]}
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* SessionManager — manages RpcClient lifecycle for background GSD execution.
|
|
3
|
+
*
|
|
4
|
+
* One active session per projectDir. Tracks events in a ring buffer,
|
|
5
|
+
* detects blockers, tracks terminal state, and accumulates cost using
|
|
6
|
+
* the cumulative-max pattern (K004).
|
|
7
|
+
*/
|
|
8
|
+
import type { ManagedSession, ExecuteOptions } from './types.js';
|
|
9
|
+
export declare class SessionManager {
|
|
10
|
+
/** Sessions keyed by projectDir for duplicate-start prevention */
|
|
11
|
+
private sessions;
|
|
12
|
+
/**
|
|
13
|
+
* Start a new GSD auto-mode session for the given project directory.
|
|
14
|
+
*
|
|
15
|
+
* Rejects if a session already exists for this projectDir.
|
|
16
|
+
* Creates an RpcClient, starts the process, performs the v2 init handshake,
|
|
17
|
+
* wires event tracking, and sends '/gsd auto' to begin execution.
|
|
18
|
+
*/
|
|
19
|
+
startSession(projectDir: string, options?: ExecuteOptions): Promise<string>;
|
|
20
|
+
/**
|
|
21
|
+
* Look up a session by sessionId.
|
|
22
|
+
* Linear scan is fine — we expect <10 concurrent sessions.
|
|
23
|
+
*/
|
|
24
|
+
getSession(sessionId: string): ManagedSession | undefined;
|
|
25
|
+
/**
|
|
26
|
+
* Look up a session by project directory (direct map lookup).
|
|
27
|
+
*/
|
|
28
|
+
getSessionByDir(projectDir: string): ManagedSession | undefined;
|
|
29
|
+
/**
|
|
30
|
+
* Resolve a pending blocker by sending a UI response.
|
|
31
|
+
*/
|
|
32
|
+
resolveBlocker(sessionId: string, response: string): Promise<void>;
|
|
33
|
+
/**
|
|
34
|
+
* Cancel a running session — abort current operation then stop the process.
|
|
35
|
+
*/
|
|
36
|
+
cancelSession(sessionId: string): Promise<void>;
|
|
37
|
+
/**
|
|
38
|
+
* Build a HeadlessJsonResult-shaped object from accumulated session state.
|
|
39
|
+
*/
|
|
40
|
+
getResult(sessionId: string): Record<string, unknown>;
|
|
41
|
+
/**
|
|
42
|
+
* Stop all active sessions and clean up resources.
|
|
43
|
+
*/
|
|
44
|
+
cleanup(): Promise<void>;
|
|
45
|
+
/**
|
|
46
|
+
* Resolve the GSD CLI path.
|
|
47
|
+
*
|
|
48
|
+
* 1. GSD_CLI_PATH env var (highest priority)
|
|
49
|
+
* 2. `which gsd` → resolve to the actual dist/cli.js
|
|
50
|
+
*/
|
|
51
|
+
static resolveCLIPath(): string;
|
|
52
|
+
private handleEvent;
|
|
53
|
+
}
|
|
54
|
+
//# sourceMappingURL=session-manager.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"session-manager.d.ts","sourceRoot":"","sources":["../src/session-manager.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAMH,OAAO,KAAK,EACV,cAAc,EACd,cAAc,EAIf,MAAM,YAAY,CAAC;AAmCpB,qBAAa,cAAc;IACzB,kEAAkE;IAClE,OAAO,CAAC,QAAQ,CAAqC;IAErD;;;;;;OAMG;IACG,YAAY,CAAC,UAAU,EAAE,MAAM,EAAE,OAAO,GAAE,cAAmB,GAAG,OAAO,CAAC,MAAM,CAAC;IA+ErF;;;OAGG;IACH,UAAU,CAAC,SAAS,EAAE,MAAM,GAAG,cAAc,GAAG,SAAS;IAOzD;;OAEG;IACH,eAAe,CAAC,UAAU,EAAE,MAAM,GAAG,cAAc,GAAG,SAAS;IAI/D;;OAEG;IACG,cAAc,CAAC,SAAS,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAaxE;;OAEG;IACG,aAAa,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAgBrD;;OAEG;IACH,SAAS,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC;IAoBrD;;OAEG;IACG,OAAO,IAAI,OAAO,CAAC,IAAI,CAAC;IAgB9B;;;;;OAKG;IACH,MAAM,CAAC,cAAc,IAAI,MAAM;IAyB/B,OAAO,CAAC,WAAW;CAsCpB"}
|