mirror of
https://github.com/anomalyco/opencode.git
synced 2026-05-16 09:33:24 +00:00
109 lines
3.9 KiB
TypeScript
109 lines
3.9 KiB
TypeScript
import { NodeHttpServer, NodeServices } from "@effect/platform-node"
|
|
import { Flag } from "@opencode-ai/core/flag/flag"
|
|
import { describe, expect } from "bun:test"
|
|
import { Config, ConfigProvider, Effect, Layer } from "effect"
|
|
import { HttpClient, HttpClientRequest, HttpRouter, HttpServer } from "effect/unstable/http"
|
|
import * as Socket from "effect/unstable/socket/Socket"
|
|
import { Server } from "../../src/server/server"
|
|
import { InstancePaths } from "../../src/server/routes/instance/httpapi/groups/instance"
|
|
import { ExperimentalHttpApiServer } from "../../src/server/routes/instance/httpapi/server"
|
|
import { resetDatabase } from "../fixture/db"
|
|
import { testEffect } from "../lib/effect"
|
|
|
|
const testStateLayer = Layer.effectDiscard(
|
|
Effect.gen(function* () {
|
|
const original = {
|
|
OPENCODE_SERVER_PASSWORD: Flag.OPENCODE_SERVER_PASSWORD,
|
|
}
|
|
Flag.OPENCODE_SERVER_PASSWORD = "secret"
|
|
yield* Effect.promise(() => resetDatabase())
|
|
yield* Effect.addFinalizer(() =>
|
|
Effect.promise(async () => {
|
|
Flag.OPENCODE_SERVER_PASSWORD = original.OPENCODE_SERVER_PASSWORD
|
|
await resetDatabase()
|
|
}),
|
|
)
|
|
}),
|
|
)
|
|
|
|
const servedRoutes: Layer.Layer<never, Config.ConfigError, HttpServer.HttpServer> = HttpRouter.serve(
|
|
ExperimentalHttpApiServer.routes,
|
|
{ disableListenLog: true, disableLogger: true },
|
|
)
|
|
|
|
const it = testEffect(
|
|
Layer.mergeAll(
|
|
testStateLayer,
|
|
servedRoutes.pipe(
|
|
Layer.provide(Socket.layerWebSocketConstructorGlobal),
|
|
Layer.provideMerge(NodeHttpServer.layerTest),
|
|
Layer.provideMerge(NodeServices.layer),
|
|
),
|
|
),
|
|
)
|
|
|
|
describe("HttpApi CORS", () => {
|
|
it.live("allows browser preflight requests without credentials", () =>
|
|
Effect.gen(function* () {
|
|
const response = yield* HttpClientRequest.options(InstancePaths.path).pipe(
|
|
HttpClientRequest.setHeaders({
|
|
origin: "http://localhost:3000",
|
|
"access-control-request-method": "GET",
|
|
"access-control-request-headers": "authorization",
|
|
}),
|
|
HttpClient.execute,
|
|
)
|
|
|
|
expect(response.status).toBe(204)
|
|
expect(response.headers["access-control-allow-origin"]).toBe("http://localhost:3000")
|
|
expect(response.headers["access-control-allow-headers"]).toBe("authorization")
|
|
}),
|
|
)
|
|
|
|
it.live("adds CORS headers to unauthorized responses", () =>
|
|
Effect.gen(function* () {
|
|
const handler = HttpRouter.toWebHandler(
|
|
ExperimentalHttpApiServer.createRoutes().pipe(
|
|
Layer.provide(ConfigProvider.layer(ConfigProvider.fromUnknown({ OPENCODE_SERVER_PASSWORD: "secret" }))),
|
|
),
|
|
{ disableLogger: true },
|
|
).handler
|
|
const response = yield* Effect.promise(() =>
|
|
handler(
|
|
new Request(new URL("/global/config", "http://localhost"), {
|
|
headers: { origin: "https://app.opencode.ai" },
|
|
}),
|
|
ExperimentalHttpApiServer.context,
|
|
),
|
|
)
|
|
|
|
expect(response.status).toBe(401)
|
|
expect(response.headers.get("access-control-allow-origin")).toBe("https://app.opencode.ai")
|
|
}),
|
|
)
|
|
|
|
it.live("uses custom CORS origins passed to the server", () =>
|
|
Effect.gen(function* () {
|
|
const listener = yield* Effect.acquireRelease(
|
|
Effect.promise(() => Server.listen({ hostname: "127.0.0.1", port: 0, cors: ["https://custom.example"] })),
|
|
(listener) => Effect.promise(() => listener.stop(true)),
|
|
)
|
|
|
|
const response = yield* Effect.promise(() =>
|
|
fetch(new URL(InstancePaths.path, listener.url), {
|
|
method: "OPTIONS",
|
|
headers: {
|
|
origin: "https://custom.example",
|
|
"access-control-request-method": "GET",
|
|
"access-control-request-headers": "authorization",
|
|
},
|
|
}),
|
|
)
|
|
|
|
expect(response.status).toBe(204)
|
|
expect(response.headers.get("access-control-allow-origin")).toBe("https://custom.example")
|
|
expect(response.headers.get("access-control-allow-headers")).toBe("authorization")
|
|
}),
|
|
)
|
|
})
|