mirror of
https://github.com/anomalyco/opencode.git
synced 2026-05-17 12:42:17 +00:00
146 lines
3.7 KiB
TypeScript
146 lines
3.7 KiB
TypeScript
export * as PluginV2 from "./plugin"
|
|
|
|
import { createDraft, finishDraft, type Draft } from "immer"
|
|
import type { LanguageModelV3 } from "@ai-sdk/provider"
|
|
import { type ProviderV2 } from "./provider"
|
|
import { Context, Effect, Layer, Schema } from "effect"
|
|
import type { ModelV2 } from "./model"
|
|
|
|
export const ID = Schema.String.pipe(Schema.brand("Plugin.ID"))
|
|
export type ID = typeof ID.Type
|
|
|
|
type HookSpec = {
|
|
"provider.update": {
|
|
input: {}
|
|
output: {
|
|
provider: ProviderV2.Info
|
|
cancel: boolean
|
|
}
|
|
}
|
|
"model.update": {
|
|
input: {}
|
|
output: {
|
|
model: ModelV2.Info
|
|
cancel: boolean
|
|
}
|
|
}
|
|
"aisdk.language": {
|
|
input: {
|
|
model: ModelV2.Info
|
|
sdk: any
|
|
options: Record<string, any>
|
|
}
|
|
output: {
|
|
language?: LanguageModelV3
|
|
}
|
|
}
|
|
"aisdk.sdk": {
|
|
input: {
|
|
model: ModelV2.Info
|
|
package: string
|
|
options: Record<string, any>
|
|
}
|
|
output: {
|
|
sdk?: any
|
|
}
|
|
}
|
|
}
|
|
|
|
export type Hooks = {
|
|
[Name in keyof HookSpec]: Readonly<HookSpec[Name]["input"]> & {
|
|
-readonly [Field in keyof HookSpec[Name]["output"]]: HookSpec[Name]["output"][Field] extends object
|
|
? Draft<HookSpec[Name]["output"][Field]>
|
|
: HookSpec[Name]["output"][Field]
|
|
}
|
|
}
|
|
|
|
export type HookFunctions = {
|
|
[key in keyof Hooks]?: (input: Hooks[key]) => Effect.Effect<void>
|
|
}
|
|
|
|
export type HookInput<Name extends keyof Hooks> = HookSpec[Name]["input"]
|
|
export type HookOutput<Name extends keyof Hooks> = HookSpec[Name]["output"]
|
|
|
|
export type Effect = Effect.Effect<HookFunctions | void, never, never>
|
|
|
|
export function define<R>(input: { id: ID; effect: Effect.Effect<HookFunctions | void, never, R> }) {
|
|
return input
|
|
}
|
|
|
|
export interface Interface {
|
|
readonly add: (input: { id: ID; effect: Effect }) => Effect.Effect<void>
|
|
readonly remove: (id: ID) => Effect.Effect<void>
|
|
readonly trigger: <Name extends keyof Hooks>(
|
|
name: Name,
|
|
input: HookInput<Name>,
|
|
output: HookOutput<Name>,
|
|
) => Effect.Effect<HookInput<Name> & HookOutput<Name>>
|
|
}
|
|
|
|
export class Service extends Context.Service<Service, Interface>()("@opencode/v2/Plugin") {}
|
|
|
|
export const layer = Layer.effect(
|
|
Service,
|
|
Effect.gen(function* () {
|
|
let hooks: {
|
|
id: ID
|
|
hooks: HookFunctions
|
|
}[] = []
|
|
|
|
const svc = Service.of({
|
|
add: Effect.fn("Plugin.add")(function* (input) {
|
|
const result = yield* input.effect
|
|
if (!result) return
|
|
hooks = [
|
|
...hooks.filter((item) => item.id !== input.id),
|
|
{
|
|
id: input.id,
|
|
hooks: result,
|
|
},
|
|
]
|
|
}),
|
|
trigger: Effect.fn("Plugin.trigger")(function* (name, input, output) {
|
|
const draftEntries = new Map<string, ReturnType<typeof createDraft>>()
|
|
const event = {
|
|
...input,
|
|
...output,
|
|
} as Record<string, unknown>
|
|
|
|
for (const [field, value] of Object.entries(output)) {
|
|
if (value && typeof value === "object") {
|
|
draftEntries.set(field, createDraft(value))
|
|
event[field] = draftEntries.get(field)
|
|
}
|
|
}
|
|
|
|
for (const item of hooks) {
|
|
const match = item.hooks[name]
|
|
if (!match) continue
|
|
yield* match(event as any).pipe(
|
|
Effect.withSpan(`Plugin.hook.${name}`, {
|
|
attributes: {
|
|
plugin: item.id,
|
|
hook: name,
|
|
},
|
|
}),
|
|
)
|
|
}
|
|
|
|
for (const [field, draft] of draftEntries) {
|
|
event[field] = finishDraft(draft)
|
|
}
|
|
|
|
return event as any
|
|
}),
|
|
remove: Effect.fn("Plugin.remove")(function* (id) {
|
|
hooks = hooks.filter((item) => item.id !== id)
|
|
}),
|
|
})
|
|
return svc
|
|
}),
|
|
)
|
|
|
|
export const defaultLayer = layer
|
|
|
|
// opencode
|
|
// sdcok
|