| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644 |
- /**
- * Security Tests
- *
- * Tests for P0/P1 security fixes:
- * - FileLock (cross-process locking)
- * - Path traversal prevention
- * - MCP input validation
- * - Atomic writes
- */
- import { describe, it, expect, beforeEach, afterEach } from 'vitest';
- import * as fs from 'fs';
- import * as path from 'path';
- import * as os from 'os';
- import { FileLock, validateProjectPath } from '../src/utils';
- import CodeGraph from '../src/index';
- import { ToolHandler, tools } from '../src/mcp/tools';
- import { scanDirectory, isSourceFile } from '../src/extraction';
- import { DatabaseConnection, getDatabasePath } from '../src/db';
- import { QueryBuilder } from '../src/db/queries';
- function createTempDir(): string {
- return fs.mkdtempSync(path.join(os.tmpdir(), 'codegraph-security-test-'));
- }
- function cleanupTempDir(dir: string): void {
- if (fs.existsSync(dir)) {
- fs.rmSync(dir, { recursive: true, force: true });
- }
- }
- describe('FileLock', () => {
- let tempDir: string;
- let lockPath: string;
- beforeEach(() => {
- tempDir = createTempDir();
- lockPath = path.join(tempDir, 'test.lock');
- });
- afterEach(() => {
- cleanupTempDir(tempDir);
- });
- it('should acquire and release a lock', () => {
- const lock = new FileLock(lockPath);
- lock.acquire();
- expect(fs.existsSync(lockPath)).toBe(true);
- const content = fs.readFileSync(lockPath, 'utf-8').trim();
- expect(parseInt(content, 10)).toBe(process.pid);
- lock.release();
- expect(fs.existsSync(lockPath)).toBe(false);
- });
- it('should prevent double acquisition within same process', () => {
- const lock1 = new FileLock(lockPath);
- const lock2 = new FileLock(lockPath);
- lock1.acquire();
- // Second lock should fail because our PID is alive
- expect(() => lock2.acquire()).toThrow(/locked by another process/);
- lock1.release();
- });
- it('should detect and remove stale locks from dead processes', () => {
- // Write a lock file with a PID that doesn't exist
- // PID 99999999 is extremely unlikely to be a real process
- fs.writeFileSync(lockPath, '99999999');
- const lock = new FileLock(lockPath);
- // Should succeed because the PID is dead
- expect(() => lock.acquire()).not.toThrow();
- lock.release();
- });
- it('should execute function with withLock', () => {
- const lock = new FileLock(lockPath);
- const result = lock.withLock(() => {
- expect(fs.existsSync(lockPath)).toBe(true);
- return 42;
- });
- expect(result).toBe(42);
- expect(fs.existsSync(lockPath)).toBe(false);
- });
- it('should release lock even if function throws', () => {
- const lock = new FileLock(lockPath);
- expect(() => {
- lock.withLock(() => {
- throw new Error('test error');
- });
- }).toThrow('test error');
- expect(fs.existsSync(lockPath)).toBe(false);
- });
- it('should execute async function with withLockAsync', async () => {
- const lock = new FileLock(lockPath);
- const result = await lock.withLockAsync(async () => {
- expect(fs.existsSync(lockPath)).toBe(true);
- return 'async-result';
- });
- expect(result).toBe('async-result');
- expect(fs.existsSync(lockPath)).toBe(false);
- });
- it('should release lock even if async function throws', async () => {
- const lock = new FileLock(lockPath);
- await expect(
- lock.withLockAsync(async () => {
- throw new Error('async error');
- })
- ).rejects.toThrow('async error');
- expect(fs.existsSync(lockPath)).toBe(false);
- });
- it('release should be idempotent', () => {
- const lock = new FileLock(lockPath);
- lock.acquire();
- lock.release();
- // Second release should not throw
- expect(() => lock.release()).not.toThrow();
- });
- });
- describe('Path Traversal Prevention', () => {
- let testDir: string;
- let cg: CodeGraph;
- beforeEach(async () => {
- testDir = createTempDir();
- const srcDir = path.join(testDir, 'src');
- fs.mkdirSync(srcDir);
- fs.writeFileSync(
- path.join(srcDir, 'hello.ts'),
- `export function hello(): string { return "hi"; }\n`
- );
- cg = CodeGraph.initSync(testDir, {
- config: { include: ['**/*.ts'], exclude: [] },
- });
- await cg.indexAll();
- });
- afterEach(() => {
- if (cg) cg.close();
- cleanupTempDir(testDir);
- });
- it('should read code for valid nodes within project', async () => {
- const nodes = cg.getNodesByKind('function');
- const hello = nodes.find((n) => n.name === 'hello');
- expect(hello).toBeDefined();
- const code = await cg.getCode(hello!.id);
- expect(code).toContain('hello');
- });
- it('should return null for non-existent node', async () => {
- const code = await cg.getCode('does-not-exist');
- expect(code).toBeNull();
- });
- });
- describe('validateProjectPath — sensitive directory blocking', () => {
- // POSIX-only: on Windows '/etc' resolves to C:\etc (non-existent), not a
- // sensitive dir — the Windows case is covered by the win32-gated test below.
- it.runIf(process.platform !== 'win32')('blocks POSIX system directories (exact match)', () => {
- expect(validateProjectPath('/')).toMatch(/sensitive system directory/i);
- expect(validateProjectPath('/etc')).toMatch(/sensitive system directory/i);
- });
- it('allows a normal, existing directory', () => {
- const dir = fs.mkdtempSync(path.join(os.tmpdir(), 'cg-validate-'));
- try {
- expect(validateProjectPath(dir)).toBeNull();
- } finally {
- fs.rmSync(dir, { recursive: true, force: true });
- }
- });
- // SENSITIVE_PATHS stores the Windows entries lowercase and validateProjectPath
- // matches via resolved.toLowerCase(), so 'C:\\Windows' and 'c:\\windows' are
- // both blocked. path.resolve is platform-specific, so this only runs on Windows.
- it.runIf(process.platform === 'win32')(
- 'blocks Windows system directories regardless of case',
- () => {
- expect(validateProjectPath('C:\\Windows')).toMatch(/sensitive system directory/i);
- expect(validateProjectPath('c:\\windows')).toMatch(/sensitive system directory/i);
- expect(validateProjectPath('C:\\WINDOWS\\System32')).toMatch(/sensitive system directory/i);
- }
- );
- });
- describe('MCP Input Validation', () => {
- let testDir: string;
- let cg: CodeGraph;
- let handler: ToolHandler;
- beforeEach(async () => {
- testDir = createTempDir();
- const srcDir = path.join(testDir, 'src');
- fs.mkdirSync(srcDir);
- fs.writeFileSync(
- path.join(srcDir, 'example.ts'),
- `export function exampleFunc(): void {}\nexport class ExampleClass {}\n`
- );
- cg = CodeGraph.initSync(testDir, {
- config: { include: ['**/*.ts'], exclude: [] },
- });
- await cg.indexAll();
- handler = new ToolHandler(cg);
- });
- afterEach(() => {
- if (cg) cg.close();
- cleanupTempDir(testDir);
- });
- it('should reject non-string query in codegraph_search', async () => {
- const result = await handler.execute('codegraph_search', { query: null });
- expect(result.isError).toBe(true);
- expect(result.content[0].text).toContain('non-empty string');
- });
- it('should reject empty string query in codegraph_search', async () => {
- const result = await handler.execute('codegraph_search', { query: '' });
- expect(result.isError).toBe(true);
- expect(result.content[0].text).toContain('non-empty string');
- });
- it('should accept valid query in codegraph_search', async () => {
- const result = await handler.execute('codegraph_search', { query: 'example' });
- expect(result.isError).toBeFalsy();
- });
- it('should clamp limit to valid range in codegraph_search', async () => {
- // Extremely large limit should still work (clamped to 100)
- const result = await handler.execute('codegraph_search', { query: 'example', limit: 999999 });
- expect(result.isError).toBeFalsy();
- });
- it('should reject non-string symbol in codegraph_callers', async () => {
- const result = await handler.execute('codegraph_callers', { symbol: 123 });
- expect(result.isError).toBe(true);
- expect(result.content[0].text).toContain('non-empty string');
- });
- it('should reject non-string task in codegraph_context', async () => {
- const result = await handler.execute('codegraph_context', { task: undefined });
- expect(result.isError).toBe(true);
- expect(result.content[0].text).toContain('non-empty string');
- });
- it('should truncate oversized codegraph_context output', async () => {
- const oversizedContext = Array.from({ length: 400 }, (_, i) => `line-${i} ${'x'.repeat(80)}`).join('\n');
- const fakeCg = {
- buildContext: async () => oversizedContext,
- };
- const fakeHandler = new ToolHandler(fakeCg as unknown as CodeGraph);
- const result = await fakeHandler.execute('codegraph_context', { task: 'find example' });
- expect(result.isError).toBeFalsy();
- expect(result.content[0].text.length).toBeLessThan(oversizedContext.length);
- expect(result.content[0].text).toContain('... (output truncated)');
- });
- it('should reject non-string symbol in codegraph_impact', async () => {
- const result = await handler.execute('codegraph_impact', { symbol: [] });
- expect(result.isError).toBe(true);
- });
- it('should reject non-string symbol in codegraph_node', async () => {
- const result = await handler.execute('codegraph_node', { symbol: false });
- expect(result.isError).toBe(true);
- });
- it('should reject non-string symbol in codegraph_callees', async () => {
- const result = await handler.execute('codegraph_callees', { symbol: {} });
- expect(result.isError).toBe(true);
- });
- it('should handle NaN limit gracefully', async () => {
- const result = await handler.execute('codegraph_search', { query: 'example', limit: 'abc' });
- expect(result.isError).toBeFalsy();
- });
- it('should handle negative limit gracefully', async () => {
- const result = await handler.execute('codegraph_search', { query: 'example', limit: -5 });
- expect(result.isError).toBeFalsy();
- });
- // #230: getCodeGraph must reject a sensitive system directory passed as
- // projectPath before opening it. The error surfaces through execute()'s
- // catch as an isError result. /etc is sensitive on POSIX; C:\Windows on
- // Windows (path.resolve is platform-specific, so each case is gated).
- it.runIf(process.platform !== 'win32')(
- 'rejects a sensitive POSIX projectPath (/etc) via the MCP handler',
- async () => {
- const result = await handler.execute('codegraph_search', {
- query: 'example',
- projectPath: '/etc',
- });
- expect(result.isError).toBe(true);
- expect(result.content[0].text).toMatch(/sensitive system directory/i);
- }
- );
- it.runIf(process.platform === 'win32')(
- 'rejects a sensitive Windows projectPath (C:\\Windows) via the MCP handler',
- async () => {
- const result = await handler.execute('codegraph_search', {
- query: 'example',
- projectPath: 'C:\\Windows',
- });
- expect(result.isError).toBe(true);
- expect(result.content[0].text).toMatch(/sensitive system directory/i);
- }
- );
- });
- describe('Atomic Writes', () => {
- let tempDir: string;
- beforeEach(() => {
- tempDir = createTempDir();
- });
- afterEach(() => {
- cleanupTempDir(tempDir);
- });
- it('should not leave temp files on success', () => {
- // We test this indirectly through the config-writer module
- // by checking that no .tmp files remain after writing
- const configDir = path.join(tempDir, '.claude');
- fs.mkdirSync(configDir, { recursive: true });
- const testFile = path.join(configDir, 'test.json');
- // Simulate what atomicWriteFileSync does
- const tmpPath = testFile + '.tmp.' + process.pid;
- fs.writeFileSync(tmpPath, '{"test": true}');
- fs.renameSync(tmpPath, testFile);
- expect(fs.existsSync(testFile)).toBe(true);
- expect(fs.existsSync(tmpPath)).toBe(false);
- const content = JSON.parse(fs.readFileSync(testFile, 'utf-8'));
- expect(content.test).toBe(true);
- });
- });
- describe('Source file detection (isSourceFile)', () => {
- it('selects files by supported extension', () => {
- expect(isSourceFile('src/index.ts')).toBe(true);
- expect(isSourceFile('src/deep/nested/file.ts')).toBe(true);
- expect(isSourceFile('src/component.tsx')).toBe(true);
- expect(isSourceFile('lib/util.js')).toBe(true);
- expect(isSourceFile('src/main.py')).toBe(true);
- });
- it('rejects unsupported extensions and extensionless files', () => {
- expect(isSourceFile('src/component.css')).toBe(false);
- expect(isSourceFile('README.md')).toBe(false);
- expect(isSourceFile('Makefile')).toBe(false);
- expect(isSourceFile('.gitignore')).toBe(false);
- });
- it('matches regardless of leading dot directories', () => {
- expect(isSourceFile('.hidden/index.ts')).toBe(true);
- });
- });
- describe('JSON.parse Error Boundaries in DB', () => {
- let tempDir: string;
- beforeEach(() => {
- tempDir = createTempDir();
- });
- afterEach(() => {
- cleanupTempDir(tempDir);
- });
- it('should not crash when node has malformed JSON in decorators column', () => {
- const dbPath = path.join(tempDir, 'test.db');
- const db = DatabaseConnection.initialize(dbPath);
- const queries = new QueryBuilder(db.getDb());
- // Insert a node with malformed JSON in the decorators column
- db.getDb().prepare(`
- INSERT INTO nodes (id, kind, name, qualified_name, file_path, language, start_line, end_line, start_column, end_column, decorators, is_exported, is_async, is_static, is_abstract, updated_at)
- VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
- `).run(
- 'test-node-1', 'function', 'myFunc', 'myFunc', 'test.ts', 'typescript',
- 1, 5, 0, 0,
- '{not valid json!!!}', // malformed decorators
- 0, 0, 0, 0, Date.now()
- );
- // Should not throw - should return node with undefined decorators
- const node = queries.getNodeById('test-node-1');
- expect(node).not.toBeNull();
- expect(node!.name).toBe('myFunc');
- expect(node!.decorators).toBeUndefined();
- db.close();
- });
- it('should not crash when edge has malformed JSON in metadata column', () => {
- const dbPath = path.join(tempDir, 'test.db');
- const db = DatabaseConnection.initialize(dbPath);
- const queries = new QueryBuilder(db.getDb());
- // Insert two nodes first
- const insertNode = db.getDb().prepare(`
- INSERT INTO nodes (id, kind, name, qualified_name, file_path, language, start_line, end_line, start_column, end_column, is_exported, is_async, is_static, is_abstract, updated_at)
- VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
- `);
- insertNode.run('node-a', 'function', 'funcA', 'funcA', 'a.ts', 'typescript', 1, 5, 0, 0, 0, 0, 0, 0, Date.now());
- insertNode.run('node-b', 'function', 'funcB', 'funcB', 'b.ts', 'typescript', 1, 5, 0, 0, 0, 0, 0, 0, Date.now());
- // Insert edge with malformed metadata
- db.getDb().prepare(`
- INSERT INTO edges (source, target, kind, metadata)
- VALUES (?, ?, ?, ?)
- `).run('node-a', 'node-b', 'calls', 'broken json {{{');
- // Should not throw - should return edge with undefined metadata
- const edges = queries.getOutgoingEdges('node-a');
- expect(edges.length).toBe(1);
- expect(edges[0].source).toBe('node-a');
- expect(edges[0].target).toBe('node-b');
- expect(edges[0].metadata).toBeUndefined();
- db.close();
- });
- it('should not crash when file record has malformed JSON in errors column', () => {
- const dbPath = path.join(tempDir, 'test.db');
- const db = DatabaseConnection.initialize(dbPath);
- const queries = new QueryBuilder(db.getDb());
- // Insert a file with malformed errors JSON
- db.getDb().prepare(`
- INSERT INTO files (path, content_hash, language, size, modified_at, indexed_at, node_count, errors)
- VALUES (?, ?, ?, ?, ?, ?, ?, ?)
- `).run('test.ts', 'abc123', 'typescript', 100, Date.now(), Date.now(), 5, 'not-an-array');
- // Should not throw - should return file with undefined errors
- const file = queries.getFileByPath('test.ts');
- expect(file).not.toBeNull();
- expect(file!.path).toBe('test.ts');
- expect(file!.errors).toBeUndefined();
- db.close();
- });
- });
- describe('Symlink Cycle Detection', () => {
- let tempDir: string;
- beforeEach(() => {
- tempDir = createTempDir();
- });
- afterEach(() => {
- cleanupTempDir(tempDir);
- });
- it('should handle symlink cycle without infinite loop', () => {
- // Create directory structure with a symlink cycle
- const srcDir = path.join(tempDir, 'src');
- fs.mkdirSync(srcDir);
- fs.writeFileSync(path.join(srcDir, 'index.ts'), 'export const x = 1;\n');
- // Create a symlink from src/loop -> tempDir (parent directory)
- try {
- fs.symlinkSync(tempDir, path.join(srcDir, 'loop'), 'dir');
- } catch {
- // Skip test if symlinks not supported (e.g., Windows without admin)
- return;
- }
- // This should complete without hanging
- const files = scanDirectory(tempDir);
- // Should find the real file but not loop infinitely
- expect(files).toContain('src/index.ts');
- // Should not find duplicates via the symlink path
- const indexFiles = files.filter(f => f.endsWith('index.ts'));
- expect(indexFiles.length).toBe(1);
- });
- it('should follow valid symlinks to directories', () => {
- // Create source directory with a file
- const realDir = path.join(tempDir, 'real');
- fs.mkdirSync(realDir);
- fs.writeFileSync(path.join(realDir, 'hello.ts'), 'export function hello() {}\n');
- // Create a symlink to realDir
- const srcDir = path.join(tempDir, 'src');
- fs.mkdirSync(srcDir);
- try {
- fs.symlinkSync(realDir, path.join(srcDir, 'linked'), 'dir');
- } catch {
- return;
- }
- const files = scanDirectory(tempDir);
- // Should find files from both the real dir and via the symlink
- // But deduplicate since they resolve to the same real path
- expect(files.some(f => f.includes('hello.ts'))).toBe(true);
- });
- it('should skip broken symlinks gracefully', () => {
- const srcDir = path.join(tempDir, 'src');
- fs.mkdirSync(srcDir);
- fs.writeFileSync(path.join(srcDir, 'valid.ts'), 'export const y = 2;\n');
- try {
- fs.symlinkSync('/nonexistent/path', path.join(srcDir, 'broken'), 'dir');
- } catch {
- return;
- }
- // Should not throw
- const files = scanDirectory(tempDir);
- expect(files).toContain('src/valid.ts');
- });
- });
- describe('Session marker symlink resistance', () => {
- // The marker write lives in src/mcp/tools.ts behind handleContext. We exercise
- // it end-to-end via ToolHandler.execute so the test exercises the same code
- // path Claude Code drives. The session id is per-test so other parallel test
- // runs can't collide with the marker file we plant a symlink at.
- const SESSION_ID = `cg-test-${process.pid}-${Date.now()}-${Math.random().toString(36).slice(2)}`;
- const crypto = require('crypto') as typeof import('crypto');
- const hash = crypto.createHash('md5').update(SESSION_ID).digest('hex').slice(0, 16);
- const markerPath = path.join(os.tmpdir(), `codegraph-consulted-${hash}`);
- let projectDir: string;
- let victimDir: string;
- let victimFile: string;
- beforeEach(async () => {
- projectDir = createTempDir();
- victimDir = createTempDir();
- victimFile = path.join(victimDir, 'private.txt');
- fs.writeFileSync(victimFile, 'SECRET-DO-NOT-OVERWRITE\n');
- if (fs.existsSync(markerPath)) fs.unlinkSync(markerPath);
- // A real .codegraph/ has to exist for handleContext to get past the
- // "not initialized" guard — index a tiny fixture so the call reaches the
- // marker write step rather than short-circuiting on missing project state.
- fs.writeFileSync(path.join(projectDir, 'a.ts'), 'export const x = 1;\n');
- const cg = await CodeGraph.init(projectDir);
- await cg.indexAll();
- cg.close();
- });
- afterEach(() => {
- if (fs.existsSync(markerPath)) fs.unlinkSync(markerPath);
- cleanupTempDir(projectDir);
- cleanupTempDir(victimDir);
- });
- it('does not follow a pre-planted symlink at the marker path', async () => {
- // Skip on platforms where the user can't create symlinks (Windows without
- // dev mode + admin). The CWE-59 risk we're guarding against doesn't apply
- // when symlinks aren't creatable, so the skip is correct, not a gap.
- try {
- fs.symlinkSync(victimFile, markerPath);
- } catch {
- return;
- }
- const cg = await CodeGraph.open(projectDir);
- const handler = new ToolHandler(cg);
- process.env.CLAUDE_SESSION_ID = SESSION_ID;
- try {
- await handler.execute('codegraph_context', { task: 'find x' });
- } finally {
- delete process.env.CLAUDE_SESSION_ID;
- cg.close();
- }
- // The victim file's contents must be untouched — the old writeFileSync
- // path would have followed the symlink and written an ISO timestamp here.
- expect(fs.readFileSync(victimFile, 'utf8')).toBe('SECRET-DO-NOT-OVERWRITE\n');
- // And the marker path itself must still be the symlink we planted —
- // no fallback path that quietly unlinked + recreated it (which would
- // also work, but is a behavior we don't want to silently rely on).
- expect(fs.lstatSync(markerPath).isSymbolicLink()).toBe(true);
- });
- it('writes the marker file with 0o600 perms on a clean path', async () => {
- // No symlink planted — happy path. Verifies the new openSync(mode: 0o600)
- // call is what actually lands on disk (regression guard for the perm
- // tightening that came with the O_NOFOLLOW fix).
- const cg = await CodeGraph.open(projectDir);
- const handler = new ToolHandler(cg);
- process.env.CLAUDE_SESSION_ID = SESSION_ID;
- try {
- await handler.execute('codegraph_context', { task: 'find x' });
- } finally {
- delete process.env.CLAUDE_SESSION_ID;
- cg.close();
- }
- expect(fs.existsSync(markerPath)).toBe(true);
- // chmod's low 9 bits — strip the file-type bits for a clean compare.
- // Windows can't enforce 0o600 in the POSIX sense; skip the assertion
- // there since the underlying OS will normalize the mode anyway.
- if (process.platform !== 'win32') {
- const mode = fs.statSync(markerPath).mode & 0o777;
- expect(mode).toBe(0o600);
- }
- });
- });
|