mirror of
https://gitea.toothfairyai.com/ToothFairyAI/tf_code.git
synced 2026-04-07 17:28:53 +00:00
269 lines
8.3 KiB
TypeScript
269 lines
8.3 KiB
TypeScript
import { Hono } from "hono"
|
|
import { describeRoute, validator, resolver } from "hono-openapi"
|
|
import z from "zod"
|
|
import { ToolRegistry } from "../../tool/registry"
|
|
import { Worktree } from "../../worktree"
|
|
import { Instance } from "../../project/instance"
|
|
import { Project } from "../../project/project"
|
|
import { MCP } from "../../mcp"
|
|
import { Session } from "../../session"
|
|
import { zodToJsonSchema } from "zod-to-json-schema"
|
|
import { errors } from "../error"
|
|
import { lazy } from "../../util/lazy"
|
|
|
|
export const ExperimentalRoutes = lazy(() =>
|
|
new Hono()
|
|
.get(
|
|
"/tool/ids",
|
|
describeRoute({
|
|
summary: "List tool IDs",
|
|
description:
|
|
"Get a list of all available tool IDs, including both built-in tools and dynamically registered tools.",
|
|
operationId: "tool.ids",
|
|
responses: {
|
|
200: {
|
|
description: "Tool IDs",
|
|
content: {
|
|
"application/json": {
|
|
schema: resolver(z.array(z.string()).meta({ ref: "ToolIDs" })),
|
|
},
|
|
},
|
|
},
|
|
...errors(400),
|
|
},
|
|
}),
|
|
async (c) => {
|
|
return c.json(await ToolRegistry.ids())
|
|
},
|
|
)
|
|
.get(
|
|
"/tool",
|
|
describeRoute({
|
|
summary: "List tools",
|
|
description:
|
|
"Get a list of available tools with their JSON schema parameters for a specific provider and model combination.",
|
|
operationId: "tool.list",
|
|
responses: {
|
|
200: {
|
|
description: "Tools",
|
|
content: {
|
|
"application/json": {
|
|
schema: resolver(
|
|
z
|
|
.array(
|
|
z
|
|
.object({
|
|
id: z.string(),
|
|
description: z.string(),
|
|
parameters: z.any(),
|
|
})
|
|
.meta({ ref: "ToolListItem" }),
|
|
)
|
|
.meta({ ref: "ToolList" }),
|
|
),
|
|
},
|
|
},
|
|
},
|
|
...errors(400),
|
|
},
|
|
}),
|
|
validator(
|
|
"query",
|
|
z.object({
|
|
provider: z.string(),
|
|
model: z.string(),
|
|
}),
|
|
),
|
|
async (c) => {
|
|
const { provider, model } = c.req.valid("query")
|
|
const tools = await ToolRegistry.tools({ providerID: provider, modelID: model })
|
|
return c.json(
|
|
tools.map((t) => ({
|
|
id: t.id,
|
|
description: t.description,
|
|
// Handle both Zod schemas and plain JSON schemas
|
|
parameters: (t.parameters as any)?._def ? zodToJsonSchema(t.parameters as any) : t.parameters,
|
|
})),
|
|
)
|
|
},
|
|
)
|
|
.post(
|
|
"/worktree",
|
|
describeRoute({
|
|
summary: "Create worktree",
|
|
description: "Create a new git worktree for the current project and run any configured startup scripts.",
|
|
operationId: "worktree.create",
|
|
responses: {
|
|
200: {
|
|
description: "Worktree created",
|
|
content: {
|
|
"application/json": {
|
|
schema: resolver(Worktree.Info),
|
|
},
|
|
},
|
|
},
|
|
...errors(400),
|
|
},
|
|
}),
|
|
validator("json", Worktree.create.schema),
|
|
async (c) => {
|
|
const body = c.req.valid("json")
|
|
const worktree = await Worktree.create(body)
|
|
return c.json(worktree)
|
|
},
|
|
)
|
|
.get(
|
|
"/worktree",
|
|
describeRoute({
|
|
summary: "List worktrees",
|
|
description: "List all sandbox worktrees for the current project.",
|
|
operationId: "worktree.list",
|
|
responses: {
|
|
200: {
|
|
description: "List of worktree directories",
|
|
content: {
|
|
"application/json": {
|
|
schema: resolver(z.array(z.string())),
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}),
|
|
async (c) => {
|
|
const sandboxes = await Project.sandboxes(Instance.project.id)
|
|
return c.json(sandboxes)
|
|
},
|
|
)
|
|
.delete(
|
|
"/worktree",
|
|
describeRoute({
|
|
summary: "Remove worktree",
|
|
description: "Remove a git worktree and delete its branch.",
|
|
operationId: "worktree.remove",
|
|
responses: {
|
|
200: {
|
|
description: "Worktree removed",
|
|
content: {
|
|
"application/json": {
|
|
schema: resolver(z.boolean()),
|
|
},
|
|
},
|
|
},
|
|
...errors(400),
|
|
},
|
|
}),
|
|
validator("json", Worktree.remove.schema),
|
|
async (c) => {
|
|
const body = c.req.valid("json")
|
|
await Worktree.remove(body)
|
|
await Project.removeSandbox(Instance.project.id, body.directory)
|
|
return c.json(true)
|
|
},
|
|
)
|
|
.post(
|
|
"/worktree/reset",
|
|
describeRoute({
|
|
summary: "Reset worktree",
|
|
description: "Reset a worktree branch to the primary default branch.",
|
|
operationId: "worktree.reset",
|
|
responses: {
|
|
200: {
|
|
description: "Worktree reset",
|
|
content: {
|
|
"application/json": {
|
|
schema: resolver(z.boolean()),
|
|
},
|
|
},
|
|
},
|
|
...errors(400),
|
|
},
|
|
}),
|
|
validator("json", Worktree.reset.schema),
|
|
async (c) => {
|
|
const body = c.req.valid("json")
|
|
await Worktree.reset(body)
|
|
return c.json(true)
|
|
},
|
|
)
|
|
.get(
|
|
"/session",
|
|
describeRoute({
|
|
summary: "List sessions",
|
|
description:
|
|
"Get a list of all OpenCode sessions across projects, sorted by most recently updated. Archived sessions are excluded by default.",
|
|
operationId: "experimental.session.list",
|
|
responses: {
|
|
200: {
|
|
description: "List of sessions",
|
|
content: {
|
|
"application/json": {
|
|
schema: resolver(Session.GlobalInfo.array()),
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}),
|
|
validator(
|
|
"query",
|
|
z.object({
|
|
directory: z.string().optional().meta({ description: "Filter sessions by project directory" }),
|
|
roots: z.coerce.boolean().optional().meta({ description: "Only return root sessions (no parentID)" }),
|
|
start: z.coerce
|
|
.number()
|
|
.optional()
|
|
.meta({ description: "Filter sessions updated on or after this timestamp (milliseconds since epoch)" }),
|
|
cursor: z.coerce
|
|
.number()
|
|
.optional()
|
|
.meta({ description: "Return sessions updated before this timestamp (milliseconds since epoch)" }),
|
|
search: z.string().optional().meta({ description: "Filter sessions by title (case-insensitive)" }),
|
|
limit: z.coerce.number().optional().meta({ description: "Maximum number of sessions to return" }),
|
|
archived: z.coerce.boolean().optional().meta({ description: "Include archived sessions (default false)" }),
|
|
}),
|
|
),
|
|
async (c) => {
|
|
const query = c.req.valid("query")
|
|
const limit = query.limit ?? 100
|
|
const sessions: Session.GlobalInfo[] = []
|
|
for await (const session of Session.listGlobal({
|
|
directory: query.directory,
|
|
roots: query.roots,
|
|
start: query.start,
|
|
cursor: query.cursor,
|
|
search: query.search,
|
|
limit: limit + 1,
|
|
archived: query.archived,
|
|
})) {
|
|
sessions.push(session)
|
|
}
|
|
const hasMore = sessions.length > limit
|
|
const list = hasMore ? sessions.slice(0, limit) : sessions
|
|
if (hasMore && list.length > 0) {
|
|
c.header("x-next-cursor", String(list[list.length - 1].time.updated))
|
|
}
|
|
return c.json(list)
|
|
},
|
|
)
|
|
.get(
|
|
"/resource",
|
|
describeRoute({
|
|
summary: "Get MCP resources",
|
|
description: "Get all available MCP resources from connected servers. Optionally filter by name.",
|
|
operationId: "experimental.resource.list",
|
|
responses: {
|
|
200: {
|
|
description: "MCP resources",
|
|
content: {
|
|
"application/json": {
|
|
schema: resolver(z.record(z.string(), MCP.Resource)),
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}),
|
|
async (c) => {
|
|
return c.json(await MCP.resources())
|
|
},
|
|
),
|
|
)
|