saeeol 1.2.0 → 1.2.2
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/package.json +14 -14
- package/src/cli/cmd/tui/component/dialog/dialog-agent.tsx +32 -0
- package/src/cli/cmd/tui/component/dialog/dialog-command.tsx +190 -0
- package/src/cli/cmd/tui/component/dialog/dialog-console-org.tsx +103 -0
- package/src/cli/cmd/tui/component/dialog/dialog-go-upsell.tsx +159 -0
- package/src/cli/cmd/tui/component/dialog/dialog-mcp.tsx +86 -0
- package/src/cli/cmd/tui/component/dialog/dialog-model.tsx +238 -0
- package/src/cli/cmd/tui/component/dialog/dialog-provider.tsx +343 -0
- package/src/cli/cmd/tui/component/dialog/dialog-session-delete-failed.tsx +103 -0
- package/src/cli/cmd/tui/component/dialog/dialog-session-list.tsx +301 -0
- package/src/cli/cmd/tui/component/dialog/dialog-session-rename.tsx +35 -0
- package/src/cli/cmd/tui/component/dialog/dialog-skill.tsx +37 -0
- package/src/cli/cmd/tui/component/dialog/dialog-stash.tsx +87 -0
- package/src/cli/cmd/tui/component/dialog/dialog-status.tsx +190 -0
- package/src/cli/cmd/tui/component/dialog/dialog-tag.tsx +44 -0
- package/src/cli/cmd/tui/component/dialog/dialog-theme-list.tsx +50 -0
- package/src/cli/cmd/tui/component/dialog/dialog-variant.tsx +39 -0
- package/src/cli/cmd/tui/component/dialog/dialog-workspace-create.tsx +200 -0
- package/src/cli/cmd/tui/component/dialog/dialog-workspace-unavailable.tsx +81 -0
- package/src/cli/cmd/tui/component/dialog-agent.tsx +1 -32
- package/src/cli/cmd/tui/component/dialog-command.tsx +1 -190
- package/src/cli/cmd/tui/component/dialog-console-org.tsx +1 -103
- package/src/cli/cmd/tui/component/dialog-go-upsell.tsx +1 -159
- package/src/cli/cmd/tui/component/dialog-mcp.tsx +1 -86
- package/src/cli/cmd/tui/component/dialog-model.tsx +1 -238
- package/src/cli/cmd/tui/component/dialog-provider.tsx +1 -343
- package/src/cli/cmd/tui/component/dialog-session-delete-failed.tsx +1 -103
- package/src/cli/cmd/tui/component/dialog-session-list.tsx +1 -301
- package/src/cli/cmd/tui/component/dialog-session-rename.tsx +1 -35
- package/src/cli/cmd/tui/component/dialog-skill.tsx +1 -37
- package/src/cli/cmd/tui/component/dialog-stash.tsx +1 -87
- package/src/cli/cmd/tui/component/dialog-status.tsx +1 -190
- package/src/cli/cmd/tui/component/dialog-tag.tsx +1 -44
- package/src/cli/cmd/tui/component/dialog-theme-list.tsx +1 -50
- package/src/cli/cmd/tui/component/dialog-variant.tsx +1 -39
- package/src/cli/cmd/tui/component/dialog-workspace-create.tsx +1 -200
- package/src/cli/cmd/tui/component/dialog-workspace-unavailable.tsx +1 -81
- package/src/session/compaction-helpers.ts +1 -169
- package/src/session/compaction.ts +1 -712
- package/src/session/core/compaction/compaction-helpers.ts +169 -0
- package/src/session/core/compaction/compaction.ts +712 -0
- package/src/session/core/compaction/overflow.ts +28 -0
- package/src/session/core/instruction.ts +234 -0
- package/src/session/core/llm.ts +504 -0
- package/src/session/core/network.ts +392 -0
- package/src/session/core/processor.ts +731 -0
- package/src/session/core/projectors.ts +139 -0
- package/src/session/core/resolve-tools.ts +241 -0
- package/src/session/core/retry.ts +149 -0
- package/src/session/core/revert.ts +173 -0
- package/src/session/core/run-state.ts +110 -0
- package/src/session/core/schema.ts +35 -0
- package/src/session/core/session-types.ts +160 -0
- package/src/session/core/session.sql.ts +124 -0
- package/src/session/core/session.ts +948 -0
- package/src/session/core/shell-exec.ts +205 -0
- package/src/session/core/status.ts +100 -0
- package/src/session/core/subtask.ts +268 -0
- package/src/session/core/summary.ts +173 -0
- package/src/session/core/system.ts +114 -0
- package/src/session/core/todo.ts +86 -0
- package/src/session/core/user-part.ts +293 -0
- package/src/session/instruction.ts +1 -234
- package/src/session/llm.ts +1 -504
- package/src/session/message/message-errors.ts +83 -0
- package/src/session/message/message-parts.ts +89 -0
- package/src/session/message/message-query.ts +107 -0
- package/src/session/message/message-transform.ts +156 -0
- package/src/session/message/message-types.ts +68 -0
- package/src/session/message/message-v2.ts +73 -0
- package/src/session/message/message.ts +192 -0
- package/src/session/message-errors.ts +1 -83
- package/src/session/message-parts.ts +1 -89
- package/src/session/message-query.ts +1 -107
- package/src/session/message-transform.ts +1 -156
- package/src/session/message-types.ts +1 -68
- package/src/session/message-v2.ts +1 -73
- package/src/session/message.ts +1 -192
- package/src/session/network.ts +1 -392
- package/src/session/overflow.ts +1 -28
- package/src/session/processor.ts +1 -731
- package/src/session/projectors.ts +2 -139
- package/src/session/prompt/prompt-command.ts +93 -0
- package/src/session/prompt/prompt-loop.ts +299 -0
- package/src/session/prompt/prompt-model.ts +44 -0
- package/src/session/prompt/prompt-reminders.ts +120 -0
- package/src/session/prompt/prompt-resolve.ts +42 -0
- package/src/session/prompt/prompt-schemas.ts +128 -0
- package/src/session/prompt/prompt-title.ts +55 -0
- package/src/session/prompt/prompt-types.ts +47 -0
- package/src/session/prompt/prompt-user-msg.ts +80 -0
- package/src/session/prompt/prompt.ts +211 -0
- package/src/session/prompt-command.ts +1 -93
- package/src/session/prompt-loop.ts +1 -299
- package/src/session/prompt-model.ts +1 -44
- package/src/session/prompt-reminders.ts +1 -120
- package/src/session/prompt-resolve.ts +1 -42
- package/src/session/prompt-schemas.ts +1 -128
- package/src/session/prompt-title.ts +1 -55
- package/src/session/prompt-types.ts +1 -47
- package/src/session/prompt-user-msg.ts +1 -80
- package/src/session/prompt.ts +1 -211
- package/src/session/resolve-tools.ts +1 -241
- package/src/session/retry.ts +1 -149
- package/src/session/revert.ts +1 -173
- package/src/session/run-state.ts +1 -110
- package/src/session/schema.ts +1 -35
- package/src/session/session-types.ts +1 -160
- package/src/session/session.sql.ts +1 -124
- package/src/session/session.ts +1 -948
- package/src/session/shell-exec.ts +1 -205
- package/src/session/status.ts +1 -100
- package/src/session/subtask.ts +1 -268
- package/src/session/summary.ts +1 -173
- package/src/session/system.ts +1 -114
- package/src/session/todo.ts +1 -86
- package/src/session/user-part.ts +1 -293
- package/src/tool/apply_patch.ts +1 -334
- package/src/tool/bash.ts +1 -656
- package/src/tool/core/external-directory.ts +55 -0
- package/src/tool/core/invalid.ts +21 -0
- package/src/tool/core/recall.ts +164 -0
- package/src/tool/core/recall.txt +12 -0
- package/src/tool/core/schema.ts +16 -0
- package/src/tool/core/tool.ts +162 -0
- package/src/tool/core/truncate.ts +160 -0
- package/src/tool/core/truncation-dir.ts +4 -0
- package/src/tool/diagnostics.ts +1 -20
- package/src/tool/edit-replacers.ts +1 -288
- package/src/tool/edit-utils.ts +1 -86
- package/src/tool/edit.ts +1 -262
- package/src/tool/external-directory.ts +1 -55
- package/src/tool/file/apply_patch.ts +334 -0
- package/src/tool/file/apply_patch.txt +33 -0
- package/src/tool/file/bash.ts +656 -0
- package/src/tool/file/bash.txt +119 -0
- package/src/tool/file/edit-replacers.ts +288 -0
- package/src/tool/file/edit-utils.ts +86 -0
- package/src/tool/file/edit.ts +262 -0
- package/src/tool/file/edit.txt +10 -0
- package/src/tool/file/read.ts +389 -0
- package/src/tool/file/read.txt +14 -0
- package/src/tool/file/write.ts +114 -0
- package/src/tool/file/write.txt +8 -0
- package/src/tool/glob.ts +1 -115
- package/src/tool/grep.ts +1 -151
- package/src/tool/integration/diagnostics.ts +20 -0
- package/src/tool/integration/lsp.ts +113 -0
- package/src/tool/integration/lsp.txt +24 -0
- package/src/tool/integration/mcp-exa.ts +73 -0
- package/src/tool/integration/package.ts +168 -0
- package/src/tool/integration/registry.ts +375 -0
- package/src/tool/invalid.ts +1 -21
- package/src/tool/lsp.ts +1 -113
- package/src/tool/mcp-exa.ts +1 -73
- package/src/tool/package.ts +1 -168
- package/src/tool/plan.ts +1 -30
- package/src/tool/question.ts +1 -52
- package/src/tool/read.ts +1 -389
- package/src/tool/recall.ts +1 -164
- package/src/tool/registry.ts +1 -375
- package/src/tool/schema.ts +1 -16
- package/src/tool/search/glob.ts +115 -0
- package/src/tool/search/glob.txt +6 -0
- package/src/tool/search/grep.ts +151 -0
- package/src/tool/search/grep.txt +8 -0
- package/src/tool/search/warpgrep.ts +107 -0
- package/src/tool/search/warpgrep.txt +10 -0
- package/src/tool/search/webfetch.ts +202 -0
- package/src/tool/search/webfetch.txt +13 -0
- package/src/tool/search/websearch.ts +71 -0
- package/src/tool/search/websearch.txt +14 -0
- package/src/tool/skill.ts +1 -91
- package/src/tool/task.ts +1 -197
- package/src/tool/todo.ts +1 -62
- package/src/tool/tool.ts +1 -162
- package/src/tool/truncate.ts +1 -160
- package/src/tool/truncation-dir.ts +1 -4
- package/src/tool/warpgrep.ts +1 -107
- package/src/tool/webfetch.ts +1 -202
- package/src/tool/websearch.ts +1 -71
- package/src/tool/workflow/plan-enter.txt +14 -0
- package/src/tool/workflow/plan-exit.txt +13 -0
- package/src/tool/workflow/plan.ts +30 -0
- package/src/tool/workflow/question.ts +52 -0
- package/src/tool/workflow/question.txt +11 -0
- package/src/tool/workflow/skill.ts +91 -0
- package/src/tool/workflow/skill.txt +5 -0
- package/src/tool/workflow/task.ts +197 -0
- package/src/tool/workflow/task.txt +57 -0
- package/src/tool/workflow/todo.ts +62 -0
- package/src/tool/workflow/todowrite.txt +167 -0
- package/src/tool/write.ts +1 -114
|
@@ -0,0 +1,334 @@
|
|
|
1
|
+
import * as path from "path"
|
|
2
|
+
import { Effect, Schema } from "effect"
|
|
3
|
+
import * as Tool from "../core/tool"
|
|
4
|
+
import { Bus } from "../../bus"
|
|
5
|
+
import { FileWatcher } from "../../file/watcher"
|
|
6
|
+
import { InstanceState } from "@/effect/instance-state"
|
|
7
|
+
import { Patch } from "../../patch"
|
|
8
|
+
import { createTwoFilesPatch, diffLines } from "diff"
|
|
9
|
+
import { assertExternalDirectoryEffect } from "../core/external-directory"
|
|
10
|
+
import { trimDiff } from "./edit"
|
|
11
|
+
import { LSP } from "@/lsp/lsp"
|
|
12
|
+
import { AppFileSystem } from "@saeeol/core/filesystem"
|
|
13
|
+
import DESCRIPTION from "./apply_patch.txt"
|
|
14
|
+
import { File } from "../../file"
|
|
15
|
+
import { filterDiagnostics } from "../integration/diagnostics"
|
|
16
|
+
import { ConfigValidation } from "../../overlay/config-validation"
|
|
17
|
+
import * as EncodedIO from "../../overlay/tool/encoded-io"
|
|
18
|
+
import { Format } from "../../format"
|
|
19
|
+
import * as Bom from "@/util/bom"
|
|
20
|
+
|
|
21
|
+
export const Parameters = Schema.Struct({
|
|
22
|
+
patchText: Schema.String.annotate({ description: "The full patch text that describes all changes to be made" }),
|
|
23
|
+
})
|
|
24
|
+
|
|
25
|
+
export const ApplyPatchTool = Tool.define(
|
|
26
|
+
"apply_patch",
|
|
27
|
+
Effect.gen(function* () {
|
|
28
|
+
const lsp = yield* LSP.Service
|
|
29
|
+
const afs = yield* AppFileSystem.Service
|
|
30
|
+
const format = yield* Format.Service
|
|
31
|
+
const bus = yield* Bus.Service
|
|
32
|
+
|
|
33
|
+
const run = Effect.fn("ApplyPatchTool.execute")(function* (
|
|
34
|
+
params: Schema.Schema.Type<typeof Parameters>,
|
|
35
|
+
ctx: Tool.Context,
|
|
36
|
+
) {
|
|
37
|
+
if (!params.patchText) {
|
|
38
|
+
return yield* Effect.fail(new Error("patchText is required"))
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
// Parse the patch to get hunks
|
|
42
|
+
let hunks: Patch.Hunk[]
|
|
43
|
+
try {
|
|
44
|
+
const parseResult = Patch.parsePatch(params.patchText)
|
|
45
|
+
hunks = parseResult.hunks
|
|
46
|
+
} catch (error) {
|
|
47
|
+
return yield* Effect.fail(new Error(`apply_patch verification failed: ${error}`))
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
if (hunks.length === 0) {
|
|
51
|
+
const normalized = params.patchText.replace(/\r\n/g, "\n").replace(/\r/g, "\n").trim()
|
|
52
|
+
if (normalized === "*** Begin Patch\n*** End Patch") {
|
|
53
|
+
return yield* Effect.fail(new Error("patch rejected: empty patch"))
|
|
54
|
+
}
|
|
55
|
+
return yield* Effect.fail(new Error("apply_patch verification failed: no hunks found"))
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
const instance = yield* InstanceState.context
|
|
59
|
+
|
|
60
|
+
// Validate file paths and check permissions
|
|
61
|
+
const fileChanges: Array<{
|
|
62
|
+
filePath: string
|
|
63
|
+
oldContent: string
|
|
64
|
+
newContent: string
|
|
65
|
+
type: "add" | "update" | "delete" | "move"
|
|
66
|
+
movePath?: string
|
|
67
|
+
diff: string
|
|
68
|
+
additions: number
|
|
69
|
+
deletions: number
|
|
70
|
+
bom: boolean
|
|
71
|
+
encoding: string
|
|
72
|
+
}> = []
|
|
73
|
+
|
|
74
|
+
let totalDiff = ""
|
|
75
|
+
|
|
76
|
+
for (const hunk of hunks) {
|
|
77
|
+
const filePath = path.resolve(instance.directory, hunk.path)
|
|
78
|
+
yield* assertExternalDirectoryEffect(ctx, filePath)
|
|
79
|
+
|
|
80
|
+
switch (hunk.type) {
|
|
81
|
+
case "add": {
|
|
82
|
+
const oldContent = ""
|
|
83
|
+
const newContent =
|
|
84
|
+
hunk.contents.length === 0 || hunk.contents.endsWith("\n") ? hunk.contents : `${hunk.contents}\n`
|
|
85
|
+
const next = Bom.split(newContent)
|
|
86
|
+
const diff = trimDiff(createTwoFilesPatch(filePath, filePath, oldContent, next.text))
|
|
87
|
+
|
|
88
|
+
let additions = 0
|
|
89
|
+
let deletions = 0
|
|
90
|
+
for (const change of diffLines(oldContent, next.text)) {
|
|
91
|
+
if (change.added) additions += change.count || 0
|
|
92
|
+
if (change.removed) deletions += change.count || 0
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
fileChanges.push({
|
|
96
|
+
filePath,
|
|
97
|
+
oldContent,
|
|
98
|
+
newContent: next.text,
|
|
99
|
+
type: "add",
|
|
100
|
+
diff,
|
|
101
|
+
additions,
|
|
102
|
+
deletions,
|
|
103
|
+
bom: next.bom,
|
|
104
|
+
encoding: "utf-8",
|
|
105
|
+
})
|
|
106
|
+
|
|
107
|
+
totalDiff += diff + "\n"
|
|
108
|
+
break
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
case "update": {
|
|
112
|
+
// Check if file exists for update
|
|
113
|
+
const stats = yield* afs.stat(filePath).pipe(Effect.catch(() => Effect.succeed(undefined)))
|
|
114
|
+
if (!stats || stats.type === "Directory") {
|
|
115
|
+
return yield* Effect.fail(
|
|
116
|
+
new Error(`apply_patch verification failed: Failed to read file to update: ${filePath}`),
|
|
117
|
+
)
|
|
118
|
+
}
|
|
119
|
+
// mojibake; the resulting diff, additions/deletions counts, and permission-prompt
|
|
120
|
+
// metadata shown to the user must reflect the real file contents.
|
|
121
|
+
const read = yield* EncodedIO.read(filePath).pipe(
|
|
122
|
+
Effect.catch((error) =>
|
|
123
|
+
Effect.fail(
|
|
124
|
+
new Error(
|
|
125
|
+
`apply_patch verification failed: ${error instanceof Error ? error.message : String(error)}`,
|
|
126
|
+
),
|
|
127
|
+
),
|
|
128
|
+
),
|
|
129
|
+
)
|
|
130
|
+
const source = Bom.split(read.text)
|
|
131
|
+
const oldContent = source.text
|
|
132
|
+
let newContent = oldContent
|
|
133
|
+
let bom = source.bom
|
|
134
|
+
let encoding = read.encoding
|
|
135
|
+
|
|
136
|
+
// Apply the update chunks to get new content
|
|
137
|
+
try {
|
|
138
|
+
const fileUpdate = Patch.deriveNewContentsFromChunks(filePath, hunk.chunks)
|
|
139
|
+
newContent = fileUpdate.content
|
|
140
|
+
bom = fileUpdate.bom
|
|
141
|
+
encoding = fileUpdate.encoding
|
|
142
|
+
} catch (error) {
|
|
143
|
+
return yield* Effect.fail(new Error(`apply_patch verification failed: ${error}`))
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
const diff = trimDiff(createTwoFilesPatch(filePath, filePath, oldContent, newContent))
|
|
147
|
+
|
|
148
|
+
let additions = 0
|
|
149
|
+
let deletions = 0
|
|
150
|
+
for (const change of diffLines(oldContent, newContent)) {
|
|
151
|
+
if (change.added) additions += change.count || 0
|
|
152
|
+
if (change.removed) deletions += change.count || 0
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
const movePath = hunk.move_path ? path.resolve(instance.directory, hunk.move_path) : undefined
|
|
156
|
+
yield* assertExternalDirectoryEffect(ctx, movePath)
|
|
157
|
+
|
|
158
|
+
fileChanges.push({
|
|
159
|
+
filePath,
|
|
160
|
+
oldContent,
|
|
161
|
+
newContent,
|
|
162
|
+
type: hunk.move_path ? "move" : "update",
|
|
163
|
+
movePath,
|
|
164
|
+
diff,
|
|
165
|
+
additions,
|
|
166
|
+
deletions,
|
|
167
|
+
bom,
|
|
168
|
+
encoding,
|
|
169
|
+
})
|
|
170
|
+
|
|
171
|
+
totalDiff += diff + "\n"
|
|
172
|
+
break
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
case "delete": {
|
|
176
|
+
const deleteRead = yield* EncodedIO.read(filePath).pipe(
|
|
177
|
+
Effect.catch((error) =>
|
|
178
|
+
Effect.fail(
|
|
179
|
+
new Error(
|
|
180
|
+
`apply_patch verification failed: ${error instanceof Error ? error.message : String(error)}`,
|
|
181
|
+
),
|
|
182
|
+
),
|
|
183
|
+
),
|
|
184
|
+
)
|
|
185
|
+
const contentToDelete = deleteRead.text
|
|
186
|
+
const source = Bom.split(contentToDelete)
|
|
187
|
+
const deleteDiff = trimDiff(createTwoFilesPatch(filePath, filePath, contentToDelete, ""))
|
|
188
|
+
|
|
189
|
+
const deletions = contentToDelete.split("\n").length
|
|
190
|
+
|
|
191
|
+
fileChanges.push({
|
|
192
|
+
filePath,
|
|
193
|
+
oldContent: contentToDelete,
|
|
194
|
+
newContent: "",
|
|
195
|
+
type: "delete",
|
|
196
|
+
diff: deleteDiff,
|
|
197
|
+
additions: 0,
|
|
198
|
+
deletions,
|
|
199
|
+
bom: source.bom,
|
|
200
|
+
encoding: deleteRead.encoding,
|
|
201
|
+
})
|
|
202
|
+
|
|
203
|
+
totalDiff += deleteDiff + "\n"
|
|
204
|
+
break
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
// Build per-file metadata for UI rendering (used for both permission and result)
|
|
210
|
+
const files = fileChanges.map((change) => ({
|
|
211
|
+
filePath: change.filePath,
|
|
212
|
+
relativePath: path.relative(instance.worktree, change.movePath ?? change.filePath).replaceAll("\\", "/"),
|
|
213
|
+
type: change.type,
|
|
214
|
+
patch: change.diff,
|
|
215
|
+
additions: change.additions,
|
|
216
|
+
deletions: change.deletions,
|
|
217
|
+
movePath: change.movePath,
|
|
218
|
+
}))
|
|
219
|
+
|
|
220
|
+
// Check permissions if needed
|
|
221
|
+
const relativePaths = fileChanges.map((c) => path.relative(instance.worktree, c.filePath).replaceAll("\\", "/"))
|
|
222
|
+
yield* ctx.ask({
|
|
223
|
+
permission: "edit",
|
|
224
|
+
patterns: relativePaths,
|
|
225
|
+
always: ["*"],
|
|
226
|
+
metadata: {
|
|
227
|
+
filepath: relativePaths.join(", "),
|
|
228
|
+
diff: totalDiff,
|
|
229
|
+
files,
|
|
230
|
+
},
|
|
231
|
+
})
|
|
232
|
+
|
|
233
|
+
// Apply the changes
|
|
234
|
+
const updates: Array<{ file: string; event: "add" | "change" | "unlink" }> = []
|
|
235
|
+
|
|
236
|
+
for (const change of fileChanges) {
|
|
237
|
+
const edited = change.type === "delete" ? undefined : (change.movePath ?? change.filePath)
|
|
238
|
+
switch (change.type) {
|
|
239
|
+
case "add":
|
|
240
|
+
// Create parent directories (recursive: true is safe on existing/root dirs)
|
|
241
|
+
yield* EncodedIO.write(change.filePath, Bom.join(change.newContent, change.bom), change.encoding)
|
|
242
|
+
updates.push({ file: change.filePath, event: "add" })
|
|
243
|
+
break
|
|
244
|
+
|
|
245
|
+
case "update":
|
|
246
|
+
yield* EncodedIO.write(change.filePath, Bom.join(change.newContent, change.bom), change.encoding)
|
|
247
|
+
updates.push({ file: change.filePath, event: "change" })
|
|
248
|
+
break
|
|
249
|
+
|
|
250
|
+
case "move":
|
|
251
|
+
if (change.movePath) {
|
|
252
|
+
// Create parent directories (recursive: true is safe on existing/root dirs)
|
|
253
|
+
yield* EncodedIO.write(change.movePath!, Bom.join(change.newContent, change.bom), change.encoding)
|
|
254
|
+
yield* afs.remove(change.filePath)
|
|
255
|
+
updates.push({ file: change.filePath, event: "unlink" })
|
|
256
|
+
updates.push({ file: change.movePath, event: "add" })
|
|
257
|
+
}
|
|
258
|
+
break
|
|
259
|
+
|
|
260
|
+
case "delete":
|
|
261
|
+
yield* afs.remove(change.filePath)
|
|
262
|
+
updates.push({ file: change.filePath, event: "unlink" })
|
|
263
|
+
break
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
if (edited) {
|
|
267
|
+
if (yield* format.file(edited)) {
|
|
268
|
+
yield* Bom.syncFile(afs, edited, change.bom)
|
|
269
|
+
}
|
|
270
|
+
yield* bus.publish(File.Event.Edited, { file: edited })
|
|
271
|
+
}
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
// Publish file change events
|
|
275
|
+
for (const update of updates) {
|
|
276
|
+
yield* bus.publish(FileWatcher.Event.Updated, update)
|
|
277
|
+
}
|
|
278
|
+
|
|
279
|
+
// Notify LSP of file changes and collect diagnostics
|
|
280
|
+
for (const change of fileChanges) {
|
|
281
|
+
if (change.type === "delete") continue
|
|
282
|
+
const target = change.movePath ?? change.filePath
|
|
283
|
+
yield* lsp.touchFile(target, "document")
|
|
284
|
+
}
|
|
285
|
+
const diagnostics = yield* lsp.diagnostics()
|
|
286
|
+
|
|
287
|
+
// Generate output summary
|
|
288
|
+
const summaryLines = fileChanges.map((change) => {
|
|
289
|
+
if (change.type === "add") {
|
|
290
|
+
return `A ${path.relative(instance.worktree, change.filePath).replaceAll("\\", "/")}`
|
|
291
|
+
}
|
|
292
|
+
if (change.type === "delete") {
|
|
293
|
+
return `D ${path.relative(instance.worktree, change.filePath).replaceAll("\\", "/")}`
|
|
294
|
+
}
|
|
295
|
+
const target = change.movePath ?? change.filePath
|
|
296
|
+
return `M ${path.relative(instance.worktree, target).replaceAll("\\", "/")}`
|
|
297
|
+
})
|
|
298
|
+
let output = `Success. Updated the following files:\n${summaryLines.join("\n")}`
|
|
299
|
+
const changedPaths = fileChanges
|
|
300
|
+
.filter((c) => c.type !== "delete")
|
|
301
|
+
.map((c) => AppFileSystem.normalizePath(c.movePath ?? c.filePath))
|
|
302
|
+
|
|
303
|
+
for (const change of fileChanges) {
|
|
304
|
+
if (change.type === "delete") continue
|
|
305
|
+
const target = change.movePath ?? change.filePath
|
|
306
|
+
const block = LSP.Diagnostic.report(target, diagnostics[AppFileSystem.normalizePath(target)] ?? [])
|
|
307
|
+
if (!block) continue
|
|
308
|
+
const rel = path.relative(instance.worktree, target).replaceAll("\\", "/")
|
|
309
|
+
output += `\n\nLSP errors detected in ${rel}, please fix:\n${block}`
|
|
310
|
+
}
|
|
311
|
+
for (const changed of fileChanges) {
|
|
312
|
+
if (changed.type === "delete") continue
|
|
313
|
+
output += yield* Effect.promise(() => ConfigValidation.check(changed.movePath ?? changed.filePath))
|
|
314
|
+
}
|
|
315
|
+
|
|
316
|
+
return {
|
|
317
|
+
title: output,
|
|
318
|
+
metadata: {
|
|
319
|
+
diff: totalDiff,
|
|
320
|
+
files,
|
|
321
|
+
diagnostics: filterDiagnostics(diagnostics, changedPaths),
|
|
322
|
+
},
|
|
323
|
+
output,
|
|
324
|
+
}
|
|
325
|
+
})
|
|
326
|
+
|
|
327
|
+
return {
|
|
328
|
+
description: DESCRIPTION,
|
|
329
|
+
parameters: Parameters,
|
|
330
|
+
execute: (params: Schema.Schema.Type<typeof Parameters>, ctx: Tool.Context) =>
|
|
331
|
+
run(params, ctx).pipe(Effect.orDie),
|
|
332
|
+
}
|
|
333
|
+
}),
|
|
334
|
+
)
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
Use the `apply_patch` tool to edit files. Your patch language is a stripped‑down, file‑oriented diff format designed to be easy to parse and safe to apply. You can think of it as a high‑level envelope:
|
|
2
|
+
|
|
3
|
+
*** Begin Patch
|
|
4
|
+
[ one or more file sections ]
|
|
5
|
+
*** End Patch
|
|
6
|
+
|
|
7
|
+
Within that envelope, you get a sequence of file operations.
|
|
8
|
+
You MUST include a header to specify the action you are taking.
|
|
9
|
+
Each operation starts with one of three headers:
|
|
10
|
+
|
|
11
|
+
*** Add File: <path> - create a new file. Every following line is a + line (the initial contents).
|
|
12
|
+
*** Delete File: <path> - remove an existing file. Nothing follows.
|
|
13
|
+
*** Update File: <path> - patch an existing file in place (optionally with a rename).
|
|
14
|
+
|
|
15
|
+
Example patch:
|
|
16
|
+
|
|
17
|
+
```
|
|
18
|
+
*** Begin Patch
|
|
19
|
+
*** Add File: hello.txt
|
|
20
|
+
+Hello world
|
|
21
|
+
*** Update File: src/app.py
|
|
22
|
+
*** Move to: src/main.py
|
|
23
|
+
@@ def greet():
|
|
24
|
+
-print("Hi")
|
|
25
|
+
+print("Hello, world!")
|
|
26
|
+
*** Delete File: obsolete.txt
|
|
27
|
+
*** End Patch
|
|
28
|
+
```
|
|
29
|
+
|
|
30
|
+
It is important to remember:
|
|
31
|
+
|
|
32
|
+
- You must include a header with your intended action (Add/Delete/Update)
|
|
33
|
+
- You must prefix new lines with `+` even when creating a new file
|