maestro/src/engine/tools/mission.ts
2026-06-03 05:08:00 +00:00

125 lines
4.8 KiB
TypeScript

/**
* Mission Brief tool — per-task pinned memo of goal / done / open /
* clarifications, always rendered at the top of every movement's system
* prompt and editable by both the LLM (via this tool) and the user
* (via the Overview tab).
*
* Design notes:
* - Partial replace semantics: only fields explicitly provided in the
* call are written. Undefined fields leave existing values intact.
* - The brief is per-LocalTask (not per-job, not per-movement) so it
* survives across iterations, ASK rounds, and follow-up messages
* within the same task conversation.
* - Storage is a single JSON column on local_tasks; see
* src/db/repository.ts.MissionBrief / updateMissionBrief.
* - Plumbing: piece-runner constructs a MissionBriefIO from the
* localTaskId + repo and threads it through ToolContext.
* Subtask contexts that aren't bound to a local_task simply leave
* the IO unset; the tool then degrades to a no-op with a clear
* error so the LLM doesn't get confused.
*/
import { ToolDef } from '../../llm/openai-compat.js';
import type { ToolContext, ToolResult } from './core.js';
const MISSION_UPDATE_DEF: ToolDef = {
type: 'function',
function: {
name: 'MissionUpdate',
description:
'タスクの Mission Brief (goal / done / open / clarifications) を更新する。常時利用可能で META_TOOL 扱い。**新規タスクの最初のツール呼び出しで goal を必ず set すること** ── ユーザー要件を verbatim に固定し、会話が長くなった後でも参照点として残す。以降は節目で done / open を更新。指定したフィールドだけ置き換わり、未指定は変更なし。詳細は ReadToolDoc({ name: "MissionUpdate" })。',
parameters: {
type: 'object',
properties: {
goal: {
type: 'string',
description: 'このタスク全体のゴール (ユーザーが最初に依頼した本質的な要件)。Markdown 可。',
},
done: {
type: 'string',
description: 'これまでに完了した主要マイルストーン。Markdown 箇条書き推奨。重複作業を避けるための参照。',
},
open: {
type: 'string',
description: '残っている作業 / 未解決のブロッカー。Markdown 箇条書き推奨。',
},
clarifications: {
type: 'string',
description: 'ユーザーから途中で追加された補足・制約。「これは壊さないで」など。Markdown 可。',
},
},
},
},
};
export const TOOL_DEFS: Record<string, ToolDef> = {
MissionUpdate: MISSION_UPDATE_DEF,
};
const FIELD_MAX_CHARS = 2000;
function clamp(value: unknown): string | undefined {
if (value === undefined) return undefined;
if (typeof value !== 'string') return undefined;
if (value.length <= FIELD_MAX_CHARS) return value;
return `${value.slice(0, FIELD_MAX_CHARS)}\n…[truncated, ${value.length} chars]`;
}
export async function executeTool(
name: string,
input: Record<string, unknown>,
ctx: ToolContext,
): Promise<ToolResult | null> {
if (name !== 'MissionUpdate') return null;
const io = ctx.missionBrief;
if (!io) {
return {
output: 'MissionUpdate はこのコンテキストでは利用できません (subtask など local_task と紐付かない実行)。',
isError: true,
};
}
const patch: Record<string, string | undefined> = {
goal: clamp(input['goal']),
done: clamp(input['done']),
open: clamp(input['open']),
clarifications: clamp(input['clarifications']),
};
// Strip undefined so the IO layer treats them as "not provided" rather
// than "set to empty string".
const filtered: Partial<{ goal: string; done: string; open: string; clarifications: string }> = {};
for (const key of ['goal', 'done', 'open', 'clarifications'] as const) {
if (patch[key] !== undefined) filtered[key] = patch[key]!;
}
if (Object.keys(filtered).length === 0) {
return {
output: '更新するフィールドが1つも指定されませんでした。goal / done / open / clarifications のいずれかを指定してください。',
isError: true,
};
}
try {
const merged = io.update(filtered);
if (!merged) {
return {
output: 'Mission Brief をクリアしました (全フィールドが空)。',
isError: false,
};
}
const summary = ['Mission Brief を更新しました:'];
for (const key of ['goal', 'done', 'open', 'clarifications'] as const) {
const value = merged[key];
if (value) summary.push(`- ${key} (${value.length} chars)`);
}
return { output: summary.join('\n'), isError: false };
} catch (err) {
return {
output: `Mission Brief の更新に失敗しました: ${err instanceof Error ? err.message : String(err)}`,
isError: true,
};
}
}