mirror of
https://gitea.toothfairyai.com/ToothFairyAI/tf_code.git
synced 2026-03-29 21:33:54 +00:00
175 lines
5.5 KiB
TypeScript
175 lines
5.5 KiB
TypeScript
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<ToolDefinition>(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<Tool.Info[]> {
|
|
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
|
|
}
|
|
}
|