|
|
@@ -1,67 +1,30 @@
|
|
|
-import { createHash } from "node:crypto"
|
|
|
import { Log } from "../util/log"
|
|
|
import { describeRoute, generateSpecs, validator, resolver, openAPIRouteHandler } from "hono-openapi"
|
|
|
import { Hono } from "hono"
|
|
|
import { compress } from "hono/compress"
|
|
|
import { cors } from "hono/cors"
|
|
|
-import { proxy } from "hono/proxy"
|
|
|
import { basicAuth } from "hono/basic-auth"
|
|
|
import z from "zod"
|
|
|
-import { Provider } from "../provider/provider"
|
|
|
-import { NamedError } from "@opencode-ai/util/error"
|
|
|
-import { LSP } from "../lsp"
|
|
|
-import { Format } from "../format"
|
|
|
-import { TuiRoutes } from "./routes/tui"
|
|
|
-import { Instance } from "../project/instance"
|
|
|
-import { Vcs } from "../project/vcs"
|
|
|
-import { Agent } from "../agent/agent"
|
|
|
-import { Skill } from "../skill"
|
|
|
import { Auth } from "../auth"
|
|
|
import { Flag } from "../flag/flag"
|
|
|
-import { Command } from "../command"
|
|
|
-import { Global } from "../global"
|
|
|
-import { WorkspaceID } from "../control-plane/schema"
|
|
|
import { ProviderID } from "../provider/schema"
|
|
|
import { WorkspaceRouterMiddleware } from "../control-plane/workspace-router-middleware"
|
|
|
-import { ProjectRoutes } from "./routes/project"
|
|
|
-import { SessionRoutes } from "./routes/session"
|
|
|
-import { PtyRoutes } from "./routes/pty"
|
|
|
-import { McpRoutes } from "./routes/mcp"
|
|
|
-import { FileRoutes } from "./routes/file"
|
|
|
-import { ConfigRoutes } from "./routes/config"
|
|
|
-import { ExperimentalRoutes } from "./routes/experimental"
|
|
|
-import { ProviderRoutes } from "./routes/provider"
|
|
|
-import { EventRoutes } from "./routes/event"
|
|
|
-import { InstanceBootstrap } from "../project/bootstrap"
|
|
|
-import { NotFoundError } from "../storage/db"
|
|
|
-import type { ContentfulStatusCode } from "hono/utils/http-status"
|
|
|
import { websocket } from "hono/bun"
|
|
|
-import { HTTPException } from "hono/http-exception"
|
|
|
import { errors } from "./error"
|
|
|
-import { Filesystem } from "@/util/filesystem"
|
|
|
-import { QuestionRoutes } from "./routes/question"
|
|
|
-import { PermissionRoutes } from "./routes/permission"
|
|
|
import { GlobalRoutes } from "./routes/global"
|
|
|
import { MDNS } from "./mdns"
|
|
|
import { lazy } from "@/util/lazy"
|
|
|
+import { errorHandler } from "./middleware"
|
|
|
+import { InstanceRoutes } from "./instance"
|
|
|
import { initProjectors } from "./projectors"
|
|
|
|
|
|
// @ts-ignore This global is needed to prevent ai-sdk from logging warnings to stdout https://github.com/vercel/ai/blob/2dc67e0ef538307f21368db32d5a12345d98831b/packages/ai/src/logger/log-warnings.ts#L85
|
|
|
globalThis.AI_SDK_LOG_WARNINGS = false
|
|
|
|
|
|
-const csp = (hash = "") =>
|
|
|
- `default-src 'self'; script-src 'self' 'wasm-unsafe-eval'${hash ? ` 'sha256-${hash}'` : ""}; style-src 'self' 'unsafe-inline'; img-src 'self' data: https:; font-src 'self' data:; media-src 'self' data:; connect-src 'self' data:`
|
|
|
-
|
|
|
initProjectors()
|
|
|
|
|
|
export namespace Server {
|
|
|
const log = Log.create({ service: "server" })
|
|
|
- const DEFAULT_CSP =
|
|
|
- "default-src 'self'; script-src 'self' 'wasm-unsafe-eval'; style-src 'self' 'unsafe-inline'; img-src 'self' data: https:; font-src 'self' data:; media-src 'self' data:; connect-src 'self' data:"
|
|
|
- const embeddedUIPromise = Flag.OPENCODE_DISABLE_EMBEDDED_WEB_UI
|
|
|
- ? Promise.resolve(null)
|
|
|
- : // @ts-expect-error - generated file at build time
|
|
|
- import("opencode-web-ui.gen.ts").then((module) => module.default as Record<string, string>).catch(() => null)
|
|
|
|
|
|
const zipped = compress()
|
|
|
|
|
|
@@ -71,30 +34,12 @@ export namespace Server {
|
|
|
return false
|
|
|
}
|
|
|
|
|
|
- export const Default = lazy(() => createApp({}))
|
|
|
+ export const Default = lazy(() => ControlPlaneRoutes())
|
|
|
|
|
|
- export const createApp = (opts: { cors?: string[] }): Hono => {
|
|
|
+ export const ControlPlaneRoutes = (opts?: { cors?: string[] }): Hono => {
|
|
|
const app = new Hono()
|
|
|
return app
|
|
|
- .onError((err, c) => {
|
|
|
- log.error("failed", {
|
|
|
- error: err,
|
|
|
- })
|
|
|
- if (err instanceof NamedError) {
|
|
|
- let status: ContentfulStatusCode
|
|
|
- if (err instanceof NotFoundError) status = 404
|
|
|
- else if (err instanceof Provider.ModelNotFoundError) status = 400
|
|
|
- else if (err.name === "ProviderAuthValidationFailed") status = 400
|
|
|
- else if (err.name.startsWith("Worktree")) status = 400
|
|
|
- else status = 500
|
|
|
- return c.json(err.toObject(), { status })
|
|
|
- }
|
|
|
- if (err instanceof HTTPException) return err.getResponse()
|
|
|
- const message = err instanceof Error && err.stack ? err.stack : err.toString()
|
|
|
- return c.json(new NamedError.Unknown({ message }).toObject(), {
|
|
|
- status: 500,
|
|
|
- })
|
|
|
- })
|
|
|
+ .onError(errorHandler(log))
|
|
|
.use((c, next) => {
|
|
|
// Allow CORS preflight requests to succeed without auth.
|
|
|
// Browser clients sending Authorization headers will preflight with OPTIONS.
|
|
|
@@ -105,8 +50,8 @@ export namespace Server {
|
|
|
return basicAuth({ username, password })(c, next)
|
|
|
})
|
|
|
.use(async (c, next) => {
|
|
|
- const skipLogging = c.req.path === "/log"
|
|
|
- if (!skipLogging) {
|
|
|
+ const skip = c.req.path === "/log"
|
|
|
+ if (!skip) {
|
|
|
log.info("request", {
|
|
|
method: c.req.method,
|
|
|
path: c.req.path,
|
|
|
@@ -117,7 +62,7 @@ export namespace Server {
|
|
|
path: c.req.path,
|
|
|
})
|
|
|
await next()
|
|
|
- if (!skipLogging) {
|
|
|
+ if (!skip) {
|
|
|
timer.stop()
|
|
|
}
|
|
|
})
|
|
|
@@ -215,27 +160,6 @@ export namespace Server {
|
|
|
return c.json(true)
|
|
|
},
|
|
|
)
|
|
|
- .use(async (c, next) => {
|
|
|
- if (c.req.path === "/log") return next()
|
|
|
- const raw = c.req.query("directory") || c.req.header("x-opencode-directory") || process.cwd()
|
|
|
- const directory = Filesystem.resolve(
|
|
|
- (() => {
|
|
|
- try {
|
|
|
- return decodeURIComponent(raw)
|
|
|
- } catch {
|
|
|
- return raw
|
|
|
- }
|
|
|
- })(),
|
|
|
- )
|
|
|
-
|
|
|
- return Instance.provide({
|
|
|
- directory,
|
|
|
- init: InstanceBootstrap,
|
|
|
- async fn() {
|
|
|
- return next()
|
|
|
- },
|
|
|
- })
|
|
|
- })
|
|
|
.get(
|
|
|
"/doc",
|
|
|
openAPIRouteHandler(app, {
|
|
|
@@ -258,126 +182,6 @@ export namespace Server {
|
|
|
}),
|
|
|
),
|
|
|
)
|
|
|
- .use(WorkspaceRouterMiddleware)
|
|
|
- .route("/project", ProjectRoutes())
|
|
|
- .route("/pty", PtyRoutes())
|
|
|
- .route("/config", ConfigRoutes())
|
|
|
- .route("/experimental", ExperimentalRoutes())
|
|
|
- .route("/session", SessionRoutes())
|
|
|
- .route("/permission", PermissionRoutes())
|
|
|
- .route("/question", QuestionRoutes())
|
|
|
- .route("/provider", ProviderRoutes())
|
|
|
- .route("/", FileRoutes())
|
|
|
- .route("/", EventRoutes())
|
|
|
- .route("/mcp", McpRoutes())
|
|
|
- .route("/tui", TuiRoutes())
|
|
|
- .post(
|
|
|
- "/instance/dispose",
|
|
|
- describeRoute({
|
|
|
- summary: "Dispose instance",
|
|
|
- description: "Clean up and dispose the current OpenCode instance, releasing all resources.",
|
|
|
- operationId: "instance.dispose",
|
|
|
- responses: {
|
|
|
- 200: {
|
|
|
- description: "Instance disposed",
|
|
|
- content: {
|
|
|
- "application/json": {
|
|
|
- schema: resolver(z.boolean()),
|
|
|
- },
|
|
|
- },
|
|
|
- },
|
|
|
- },
|
|
|
- }),
|
|
|
- async (c) => {
|
|
|
- await Instance.dispose()
|
|
|
- return c.json(true)
|
|
|
- },
|
|
|
- )
|
|
|
- .get(
|
|
|
- "/path",
|
|
|
- describeRoute({
|
|
|
- summary: "Get paths",
|
|
|
- description: "Retrieve the current working directory and related path information for the OpenCode instance.",
|
|
|
- operationId: "path.get",
|
|
|
- responses: {
|
|
|
- 200: {
|
|
|
- description: "Path",
|
|
|
- content: {
|
|
|
- "application/json": {
|
|
|
- schema: resolver(
|
|
|
- z
|
|
|
- .object({
|
|
|
- home: z.string(),
|
|
|
- state: z.string(),
|
|
|
- config: z.string(),
|
|
|
- worktree: z.string(),
|
|
|
- directory: z.string(),
|
|
|
- })
|
|
|
- .meta({
|
|
|
- ref: "Path",
|
|
|
- }),
|
|
|
- ),
|
|
|
- },
|
|
|
- },
|
|
|
- },
|
|
|
- },
|
|
|
- }),
|
|
|
- async (c) => {
|
|
|
- return c.json({
|
|
|
- home: Global.Path.home,
|
|
|
- state: Global.Path.state,
|
|
|
- config: Global.Path.config,
|
|
|
- worktree: Instance.worktree,
|
|
|
- directory: Instance.directory,
|
|
|
- })
|
|
|
- },
|
|
|
- )
|
|
|
- .get(
|
|
|
- "/vcs",
|
|
|
- describeRoute({
|
|
|
- summary: "Get VCS info",
|
|
|
- description: "Retrieve version control system (VCS) information for the current project, such as git branch.",
|
|
|
- operationId: "vcs.get",
|
|
|
- responses: {
|
|
|
- 200: {
|
|
|
- description: "VCS info",
|
|
|
- content: {
|
|
|
- "application/json": {
|
|
|
- schema: resolver(Vcs.Info),
|
|
|
- },
|
|
|
- },
|
|
|
- },
|
|
|
- },
|
|
|
- }),
|
|
|
- async (c) => {
|
|
|
- const branch = await Vcs.branch()
|
|
|
- return c.json({
|
|
|
- branch,
|
|
|
- })
|
|
|
- },
|
|
|
- )
|
|
|
- .get(
|
|
|
- "/command",
|
|
|
- describeRoute({
|
|
|
- summary: "List commands",
|
|
|
- description: "Get a list of all available commands in the OpenCode system.",
|
|
|
- operationId: "command.list",
|
|
|
- responses: {
|
|
|
- 200: {
|
|
|
- description: "List of commands",
|
|
|
- content: {
|
|
|
- "application/json": {
|
|
|
- schema: resolver(Command.Info.array()),
|
|
|
- },
|
|
|
- },
|
|
|
- },
|
|
|
- },
|
|
|
- }),
|
|
|
- async (c) => {
|
|
|
- const commands = await Command.list()
|
|
|
- return c.json(commands)
|
|
|
- },
|
|
|
- )
|
|
|
.post(
|
|
|
"/log",
|
|
|
describeRoute({
|
|
|
@@ -430,132 +234,21 @@ export namespace Server {
|
|
|
return c.json(true)
|
|
|
},
|
|
|
)
|
|
|
- .get(
|
|
|
- "/agent",
|
|
|
- describeRoute({
|
|
|
- summary: "List agents",
|
|
|
- description: "Get a list of all available AI agents in the OpenCode system.",
|
|
|
- operationId: "app.agents",
|
|
|
- responses: {
|
|
|
- 200: {
|
|
|
- description: "List of agents",
|
|
|
- content: {
|
|
|
- "application/json": {
|
|
|
- schema: resolver(Agent.Info.array()),
|
|
|
- },
|
|
|
- },
|
|
|
- },
|
|
|
- },
|
|
|
- }),
|
|
|
- async (c) => {
|
|
|
- const modes = await Agent.list()
|
|
|
- return c.json(modes)
|
|
|
- },
|
|
|
- )
|
|
|
- .get(
|
|
|
- "/skill",
|
|
|
- describeRoute({
|
|
|
- summary: "List skills",
|
|
|
- description: "Get a list of all available skills in the OpenCode system.",
|
|
|
- operationId: "app.skills",
|
|
|
- responses: {
|
|
|
- 200: {
|
|
|
- description: "List of skills",
|
|
|
- content: {
|
|
|
- "application/json": {
|
|
|
- schema: resolver(Skill.Info.array()),
|
|
|
- },
|
|
|
- },
|
|
|
- },
|
|
|
- },
|
|
|
- }),
|
|
|
- async (c) => {
|
|
|
- const skills = await Skill.all()
|
|
|
- return c.json(skills)
|
|
|
- },
|
|
|
- )
|
|
|
- .get(
|
|
|
- "/lsp",
|
|
|
- describeRoute({
|
|
|
- summary: "Get LSP status",
|
|
|
- description: "Get LSP server status",
|
|
|
- operationId: "lsp.status",
|
|
|
- responses: {
|
|
|
- 200: {
|
|
|
- description: "LSP server status",
|
|
|
- content: {
|
|
|
- "application/json": {
|
|
|
- schema: resolver(LSP.Status.array()),
|
|
|
- },
|
|
|
- },
|
|
|
- },
|
|
|
- },
|
|
|
- }),
|
|
|
- async (c) => {
|
|
|
- return c.json(await LSP.status())
|
|
|
- },
|
|
|
- )
|
|
|
- .get(
|
|
|
- "/formatter",
|
|
|
- describeRoute({
|
|
|
- summary: "Get formatter status",
|
|
|
- description: "Get formatter status",
|
|
|
- operationId: "formatter.status",
|
|
|
- responses: {
|
|
|
- 200: {
|
|
|
- description: "Formatter status",
|
|
|
- content: {
|
|
|
- "application/json": {
|
|
|
- schema: resolver(Format.Status.array()),
|
|
|
- },
|
|
|
- },
|
|
|
- },
|
|
|
- },
|
|
|
- }),
|
|
|
- async (c) => {
|
|
|
- return c.json(await Format.status())
|
|
|
- },
|
|
|
- )
|
|
|
- .all("/*", async (c) => {
|
|
|
- const embeddedWebUI = await embeddedUIPromise
|
|
|
- const path = c.req.path
|
|
|
+ .use(WorkspaceRouterMiddleware)
|
|
|
+ }
|
|
|
|
|
|
- if (embeddedWebUI) {
|
|
|
- const match = embeddedWebUI[path.replace(/^\//, "")] ?? embeddedWebUI["index.html"] ?? null
|
|
|
- if (!match) return c.json({ error: "Not Found" }, 404)
|
|
|
- const file = Bun.file(match)
|
|
|
- if (await file.exists()) {
|
|
|
- c.header("Content-Type", file.type)
|
|
|
- if (file.type.startsWith("text/html")) {
|
|
|
- c.header("Content-Security-Policy", DEFAULT_CSP)
|
|
|
- }
|
|
|
- return c.body(await file.arrayBuffer())
|
|
|
- } else {
|
|
|
- return c.json({ error: "Not Found" }, 404)
|
|
|
- }
|
|
|
- } else {
|
|
|
- const response = await proxy(`https://app.opencode.ai${path}`, {
|
|
|
- ...c.req,
|
|
|
- headers: {
|
|
|
- ...c.req.raw.headers,
|
|
|
- host: "app.opencode.ai",
|
|
|
- },
|
|
|
- })
|
|
|
- const match = response.headers.get("content-type")?.includes("text/html")
|
|
|
- ? (await response.clone().text()).match(
|
|
|
- /<script\b(?![^>]*\bsrc\s*=)[^>]*\bid=(['"])oc-theme-preload-script\1[^>]*>([\s\S]*?)<\/script>/i,
|
|
|
- )
|
|
|
- : undefined
|
|
|
- const hash = match ? createHash("sha256").update(match[2]).digest("base64") : ""
|
|
|
- response.headers.set("Content-Security-Policy", csp(hash))
|
|
|
- return response
|
|
|
- }
|
|
|
- }) as unknown as Hono
|
|
|
+ export function createApp(opts: { cors?: string[] }) {
|
|
|
+ return ControlPlaneRoutes(opts)
|
|
|
}
|
|
|
|
|
|
export async function openapi() {
|
|
|
- // Cast to break excessive type recursion from long route chains
|
|
|
- const result = await generateSpecs(Default(), {
|
|
|
+ // Build a fresh app with all routes registered directly so
|
|
|
+ // hono-openapi can see describeRoute metadata (`.route()` wraps
|
|
|
+ // handlers when the sub-app has a custom errorHandler, which
|
|
|
+ // strips the metadata symbol).
|
|
|
+ const app = ControlPlaneRoutes()
|
|
|
+ InstanceRoutes(app)
|
|
|
+ const result = await generateSpecs(app, {
|
|
|
documentation: {
|
|
|
info: {
|
|
|
title: "opencode",
|
|
|
@@ -579,7 +272,7 @@ export namespace Server {
|
|
|
cors?: string[]
|
|
|
}) {
|
|
|
url = new URL(`http://${opts.hostname}:${opts.port}`)
|
|
|
- const app = createApp(opts)
|
|
|
+ const app = ControlPlaneRoutes({ cors: opts.cors })
|
|
|
const args = {
|
|
|
hostname: opts.hostname,
|
|
|
idleTimeout: 0,
|