@debugmcp/mcp-debugger
Version:
Run-time step-through debugging for LLM agents.
111 lines (99 loc) • 4.88 kB
text/typescript
/**
* Python Adapter Factory
*
* Factory for creating Python debug adapter instances.
* Implements the adapter factory interface for dependency injection.
*
* @since 2.0.0
*/
import { IDebugAdapter } from '../debug-adapter-interface.js';
import { IAdapterFactory, AdapterDependencies, AdapterMetadata, FactoryValidationResult } from '../adapter-registry-interface.js';
import { PythonDebugAdapter } from './python-debug-adapter.js';
import { DebugLanguage } from '../../session/models.js';
import { findPythonExecutable, getPythonVersion } from '../../utils/python-utils.js';
import { spawn } from 'child_process';
/**
* Factory for creating Python debug adapters
*/
export class PythonAdapterFactory implements IAdapterFactory {
/**
* Create a new Python debug adapter instance
*/
createAdapter(dependencies: AdapterDependencies): IDebugAdapter {
return new PythonDebugAdapter(dependencies);
}
/**
* Get metadata about the Python adapter
*/
getMetadata(): AdapterMetadata {
return {
language: DebugLanguage.PYTHON,
displayName: 'Python',
version: '2.0.0',
author: 'mcp-debugger team',
description: 'Debug Python applications using debugpy',
documentationUrl: 'https://github.com/debugmcp/mcp-debugger/docs/python',
minimumDebuggerVersion: '1.0.0',
fileExtensions: ['.py', '.pyw'],
icon: ''
};
}
/**
* Validate that the factory can create adapters in current environment
*/
async validate(): Promise<FactoryValidationResult> {
const errors: string[] = [];
const warnings: string[] = [];
let pythonPath: string | undefined;
let pythonVersion: string | undefined;
try {
// Check Python executable
pythonPath = await findPythonExecutable();
// Check Python version
pythonVersion = await getPythonVersion(pythonPath) || undefined;
if (pythonVersion) {
const [major, minor] = pythonVersion.split('.').map(Number);
if (major < 3 || (major === 3 && minor < 7)) {
errors.push(`Python 3.7 or higher required. Current version: ${pythonVersion}`);
}
} else {
warnings.push('Could not determine Python version');
}
// Check debugpy installation
const hasDebugpy = await this.checkDebugpyInstalled(pythonPath);
if (!hasDebugpy) {
errors.push('debugpy not installed. Run: pip install debugpy');
}
} catch (error) {
errors.push(error instanceof Error ? error.message : 'Python executable not found');
}
return {
valid: errors.length === 0,
errors,
warnings,
details: {
pythonPath,
pythonVersion,
pythonDetectionMethod: process.platform === 'win32' ? 'py launcher' : 'python3',
platform: process.platform,
timestamp: new Date().toISOString()
}
};
}
/**
* Check if debugpy is installed
*/
private checkDebugpyInstalled(pythonPath: string): Promise<boolean> {
return new Promise((resolve) => {
const child = spawn(pythonPath, ['-c', 'import debugpy; print(debugpy.__version__)'], {
stdio: ['ignore', 'pipe', 'pipe']
});
let output = '';
child.stdout?.on('data', (data) => { output += data.toString(); });
child.on('error', () => resolve(false));
child.on('exit', (code) => {
resolve(code === 0 && output.trim().length > 0);
});
});
}
}