kantban-cli 0.1.46 → 0.1.49
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/{chunk-DAFLEMLK.js → chunk-4R27WTCJ.js} +31 -17
- package/dist/chunk-4R27WTCJ.js.map +1 -0
- package/dist/{chunk-SDMNXFPV.js → chunk-DENXSVKE.js} +9 -3
- package/dist/chunk-DENXSVKE.js.map +1 -0
- package/dist/{chunk-N4ZHMJD7.js → chunk-QHJZIGEE.js} +146 -7
- package/dist/chunk-QHJZIGEE.js.map +1 -0
- package/dist/{cron-QUEYUVIX.js → cron-F6D6475M.js} +3 -3
- package/dist/index.js +3 -3
- package/dist/lib/gate-proxy-server.js +38 -18
- package/dist/lib/gate-proxy-server.js.map +1 -1
- package/dist/{pipeline-IT7LJXKA.js → pipeline-GZOSDNPF.js} +245 -52
- package/dist/pipeline-GZOSDNPF.js.map +1 -0
- package/dist/{pipeline-init-IGZZOOLK.js → pipeline-init-AUKPFJYE.js} +5 -2
- package/dist/pipeline-init-AUKPFJYE.js.map +1 -0
- package/package.json +1 -1
- package/dist/chunk-DAFLEMLK.js.map +0 -1
- package/dist/chunk-N4ZHMJD7.js.map +0 -1
- package/dist/chunk-SDMNXFPV.js.map +0 -1
- package/dist/pipeline-IT7LJXKA.js.map +0 -1
- package/dist/pipeline-init-IGZZOOLK.js.map +0 -1
- /package/dist/{cron-QUEYUVIX.js.map → cron-F6D6475M.js.map} +0 -0
|
@@ -5,12 +5,12 @@ import {
|
|
|
5
5
|
import {
|
|
6
6
|
formatGateErrors,
|
|
7
7
|
runGates
|
|
8
|
-
} from "../chunk-
|
|
8
|
+
} from "../chunk-DENXSVKE.js";
|
|
9
9
|
import {
|
|
10
10
|
parseGateConfig,
|
|
11
11
|
parseTimeout,
|
|
12
12
|
resolveGatesForColumn
|
|
13
|
-
} from "../chunk-
|
|
13
|
+
} from "../chunk-4R27WTCJ.js";
|
|
14
14
|
import "../chunk-5ZU2OOES.js";
|
|
15
15
|
|
|
16
16
|
// src/lib/gate-proxy-server.ts
|
|
@@ -45,7 +45,7 @@ var GateProxy = class {
|
|
|
45
45
|
return opts;
|
|
46
46
|
}
|
|
47
47
|
async handleRunGates(columnName2, ticketId) {
|
|
48
|
-
const allGates = resolveGatesForColumn(this.config, columnName2);
|
|
48
|
+
const { gates: allGates } = resolveGatesForColumn(this.config, columnName2);
|
|
49
49
|
let gates = allGates;
|
|
50
50
|
if (ticketId) {
|
|
51
51
|
const waivers = await this.deps.getTicketGateWaivers(ticketId);
|
|
@@ -60,19 +60,22 @@ var GateProxy = class {
|
|
|
60
60
|
}
|
|
61
61
|
async handleMoveTicket(move) {
|
|
62
62
|
try {
|
|
63
|
-
const
|
|
64
|
-
if (
|
|
65
|
-
const
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
63
|
+
const target = await this.deps.getColumn(move.columnId);
|
|
64
|
+
if (target?.type === "done") {
|
|
65
|
+
const { source } = resolveGatesForColumn(this.config, target.name);
|
|
66
|
+
if (source === "default") {
|
|
67
|
+
const forwardResult2 = await this.deps.forwardMoveTicket({
|
|
68
|
+
...move.args,
|
|
69
|
+
projectId: move.projectId,
|
|
70
|
+
ticketId: move.ticketId,
|
|
71
|
+
column_id: move.columnId
|
|
72
|
+
});
|
|
73
|
+
return { forwardResult: forwardResult2 };
|
|
74
|
+
}
|
|
72
75
|
}
|
|
73
76
|
} catch {
|
|
74
77
|
}
|
|
75
|
-
const allGates = resolveGatesForColumn(this.config, move.currentColumnName);
|
|
78
|
+
const { gates: allGates } = resolveGatesForColumn(this.config, move.currentColumnName);
|
|
76
79
|
const waivers = await this.deps.getTicketGateWaivers(move.ticketId);
|
|
77
80
|
const waiverSet = new Set(waivers);
|
|
78
81
|
const gates = allGates.filter((g) => !waiverSet.has(g.name));
|
|
@@ -108,7 +111,7 @@ var GateProxy = class {
|
|
|
108
111
|
return { forwardResult };
|
|
109
112
|
}
|
|
110
113
|
async handleCompleteTask(complete) {
|
|
111
|
-
const allGates = resolveGatesForColumn(this.config, complete.currentColumnName);
|
|
114
|
+
const { gates: allGates } = resolveGatesForColumn(this.config, complete.currentColumnName);
|
|
112
115
|
const waivers = await this.deps.getTicketGateWaivers(complete.ticketId);
|
|
113
116
|
const waiverSet = new Set(waivers);
|
|
114
117
|
const gates = allGates.filter((g) => !waiverSet.has(g.name));
|
|
@@ -284,14 +287,15 @@ var deps = {
|
|
|
284
287
|
return { title: "", description: "" };
|
|
285
288
|
}
|
|
286
289
|
},
|
|
287
|
-
async
|
|
290
|
+
async getColumn(columnIdArg) {
|
|
288
291
|
try {
|
|
289
292
|
validateUuid(columnIdArg, "columnId");
|
|
290
293
|
const data = await apiGet(`/projects/${projectId}/columns/${columnIdArg}`);
|
|
291
|
-
|
|
294
|
+
if (!data.name || !data.type) return null;
|
|
295
|
+
return { name: data.name, type: data.type };
|
|
292
296
|
} catch (err) {
|
|
293
297
|
process.stderr.write(
|
|
294
|
-
`gate-proxy-server: column
|
|
298
|
+
`gate-proxy-server: column fetch failed for ${columnIdArg}: ${err instanceof Error ? err.message : String(err)}
|
|
295
299
|
`
|
|
296
300
|
);
|
|
297
301
|
return null;
|
|
@@ -380,6 +384,13 @@ async function handleToolCall(name, args) {
|
|
|
380
384
|
if (isError && result.results) {
|
|
381
385
|
process.stderr.write(`gate-proxy-server: move blocked
|
|
382
386
|
${formatGateErrors(result.results)}`);
|
|
387
|
+
for (const r of result.results) {
|
|
388
|
+
if (!r.passed) {
|
|
389
|
+
console.error(
|
|
390
|
+
`[gate-error] ticket=${moveTicketId} column=${columnName} gate=${r.name} exit_code=${r.exit_code} errno=${r.errno ?? ""} syscall=${r.syscall ?? ""} stderr=${JSON.stringify((r.stderrTail ?? "").slice(0, 500))}`
|
|
391
|
+
);
|
|
392
|
+
}
|
|
393
|
+
}
|
|
383
394
|
}
|
|
384
395
|
return {
|
|
385
396
|
content: [{ type: "text", text: JSON.stringify(result, null, 2) }],
|
|
@@ -403,6 +414,13 @@ ${formatGateErrors(result.results)}`);
|
|
|
403
414
|
if (isError && result.results) {
|
|
404
415
|
process.stderr.write(`gate-proxy-server: complete blocked
|
|
405
416
|
${formatGateErrors(result.results)}`);
|
|
417
|
+
for (const r of result.results) {
|
|
418
|
+
if (!r.passed) {
|
|
419
|
+
console.error(
|
|
420
|
+
`[gate-error] ticket=${completeTicketId} column=${columnName} gate=${r.name} exit_code=${r.exit_code} errno=${r.errno ?? ""} syscall=${r.syscall ?? ""} stderr=${JSON.stringify((r.stderrTail ?? "").slice(0, 500))}`
|
|
421
|
+
);
|
|
422
|
+
}
|
|
423
|
+
}
|
|
406
424
|
}
|
|
407
425
|
return {
|
|
408
426
|
content: [{ type: "text", text: JSON.stringify(result, null, 2) }],
|
|
@@ -493,7 +511,7 @@ async function handleMessage(msg) {
|
|
|
493
511
|
}
|
|
494
512
|
var rl = createInterface({ input: process.stdin });
|
|
495
513
|
var messageQueue = Promise.resolve();
|
|
496
|
-
var HANDLER_TIMEOUT_MS =
|
|
514
|
+
var HANDLER_TIMEOUT_MS = Number(process.env["GATE_PROXY_HANDLER_TIMEOUT_MS"]) || 30 * 6e4;
|
|
497
515
|
rl.on("line", (line) => {
|
|
498
516
|
const trimmed = line.trim();
|
|
499
517
|
if (!trimmed) return;
|
|
@@ -553,4 +571,6 @@ process.on("unhandledRejection", (err) => {
|
|
|
553
571
|
});
|
|
554
572
|
process.stderr.write(`gate-proxy-server: started (column="${columnName}", gates=${gateConfig.default.length})
|
|
555
573
|
`);
|
|
574
|
+
process.stderr.write(`[gate-proxy] handler timeout: ${HANDLER_TIMEOUT_MS}ms
|
|
575
|
+
`);
|
|
556
576
|
//# sourceMappingURL=gate-proxy-server.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../src/lib/gate-proxy-server.ts","../../src/lib/gate-proxy.ts"],"sourcesContent":["#!/usr/bin/env node\n\n/**\n * gate-proxy-server.ts — MCP stdio server for gate enforcement.\n *\n * Spawned by the pipeline orchestrator as a child process. Reads gate config\n * from env vars, instantiates GateProxy, and exposes three MCP tools over\n * JSON-RPC on stdin/stdout:\n *\n * - kantban_run_gates — Run column gates and report results\n * - kantban_move_ticket — Run gates then forward move to KantBan API\n * - kantban_complete_task — Run gates then forward complete to KantBan API\n *\n * Uses raw JSON-RPC (no @modelcontextprotocol/sdk dependency).\n */\n\nimport { readFileSync } from 'node:fs';\nimport { createInterface } from 'node:readline';\nimport { parseGateConfig } from './gate-config.js';\nimport { runGates, formatGateErrors } from './gate-runner.js';\nimport { GateProxy } from './gate-proxy.js';\nimport { fetchWithRetry } from '../client.js';\nimport type { GateProxyDeps } from './gate-proxy.js';\n\n// ── UUID validation ──────────────────────────────────────────────────────\n\nconst UUID_RE = /^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/i;\n\nfunction validateUuid(value: unknown, name: string): string {\n if (typeof value !== 'string' || !UUID_RE.test(value)) {\n throw new Error(`Invalid ${name}: expected UUID, got \"${String(value)}\"`);\n }\n return value;\n}\n\n// ── env ──────────────────────────────────────────────────────────────────\n\nconst GATE_CONFIG_PATH = process.env['GATE_CONFIG_PATH'];\nconst COLUMN_ID = process.env['COLUMN_ID'];\nconst COLUMN_NAME = process.env['COLUMN_NAME'];\nconst PROJECT_ID = process.env['PROJECT_ID'];\nconst API_TOKEN = process.env['KANTBAN_API_TOKEN'];\nconst API_URL = process.env['KANTBAN_API_URL'];\n\nfunction requireEnv(name: string, value: string | undefined): string {\n if (!value) {\n process.stderr.write(`gate-proxy-server: missing required env var ${name}\\n`);\n process.exit(1);\n }\n return value;\n}\n\nconst gateConfigPath = requireEnv('GATE_CONFIG_PATH', GATE_CONFIG_PATH);\nconst columnId = requireEnv('COLUMN_ID', COLUMN_ID);\nconst columnName = requireEnv('COLUMN_NAME', COLUMN_NAME);\nconst projectId = requireEnv('PROJECT_ID', PROJECT_ID);\nconst apiToken = requireEnv('KANTBAN_API_TOKEN', API_TOKEN);\nconst apiUrl = requireEnv('KANTBAN_API_URL', API_URL);\n\n// ── gate config ──────────────────────────────────────────────────────────\n\nconst yamlContent = readFileSync(gateConfigPath, 'utf-8');\nconst gateConfig = parseGateConfig(yamlContent);\n\n// GATE_CWD overrides the YAML settings.cwd — set by the orchestrator to the\n// agent's worktree path so gates run against the agent's checked-out code,\n// not the root repo.\nconst GATE_CWD = process.env['GATE_CWD'];\nif (GATE_CWD) {\n if (!gateConfig.settings) gateConfig.settings = {};\n gateConfig.settings.cwd = GATE_CWD;\n}\n\n// ── API helpers ──────────────────────────────────────────────────────────\n\nfunction apiHeaders(): Record<string, string> {\n return {\n 'Authorization': `Bearer ${apiToken}`,\n 'Content-Type': 'application/json',\n 'X-KantBan-Via': 'cli-gate-proxy',\n };\n}\n\nasync function apiPost(path: string, body: Record<string, unknown>): Promise<unknown> {\n const url = new URL(path, apiUrl);\n const res = await fetchWithRetry(url.toString(), {\n method: 'POST',\n headers: apiHeaders(),\n body: JSON.stringify(body),\n });\n if (!res.ok) {\n const text = await res.text();\n throw new Error(`API error ${res.status}: ${text}`);\n }\n const json = (await res.json()) as { success: boolean; data: unknown };\n if (!json.success) throw new Error('API responded with success: false');\n return json.data;\n}\n\nasync function apiPatch(path: string, body: Record<string, unknown>): Promise<unknown> {\n const url = new URL(path, apiUrl);\n const res = await fetchWithRetry(url.toString(), {\n method: 'PATCH',\n headers: apiHeaders(),\n body: JSON.stringify(body),\n });\n if (!res.ok) {\n const text = await res.text();\n throw new Error(`API error ${res.status}: ${text}`);\n }\n const json = (await res.json()) as { success: boolean; data: unknown };\n if (!json.success) throw new Error('API responded with success: false');\n return json.data;\n}\n\nasync function apiGet(path: string, params?: Record<string, string>): Promise<unknown> {\n const url = new URL(path, apiUrl);\n if (params) {\n for (const [k, v] of Object.entries(params)) {\n url.searchParams.set(k, v);\n }\n }\n const res = await fetchWithRetry(url.toString(), {\n headers: apiHeaders(),\n });\n if (!res.ok) {\n const text = await res.text();\n throw new Error(`API error ${res.status}: ${text}`);\n }\n const json = (await res.json()) as { success: boolean; data: unknown };\n if (!json.success) throw new Error('API responded with success: false');\n return json.data;\n}\n\n// ── GateProxy deps ──────────────────────────────────────────────────────\n\nconst deps: GateProxyDeps = {\n runGates,\n\n async forwardMoveTicket(args: Record<string, unknown>): Promise<unknown> {\n const ticketProjectId = validateUuid(args['projectId'], 'projectId');\n const ticketId = validateUuid(args['ticketId'], 'ticketId');\n const body = { ...args };\n delete body['projectId'];\n delete body['ticketId'];\n return apiPatch(`/projects/${ticketProjectId}/tickets/${ticketId}/move`, body);\n },\n\n async forwardCompleteTask(args: Record<string, unknown>): Promise<unknown> {\n const projectId = validateUuid(args['projectId'], 'projectId');\n const ticketId = validateUuid(args['ticketId'], 'ticketId');\n const body = { ...args };\n delete body['projectId'];\n delete body['ticketId'];\n return apiPost(`/projects/${projectId}/tickets/${ticketId}/complete`, body);\n },\n\n async getTicketGateWaivers(ticketId: string): Promise<string[]> {\n try {\n // Field values require both projectId and ticketId in the path.\n // projectId is injected as an env var (PROJECT_ID) at server startup.\n validateUuid(ticketId, 'ticketId');\n const data = (await apiGet(`/projects/${projectId}/tickets/${ticketId}/field-values`)) as\n Array<{ field_name: string; value: unknown }>;\n const waiver = data.find((fv) => fv.field_name === 'gate_waiver');\n if (!waiver) return [];\n if (Array.isArray(waiver.value)) {\n return waiver.value.filter((v): v is string => typeof v === 'string' && v.length > 0);\n }\n return [];\n } catch (err) {\n process.stderr.write(`gate-proxy-server: waiver fetch failed (running all gates): ${err instanceof Error ? err.message : String(err)}\\n`);\n return [];\n }\n },\n\n async getTicketDetails(ticketId: string): Promise<{ title: string; description: string }> {\n try {\n validateUuid(ticketId, 'ticketId');\n const data = (await apiGet(`/projects/${projectId}/tickets/${ticketId}`)) as {\n title?: string;\n description?: string;\n };\n return {\n title: data.title ?? '',\n description: data.description ?? '',\n };\n } catch (err) {\n process.stderr.write(\n `gate-proxy-server: ticket details fetch failed: ${err instanceof Error ? err.message : String(err)}\\n`,\n );\n return { title: '', description: '' };\n }\n },\n\n async getColumnType(columnIdArg: string): Promise<string | null> {\n try {\n validateUuid(columnIdArg, 'columnId');\n const data = (await apiGet(`/projects/${projectId}/columns/${columnIdArg}`)) as {\n type?: string;\n };\n return data.type ?? null;\n } catch (err) {\n process.stderr.write(\n `gate-proxy-server: column type fetch failed for ${columnIdArg}: ${err instanceof Error ? err.message : String(err)}\\n`,\n );\n return null;\n }\n },\n};\n\nconst proxy = new GateProxy(gateConfig, deps);\n\n// ── JSON-RPC types ──────────────────────────────────────────────────────\n\ninterface JsonRpcRequest {\n jsonrpc: '2.0';\n id: number | string;\n method: string;\n params?: Record<string, unknown>;\n}\n\ninterface JsonRpcResponse {\n jsonrpc: '2.0';\n id: number | string | null;\n result?: unknown;\n error?: { code: number; message: string; data?: unknown };\n}\n\n// ── Tool definitions ────────────────────────────────────────────────────\n\nconst TOOLS = [\n {\n name: 'kantban_run_gates',\n description:\n 'Run all configured gates for the current column and report results. ' +\n 'Use this to check gate status before attempting to move a ticket. ' +\n 'Pass ticketId to filter out waived gates for that ticket.',\n inputSchema: {\n type: 'object' as const,\n properties: {\n ticketId: { type: 'string', description: 'Ticket ID (UUID) — filters out waived gates for this ticket' },\n },\n required: [] as string[],\n },\n },\n {\n name: 'kantban_move_ticket',\n description:\n 'Move a ticket to a different column. Gates are automatically enforced — ' +\n 'if any required gate fails, the move is blocked and failure details are returned.',\n inputSchema: {\n type: 'object' as const,\n properties: {\n projectId: { type: 'string', description: 'Project ID (UUID)' },\n ticketId: { type: 'string', description: 'Ticket ID (UUID)' },\n columnId: { type: 'string', description: 'Target column ID (UUID)' },\n currentColumnName: { type: 'string', description: 'Current column name' },\n handoff: {\n type: 'object',\n description: 'Structured handoff data for the next pipeline stage',\n additionalProperties: true,\n },\n },\n required: ['projectId', 'ticketId', 'columnId'],\n },\n },\n {\n name: 'kantban_complete_task',\n description:\n 'Mark a ticket as complete. Gates are automatically enforced — ' +\n 'if any required gate fails, the completion is blocked and failure details are returned.',\n inputSchema: {\n type: 'object' as const,\n properties: {\n projectId: { type: 'string', description: 'Project ID (UUID)' },\n ticketId: { type: 'string', description: 'Ticket ID (UUID)' },\n currentColumnName: { type: 'string', description: 'Current column name' },\n moveToColumn: { type: 'string', description: 'Column name to move the ticket to (e.g. \"Done\")' },\n completionComment: { type: 'string', description: 'Comment to add upon completion' },\n handoff: {\n type: 'object',\n description: 'Structured handoff data for the next pipeline stage',\n additionalProperties: true,\n },\n },\n required: ['projectId', 'ticketId'],\n },\n },\n];\n\n// ── Tool handlers ───────────────────────────────────────────────────────\n\nasync function handleToolCall(\n name: string,\n args: Record<string, unknown>,\n): Promise<{ content: Array<{ type: 'text'; text: string }>; isError?: boolean }> {\n try {\n switch (name) {\n case 'kantban_run_gates': {\n const ticketIdArg = args['ticketId'] as string | undefined;\n if (ticketIdArg) validateUuid(ticketIdArg, 'ticketId');\n const result = await proxy.handleRunGates(columnName, ticketIdArg);\n return {\n content: [{ type: 'text', text: JSON.stringify(result, null, 2) }],\n };\n }\n\n case 'kantban_move_ticket': {\n const moveProjectId = validateUuid(args['projectId'], 'projectId');\n const moveTicketId = validateUuid(args['ticketId'], 'ticketId');\n const moveColumnId = validateUuid(args['columnId'], 'columnId');\n const result = await proxy.handleMoveTicket({\n projectId: moveProjectId,\n ticketId: moveTicketId,\n columnId: moveColumnId,\n currentColumnName: (args['currentColumnName'] as string | undefined) ?? columnName,\n args: {\n ...(args['handoff'] !== undefined ? { handoff: args['handoff'] } : {}),\n ...(args['column_id'] !== undefined ? { column_id: args['column_id'] } : {}),\n },\n });\n const isError = result.error === 'GATE_FAILURE';\n if (isError && result.results) {\n process.stderr.write(`gate-proxy-server: move blocked\\n${formatGateErrors(result.results)}`);\n }\n return {\n content: [{ type: 'text', text: JSON.stringify(result, null, 2) }],\n ...(isError ? { isError: true } : {}),\n };\n }\n\n case 'kantban_complete_task': {\n const completeProjectId = validateUuid(args['projectId'], 'projectId');\n const completeTicketId = validateUuid(args['ticketId'], 'ticketId');\n const result = await proxy.handleCompleteTask({\n projectId: completeProjectId,\n ticketId: completeTicketId,\n currentColumnName: (args['currentColumnName'] as string | undefined) ?? columnName,\n args: {\n ...(args['moveToColumn'] !== undefined ? { moveToColumn: args['moveToColumn'] } : {}),\n ...(args['completionComment'] !== undefined ? { completionComment: args['completionComment'] } : {}),\n ...(args['handoff'] !== undefined ? { handoff: args['handoff'] } : {}),\n },\n });\n const isError = result.error === 'GATE_FAILURE';\n if (isError && result.results) {\n process.stderr.write(`gate-proxy-server: complete blocked\\n${formatGateErrors(result.results)}`);\n }\n return {\n content: [{ type: 'text', text: JSON.stringify(result, null, 2) }],\n ...(isError ? { isError: true } : {}),\n };\n }\n\n default:\n return {\n content: [{ type: 'text', text: `Unknown tool: ${name}` }],\n isError: true,\n };\n }\n } catch (err) {\n const message = err instanceof Error ? err.message : String(err);\n return {\n content: [{ type: 'text', text: `Error: ${message}` }],\n isError: true,\n };\n }\n}\n\n// ── JSON-RPC dispatch ───────────────────────────────────────────────────\n\nfunction sendResponse(response: JsonRpcResponse): void {\n const json = JSON.stringify(response);\n process.stdout.write(json + '\\n');\n}\n\nasync function handleMessage(msg: JsonRpcRequest): Promise<void> {\n const { id, method, params } = msg;\n\n switch (method) {\n case 'initialize': {\n sendResponse({\n jsonrpc: '2.0',\n id,\n result: {\n protocolVersion: '2024-11-05',\n capabilities: { tools: {} },\n serverInfo: {\n name: 'kantban-gates',\n version: '0.1.0',\n },\n },\n });\n return;\n }\n\n case 'notifications/initialized': {\n // Client acknowledgement — no response needed for notifications\n return;\n }\n\n case 'tools/list': {\n sendResponse({\n jsonrpc: '2.0',\n id,\n result: { tools: TOOLS },\n });\n return;\n }\n\n case 'tools/call': {\n const toolName = params?.['name'] as string | undefined;\n const toolArgs = (params?.['arguments'] as Record<string, unknown> | undefined) ?? {};\n\n if (!toolName) {\n sendResponse({\n jsonrpc: '2.0',\n id,\n error: { code: -32602, message: 'Missing tool name' },\n });\n return;\n }\n\n const result = await handleToolCall(toolName, toolArgs);\n sendResponse({\n jsonrpc: '2.0',\n id,\n result,\n });\n return;\n }\n\n case 'ping': {\n sendResponse({ jsonrpc: '2.0', id, result: {} });\n return;\n }\n\n default: {\n // Unknown method — return method not found per JSON-RPC spec\n if (method.startsWith('notifications/')) {\n // Notifications don't get responses\n return;\n }\n sendResponse({\n jsonrpc: '2.0',\n id,\n error: { code: -32601, message: `Method not found: ${method}` },\n });\n }\n }\n}\n\n// ── stdin reader ────────────────────────────────────────────────────────\n\nconst rl = createInterface({ input: process.stdin });\n\n// Serialize message processing to prevent concurrent double-moves\nlet messageQueue: Promise<void> = Promise.resolve();\n\n/**\n * Per-message handler timeout. Bounds the serialized queue so one hung handler\n * cannot block every subsequent tool call forever (classic symptom: a gate's\n * child process leaves grandchildren holding stdio pipes open past execFile's\n * SIGTERM, the gate promise never resolves, and the next kantban_move_ticket\n * waits behind it in `messageQueue` indefinitely).\n *\n * 15 min is longer than any realistic single gate suite but short enough that\n * a true deadlock surfaces as an error response instead of an infinite wait.\n */\nconst HANDLER_TIMEOUT_MS = 15 * 60 * 1000;\n\nrl.on('line', (line: string) => {\n const trimmed = line.trim();\n if (!trimmed) return;\n\n let msg: JsonRpcRequest;\n try {\n msg = JSON.parse(trimmed) as JsonRpcRequest;\n } catch {\n sendResponse({\n jsonrpc: '2.0',\n id: null as unknown as number,\n error: { code: -32700, message: 'Parse error' },\n });\n return;\n }\n\n messageQueue = messageQueue\n .then(() => {\n let timer: NodeJS.Timeout | undefined;\n const timeout = new Promise<never>((_, reject) => {\n timer = setTimeout(\n () => reject(new Error(`handler timed out after ${HANDLER_TIMEOUT_MS}ms`)),\n HANDLER_TIMEOUT_MS,\n );\n });\n return Promise.race([handleMessage(msg), timeout]).finally(() => {\n if (timer) clearTimeout(timer);\n });\n })\n .catch((err) => {\n const message = err instanceof Error ? err.message : String(err);\n process.stderr.write(`gate-proxy-server: handler failed: ${message}\\n`);\n // Send an error response so the client doesn't wait forever. Requests\n // (with an id) expect a response; notifications (id === undefined)\n // legitimately don't.\n if (msg.id !== undefined && msg.id !== null) {\n try {\n sendResponse({\n jsonrpc: '2.0',\n id: msg.id,\n error: {\n code: -32603,\n message: `gate-proxy handler error: ${message}`,\n },\n });\n } catch (sendErr) {\n process.stderr.write(\n `gate-proxy-server: failed to send error response: ${sendErr instanceof Error ? sendErr.message : String(sendErr)}\\n`,\n );\n }\n }\n });\n});\n\nrl.on('close', () => {\n process.exit(0);\n});\n\n// Suppress unhandled rejection crashes — log to stderr and continue\nprocess.on('unhandledRejection', (err) => {\n process.stderr.write(\n `gate-proxy-server: unhandled rejection: ${err instanceof Error ? err.message : String(err)}\\n`,\n );\n});\n\nprocess.stderr.write(`gate-proxy-server: started (column=\"${columnName}\", gates=${gateConfig.default.length})\\n`);\n","import { resolveGatesForColumn, parseTimeout } from './gate-config.js';\nimport { formatGateErrors } from './gate-runner.js';\nimport type { GateConfig, GateDefinition, GateResult } from '@kantban/types';\n\nexport interface GateProxyDeps {\n runGates: (gates: GateDefinition[], options?: { cwd?: string; env?: Record<string, string>; totalTimeoutMs?: number }) => Promise<GateResult[]>;\n forwardMoveTicket: (args: Record<string, unknown>) => Promise<unknown>;\n forwardCompleteTask: (args: Record<string, unknown>) => Promise<unknown>;\n getTicketGateWaivers: (ticketId: string) => Promise<string[]>;\n getTicketDetails: (ticketId: string) => Promise<{ title: string; description: string }>;\n /**\n * Fetch the type of a column (`start`, `in_progress`, `done`, `default`).\n * Used to bypass gates on moves into terminal columns — gates enforce\n * \"work in the source column is complete\", but entering a `done` column\n * is by definition post-work cleanup, and running source-column gates at\n * that point can strand tickets in Escalation when main is transiently\n * broken (the merge commit has already been pushed; the ticket just\n * needs to follow).\n */\n getColumnType: (columnId: string) => Promise<string | null>;\n}\n\nexport interface RunGatesResult {\n passed: boolean;\n results: GateResult[];\n}\n\nexport interface MoveArgs {\n projectId: string;\n ticketId: string;\n columnId: string;\n currentColumnName: string;\n args: Record<string, unknown>;\n}\n\nexport interface CompleteArgs {\n projectId: string;\n ticketId: string;\n currentColumnName: string;\n args: Record<string, unknown>;\n}\n\nexport interface InterceptResult {\n error?: 'GATE_FAILURE';\n message?: string;\n formatted?: string;\n results?: GateResult[];\n hint?: string;\n forwardResult?: unknown;\n}\n\nexport class GateProxy {\n private config: GateConfig;\n private deps: GateProxyDeps;\n\n constructor(config: GateConfig, deps: GateProxyDeps) {\n this.config = config;\n this.deps = deps;\n }\n\n private async buildTicketEnv(\n ticketId: string,\n projectId: string,\n columnName: string,\n ): Promise<Record<string, string>> {\n const details = await this.deps.getTicketDetails(ticketId);\n return {\n KANTBAN_TICKET_ID: ticketId,\n KANTBAN_TICKET_TITLE: details.title,\n KANTBAN_TICKET_DESCRIPTION: details.description,\n KANTBAN_TICKET_COLUMN: columnName,\n KANTBAN_PROJECT_ID: projectId,\n };\n }\n\n private buildRunOptions(): { cwd?: string; env?: Record<string, string>; totalTimeoutMs?: number } {\n const opts: { cwd?: string; env?: Record<string, string>; totalTimeoutMs?: number } = {};\n if (this.config.settings?.cwd !== undefined) opts.cwd = this.config.settings.cwd;\n if (this.config.settings?.env !== undefined) opts.env = this.config.settings.env;\n if (this.config.settings?.total_timeout !== undefined) {\n opts.totalTimeoutMs = parseTimeout(this.config.settings.total_timeout);\n }\n return opts;\n }\n\n async handleRunGates(columnName: string, ticketId?: string): Promise<RunGatesResult> {\n const allGates = resolveGatesForColumn(this.config, columnName);\n\n let gates = allGates;\n if (ticketId) {\n const waivers = await this.deps.getTicketGateWaivers(ticketId);\n const waiverSet = new Set(waivers);\n gates = allGates.filter((g) => !waiverSet.has(g.name));\n }\n\n const results = await this.deps.runGates(gates, this.buildRunOptions());\n return {\n passed: results.filter((r) => r.required).every((r) => r.passed),\n results,\n };\n }\n\n async handleMoveTicket(move: MoveArgs): Promise<InterceptResult> {\n // Terminal-column bypass: moves INTO a `done`-type column are post-work\n // cleanup, not work-in-progress. Running source-column gates at this\n // point can strand tickets (classic scenario: Merge agent successfully\n // pushes feature branch to main, then this move-to-Done runs the merge\n // column's gates against a transiently-broken main and fails — but the\n // push already happened, so the ticket MUST follow to Done). Gates\n // enforce entry criteria when work arrives at a column; terminal\n // columns have no further work to gate.\n try {\n const targetType = await this.deps.getColumnType(move.columnId);\n if (targetType === 'done') {\n const forwardResult = await this.deps.forwardMoveTicket({\n ...move.args,\n projectId: move.projectId,\n ticketId: move.ticketId,\n column_id: move.columnId,\n });\n return { forwardResult };\n }\n } catch {\n // If the type lookup fails, fall through to the normal gate path so\n // behavior stays conservative (gates enforced by default).\n }\n\n const allGates = resolveGatesForColumn(this.config, move.currentColumnName);\n\n // Filter out waived gates\n const waivers = await this.deps.getTicketGateWaivers(move.ticketId);\n const waiverSet = new Set(waivers);\n const gates = allGates.filter((g) => !waiverSet.has(g.name));\n\n let results: GateResult[];\n try {\n const opts = this.buildRunOptions();\n const ticketEnv = await this.buildTicketEnv(move.ticketId, move.projectId, move.currentColumnName);\n opts.env = { ...opts.env, ...ticketEnv };\n results = await this.deps.runGates(gates, opts);\n } catch (err) {\n return {\n error: 'GATE_FAILURE',\n message: `Gate evaluation error: ${err instanceof Error ? err.message : String(err)}`,\n hint: 'Fix the gate environment and retry',\n };\n }\n\n const requiredFailures = results.filter((r) => r.required && !r.passed);\n if (requiredFailures.length > 0) {\n return {\n error: 'GATE_FAILURE',\n message: `Cannot move ticket — ${requiredFailures.length} required gate(s) failed`,\n formatted: formatGateErrors(results.filter((r) => r.required)),\n results,\n hint: 'Fix the failing gate(s) and try move_ticket again',\n };\n }\n\n const forwardResult = await this.deps.forwardMoveTicket({\n ...move.args,\n projectId: move.projectId,\n ticketId: move.ticketId,\n column_id: move.columnId,\n });\n\n return { forwardResult };\n }\n\n async handleCompleteTask(complete: CompleteArgs): Promise<InterceptResult> {\n const allGates = resolveGatesForColumn(this.config, complete.currentColumnName);\n const waivers = await this.deps.getTicketGateWaivers(complete.ticketId);\n const waiverSet = new Set(waivers);\n const gates = allGates.filter((g) => !waiverSet.has(g.name));\n\n let results: GateResult[];\n try {\n const opts = this.buildRunOptions();\n const ticketEnv = await this.buildTicketEnv(complete.ticketId, complete.projectId, complete.currentColumnName);\n opts.env = { ...opts.env, ...ticketEnv };\n results = await this.deps.runGates(gates, opts);\n } catch (err) {\n return {\n error: 'GATE_FAILURE',\n message: `Gate evaluation error: ${err instanceof Error ? err.message : String(err)}`,\n hint: 'Fix the gate environment and retry',\n };\n }\n\n const requiredFailures = results.filter((r) => r.required && !r.passed);\n if (requiredFailures.length > 0) {\n return {\n error: 'GATE_FAILURE',\n message: `Cannot complete task — ${requiredFailures.length} required gate(s) failed`,\n formatted: formatGateErrors(results.filter((r) => r.required)),\n results,\n hint: 'Fix the failing gate(s) and try complete_task again',\n };\n }\n\n const forwardResult = await this.deps.forwardCompleteTask({\n ...complete.args,\n projectId: complete.projectId,\n ticketId: complete.ticketId,\n });\n\n return { forwardResult };\n }\n}\n"],"mappings":"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","names":["deps","projectId","columnName","forwardResult","projectId"]}
|
|
1
|
+
{"version":3,"sources":["../../src/lib/gate-proxy-server.ts","../../src/lib/gate-proxy.ts"],"sourcesContent":["#!/usr/bin/env node\n\n/**\n * gate-proxy-server.ts — MCP stdio server for gate enforcement.\n *\n * Spawned by the pipeline orchestrator as a child process. Reads gate config\n * from env vars, instantiates GateProxy, and exposes three MCP tools over\n * JSON-RPC on stdin/stdout:\n *\n * - kantban_run_gates — Run column gates and report results\n * - kantban_move_ticket — Run gates then forward move to KantBan API\n * - kantban_complete_task — Run gates then forward complete to KantBan API\n *\n * Uses raw JSON-RPC (no @modelcontextprotocol/sdk dependency).\n */\n\nimport { readFileSync } from 'node:fs';\nimport { createInterface } from 'node:readline';\nimport { parseGateConfig } from './gate-config.js';\nimport { runGates, formatGateErrors } from './gate-runner.js';\nimport { GateProxy } from './gate-proxy.js';\nimport { fetchWithRetry } from '../client.js';\nimport type { GateProxyDeps } from './gate-proxy.js';\n\n// ── UUID validation ──────────────────────────────────────────────────────\n\nconst UUID_RE = /^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/i;\n\nfunction validateUuid(value: unknown, name: string): string {\n if (typeof value !== 'string' || !UUID_RE.test(value)) {\n throw new Error(`Invalid ${name}: expected UUID, got \"${String(value)}\"`);\n }\n return value;\n}\n\n// ── env ──────────────────────────────────────────────────────────────────\n\nconst GATE_CONFIG_PATH = process.env['GATE_CONFIG_PATH'];\nconst COLUMN_ID = process.env['COLUMN_ID'];\nconst COLUMN_NAME = process.env['COLUMN_NAME'];\nconst PROJECT_ID = process.env['PROJECT_ID'];\nconst API_TOKEN = process.env['KANTBAN_API_TOKEN'];\nconst API_URL = process.env['KANTBAN_API_URL'];\n\nfunction requireEnv(name: string, value: string | undefined): string {\n if (!value) {\n process.stderr.write(`gate-proxy-server: missing required env var ${name}\\n`);\n process.exit(1);\n }\n return value;\n}\n\nconst gateConfigPath = requireEnv('GATE_CONFIG_PATH', GATE_CONFIG_PATH);\nconst columnId = requireEnv('COLUMN_ID', COLUMN_ID);\nconst columnName = requireEnv('COLUMN_NAME', COLUMN_NAME);\nconst projectId = requireEnv('PROJECT_ID', PROJECT_ID);\nconst apiToken = requireEnv('KANTBAN_API_TOKEN', API_TOKEN);\nconst apiUrl = requireEnv('KANTBAN_API_URL', API_URL);\n\n// ── gate config ──────────────────────────────────────────────────────────\n\nconst yamlContent = readFileSync(gateConfigPath, 'utf-8');\nconst gateConfig = parseGateConfig(yamlContent);\n\n// GATE_CWD overrides the YAML settings.cwd — set by the orchestrator to the\n// agent's worktree path so gates run against the agent's checked-out code,\n// not the root repo.\nconst GATE_CWD = process.env['GATE_CWD'];\nif (GATE_CWD) {\n if (!gateConfig.settings) gateConfig.settings = {};\n gateConfig.settings.cwd = GATE_CWD;\n}\n\n// ── API helpers ──────────────────────────────────────────────────────────\n\nfunction apiHeaders(): Record<string, string> {\n return {\n 'Authorization': `Bearer ${apiToken}`,\n 'Content-Type': 'application/json',\n 'X-KantBan-Via': 'cli-gate-proxy',\n };\n}\n\nasync function apiPost(path: string, body: Record<string, unknown>): Promise<unknown> {\n const url = new URL(path, apiUrl);\n const res = await fetchWithRetry(url.toString(), {\n method: 'POST',\n headers: apiHeaders(),\n body: JSON.stringify(body),\n });\n if (!res.ok) {\n const text = await res.text();\n throw new Error(`API error ${res.status}: ${text}`);\n }\n const json = (await res.json()) as { success: boolean; data: unknown };\n if (!json.success) throw new Error('API responded with success: false');\n return json.data;\n}\n\nasync function apiPatch(path: string, body: Record<string, unknown>): Promise<unknown> {\n const url = new URL(path, apiUrl);\n const res = await fetchWithRetry(url.toString(), {\n method: 'PATCH',\n headers: apiHeaders(),\n body: JSON.stringify(body),\n });\n if (!res.ok) {\n const text = await res.text();\n throw new Error(`API error ${res.status}: ${text}`);\n }\n const json = (await res.json()) as { success: boolean; data: unknown };\n if (!json.success) throw new Error('API responded with success: false');\n return json.data;\n}\n\nasync function apiGet(path: string, params?: Record<string, string>): Promise<unknown> {\n const url = new URL(path, apiUrl);\n if (params) {\n for (const [k, v] of Object.entries(params)) {\n url.searchParams.set(k, v);\n }\n }\n const res = await fetchWithRetry(url.toString(), {\n headers: apiHeaders(),\n });\n if (!res.ok) {\n const text = await res.text();\n throw new Error(`API error ${res.status}: ${text}`);\n }\n const json = (await res.json()) as { success: boolean; data: unknown };\n if (!json.success) throw new Error('API responded with success: false');\n return json.data;\n}\n\n// ── GateProxy deps ──────────────────────────────────────────────────────\n\nconst deps: GateProxyDeps = {\n runGates,\n\n async forwardMoveTicket(args: Record<string, unknown>): Promise<unknown> {\n const ticketProjectId = validateUuid(args['projectId'], 'projectId');\n const ticketId = validateUuid(args['ticketId'], 'ticketId');\n const body = { ...args };\n delete body['projectId'];\n delete body['ticketId'];\n return apiPatch(`/projects/${ticketProjectId}/tickets/${ticketId}/move`, body);\n },\n\n async forwardCompleteTask(args: Record<string, unknown>): Promise<unknown> {\n const projectId = validateUuid(args['projectId'], 'projectId');\n const ticketId = validateUuid(args['ticketId'], 'ticketId');\n const body = { ...args };\n delete body['projectId'];\n delete body['ticketId'];\n return apiPost(`/projects/${projectId}/tickets/${ticketId}/complete`, body);\n },\n\n async getTicketGateWaivers(ticketId: string): Promise<string[]> {\n try {\n // Field values require both projectId and ticketId in the path.\n // projectId is injected as an env var (PROJECT_ID) at server startup.\n validateUuid(ticketId, 'ticketId');\n const data = (await apiGet(`/projects/${projectId}/tickets/${ticketId}/field-values`)) as\n Array<{ field_name: string; value: unknown }>;\n const waiver = data.find((fv) => fv.field_name === 'gate_waiver');\n if (!waiver) return [];\n if (Array.isArray(waiver.value)) {\n return waiver.value.filter((v): v is string => typeof v === 'string' && v.length > 0);\n }\n return [];\n } catch (err) {\n process.stderr.write(`gate-proxy-server: waiver fetch failed (running all gates): ${err instanceof Error ? err.message : String(err)}\\n`);\n return [];\n }\n },\n\n async getTicketDetails(ticketId: string): Promise<{ title: string; description: string }> {\n try {\n validateUuid(ticketId, 'ticketId');\n const data = (await apiGet(`/projects/${projectId}/tickets/${ticketId}`)) as {\n title?: string;\n description?: string;\n };\n return {\n title: data.title ?? '',\n description: data.description ?? '',\n };\n } catch (err) {\n process.stderr.write(\n `gate-proxy-server: ticket details fetch failed: ${err instanceof Error ? err.message : String(err)}\\n`,\n );\n return { title: '', description: '' };\n }\n },\n\n async getColumn(columnIdArg: string): Promise<{ name: string; type: string } | null> {\n try {\n validateUuid(columnIdArg, 'columnId');\n const data = (await apiGet(`/projects/${projectId}/columns/${columnIdArg}`)) as {\n name?: string;\n type?: string;\n };\n if (!data.name || !data.type) return null;\n return { name: data.name, type: data.type };\n } catch (err) {\n process.stderr.write(\n `gate-proxy-server: column fetch failed for ${columnIdArg}: ${err instanceof Error ? err.message : String(err)}\\n`,\n );\n return null;\n }\n },\n};\n\nconst proxy = new GateProxy(gateConfig, deps);\n\n// ── JSON-RPC types ──────────────────────────────────────────────────────\n\ninterface JsonRpcRequest {\n jsonrpc: '2.0';\n id: number | string;\n method: string;\n params?: Record<string, unknown>;\n}\n\ninterface JsonRpcResponse {\n jsonrpc: '2.0';\n id: number | string | null;\n result?: unknown;\n error?: { code: number; message: string; data?: unknown };\n}\n\n// ── Tool definitions ────────────────────────────────────────────────────\n\nconst TOOLS = [\n {\n name: 'kantban_run_gates',\n description:\n 'Run all configured gates for the current column and report results. ' +\n 'Use this to check gate status before attempting to move a ticket. ' +\n 'Pass ticketId to filter out waived gates for that ticket.',\n inputSchema: {\n type: 'object' as const,\n properties: {\n ticketId: { type: 'string', description: 'Ticket ID (UUID) — filters out waived gates for this ticket' },\n },\n required: [] as string[],\n },\n },\n {\n name: 'kantban_move_ticket',\n description:\n 'Move a ticket to a different column. Gates are automatically enforced — ' +\n 'if any required gate fails, the move is blocked and failure details are returned.',\n inputSchema: {\n type: 'object' as const,\n properties: {\n projectId: { type: 'string', description: 'Project ID (UUID)' },\n ticketId: { type: 'string', description: 'Ticket ID (UUID)' },\n columnId: { type: 'string', description: 'Target column ID (UUID)' },\n currentColumnName: { type: 'string', description: 'Current column name' },\n handoff: {\n type: 'object',\n description: 'Structured handoff data for the next pipeline stage',\n additionalProperties: true,\n },\n },\n required: ['projectId', 'ticketId', 'columnId'],\n },\n },\n {\n name: 'kantban_complete_task',\n description:\n 'Mark a ticket as complete. Gates are automatically enforced — ' +\n 'if any required gate fails, the completion is blocked and failure details are returned.',\n inputSchema: {\n type: 'object' as const,\n properties: {\n projectId: { type: 'string', description: 'Project ID (UUID)' },\n ticketId: { type: 'string', description: 'Ticket ID (UUID)' },\n currentColumnName: { type: 'string', description: 'Current column name' },\n moveToColumn: { type: 'string', description: 'Column name to move the ticket to (e.g. \"Done\")' },\n completionComment: { type: 'string', description: 'Comment to add upon completion' },\n handoff: {\n type: 'object',\n description: 'Structured handoff data for the next pipeline stage',\n additionalProperties: true,\n },\n },\n required: ['projectId', 'ticketId'],\n },\n },\n];\n\n// ── Tool handlers ───────────────────────────────────────────────────────\n\nasync function handleToolCall(\n name: string,\n args: Record<string, unknown>,\n): Promise<{ content: Array<{ type: 'text'; text: string }>; isError?: boolean }> {\n try {\n switch (name) {\n case 'kantban_run_gates': {\n const ticketIdArg = args['ticketId'] as string | undefined;\n if (ticketIdArg) validateUuid(ticketIdArg, 'ticketId');\n const result = await proxy.handleRunGates(columnName, ticketIdArg);\n return {\n content: [{ type: 'text', text: JSON.stringify(result, null, 2) }],\n };\n }\n\n case 'kantban_move_ticket': {\n const moveProjectId = validateUuid(args['projectId'], 'projectId');\n const moveTicketId = validateUuid(args['ticketId'], 'ticketId');\n const moveColumnId = validateUuid(args['columnId'], 'columnId');\n const result = await proxy.handleMoveTicket({\n projectId: moveProjectId,\n ticketId: moveTicketId,\n columnId: moveColumnId,\n currentColumnName: (args['currentColumnName'] as string | undefined) ?? columnName,\n args: {\n ...(args['handoff'] !== undefined ? { handoff: args['handoff'] } : {}),\n ...(args['column_id'] !== undefined ? { column_id: args['column_id'] } : {}),\n },\n });\n const isError = result.error === 'GATE_FAILURE';\n if (isError && result.results) {\n process.stderr.write(`gate-proxy-server: move blocked\\n${formatGateErrors(result.results)}`);\n for (const r of result.results) {\n if (!r.passed) {\n console.error(\n `[gate-error] ticket=${moveTicketId} column=${columnName} ` +\n `gate=${r.name} exit_code=${r.exit_code} errno=${r.errno ?? ''} ` +\n `syscall=${r.syscall ?? ''} stderr=${JSON.stringify((r.stderrTail ?? '').slice(0, 500))}`,\n );\n }\n }\n }\n return {\n content: [{ type: 'text', text: JSON.stringify(result, null, 2) }],\n ...(isError ? { isError: true } : {}),\n };\n }\n\n case 'kantban_complete_task': {\n const completeProjectId = validateUuid(args['projectId'], 'projectId');\n const completeTicketId = validateUuid(args['ticketId'], 'ticketId');\n const result = await proxy.handleCompleteTask({\n projectId: completeProjectId,\n ticketId: completeTicketId,\n currentColumnName: (args['currentColumnName'] as string | undefined) ?? columnName,\n args: {\n ...(args['moveToColumn'] !== undefined ? { moveToColumn: args['moveToColumn'] } : {}),\n ...(args['completionComment'] !== undefined ? { completionComment: args['completionComment'] } : {}),\n ...(args['handoff'] !== undefined ? { handoff: args['handoff'] } : {}),\n },\n });\n const isError = result.error === 'GATE_FAILURE';\n if (isError && result.results) {\n process.stderr.write(`gate-proxy-server: complete blocked\\n${formatGateErrors(result.results)}`);\n for (const r of result.results) {\n if (!r.passed) {\n console.error(\n `[gate-error] ticket=${completeTicketId} column=${columnName} ` +\n `gate=${r.name} exit_code=${r.exit_code} errno=${r.errno ?? ''} ` +\n `syscall=${r.syscall ?? ''} stderr=${JSON.stringify((r.stderrTail ?? '').slice(0, 500))}`,\n );\n }\n }\n }\n return {\n content: [{ type: 'text', text: JSON.stringify(result, null, 2) }],\n ...(isError ? { isError: true } : {}),\n };\n }\n\n default:\n return {\n content: [{ type: 'text', text: `Unknown tool: ${name}` }],\n isError: true,\n };\n }\n } catch (err) {\n const message = err instanceof Error ? err.message : String(err);\n return {\n content: [{ type: 'text', text: `Error: ${message}` }],\n isError: true,\n };\n }\n}\n\n// ── JSON-RPC dispatch ───────────────────────────────────────────────────\n\nfunction sendResponse(response: JsonRpcResponse): void {\n const json = JSON.stringify(response);\n process.stdout.write(json + '\\n');\n}\n\nasync function handleMessage(msg: JsonRpcRequest): Promise<void> {\n const { id, method, params } = msg;\n\n switch (method) {\n case 'initialize': {\n sendResponse({\n jsonrpc: '2.0',\n id,\n result: {\n protocolVersion: '2024-11-05',\n capabilities: { tools: {} },\n serverInfo: {\n name: 'kantban-gates',\n version: '0.1.0',\n },\n },\n });\n return;\n }\n\n case 'notifications/initialized': {\n // Client acknowledgement — no response needed for notifications\n return;\n }\n\n case 'tools/list': {\n sendResponse({\n jsonrpc: '2.0',\n id,\n result: { tools: TOOLS },\n });\n return;\n }\n\n case 'tools/call': {\n const toolName = params?.['name'] as string | undefined;\n const toolArgs = (params?.['arguments'] as Record<string, unknown> | undefined) ?? {};\n\n if (!toolName) {\n sendResponse({\n jsonrpc: '2.0',\n id,\n error: { code: -32602, message: 'Missing tool name' },\n });\n return;\n }\n\n const result = await handleToolCall(toolName, toolArgs);\n sendResponse({\n jsonrpc: '2.0',\n id,\n result,\n });\n return;\n }\n\n case 'ping': {\n sendResponse({ jsonrpc: '2.0', id, result: {} });\n return;\n }\n\n default: {\n // Unknown method — return method not found per JSON-RPC spec\n if (method.startsWith('notifications/')) {\n // Notifications don't get responses\n return;\n }\n sendResponse({\n jsonrpc: '2.0',\n id,\n error: { code: -32601, message: `Method not found: ${method}` },\n });\n }\n }\n}\n\n// ── stdin reader ────────────────────────────────────────────────────────\n\nconst rl = createInterface({ input: process.stdin });\n\n// Serialize message processing to prevent concurrent double-moves\nlet messageQueue: Promise<void> = Promise.resolve();\n\n/**\n * Per-message handler timeout. Bounds the serialized queue so one hung handler\n * cannot block every subsequent tool call forever (classic symptom: a gate's\n * child process leaves grandchildren holding stdio pipes open past execFile's\n * SIGTERM, the gate promise never resolves, and the next kantban_move_ticket\n * waits behind it in `messageQueue` indefinitely).\n *\n * Default is 30 min (env GATE_PROXY_HANDLER_TIMEOUT_MS overrides). 30 min\n * comfortably covers a 20-gate suite at ~60 s/gate while still surfacing a\n * true deadlock as an error response rather than an infinite wait.\n */\nconst HANDLER_TIMEOUT_MS =\n Number(process.env['GATE_PROXY_HANDLER_TIMEOUT_MS']) || 30 * 60_000;\n\nrl.on('line', (line: string) => {\n const trimmed = line.trim();\n if (!trimmed) return;\n\n let msg: JsonRpcRequest;\n try {\n msg = JSON.parse(trimmed) as JsonRpcRequest;\n } catch {\n sendResponse({\n jsonrpc: '2.0',\n id: null as unknown as number,\n error: { code: -32700, message: 'Parse error' },\n });\n return;\n }\n\n messageQueue = messageQueue\n .then(() => {\n let timer: NodeJS.Timeout | undefined;\n const timeout = new Promise<never>((_, reject) => {\n timer = setTimeout(\n () => reject(new Error(`handler timed out after ${HANDLER_TIMEOUT_MS}ms`)),\n HANDLER_TIMEOUT_MS,\n );\n });\n return Promise.race([handleMessage(msg), timeout]).finally(() => {\n if (timer) clearTimeout(timer);\n });\n })\n .catch((err) => {\n const message = err instanceof Error ? err.message : String(err);\n process.stderr.write(`gate-proxy-server: handler failed: ${message}\\n`);\n // Send an error response so the client doesn't wait forever. Requests\n // (with an id) expect a response; notifications (id === undefined)\n // legitimately don't.\n if (msg.id !== undefined && msg.id !== null) {\n try {\n sendResponse({\n jsonrpc: '2.0',\n id: msg.id,\n error: {\n code: -32603,\n message: `gate-proxy handler error: ${message}`,\n },\n });\n } catch (sendErr) {\n process.stderr.write(\n `gate-proxy-server: failed to send error response: ${sendErr instanceof Error ? sendErr.message : String(sendErr)}\\n`,\n );\n }\n }\n });\n});\n\nrl.on('close', () => {\n process.exit(0);\n});\n\n// Suppress unhandled rejection crashes — log to stderr and continue\nprocess.on('unhandledRejection', (err) => {\n process.stderr.write(\n `gate-proxy-server: unhandled rejection: ${err instanceof Error ? err.message : String(err)}\\n`,\n );\n});\n\nprocess.stderr.write(`gate-proxy-server: started (column=\"${columnName}\", gates=${gateConfig.default.length})\\n`);\nprocess.stderr.write(`[gate-proxy] handler timeout: ${HANDLER_TIMEOUT_MS}ms\\n`);\n","import { resolveGatesForColumn, parseTimeout } from './gate-config.js';\nimport { formatGateErrors } from './gate-runner.js';\nimport type { GateConfig, GateDefinition, GateResult } from '@kantban/types';\n\nexport interface GateProxyDeps {\n runGates: (gates: GateDefinition[], options?: { cwd?: string; env?: Record<string, string>; totalTimeoutMs?: number }) => Promise<GateResult[]>;\n forwardMoveTicket: (args: Record<string, unknown>) => Promise<unknown>;\n forwardCompleteTask: (args: Record<string, unknown>) => Promise<unknown>;\n getTicketGateWaivers: (ticketId: string) => Promise<string[]>;\n getTicketDetails: (ticketId: string) => Promise<{ title: string; description: string }>;\n /**\n * Fetch the name + type of a column. Type distinguishes terminal (`done`)\n * from in-flight columns for the bypass decision; name is used to look up\n * per-column overrides in gate-config so an explicit `columns.<name>:`\n * entry for a done-type target column can disable the bypass.\n *\n * The bypass exists because gates enforce \"work in the source column is\n * complete\", but entering a `done` column is by definition post-work\n * cleanup, and running source-column gates at that point can strand\n * tickets in Escalation when main is transiently broken (the merge commit\n * has already been pushed; the ticket just needs to follow). Returning\n * `{name, type}` in one call lets us check both the bypass condition and\n * whether an operator has explicitly configured gates for this terminal\n * column without a second API hop.\n */\n getColumn: (columnId: string) => Promise<{ name: string; type: string } | null>;\n}\n\nexport interface RunGatesResult {\n passed: boolean;\n results: GateResult[];\n}\n\nexport interface MoveArgs {\n projectId: string;\n ticketId: string;\n columnId: string;\n currentColumnName: string;\n args: Record<string, unknown>;\n}\n\nexport interface CompleteArgs {\n projectId: string;\n ticketId: string;\n currentColumnName: string;\n args: Record<string, unknown>;\n}\n\nexport interface InterceptResult {\n error?: 'GATE_FAILURE';\n message?: string;\n formatted?: string;\n results?: GateResult[];\n hint?: string;\n forwardResult?: unknown;\n}\n\nexport class GateProxy {\n private config: GateConfig;\n private deps: GateProxyDeps;\n\n constructor(config: GateConfig, deps: GateProxyDeps) {\n this.config = config;\n this.deps = deps;\n }\n\n private async buildTicketEnv(\n ticketId: string,\n projectId: string,\n columnName: string,\n ): Promise<Record<string, string>> {\n const details = await this.deps.getTicketDetails(ticketId);\n return {\n KANTBAN_TICKET_ID: ticketId,\n KANTBAN_TICKET_TITLE: details.title,\n KANTBAN_TICKET_DESCRIPTION: details.description,\n KANTBAN_TICKET_COLUMN: columnName,\n KANTBAN_PROJECT_ID: projectId,\n };\n }\n\n private buildRunOptions(): { cwd?: string; env?: Record<string, string>; totalTimeoutMs?: number } {\n const opts: { cwd?: string; env?: Record<string, string>; totalTimeoutMs?: number } = {};\n if (this.config.settings?.cwd !== undefined) opts.cwd = this.config.settings.cwd;\n if (this.config.settings?.env !== undefined) opts.env = this.config.settings.env;\n if (this.config.settings?.total_timeout !== undefined) {\n opts.totalTimeoutMs = parseTimeout(this.config.settings.total_timeout);\n }\n return opts;\n }\n\n async handleRunGates(columnName: string, ticketId?: string): Promise<RunGatesResult> {\n const { gates: allGates } = resolveGatesForColumn(this.config, columnName);\n\n let gates = allGates;\n if (ticketId) {\n const waivers = await this.deps.getTicketGateWaivers(ticketId);\n const waiverSet = new Set(waivers);\n gates = allGates.filter((g) => !waiverSet.has(g.name));\n }\n\n const results = await this.deps.runGates(gates, this.buildRunOptions());\n return {\n passed: results.filter((r) => r.required).every((r) => r.passed),\n results,\n };\n }\n\n async handleMoveTicket(move: MoveArgs): Promise<InterceptResult> {\n // Terminal-column bypass: moves INTO a `done`-type column are post-work\n // cleanup, not work-in-progress. Running source-column gates at this\n // point can strand tickets (classic scenario: Merge agent successfully\n // pushes feature branch to main, then this move-to-Done runs the merge\n // column's gates against a transiently-broken main and fails — but the\n // push already happened, so the ticket MUST follow to Done). Gates\n // enforce entry criteria when work arrives at a column; terminal\n // columns have no further work to gate.\n //\n // Exception: if the user has written an explicit `columns.<TargetName>:`\n // entry in pipeline.gates.yaml for the target done column, they've\n // opted INTO enforcement on moves-to-done. Honor that override by\n // disabling the bypass. Note the \"fall through\" path below runs gates\n // for `move.currentColumnName` (the SOURCE column), not the target\n // column's configured gates — the target override is a SIGNAL to\n // disable the bypass, not a replacement gate set. Source-column gates\n // still encode \"work in the source column is complete\"; the override\n // just re-enables that enforcement on moves into terminal columns.\n try {\n const target = await this.deps.getColumn(move.columnId);\n if (target?.type === 'done') {\n const { source } = resolveGatesForColumn(this.config, target.name);\n if (source === 'default') {\n const forwardResult = await this.deps.forwardMoveTicket({\n ...move.args,\n projectId: move.projectId,\n ticketId: move.ticketId,\n column_id: move.columnId,\n });\n return { forwardResult };\n }\n // Explicit override exists — do NOT bypass; fall through to gate execution.\n }\n } catch {\n // If the lookup fails, fall through to the normal gate path so\n // behavior stays conservative (gates enforced by default).\n }\n\n const { gates: allGates } = resolveGatesForColumn(this.config, move.currentColumnName);\n\n // Filter out waived gates\n const waivers = await this.deps.getTicketGateWaivers(move.ticketId);\n const waiverSet = new Set(waivers);\n const gates = allGates.filter((g) => !waiverSet.has(g.name));\n\n let results: GateResult[];\n try {\n const opts = this.buildRunOptions();\n const ticketEnv = await this.buildTicketEnv(move.ticketId, move.projectId, move.currentColumnName);\n opts.env = { ...opts.env, ...ticketEnv };\n results = await this.deps.runGates(gates, opts);\n } catch (err) {\n return {\n error: 'GATE_FAILURE',\n message: `Gate evaluation error: ${err instanceof Error ? err.message : String(err)}`,\n hint: 'Fix the gate environment and retry',\n };\n }\n\n const requiredFailures = results.filter((r) => r.required && !r.passed);\n if (requiredFailures.length > 0) {\n return {\n error: 'GATE_FAILURE',\n message: `Cannot move ticket — ${requiredFailures.length} required gate(s) failed`,\n formatted: formatGateErrors(results.filter((r) => r.required)),\n results,\n hint: 'Fix the failing gate(s) and try move_ticket again',\n };\n }\n\n const forwardResult = await this.deps.forwardMoveTicket({\n ...move.args,\n projectId: move.projectId,\n ticketId: move.ticketId,\n column_id: move.columnId,\n });\n\n return { forwardResult };\n }\n\n async handleCompleteTask(complete: CompleteArgs): Promise<InterceptResult> {\n const { gates: allGates } = resolveGatesForColumn(this.config, complete.currentColumnName);\n const waivers = await this.deps.getTicketGateWaivers(complete.ticketId);\n const waiverSet = new Set(waivers);\n const gates = allGates.filter((g) => !waiverSet.has(g.name));\n\n let results: GateResult[];\n try {\n const opts = this.buildRunOptions();\n const ticketEnv = await this.buildTicketEnv(complete.ticketId, complete.projectId, complete.currentColumnName);\n opts.env = { ...opts.env, ...ticketEnv };\n results = await this.deps.runGates(gates, opts);\n } catch (err) {\n return {\n error: 'GATE_FAILURE',\n message: `Gate evaluation error: ${err instanceof Error ? err.message : String(err)}`,\n hint: 'Fix the gate environment and retry',\n };\n }\n\n const requiredFailures = results.filter((r) => r.required && !r.passed);\n if (requiredFailures.length > 0) {\n return {\n error: 'GATE_FAILURE',\n message: `Cannot complete task — ${requiredFailures.length} required gate(s) failed`,\n formatted: formatGateErrors(results.filter((r) => r.required)),\n results,\n hint: 'Fix the failing gate(s) and try complete_task again',\n };\n }\n\n const forwardResult = await this.deps.forwardCompleteTask({\n ...complete.args,\n projectId: complete.projectId,\n ticketId: complete.ticketId,\n });\n\n return { forwardResult };\n }\n}\n"],"mappings":"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","names":["deps","projectId","columnName","forwardResult","projectId"]}
|