mirror of
https://gitea.toothfairyai.com/ToothFairyAI/tf_code.git
synced 2026-03-30 22:03:58 +00:00
139 lines
4.5 KiB
TypeScript
139 lines
4.5 KiB
TypeScript
import { Ripgrep } from "../file/ripgrep"
|
|
import { Global } from "../global"
|
|
import { Filesystem } from "../util/filesystem"
|
|
import { Config } from "../config/config"
|
|
|
|
import { Instance } from "../project/instance"
|
|
import path from "path"
|
|
import os from "os"
|
|
|
|
import PROMPT_ANTHROPIC from "./prompt/anthropic.txt"
|
|
import PROMPT_ANTHROPIC_WITHOUT_TODO from "./prompt/qwen.txt"
|
|
import PROMPT_BEAST from "./prompt/beast.txt"
|
|
import PROMPT_GEMINI from "./prompt/gemini.txt"
|
|
import PROMPT_ANTHROPIC_SPOOF from "./prompt/anthropic_spoof.txt"
|
|
|
|
import PROMPT_CODEX from "./prompt/codex.txt"
|
|
import PROMPT_CODEX_INSTRUCTIONS from "./prompt/codex_header.txt"
|
|
import type { Provider } from "@/provider/provider"
|
|
import { Flag } from "@/flag/flag"
|
|
|
|
export namespace SystemPrompt {
|
|
export function header(providerID: string) {
|
|
if (providerID.includes("anthropic")) return [PROMPT_ANTHROPIC_SPOOF.trim()]
|
|
return []
|
|
}
|
|
|
|
export function instructions() {
|
|
return PROMPT_CODEX_INSTRUCTIONS.trim()
|
|
}
|
|
|
|
export function provider(model: Provider.Model) {
|
|
if (model.api.id.includes("gpt-5")) return [PROMPT_CODEX]
|
|
if (model.api.id.includes("gpt-") || model.api.id.includes("o1") || model.api.id.includes("o3"))
|
|
return [PROMPT_BEAST]
|
|
if (model.api.id.includes("gemini-")) return [PROMPT_GEMINI]
|
|
if (model.api.id.includes("claude")) return [PROMPT_ANTHROPIC]
|
|
return [PROMPT_ANTHROPIC_WITHOUT_TODO]
|
|
}
|
|
|
|
export async function environment() {
|
|
const project = Instance.project
|
|
return [
|
|
[
|
|
`Here is some useful information about the environment you are running in:`,
|
|
`<env>`,
|
|
` Working directory: ${Instance.directory}`,
|
|
` Is directory a git repo: ${project.vcs === "git" ? "yes" : "no"}`,
|
|
` Platform: ${process.platform}`,
|
|
` Today's date: ${new Date().toDateString()}`,
|
|
`</env>`,
|
|
`<files>`,
|
|
` ${
|
|
project.vcs === "git" && false
|
|
? await Ripgrep.tree({
|
|
cwd: Instance.directory,
|
|
limit: 200,
|
|
})
|
|
: ""
|
|
}`,
|
|
`</files>`,
|
|
].join("\n"),
|
|
]
|
|
}
|
|
|
|
const LOCAL_RULE_FILES = [
|
|
"AGENTS.md",
|
|
"CLAUDE.md",
|
|
"CONTEXT.md", // deprecated
|
|
]
|
|
const GLOBAL_RULE_FILES = [path.join(Global.Path.config, "AGENTS.md")]
|
|
if (!Flag.OPENCODE_DISABLE_CLAUDE_CODE_PROMPT) {
|
|
GLOBAL_RULE_FILES.push(path.join(os.homedir(), ".claude", "CLAUDE.md"))
|
|
}
|
|
|
|
if (Flag.OPENCODE_CONFIG_DIR) {
|
|
GLOBAL_RULE_FILES.push(path.join(Flag.OPENCODE_CONFIG_DIR, "AGENTS.md"))
|
|
}
|
|
|
|
export async function custom() {
|
|
const config = await Config.get()
|
|
const paths = new Set<string>()
|
|
|
|
for (const localRuleFile of LOCAL_RULE_FILES) {
|
|
const matches = await Filesystem.findUp(localRuleFile, Instance.directory, Instance.worktree)
|
|
if (matches.length > 0) {
|
|
matches.forEach((path) => paths.add(path))
|
|
break
|
|
}
|
|
}
|
|
|
|
for (const globalRuleFile of GLOBAL_RULE_FILES) {
|
|
if (await Bun.file(globalRuleFile).exists()) {
|
|
paths.add(globalRuleFile)
|
|
break
|
|
}
|
|
}
|
|
|
|
const urls: string[] = []
|
|
if (config.instructions) {
|
|
for (let instruction of config.instructions) {
|
|
if (instruction.startsWith("https://") || instruction.startsWith("http://")) {
|
|
urls.push(instruction)
|
|
continue
|
|
}
|
|
if (instruction.startsWith("~/")) {
|
|
instruction = path.join(os.homedir(), instruction.slice(2))
|
|
}
|
|
let matches: string[] = []
|
|
if (path.isAbsolute(instruction)) {
|
|
matches = await Array.fromAsync(
|
|
new Bun.Glob(path.basename(instruction)).scan({
|
|
cwd: path.dirname(instruction),
|
|
absolute: true,
|
|
onlyFiles: true,
|
|
}),
|
|
).catch(() => [])
|
|
} else {
|
|
matches = await Filesystem.globUp(instruction, Instance.directory, Instance.worktree).catch(() => [])
|
|
}
|
|
matches.forEach((path) => paths.add(path))
|
|
}
|
|
}
|
|
|
|
const foundFiles = Array.from(paths).map((p) =>
|
|
Bun.file(p)
|
|
.text()
|
|
.catch(() => "")
|
|
.then((x) => "Instructions from: " + p + "\n" + x),
|
|
)
|
|
const foundUrls = urls.map((url) =>
|
|
fetch(url, { signal: AbortSignal.timeout(5000) })
|
|
.then((res) => (res.ok ? res.text() : ""))
|
|
.catch(() => "")
|
|
.then((x) => (x ? "Instructions from: " + url + "\n" + x : "")),
|
|
)
|
|
return Promise.all([...foundFiles, ...foundUrls]).then((result) => result.filter(Boolean))
|
|
}
|
|
}
|