188 lines
7.0 KiB
TypeScript
188 lines
7.0 KiB
TypeScript
import fs from "fs/promises"
|
|
import path from "path"
|
|
import { describe, expect } from "bun:test"
|
|
import { Deferred, Effect, Equal, Hash, Layer, Schema, Stream } from "effect"
|
|
import { Tool } from "@opencode-ai/core/public"
|
|
import { define } from "@opencode-ai/plugin/v2/effect"
|
|
import { AgentV2 } from "@opencode-ai/core/agent"
|
|
import { Catalog } from "@opencode-ai/core/catalog"
|
|
import { LocationServiceMap } from "@opencode-ai/core/location-layer"
|
|
import { Location } from "@opencode-ai/core/location"
|
|
import { PluginBoot } from "@opencode-ai/core/plugin/boot"
|
|
import { ProviderV2 } from "@opencode-ai/core/provider"
|
|
import { AbsolutePath } from "@opencode-ai/core/schema"
|
|
import { tmpdir } from "./fixture/tmpdir"
|
|
import { testEffect } from "./lib/effect"
|
|
import { toolDefinitions } from "./lib/tool"
|
|
import { FSUtil } from "../src/fs-util"
|
|
import { Credential } from "../src/credential"
|
|
import { Database } from "../src/database/database"
|
|
import { EventV2 } from "../src/event"
|
|
import { Global } from "../src/global"
|
|
import { ModelsDev } from "../src/models-dev"
|
|
import { Npm } from "../src/npm"
|
|
import { Project } from "../src/project"
|
|
import { Reference } from "../src/reference"
|
|
import { ToolRegistry } from "../src/tool/registry"
|
|
import { ApplicationTools } from "../src/tool/application-tools"
|
|
|
|
const applicationTools = ApplicationTools.layer
|
|
const it = testEffect(
|
|
Layer.merge(
|
|
Layer.mergeAll(applicationTools, Database.defaultLayer, EventV2.defaultLayer),
|
|
LocationServiceMap.layer.pipe(
|
|
Layer.provide(applicationTools),
|
|
Layer.provide(
|
|
Layer.mergeAll(
|
|
Project.defaultLayer,
|
|
EventV2.defaultLayer,
|
|
Credential.defaultLayer.pipe(Layer.fresh),
|
|
Npm.defaultLayer,
|
|
ModelsDev.defaultLayer,
|
|
FSUtil.defaultLayer,
|
|
Global.defaultLayer,
|
|
),
|
|
),
|
|
),
|
|
),
|
|
)
|
|
|
|
describe("LocationServiceMap", () => {
|
|
it.effect("compares equivalent location refs by value", () =>
|
|
Effect.sync(() => {
|
|
const directory = AbsolutePath.make("/project")
|
|
expect(Equal.equals(Location.Ref.make({ directory }), Location.Ref.make({ directory }))).toBe(true)
|
|
expect(Hash.hash(Location.Ref.make({ directory }))).toBe(
|
|
Hash.hash(Location.Ref.make({ directory, workspaceID: undefined })),
|
|
)
|
|
}),
|
|
)
|
|
|
|
it.live("isolates location state while sharing location policy with catalog", () =>
|
|
Effect.acquireRelease(
|
|
Effect.promise(() => Promise.all([tmpdir(), tmpdir()])),
|
|
(dirs) => Effect.promise(() => Promise.all(dirs.map((dir) => dir[Symbol.asyncDispose]())).then(() => undefined)),
|
|
).pipe(
|
|
Effect.flatMap(([blocked, allowed]) =>
|
|
Effect.gen(function* () {
|
|
yield* (yield* ApplicationTools.Service).register({
|
|
application_context: Tool.make({
|
|
description: "Read application context",
|
|
input: Schema.Struct({}),
|
|
output: Schema.Struct({ ok: Schema.Boolean }),
|
|
execute: () => Effect.succeed({ ok: true }),
|
|
}),
|
|
})
|
|
yield* Effect.promise(() =>
|
|
fs.writeFile(
|
|
path.join(blocked.path, "opencode.json"),
|
|
JSON.stringify({
|
|
experimental: { policies: [{ effect: "deny", action: "provider.use", resource: "test" }] },
|
|
}),
|
|
),
|
|
)
|
|
|
|
const update = (directory: string) =>
|
|
Effect.gen(function* () {
|
|
yield* PluginBoot.Service.use((boot) => boot.wait())
|
|
yield* Reference.Service
|
|
const catalog = yield* Catalog.Service
|
|
yield* catalog.transform((editor) => editor.provider.update(ProviderV2.ID.make("test"), () => {}))
|
|
return {
|
|
providers: yield* catalog.provider.all(),
|
|
tools: yield* toolDefinitions(yield* ToolRegistry.Service),
|
|
}
|
|
}).pipe(
|
|
Effect.scoped,
|
|
Effect.provide(LocationServiceMap.get(Location.Ref.make({ directory: AbsolutePath.make(directory) }))),
|
|
)
|
|
|
|
const blockedState = yield* update(blocked.path)
|
|
expect(blockedState.providers.some((provider) => provider.id === ProviderV2.ID.make("test"))).toBe(false)
|
|
expect(blockedState.tools.map((tool) => tool.name).sort()).toEqual([
|
|
"application_context",
|
|
"apply_patch",
|
|
"bash",
|
|
"edit",
|
|
"glob",
|
|
"grep",
|
|
"question",
|
|
"read",
|
|
"skill",
|
|
"todowrite",
|
|
"webfetch",
|
|
"websearch",
|
|
"write",
|
|
])
|
|
const allowedState = yield* update(allowed.path)
|
|
expect(allowedState.providers.some((provider) => provider.id === ProviderV2.ID.make("test"))).toBe(true)
|
|
expect(allowedState.tools.map((tool) => tool.name).sort()).toEqual([
|
|
"application_context",
|
|
"apply_patch",
|
|
"bash",
|
|
"edit",
|
|
"glob",
|
|
"grep",
|
|
"question",
|
|
"read",
|
|
"skill",
|
|
"todowrite",
|
|
"webfetch",
|
|
"websearch",
|
|
"write",
|
|
])
|
|
}),
|
|
),
|
|
),
|
|
)
|
|
|
|
it.live("installs public plugins into a location", () =>
|
|
Effect.acquireRelease(
|
|
Effect.promise(() => tmpdir()),
|
|
(dir) => Effect.promise(() => dir[Symbol.asyncDispose]()),
|
|
).pipe(
|
|
Effect.flatMap((dir) =>
|
|
Effect.gen(function* () {
|
|
const boot = yield* PluginBoot.Service
|
|
const catalogUpdated = yield* Deferred.make<void>()
|
|
const seen: string[] = []
|
|
yield* boot.add(
|
|
define({
|
|
id: "reviewer",
|
|
effect: (ctx) =>
|
|
Effect.gen(function* () {
|
|
yield* ctx.event.subscribe("catalog.updated").pipe(
|
|
Stream.runForEach(() => Deferred.succeed(catalogUpdated, undefined).pipe(Effect.asVoid)),
|
|
Effect.forkScoped({ startImmediately: true }),
|
|
)
|
|
yield* ctx.agent.transform((agent) => {
|
|
agent.update("reviewer", (item) => {
|
|
item.description = "Reviews code"
|
|
item.mode = "subagent"
|
|
})
|
|
})
|
|
seen.push((yield* ctx.agent.get("reviewer"))?.description ?? "")
|
|
yield* ctx.catalog.transform((catalog) => {
|
|
catalog.provider.update("public", (provider) => {
|
|
provider.name = "Public provider"
|
|
})
|
|
})
|
|
}),
|
|
}),
|
|
)
|
|
|
|
yield* Deferred.await(catalogUpdated)
|
|
expect(seen).toEqual(["Reviews code"])
|
|
expect(yield* (yield* AgentV2.Service).get(AgentV2.ID.make("reviewer"))).toMatchObject({
|
|
description: "Reviews code",
|
|
mode: "subagent",
|
|
})
|
|
}).pipe(
|
|
Effect.scoped,
|
|
Effect.provide(LocationServiceMap.get(Location.Ref.make({ directory: AbsolutePath.make(dir.path) }))),
|
|
),
|
|
),
|
|
),
|
|
)
|
|
})
|