import { describe, expect } from "bun:test" import { Effect } from "effect" import { Catalog } from "@opencode-ai/core/catalog" import { Credential } from "@opencode-ai/core/credential" import { Integration } from "@opencode-ai/core/integration" import { ModelV2 } from "@opencode-ai/core/model" import { PluginV2 } from "@opencode-ai/core/plugin" import { PluginHost } from "@opencode-ai/core/plugin/host" import { OpencodePlugin } from "@opencode-ai/core/plugin/provider/opencode" import { ProviderV2 } from "@opencode-ai/core/provider" import { testEffect } from "../lib/effect" import { PluginTestLayer } from "./fixture" const it = testEffect(PluginTestLayer) const addPlugin = Effect.fn(function* () { const plugin = yield* PluginV2.Service const host = yield* PluginHost.make(plugin) const integration = yield* Integration.Service yield* OpencodePlugin.effect(host).pipe(Effect.provideService(Integration.Service, integration)) }) function required(value: T | undefined): T { if (value === undefined) throw new Error("Expected value") return value } function withEnv(vars: Record, effect: () => Effect.Effect) { return Effect.acquireUseRelease( Effect.sync(() => { const previous = Object.fromEntries(Object.keys(vars).map((key) => [key, process.env[key]])) Object.entries(vars).forEach(([key, value]) => { if (value === undefined) delete process.env[key] else process.env[key] = value }) return previous }), effect, (previous) => Effect.sync(() => Object.entries(previous).forEach(([key, value]) => { if (value === undefined) delete process.env[key] else process.env[key] = value }), ), ) } const cost = (input: number, output = 0) => [{ input, output, cache: { read: 0, write: 0 } }] describe("OpencodePlugin", () => { it.effect("registers OAuth and API key methods", () => Effect.gen(function* () { yield* addPlugin() expect((yield* (yield* Integration.Service).get(Integration.ID.make("opencode")))?.methods).toEqual([ { id: Integration.MethodID.make("device"), type: "oauth", label: "Sign in with OpenCode", prompts: [ { type: "text", key: "server", message: "OpenCode server", placeholder: "https://console.opencode.ai", }, ], }, { type: "key", label: "API key" }, ]) }), ) it.live("loads providers and models from the connected OpenCode server", () => Effect.acquireUseRelease( Effect.sync(() => { const authorization: Array = [] return { authorization, server: Bun.serve({ port: 0, fetch: (request) => { authorization.push(request.headers.get("authorization")) return Response.json({ config: { providers: { remote: { name: "Remote", models: { model: { name: "Remote Model" }, }, }, }, }, }) }, }), } }), ({ authorization, server }) => Effect.gen(function* () { const credentials = yield* Credential.Service yield* credentials.create({ integrationID: Integration.ID.make("opencode"), value: new Credential.Key({ type: "key", key: "secret", metadata: { server: server.url.origin }, }), }) yield* addPlugin() const catalog = yield* Catalog.Service expect((yield* catalog.provider.get(ProviderV2.ID.make("remote")))?.name).toBe("Remote") expect((yield* catalog.model.get(ProviderV2.ID.make("remote"), ModelV2.ID.make("model")))?.name).toBe( "Remote Model", ) expect(authorization).toContain("Bearer secret") }), ({ server }) => Effect.promise(() => server.stop(true)), ), ) it.effect("uses a public key and disables paid models without credentials", () => withEnv({ OPENCODE_API_KEY: undefined }, () => Effect.gen(function* () { const catalog = yield* Catalog.Service yield* catalog.transform((catalog) => { const provider = new ProviderV2.Info({ ...ProviderV2.Info.empty(ProviderV2.ID.opencode), api: { type: "aisdk", package: "test-provider" }, }) const model = new ModelV2.Info({ ...ModelV2.Info.empty(provider.id, ModelV2.ID.make("paid")), api: { id: ModelV2.ID.make("paid"), type: "aisdk", package: "test-provider" }, cost: cost(1), }) catalog.provider.update(provider.id, () => {}) catalog.model.update(provider.id, model.id, (draft) => { draft.cost = [...model.cost] }) }) yield* addPlugin() expect(required(yield* catalog.provider.get(ProviderV2.ID.opencode)).request.body.apiKey).toBe("public") expect(required(yield* catalog.model.get(ProviderV2.ID.opencode, ModelV2.ID.make("paid"))).enabled).toBe(false) }), ), ) it.effect("keeps free models without credentials", () => withEnv({ OPENCODE_API_KEY: undefined }, () => Effect.gen(function* () { const catalog = yield* Catalog.Service yield* catalog.transform((catalog) => { const provider = new ProviderV2.Info({ ...ProviderV2.Info.empty(ProviderV2.ID.opencode), api: { type: "aisdk", package: "test-provider" }, }) const model = new ModelV2.Info({ ...ModelV2.Info.empty(provider.id, ModelV2.ID.make("free")), api: { id: ModelV2.ID.make("free"), type: "aisdk", package: "test-provider" }, cost: cost(0), }) catalog.provider.update(provider.id, () => {}) catalog.model.update(provider.id, model.id, (draft) => { draft.cost = [...model.cost] }) }) yield* addPlugin() expect(required(yield* catalog.provider.get(ProviderV2.ID.opencode)).request.body.apiKey).toBe("public") expect(required(yield* catalog.model.get(ProviderV2.ID.opencode, ModelV2.ID.make("free"))).enabled).toBe(true) }), ), ) it.effect("treats output-only cost as free without credentials", () => withEnv({ OPENCODE_API_KEY: undefined }, () => Effect.gen(function* () { const catalog = yield* Catalog.Service yield* catalog.transform((catalog) => { const provider = new ProviderV2.Info({ ...ProviderV2.Info.empty(ProviderV2.ID.opencode), api: { type: "aisdk", package: "test-provider" }, }) const model = new ModelV2.Info({ ...ModelV2.Info.empty(provider.id, ModelV2.ID.make("output-only")), api: { id: ModelV2.ID.make("output-only"), type: "aisdk", package: "test-provider" }, cost: cost(0, 1), }) catalog.provider.update(provider.id, () => {}) catalog.model.update(provider.id, model.id, (draft) => { draft.cost = [...model.cost] }) }) yield* addPlugin() expect(required(yield* catalog.provider.get(ProviderV2.ID.opencode)).request.body.apiKey).toBe("public") expect(required(yield* catalog.model.get(ProviderV2.ID.opencode, ModelV2.ID.make("output-only"))).enabled).toBe( true, ) }), ), ) it.effect("uses OPENCODE_API_KEY as credentials", () => withEnv({ OPENCODE_API_KEY: "secret" }, () => Effect.gen(function* () { const catalog = yield* Catalog.Service yield* catalog.transform((catalog) => { const provider = new ProviderV2.Info({ ...ProviderV2.Info.empty(ProviderV2.ID.opencode), api: { type: "aisdk", package: "test-provider" }, }) const model = new ModelV2.Info({ ...ModelV2.Info.empty(provider.id, ModelV2.ID.make("paid")), api: { id: ModelV2.ID.make("paid"), type: "aisdk", package: "test-provider" }, cost: cost(1), }) catalog.provider.update(provider.id, () => {}) catalog.model.update(provider.id, model.id, (draft) => { draft.cost = [...model.cost] }) }) yield* addPlugin() expect(required(yield* catalog.provider.get(ProviderV2.ID.opencode)).request.body.apiKey).toBeUndefined() expect(required(yield* catalog.model.get(ProviderV2.ID.opencode, ModelV2.ID.make("paid"))).enabled).toBe(true) }), ), ) it.effect("uses configured provider env vars as credentials", () => withEnv({ OPENCODE_API_KEY: undefined, CUSTOM_OPENCODE_API_KEY: "secret" }, () => Effect.gen(function* () { const catalog = yield* Catalog.Service const integrations = yield* Integration.Service yield* integrations.transform((editor) => { editor.method.update({ integrationID: Integration.ID.make("opencode"), method: { type: "env", names: ["CUSTOM_OPENCODE_API_KEY"] }, }) }) yield* catalog.transform((catalog) => { const provider = new ProviderV2.Info({ ...ProviderV2.Info.empty(ProviderV2.ID.opencode), api: { type: "aisdk", package: "test-provider" }, }) const model = new ModelV2.Info({ ...ModelV2.Info.empty(provider.id, ModelV2.ID.make("paid")), api: { id: ModelV2.ID.make("paid"), type: "aisdk", package: "test-provider" }, cost: cost(1), }) catalog.provider.update(provider.id, () => {}) catalog.model.update(provider.id, model.id, (draft) => { draft.cost = [...model.cost] }) }) yield* addPlugin() expect(required(yield* catalog.provider.get(ProviderV2.ID.opencode)).request.body.apiKey).toBeUndefined() expect(required(yield* catalog.model.get(ProviderV2.ID.opencode, ModelV2.ID.make("paid"))).enabled).toBe(true) }), ), ) it.effect("uses configured apiKey as credentials", () => withEnv({ OPENCODE_API_KEY: undefined }, () => Effect.gen(function* () { const catalog = yield* Catalog.Service yield* catalog.transform((catalog) => { const provider = new ProviderV2.Info({ ...ProviderV2.Info.empty(ProviderV2.ID.opencode), api: { type: "aisdk", package: "test-provider" }, request: { headers: {}, body: { apiKey: "configured" }, }, }) const model = new ModelV2.Info({ ...ModelV2.Info.empty(provider.id, ModelV2.ID.make("paid")), api: { id: ModelV2.ID.make("paid"), type: "aisdk", package: "test-provider" }, cost: cost(1), }) catalog.provider.update(provider.id, (draft) => { draft.request = provider.request }) catalog.model.update(provider.id, model.id, (draft) => { draft.cost = [...model.cost] }) }) yield* addPlugin() expect(required(yield* catalog.provider.get(ProviderV2.ID.opencode)).request.body.apiKey).toBe("configured") expect(required(yield* catalog.model.get(ProviderV2.ID.opencode, ModelV2.ID.make("paid"))).enabled).toBe(true) }), ), ) it.effect("ignores non-opencode providers and models", () => withEnv({ OPENCODE_API_KEY: undefined }, () => Effect.gen(function* () { const catalog = yield* Catalog.Service yield* catalog.transform((catalog) => { const provider = new ProviderV2.Info({ ...ProviderV2.Info.empty(ProviderV2.ID.openai), api: { type: "aisdk", package: "test-provider" }, }) const model = new ModelV2.Info({ ...ModelV2.Info.empty(provider.id, ModelV2.ID.make("paid")), api: { id: ModelV2.ID.make("paid"), type: "aisdk", package: "test-provider" }, cost: cost(1), }) catalog.provider.update(provider.id, () => {}) catalog.model.update(provider.id, model.id, (draft) => { draft.cost = [...model.cost] }) }) yield* addPlugin() expect(required(yield* catalog.provider.get(ProviderV2.ID.openai)).request.body.apiKey).toBeUndefined() expect(required(yield* catalog.model.get(ProviderV2.ID.openai, ModelV2.ID.make("paid"))).enabled).toBe(true) }), ), ) it.effect("prefers gpt-5-nano as the opencode small model", () => Effect.gen(function* () { const catalog = yield* Catalog.Service const providerID = ProviderV2.ID.opencode yield* catalog.transform((catalog) => { catalog.provider.update(providerID, () => {}) catalog.model.update(providerID, ModelV2.ID.make("cheap-mini"), (model) => { model.capabilities.input = ["text"] model.capabilities.output = ["text"] model.cost = [...cost(1, 1)] model.time.released = Date.now() }) catalog.model.update(providerID, ModelV2.ID.make("gpt-5-nano"), (model) => { model.capabilities.input = ["text"] model.capabilities.output = ["text"] model.cost = [...cost(10, 10)] model.time.released = Date.now() }) }) const selected = yield* catalog.model.small(providerID) expect(selected?.id).toBe(ModelV2.ID.make("gpt-5-nano")) }), ) })