mirror of
https://github.com/anomalyco/opencode.git
synced 2026-05-25 06:35:35 +00:00
- workspace-routing best-effort probe: catch ONLY SessionNotFound (was Effect.catch(() => undefined) which swallowed all E-channel errors, broader than the prior catchDefect-only behavior). Drop the now-redundant catchDefect since session.get fails typed instead of throwing. - trace.ts: TODO(typed-errors) marker on the legacy Hono shim so it greps as a known retirement target. - httpapi-parity.test.ts: drop stale FIXME comment above the now- passing reproducer.
126 lines
6 KiB
TypeScript
126 lines
6 KiB
TypeScript
import { afterEach, describe, expect, test } from "bun:test"
|
|
import { Effect } from "effect"
|
|
import { Flag } from "@opencode-ai/core/flag/flag"
|
|
import * as Log from "@opencode-ai/core/util/log"
|
|
import { WithInstance } from "../../src/project/with-instance"
|
|
import { Server } from "../../src/server/server"
|
|
import { Session } from "@/session/session"
|
|
import { MessageID } from "../../src/session/schema"
|
|
import { ModelID, ProviderID } from "../../src/provider/schema"
|
|
import { resetDatabase } from "../fixture/db"
|
|
import { disposeAllInstances, tmpdir } from "../fixture/fixture"
|
|
|
|
void Log.init({ print: false })
|
|
|
|
const original = Flag.OPENCODE_EXPERIMENTAL_HTTPAPI
|
|
|
|
afterEach(async () => {
|
|
Flag.OPENCODE_EXPERIMENTAL_HTTPAPI = original
|
|
await disposeAllInstances()
|
|
await resetDatabase()
|
|
})
|
|
|
|
function app(experimental: boolean) {
|
|
Flag.OPENCODE_EXPERIMENTAL_HTTPAPI = experimental
|
|
return experimental ? Server.Default().app : Server.Legacy().app
|
|
}
|
|
|
|
function runSession<A, E>(fx: Effect.Effect<A, E, Session.Service>) {
|
|
return Effect.runPromise(fx.pipe(Effect.provide(Session.defaultLayer)))
|
|
}
|
|
|
|
function createSessionWithMessages(directory: string, count: number) {
|
|
return WithInstance.provide({
|
|
directory,
|
|
fn: async () => {
|
|
const session = await runSession(Session.Service.use((svc) => svc.create({})))
|
|
for (let i = 0; i < count; i++) {
|
|
await runSession(
|
|
Effect.gen(function* () {
|
|
const svc = yield* Session.Service
|
|
yield* svc.updateMessage({
|
|
id: MessageID.ascending(),
|
|
role: "user",
|
|
sessionID: session.id,
|
|
agent: "build",
|
|
model: { providerID: ProviderID.make("test"), modelID: ModelID.make("test") },
|
|
time: { created: Date.now() },
|
|
})
|
|
}),
|
|
)
|
|
}
|
|
return session.id
|
|
},
|
|
})
|
|
}
|
|
|
|
// ──────────────────────────────────────────────────────────────────────────────
|
|
// Reproducer 1: Link header should reflect the request's actual Host header,
|
|
// not "localhost". HttpApi uses `new URL(request.url, "http://localhost")`
|
|
// which embeds localhost because request.url is path-only. Fix: use
|
|
// `HttpServerRequest.toURL(request)` which honors the Host header.
|
|
// ──────────────────────────────────────────────────────────────────────────────
|
|
describe("Link header host", () => {
|
|
test("HttpApi pagination Link header echoes request host", async () => {
|
|
await using tmp = await tmpdir({ config: { formatter: false, lsp: false } })
|
|
const sessionID = await createSessionWithMessages(tmp.path, 3)
|
|
|
|
const response = await app(true).request(`/session/${sessionID}/message?limit=2`, {
|
|
headers: {
|
|
host: "opencode.test:4096",
|
|
"x-opencode-directory": tmp.path,
|
|
},
|
|
})
|
|
|
|
expect(response.status).toBe(200)
|
|
const link = response.headers.get("link")
|
|
expect(link).not.toBeNull()
|
|
// Link should contain the request's Host, not "localhost".
|
|
expect(link).toContain("opencode.test")
|
|
expect(link).not.toContain("localhost")
|
|
})
|
|
})
|
|
|
|
// ──────────────────────────────────────────────────────────────────────────────
|
|
// Reproducer 2: GET /session/{missing-id}/todo should return 404, not 500.
|
|
// The session.todo handler in HttpApi doesn't wrap with `mapNotFound`, so a
|
|
// `NotFoundError` from the service surfaces as a defect → 500. Hono's
|
|
// equivalent maps to 404 via `errors.notFound`.
|
|
//
|
|
// Affected endpoints (handlers without mapNotFound): todo, diff, summarize,
|
|
// fork, abort, init, deleteMessage, command, shell, revert, unrevert.
|
|
//
|
|
// FIXME: unskip when mapNotFound coverage is added (next PR).
|
|
// ──────────────────────────────────────────────────────────────────────────────
|
|
describe("404 mapping for missing session", () => {
|
|
test.todo("HttpApi /session/{missing}/todo returns 404 not 500", async () => {
|
|
await using tmp = await tmpdir({ config: { formatter: false, lsp: false } })
|
|
|
|
const response = await app(true).request("/session/ses_does_not_exist/todo", {
|
|
headers: { "x-opencode-directory": tmp.path },
|
|
})
|
|
|
|
expect(response.status).toBe(404)
|
|
})
|
|
})
|
|
|
|
// ──────────────────────────────────────────────────────────────────────────────
|
|
// Reproducer 3: 404 response body matches Hono's NamedError envelope
|
|
// `{ name, data: { message } }`. `Session.get` now fails with a typed
|
|
// `SessionNotFound` annotated `httpApiStatus: 404`. The endpoint declares the
|
|
// schema; HttpApi auto-routes status + body — no `mapNotFound` wrapper.
|
|
// ──────────────────────────────────────────────────────────────────────────────
|
|
describe("Error JSON shape parity", () => {
|
|
test("HttpApi 404 body matches NamedError shape (sketch: session.get)", async () => {
|
|
await using tmp = await tmpdir({ config: { formatter: false, lsp: false } })
|
|
|
|
const response = await app(true).request("/session/ses_does_not_exist", {
|
|
headers: { "x-opencode-directory": tmp.path },
|
|
})
|
|
|
|
expect(response.status).toBe(404)
|
|
const body = (await response.json()) as { name?: string; data?: { message?: string } }
|
|
expect(body.name).toBe("NotFoundError")
|
|
expect(typeof body.data?.message).toBe("string")
|
|
})
|
|
})
|