mirror of
https://github.com/anomalyco/opencode.git
synced 2026-04-25 07:15:19 +00:00
Delete get() and waitForDependencies() facade functions from config/tui.ts. Add TuiConfig.defaultLayer to AppLayer so the service is available via AppRuntime. Migrate all callers (attach.ts, thread.ts, plugin/runtime.ts) to AppRuntime.runPromise(TuiConfig.Service.use(...)). Migrate tui.test.ts (~28 calls) to the same pattern. Rework test fixture: replace spyOn(TuiConfig, 'get') with mockTuiService helper that mocks TuiConfig.Service.use at the Effect level. Update all 9 test files that spied on the old facades.
156 lines
4.3 KiB
TypeScript
156 lines
4.3 KiB
TypeScript
import { expect, spyOn, test } from "bun:test"
|
|
import fs from "fs/promises"
|
|
import path from "path"
|
|
import { pathToFileURL } from "url"
|
|
import { tmpdir } from "../../fixture/fixture"
|
|
import { createTuiPluginApi } from "../../fixture/tui-plugin"
|
|
import { mockTuiService } from "../../fixture/tui-runtime"
|
|
|
|
const { TuiPluginRuntime } = await import("../../../src/cli/cmd/tui/plugin/runtime")
|
|
|
|
test("toggles plugin runtime state by exported id", async () => {
|
|
await using tmp = await tmpdir({
|
|
init: async (dir) => {
|
|
const file = path.join(dir, "toggle-plugin.ts")
|
|
const spec = pathToFileURL(file).href
|
|
const marker = path.join(dir, "toggle.txt")
|
|
|
|
await Bun.write(
|
|
file,
|
|
`export default {
|
|
id: "demo.toggle",
|
|
tui: async (api, options) => {
|
|
const text = await Bun.file(options.marker).text().catch(() => "")
|
|
await Bun.write(options.marker, text + "start\\n")
|
|
api.lifecycle.onDispose(async () => {
|
|
const next = await Bun.file(options.marker).text().catch(() => "")
|
|
await Bun.write(options.marker, next + "stop\\n")
|
|
})
|
|
},
|
|
}
|
|
`,
|
|
)
|
|
|
|
return {
|
|
spec,
|
|
marker,
|
|
}
|
|
},
|
|
})
|
|
|
|
process.env.OPENCODE_PLUGIN_META_FILE = path.join(tmp.path, "plugin-meta.json")
|
|
const restore = mockTuiService({
|
|
plugin: [[tmp.extra.spec, { marker: tmp.extra.marker }]],
|
|
plugin_enabled: {
|
|
"demo.toggle": false,
|
|
},
|
|
plugin_origins: [
|
|
{
|
|
spec: [tmp.extra.spec, { marker: tmp.extra.marker }],
|
|
scope: "local",
|
|
source: path.join(tmp.path, "tui.json"),
|
|
},
|
|
],
|
|
})
|
|
const cwd = spyOn(process, "cwd").mockImplementation(() => tmp.path)
|
|
const api = createTuiPluginApi()
|
|
|
|
try {
|
|
await TuiPluginRuntime.init(api)
|
|
|
|
await expect(fs.readFile(tmp.extra.marker, "utf8")).rejects.toThrow()
|
|
expect(TuiPluginRuntime.list().find((item) => item.id === "demo.toggle")).toEqual({
|
|
id: "demo.toggle",
|
|
source: "file",
|
|
spec: tmp.extra.spec,
|
|
target: tmp.extra.spec,
|
|
enabled: false,
|
|
active: false,
|
|
})
|
|
|
|
await expect(TuiPluginRuntime.activatePlugin("demo.toggle")).resolves.toBe(true)
|
|
await expect(fs.readFile(tmp.extra.marker, "utf8")).resolves.toBe("start\n")
|
|
expect(api.kv.get("plugin_enabled", {})).toEqual({
|
|
"demo.toggle": true,
|
|
})
|
|
|
|
await expect(TuiPluginRuntime.deactivatePlugin("demo.toggle")).resolves.toBe(true)
|
|
await expect(fs.readFile(tmp.extra.marker, "utf8")).resolves.toBe("start\nstop\n")
|
|
expect(api.kv.get("plugin_enabled", {})).toEqual({
|
|
"demo.toggle": false,
|
|
})
|
|
|
|
await expect(TuiPluginRuntime.activatePlugin("missing.id")).resolves.toBe(false)
|
|
} finally {
|
|
await TuiPluginRuntime.dispose()
|
|
cwd.mockRestore()
|
|
restore()
|
|
delete process.env.OPENCODE_PLUGIN_META_FILE
|
|
}
|
|
})
|
|
|
|
test("kv plugin_enabled overrides tui config on startup", async () => {
|
|
await using tmp = await tmpdir({
|
|
init: async (dir) => {
|
|
const file = path.join(dir, "startup-plugin.ts")
|
|
const spec = pathToFileURL(file).href
|
|
const marker = path.join(dir, "startup.txt")
|
|
|
|
await Bun.write(
|
|
file,
|
|
`export default {
|
|
id: "demo.startup",
|
|
tui: async (_api, options) => {
|
|
await Bun.write(options.marker, "on")
|
|
},
|
|
}
|
|
`,
|
|
)
|
|
|
|
return {
|
|
spec,
|
|
marker,
|
|
}
|
|
},
|
|
})
|
|
|
|
process.env.OPENCODE_PLUGIN_META_FILE = path.join(tmp.path, "plugin-meta.json")
|
|
const restore = mockTuiService({
|
|
plugin: [[tmp.extra.spec, { marker: tmp.extra.marker }]],
|
|
plugin_enabled: {
|
|
"demo.startup": false,
|
|
},
|
|
plugin_origins: [
|
|
{
|
|
spec: [tmp.extra.spec, { marker: tmp.extra.marker }],
|
|
scope: "local",
|
|
source: path.join(tmp.path, "tui.json"),
|
|
},
|
|
],
|
|
})
|
|
const cwd = spyOn(process, "cwd").mockImplementation(() => tmp.path)
|
|
const api = createTuiPluginApi()
|
|
api.kv.set("plugin_enabled", {
|
|
"demo.startup": true,
|
|
})
|
|
|
|
try {
|
|
await TuiPluginRuntime.init(api)
|
|
|
|
await expect(fs.readFile(tmp.extra.marker, "utf8")).resolves.toBe("on")
|
|
expect(TuiPluginRuntime.list().find((item) => item.id === "demo.startup")).toEqual({
|
|
id: "demo.startup",
|
|
source: "file",
|
|
spec: tmp.extra.spec,
|
|
target: tmp.extra.spec,
|
|
enabled: true,
|
|
active: true,
|
|
})
|
|
} finally {
|
|
await TuiPluginRuntime.dispose()
|
|
cwd.mockRestore()
|
|
restore()
|
|
delete process.env.OPENCODE_PLUGIN_META_FILE
|
|
}
|
|
})
|