/** * tools/update-subtask.js * Tool to append additional information to a specific subtask */ import { z } from 'zod'; import { handleApiResult, createErrorResponse, getProjectRootFromSession } from './utils.js'; import { updateSubtaskByIdDirect } from '../core/task-master-core.js'; import { findTasksJsonPath } from '../core/utils/path-utils.js'; /** * Register the update-subtask tool with the MCP server * @param {Object} server - FastMCP server instance */ export function registerUpdateSubtaskTool(server) { server.addTool({ name: 'update_subtask', description: 'Appends timestamped information to a specific subtask without replacing existing content', parameters: z.object({ id: z .string() .describe( 'ID of the subtask to update in format "parentId.subtaskId" (e.g., "5.2"). Parent ID is the ID of the task that contains the subtask.' ), prompt: z.string().describe('Information to add to the subtask'), research: z .boolean() .optional() .describe('Use Perplexity AI for research-backed updates'), file: z.string().optional().describe('Absolute path to the tasks file'), projectRoot: z .string() .describe('The directory of the project. Must be an absolute path.') }), execute: async (args, { log, session }) => { try { log.info(`Updating subtask with args: ${JSON.stringify(args)}`); // Get project root from args or session const rootFolder = args.projectRoot || getProjectRootFromSession(session, log); // Ensure project root was determined if (!rootFolder) { return createErrorResponse( 'Could not determine project root. Please provide it explicitly or ensure your session contains valid root information.' ); } // Resolve the path to tasks.json let tasksJsonPath; try { tasksJsonPath = findTasksJsonPath( { projectRoot: rootFolder, file: args.file }, log ); } catch (error) { log.error(`Error finding tasks.json: ${error.message}`); return createErrorResponse( `Failed to find tasks.json: ${error.message}` ); } const result = await updateSubtaskByIdDirect( { // Pass the explicitly resolved path tasksJsonPath: tasksJsonPath, // Pass other relevant args id: args.id, prompt: args.prompt, research: args.research }, log, { session } ); if (result.success) { log.info(`Successfully updated subtask with ID ${args.id}`); } else { log.error( `Failed to update subtask: ${result.error?.message || 'Unknown error'}` ); } return handleApiResult(result, log, 'Error updating subtask'); } catch (error) { log.error(`Error in update_subtask tool: ${error.message}`); return createErrorResponse(error.message); } } }); }