mirror of
https://github.com/anomalyco/opencode.git
synced 2026-05-19 16:40:48 +00:00
269 lines
10 KiB
TypeScript
269 lines
10 KiB
TypeScript
export * as Catalog from "./catalog"
|
|
|
|
import { Context, Effect, HashMap, Layer, Option, Order, pipe, Schema, Array } from "effect"
|
|
import { produce, type Draft } from "immer"
|
|
import { ModelV2 } from "./model"
|
|
import { PluginV2 } from "./plugin"
|
|
import { ProviderV2 } from "./provider"
|
|
import { Location } from "./location"
|
|
import { EventV2 } from "./event"
|
|
|
|
type ProviderRecord = {
|
|
provider: ProviderV2.Info
|
|
models: HashMap.HashMap<ModelV2.ID, ModelV2.Info>
|
|
}
|
|
|
|
export class ProviderNotFoundError extends Schema.TaggedErrorClass<ProviderNotFoundError>()(
|
|
"CatalogV2.ProviderNotFound",
|
|
{
|
|
providerID: ProviderV2.ID,
|
|
},
|
|
) {}
|
|
|
|
export class ModelNotFoundError extends Schema.TaggedErrorClass<ModelNotFoundError>()("CatalogV2.ModelNotFound", {
|
|
providerID: ProviderV2.ID,
|
|
modelID: ModelV2.ID,
|
|
}) {}
|
|
|
|
export const Event = {
|
|
ModelUpdated: EventV2.define({
|
|
type: "catalog.model.updated",
|
|
schema: {
|
|
model: ModelV2.Info,
|
|
},
|
|
}),
|
|
}
|
|
|
|
export interface Interface {
|
|
readonly provider: {
|
|
readonly get: (providerID: ProviderV2.ID) => Effect.Effect<ProviderV2.Info, ProviderNotFoundError>
|
|
readonly update: (providerID: ProviderV2.ID, fn: (provider: Draft<ProviderV2.Info>) => void) => Effect.Effect<void>
|
|
readonly all: () => Effect.Effect<ProviderV2.Info[]>
|
|
readonly available: () => Effect.Effect<ProviderV2.Info[]>
|
|
}
|
|
readonly model: {
|
|
readonly get: (
|
|
providerID: ProviderV2.ID,
|
|
modelID: ModelV2.ID,
|
|
) => Effect.Effect<ModelV2.Info, ProviderNotFoundError | ModelNotFoundError>
|
|
readonly update: (
|
|
providerID: ProviderV2.ID,
|
|
modelID: ModelV2.ID,
|
|
fn: (model: Draft<ModelV2.Info>) => void,
|
|
) => Effect.Effect<void, ProviderNotFoundError>
|
|
readonly all: () => Effect.Effect<ModelV2.Info[]>
|
|
readonly available: () => Effect.Effect<ModelV2.Info[]>
|
|
readonly default: () => Effect.Effect<Option.Option<ModelV2.Info>>
|
|
readonly setDefault: (
|
|
providerID: ProviderV2.ID,
|
|
modelID: ModelV2.ID,
|
|
) => Effect.Effect<void, ProviderNotFoundError | ModelNotFoundError>
|
|
readonly small: (providerID: ProviderV2.ID) => Effect.Effect<Option.Option<ModelV2.Info>>
|
|
}
|
|
}
|
|
|
|
export class Service extends Context.Service<Service, Interface>()("@opencode/v2/Catalog") {}
|
|
|
|
export const layer = Layer.effect(
|
|
Service,
|
|
Effect.gen(function* () {
|
|
yield* Location.Service
|
|
let records = HashMap.empty<ProviderV2.ID, ProviderRecord>()
|
|
let defaultModel: { providerID: ProviderV2.ID; modelID: ModelV2.ID } | undefined
|
|
const plugin = yield* PluginV2.Service
|
|
const events = yield* EventV2.Service
|
|
|
|
const resolve = (model: ModelV2.Info) => {
|
|
const provider = Option.getOrThrow(HashMap.get(records, model.providerID)).provider
|
|
const endpoint =
|
|
model.endpoint.type === "unknown"
|
|
? provider.endpoint
|
|
: model.endpoint.type === "aisdk" && provider.endpoint.type === "aisdk" && !model.endpoint.url
|
|
? { ...model.endpoint, url: provider.endpoint.url }
|
|
: model.endpoint
|
|
const options = {
|
|
headers: {
|
|
...provider.options.headers,
|
|
...model.options.headers,
|
|
},
|
|
body: {
|
|
...provider.options.body,
|
|
...model.options.body,
|
|
},
|
|
aisdk: {
|
|
provider: {
|
|
...provider.options.aisdk.provider,
|
|
...model.options.aisdk.provider,
|
|
},
|
|
request: model.options.aisdk.request,
|
|
},
|
|
variant: model.options.variant,
|
|
}
|
|
return new ModelV2.Info({
|
|
...model,
|
|
endpoint,
|
|
options,
|
|
})
|
|
}
|
|
|
|
function* getRecord(providerID: ProviderV2.ID) {
|
|
const match = HashMap.get(records, providerID)
|
|
if (!match.valueOrUndefined) return yield* new ProviderNotFoundError({ providerID })
|
|
return match.value
|
|
}
|
|
|
|
const result: Interface = {
|
|
provider: {
|
|
get: Effect.fn("CatalogV2.provider.get")(function* (providerID) {
|
|
const record = yield* getRecord(providerID)
|
|
return record.provider
|
|
}),
|
|
|
|
update: Effect.fnUntraced(function* (providerID, fn) {
|
|
const current = Option.getOrUndefined(HashMap.get(records, providerID))
|
|
const provider = produce(current?.provider ?? ProviderV2.Info.empty(providerID), (draft) => {
|
|
fn(draft)
|
|
if (draft.endpoint.type === "aisdk" && typeof draft.options.aisdk.provider.baseURL === "string") {
|
|
draft.endpoint.url = draft.options.aisdk.provider.baseURL
|
|
delete draft.options.aisdk.provider.baseURL
|
|
}
|
|
})
|
|
const updated = yield* plugin.trigger("provider.update", {}, { provider, cancel: false })
|
|
records = HashMap.set(records, providerID, {
|
|
provider: updated.provider,
|
|
models: current?.models ?? HashMap.empty<ModelV2.ID, ModelV2.Info>(),
|
|
})
|
|
}),
|
|
|
|
all: Effect.fn("CatalogV2.provider.all")(function* () {
|
|
return globalThis.Array.from(HashMap.values(records)).map((record) => record.provider)
|
|
}),
|
|
|
|
available: Effect.fn("CatalogV2.provider.available")(function* () {
|
|
return globalThis.Array.from(HashMap.values(records))
|
|
.map((record) => record.provider)
|
|
.filter((provider) => provider.enabled)
|
|
}),
|
|
},
|
|
|
|
model: {
|
|
get: Effect.fn("CatalogV2.model.get")(function* (providerID, modelID) {
|
|
const record = yield* getRecord(providerID)
|
|
const model = Option.getOrUndefined(HashMap.get(record.models, modelID))
|
|
if (!model) return yield* new ModelNotFoundError({ providerID, modelID })
|
|
return resolve(model)
|
|
}),
|
|
|
|
update: Effect.fnUntraced(function* (providerID, modelID, fn) {
|
|
const record = yield* getRecord(providerID)
|
|
const model = produce(
|
|
HashMap.get(record.models, modelID).pipe(Option.getOrElse(() => ModelV2.Info.empty(providerID, modelID))),
|
|
(draft) => {
|
|
fn(draft)
|
|
if (draft.endpoint.type === "aisdk" && typeof draft.options.aisdk.provider.baseURL === "string") {
|
|
draft.endpoint.url = draft.options.aisdk.provider.baseURL
|
|
delete draft.options.aisdk.provider.baseURL
|
|
}
|
|
},
|
|
)
|
|
const updated = yield* plugin.trigger("model.update", {}, { model, cancel: false })
|
|
if (updated.cancel) return
|
|
const next = new ModelV2.Info({ ...updated.model, id: modelID, providerID })
|
|
records = HashMap.set(records, providerID, {
|
|
provider: record.provider,
|
|
models: HashMap.set(record.models, modelID, next),
|
|
})
|
|
yield* events.publish(Event.ModelUpdated, { model: resolve(next) })
|
|
return
|
|
}),
|
|
|
|
all: Effect.fn("CatalogV2.model.all")(function* () {
|
|
return pipe(
|
|
records,
|
|
HashMap.toValues,
|
|
Array.flatMap((record) => HashMap.toValues(record.models)),
|
|
Array.map(resolve),
|
|
Array.sortWith((item) => item.time.released.epochMilliseconds, Order.flip(Order.Number)),
|
|
)
|
|
}),
|
|
|
|
available: Effect.fn("CatalogV2.model.available")(function* () {
|
|
return (yield* result.model.all()).filter((model) => {
|
|
const record = Option.getOrUndefined(HashMap.get(records, model.providerID))
|
|
return record?.provider.enabled !== false && model.enabled
|
|
})
|
|
}),
|
|
|
|
default: Effect.fn("CatalogV2.model.default")(function* () {
|
|
if (defaultModel) {
|
|
const model = yield* result.model.get(defaultModel.providerID, defaultModel.modelID).pipe(Effect.option)
|
|
if (Option.isSome(model) && model.value.enabled) return model
|
|
}
|
|
|
|
return pipe(
|
|
yield* result.model.available(),
|
|
Array.sortWith((item) => item.time.released.epochMilliseconds, Order.flip(Order.Number)),
|
|
Array.head,
|
|
)
|
|
}),
|
|
|
|
setDefault: Effect.fn("CatalogV2.model.setDefault")(function* (providerID, modelID) {
|
|
yield* result.model.get(providerID, modelID)
|
|
defaultModel = { providerID, modelID }
|
|
}),
|
|
|
|
small: Effect.fn("CatalogV2.model.small")(function* (providerID) {
|
|
const record = Option.getOrUndefined(HashMap.get(records, providerID))
|
|
if (!record) return Option.none<ModelV2.Info>()
|
|
|
|
if (providerID === ProviderV2.ID.opencode) {
|
|
const gpt5Nano = Option.getOrUndefined(HashMap.get(record.models, ModelV2.ID.make("gpt-5-nano")))
|
|
if (gpt5Nano?.enabled && gpt5Nano.status === "active") return Option.some(resolve(gpt5Nano))
|
|
}
|
|
|
|
const candidates = pipe(
|
|
HashMap.toValues(record.models),
|
|
Array.filter(
|
|
(model) =>
|
|
model.providerID === providerID &&
|
|
model.enabled &&
|
|
model.status === "active" &&
|
|
model.capabilities.input.some((item) => item.startsWith("text")) &&
|
|
model.capabilities.output.some((item) => item.startsWith("text")),
|
|
),
|
|
Array.map((model) => ({
|
|
model,
|
|
cost: model.cost[0] ? model.cost[0].input + model.cost[0].output : 999,
|
|
age: (Date.now() - model.time.released.epochMilliseconds) / (1000 * 60 * 60 * 24 * 30),
|
|
small: SMALL_MODEL_RE.test(`${model.id} ${model.family ?? ""} ${model.name}`.toLowerCase()),
|
|
})),
|
|
Array.filter((item) => item.cost > 0 && item.age <= 18),
|
|
)
|
|
|
|
const pick = (items: typeof candidates) => {
|
|
const maxCost = Math.max(...items.map((item) => item.cost), 0.01)
|
|
const maxAge = Math.max(...items.map((item) => item.age), 0.01)
|
|
return pipe(
|
|
items,
|
|
Array.sortWith((item) => (item.cost / maxCost) * 0.8 + (item.age / maxAge) * 0.2, Order.Number),
|
|
Array.map((item) => resolve(item.model)),
|
|
Array.head,
|
|
)
|
|
}
|
|
|
|
return pipe(
|
|
candidates,
|
|
Array.filter((item) => item.small),
|
|
(items) => (items.length > 0 ? pick(items) : pick(candidates)),
|
|
)
|
|
}),
|
|
},
|
|
}
|
|
|
|
return Service.of(result)
|
|
}),
|
|
)
|
|
|
|
const SMALL_MODEL_RE = /\b(nano|flash|lite|mini|haiku|small|fast)\b/
|
|
|
|
export const defaultLayer = layer.pipe(Layer.provideMerge(EventV2.defaultLayer), Layer.provide(PluginV2.defaultLayer))
|