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 account and service account 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: "OpenCode Console account", }, { type: "key", label: "API key (service account)" }, ]) }), ) 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")) const origin = new URL(request.url).origin return Response.json({ config: { enterprise: { url: origin }, provider: { remote: { name: "Remote", npm: "@ai-sdk/openai-compatible", api: `${origin}/v1`, env: ["REMOTE_API_KEY"], options: { apiKey: "{env:REMOTE_API_KEY}", headers: { "x-org-id": "org" }, custom: "value", }, models: { model: { name: "Remote Model", family: "remote", release_date: "2026-01-02", tool_call: true, modalities: { input: ["text", "image"], output: ["text"] }, options: { apiKey: "model-secret", temperature: 0.5 }, variants: { high: { apiKey: "variant-secret", temperature: 0.2 } }, cost: { input: 1, output: 2, cache_read: 0.1 }, limit: { context: 1000, output: 100 }, }, disabled: { name: "Disabled", status: "deprecated" }, }, }, }, }, }) }, }), } }), ({ authorization, server }) => Effect.gen(function* () { const credentials = yield* Credential.Service const catalog = yield* Catalog.Service yield* catalog.transform((draft) => { draft.provider.update(ProviderV2.ID.make("remote"), () => {}) draft.model.update(ProviderV2.ID.make("remote"), ModelV2.ID.make("stale"), () => {}) }) yield* credentials.create({ integrationID: Integration.ID.make("opencode"), value: new Credential.Key({ type: "key", key: "secret", metadata: { server: server.url.origin }, }), }) yield* addPlugin() const provider = required(yield* catalog.provider.get(ProviderV2.ID.make("remote"))) expect(provider).toMatchObject({ name: "Remote", integrationID: "opencode", api: { type: "aisdk", package: "@ai-sdk/openai-compatible", url: `${server.url.origin}/v1`, }, }) expect(provider.request).toEqual({ headers: { "x-org-id": "org" }, body: { custom: "value" } }) expect(yield* (yield* Integration.Service).get(Integration.ID.make("remote"))).toBeUndefined() const model = required(yield* catalog.model.get(ProviderV2.ID.make("remote"), ModelV2.ID.make("model"))) expect(model).toMatchObject({ name: "Remote Model", family: "remote", capabilities: { tools: true, input: ["text", "image"], output: ["text"] }, cost: [{ input: 1, output: 2, cache: { read: 0.1, write: 0 } }], limit: { context: 1000, output: 100 }, }) expect(model.request).toMatchObject({ body: { custom: "value" }, generation: { temperature: 0.5 } }) expect(model.request.body).toEqual({ custom: "value" }) expect(model.variants).toEqual([ { id: ModelV2.VariantID.make("high"), headers: {}, body: {}, generation: { temperature: 0.2 }, options: {}, }, ]) expect( required(yield* catalog.model.get(ProviderV2.ID.make("remote"), ModelV2.ID.make("disabled"))).enabled, ).toBe(false) expect(yield* catalog.model.get(ProviderV2.ID.make("remote"), ModelV2.ID.make("stale"))).toBeUndefined() 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")) }), ) })