VS Code Extension Refactorer
Overview
This skill enables systematic and safe refactoring of VS Code extension code. It provides structured workflows for identifying improvement opportunities, applying proven refactoring techniques, and ensuring code quality while maintaining functionality.
When to Use This Skill
-
Extracting classes, methods, or modules from large files
-
Reducing code duplication across the codebase
-
Improving TypeScript type safety and interfaces
-
Reorganizing module structure and dependencies
-
Applying design patterns (Manager, Coordinator, Factory, etc.)
-
Optimizing performance-critical code paths
-
Preparing code for new feature development
-
Cleaning up after rapid prototyping
Refactoring Workflow
Phase 1: Analysis and Planning
Code Smell Detection
Smell Indicators Refactoring
Long Method
50 lines, multiple responsibilities Extract Method
Large Class
500 lines, >10 public methods Extract Class
Duplicate Code Similar blocks in 2+ places Extract Function/Class
Feature Envy Method uses other class data extensively Move Method
Data Clumps Same parameters passed together Introduce Parameter Object
Primitive Obsession Overuse of primitives for domain concepts Replace with Value Object
Long Parameter List
4 parameters Introduce Parameter Object
Divergent Change Class changes for multiple reasons Extract Class (SRP)
Shotgun Surgery One change requires many file edits Move Method, Inline Class
God Class Class knows/does too much Extract Class, Delegate
Impact Assessment
// Before refactoring, assess: interface RefactoringAssessment { // Scope filesAffected: string[]; publicApiChanges: boolean; breakingChanges: boolean;
// Risk testCoverage: 'high' | 'medium' | 'low'; complexity: 'high' | 'medium' | 'low';
// Dependencies internalDependents: string[]; externalDependents: string[]; // Other extensions, APIs }
Phase 2: Safe Refactoring Techniques
Extract Method
Before:
async function processTerminals(): Promise<void> { // Validation logic (10 lines) if (!this.terminals) { throw new Error('Terminals not initialized'); } if (this.terminals.size === 0) { console.log('No terminals to process'); return; }
// Processing logic (20 lines) for (const [id, terminal] of this.terminals) { const state = terminal.getState(); if (state === 'running') { await terminal.sendCommand('status'); const output = await terminal.waitForOutput(); this.results.set(id, output); } }
// Cleanup logic (10 lines) this.results.forEach((result, id) => { if (result.includes('error')) { this.terminals.get(id)?.restart(); } }); }
After:
async function processTerminals(): Promise<void> { this.validateTerminals(); await this.collectTerminalStatuses(); this.handleErrorResults(); }
private validateTerminals(): void { if (!this.terminals) { throw new Error('Terminals not initialized'); } if (this.terminals.size === 0) { console.log('No terminals to process'); return; } }
private async collectTerminalStatuses(): Promise<void> { for (const [id, terminal] of this.terminals) { const state = terminal.getState(); if (state === 'running') { await terminal.sendCommand('status'); const output = await terminal.waitForOutput(); this.results.set(id, output); } } }
private handleErrorResults(): void { this.results.forEach((result, id) => { if (result.includes('error')) { this.terminals.get(id)?.restart(); } }); }
Extract Class
Before (God Class):
class TerminalManager { // Terminal management (proper responsibility) private terminals: Map<number, Terminal>; createTerminal(): Terminal { } disposeTerminal(id: number): void { }
// UI concerns (wrong place) private statusBarItem: vscode.StatusBarItem; updateStatusBar(): void { } showNotification(msg: string): void { }
// Persistence concerns (wrong place) saveState(): void { } loadState(): void { } migrateOldState(): void { }
// Configuration concerns (wrong place) getConfig(key: string): unknown { } updateConfig(key: string, value: unknown): void { } }
After (Single Responsibility):
// Core terminal management class TerminalManager { private terminals: Map<number, Terminal>;
constructor( private ui: TerminalUIManager, private persistence: TerminalPersistence, private config: TerminalConfig ) {}
createTerminal(): Terminal { const terminal = new Terminal(this.config.getShellPath()); this.terminals.set(terminal.id, terminal); this.ui.updateStatusBar(this.terminals.size); this.persistence.saveState(this.getState()); return terminal; }
disposeTerminal(id: number): void { this.terminals.delete(id); this.ui.updateStatusBar(this.terminals.size); this.persistence.saveState(this.getState()); } }
// UI responsibility class TerminalUIManager { private statusBarItem: vscode.StatusBarItem;
updateStatusBar(count: number): void { } showNotification(msg: string): void { } }
// Persistence responsibility class TerminalPersistence { saveState(state: TerminalState): void { } loadState(): TerminalState { } migrateOldState(): void { } }
// Configuration responsibility class TerminalConfig { getShellPath(): string { } get(key: string): unknown { } update(key: string, value: unknown): void { } }
Replace Conditional with Polymorphism
Before:
function handleMessage(message: Message): void { switch (message.type) { case 'create': const terminal = createTerminal(message.config); sendResponse({ type: 'created', id: terminal.id }); break; case 'write': const t = getTerminal(message.id); if (t) t.write(message.data); break; case 'resize': const term = getTerminal(message.id); if (term) term.resize(message.cols, message.rows); break; case 'dispose': disposeTerminal(message.id); sendResponse({ type: 'disposed', id: message.id }); break; default: console.warn('Unknown message type:', message.type); } }
After:
interface MessageHandler { handle(message: Message): void; }
class CreateHandler implements MessageHandler { handle(message: CreateMessage): void { const terminal = this.manager.createTerminal(message.config); this.sender.send({ type: 'created', id: terminal.id }); } }
class WriteHandler implements MessageHandler { handle(message: WriteMessage): void { this.manager.getTerminal(message.id)?.write(message.data); } }
class ResizeHandler implements MessageHandler { handle(message: ResizeMessage): void { this.manager.getTerminal(message.id)?.resize(message.cols, message.rows); } }
class DisposeHandler implements MessageHandler { handle(message: DisposeMessage): void { this.manager.disposeTerminal(message.id); this.sender.send({ type: 'disposed', id: message.id }); } }
// Message router class MessageRouter { private handlers = new Map<string, MessageHandler>([ ['create', new CreateHandler()], ['write', new WriteHandler()], ['resize', new ResizeHandler()], ['dispose', new DisposeHandler()] ]);
route(message: Message): void { const handler = this.handlers.get(message.type); if (handler) { handler.handle(message); } else { console.warn('Unknown message type:', message.type); } } }
Introduce Parameter Object
Before:
function createTerminal( shell: string, cwd: string, env: Record<string, string>, cols: number, rows: number, scrollback: number, name?: string ): Terminal { // ... }
// Caller must remember order createTerminal('/bin/bash', '/home', {}, 80, 24, 1000, 'Main');
After:
interface TerminalOptions { shell: string; cwd: string; env?: Record<string, string>; dimensions?: { cols: number; rows: number; }; scrollback?: number; name?: string; }
const DEFAULT_OPTIONS: Partial<TerminalOptions> = { env: {}, dimensions: { cols: 80, rows: 24 }, scrollback: 1000 };
function createTerminal(options: TerminalOptions): Terminal { const opts = { ...DEFAULT_OPTIONS, ...options }; // ... }
// Clear, self-documenting call createTerminal({ shell: '/bin/bash', cwd: '/home', name: 'Main' });
Phase 3: VS Code-Specific Patterns
Manager-Coordinator Pattern
Standard pattern for VS Code extensions with multiple concerns:
// Coordinator orchestrates managers class TerminalWebviewManager implements ICoordinator { private managers: Map<string, IManager> = new Map();
constructor(context: vscode.ExtensionContext) { // Initialize managers this.managers.set('message', new MessageManager(this)); this.managers.set('ui', new UIManager(this)); this.managers.set('input', new InputManager(this)); this.managers.set('performance', new PerformanceManager(this)); }
getManager<T extends IManager>(name: string): T { return this.managers.get(name) as T; }
async initialize(): Promise<void> { for (const manager of this.managers.values()) { await manager.initialize(); } }
dispose(): void { // Dispose in reverse order const managers = Array.from(this.managers.values()).reverse(); for (const manager of managers) { manager.dispose(); } } }
// Individual manager interface interface IManager extends vscode.Disposable { initialize(): Promise<void>; }
// Example manager implementation class MessageManager implements IManager { constructor(private coordinator: ICoordinator) {}
async initialize(): Promise<void> { // Setup message handling }
dispose(): void { // Cleanup } }
Service Locator Pattern
For complex dependency management:
class ServiceContainer { private services = new Map<string, unknown>(); private factories = new Map<string, () => unknown>();
register<T>(key: string, instance: T): void { this.services.set(key, instance); }
registerFactory<T>(key: string, factory: () => T): void { this.factories.set(key, factory); }
get<T>(key: string): T { if (this.services.has(key)) { return this.services.get(key) as T; }
const factory = this.factories.get(key);
if (factory) {
const instance = factory() as T;
this.services.set(key, instance);
return instance;
}
throw new Error(`Service not found: ${key}`);
} }
// Usage const container = new ServiceContainer(); container.register('config', new ConfigService()); container.registerFactory('terminal', () => new TerminalService( container.get('config') ));
Disposable Chain Pattern
Proper resource cleanup:
class DisposableChain implements vscode.Disposable { private chain: vscode.Disposable[] = [];
add<T extends vscode.Disposable>(disposable: T): T { this.chain.push(disposable); return disposable; }
addFunction(fn: () => void): void { this.chain.push({ dispose: fn }); }
dispose(): void { // LIFO disposal while (this.chain.length) { const d = this.chain.pop(); try { d?.dispose(); } catch (e) { console.error('Dispose error:', e); } } } }
// Usage in extension export function activate(context: vscode.ExtensionContext) { const disposables = new DisposableChain();
const config = disposables.add(new ConfigService()); const terminal = disposables.add(new TerminalService(config)); const ui = disposables.add(new UIService(terminal));
context.subscriptions.push(disposables); }
Phase 4: Type Safety Improvements
Replace any with Proper Types
Before:
function processMessage(message: any): any { if (message.type === 'data') { return message.payload.items.map((item: any) => item.value); } return null; }
After:
interface DataMessage { type: 'data'; payload: { items: Array<{ value: string }>; }; }
interface ErrorMessage { type: 'error'; error: string; }
type Message = DataMessage | ErrorMessage;
function processMessage(message: Message): string[] | null { if (message.type === 'data') { return message.payload.items.map(item => item.value); } return null; }
Discriminated Unions
Before:
interface TerminalState { status: string; data?: string; error?: Error; progress?: number; }
// Caller must check multiple fields function render(state: TerminalState): void { if (state.status === 'loading' && state.progress !== undefined) { showProgress(state.progress); } else if (state.status === 'success' && state.data) { showData(state.data); } else if (state.status === 'error' && state.error) { showError(state.error); } }
After:
type TerminalState = | { status: 'idle' } | { status: 'loading'; progress: number } | { status: 'success'; data: string } | { status: 'error'; error: Error };
// Type narrowing with exhaustive check
function render(state: TerminalState): void {
switch (state.status) {
case 'idle':
showIdle();
break;
case 'loading':
showProgress(state.progress); // progress guaranteed
break;
case 'success':
showData(state.data); // data guaranteed
break;
case 'error':
showError(state.error); // error guaranteed
break;
default:
const _exhaustive: never = state;
throw new Error(Unhandled state: ${_exhaustive});
}
}
Generic Constraints
Before:
class Repository<T> { private items: T[] = [];
add(item: T): void { this.items.push(item); }
findById(id: number): T | undefined { // Can't access .id because T is unconstrained return this.items.find((item: any) => item.id === id); } }
After:
interface Identifiable { id: number; }
class Repository<T extends Identifiable> { private items: T[] = [];
add(item: T): void { this.items.push(item); }
findById(id: number): T | undefined { return this.items.find(item => item.id === id); // Type-safe }
update(id: number, updates: Partial<Omit<T, 'id'>>): boolean { const item = this.findById(id); if (item) { Object.assign(item, updates); return true; } return false; } }
Phase 5: Verification
Refactoring Checklist
-
All tests pass after refactoring
-
No new TypeScript errors introduced
-
Public API maintained (or documented changes)
-
Performance not degraded
-
Memory usage stable
-
No circular dependencies introduced
-
Documentation updated if needed
Testing Strategy
// Before refactoring: Capture behavior describe('TerminalManager (before refactor)', () => { it('creates terminal with correct config', () => { const result = manager.createTerminal(config); expect(result).toMatchSnapshot(); }); });
// After refactoring: Same tests must pass describe('TerminalManager (after refactor)', () => { it('creates terminal with correct config', () => { const result = manager.createTerminal(config); expect(result).toMatchSnapshot(); // Same snapshot }); });
Common Refactoring Scenarios
Scenario 1: Breaking Up a Large File
src/terminal.ts (2000 lines) ↓ Split by responsibility src/terminal/ ├── index.ts # Public exports ├── TerminalManager.ts # Core management ├── TerminalFactory.ts # Creation logic ├── TerminalState.ts # State management ├── types.ts # Interfaces and types └── utils.ts # Helper functions
Scenario 2: Reducing Coupling
// Before: Tight coupling class TerminalManager { private ui = new UIManager(); // Direct instantiation private config = new ConfigManager(); }
// After: Dependency injection class TerminalManager { constructor( private ui: IUIManager, private config: IConfigManager ) {} }
// Factory handles wiring function createTerminalManager(): TerminalManager { return new TerminalManager( new UIManager(), new ConfigManager() ); }
Scenario 3: Async Refactoring
// Before: Callback hell function loadData(callback: (data: Data) => void): void { readFile(path, (err, content) => { if (err) throw err; parseData(content, (err, parsed) => { if (err) throw err; validateData(parsed, (err, valid) => { if (err) throw err; callback(valid); }); }); }); }
// After: Async/await async function loadData(): Promise<Data> { const content = await readFile(path); const parsed = await parseData(content); return validateData(parsed); }
Resources
For detailed reference documentation:
-
references/code-smells.md
-
Complete code smell catalog with examples
-
references/design-patterns.md
-
VS Code-specific design pattern implementations
-
references/type-patterns.md
-
Advanced TypeScript type patterns