@bloxystudios/bloxycode 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +256 -0
- package/bin/bloxycode +84 -0
- package/package.json +133 -0
- package/src/acp/README.md +164 -0
- package/src/acp/agent.ts +1437 -0
- package/src/acp/session.ts +105 -0
- package/src/acp/types.ts +22 -0
- package/src/agent/agent.ts +356 -0
- package/src/agent/generate.txt +75 -0
- package/src/agent/prompt/bloxy.txt +46 -0
- package/src/agent/prompt/compaction.txt +12 -0
- package/src/agent/prompt/explore.txt +18 -0
- package/src/agent/prompt/summary.txt +11 -0
- package/src/agent/prompt/title.txt +44 -0
- package/src/auth/index.ts +73 -0
- package/src/bloxy/event.ts +41 -0
- package/src/bloxy/index.ts +5 -0
- package/src/bloxy/parser.ts +263 -0
- package/src/bloxy/prompt.ts +121 -0
- package/src/bloxy/runner.ts +193 -0
- package/src/bloxy/state.ts +246 -0
- package/src/bun/index.ts +134 -0
- package/src/bus/bus-event.ts +43 -0
- package/src/bus/global.ts +10 -0
- package/src/bus/index.ts +105 -0
- package/src/cli/bootstrap.ts +17 -0
- package/src/cli/cmd/acp.ts +69 -0
- package/src/cli/cmd/agent.ts +257 -0
- package/src/cli/cmd/auth.ts +400 -0
- package/src/cli/cmd/cmd.ts +7 -0
- package/src/cli/cmd/debug/agent.ts +167 -0
- package/src/cli/cmd/debug/config.ts +16 -0
- package/src/cli/cmd/debug/file.ts +97 -0
- package/src/cli/cmd/debug/index.ts +48 -0
- package/src/cli/cmd/debug/lsp.ts +52 -0
- package/src/cli/cmd/debug/ripgrep.ts +87 -0
- package/src/cli/cmd/debug/scrap.ts +16 -0
- package/src/cli/cmd/debug/skill.ts +16 -0
- package/src/cli/cmd/debug/snapshot.ts +52 -0
- package/src/cli/cmd/export.ts +88 -0
- package/src/cli/cmd/generate.ts +38 -0
- package/src/cli/cmd/github.ts +1548 -0
- package/src/cli/cmd/import.ts +98 -0
- package/src/cli/cmd/mcp.ts +755 -0
- package/src/cli/cmd/models.ts +77 -0
- package/src/cli/cmd/pr.ts +112 -0
- package/src/cli/cmd/run.ts +395 -0
- package/src/cli/cmd/serve.ts +20 -0
- package/src/cli/cmd/session.ts +135 -0
- package/src/cli/cmd/stats.ts +402 -0
- package/src/cli/cmd/tui/app.tsx +771 -0
- package/src/cli/cmd/tui/attach.ts +39 -0
- package/src/cli/cmd/tui/component/border.tsx +21 -0
- package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
- package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
- package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
- package/src/cli/cmd/tui/component/dialog-model.tsx +234 -0
- package/src/cli/cmd/tui/component/dialog-provider.tsx +256 -0
- package/src/cli/cmd/tui/component/dialog-session-list.tsx +114 -0
- package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
- package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
- package/src/cli/cmd/tui/component/dialog-status.tsx +164 -0
- package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
- package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
- package/src/cli/cmd/tui/component/logo.tsx +102 -0
- package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +653 -0
- package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
- package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
- package/src/cli/cmd/tui/component/prompt/index.tsx +1138 -0
- package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
- package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
- package/src/cli/cmd/tui/component/tips.tsx +153 -0
- package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
- package/src/cli/cmd/tui/context/args.tsx +14 -0
- package/src/cli/cmd/tui/context/directory.ts +13 -0
- package/src/cli/cmd/tui/context/exit.tsx +23 -0
- package/src/cli/cmd/tui/context/helper.tsx +25 -0
- package/src/cli/cmd/tui/context/keybind.tsx +101 -0
- package/src/cli/cmd/tui/context/kv.tsx +52 -0
- package/src/cli/cmd/tui/context/local.tsx +402 -0
- package/src/cli/cmd/tui/context/prompt.tsx +18 -0
- package/src/cli/cmd/tui/context/route.tsx +46 -0
- package/src/cli/cmd/tui/context/sdk.tsx +94 -0
- package/src/cli/cmd/tui/context/sync.tsx +470 -0
- package/src/cli/cmd/tui/context/theme/aura.json +69 -0
- package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
- package/src/cli/cmd/tui/context/theme/bloxycode.json +245 -0
- package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
- package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
- package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
- package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
- package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
- package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
- package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
- package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
- package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
- package/src/cli/cmd/tui/context/theme/github.json +233 -0
- package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
- package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
- package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
- package/src/cli/cmd/tui/context/theme/material.json +235 -0
- package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
- package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
- package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
- package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
- package/src/cli/cmd/tui/context/theme/nord.json +223 -0
- package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
- package/src/cli/cmd/tui/context/theme/orng.json +249 -0
- package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
- package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
- package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
- package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
- package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
- package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
- package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
- package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
- package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
- package/src/cli/cmd/tui/context/theme.tsx +1152 -0
- package/src/cli/cmd/tui/event.ts +48 -0
- package/src/cli/cmd/tui/routes/home.tsx +140 -0
- package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
- package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
- package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
- package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
- package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
- package/src/cli/cmd/tui/routes/session/header.tsx +142 -0
- package/src/cli/cmd/tui/routes/session/index.tsx +2048 -0
- package/src/cli/cmd/tui/routes/session/permission.tsx +508 -0
- package/src/cli/cmd/tui/routes/session/question.tsx +453 -0
- package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
- package/src/cli/cmd/tui/thread.ts +165 -0
- package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
- package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
- package/src/cli/cmd/tui/ui/dialog-export-options.tsx +204 -0
- package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
- package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
- package/src/cli/cmd/tui/ui/dialog-select.tsx +385 -0
- package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
- package/src/cli/cmd/tui/ui/link.tsx +28 -0
- package/src/cli/cmd/tui/ui/spinner.ts +368 -0
- package/src/cli/cmd/tui/ui/toast.tsx +100 -0
- package/src/cli/cmd/tui/util/clipboard.ts +160 -0
- package/src/cli/cmd/tui/util/editor.ts +32 -0
- package/src/cli/cmd/tui/util/signal.ts +7 -0
- package/src/cli/cmd/tui/util/terminal.ts +114 -0
- package/src/cli/cmd/tui/util/transcript.ts +98 -0
- package/src/cli/cmd/tui/worker.ts +152 -0
- package/src/cli/cmd/uninstall.ts +357 -0
- package/src/cli/cmd/upgrade.ts +73 -0
- package/src/cli/cmd/web.ts +81 -0
- package/src/cli/error.ts +57 -0
- package/src/cli/network.ts +53 -0
- package/src/cli/ui.ts +86 -0
- package/src/cli/upgrade.ts +25 -0
- package/src/command/index.ts +173 -0
- package/src/command/template/bloxy-resume.txt +15 -0
- package/src/command/template/bloxy-status.txt +25 -0
- package/src/command/template/bloxy-validate.txt +22 -0
- package/src/command/template/bloxy.txt +14 -0
- package/src/command/template/initialize.txt +10 -0
- package/src/command/template/review.txt +99 -0
- package/src/config/config.ts +1367 -0
- package/src/config/markdown.ts +93 -0
- package/src/env/index.ts +26 -0
- package/src/file/ignore.ts +83 -0
- package/src/file/index.ts +415 -0
- package/src/file/ripgrep.ts +407 -0
- package/src/file/time.ts +69 -0
- package/src/file/watcher.ts +127 -0
- package/src/flag/flag.ts +79 -0
- package/src/format/formatter.ts +357 -0
- package/src/format/index.ts +137 -0
- package/src/global/index.ts +55 -0
- package/src/id/id.ts +83 -0
- package/src/ide/index.ts +76 -0
- package/src/index.ts +159 -0
- package/src/installation/index.ts +246 -0
- package/src/lsp/client.ts +252 -0
- package/src/lsp/index.ts +485 -0
- package/src/lsp/language.ts +119 -0
- package/src/lsp/server.ts +2046 -0
- package/src/mcp/auth.ts +135 -0
- package/src/mcp/index.ts +934 -0
- package/src/mcp/oauth-callback.ts +200 -0
- package/src/mcp/oauth-provider.ts +154 -0
- package/src/patch/index.ts +680 -0
- package/src/permission/arity.ts +163 -0
- package/src/permission/index.ts +210 -0
- package/src/permission/next.ts +280 -0
- package/src/plugin/antigravity.ts +378 -0
- package/src/plugin/codex.ts +506 -0
- package/src/plugin/copilot.ts +298 -0
- package/src/plugin/index.ts +136 -0
- package/src/project/bootstrap.ts +35 -0
- package/src/project/instance.ts +91 -0
- package/src/project/project.ts +371 -0
- package/src/project/state.ts +66 -0
- package/src/project/vcs.ts +76 -0
- package/src/provider/auth.ts +147 -0
- package/src/provider/models-snapshot.ts +2 -0
- package/src/provider/models.ts +133 -0
- package/src/provider/provider.ts +1241 -0
- package/src/provider/sdk/openai-compatible/src/README.md +5 -0
- package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
- package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
- package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
- package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +22 -0
- package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
- package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
- package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
- package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1732 -0
- package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
- package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
- package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
- package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
- package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
- package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
- package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
- package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
- package/src/provider/transform.ts +741 -0
- package/src/pty/index.ts +241 -0
- package/src/question/index.ts +171 -0
- package/src/scheduler/index.ts +61 -0
- package/src/server/error.ts +36 -0
- package/src/server/event.ts +7 -0
- package/src/server/mdns.ts +59 -0
- package/src/server/routes/config.ts +92 -0
- package/src/server/routes/experimental.ts +208 -0
- package/src/server/routes/file.ts +197 -0
- package/src/server/routes/global.ts +135 -0
- package/src/server/routes/mcp.ts +225 -0
- package/src/server/routes/permission.ts +68 -0
- package/src/server/routes/project.ts +82 -0
- package/src/server/routes/provider.ts +165 -0
- package/src/server/routes/pty.ts +169 -0
- package/src/server/routes/question.ts +98 -0
- package/src/server/routes/session.ts +939 -0
- package/src/server/routes/tui.ts +379 -0
- package/src/server/server.ts +604 -0
- package/src/session/compaction.ts +225 -0
- package/src/session/fallback.ts +246 -0
- package/src/session/index.ts +498 -0
- package/src/session/instruction.ts +164 -0
- package/src/session/llm.ts +298 -0
- package/src/session/message-v2.ts +747 -0
- package/src/session/message.ts +189 -0
- package/src/session/processor.ts +450 -0
- package/src/session/prompt/anthropic-20250930.txt +166 -0
- package/src/session/prompt/anthropic.txt +105 -0
- package/src/session/prompt/beast.txt +147 -0
- package/src/session/prompt/build-switch.txt +5 -0
- package/src/session/prompt/codex_header.txt +79 -0
- package/src/session/prompt/copilot-gpt-5.txt +143 -0
- package/src/session/prompt/gemini.txt +155 -0
- package/src/session/prompt/max-steps.txt +16 -0
- package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
- package/src/session/prompt/plan.txt +26 -0
- package/src/session/prompt/qwen.txt +109 -0
- package/src/session/prompt.ts +1822 -0
- package/src/session/retry.ts +99 -0
- package/src/session/revert.ts +121 -0
- package/src/session/status.ts +100 -0
- package/src/session/summary.ts +217 -0
- package/src/session/system.ts +52 -0
- package/src/session/todo.ts +37 -0
- package/src/share/share-next.ts +200 -0
- package/src/share/share.ts +92 -0
- package/src/shell/shell.ts +67 -0
- package/src/skill/index.ts +1 -0
- package/src/skill/skill.ts +135 -0
- package/src/snapshot/index.ts +236 -0
- package/src/storage/storage.ts +227 -0
- package/src/tool/apply_patch.ts +281 -0
- package/src/tool/apply_patch.txt +33 -0
- package/src/tool/bash.ts +258 -0
- package/src/tool/bash.txt +115 -0
- package/src/tool/batch.ts +175 -0
- package/src/tool/batch.txt +24 -0
- package/src/tool/bloxy-control.ts +123 -0
- package/src/tool/bloxy-control.txt +13 -0
- package/src/tool/codesearch.ts +132 -0
- package/src/tool/codesearch.txt +12 -0
- package/src/tool/edit.ts +655 -0
- package/src/tool/edit.txt +10 -0
- package/src/tool/external-directory.ts +32 -0
- package/src/tool/glob.ts +77 -0
- package/src/tool/glob.txt +6 -0
- package/src/tool/grep.ts +154 -0
- package/src/tool/grep.txt +8 -0
- package/src/tool/invalid.ts +17 -0
- package/src/tool/ls.ts +121 -0
- package/src/tool/ls.txt +1 -0
- package/src/tool/lsp.ts +96 -0
- package/src/tool/lsp.txt +19 -0
- package/src/tool/multiedit.ts +46 -0
- package/src/tool/multiedit.txt +41 -0
- package/src/tool/plan-enter.txt +14 -0
- package/src/tool/plan-exit.txt +13 -0
- package/src/tool/plan.ts +130 -0
- package/src/tool/question.ts +33 -0
- package/src/tool/question.txt +10 -0
- package/src/tool/read.ts +211 -0
- package/src/tool/read.txt +12 -0
- package/src/tool/registry.ts +161 -0
- package/src/tool/skill.ts +82 -0
- package/src/tool/task.ts +191 -0
- package/src/tool/task.txt +60 -0
- package/src/tool/todo.ts +53 -0
- package/src/tool/todoread.txt +14 -0
- package/src/tool/todowrite.txt +167 -0
- package/src/tool/tool.ts +89 -0
- package/src/tool/truncation.ts +106 -0
- package/src/tool/webfetch.ts +188 -0
- package/src/tool/webfetch.txt +13 -0
- package/src/tool/websearch.ts +150 -0
- package/src/tool/websearch.txt +14 -0
- package/src/tool/write.ts +85 -0
- package/src/tool/write.txt +8 -0
- package/src/util/archive.ts +16 -0
- package/src/util/binary.ts +41 -0
- package/src/util/color.ts +19 -0
- package/src/util/context.ts +25 -0
- package/src/util/defer.ts +12 -0
- package/src/util/error.ts +54 -0
- package/src/util/eventloop.ts +20 -0
- package/src/util/filesystem.ts +93 -0
- package/src/util/fn.ts +11 -0
- package/src/util/format.ts +20 -0
- package/src/util/iife.ts +3 -0
- package/src/util/keybind.ts +103 -0
- package/src/util/lazy.ts +23 -0
- package/src/util/locale.ts +81 -0
- package/src/util/lock.ts +98 -0
- package/src/util/log.ts +180 -0
- package/src/util/queue.ts +32 -0
- package/src/util/rpc.ts +66 -0
- package/src/util/scrap.ts +10 -0
- package/src/util/signal.ts +12 -0
- package/src/util/slug.ts +74 -0
- package/src/util/timeout.ts +14 -0
- package/src/util/token.ts +7 -0
- package/src/util/wildcard.ts +56 -0
- package/src/worktree/index.ts +549 -0
|
@@ -0,0 +1,498 @@
|
|
|
1
|
+
import { Slug } from "@/util/slug"
|
|
2
|
+
import path from "path"
|
|
3
|
+
import { BusEvent } from "@/bus/bus-event"
|
|
4
|
+
import { Bus } from "@/bus"
|
|
5
|
+
import { Decimal } from "decimal.js"
|
|
6
|
+
import z from "zod"
|
|
7
|
+
import { type LanguageModelUsage, type ProviderMetadata } from "ai"
|
|
8
|
+
import { Config } from "../config/config"
|
|
9
|
+
import { Flag } from "../flag/flag"
|
|
10
|
+
import { Identifier } from "../id/id"
|
|
11
|
+
import { Installation } from "../installation"
|
|
12
|
+
|
|
13
|
+
import { Storage } from "../storage/storage"
|
|
14
|
+
import { Log } from "../util/log"
|
|
15
|
+
import { MessageV2 } from "./message-v2"
|
|
16
|
+
import { Instance } from "../project/instance"
|
|
17
|
+
import { SessionPrompt } from "./prompt"
|
|
18
|
+
import { fn } from "@/util/fn"
|
|
19
|
+
import { Command } from "../command"
|
|
20
|
+
import { Snapshot } from "@/snapshot"
|
|
21
|
+
|
|
22
|
+
import type { Provider } from "@/provider/provider"
|
|
23
|
+
import { PermissionNext } from "@/permission/next"
|
|
24
|
+
import { Global } from "@/global"
|
|
25
|
+
|
|
26
|
+
export namespace Session {
|
|
27
|
+
const log = Log.create({ service: "session" })
|
|
28
|
+
|
|
29
|
+
const parentTitlePrefix = "New session - "
|
|
30
|
+
const childTitlePrefix = "Child session - "
|
|
31
|
+
|
|
32
|
+
function createDefaultTitle(isChild = false) {
|
|
33
|
+
return (isChild ? childTitlePrefix : parentTitlePrefix) + new Date().toISOString()
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
export function isDefaultTitle(title: string) {
|
|
37
|
+
return new RegExp(
|
|
38
|
+
`^(${parentTitlePrefix}|${childTitlePrefix})\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}\\.\\d{3}Z$`,
|
|
39
|
+
).test(title)
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
export const Info = z
|
|
43
|
+
.object({
|
|
44
|
+
id: Identifier.schema("session"),
|
|
45
|
+
slug: z.string(),
|
|
46
|
+
projectID: z.string(),
|
|
47
|
+
directory: z.string(),
|
|
48
|
+
parentID: Identifier.schema("session").optional(),
|
|
49
|
+
summary: z
|
|
50
|
+
.object({
|
|
51
|
+
additions: z.number(),
|
|
52
|
+
deletions: z.number(),
|
|
53
|
+
files: z.number(),
|
|
54
|
+
diffs: Snapshot.FileDiff.array().optional(),
|
|
55
|
+
})
|
|
56
|
+
.optional(),
|
|
57
|
+
share: z
|
|
58
|
+
.object({
|
|
59
|
+
url: z.string(),
|
|
60
|
+
})
|
|
61
|
+
.optional(),
|
|
62
|
+
title: z.string(),
|
|
63
|
+
version: z.string(),
|
|
64
|
+
time: z.object({
|
|
65
|
+
created: z.number(),
|
|
66
|
+
updated: z.number(),
|
|
67
|
+
compacting: z.number().optional(),
|
|
68
|
+
archived: z.number().optional(),
|
|
69
|
+
}),
|
|
70
|
+
permission: PermissionNext.Ruleset.optional(),
|
|
71
|
+
revert: z
|
|
72
|
+
.object({
|
|
73
|
+
messageID: z.string(),
|
|
74
|
+
partID: z.string().optional(),
|
|
75
|
+
snapshot: z.string().optional(),
|
|
76
|
+
diff: z.string().optional(),
|
|
77
|
+
})
|
|
78
|
+
.optional(),
|
|
79
|
+
})
|
|
80
|
+
.meta({
|
|
81
|
+
ref: "Session",
|
|
82
|
+
})
|
|
83
|
+
export type Info = z.output<typeof Info>
|
|
84
|
+
|
|
85
|
+
export const ShareInfo = z
|
|
86
|
+
.object({
|
|
87
|
+
secret: z.string(),
|
|
88
|
+
url: z.string(),
|
|
89
|
+
})
|
|
90
|
+
.meta({
|
|
91
|
+
ref: "SessionShare",
|
|
92
|
+
})
|
|
93
|
+
export type ShareInfo = z.output<typeof ShareInfo>
|
|
94
|
+
|
|
95
|
+
export const Event = {
|
|
96
|
+
Created: BusEvent.define(
|
|
97
|
+
"session.created",
|
|
98
|
+
z.object({
|
|
99
|
+
info: Info,
|
|
100
|
+
}),
|
|
101
|
+
),
|
|
102
|
+
Updated: BusEvent.define(
|
|
103
|
+
"session.updated",
|
|
104
|
+
z.object({
|
|
105
|
+
info: Info,
|
|
106
|
+
}),
|
|
107
|
+
),
|
|
108
|
+
Deleted: BusEvent.define(
|
|
109
|
+
"session.deleted",
|
|
110
|
+
z.object({
|
|
111
|
+
info: Info,
|
|
112
|
+
}),
|
|
113
|
+
),
|
|
114
|
+
Diff: BusEvent.define(
|
|
115
|
+
"session.diff",
|
|
116
|
+
z.object({
|
|
117
|
+
sessionID: z.string(),
|
|
118
|
+
diff: Snapshot.FileDiff.array(),
|
|
119
|
+
}),
|
|
120
|
+
),
|
|
121
|
+
Error: BusEvent.define(
|
|
122
|
+
"session.error",
|
|
123
|
+
z.object({
|
|
124
|
+
sessionID: z.string().optional(),
|
|
125
|
+
error: MessageV2.Assistant.shape.error,
|
|
126
|
+
}),
|
|
127
|
+
),
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
export const create = fn(
|
|
131
|
+
z
|
|
132
|
+
.object({
|
|
133
|
+
parentID: Identifier.schema("session").optional(),
|
|
134
|
+
title: z.string().optional(),
|
|
135
|
+
permission: Info.shape.permission,
|
|
136
|
+
})
|
|
137
|
+
.optional(),
|
|
138
|
+
async (input) => {
|
|
139
|
+
return createNext({
|
|
140
|
+
parentID: input?.parentID,
|
|
141
|
+
directory: Instance.directory,
|
|
142
|
+
title: input?.title,
|
|
143
|
+
permission: input?.permission,
|
|
144
|
+
})
|
|
145
|
+
},
|
|
146
|
+
)
|
|
147
|
+
|
|
148
|
+
export const fork = fn(
|
|
149
|
+
z.object({
|
|
150
|
+
sessionID: Identifier.schema("session"),
|
|
151
|
+
messageID: Identifier.schema("message").optional(),
|
|
152
|
+
}),
|
|
153
|
+
async (input) => {
|
|
154
|
+
const session = await createNext({
|
|
155
|
+
directory: Instance.directory,
|
|
156
|
+
})
|
|
157
|
+
const msgs = await messages({ sessionID: input.sessionID })
|
|
158
|
+
const idMap = new Map<string, string>()
|
|
159
|
+
|
|
160
|
+
for (const msg of msgs) {
|
|
161
|
+
if (input.messageID && msg.info.id >= input.messageID) break
|
|
162
|
+
const newID = Identifier.ascending("message")
|
|
163
|
+
idMap.set(msg.info.id, newID)
|
|
164
|
+
|
|
165
|
+
const parentID = msg.info.role === "assistant" && msg.info.parentID ? idMap.get(msg.info.parentID) : undefined
|
|
166
|
+
const cloned = await updateMessage({
|
|
167
|
+
...msg.info,
|
|
168
|
+
sessionID: session.id,
|
|
169
|
+
id: newID,
|
|
170
|
+
...(parentID && { parentID }),
|
|
171
|
+
})
|
|
172
|
+
|
|
173
|
+
for (const part of msg.parts) {
|
|
174
|
+
await updatePart({
|
|
175
|
+
...part,
|
|
176
|
+
id: Identifier.ascending("part"),
|
|
177
|
+
messageID: cloned.id,
|
|
178
|
+
sessionID: session.id,
|
|
179
|
+
})
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
return session
|
|
183
|
+
},
|
|
184
|
+
)
|
|
185
|
+
|
|
186
|
+
export const touch = fn(Identifier.schema("session"), async (sessionID) => {
|
|
187
|
+
await update(sessionID, (draft) => {
|
|
188
|
+
draft.time.updated = Date.now()
|
|
189
|
+
})
|
|
190
|
+
})
|
|
191
|
+
|
|
192
|
+
export async function createNext(input: {
|
|
193
|
+
id?: string
|
|
194
|
+
title?: string
|
|
195
|
+
parentID?: string
|
|
196
|
+
directory: string
|
|
197
|
+
permission?: PermissionNext.Ruleset
|
|
198
|
+
}) {
|
|
199
|
+
const result: Info = {
|
|
200
|
+
id: Identifier.descending("session", input.id),
|
|
201
|
+
slug: Slug.create(),
|
|
202
|
+
version: Installation.VERSION,
|
|
203
|
+
projectID: Instance.project.id,
|
|
204
|
+
directory: input.directory,
|
|
205
|
+
parentID: input.parentID,
|
|
206
|
+
title: input.title ?? createDefaultTitle(!!input.parentID),
|
|
207
|
+
permission: input.permission,
|
|
208
|
+
time: {
|
|
209
|
+
created: Date.now(),
|
|
210
|
+
updated: Date.now(),
|
|
211
|
+
},
|
|
212
|
+
}
|
|
213
|
+
log.info("created", result)
|
|
214
|
+
await Storage.write(["session", Instance.project.id, result.id], result)
|
|
215
|
+
Bus.publish(Event.Created, {
|
|
216
|
+
info: result,
|
|
217
|
+
})
|
|
218
|
+
const cfg = await Config.get()
|
|
219
|
+
if (!result.parentID && (Flag.BLOXYCODE_AUTO_SHARE || cfg.share === "auto"))
|
|
220
|
+
share(result.id)
|
|
221
|
+
.then((share) => {
|
|
222
|
+
update(result.id, (draft) => {
|
|
223
|
+
draft.share = share
|
|
224
|
+
})
|
|
225
|
+
})
|
|
226
|
+
.catch(() => {
|
|
227
|
+
// Silently ignore sharing errors during session creation
|
|
228
|
+
})
|
|
229
|
+
Bus.publish(Event.Updated, {
|
|
230
|
+
info: result,
|
|
231
|
+
})
|
|
232
|
+
return result
|
|
233
|
+
}
|
|
234
|
+
|
|
235
|
+
export function plan(input: { slug: string; time: { created: number } }) {
|
|
236
|
+
const base = Instance.project.vcs
|
|
237
|
+
? path.join(Instance.worktree, ".bloxycode", "plans")
|
|
238
|
+
: path.join(Global.Path.data, "plans")
|
|
239
|
+
return path.join(base, [input.time.created, input.slug].join("-") + ".md")
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
export const get = fn(Identifier.schema("session"), async (id) => {
|
|
243
|
+
const read = await Storage.read<Info>(["session", Instance.project.id, id])
|
|
244
|
+
return read as Info
|
|
245
|
+
})
|
|
246
|
+
|
|
247
|
+
export const getShare = fn(Identifier.schema("session"), async (id) => {
|
|
248
|
+
return Storage.read<ShareInfo>(["share", id])
|
|
249
|
+
})
|
|
250
|
+
|
|
251
|
+
export const share = fn(Identifier.schema("session"), async (id) => {
|
|
252
|
+
const cfg = await Config.get()
|
|
253
|
+
if (cfg.share === "disabled") {
|
|
254
|
+
throw new Error("Sharing is disabled in configuration")
|
|
255
|
+
}
|
|
256
|
+
const { ShareNext } = await import("@/share/share-next")
|
|
257
|
+
const share = await ShareNext.create(id)
|
|
258
|
+
await update(
|
|
259
|
+
id,
|
|
260
|
+
(draft) => {
|
|
261
|
+
draft.share = {
|
|
262
|
+
url: share.url,
|
|
263
|
+
}
|
|
264
|
+
},
|
|
265
|
+
{ touch: false },
|
|
266
|
+
)
|
|
267
|
+
return share
|
|
268
|
+
})
|
|
269
|
+
|
|
270
|
+
export const unshare = fn(Identifier.schema("session"), async (id) => {
|
|
271
|
+
// Use ShareNext to remove the share (same as share function uses ShareNext to create)
|
|
272
|
+
const { ShareNext } = await import("@/share/share-next")
|
|
273
|
+
await ShareNext.remove(id)
|
|
274
|
+
await update(
|
|
275
|
+
id,
|
|
276
|
+
(draft) => {
|
|
277
|
+
draft.share = undefined
|
|
278
|
+
},
|
|
279
|
+
{ touch: false },
|
|
280
|
+
)
|
|
281
|
+
})
|
|
282
|
+
|
|
283
|
+
export async function update(id: string, editor: (session: Info) => void, options?: { touch?: boolean }) {
|
|
284
|
+
const project = Instance.project
|
|
285
|
+
const result = await Storage.update<Info>(["session", project.id, id], (draft) => {
|
|
286
|
+
editor(draft)
|
|
287
|
+
if (options?.touch !== false) {
|
|
288
|
+
draft.time.updated = Date.now()
|
|
289
|
+
}
|
|
290
|
+
})
|
|
291
|
+
Bus.publish(Event.Updated, {
|
|
292
|
+
info: result,
|
|
293
|
+
})
|
|
294
|
+
return result
|
|
295
|
+
}
|
|
296
|
+
|
|
297
|
+
export const diff = fn(Identifier.schema("session"), async (sessionID) => {
|
|
298
|
+
const diffs = await Storage.read<Snapshot.FileDiff[]>(["session_diff", sessionID])
|
|
299
|
+
return diffs ?? []
|
|
300
|
+
})
|
|
301
|
+
|
|
302
|
+
export const messages = fn(
|
|
303
|
+
z.object({
|
|
304
|
+
sessionID: Identifier.schema("session"),
|
|
305
|
+
limit: z.number().optional(),
|
|
306
|
+
}),
|
|
307
|
+
async (input) => {
|
|
308
|
+
const result = [] as MessageV2.WithParts[]
|
|
309
|
+
for await (const msg of MessageV2.stream(input.sessionID)) {
|
|
310
|
+
if (input.limit && result.length >= input.limit) break
|
|
311
|
+
result.push(msg)
|
|
312
|
+
}
|
|
313
|
+
result.reverse()
|
|
314
|
+
return result
|
|
315
|
+
},
|
|
316
|
+
)
|
|
317
|
+
|
|
318
|
+
export async function* list() {
|
|
319
|
+
const project = Instance.project
|
|
320
|
+
for (const item of await Storage.list(["session", project.id])) {
|
|
321
|
+
yield Storage.read<Info>(item)
|
|
322
|
+
}
|
|
323
|
+
}
|
|
324
|
+
|
|
325
|
+
export const children = fn(Identifier.schema("session"), async (parentID) => {
|
|
326
|
+
const project = Instance.project
|
|
327
|
+
const result = [] as Session.Info[]
|
|
328
|
+
for (const item of await Storage.list(["session", project.id])) {
|
|
329
|
+
const session = await Storage.read<Info>(item)
|
|
330
|
+
if (session.parentID !== parentID) continue
|
|
331
|
+
result.push(session)
|
|
332
|
+
}
|
|
333
|
+
return result
|
|
334
|
+
})
|
|
335
|
+
|
|
336
|
+
export const remove = fn(Identifier.schema("session"), async (sessionID) => {
|
|
337
|
+
const project = Instance.project
|
|
338
|
+
try {
|
|
339
|
+
const session = await get(sessionID)
|
|
340
|
+
for (const child of await children(sessionID)) {
|
|
341
|
+
await remove(child.id)
|
|
342
|
+
}
|
|
343
|
+
await unshare(sessionID).catch(() => {})
|
|
344
|
+
for (const msg of await Storage.list(["message", sessionID])) {
|
|
345
|
+
for (const part of await Storage.list(["part", msg.at(-1)!])) {
|
|
346
|
+
await Storage.remove(part)
|
|
347
|
+
}
|
|
348
|
+
await Storage.remove(msg)
|
|
349
|
+
}
|
|
350
|
+
await Storage.remove(["session", project.id, sessionID])
|
|
351
|
+
Bus.publish(Event.Deleted, {
|
|
352
|
+
info: session,
|
|
353
|
+
})
|
|
354
|
+
} catch (e) {
|
|
355
|
+
log.error(e)
|
|
356
|
+
}
|
|
357
|
+
})
|
|
358
|
+
|
|
359
|
+
export const updateMessage = fn(MessageV2.Info, async (msg) => {
|
|
360
|
+
await Storage.write(["message", msg.sessionID, msg.id], msg)
|
|
361
|
+
Bus.publish(MessageV2.Event.Updated, {
|
|
362
|
+
info: msg,
|
|
363
|
+
})
|
|
364
|
+
return msg
|
|
365
|
+
})
|
|
366
|
+
|
|
367
|
+
export const removeMessage = fn(
|
|
368
|
+
z.object({
|
|
369
|
+
sessionID: Identifier.schema("session"),
|
|
370
|
+
messageID: Identifier.schema("message"),
|
|
371
|
+
}),
|
|
372
|
+
async (input) => {
|
|
373
|
+
await Storage.remove(["message", input.sessionID, input.messageID])
|
|
374
|
+
Bus.publish(MessageV2.Event.Removed, {
|
|
375
|
+
sessionID: input.sessionID,
|
|
376
|
+
messageID: input.messageID,
|
|
377
|
+
})
|
|
378
|
+
return input.messageID
|
|
379
|
+
},
|
|
380
|
+
)
|
|
381
|
+
|
|
382
|
+
export const removePart = fn(
|
|
383
|
+
z.object({
|
|
384
|
+
sessionID: Identifier.schema("session"),
|
|
385
|
+
messageID: Identifier.schema("message"),
|
|
386
|
+
partID: Identifier.schema("part"),
|
|
387
|
+
}),
|
|
388
|
+
async (input) => {
|
|
389
|
+
await Storage.remove(["part", input.messageID, input.partID])
|
|
390
|
+
Bus.publish(MessageV2.Event.PartRemoved, {
|
|
391
|
+
sessionID: input.sessionID,
|
|
392
|
+
messageID: input.messageID,
|
|
393
|
+
partID: input.partID,
|
|
394
|
+
})
|
|
395
|
+
return input.partID
|
|
396
|
+
},
|
|
397
|
+
)
|
|
398
|
+
|
|
399
|
+
const UpdatePartInput = z.union([
|
|
400
|
+
MessageV2.Part,
|
|
401
|
+
z.object({
|
|
402
|
+
part: MessageV2.TextPart,
|
|
403
|
+
delta: z.string(),
|
|
404
|
+
}),
|
|
405
|
+
z.object({
|
|
406
|
+
part: MessageV2.ReasoningPart,
|
|
407
|
+
delta: z.string(),
|
|
408
|
+
}),
|
|
409
|
+
])
|
|
410
|
+
|
|
411
|
+
export const updatePart = fn(UpdatePartInput, async (input) => {
|
|
412
|
+
const part = "delta" in input ? input.part : input
|
|
413
|
+
const delta = "delta" in input ? input.delta : undefined
|
|
414
|
+
await Storage.write(["part", part.messageID, part.id], part)
|
|
415
|
+
Bus.publish(MessageV2.Event.PartUpdated, {
|
|
416
|
+
part,
|
|
417
|
+
delta,
|
|
418
|
+
})
|
|
419
|
+
return part
|
|
420
|
+
})
|
|
421
|
+
|
|
422
|
+
export const getUsage = fn(
|
|
423
|
+
z.object({
|
|
424
|
+
model: z.custom<Provider.Model>(),
|
|
425
|
+
usage: z.custom<LanguageModelUsage>(),
|
|
426
|
+
metadata: z.custom<ProviderMetadata>().optional(),
|
|
427
|
+
}),
|
|
428
|
+
(input) => {
|
|
429
|
+
const cachedInputTokens = input.usage.cachedInputTokens ?? 0
|
|
430
|
+
const excludesCachedTokens = !!(input.metadata?.["anthropic"] || input.metadata?.["bedrock"])
|
|
431
|
+
const adjustedInputTokens = excludesCachedTokens
|
|
432
|
+
? (input.usage.inputTokens ?? 0)
|
|
433
|
+
: (input.usage.inputTokens ?? 0) - cachedInputTokens
|
|
434
|
+
const safe = (value: number) => {
|
|
435
|
+
if (!Number.isFinite(value)) return 0
|
|
436
|
+
return value
|
|
437
|
+
}
|
|
438
|
+
|
|
439
|
+
const tokens = {
|
|
440
|
+
input: safe(adjustedInputTokens),
|
|
441
|
+
output: safe(input.usage.outputTokens ?? 0),
|
|
442
|
+
reasoning: safe(input.usage?.reasoningTokens ?? 0),
|
|
443
|
+
cache: {
|
|
444
|
+
write: safe(
|
|
445
|
+
(input.metadata?.["anthropic"]?.["cacheCreationInputTokens"] ??
|
|
446
|
+
// @ts-expect-error
|
|
447
|
+
input.metadata?.["bedrock"]?.["usage"]?.["cacheWriteInputTokens"] ??
|
|
448
|
+
0) as number,
|
|
449
|
+
),
|
|
450
|
+
read: safe(cachedInputTokens),
|
|
451
|
+
},
|
|
452
|
+
}
|
|
453
|
+
|
|
454
|
+
const costInfo =
|
|
455
|
+
input.model.cost?.experimentalOver200K && tokens.input + tokens.cache.read > 200_000
|
|
456
|
+
? input.model.cost.experimentalOver200K
|
|
457
|
+
: input.model.cost
|
|
458
|
+
return {
|
|
459
|
+
cost: safe(
|
|
460
|
+
new Decimal(0)
|
|
461
|
+
.add(new Decimal(tokens.input).mul(costInfo?.input ?? 0).div(1_000_000))
|
|
462
|
+
.add(new Decimal(tokens.output).mul(costInfo?.output ?? 0).div(1_000_000))
|
|
463
|
+
.add(new Decimal(tokens.cache.read).mul(costInfo?.cache?.read ?? 0).div(1_000_000))
|
|
464
|
+
.add(new Decimal(tokens.cache.write).mul(costInfo?.cache?.write ?? 0).div(1_000_000))
|
|
465
|
+
// TODO: update models.dev to have better pricing model, for now:
|
|
466
|
+
// charge reasoning tokens at the same rate as output tokens
|
|
467
|
+
.add(new Decimal(tokens.reasoning).mul(costInfo?.output ?? 0).div(1_000_000))
|
|
468
|
+
.toNumber(),
|
|
469
|
+
),
|
|
470
|
+
tokens,
|
|
471
|
+
}
|
|
472
|
+
},
|
|
473
|
+
)
|
|
474
|
+
|
|
475
|
+
export class BusyError extends Error {
|
|
476
|
+
constructor(public readonly sessionID: string) {
|
|
477
|
+
super(`Session ${sessionID} is busy`)
|
|
478
|
+
}
|
|
479
|
+
}
|
|
480
|
+
|
|
481
|
+
export const initialize = fn(
|
|
482
|
+
z.object({
|
|
483
|
+
sessionID: Identifier.schema("session"),
|
|
484
|
+
modelID: z.string(),
|
|
485
|
+
providerID: z.string(),
|
|
486
|
+
messageID: Identifier.schema("message"),
|
|
487
|
+
}),
|
|
488
|
+
async (input) => {
|
|
489
|
+
await SessionPrompt.command({
|
|
490
|
+
sessionID: input.sessionID,
|
|
491
|
+
messageID: input.messageID,
|
|
492
|
+
model: input.providerID + "/" + input.modelID,
|
|
493
|
+
command: Command.Default.INIT,
|
|
494
|
+
arguments: "",
|
|
495
|
+
})
|
|
496
|
+
},
|
|
497
|
+
)
|
|
498
|
+
}
|
|
@@ -0,0 +1,164 @@
|
|
|
1
|
+
import path from "path"
|
|
2
|
+
import os from "os"
|
|
3
|
+
import { Global } from "../global"
|
|
4
|
+
import { Filesystem } from "../util/filesystem"
|
|
5
|
+
import { Config } from "../config/config"
|
|
6
|
+
import { Instance } from "../project/instance"
|
|
7
|
+
import { Flag } from "@/flag/flag"
|
|
8
|
+
import { Log } from "../util/log"
|
|
9
|
+
import type { MessageV2 } from "./message-v2"
|
|
10
|
+
|
|
11
|
+
const log = Log.create({ service: "instruction" })
|
|
12
|
+
|
|
13
|
+
const FILES = [
|
|
14
|
+
"AGENTS.md",
|
|
15
|
+
"CLAUDE.md",
|
|
16
|
+
"CONTEXT.md", // deprecated
|
|
17
|
+
]
|
|
18
|
+
|
|
19
|
+
function globalFiles() {
|
|
20
|
+
const files = [path.join(Global.Path.config, "AGENTS.md")]
|
|
21
|
+
if (!Flag.BLOXYCODE_DISABLE_CLAUDE_CODE_PROMPT) {
|
|
22
|
+
files.push(path.join(os.homedir(), ".claude", "CLAUDE.md"))
|
|
23
|
+
}
|
|
24
|
+
if (Flag.BLOXYCODE_CONFIG_DIR) {
|
|
25
|
+
files.push(path.join(Flag.BLOXYCODE_CONFIG_DIR, "AGENTS.md"))
|
|
26
|
+
}
|
|
27
|
+
return files
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
async function resolveRelative(instruction: string): Promise<string[]> {
|
|
31
|
+
if (!Flag.BLOXYCODE_DISABLE_PROJECT_CONFIG) {
|
|
32
|
+
return Filesystem.globUp(instruction, Instance.directory, Instance.worktree).catch(() => [])
|
|
33
|
+
}
|
|
34
|
+
if (!Flag.BLOXYCODE_CONFIG_DIR) {
|
|
35
|
+
log.warn(
|
|
36
|
+
`Skipping relative instruction "${instruction}" - no BLOXYCODE_CONFIG_DIR set while project config is disabled`,
|
|
37
|
+
)
|
|
38
|
+
return []
|
|
39
|
+
}
|
|
40
|
+
return Filesystem.globUp(instruction, Flag.BLOXYCODE_CONFIG_DIR, Flag.BLOXYCODE_CONFIG_DIR).catch(() => [])
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
export namespace InstructionPrompt {
|
|
44
|
+
export async function systemPaths() {
|
|
45
|
+
const config = await Config.get()
|
|
46
|
+
const paths = new Set<string>()
|
|
47
|
+
|
|
48
|
+
if (!Flag.BLOXYCODE_DISABLE_PROJECT_CONFIG) {
|
|
49
|
+
for (const file of FILES) {
|
|
50
|
+
const matches = await Filesystem.findUp(file, Instance.directory, Instance.worktree)
|
|
51
|
+
if (matches.length > 0) {
|
|
52
|
+
matches.forEach((p) => paths.add(path.resolve(p)))
|
|
53
|
+
break
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
for (const file of globalFiles()) {
|
|
59
|
+
if (await Bun.file(file).exists()) {
|
|
60
|
+
paths.add(path.resolve(file))
|
|
61
|
+
break
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
if (config.instructions) {
|
|
66
|
+
for (let instruction of config.instructions) {
|
|
67
|
+
if (instruction.startsWith("https://") || instruction.startsWith("http://")) continue
|
|
68
|
+
if (instruction.startsWith("~/")) {
|
|
69
|
+
instruction = path.join(os.homedir(), instruction.slice(2))
|
|
70
|
+
}
|
|
71
|
+
const matches = path.isAbsolute(instruction)
|
|
72
|
+
? await Array.fromAsync(
|
|
73
|
+
new Bun.Glob(path.basename(instruction)).scan({
|
|
74
|
+
cwd: path.dirname(instruction),
|
|
75
|
+
absolute: true,
|
|
76
|
+
onlyFiles: true,
|
|
77
|
+
}),
|
|
78
|
+
).catch(() => [])
|
|
79
|
+
: await resolveRelative(instruction)
|
|
80
|
+
matches.forEach((p) => paths.add(path.resolve(p)))
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
return paths
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
export async function system() {
|
|
88
|
+
const config = await Config.get()
|
|
89
|
+
const paths = await systemPaths()
|
|
90
|
+
|
|
91
|
+
const files = Array.from(paths).map(async (p) => {
|
|
92
|
+
const content = await Bun.file(p)
|
|
93
|
+
.text()
|
|
94
|
+
.catch(() => "")
|
|
95
|
+
return content ? "Instructions from: " + p + "\n" + content : ""
|
|
96
|
+
})
|
|
97
|
+
|
|
98
|
+
const urls: string[] = []
|
|
99
|
+
if (config.instructions) {
|
|
100
|
+
for (const instruction of config.instructions) {
|
|
101
|
+
if (instruction.startsWith("https://") || instruction.startsWith("http://")) {
|
|
102
|
+
urls.push(instruction)
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
const fetches = urls.map((url) =>
|
|
107
|
+
fetch(url, { signal: AbortSignal.timeout(5000) })
|
|
108
|
+
.then((res) => (res.ok ? res.text() : ""))
|
|
109
|
+
.catch(() => "")
|
|
110
|
+
.then((x) => (x ? "Instructions from: " + url + "\n" + x : "")),
|
|
111
|
+
)
|
|
112
|
+
|
|
113
|
+
return Promise.all([...files, ...fetches]).then((result) => result.filter(Boolean))
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
export function loaded(messages: MessageV2.WithParts[]) {
|
|
117
|
+
const paths = new Set<string>()
|
|
118
|
+
for (const msg of messages) {
|
|
119
|
+
for (const part of msg.parts) {
|
|
120
|
+
if (part.type === "tool" && part.tool === "read" && part.state.status === "completed") {
|
|
121
|
+
if (part.state.time.compacted) continue
|
|
122
|
+
const loaded = part.state.metadata?.loaded
|
|
123
|
+
if (!loaded || !Array.isArray(loaded)) continue
|
|
124
|
+
for (const p of loaded) {
|
|
125
|
+
if (typeof p === "string") paths.add(p)
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
return paths
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
export async function find(dir: string) {
|
|
134
|
+
for (const file of FILES) {
|
|
135
|
+
const filepath = path.resolve(path.join(dir, file))
|
|
136
|
+
if (await Bun.file(filepath).exists()) return filepath
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
export async function resolve(messages: MessageV2.WithParts[], filepath: string) {
|
|
141
|
+
const system = await systemPaths()
|
|
142
|
+
const already = loaded(messages)
|
|
143
|
+
const results: { filepath: string; content: string }[] = []
|
|
144
|
+
|
|
145
|
+
let current = path.dirname(path.resolve(filepath))
|
|
146
|
+
const root = path.resolve(Instance.directory)
|
|
147
|
+
|
|
148
|
+
while (current.startsWith(root)) {
|
|
149
|
+
const found = await find(current)
|
|
150
|
+
if (found && !system.has(found) && !already.has(found)) {
|
|
151
|
+
const content = await Bun.file(found)
|
|
152
|
+
.text()
|
|
153
|
+
.catch(() => undefined)
|
|
154
|
+
if (content) {
|
|
155
|
+
results.push({ filepath: found, content: "Instructions from: " + found + "\n" + content })
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
if (current === root) break
|
|
159
|
+
current = path.dirname(current)
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
return results
|
|
163
|
+
}
|
|
164
|
+
}
|