- Added 'defaultNumTasks: 10' to default config, now used in 'parse-prd' - Adjusted 'parse-prd' and 'expand-task' to: - Accept a 'numTasks' value of 0 - Updated tool and command descriptions - Updated prompts to 'an appropriate number of' when value is 0 - Updated 'README-task-master.md' and 'command-reference.md' docs - Added more tests for: 'parse-prd', 'expand-task' and 'config-manager' Co-authored-by: Ralph Khreish <35776126+Crunchyman-ralph@users.noreply.github.com>
207 lines
5.5 KiB
JavaScript
207 lines
5.5 KiB
JavaScript
/**
|
|
* parse-prd.js
|
|
* Direct function implementation for parsing PRD documents
|
|
*/
|
|
|
|
import path from 'path';
|
|
import fs from 'fs';
|
|
import { parsePRD } from '../../../../scripts/modules/task-manager.js';
|
|
import {
|
|
enableSilentMode,
|
|
disableSilentMode,
|
|
isSilentMode
|
|
} from '../../../../scripts/modules/utils.js';
|
|
import { createLogWrapper } from '../../tools/utils.js';
|
|
import { getDefaultNumTasks } from '../../../../scripts/modules/config-manager.js';
|
|
import { resolvePrdPath, resolveProjectPath } from '../utils/path-utils.js';
|
|
import { TASKMASTER_TASKS_FILE } from '../../../../src/constants/paths.js';
|
|
|
|
/**
|
|
* Direct function wrapper for parsing PRD documents and generating tasks.
|
|
*
|
|
* @param {Object} args - Command arguments containing projectRoot, input, output, numTasks options.
|
|
* @param {Object} log - Logger object.
|
|
* @param {Object} context - Context object containing session data.
|
|
* @returns {Promise<Object>} - Result object with success status and data/error information.
|
|
*/
|
|
export async function parsePRDDirect(args, log, context = {}) {
|
|
const { session } = context;
|
|
// Extract projectRoot from args
|
|
const {
|
|
input: inputArg,
|
|
output: outputArg,
|
|
numTasks: numTasksArg,
|
|
force,
|
|
append,
|
|
research,
|
|
projectRoot
|
|
} = args;
|
|
|
|
// Create the standard logger wrapper
|
|
const logWrapper = createLogWrapper(log);
|
|
|
|
// --- Input Validation and Path Resolution ---
|
|
if (!projectRoot) {
|
|
logWrapper.error('parsePRDDirect requires a projectRoot argument.');
|
|
return {
|
|
success: false,
|
|
error: {
|
|
code: 'MISSING_ARGUMENT',
|
|
message: 'projectRoot is required.'
|
|
}
|
|
};
|
|
}
|
|
|
|
// Resolve input path using path utilities
|
|
let inputPath;
|
|
if (inputArg) {
|
|
try {
|
|
inputPath = resolvePrdPath({ input: inputArg, projectRoot }, session);
|
|
} catch (error) {
|
|
logWrapper.error(`Error resolving PRD path: ${error.message}`);
|
|
return {
|
|
success: false,
|
|
error: { code: 'FILE_NOT_FOUND', message: error.message }
|
|
};
|
|
}
|
|
} else {
|
|
logWrapper.error('parsePRDDirect called without input path');
|
|
return {
|
|
success: false,
|
|
error: { code: 'MISSING_ARGUMENT', message: 'Input path is required' }
|
|
};
|
|
}
|
|
|
|
// Resolve output path - use new path utilities for default
|
|
const outputPath = outputArg
|
|
? path.isAbsolute(outputArg)
|
|
? outputArg
|
|
: path.resolve(projectRoot, outputArg)
|
|
: resolveProjectPath(TASKMASTER_TASKS_FILE, args) ||
|
|
path.resolve(projectRoot, TASKMASTER_TASKS_FILE);
|
|
|
|
// Check if input file exists
|
|
if (!fs.existsSync(inputPath)) {
|
|
const errorMsg = `Input PRD file not found at resolved path: ${inputPath}`;
|
|
logWrapper.error(errorMsg);
|
|
return {
|
|
success: false,
|
|
error: { code: 'FILE_NOT_FOUND', message: errorMsg }
|
|
};
|
|
}
|
|
|
|
const outputDir = path.dirname(outputPath);
|
|
try {
|
|
if (!fs.existsSync(outputDir)) {
|
|
logWrapper.info(`Creating output directory: ${outputDir}`);
|
|
fs.mkdirSync(outputDir, { recursive: true });
|
|
}
|
|
} catch (error) {
|
|
const errorMsg = `Failed to create output directory ${outputDir}: ${error.message}`;
|
|
logWrapper.error(errorMsg);
|
|
return {
|
|
success: false,
|
|
error: { code: 'DIRECTORY_CREATE_FAILED', message: errorMsg }
|
|
};
|
|
}
|
|
|
|
let numTasks = getDefaultNumTasks(projectRoot);
|
|
if (numTasksArg) {
|
|
numTasks =
|
|
typeof numTasksArg === 'string' ? parseInt(numTasksArg, 10) : numTasksArg;
|
|
if (Number.isNaN(numTasks) || numTasks < 0) {
|
|
// Ensure positive number
|
|
numTasks = getDefaultNumTasks(projectRoot); // Fallback to default if parsing fails or invalid
|
|
logWrapper.warn(
|
|
`Invalid numTasks value: ${numTasksArg}. Using default: ${numTasks}`
|
|
);
|
|
}
|
|
}
|
|
|
|
if (append) {
|
|
logWrapper.info('Append mode enabled.');
|
|
if (force) {
|
|
logWrapper.warn(
|
|
'Both --force and --append flags were provided. --force takes precedence; append mode will be ignored.'
|
|
);
|
|
}
|
|
}
|
|
|
|
if (research) {
|
|
logWrapper.info(
|
|
'Research mode enabled. Using Perplexity AI for enhanced PRD analysis.'
|
|
);
|
|
}
|
|
|
|
logWrapper.info(
|
|
`Parsing PRD via direct function. Input: ${inputPath}, Output: ${outputPath}, NumTasks: ${numTasks}, Force: ${force}, Append: ${append}, Research: ${research}, ProjectRoot: ${projectRoot}`
|
|
);
|
|
|
|
const wasSilent = isSilentMode();
|
|
if (!wasSilent) {
|
|
enableSilentMode();
|
|
}
|
|
|
|
try {
|
|
// Call the core parsePRD function
|
|
const result = await parsePRD(
|
|
inputPath,
|
|
outputPath,
|
|
numTasks,
|
|
{
|
|
session,
|
|
mcpLog: logWrapper,
|
|
projectRoot,
|
|
force,
|
|
append,
|
|
research,
|
|
commandName: 'parse-prd',
|
|
outputType: 'mcp'
|
|
},
|
|
'json'
|
|
);
|
|
|
|
// Adjust check for the new return structure
|
|
if (result && result.success) {
|
|
const successMsg = `Successfully parsed PRD and generated tasks in ${result.tasksPath}`;
|
|
logWrapper.success(successMsg);
|
|
return {
|
|
success: true,
|
|
data: {
|
|
message: successMsg,
|
|
outputPath: result.tasksPath,
|
|
telemetryData: result.telemetryData,
|
|
tagInfo: result.tagInfo
|
|
}
|
|
};
|
|
} else {
|
|
// Handle case where core function didn't return expected success structure
|
|
logWrapper.error(
|
|
'Core parsePRD function did not return a successful structure.'
|
|
);
|
|
return {
|
|
success: false,
|
|
error: {
|
|
code: 'CORE_FUNCTION_ERROR',
|
|
message:
|
|
result?.message ||
|
|
'Core function failed to parse PRD or returned unexpected result.'
|
|
}
|
|
};
|
|
}
|
|
} catch (error) {
|
|
logWrapper.error(`Error executing core parsePRD: ${error.message}`);
|
|
return {
|
|
success: false,
|
|
error: {
|
|
code: 'PARSE_PRD_CORE_ERROR',
|
|
message: error.message || 'Unknown error parsing PRD'
|
|
}
|
|
};
|
|
} finally {
|
|
if (!wasSilent && isSilentMode()) {
|
|
disableSilentMode();
|
|
}
|
|
}
|
|
}
|