Files
claude-task-master/tests/unit/commands.test.js
2025-05-31 16:21:03 +02:00

322 lines
9.2 KiB
JavaScript

/**
* Commands module tests - Focus on CLI setup and integration
*/
import { jest } from '@jest/globals';
// Mock modules first
jest.mock('fs', () => ({
existsSync: jest.fn(),
readFileSync: jest.fn()
}));
jest.mock('path', () => ({
join: jest.fn((dir, file) => `${dir}/${file}`)
}));
jest.mock('chalk', () => ({
red: jest.fn((text) => text),
blue: jest.fn((text) => text),
green: jest.fn((text) => text),
yellow: jest.fn((text) => text),
white: jest.fn((text) => ({
bold: jest.fn((text) => text)
})),
reset: jest.fn((text) => text)
}));
// Mock config-manager to prevent file system discovery issues
jest.mock('../../scripts/modules/config-manager.js', () => ({
getLogLevel: jest.fn(() => 'info'),
getDebugFlag: jest.fn(() => false),
getConfig: jest.fn(() => ({})), // Return empty config to prevent real loading
getGlobalConfig: jest.fn(() => ({}))
}));
// Mock path-utils to prevent file system discovery issues
jest.mock('../../src/utils/path-utils.js', () => ({
__esModule: true,
findProjectRoot: jest.fn(() => '/mock/project'),
findConfigPath: jest.fn(() => null),
findTasksPath: jest.fn(() => '/mock/tasks.json'),
findComplexityReportPath: jest.fn(() => null),
resolveTasksOutputPath: jest.fn(() => '/mock/tasks.json'),
resolveComplexityReportOutputPath: jest.fn(() => '/mock/report.json')
}));
jest.mock('../../scripts/modules/ui.js', () => ({
displayBanner: jest.fn(),
displayHelp: jest.fn()
}));
// Add utility functions for testing
const toKebabCase = (str) => {
return str
.replace(/([a-z0-9])([A-Z])/g, '$1-$2')
.toLowerCase()
.replace(/^-/, '');
};
function detectCamelCaseFlags(args) {
const camelCaseFlags = [];
for (const arg of args) {
if (arg.startsWith('--')) {
const flagName = arg.split('=')[0].slice(2);
if (!flagName.includes('-')) {
if (/[a-z][A-Z]/.test(flagName)) {
const kebabVersion = toKebabCase(flagName);
if (kebabVersion !== flagName) {
camelCaseFlags.push({
original: flagName,
kebabCase: kebabVersion
});
}
}
}
}
}
return camelCaseFlags;
}
jest.mock('../../scripts/modules/utils.js', () => ({
CONFIG: {
projectVersion: '1.5.0'
},
log: jest.fn(() => {}), // Prevent any real logging that could trigger config discovery
toKebabCase: toKebabCase,
detectCamelCaseFlags: detectCamelCaseFlags
}));
// Import all modules after mocking
import fs from 'fs';
import path from 'path';
import { setupCLI } from '../../scripts/modules/commands.js';
describe('Commands Module - CLI Setup and Integration', () => {
const mockExistsSync = jest.spyOn(fs, 'existsSync');
const mockReadFileSync = jest.spyOn(fs, 'readFileSync');
const mockJoin = jest.spyOn(path, 'join');
beforeEach(() => {
jest.clearAllMocks();
mockExistsSync.mockReturnValue(true);
});
afterAll(() => {
jest.restoreAllMocks();
});
describe('setupCLI function', () => {
test('should return Commander program instance', () => {
const program = setupCLI();
expect(program).toBeDefined();
expect(program.name()).toBe('dev');
});
test('should read version from package.json when available', () => {
mockExistsSync.mockReturnValue(true);
mockReadFileSync.mockReturnValue('{"version": "1.0.0"}');
mockJoin.mockReturnValue('package.json');
const program = setupCLI();
const version = program._version();
expect(mockReadFileSync).toHaveBeenCalledWith('package.json', 'utf8');
expect(version).toBe('1.0.0');
});
test('should use default version when package.json is not available', () => {
mockExistsSync.mockReturnValue(false);
const program = setupCLI();
const version = program._version();
expect(mockReadFileSync).not.toHaveBeenCalled();
expect(version).toBe('unknown');
});
test('should use default version when package.json reading throws an error', () => {
mockExistsSync.mockReturnValue(true);
mockReadFileSync.mockImplementation(() => {
throw new Error('Read error');
});
// Mock console methods to prevent chalk formatting conflicts
const consoleErrorSpy = jest
.spyOn(console, 'error')
.mockImplementation(() => {});
const consoleLogSpy = jest
.spyOn(console, 'log')
.mockImplementation(() => {});
const consoleWarnSpy = jest
.spyOn(console, 'warn')
.mockImplementation(() => {});
const program = setupCLI();
const version = program._version();
expect(mockReadFileSync).toHaveBeenCalled();
expect(version).toBe('unknown');
// Restore console methods
consoleErrorSpy.mockRestore();
consoleLogSpy.mockRestore();
consoleWarnSpy.mockRestore();
});
});
describe('CLI Flag Format Validation', () => {
test('should detect camelCase flags correctly', () => {
const args = ['node', 'task-master', '--camelCase', '--kebab-case'];
const camelCaseFlags = args.filter(
(arg) =>
arg.startsWith('--') && /[A-Z]/.test(arg) && !arg.includes('-[A-Z]')
);
expect(camelCaseFlags).toContain('--camelCase');
expect(camelCaseFlags).not.toContain('--kebab-case');
});
test('should accept kebab-case flags correctly', () => {
const args = ['node', 'task-master', '--kebab-case'];
const camelCaseFlags = args.filter(
(arg) =>
arg.startsWith('--') && /[A-Z]/.test(arg) && !arg.includes('-[A-Z]')
);
expect(camelCaseFlags).toHaveLength(0);
});
test('toKebabCase should convert camelCase to kebab-case', () => {
expect(toKebabCase('promptText')).toBe('prompt-text');
expect(toKebabCase('userID')).toBe('user-id');
expect(toKebabCase('numTasks')).toBe('num-tasks');
expect(toKebabCase('alreadyKebabCase')).toBe('already-kebab-case');
});
test('detectCamelCaseFlags should identify camelCase flags', () => {
const args = [
'node',
'task-master',
'add-task',
'--promptText=test',
'--userID=123'
];
const flags = detectCamelCaseFlags(args);
expect(flags).toHaveLength(2);
expect(flags).toContainEqual({
original: 'promptText',
kebabCase: 'prompt-text'
});
expect(flags).toContainEqual({
original: 'userID',
kebabCase: 'user-id'
});
});
test('detectCamelCaseFlags should not flag kebab-case flags', () => {
const args = [
'node',
'task-master',
'add-task',
'--prompt-text=test',
'--user-id=123'
];
const flags = detectCamelCaseFlags(args);
expect(flags).toHaveLength(0);
});
test('detectCamelCaseFlags should respect single-word flags', () => {
const args = [
'node',
'task-master',
'add-task',
'--prompt=test',
'--file=test.json',
'--priority=high',
'--promptText=test'
];
const flags = detectCamelCaseFlags(args);
expect(flags).toHaveLength(1);
expect(flags).toContainEqual({
original: 'promptText',
kebabCase: 'prompt-text'
});
});
});
describe('Command Validation Logic', () => {
test('should validate task ID parameter correctly', () => {
// Test valid task IDs
const validId = '5';
const taskId = parseInt(validId, 10);
expect(Number.isNaN(taskId) || taskId <= 0).toBe(false);
// Test invalid task IDs
const invalidId = 'not-a-number';
const invalidTaskId = parseInt(invalidId, 10);
expect(Number.isNaN(invalidTaskId) || invalidTaskId <= 0).toBe(true);
// Test zero or negative IDs
const zeroId = '0';
const zeroTaskId = parseInt(zeroId, 10);
expect(Number.isNaN(zeroTaskId) || zeroTaskId <= 0).toBe(true);
});
test('should handle environment variable cleanup correctly', () => {
// Instead of using delete operator, test setting to undefined
const testEnv = { PERPLEXITY_API_KEY: 'test-key' };
testEnv.PERPLEXITY_API_KEY = undefined;
expect(testEnv.PERPLEXITY_API_KEY).toBeUndefined();
});
});
});
// Test utility functions that commands rely on
describe('Version comparison utility', () => {
let compareVersions;
beforeAll(async () => {
const commandsModule = await import('../../scripts/modules/commands.js');
compareVersions = commandsModule.compareVersions;
});
test('compareVersions correctly compares semantic versions', () => {
expect(compareVersions('1.0.0', '1.0.0')).toBe(0);
expect(compareVersions('1.0.0', '1.0.1')).toBe(-1);
expect(compareVersions('1.0.1', '1.0.0')).toBe(1);
expect(compareVersions('1.0.0', '1.1.0')).toBe(-1);
expect(compareVersions('1.1.0', '1.0.0')).toBe(1);
expect(compareVersions('1.0.0', '2.0.0')).toBe(-1);
expect(compareVersions('2.0.0', '1.0.0')).toBe(1);
expect(compareVersions('1.0', '1.0.0')).toBe(0);
expect(compareVersions('1.0.0.0', '1.0.0')).toBe(0);
expect(compareVersions('1.0.0', '1.0.0.1')).toBe(-1);
});
});
describe('Update check functionality', () => {
let displayUpgradeNotification;
let consoleLogSpy;
beforeAll(async () => {
const commandsModule = await import('../../scripts/modules/commands.js');
displayUpgradeNotification = commandsModule.displayUpgradeNotification;
});
beforeEach(() => {
consoleLogSpy = jest.spyOn(console, 'log').mockImplementation(() => {});
});
afterEach(() => {
consoleLogSpy.mockRestore();
});
test('displays upgrade notification when newer version is available', () => {
displayUpgradeNotification('1.0.0', '1.1.0');
expect(consoleLogSpy).toHaveBeenCalled();
expect(consoleLogSpy.mock.calls[0][0]).toContain('Update Available!');
expect(consoleLogSpy.mock.calls[0][0]).toContain('1.0.0');
expect(consoleLogSpy.mock.calls[0][0]).toContain('1.1.0');
});
});