* feat(hooks): add tasks-todowrite-disabler hook to block TodoRead/TodoWrite Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-opencode) Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai> * feat(task-tools): add parallel execution guidance to descriptions Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-opencode) Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai> * refactor(index): migrate task system to experimental.task_system flag Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-opencode) Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai> * docs: update AGENTS.md for experimental task system Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-opencode) Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai> * fix(task-tests): align test field names with Claude Code spec (subject, blockedBy, addBlockedBy) * fix: address Cubic review feedback Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-opencode) Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai> * fix: add optional chaining for tasksTodowriteDisabler null check --------- Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai>
114 lines
3.7 KiB
TypeScript
114 lines
3.7 KiB
TypeScript
import type { PluginInput } from "@opencode-ai/plugin";
|
|
import { tool, type ToolDefinition } from "@opencode-ai/plugin/tool";
|
|
import { join } from "path";
|
|
import type { OhMyOpenCodeConfig } from "../../config/schema";
|
|
import type { TaskObject } from "./types";
|
|
import { TaskObjectSchema, TaskCreateInputSchema } from "./types";
|
|
import {
|
|
getTaskDir,
|
|
writeJsonAtomic,
|
|
acquireLock,
|
|
generateTaskId,
|
|
} from "../../features/claude-tasks/storage";
|
|
import { syncTaskTodoUpdate } from "./todo-sync";
|
|
|
|
export function createTaskCreateTool(
|
|
config: Partial<OhMyOpenCodeConfig>,
|
|
ctx?: PluginInput,
|
|
): ToolDefinition {
|
|
return tool({
|
|
description: `Create a new task with auto-generated ID and threadID recording.
|
|
|
|
Auto-generates T-{uuid} ID, records threadID from context, sets status to "pending".
|
|
Returns minimal response with task ID and subject.
|
|
|
|
**IMPORTANT - Dependency Planning for Parallel Execution:**
|
|
Use \`blockedBy\` to specify task IDs that must complete before this task can start.
|
|
Calculate dependencies carefully to maximize parallel execution:
|
|
- Tasks with no dependencies can run simultaneously
|
|
- Only block a task if it truly depends on another's output
|
|
- Minimize dependency chains to reduce sequential bottlenecks`,
|
|
args: {
|
|
subject: tool.schema.string().describe("Task subject (required)"),
|
|
description: tool.schema.string().optional().describe("Task description"),
|
|
activeForm: tool.schema
|
|
.string()
|
|
.optional()
|
|
.describe("Active form (present continuous)"),
|
|
metadata: tool.schema
|
|
.record(tool.schema.string(), tool.schema.unknown())
|
|
.optional()
|
|
.describe("Task metadata"),
|
|
blockedBy: tool.schema
|
|
.array(tool.schema.string())
|
|
.optional()
|
|
.describe("Task IDs blocking this task"),
|
|
blocks: tool.schema
|
|
.array(tool.schema.string())
|
|
.optional()
|
|
.describe("Task IDs this task blocks"),
|
|
repoURL: tool.schema.string().optional().describe("Repository URL"),
|
|
parentID: tool.schema.string().optional().describe("Parent task ID"),
|
|
},
|
|
execute: async (args, context) => {
|
|
return handleCreate(args, config, ctx, context);
|
|
},
|
|
});
|
|
}
|
|
|
|
async function handleCreate(
|
|
args: Record<string, unknown>,
|
|
config: Partial<OhMyOpenCodeConfig>,
|
|
ctx: PluginInput | undefined,
|
|
context: { sessionID: string },
|
|
): Promise<string> {
|
|
try {
|
|
const validatedArgs = TaskCreateInputSchema.parse(args);
|
|
const taskDir = getTaskDir(config);
|
|
const lock = acquireLock(taskDir);
|
|
|
|
if (!lock.acquired) {
|
|
return JSON.stringify({ error: "task_lock_unavailable" });
|
|
}
|
|
|
|
try {
|
|
const taskId = generateTaskId();
|
|
const task: TaskObject = {
|
|
id: taskId,
|
|
subject: validatedArgs.subject,
|
|
description: validatedArgs.description ?? "",
|
|
status: "pending",
|
|
blocks: validatedArgs.blocks ?? [],
|
|
blockedBy: validatedArgs.blockedBy ?? [],
|
|
activeForm: validatedArgs.activeForm,
|
|
metadata: validatedArgs.metadata,
|
|
repoURL: validatedArgs.repoURL,
|
|
parentID: validatedArgs.parentID,
|
|
threadID: context.sessionID,
|
|
};
|
|
|
|
const validatedTask = TaskObjectSchema.parse(task);
|
|
writeJsonAtomic(join(taskDir, `${taskId}.json`), validatedTask);
|
|
|
|
await syncTaskTodoUpdate(ctx, validatedTask, context.sessionID);
|
|
|
|
return JSON.stringify({
|
|
task: {
|
|
id: validatedTask.id,
|
|
subject: validatedTask.subject,
|
|
},
|
|
});
|
|
} finally {
|
|
lock.release();
|
|
}
|
|
} catch (error) {
|
|
if (error instanceof Error && error.message.includes("Required")) {
|
|
return JSON.stringify({
|
|
error: "validation_error",
|
|
message: error.message,
|
|
});
|
|
}
|
|
return JSON.stringify({ error: "internal_error" });
|
|
}
|
|
}
|