mirror of
https://github.com/anomalyco/opencode.git
synced 2026-05-15 09:02:35 +00:00
534 lines
17 KiB
TypeScript
534 lines
17 KiB
TypeScript
import { afterEach, describe, expect } from "bun:test"
|
|
import path from "path"
|
|
import fs from "fs/promises"
|
|
import { Cause, Deferred, Effect, Exit, Fiber, Layer } from "effect"
|
|
import { EditTool } from "../../src/tool/edit"
|
|
import { disposeAllInstances, TestInstance } from "../fixture/fixture"
|
|
import { LSP } from "@/lsp/lsp"
|
|
import { AppFileSystem } from "@opencode-ai/core/filesystem"
|
|
import { Format } from "../../src/format"
|
|
import { Agent } from "../../src/agent/agent"
|
|
import { Bus } from "../../src/bus"
|
|
import { Truncate } from "@/tool/truncate"
|
|
import { SessionID, MessageID } from "../../src/session/schema"
|
|
import * as Tool from "../../src/tool/tool"
|
|
import { testEffect } from "../lib/effect"
|
|
import { FileWatcher } from "../../src/file/watcher"
|
|
|
|
const ctx = {
|
|
sessionID: SessionID.make("ses_test-edit-session"),
|
|
messageID: MessageID.make("msg_test"),
|
|
callID: "",
|
|
agent: "build",
|
|
abort: AbortSignal.any([]),
|
|
messages: [],
|
|
metadata: () => Effect.void,
|
|
ask: () => Effect.void,
|
|
}
|
|
|
|
afterEach(async () => {
|
|
await disposeAllInstances()
|
|
})
|
|
|
|
const layer = Layer.mergeAll(
|
|
LSP.defaultLayer,
|
|
AppFileSystem.defaultLayer,
|
|
Format.defaultLayer,
|
|
Bus.layer,
|
|
Truncate.defaultLayer,
|
|
Agent.defaultLayer,
|
|
)
|
|
|
|
const it = testEffect(layer)
|
|
|
|
const init = Effect.fn("EditToolTest.init")(function* () {
|
|
const info = yield* EditTool
|
|
return yield* info.init()
|
|
})
|
|
|
|
const run = Effect.fn("EditToolTest.run")(function* (
|
|
args: Tool.InferParameters<typeof EditTool>,
|
|
next: Tool.Context = ctx,
|
|
) {
|
|
const tool = yield* init()
|
|
return yield* tool.execute(args, next)
|
|
})
|
|
|
|
const fail = Effect.fn("EditToolTest.fail")(function* (args: Tool.InferParameters<typeof EditTool>) {
|
|
const exit = yield* run(args).pipe(Effect.exit)
|
|
if (Exit.isFailure(exit)) {
|
|
const err = Cause.squash(exit.cause)
|
|
return err instanceof Error ? err : new Error(String(err))
|
|
}
|
|
throw new Error("expected edit to fail")
|
|
})
|
|
|
|
const put = Effect.fn("EditToolTest.put")(function* (p: string, content: string) {
|
|
const fs = yield* AppFileSystem.Service
|
|
yield* fs.writeWithDirs(p, content)
|
|
})
|
|
|
|
const load = Effect.fn("EditToolTest.load")(function* (p: string) {
|
|
const fs = yield* AppFileSystem.Service
|
|
return yield* fs.readFileString(p)
|
|
})
|
|
|
|
const loadRaw = Effect.fn("EditToolTest.loadRaw")(function* (p: string) {
|
|
return yield* Effect.promise(() => fs.readFile(p, "utf-8"))
|
|
})
|
|
|
|
const makeDirectory = Effect.fn("EditToolTest.makeDirectory")(function* (p: string) {
|
|
const fs = yield* AppFileSystem.Service
|
|
yield* fs.makeDirectory(p)
|
|
})
|
|
|
|
const onceBus = Effect.fn("EditToolTest.onceBus")(function* (def: typeof FileWatcher.Event.Updated) {
|
|
const bus = yield* Bus.Service
|
|
const deferred = yield* Deferred.make<void>()
|
|
const unsub = yield* bus.subscribeCallback(def, () => Effect.runSync(Deferred.succeed(deferred, undefined)))
|
|
yield* Effect.addFinalizer(() => Effect.sync(unsub))
|
|
return deferred
|
|
})
|
|
|
|
describe("tool.edit", () => {
|
|
describe("creating new files", () => {
|
|
it.instance("creates new file when oldString is empty", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "newfile.txt")
|
|
const result = yield* run({ filePath: filepath, oldString: "", newString: "new content" })
|
|
|
|
expect(result.metadata.diff).toContain("new content")
|
|
expect(yield* load(filepath)).toBe("new content")
|
|
}),
|
|
)
|
|
|
|
it.instance("preserves BOM when oldString is empty on existing files", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "existing.cs")
|
|
const bom = String.fromCharCode(0xfeff)
|
|
yield* put(filepath, `${bom}using System;\n`)
|
|
|
|
const result = yield* run({ filePath: filepath, oldString: "", newString: "using Up;\n" })
|
|
|
|
expect(result.metadata.diff).toContain("-using System;")
|
|
expect(result.metadata.diff).toContain("+using Up;")
|
|
|
|
const content = yield* loadRaw(filepath)
|
|
expect(content.charCodeAt(0)).toBe(0xfeff)
|
|
expect(content.slice(1)).toBe("using Up;\n")
|
|
}),
|
|
)
|
|
|
|
it.instance("creates new file with nested directories", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "nested", "dir", "file.txt")
|
|
|
|
yield* run({ filePath: filepath, oldString: "", newString: "nested file" })
|
|
|
|
expect(yield* load(filepath)).toBe("nested file")
|
|
}),
|
|
)
|
|
|
|
it.instance("emits add event for new files", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const updated = yield* onceBus(FileWatcher.Event.Updated)
|
|
|
|
yield* run({ filePath: path.join(test.directory, "new.txt"), oldString: "", newString: "content" })
|
|
yield* Deferred.await(updated)
|
|
}),
|
|
)
|
|
})
|
|
|
|
describe("editing existing files", () => {
|
|
it.instance("replaces text in existing file", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "existing.txt")
|
|
yield* put(filepath, "old content here")
|
|
|
|
const result = yield* run({ filePath: filepath, oldString: "old content", newString: "new content" })
|
|
|
|
expect(result.output).toContain("Edit applied successfully")
|
|
expect(yield* load(filepath)).toBe("new content here")
|
|
}),
|
|
)
|
|
|
|
it.instance("replaces the first visible line in BOM files", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "existing.cs")
|
|
const bom = String.fromCharCode(0xfeff)
|
|
yield* put(filepath, `${bom}using System;\nclass Test {}\n`)
|
|
|
|
const result = yield* run({ filePath: filepath, oldString: "using System;", newString: "using Up;" })
|
|
|
|
expect(result.metadata.diff).toContain("-using System;")
|
|
expect(result.metadata.diff).toContain("+using Up;")
|
|
expect(result.metadata.diff).not.toContain(bom)
|
|
|
|
const content = yield* loadRaw(filepath)
|
|
expect(content.charCodeAt(0)).toBe(0xfeff)
|
|
expect(content.slice(1)).toBe("using Up;\nclass Test {}\n")
|
|
}),
|
|
)
|
|
|
|
it.instance("throws error when file does not exist", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
expect(
|
|
(yield* fail({ filePath: path.join(test.directory, "nonexistent.txt"), oldString: "old", newString: "new" }))
|
|
.message,
|
|
).toContain("not found")
|
|
}),
|
|
)
|
|
|
|
it.instance("throws error when oldString equals newString", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "file.txt")
|
|
yield* put(filepath, "content")
|
|
|
|
expect((yield* fail({ filePath: filepath, oldString: "same", newString: "same" })).message).toContain(
|
|
"identical",
|
|
)
|
|
}),
|
|
)
|
|
|
|
it.instance("throws error when oldString not found in file", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "file.txt")
|
|
yield* put(filepath, "actual content")
|
|
|
|
expect(yield* fail({ filePath: filepath, oldString: "not in file", newString: "replacement" })).toBeInstanceOf(
|
|
Error,
|
|
)
|
|
}),
|
|
)
|
|
|
|
it.instance("replaces all occurrences with replaceAll option", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "file.txt")
|
|
yield* put(filepath, "foo bar foo baz foo")
|
|
|
|
yield* run({ filePath: filepath, oldString: "foo", newString: "qux", replaceAll: true })
|
|
|
|
expect(yield* load(filepath)).toBe("qux bar qux baz qux")
|
|
}),
|
|
)
|
|
|
|
it.instance("emits change event for existing files", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "file.txt")
|
|
yield* put(filepath, "original")
|
|
const updated = yield* onceBus(FileWatcher.Event.Updated)
|
|
|
|
yield* run({ filePath: filepath, oldString: "original", newString: "modified" })
|
|
yield* Deferred.await(updated)
|
|
}),
|
|
)
|
|
})
|
|
|
|
describe("edge cases", () => {
|
|
it.instance("handles multiline replacements", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "file.txt")
|
|
yield* put(filepath, "line1\nline2\nline3")
|
|
|
|
yield* run({ filePath: filepath, oldString: "line2", newString: "new line 2\nextra line" })
|
|
|
|
expect(yield* load(filepath)).toBe("line1\nnew line 2\nextra line\nline3")
|
|
}),
|
|
)
|
|
|
|
it.instance("handles CRLF line endings", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "file.txt")
|
|
yield* put(filepath, "line1\r\nold\r\nline3")
|
|
|
|
yield* run({ filePath: filepath, oldString: "old", newString: "new" })
|
|
|
|
expect(yield* load(filepath)).toBe("line1\r\nnew\r\nline3")
|
|
}),
|
|
)
|
|
|
|
it.instance("throws error when oldString equals newString", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "file.txt")
|
|
yield* put(filepath, "content")
|
|
|
|
expect((yield* fail({ filePath: filepath, oldString: "", newString: "" })).message).toContain("identical")
|
|
}),
|
|
)
|
|
|
|
it.instance("throws error when path is directory", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const dirpath = path.join(test.directory, "adir")
|
|
yield* makeDirectory(dirpath)
|
|
|
|
expect((yield* fail({ filePath: dirpath, oldString: "old", newString: "new" })).message).toContain("directory")
|
|
}),
|
|
)
|
|
|
|
it.instance("tracks file diff statistics", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "file.txt")
|
|
yield* put(filepath, "line1\nline2\nline3")
|
|
|
|
const result = yield* run({ filePath: filepath, oldString: "line2", newString: "new line a\nnew line b" })
|
|
|
|
expect(result.metadata.filediff).toBeDefined()
|
|
expect(result.metadata.filediff.file).toBe(filepath)
|
|
expect(result.metadata.filediff.additions).toBeGreaterThan(0)
|
|
}),
|
|
)
|
|
})
|
|
|
|
describe("line endings", () => {
|
|
const old = "alpha\nbeta\ngamma"
|
|
const next = "alpha\nbeta-updated\ngamma"
|
|
const alt = "alpha\nbeta\nomega"
|
|
|
|
const normalize = (text: string, ending: "\n" | "\r\n") => {
|
|
const normalized = text.replaceAll("\r\n", "\n")
|
|
if (ending === "\n") return normalized
|
|
return normalized.replaceAll("\n", "\r\n")
|
|
}
|
|
|
|
const count = (content: string) => {
|
|
const crlf = content.match(/\r\n/g)?.length ?? 0
|
|
const lf = content.match(/\n/g)?.length ?? 0
|
|
return {
|
|
crlf,
|
|
lf: lf - crlf,
|
|
}
|
|
}
|
|
|
|
const expectLf = (content: string) => {
|
|
const counts = count(content)
|
|
expect(counts.crlf).toBe(0)
|
|
expect(counts.lf).toBeGreaterThan(0)
|
|
}
|
|
|
|
const expectCrlf = (content: string) => {
|
|
const counts = count(content)
|
|
expect(counts.lf).toBe(0)
|
|
expect(counts.crlf).toBeGreaterThan(0)
|
|
}
|
|
|
|
type Input = {
|
|
content: string
|
|
oldString: string
|
|
newString: string
|
|
replaceAll?: boolean
|
|
}
|
|
|
|
const apply = Effect.fn("EditToolTest.lineEndings.apply")(function* (input: Input) {
|
|
const test = yield* TestInstance
|
|
const filePath = path.join(test.directory, "test.txt")
|
|
yield* put(filePath, input.content)
|
|
yield* run({
|
|
filePath,
|
|
oldString: input.oldString,
|
|
newString: input.newString,
|
|
replaceAll: input.replaceAll,
|
|
})
|
|
return yield* load(filePath)
|
|
})
|
|
|
|
it.instance("preserves LF with LF multi-line strings", () =>
|
|
Effect.gen(function* () {
|
|
const content = normalize(old + "\n", "\n")
|
|
const output = yield* apply({
|
|
content,
|
|
oldString: normalize(old, "\n"),
|
|
newString: normalize(next, "\n"),
|
|
})
|
|
expect(output).toBe(normalize(next + "\n", "\n"))
|
|
expectLf(output)
|
|
}),
|
|
)
|
|
|
|
it.instance("preserves CRLF with CRLF multi-line strings", () =>
|
|
Effect.gen(function* () {
|
|
const content = normalize(old + "\n", "\r\n")
|
|
const output = yield* apply({
|
|
content,
|
|
oldString: normalize(old, "\r\n"),
|
|
newString: normalize(next, "\r\n"),
|
|
})
|
|
expect(output).toBe(normalize(next + "\n", "\r\n"))
|
|
expectCrlf(output)
|
|
}),
|
|
)
|
|
|
|
it.instance("preserves LF when old/new use CRLF", () =>
|
|
Effect.gen(function* () {
|
|
const content = normalize(old + "\n", "\n")
|
|
const output = yield* apply({
|
|
content,
|
|
oldString: normalize(old, "\r\n"),
|
|
newString: normalize(next, "\r\n"),
|
|
})
|
|
expect(output).toBe(normalize(next + "\n", "\n"))
|
|
expectLf(output)
|
|
}),
|
|
)
|
|
|
|
it.instance("preserves CRLF when old/new use LF", () =>
|
|
Effect.gen(function* () {
|
|
const content = normalize(old + "\n", "\r\n")
|
|
const output = yield* apply({
|
|
content,
|
|
oldString: normalize(old, "\n"),
|
|
newString: normalize(next, "\n"),
|
|
})
|
|
expect(output).toBe(normalize(next + "\n", "\r\n"))
|
|
expectCrlf(output)
|
|
}),
|
|
)
|
|
|
|
it.instance("preserves LF when newString uses CRLF", () =>
|
|
Effect.gen(function* () {
|
|
const content = normalize(old + "\n", "\n")
|
|
const output = yield* apply({
|
|
content,
|
|
oldString: normalize(old, "\n"),
|
|
newString: normalize(next, "\r\n"),
|
|
})
|
|
expect(output).toBe(normalize(next + "\n", "\n"))
|
|
expectLf(output)
|
|
}),
|
|
)
|
|
|
|
it.instance("preserves CRLF when newString uses LF", () =>
|
|
Effect.gen(function* () {
|
|
const content = normalize(old + "\n", "\r\n")
|
|
const output = yield* apply({
|
|
content,
|
|
oldString: normalize(old, "\r\n"),
|
|
newString: normalize(next, "\n"),
|
|
})
|
|
expect(output).toBe(normalize(next + "\n", "\r\n"))
|
|
expectCrlf(output)
|
|
}),
|
|
)
|
|
|
|
it.instance("preserves LF with mixed old/new line endings", () =>
|
|
Effect.gen(function* () {
|
|
const content = normalize(old + "\n", "\n")
|
|
const output = yield* apply({
|
|
content,
|
|
oldString: "alpha\nbeta\r\ngamma",
|
|
newString: "alpha\r\nbeta\nomega",
|
|
})
|
|
expect(output).toBe(normalize(alt + "\n", "\n"))
|
|
expectLf(output)
|
|
}),
|
|
)
|
|
|
|
it.instance("preserves CRLF with mixed old/new line endings", () =>
|
|
Effect.gen(function* () {
|
|
const content = normalize(old + "\n", "\r\n")
|
|
const output = yield* apply({
|
|
content,
|
|
oldString: "alpha\r\nbeta\ngamma",
|
|
newString: "alpha\nbeta\r\nomega",
|
|
})
|
|
expect(output).toBe(normalize(alt + "\n", "\r\n"))
|
|
expectCrlf(output)
|
|
}),
|
|
)
|
|
|
|
it.instance("replaceAll preserves LF for multi-line blocks", () =>
|
|
Effect.gen(function* () {
|
|
const blockOld = "alpha\nbeta"
|
|
const blockNew = "alpha\nbeta-updated"
|
|
const content = normalize(blockOld + "\n" + blockOld + "\n", "\n")
|
|
const output = yield* apply({
|
|
content,
|
|
oldString: normalize(blockOld, "\n"),
|
|
newString: normalize(blockNew, "\n"),
|
|
replaceAll: true,
|
|
})
|
|
expect(output).toBe(normalize(blockNew + "\n" + blockNew + "\n", "\n"))
|
|
expectLf(output)
|
|
}),
|
|
)
|
|
|
|
it.instance("replaceAll preserves CRLF for multi-line blocks", () =>
|
|
Effect.gen(function* () {
|
|
const blockOld = "alpha\nbeta"
|
|
const blockNew = "alpha\nbeta-updated"
|
|
const content = normalize(blockOld + "\n" + blockOld + "\n", "\r\n")
|
|
const output = yield* apply({
|
|
content,
|
|
oldString: normalize(blockOld, "\r\n"),
|
|
newString: normalize(blockNew, "\r\n"),
|
|
replaceAll: true,
|
|
})
|
|
expect(output).toBe(normalize(blockNew + "\n" + blockNew + "\n", "\r\n"))
|
|
expectCrlf(output)
|
|
}),
|
|
)
|
|
})
|
|
|
|
describe("concurrent editing", () => {
|
|
it.instance("preserves concurrent edits to different sections of the same file", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "file.txt")
|
|
yield* put(filepath, "top = 0\nmiddle = keep\nbottom = 0\n")
|
|
|
|
const firstAsk = yield* Deferred.make<void>()
|
|
let asks = 0
|
|
const delayedCtx = {
|
|
...ctx,
|
|
ask: () =>
|
|
Effect.gen(function* () {
|
|
asks++
|
|
if (asks !== 1) return
|
|
yield* Deferred.succeed(firstAsk, undefined)
|
|
yield* Effect.promise(() => Bun.sleep(50))
|
|
}),
|
|
}
|
|
|
|
const first = yield* run(
|
|
{
|
|
filePath: filepath,
|
|
oldString: "top = 0",
|
|
newString: "top = 1",
|
|
},
|
|
delayedCtx,
|
|
).pipe(Effect.forkScoped)
|
|
|
|
yield* Deferred.await(firstAsk)
|
|
yield* Effect.all([
|
|
Fiber.join(first),
|
|
run(
|
|
{
|
|
filePath: filepath,
|
|
oldString: "bottom = 0",
|
|
newString: "bottom = 2",
|
|
},
|
|
delayedCtx,
|
|
),
|
|
])
|
|
|
|
expect(yield* load(filepath)).toBe("top = 1\nmiddle = keep\nbottom = 2\n")
|
|
}),
|
|
)
|
|
})
|
|
})
|