import { PlanExitTool } from "./plan" import { QuestionTool } from "./question" import { BashTool } from "./bash" import { EditTool } from "./edit" import { GlobTool } from "./glob" import { GrepTool } from "./grep" import { BatchTool } from "./batch" import { ReadTool } from "./read" import { TaskTool } from "./task" import { TodoWriteTool, TodoReadTool } from "./todo" import { WebFetchTool } from "./webfetch" import { WriteTool } from "./write" import { InvalidTool } from "./invalid" import { SkillTool } from "./skill" import type { Agent } from "../agent/agent" import { Tool } from "./tool" import { Instance } from "../project/instance" import { Config } from "../config/config" import path from "path" import { type ToolContext as PluginToolContext, type ToolDefinition } from "@opencode-ai/plugin" import z from "zod" import { Plugin } from "../plugin" import { ProviderID, type ModelID } from "../provider/schema" import { WebSearchTool } from "./websearch" import { CodeSearchTool } from "./codesearch" import { Flag } from "@/flag/flag" import { Log } from "@/util/log" import { LspTool } from "./lsp" import { Truncate } from "./truncate" import { ApplyPatchTool } from "./apply_patch" import { Glob } from "../util/glob" import { pathToFileURL } from "url" export namespace ToolRegistry { const log = Log.create({ service: "tool.registry" }) export const state = Instance.state(async () => { const custom = [] as Tool.Info[] const matches = await Config.directories().then((dirs) => dirs.flatMap((dir) => Glob.scanSync("{tool,tools}/*.{js,ts}", { cwd: dir, absolute: true, dot: true, symlink: true }), ), ) if (matches.length) await Config.waitForDependencies() for (const match of matches) { const namespace = path.basename(match, path.extname(match)) const mod = await import(process.platform === "win32" ? match : pathToFileURL(match).href) for (const [id, def] of Object.entries(mod)) { custom.push(fromPlugin(id === "default" ? namespace : `${namespace}_${id}`, def)) } } const plugins = await Plugin.list() for (const plugin of plugins) { for (const [id, def] of Object.entries(plugin.tool ?? {})) { custom.push(fromPlugin(id, def)) } } return { custom } }) function fromPlugin(id: string, def: ToolDefinition): Tool.Info { return { id, init: async (initCtx) => ({ parameters: z.object(def.args), description: def.description, execute: async (args, ctx) => { const pluginCtx = { ...ctx, directory: Instance.directory, worktree: Instance.worktree, } as unknown as PluginToolContext const result = await def.execute(args as any, pluginCtx) const out = await Truncate.output(result, {}, initCtx?.agent) return { title: "", output: out.truncated ? out.content : result, metadata: { truncated: out.truncated, outputPath: out.truncated ? out.outputPath : undefined }, } }, }), } } export async function register(tool: Tool.Info) { const { custom } = await state() const idx = custom.findIndex((t) => t.id === tool.id) if (idx >= 0) { custom.splice(idx, 1, tool) return } custom.push(tool) } async function all(): Promise { const custom = await state().then((x) => x.custom) const config = await Config.get() const question = ["app", "cli", "desktop"].includes(Flag.OPENCODE_CLIENT) || Flag.OPENCODE_ENABLE_QUESTION_TOOL return [ InvalidTool, ...(question ? [QuestionTool] : []), BashTool, ReadTool, GlobTool, GrepTool, EditTool, WriteTool, TaskTool, WebFetchTool, TodoWriteTool, // TodoReadTool, WebSearchTool, CodeSearchTool, SkillTool, ApplyPatchTool, ...(Flag.OPENCODE_EXPERIMENTAL_LSP_TOOL ? [LspTool] : []), ...(config.experimental?.batch_tool === true ? [BatchTool] : []), ...(Flag.OPENCODE_EXPERIMENTAL_PLAN_MODE && Flag.OPENCODE_CLIENT === "cli" ? [PlanExitTool] : []), ...custom, ] } export async function ids() { return all().then((x) => x.map((t) => t.id)) } export async function tools( model: { providerID: ProviderID modelID: ModelID }, agent?: Agent.Info, ) { const tools = await all() const result = await Promise.all( tools .filter((t) => { // Enable websearch/codesearch for zen users OR via enable flag if (t.id === "codesearch" || t.id === "websearch") { return model.providerID === ProviderID.opencode || Flag.OPENCODE_ENABLE_EXA } // use apply tool in same format as codex const usePatch = model.modelID.includes("gpt-") && !model.modelID.includes("oss") && !model.modelID.includes("gpt-4") if (t.id === "apply_patch") return usePatch if (t.id === "edit" || t.id === "write") return !usePatch return true }) .map(async (t) => { using _ = log.time(t.id) const tool = await t.init({ agent }) const output = { description: tool.description, parameters: tool.parameters, } await Plugin.trigger("tool.definition", { toolID: t.id }, output) return { id: t.id, ...tool, description: output.description, parameters: output.parameters, } }), ) return result } }