Files
claude-task-master/packages/tm-bridge/src/update-bridge.ts
2025-10-25 19:25:17 +02:00

184 lines
4.8 KiB
TypeScript

import chalk from 'chalk';
import boxen from 'boxen';
import { createTmCore, type TmCore } from '@tm/core';
/**
* Parameters for the update bridge function
*/
export interface UpdateBridgeParams {
/** Task ID (can be numeric "1", alphanumeric "TAS-49", or dotted "1.2" or "TAS-49.1") */
taskId: string | number;
/** Update prompt for AI */
prompt: string;
/** Project root directory */
projectRoot: string;
/** Optional tag for task organization */
tag?: string;
/** Whether to append or full update (default: false) */
appendMode?: boolean;
/** Whether called from MCP context (default: false) */
isMCP?: boolean;
/** Output format (default: 'text') */
outputFormat?: 'text' | 'json';
/** Logging function */
report: (level: string, ...args: unknown[]) => void;
}
/**
* Result returned when API storage handles the update
*/
export interface RemoteUpdateResult {
success: boolean;
taskId: string | number;
message: string;
telemetryData: null;
tagInfo: null;
}
/**
* Shared bridge function for update-task and update-subtask commands.
* Checks if using API storage and delegates to remote AI service if so.
*
* In API storage, tasks and subtasks are treated identically - there's no
* parent/child hierarchy, so update-task and update-subtask can be used
* interchangeably.
*
* @param params - Bridge parameters
* @returns Result object if API storage handled it, null if should fall through to file storage
*/
export async function tryUpdateViaRemote(
params: UpdateBridgeParams
): Promise<RemoteUpdateResult | null> {
const {
taskId,
prompt,
projectRoot,
tag,
appendMode = false,
isMCP = false,
outputFormat = 'text',
report
} = params;
let tmCore: TmCore;
try {
tmCore = await createTmCore({
projectPath: projectRoot || process.cwd()
});
} catch (tmCoreError) {
const errorMessage =
tmCoreError instanceof Error ? tmCoreError.message : String(tmCoreError);
report(
'warn',
`TmCore check failed, falling back to file-based update: ${errorMessage}`
);
// Return null to signal fall-through to file storage logic
return null;
}
// Check if we're using API storage (use resolved storage type, not config)
const storageType = tmCore.tasks.getStorageType();
if (storageType !== 'api') {
// Not API storage - signal caller to fall through to file-based logic
report(
'info',
`Using file storage - processing update locally for task ${taskId}`
);
return null;
}
// API STORAGE PATH: Delegate to remote AI service
report('info', `Delegating update to Hamster for task ${taskId}`);
const mode = appendMode ? 'append' : 'update';
// Show CLI output if not MCP
if (!isMCP && outputFormat === 'text') {
const showDebug = process.env.TM_DEBUG === '1';
const promptPreview = showDebug
? `${prompt.substring(0, 60)}${prompt.length > 60 ? '...' : ''}`
: '[hidden]';
console.log(
boxen(
chalk.blue.bold(`Updating Task via Hamster`) +
'\n\n' +
chalk.white(`Task ID: ${taskId}`) +
'\n' +
chalk.white(`Mode: ${mode}`) +
'\n' +
chalk.white(`Prompt: ${promptPreview}`),
{
padding: 1,
borderColor: 'blue',
borderStyle: 'round',
margin: { top: 1, bottom: 1 }
}
)
);
}
let loadingIndicator: NodeJS.Timeout | null = null;
if (!isMCP && outputFormat === 'text') {
// Simple loading indicator simulation (replace with actual startLoadingIndicator if available)
const frames = ['⠋', '⠙', '⠹', '⠸', '⠼', '⠴', '⠦', '⠧', '⠇', '⠏'];
let frameIndex = 0;
loadingIndicator = setInterval(() => {
process.stdout.write(
`\r${frames[frameIndex]} Updating task on Hamster...`
);
frameIndex = (frameIndex + 1) % frames.length;
}, 80);
}
try {
// Call the API storage method which handles the remote update
await tmCore.tasks.updateWithPrompt(String(taskId), prompt, tag, {
mode
});
if (loadingIndicator) {
clearInterval(loadingIndicator);
process.stdout.write('\r✓ Task updated successfully.\n');
}
if (outputFormat === 'text') {
console.log(
boxen(
chalk.green(`Successfully updated task ${taskId} via remote AI`) +
'\n\n' +
chalk.white('The task has been updated on the remote server.') +
'\n' +
chalk.white(
`Run ${chalk.yellow(`task-master show ${taskId}`)} to view the updated task.`
),
{
padding: 1,
borderColor: 'green',
borderStyle: 'round'
}
)
);
}
// Return success result - signals that we handled it
return {
success: true,
taskId: taskId,
message: 'Task updated via remote AI service',
telemetryData: null,
tagInfo: null
};
} catch (updateError) {
if (loadingIndicator) {
clearInterval(loadingIndicator);
process.stdout.write('\r✗ Update failed.\n');
}
// tm-core already formatted the error properly, just re-throw
throw updateError;
}
}