| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157 |
- /**
- * MCP `initialize` handshake regression tests.
- *
- * Issue #172: on slow filesystems (Docker Desktop VirtioFS on macOS, WSL2),
- * the MCP server was blocking the initialize response on CodeGraph.open() and
- * Parser.init() (web-tree-sitter WASM bootstrap), which could take longer than
- * Claude Code's ~30s handshake timeout. The child process stayed alive and
- * had received the request, but never sent a response, so tools never
- * appeared in the client. The fix sends the initialize response before
- * kicking off the heavy init in the background. These tests guard the
- * contract that initialize is fast regardless of how much work init does.
- */
- import { describe, it, expect, beforeEach, afterEach } from 'vitest';
- import { spawn, ChildProcessWithoutNullStreams } from 'child_process';
- import * as fs from 'fs';
- import * as path from 'path';
- import * as os from 'os';
- import { CodeGraph } from '../src';
- const BIN = path.resolve(__dirname, '../dist/bin/codegraph.js');
- function spawnServer(cwd: string): ChildProcessWithoutNullStreams {
- return spawn(process.execPath, [BIN, 'serve', '--mcp'], {
- cwd,
- stdio: ['pipe', 'pipe', 'pipe'],
- // Pin to direct (in-process) mode. #172 is a contract about the in-process
- // server's init ordering — the "File watcher active" log this test observes
- // is emitted in-process. In daemon mode the watcher runs in the detached
- // daemon (logging to .codegraph/daemon.log, not the child's stderr); the
- // same response-before-init guarantee lives in the shared session code and
- // is covered by mcp-daemon.test.ts. Direct mode also avoids leaking a
- // detached daemon from this suite.
- env: { ...process.env, CODEGRAPH_NO_DAEMON: '1' },
- }) as ChildProcessWithoutNullStreams;
- }
- function sendInitialize(child: ChildProcessWithoutNullStreams, projectPath: string) {
- const msg = JSON.stringify({
- jsonrpc: '2.0',
- id: 0,
- method: 'initialize',
- params: {
- protocolVersion: '2025-11-25',
- capabilities: {},
- clientInfo: { name: 'test', version: '0.0.0' },
- rootUri: `file://${projectPath}`,
- },
- });
- child.stdin.write(msg + '\n');
- }
- /**
- * Collect stdout lines and stderr text from the child, tagging each piece
- * with a monotonic sequence number. Lets us assert ordering between the
- * JSON-RPC response (stdout) and side-effect logs (stderr).
- */
- function tagStreams(child: ChildProcessWithoutNullStreams) {
- const events: Array<{ seq: number; stream: 'stdout' | 'stderr'; text: string }> = [];
- let seq = 0;
- let stdoutBuf = '';
- let stderrBuf = '';
- child.stdout.on('data', (chunk) => {
- stdoutBuf += chunk.toString('utf8');
- let idx;
- while ((idx = stdoutBuf.indexOf('\n')) !== -1) {
- const line = stdoutBuf.slice(0, idx);
- stdoutBuf = stdoutBuf.slice(idx + 1);
- events.push({ seq: seq++, stream: 'stdout', text: line });
- }
- });
- child.stderr.on('data', (chunk) => {
- stderrBuf += chunk.toString('utf8');
- let idx;
- while ((idx = stderrBuf.indexOf('\n')) !== -1) {
- const line = stderrBuf.slice(0, idx);
- stderrBuf = stderrBuf.slice(idx + 1);
- events.push({ seq: seq++, stream: 'stderr', text: line });
- }
- });
- return events;
- }
- function waitFor<T>(
- events: ReadonlyArray<{ seq: number; stream: string; text: string }>,
- predicate: (e: { seq: number; stream: string; text: string }) => boolean,
- timeoutMs: number,
- ): Promise<{ seq: number; stream: string; text: string }> {
- return new Promise((resolve, reject) => {
- const started = Date.now();
- const tick = () => {
- const hit = events.find(predicate);
- if (hit) return resolve(hit);
- if (Date.now() - started > timeoutMs) {
- return reject(new Error(`Timed out waiting for predicate. Events: ${JSON.stringify(events)}`));
- }
- setTimeout(tick, 20);
- };
- tick();
- });
- }
- describe('MCP initialize handshake (issue #172)', () => {
- let tempDir: string;
- let child: ChildProcessWithoutNullStreams | null = null;
- beforeEach(() => {
- tempDir = fs.mkdtempSync(path.join(os.tmpdir(), 'codegraph-mcp-init-'));
- });
- afterEach(() => {
- if (child && !child.killed) {
- child.kill('SIGKILL');
- child = null;
- }
- fs.rmSync(tempDir, { recursive: true, force: true });
- });
- it('responds to initialize quickly when no .codegraph exists in cwd', async () => {
- child = spawnServer(tempDir);
- const events = tagStreams(child);
- sendInitialize(child, tempDir);
- const response = await waitFor(events, (e) => e.stream === 'stdout', 5000);
- const json = JSON.parse(response.text);
- expect(json.jsonrpc).toBe('2.0');
- expect(json.id).toBe(0);
- expect(json.result.protocolVersion).toBeDefined();
- expect(json.result.capabilities.tools).toBeDefined();
- }, 10000);
- it('sends initialize response BEFORE tryInitializeDefault finishes', async () => {
- // Seed a real .codegraph so the server's tryInitializeDefault path runs
- // its full body: CodeGraph.open() (which awaits initGrammars()) and then
- // startWatching() (which logs "File watcher active" to stderr). On any
- // platform, that stderr log is observable evidence that tryInitializeDefault
- // has completed. The contract we're protecting: the JSON-RPC response on
- // stdout must arrive BEFORE that stderr log. If a future change re-awaits
- // tryInitializeDefault before sendResult, this ordering inverts and the
- // test fails — regardless of how fast the local filesystem is.
- const cg = await CodeGraph.init(tempDir);
- cg.close();
- child = spawnServer(tempDir);
- const events = tagStreams(child);
- sendInitialize(child, tempDir);
- const response = await waitFor(events, (e) => e.stream === 'stdout', 10000);
- const watcherLog = await waitFor(
- events,
- (e) => e.stream === 'stderr' && e.text.includes('File watcher active'),
- 10000,
- );
- expect(response.seq).toBeLessThan(watcherLog.seq);
- const json = JSON.parse(response.text);
- expect(json.id).toBe(0);
- expect(json.result.serverInfo.name).toBe('codegraph');
- }, 20000);
- });
|