| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337 |
- import { Config } from "../config/config"
- import z from "zod"
- import { Provider } from "../provider/provider"
- import { generateObject, streamObject, type ModelMessage } from "ai"
- import { SystemPrompt } from "../session/system"
- import { Instance } from "../project/instance"
- import { Truncate } from "../tool/truncation"
- import { Auth } from "../auth"
- import { ProviderTransform } from "../provider/transform"
- import PROMPT_GENERATE from "./generate.txt"
- import PROMPT_COMPACTION from "./prompt/compaction.txt"
- import PROMPT_EXPLORE from "./prompt/explore.txt"
- import PROMPT_SUMMARY from "./prompt/summary.txt"
- import PROMPT_TITLE from "./prompt/title.txt"
- import { PermissionNext } from "@/permission/next"
- import { mergeDeep, pipe, sortBy, values } from "remeda"
- import { Global } from "@/global"
- import path from "path"
- import { Plugin } from "@/plugin"
- export namespace Agent {
- export const Info = z
- .object({
- name: z.string(),
- description: z.string().optional(),
- mode: z.enum(["subagent", "primary", "all"]),
- native: z.boolean().optional(),
- hidden: z.boolean().optional(),
- topP: z.number().optional(),
- temperature: z.number().optional(),
- color: z.string().optional(),
- permission: PermissionNext.Ruleset,
- model: z
- .object({
- modelID: z.string(),
- providerID: z.string(),
- })
- .optional(),
- variant: z.string().optional(),
- prompt: z.string().optional(),
- options: z.record(z.string(), z.any()),
- steps: z.number().int().positive().optional(),
- })
- .meta({
- ref: "Agent",
- })
- export type Info = z.infer<typeof Info>
- const state = Instance.state(async () => {
- const cfg = await Config.get()
- const defaults = PermissionNext.fromConfig({
- "*": "allow",
- doom_loop: "ask",
- external_directory: {
- "*": "ask",
- [Truncate.DIR]: "allow",
- [Truncate.GLOB]: "allow",
- },
- question: "deny",
- plan_enter: "deny",
- plan_exit: "deny",
- // mirrors github.com/github/gitignore Node.gitignore pattern for .env files
- read: {
- "*": "allow",
- "*.env": "ask",
- "*.env.*": "ask",
- "*.env.example": "allow",
- },
- })
- const user = PermissionNext.fromConfig(cfg.permission ?? {})
- const result: Record<string, Info> = {
- build: {
- name: "build",
- description: "The default agent. Executes tools based on configured permissions.",
- options: {},
- permission: PermissionNext.merge(
- defaults,
- PermissionNext.fromConfig({
- question: "allow",
- plan_enter: "allow",
- }),
- user,
- ),
- mode: "primary",
- native: true,
- },
- plan: {
- name: "plan",
- description: "Plan mode. Disallows all edit tools.",
- options: {},
- permission: PermissionNext.merge(
- defaults,
- PermissionNext.fromConfig({
- question: "allow",
- plan_exit: "allow",
- external_directory: {
- [path.join(Global.Path.data, "plans", "*")]: "allow",
- },
- edit: {
- "*": "deny",
- [path.join(".opencode", "plans", "*.md")]: "allow",
- [path.relative(Instance.worktree, path.join(Global.Path.data, path.join("plans", "*.md")))]: "allow",
- },
- }),
- user,
- ),
- mode: "primary",
- native: true,
- },
- general: {
- name: "general",
- description: `General-purpose agent for researching complex questions and executing multi-step tasks. Use this agent to execute multiple units of work in parallel.`,
- permission: PermissionNext.merge(
- defaults,
- PermissionNext.fromConfig({
- todoread: "deny",
- todowrite: "deny",
- }),
- user,
- ),
- options: {},
- mode: "subagent",
- native: true,
- },
- explore: {
- name: "explore",
- permission: PermissionNext.merge(
- defaults,
- PermissionNext.fromConfig({
- "*": "deny",
- grep: "allow",
- glob: "allow",
- list: "allow",
- bash: "allow",
- webfetch: "allow",
- websearch: "allow",
- codesearch: "allow",
- read: "allow",
- external_directory: {
- [Truncate.DIR]: "allow",
- [Truncate.GLOB]: "allow",
- },
- }),
- user,
- ),
- description: `Fast agent specialized for exploring codebases. Use this when you need to quickly find files by patterns (eg. "src/components/**/*.tsx"), search code for keywords (eg. "API endpoints"), or answer questions about the codebase (eg. "how do API endpoints work?"). When calling this agent, specify the desired thoroughness level: "quick" for basic searches, "medium" for moderate exploration, or "very thorough" for comprehensive analysis across multiple locations and naming conventions.`,
- prompt: PROMPT_EXPLORE,
- options: {},
- mode: "subagent",
- native: true,
- },
- compaction: {
- name: "compaction",
- mode: "primary",
- native: true,
- hidden: true,
- prompt: PROMPT_COMPACTION,
- permission: PermissionNext.merge(
- defaults,
- PermissionNext.fromConfig({
- "*": "deny",
- }),
- user,
- ),
- options: {},
- },
- title: {
- name: "title",
- mode: "primary",
- options: {},
- native: true,
- hidden: true,
- temperature: 0.5,
- permission: PermissionNext.merge(
- defaults,
- PermissionNext.fromConfig({
- "*": "deny",
- }),
- user,
- ),
- prompt: PROMPT_TITLE,
- },
- summary: {
- name: "summary",
- mode: "primary",
- options: {},
- native: true,
- hidden: true,
- permission: PermissionNext.merge(
- defaults,
- PermissionNext.fromConfig({
- "*": "deny",
- }),
- user,
- ),
- prompt: PROMPT_SUMMARY,
- },
- }
- for (const [key, value] of Object.entries(cfg.agent ?? {})) {
- if (value.disable) {
- delete result[key]
- continue
- }
- let item = result[key]
- if (!item)
- item = result[key] = {
- name: key,
- mode: "all",
- permission: PermissionNext.merge(defaults, user),
- options: {},
- native: false,
- }
- if (value.model) item.model = Provider.parseModel(value.model)
- item.variant = value.variant ?? item.variant
- item.prompt = value.prompt ?? item.prompt
- item.description = value.description ?? item.description
- item.temperature = value.temperature ?? item.temperature
- item.topP = value.top_p ?? item.topP
- item.mode = value.mode ?? item.mode
- item.color = value.color ?? item.color
- item.hidden = value.hidden ?? item.hidden
- item.name = value.name ?? item.name
- item.steps = value.steps ?? item.steps
- item.options = mergeDeep(item.options, value.options ?? {})
- item.permission = PermissionNext.merge(item.permission, PermissionNext.fromConfig(value.permission ?? {}))
- }
- // Ensure Truncate.DIR is allowed unless explicitly configured
- for (const name in result) {
- const agent = result[name]
- const explicit = agent.permission.some((r) => {
- if (r.permission !== "external_directory") return false
- if (r.action !== "deny") return false
- return r.pattern === Truncate.DIR || r.pattern === Truncate.GLOB
- })
- if (explicit) continue
- result[name].permission = PermissionNext.merge(
- result[name].permission,
- PermissionNext.fromConfig({ external_directory: { [Truncate.DIR]: "allow", [Truncate.GLOB]: "allow" } }),
- )
- }
- return result
- })
- export async function get(agent: string) {
- return state().then((x) => x[agent])
- }
- export async function list() {
- const cfg = await Config.get()
- return pipe(
- await state(),
- values(),
- sortBy([(x) => (cfg.default_agent ? x.name === cfg.default_agent : x.name === "build"), "desc"]),
- )
- }
- export async function defaultAgent() {
- const cfg = await Config.get()
- const agents = await state()
- if (cfg.default_agent) {
- const agent = agents[cfg.default_agent]
- if (!agent) throw new Error(`default agent "${cfg.default_agent}" not found`)
- if (agent.mode === "subagent") throw new Error(`default agent "${cfg.default_agent}" is a subagent`)
- if (agent.hidden === true) throw new Error(`default agent "${cfg.default_agent}" is hidden`)
- return agent.name
- }
- const primaryVisible = Object.values(agents).find((a) => a.mode !== "subagent" && a.hidden !== true)
- if (!primaryVisible) throw new Error("no primary visible agent found")
- return primaryVisible.name
- }
- export async function generate(input: { description: string; model?: { providerID: string; modelID: string } }) {
- const cfg = await Config.get()
- const defaultModel = input.model ?? (await Provider.defaultModel())
- const model = await Provider.getModel(defaultModel.providerID, defaultModel.modelID)
- const language = await Provider.getLanguage(model)
- const system = [PROMPT_GENERATE]
- await Plugin.trigger("experimental.chat.system.transform", { model }, { system })
- const existing = await list()
- const params = {
- experimental_telemetry: {
- isEnabled: cfg.experimental?.openTelemetry,
- metadata: {
- userId: cfg.username ?? "unknown",
- },
- },
- temperature: 0.3,
- messages: [
- ...system.map(
- (item): ModelMessage => ({
- role: "system",
- content: item,
- }),
- ),
- {
- role: "user",
- content: `Create an agent configuration based on this request: \"${input.description}\".\n\nIMPORTANT: The following identifiers already exist and must NOT be used: ${existing.map((i) => i.name).join(", ")}\n Return ONLY the JSON object, no other text, do not wrap in backticks`,
- },
- ],
- model: language,
- schema: z.object({
- identifier: z.string(),
- whenToUse: z.string(),
- systemPrompt: z.string(),
- }),
- } satisfies Parameters<typeof generateObject>[0]
- if (defaultModel.providerID === "openai" && (await Auth.get(defaultModel.providerID))?.type === "oauth") {
- const result = streamObject({
- ...params,
- providerOptions: ProviderTransform.providerOptions(model, {
- instructions: SystemPrompt.instructions(),
- store: false,
- }),
- onError: () => {},
- })
- for await (const part of result.fullStream) {
- if (part.type === "error") throw part.error
- }
- return result.object
- }
- const result = await generateObject(params)
- return result.object
- }
- }
|