mirror of
https://github.com/anomalyco/opencode.git
synced 2026-05-13 23:52:06 +00:00
This reverts: -28b03595bresearch: delete Hono backend (do not merge) (#25667) -b24a4e897chore(server): clean up post-Hono-deletion scar tissue (#26542) v1.14.42 broke startup for users with plugins that depend on the Hono wire format (most visibly opencode-gemini-auth, see #26546). Restoring Hono as the default backend on stable channels while we investigate the actual plugin compatibility story. OPENCODE_EXPERIMENTAL_HTTPAPI flag and dual-backend selection come back. Stable installs default to Hono; dev/beta default to HTTP API. Conflict resolution: took the pre-deletion side for control-plane schemas and the seven test files where post-deletion follow-up PRs had also touched the conflicting lines. The HTTP API code added since the deletion (compression, cors-vary, fence, lifecycle log, account error mapping, etc.) is preserved as-is — those still apply on the HTTP API path for users on dev/beta channels.
69 lines
2.7 KiB
TypeScript
69 lines
2.7 KiB
TypeScript
import { afterEach, describe, expect, test } from "bun:test"
|
|
import { Flag } from "@opencode-ai/core/flag/flag"
|
|
import { Instance } from "../../src/project/instance"
|
|
import { Server } from "../../src/server/server"
|
|
import { EventPaths } from "../../src/server/routes/instance/httpapi/event"
|
|
import * as Log from "@opencode-ai/core/util/log"
|
|
import { resetDatabase } from "../fixture/db"
|
|
import { disposeAllInstances, tmpdir } from "../fixture/fixture"
|
|
|
|
void Log.init({ print: false })
|
|
|
|
const original = Flag.OPENCODE_EXPERIMENTAL_HTTPAPI
|
|
|
|
function app(experimental = true) {
|
|
Flag.OPENCODE_EXPERIMENTAL_HTTPAPI = experimental
|
|
return experimental ? Server.Default().app : Server.Legacy().app
|
|
}
|
|
|
|
async function readFirstChunk(response: Response) {
|
|
if (!response.body) throw new Error("missing response body")
|
|
const reader = response.body.getReader()
|
|
const result = await Promise.race([
|
|
reader.read(),
|
|
new Promise<never>((_, reject) => setTimeout(() => reject(new Error("timed out waiting for event")), 5_000)),
|
|
])
|
|
await reader.cancel()
|
|
return new TextDecoder().decode(result.value)
|
|
}
|
|
|
|
async function readFirstEvent(response: Response) {
|
|
return JSON.parse((await readFirstChunk(response)).replace(/^data: /, "")) as {
|
|
id?: string
|
|
type: string
|
|
properties: Record<string, unknown>
|
|
}
|
|
}
|
|
|
|
afterEach(async () => {
|
|
Flag.OPENCODE_EXPERIMENTAL_HTTPAPI = original
|
|
await disposeAllInstances()
|
|
await resetDatabase()
|
|
})
|
|
|
|
describe("event HttpApi bridge", () => {
|
|
test("serves event stream through experimental Effect route", async () => {
|
|
await using tmp = await tmpdir({ git: true, config: { formatter: false, lsp: false } })
|
|
const response = await app().request(EventPaths.event, { headers: { "x-opencode-directory": tmp.path } })
|
|
|
|
expect(response.status).toBe(200)
|
|
expect(response.headers.get("content-type")).toContain("text/event-stream")
|
|
expect(response.headers.get("cache-control")).toBe("no-cache, no-transform")
|
|
expect(response.headers.get("x-accel-buffering")).toBe("no")
|
|
expect(response.headers.get("x-content-type-options")).toBe("nosniff")
|
|
expect(await readFirstEvent(response)).toMatchObject({ type: "server.connected", properties: {} })
|
|
})
|
|
|
|
test("matches legacy first event frame", async () => {
|
|
await using tmp = await tmpdir({ git: true, config: { formatter: false, lsp: false } })
|
|
const headers = { "x-opencode-directory": tmp.path }
|
|
const legacy = await app(false).request(EventPaths.event, { headers })
|
|
const effect = await app(true).request(EventPaths.event, { headers })
|
|
|
|
const legacyEvent = await readFirstEvent(legacy)
|
|
const effectEvent = await readFirstEvent(effect)
|
|
expect(effectEvent.type).toBe(legacyEvent.type)
|
|
expect(effectEvent.properties).toEqual(legacyEvent.properties)
|
|
})
|
|
})
|