@link-assistant/agent 0.0.8
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/EXAMPLES.md +383 -0
- package/LICENSE +24 -0
- package/MODELS.md +95 -0
- package/README.md +388 -0
- package/TOOLS.md +134 -0
- package/package.json +89 -0
- package/src/agent/agent.ts +150 -0
- package/src/agent/generate.txt +75 -0
- package/src/auth/index.ts +64 -0
- package/src/bun/index.ts +96 -0
- package/src/bus/global.ts +10 -0
- package/src/bus/index.ts +119 -0
- package/src/cli/bootstrap.js +41 -0
- package/src/cli/bootstrap.ts +17 -0
- package/src/cli/cmd/agent.ts +165 -0
- package/src/cli/cmd/cmd.ts +5 -0
- package/src/cli/cmd/export.ts +88 -0
- package/src/cli/cmd/mcp.ts +80 -0
- package/src/cli/cmd/models.ts +58 -0
- package/src/cli/cmd/run.ts +359 -0
- package/src/cli/cmd/stats.ts +276 -0
- package/src/cli/error.ts +27 -0
- package/src/command/index.ts +73 -0
- package/src/command/template/initialize.txt +10 -0
- package/src/config/config.ts +705 -0
- package/src/config/markdown.ts +41 -0
- package/src/file/ripgrep.ts +391 -0
- package/src/file/time.ts +38 -0
- package/src/file/watcher.ts +75 -0
- package/src/file.ts +6 -0
- package/src/flag/flag.ts +19 -0
- package/src/format/formatter.ts +248 -0
- package/src/format/index.ts +137 -0
- package/src/global/index.ts +52 -0
- package/src/id/id.ts +72 -0
- package/src/index.js +371 -0
- package/src/mcp/index.ts +289 -0
- package/src/patch/index.ts +622 -0
- package/src/project/bootstrap.ts +22 -0
- package/src/project/instance.ts +67 -0
- package/src/project/project.ts +105 -0
- package/src/project/state.ts +65 -0
- package/src/provider/models-macro.ts +11 -0
- package/src/provider/models.ts +98 -0
- package/src/provider/opencode.js +47 -0
- package/src/provider/provider.ts +636 -0
- package/src/provider/transform.ts +241 -0
- package/src/server/project.ts +48 -0
- package/src/server/server.ts +249 -0
- package/src/session/agent.js +204 -0
- package/src/session/compaction.ts +249 -0
- package/src/session/index.ts +380 -0
- package/src/session/message-v2.ts +758 -0
- package/src/session/message.ts +189 -0
- package/src/session/processor.ts +356 -0
- package/src/session/prompt/anthropic-20250930.txt +166 -0
- package/src/session/prompt/anthropic.txt +105 -0
- package/src/session/prompt/anthropic_spoof.txt +1 -0
- package/src/session/prompt/beast.txt +147 -0
- package/src/session/prompt/build-switch.txt +5 -0
- package/src/session/prompt/codex.txt +318 -0
- package/src/session/prompt/copilot-gpt-5.txt +143 -0
- package/src/session/prompt/gemini.txt +155 -0
- package/src/session/prompt/grok-code.txt +1 -0
- package/src/session/prompt/plan.txt +8 -0
- package/src/session/prompt/polaris.txt +107 -0
- package/src/session/prompt/qwen.txt +109 -0
- package/src/session/prompt/summarize-turn.txt +5 -0
- package/src/session/prompt/summarize.txt +10 -0
- package/src/session/prompt/title.txt +25 -0
- package/src/session/prompt.ts +1390 -0
- package/src/session/retry.ts +53 -0
- package/src/session/revert.ts +108 -0
- package/src/session/status.ts +75 -0
- package/src/session/summary.ts +179 -0
- package/src/session/system.ts +138 -0
- package/src/session/todo.ts +36 -0
- package/src/snapshot/index.ts +197 -0
- package/src/storage/storage.ts +226 -0
- package/src/tool/bash.ts +193 -0
- package/src/tool/bash.txt +121 -0
- package/src/tool/batch.ts +173 -0
- package/src/tool/batch.txt +28 -0
- package/src/tool/codesearch.ts +123 -0
- package/src/tool/codesearch.txt +12 -0
- package/src/tool/edit.ts +604 -0
- package/src/tool/edit.txt +10 -0
- package/src/tool/glob.ts +65 -0
- package/src/tool/glob.txt +6 -0
- package/src/tool/grep.ts +116 -0
- package/src/tool/grep.txt +8 -0
- package/src/tool/invalid.ts +17 -0
- package/src/tool/ls.ts +110 -0
- package/src/tool/ls.txt +1 -0
- package/src/tool/multiedit.ts +46 -0
- package/src/tool/multiedit.txt +41 -0
- package/src/tool/patch.ts +188 -0
- package/src/tool/patch.txt +1 -0
- package/src/tool/read.ts +201 -0
- package/src/tool/read.txt +12 -0
- package/src/tool/registry.ts +87 -0
- package/src/tool/task.ts +126 -0
- package/src/tool/task.txt +60 -0
- package/src/tool/todo.ts +39 -0
- package/src/tool/todoread.txt +14 -0
- package/src/tool/todowrite.txt +167 -0
- package/src/tool/tool.ts +66 -0
- package/src/tool/webfetch.ts +171 -0
- package/src/tool/webfetch.txt +14 -0
- package/src/tool/websearch.ts +133 -0
- package/src/tool/websearch.txt +11 -0
- package/src/tool/write.ts +33 -0
- package/src/tool/write.txt +8 -0
- package/src/util/binary.ts +41 -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 +69 -0
- package/src/util/fn.ts +11 -0
- package/src/util/iife.ts +3 -0
- package/src/util/keybind.ts +79 -0
- package/src/util/lazy.ts +11 -0
- package/src/util/locale.ts +39 -0
- package/src/util/lock.ts +98 -0
- package/src/util/log.ts +177 -0
- package/src/util/queue.ts +19 -0
- package/src/util/rpc.ts +42 -0
- package/src/util/scrap.ts +10 -0
- package/src/util/signal.ts +12 -0
- package/src/util/timeout.ts +14 -0
- package/src/util/token.ts +7 -0
- package/src/util/wildcard.ts +54 -0
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
import { NamedError } from "../util/error"
|
|
2
|
+
import matter from "gray-matter"
|
|
3
|
+
import { z } from "zod"
|
|
4
|
+
|
|
5
|
+
export namespace ConfigMarkdown {
|
|
6
|
+
export const FILE_REGEX = /(?<![\w`])@(\.?[^\s`,.]*(?:\.[^\s`,.]+)*)/g
|
|
7
|
+
export const SHELL_REGEX = /!`([^`]+)`/g
|
|
8
|
+
|
|
9
|
+
export function files(template: string) {
|
|
10
|
+
return Array.from(template.matchAll(FILE_REGEX))
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export function shell(template: string) {
|
|
14
|
+
return Array.from(template.matchAll(SHELL_REGEX))
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
export async function parse(filePath: string) {
|
|
18
|
+
const template = await Bun.file(filePath).text()
|
|
19
|
+
|
|
20
|
+
try {
|
|
21
|
+
const md = matter(template)
|
|
22
|
+
return md
|
|
23
|
+
} catch (err) {
|
|
24
|
+
throw new FrontmatterError(
|
|
25
|
+
{
|
|
26
|
+
path: filePath,
|
|
27
|
+
message: `Failed to parse YAML frontmatter: ${err instanceof Error ? err.message : String(err)}`,
|
|
28
|
+
},
|
|
29
|
+
{ cause: err },
|
|
30
|
+
)
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
export const FrontmatterError = NamedError.create(
|
|
35
|
+
"ConfigFrontmatterError",
|
|
36
|
+
z.object({
|
|
37
|
+
path: z.string(),
|
|
38
|
+
message: z.string(),
|
|
39
|
+
}),
|
|
40
|
+
)
|
|
41
|
+
}
|
|
@@ -0,0 +1,391 @@
|
|
|
1
|
+
// Ripgrep utility functions
|
|
2
|
+
import path from "path"
|
|
3
|
+
import { Global } from "../global"
|
|
4
|
+
import fs from "fs/promises"
|
|
5
|
+
import z from "zod"
|
|
6
|
+
import { NamedError } from "../util/error"
|
|
7
|
+
import { lazy } from "../util/lazy"
|
|
8
|
+
import { $ } from "bun"
|
|
9
|
+
|
|
10
|
+
import { ZipReader, BlobReader, BlobWriter } from "@zip.js/zip.js"
|
|
11
|
+
import { Log } from "../util/log"
|
|
12
|
+
|
|
13
|
+
export namespace Ripgrep {
|
|
14
|
+
const log = Log.create({ service: "ripgrep" })
|
|
15
|
+
const Stats = z.object({
|
|
16
|
+
elapsed: z.object({
|
|
17
|
+
secs: z.number(),
|
|
18
|
+
nanos: z.number(),
|
|
19
|
+
human: z.string(),
|
|
20
|
+
}),
|
|
21
|
+
searches: z.number(),
|
|
22
|
+
searches_with_match: z.number(),
|
|
23
|
+
bytes_searched: z.number(),
|
|
24
|
+
bytes_printed: z.number(),
|
|
25
|
+
matched_lines: z.number(),
|
|
26
|
+
matches: z.number(),
|
|
27
|
+
})
|
|
28
|
+
|
|
29
|
+
const Begin = z.object({
|
|
30
|
+
type: z.literal("begin"),
|
|
31
|
+
data: z.object({
|
|
32
|
+
path: z.object({
|
|
33
|
+
text: z.string(),
|
|
34
|
+
}),
|
|
35
|
+
}),
|
|
36
|
+
})
|
|
37
|
+
|
|
38
|
+
export const Match = z.object({
|
|
39
|
+
type: z.literal("match"),
|
|
40
|
+
data: z.object({
|
|
41
|
+
path: z.object({
|
|
42
|
+
text: z.string(),
|
|
43
|
+
}),
|
|
44
|
+
lines: z.object({
|
|
45
|
+
text: z.string(),
|
|
46
|
+
}),
|
|
47
|
+
line_number: z.number(),
|
|
48
|
+
absolute_offset: z.number(),
|
|
49
|
+
submatches: z.array(
|
|
50
|
+
z.object({
|
|
51
|
+
match: z.object({
|
|
52
|
+
text: z.string(),
|
|
53
|
+
}),
|
|
54
|
+
start: z.number(),
|
|
55
|
+
end: z.number(),
|
|
56
|
+
}),
|
|
57
|
+
),
|
|
58
|
+
}),
|
|
59
|
+
})
|
|
60
|
+
|
|
61
|
+
const End = z.object({
|
|
62
|
+
type: z.literal("end"),
|
|
63
|
+
data: z.object({
|
|
64
|
+
path: z.object({
|
|
65
|
+
text: z.string(),
|
|
66
|
+
}),
|
|
67
|
+
binary_offset: z.number().nullable(),
|
|
68
|
+
stats: Stats,
|
|
69
|
+
}),
|
|
70
|
+
})
|
|
71
|
+
|
|
72
|
+
const Summary = z.object({
|
|
73
|
+
type: z.literal("summary"),
|
|
74
|
+
data: z.object({
|
|
75
|
+
elapsed_total: z.object({
|
|
76
|
+
human: z.string(),
|
|
77
|
+
nanos: z.number(),
|
|
78
|
+
secs: z.number(),
|
|
79
|
+
}),
|
|
80
|
+
stats: Stats,
|
|
81
|
+
}),
|
|
82
|
+
})
|
|
83
|
+
|
|
84
|
+
const Result = z.union([Begin, Match, End, Summary])
|
|
85
|
+
|
|
86
|
+
export type Result = z.infer<typeof Result>
|
|
87
|
+
export type Match = z.infer<typeof Match>
|
|
88
|
+
export type Begin = z.infer<typeof Begin>
|
|
89
|
+
export type End = z.infer<typeof End>
|
|
90
|
+
export type Summary = z.infer<typeof Summary>
|
|
91
|
+
const PLATFORM = {
|
|
92
|
+
"arm64-darwin": { platform: "aarch64-apple-darwin", extension: "tar.gz" },
|
|
93
|
+
"arm64-linux": {
|
|
94
|
+
platform: "aarch64-unknown-linux-gnu",
|
|
95
|
+
extension: "tar.gz",
|
|
96
|
+
},
|
|
97
|
+
"x64-darwin": { platform: "x86_64-apple-darwin", extension: "tar.gz" },
|
|
98
|
+
"x64-linux": { platform: "x86_64-unknown-linux-musl", extension: "tar.gz" },
|
|
99
|
+
"x64-win32": { platform: "x86_64-pc-windows-msvc", extension: "zip" },
|
|
100
|
+
} as const
|
|
101
|
+
|
|
102
|
+
export const ExtractionFailedError = NamedError.create(
|
|
103
|
+
"RipgrepExtractionFailedError",
|
|
104
|
+
z.object({
|
|
105
|
+
filepath: z.string(),
|
|
106
|
+
stderr: z.string(),
|
|
107
|
+
}),
|
|
108
|
+
)
|
|
109
|
+
|
|
110
|
+
export const UnsupportedPlatformError = NamedError.create(
|
|
111
|
+
"RipgrepUnsupportedPlatformError",
|
|
112
|
+
z.object({
|
|
113
|
+
platform: z.string(),
|
|
114
|
+
}),
|
|
115
|
+
)
|
|
116
|
+
|
|
117
|
+
export const DownloadFailedError = NamedError.create(
|
|
118
|
+
"RipgrepDownloadFailedError",
|
|
119
|
+
z.object({
|
|
120
|
+
url: z.string(),
|
|
121
|
+
status: z.number(),
|
|
122
|
+
}),
|
|
123
|
+
)
|
|
124
|
+
|
|
125
|
+
const state = lazy(async () => {
|
|
126
|
+
let filepath = Bun.which("rg")
|
|
127
|
+
if (filepath) return { filepath }
|
|
128
|
+
filepath = path.join(Global.Path.bin, "rg" + (process.platform === "win32" ? ".exe" : ""))
|
|
129
|
+
|
|
130
|
+
const file = Bun.file(filepath)
|
|
131
|
+
if (!(await file.exists())) {
|
|
132
|
+
const platformKey = `${process.arch}-${process.platform}` as keyof typeof PLATFORM
|
|
133
|
+
const config = PLATFORM[platformKey]
|
|
134
|
+
if (!config) throw new UnsupportedPlatformError({ platform: platformKey })
|
|
135
|
+
|
|
136
|
+
const version = "14.1.1"
|
|
137
|
+
const filename = `ripgrep-${version}-${config.platform}.${config.extension}`
|
|
138
|
+
const url = `https://github.com/BurntSushi/ripgrep/releases/download/${version}/${filename}`
|
|
139
|
+
|
|
140
|
+
const response = await fetch(url)
|
|
141
|
+
if (!response.ok) throw new DownloadFailedError({ url, status: response.status })
|
|
142
|
+
|
|
143
|
+
const buffer = await response.arrayBuffer()
|
|
144
|
+
const archivePath = path.join(Global.Path.bin, filename)
|
|
145
|
+
await Bun.write(archivePath, buffer)
|
|
146
|
+
if (config.extension === "tar.gz") {
|
|
147
|
+
const args = ["tar", "-xzf", archivePath, "--strip-components=1"]
|
|
148
|
+
|
|
149
|
+
if (platformKey.endsWith("-darwin")) args.push("--include=*/rg")
|
|
150
|
+
if (platformKey.endsWith("-linux")) args.push("--wildcards", "*/rg")
|
|
151
|
+
|
|
152
|
+
const proc = Bun.spawn(args, {
|
|
153
|
+
cwd: Global.Path.bin,
|
|
154
|
+
stderr: "pipe",
|
|
155
|
+
stdout: "pipe",
|
|
156
|
+
})
|
|
157
|
+
await proc.exited
|
|
158
|
+
if (proc.exitCode !== 0)
|
|
159
|
+
throw new ExtractionFailedError({
|
|
160
|
+
filepath,
|
|
161
|
+
stderr: await Bun.readableStreamToText(proc.stderr),
|
|
162
|
+
})
|
|
163
|
+
}
|
|
164
|
+
if (config.extension === "zip") {
|
|
165
|
+
if (config.extension === "zip") {
|
|
166
|
+
const zipFileReader = new ZipReader(new BlobReader(new Blob([await Bun.file(archivePath).arrayBuffer()])))
|
|
167
|
+
const entries = await zipFileReader.getEntries()
|
|
168
|
+
let rgEntry: any
|
|
169
|
+
for (const entry of entries) {
|
|
170
|
+
if (entry.filename.endsWith("rg.exe")) {
|
|
171
|
+
rgEntry = entry
|
|
172
|
+
break
|
|
173
|
+
}
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
if (!rgEntry) {
|
|
177
|
+
throw new ExtractionFailedError({
|
|
178
|
+
filepath: archivePath,
|
|
179
|
+
stderr: "rg.exe not found in zip archive",
|
|
180
|
+
})
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
const rgBlob = await rgEntry.getData(new BlobWriter())
|
|
184
|
+
if (!rgBlob) {
|
|
185
|
+
throw new ExtractionFailedError({
|
|
186
|
+
filepath: archivePath,
|
|
187
|
+
stderr: "Failed to extract rg.exe from zip archive",
|
|
188
|
+
})
|
|
189
|
+
}
|
|
190
|
+
await Bun.write(filepath, await rgBlob.arrayBuffer())
|
|
191
|
+
await zipFileReader.close()
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
await fs.unlink(archivePath)
|
|
195
|
+
if (!platformKey.endsWith("-win32")) await fs.chmod(filepath, 0o755)
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
return {
|
|
199
|
+
filepath,
|
|
200
|
+
}
|
|
201
|
+
})
|
|
202
|
+
|
|
203
|
+
export async function filepath() {
|
|
204
|
+
const { filepath } = await state()
|
|
205
|
+
return filepath
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
export async function* files(input: { cwd: string; glob?: string[] }) {
|
|
209
|
+
const args = [await filepath(), "--files", "--follow", "--hidden", "--glob=!.git/*"]
|
|
210
|
+
if (input.glob) {
|
|
211
|
+
for (const g of input.glob) {
|
|
212
|
+
args.push(`--glob=${g}`)
|
|
213
|
+
}
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
// Bun.spawn should throw this, but it incorrectly reports that the executable does not exist.
|
|
217
|
+
// See https://github.com/oven-sh/bun/issues/24012
|
|
218
|
+
if (!(await fs.stat(input.cwd).catch(() => undefined))?.isDirectory()) {
|
|
219
|
+
throw Object.assign(new Error(`No such file or directory: '${input.cwd}'`), {
|
|
220
|
+
code: "ENOENT",
|
|
221
|
+
errno: -2,
|
|
222
|
+
path: input.cwd,
|
|
223
|
+
})
|
|
224
|
+
}
|
|
225
|
+
|
|
226
|
+
const proc = Bun.spawn(args, {
|
|
227
|
+
cwd: input.cwd,
|
|
228
|
+
stdout: "pipe",
|
|
229
|
+
stderr: "ignore",
|
|
230
|
+
maxBuffer: 1024 * 1024 * 20,
|
|
231
|
+
})
|
|
232
|
+
|
|
233
|
+
const reader = proc.stdout.getReader()
|
|
234
|
+
const decoder = new TextDecoder()
|
|
235
|
+
let buffer = ""
|
|
236
|
+
|
|
237
|
+
try {
|
|
238
|
+
while (true) {
|
|
239
|
+
const { done, value } = await reader.read()
|
|
240
|
+
if (done) break
|
|
241
|
+
|
|
242
|
+
buffer += decoder.decode(value, { stream: true })
|
|
243
|
+
const lines = buffer.split("\n")
|
|
244
|
+
buffer = lines.pop() || ""
|
|
245
|
+
|
|
246
|
+
for (const line of lines) {
|
|
247
|
+
if (line) yield line
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
if (buffer) yield buffer
|
|
252
|
+
} finally {
|
|
253
|
+
reader.releaseLock()
|
|
254
|
+
await proc.exited
|
|
255
|
+
}
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
export async function tree(input: { cwd: string; limit?: number }) {
|
|
259
|
+
log.info("tree", input)
|
|
260
|
+
const files = await Array.fromAsync(Ripgrep.files({ cwd: input.cwd }))
|
|
261
|
+
interface Node {
|
|
262
|
+
path: string[]
|
|
263
|
+
children: Node[]
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
function getPath(node: Node, parts: string[], create: boolean) {
|
|
267
|
+
if (parts.length === 0) return node
|
|
268
|
+
let current = node
|
|
269
|
+
for (const part of parts) {
|
|
270
|
+
let existing = current.children.find((x) => x.path.at(-1) === part)
|
|
271
|
+
if (!existing) {
|
|
272
|
+
if (!create) return
|
|
273
|
+
existing = {
|
|
274
|
+
path: current.path.concat(part),
|
|
275
|
+
children: [],
|
|
276
|
+
}
|
|
277
|
+
current.children.push(existing)
|
|
278
|
+
}
|
|
279
|
+
current = existing
|
|
280
|
+
}
|
|
281
|
+
return current
|
|
282
|
+
}
|
|
283
|
+
|
|
284
|
+
const root: Node = {
|
|
285
|
+
path: [],
|
|
286
|
+
children: [],
|
|
287
|
+
}
|
|
288
|
+
for (const file of files) {
|
|
289
|
+
if (file.includes(".opencode")) continue
|
|
290
|
+
const parts = file.split(path.sep)
|
|
291
|
+
getPath(root, parts, true)
|
|
292
|
+
}
|
|
293
|
+
|
|
294
|
+
function sort(node: Node) {
|
|
295
|
+
node.children.sort((a, b) => {
|
|
296
|
+
if (!a.children.length && b.children.length) return 1
|
|
297
|
+
if (!b.children.length && a.children.length) return -1
|
|
298
|
+
return a.path.at(-1)!.localeCompare(b.path.at(-1)!)
|
|
299
|
+
})
|
|
300
|
+
for (const child of node.children) {
|
|
301
|
+
sort(child)
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
sort(root)
|
|
305
|
+
|
|
306
|
+
let current = [root]
|
|
307
|
+
const result: Node = {
|
|
308
|
+
path: [],
|
|
309
|
+
children: [],
|
|
310
|
+
}
|
|
311
|
+
|
|
312
|
+
let processed = 0
|
|
313
|
+
const limit = input.limit ?? 50
|
|
314
|
+
while (current.length > 0) {
|
|
315
|
+
const next = []
|
|
316
|
+
for (const node of current) {
|
|
317
|
+
if (node.children.length) next.push(...node.children)
|
|
318
|
+
}
|
|
319
|
+
const max = Math.max(...current.map((x) => x.children.length))
|
|
320
|
+
for (let i = 0; i < max && processed < limit; i++) {
|
|
321
|
+
for (const node of current) {
|
|
322
|
+
const child = node.children[i]
|
|
323
|
+
if (!child) continue
|
|
324
|
+
getPath(result, child.path, true)
|
|
325
|
+
processed++
|
|
326
|
+
if (processed >= limit) break
|
|
327
|
+
}
|
|
328
|
+
}
|
|
329
|
+
if (processed >= limit) {
|
|
330
|
+
for (const node of [...current, ...next]) {
|
|
331
|
+
const compare = getPath(result, node.path, false)
|
|
332
|
+
if (!compare) continue
|
|
333
|
+
if (compare?.children.length !== node.children.length) {
|
|
334
|
+
const diff = node.children.length - compare.children.length
|
|
335
|
+
compare.children.push({
|
|
336
|
+
path: compare.path.concat(`[${diff} truncated]`),
|
|
337
|
+
children: [],
|
|
338
|
+
})
|
|
339
|
+
}
|
|
340
|
+
}
|
|
341
|
+
break
|
|
342
|
+
}
|
|
343
|
+
current = next
|
|
344
|
+
}
|
|
345
|
+
|
|
346
|
+
const lines: string[] = []
|
|
347
|
+
|
|
348
|
+
function render(node: Node, depth: number) {
|
|
349
|
+
const indent = "\t".repeat(depth)
|
|
350
|
+
lines.push(indent + node.path.at(-1) + (node.children.length ? "/" : ""))
|
|
351
|
+
for (const child of node.children) {
|
|
352
|
+
render(child, depth + 1)
|
|
353
|
+
}
|
|
354
|
+
}
|
|
355
|
+
result.children.map((x) => render(x, 0))
|
|
356
|
+
|
|
357
|
+
return lines.join("\n")
|
|
358
|
+
}
|
|
359
|
+
|
|
360
|
+
export async function search(input: { cwd: string; pattern: string; glob?: string[]; limit?: number }) {
|
|
361
|
+
const args = [`${await filepath()}`, "--json", "--hidden", "--glob='!.git/*'"]
|
|
362
|
+
|
|
363
|
+
if (input.glob) {
|
|
364
|
+
for (const g of input.glob) {
|
|
365
|
+
args.push(`--glob=${g}`)
|
|
366
|
+
}
|
|
367
|
+
}
|
|
368
|
+
|
|
369
|
+
if (input.limit) {
|
|
370
|
+
args.push(`--max-count=${input.limit}`)
|
|
371
|
+
}
|
|
372
|
+
|
|
373
|
+
args.push("--")
|
|
374
|
+
args.push(input.pattern)
|
|
375
|
+
|
|
376
|
+
const command = args.join(" ")
|
|
377
|
+
const result = await $`${{ raw: command }}`.cwd(input.cwd).quiet().nothrow()
|
|
378
|
+
if (result.exitCode !== 0) {
|
|
379
|
+
return []
|
|
380
|
+
}
|
|
381
|
+
|
|
382
|
+
const lines = result.text().trim().split("\n").filter(Boolean)
|
|
383
|
+
// Parse JSON lines from ripgrep output
|
|
384
|
+
|
|
385
|
+
return lines
|
|
386
|
+
.map((line) => JSON.parse(line))
|
|
387
|
+
.map((parsed) => Result.parse(parsed))
|
|
388
|
+
.filter((r) => r.type === "match")
|
|
389
|
+
.map((r) => r.data)
|
|
390
|
+
}
|
|
391
|
+
}
|
package/src/file/time.ts
ADDED
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import { Instance } from "../project/instance"
|
|
2
|
+
import { Log } from "../util/log"
|
|
3
|
+
|
|
4
|
+
export namespace FileTime {
|
|
5
|
+
const log = Log.create({ service: "file.time" })
|
|
6
|
+
export const state = Instance.state(() => {
|
|
7
|
+
const read: {
|
|
8
|
+
[sessionID: string]: {
|
|
9
|
+
[path: string]: Date | undefined
|
|
10
|
+
}
|
|
11
|
+
} = {}
|
|
12
|
+
return {
|
|
13
|
+
read,
|
|
14
|
+
}
|
|
15
|
+
})
|
|
16
|
+
|
|
17
|
+
export function read(sessionID: string, file: string) {
|
|
18
|
+
log.info("read", { sessionID, file })
|
|
19
|
+
const { read } = state()
|
|
20
|
+
read[sessionID] = read[sessionID] || {}
|
|
21
|
+
read[sessionID][file] = new Date()
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
export function get(sessionID: string, file: string) {
|
|
25
|
+
return state().read[sessionID]?.[file]
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
export async function assert(sessionID: string, filepath: string) {
|
|
29
|
+
const time = get(sessionID, filepath)
|
|
30
|
+
if (!time) throw new Error(`You must read the file ${filepath} before overwriting it. Use the Read tool first`)
|
|
31
|
+
const stats = await Bun.file(filepath).stat()
|
|
32
|
+
if (stats.mtime.getTime() > time.getTime()) {
|
|
33
|
+
throw new Error(
|
|
34
|
+
`File ${filepath} has been modified since it was last read.\nLast modification: ${stats.mtime.toISOString()}\nLast read: ${time.toISOString()}\n\nPlease read the file again before modifying it.`,
|
|
35
|
+
)
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
}
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
import z from "zod"
|
|
2
|
+
import { Bus } from "../bus"
|
|
3
|
+
import { Flag } from "../flag/flag"
|
|
4
|
+
import { Instance } from "../project/instance"
|
|
5
|
+
import { Log } from "../util/log"
|
|
6
|
+
import { FileIgnore } from "./ignore"
|
|
7
|
+
import { Config } from "../config/config"
|
|
8
|
+
// @ts-ignore
|
|
9
|
+
import { createWrapper } from "@parcel/watcher/wrapper"
|
|
10
|
+
import { lazy } from "../util/lazy"
|
|
11
|
+
|
|
12
|
+
export namespace FileWatcher {
|
|
13
|
+
const log = Log.create({ service: "file.watcher" })
|
|
14
|
+
|
|
15
|
+
export const Event = {
|
|
16
|
+
Updated: Bus.event(
|
|
17
|
+
"file.watcher.updated",
|
|
18
|
+
z.object({
|
|
19
|
+
file: z.string(),
|
|
20
|
+
event: z.union([z.literal("add"), z.literal("change"), z.literal("unlink")]),
|
|
21
|
+
}),
|
|
22
|
+
),
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
const watcher = lazy(() => {
|
|
26
|
+
const binding = require(
|
|
27
|
+
`@parcel/watcher-${process.platform}-${process.arch}${process.platform === "linux" ? "-glibc" : ""}`,
|
|
28
|
+
)
|
|
29
|
+
return createWrapper(binding) as typeof import("@parcel/watcher")
|
|
30
|
+
})
|
|
31
|
+
|
|
32
|
+
const state = Instance.state(
|
|
33
|
+
async () => {
|
|
34
|
+
if (Instance.project.vcs !== "git") return {}
|
|
35
|
+
log.info("init")
|
|
36
|
+
const cfg = await Config.get()
|
|
37
|
+
const backend = (() => {
|
|
38
|
+
if (process.platform === "win32") return "windows"
|
|
39
|
+
if (process.platform === "darwin") return "fs-events"
|
|
40
|
+
if (process.platform === "linux") return "inotify"
|
|
41
|
+
})()
|
|
42
|
+
if (!backend) {
|
|
43
|
+
log.error("watcher backend not supported", { platform: process.platform })
|
|
44
|
+
return {}
|
|
45
|
+
}
|
|
46
|
+
log.info("watcher backend", { platform: process.platform, backend })
|
|
47
|
+
const sub = await watcher().subscribe(
|
|
48
|
+
Instance.directory,
|
|
49
|
+
(err, evts) => {
|
|
50
|
+
if (err) return
|
|
51
|
+
for (const evt of evts) {
|
|
52
|
+
log.info("event", evt)
|
|
53
|
+
if (evt.type === "create") Bus.publish(Event.Updated, { file: evt.path, event: "add" })
|
|
54
|
+
if (evt.type === "update") Bus.publish(Event.Updated, { file: evt.path, event: "change" })
|
|
55
|
+
if (evt.type === "delete") Bus.publish(Event.Updated, { file: evt.path, event: "unlink" })
|
|
56
|
+
}
|
|
57
|
+
},
|
|
58
|
+
{
|
|
59
|
+
ignore: [...FileIgnore.PATTERNS, ...(cfg.watcher?.ignore ?? [])],
|
|
60
|
+
backend,
|
|
61
|
+
},
|
|
62
|
+
)
|
|
63
|
+
return { sub }
|
|
64
|
+
},
|
|
65
|
+
async (state) => {
|
|
66
|
+
if (!state.sub) return
|
|
67
|
+
await state.sub?.unsubscribe()
|
|
68
|
+
},
|
|
69
|
+
)
|
|
70
|
+
|
|
71
|
+
export function init() {
|
|
72
|
+
if (!Flag.OPENCODE_EXPERIMENTAL_WATCHER) return
|
|
73
|
+
state()
|
|
74
|
+
}
|
|
75
|
+
}
|
package/src/file.ts
ADDED
package/src/flag/flag.ts
ADDED
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
export namespace Flag {
|
|
2
|
+
// OPENCODE_AUTO_SHARE removed - no sharing support
|
|
3
|
+
export const OPENCODE_CONFIG = process.env["OPENCODE_CONFIG"]
|
|
4
|
+
export const OPENCODE_CONFIG_DIR = process.env["OPENCODE_CONFIG_DIR"]
|
|
5
|
+
export const OPENCODE_CONFIG_CONTENT = process.env["OPENCODE_CONFIG_CONTENT"]
|
|
6
|
+
export const OPENCODE_DISABLE_AUTOUPDATE = truthy("OPENCODE_DISABLE_AUTOUPDATE")
|
|
7
|
+
export const OPENCODE_DISABLE_PRUNE = truthy("OPENCODE_DISABLE_PRUNE")
|
|
8
|
+
export const OPENCODE_ENABLE_EXPERIMENTAL_MODELS = truthy("OPENCODE_ENABLE_EXPERIMENTAL_MODELS")
|
|
9
|
+
export const OPENCODE_DISABLE_AUTOCOMPACT = truthy("OPENCODE_DISABLE_AUTOCOMPACT")
|
|
10
|
+
|
|
11
|
+
// Experimental
|
|
12
|
+
export const OPENCODE_EXPERIMENTAL = truthy("OPENCODE_EXPERIMENTAL")
|
|
13
|
+
export const OPENCODE_EXPERIMENTAL_WATCHER = OPENCODE_EXPERIMENTAL || truthy("OPENCODE_EXPERIMENTAL_WATCHER")
|
|
14
|
+
|
|
15
|
+
function truthy(key: string) {
|
|
16
|
+
const value = process.env[key]?.toLowerCase()
|
|
17
|
+
return value === "true" || value === "1"
|
|
18
|
+
}
|
|
19
|
+
}
|