| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770 |
- import { afterAll, afterEach, describe, test, expect } from "bun:test"
- import path from "path"
- import fs from "fs/promises"
- import { Effect, Layer, ManagedRuntime } from "effect"
- import { EditTool } from "../../src/tool/edit"
- import { Instance } from "../../src/project/instance"
- import { tmpdir } from "../fixture/fixture"
- import { FileTime } from "../../src/file/time"
- import { LSP } from "../../src/lsp"
- import { AppFileSystem } from "../../src/filesystem"
- import { Format } from "../../src/format"
- import { Agent } from "../../src/agent/agent"
- import { Bus } from "../../src/bus"
- import { BusEvent } from "../../src/bus/bus-event"
- import { Truncate } from "../../src/tool/truncate"
- import { SessionID, MessageID } from "../../src/session/schema"
- const ctx = {
- sessionID: SessionID.make("ses_test-edit-session"),
- messageID: MessageID.make(""),
- callID: "",
- agent: "build",
- abort: AbortSignal.any([]),
- messages: [],
- metadata: () => Effect.void,
- ask: () => Effect.void,
- }
- afterEach(async () => {
- await Instance.disposeAll()
- })
- async function touch(file: string, time: number) {
- const date = new Date(time)
- await fs.utimes(file, date, date)
- }
- const runtime = ManagedRuntime.make(
- Layer.mergeAll(
- LSP.defaultLayer,
- FileTime.defaultLayer,
- AppFileSystem.defaultLayer,
- Format.defaultLayer,
- Bus.layer,
- Truncate.defaultLayer,
- Agent.defaultLayer,
- ),
- )
- afterAll(async () => {
- await runtime.dispose()
- })
- const resolve = () =>
- runtime.runPromise(
- Effect.gen(function* () {
- const info = yield* EditTool
- return yield* info.init()
- }),
- )
- const readFileTime = (sessionID: SessionID, filepath: string) =>
- runtime.runPromise(FileTime.Service.use((ft) => ft.read(sessionID, filepath)))
- const subscribeBus = <D extends BusEvent.Definition>(def: D, callback: () => unknown) =>
- runtime.runPromise(Bus.Service.use((bus) => bus.subscribeCallback(def, callback)))
- async function onceBus<D extends BusEvent.Definition>(def: D) {
- const result = Promise.withResolvers<void>()
- const unsub = await subscribeBus(def, () => {
- unsub()
- result.resolve()
- })
- return {
- wait: result.promise,
- unsub,
- }
- }
- describe("tool.edit", () => {
- describe("creating new files", () => {
- test("creates new file when oldString is empty", async () => {
- await using tmp = await tmpdir()
- const filepath = path.join(tmp.path, "newfile.txt")
- await Instance.provide({
- directory: tmp.path,
- fn: async () => {
- const edit = await resolve()
- const result = await Effect.runPromise(
- edit.execute(
- {
- filePath: filepath,
- oldString: "",
- newString: "new content",
- },
- ctx,
- ),
- )
- expect(result.metadata.diff).toContain("new content")
- const content = await fs.readFile(filepath, "utf-8")
- expect(content).toBe("new content")
- },
- })
- })
- test("creates new file with nested directories", async () => {
- await using tmp = await tmpdir()
- const filepath = path.join(tmp.path, "nested", "dir", "file.txt")
- await Instance.provide({
- directory: tmp.path,
- fn: async () => {
- const edit = await resolve()
- await Effect.runPromise(
- edit.execute(
- {
- filePath: filepath,
- oldString: "",
- newString: "nested file",
- },
- ctx,
- ),
- )
- const content = await fs.readFile(filepath, "utf-8")
- expect(content).toBe("nested file")
- },
- })
- })
- test("emits add event for new files", async () => {
- await using tmp = await tmpdir()
- const filepath = path.join(tmp.path, "new.txt")
- await Instance.provide({
- directory: tmp.path,
- fn: async () => {
- const { FileWatcher } = await import("../../src/file/watcher")
- const updated = await onceBus(FileWatcher.Event.Updated)
- try {
- const edit = await resolve()
- await Effect.runPromise(
- edit.execute(
- {
- filePath: filepath,
- oldString: "",
- newString: "content",
- },
- ctx,
- ),
- )
- await updated.wait
- } finally {
- updated.unsub()
- }
- },
- })
- })
- })
- describe("editing existing files", () => {
- test("replaces text in existing file", async () => {
- await using tmp = await tmpdir()
- const filepath = path.join(tmp.path, "existing.txt")
- await fs.writeFile(filepath, "old content here", "utf-8")
- await Instance.provide({
- directory: tmp.path,
- fn: async () => {
- await readFileTime(ctx.sessionID, filepath)
- const edit = await resolve()
- const result = await Effect.runPromise(
- edit.execute(
- {
- filePath: filepath,
- oldString: "old content",
- newString: "new content",
- },
- ctx,
- ),
- )
- expect(result.output).toContain("Edit applied successfully")
- const content = await fs.readFile(filepath, "utf-8")
- expect(content).toBe("new content here")
- },
- })
- })
- test("throws error when file does not exist", async () => {
- await using tmp = await tmpdir()
- const filepath = path.join(tmp.path, "nonexistent.txt")
- await Instance.provide({
- directory: tmp.path,
- fn: async () => {
- await readFileTime(ctx.sessionID, filepath)
- const edit = await resolve()
- await expect(
- Effect.runPromise(
- edit.execute(
- {
- filePath: filepath,
- oldString: "old",
- newString: "new",
- },
- ctx,
- ),
- ),
- ).rejects.toThrow("not found")
- },
- })
- })
- test("throws error when oldString equals newString", async () => {
- await using tmp = await tmpdir()
- const filepath = path.join(tmp.path, "file.txt")
- await fs.writeFile(filepath, "content", "utf-8")
- await Instance.provide({
- directory: tmp.path,
- fn: async () => {
- const edit = await resolve()
- await expect(
- Effect.runPromise(
- edit.execute(
- {
- filePath: filepath,
- oldString: "same",
- newString: "same",
- },
- ctx,
- ),
- ),
- ).rejects.toThrow("identical")
- },
- })
- })
- test("throws error when oldString not found in file", async () => {
- await using tmp = await tmpdir()
- const filepath = path.join(tmp.path, "file.txt")
- await fs.writeFile(filepath, "actual content", "utf-8")
- await Instance.provide({
- directory: tmp.path,
- fn: async () => {
- await readFileTime(ctx.sessionID, filepath)
- const edit = await resolve()
- await expect(
- Effect.runPromise(
- edit.execute(
- {
- filePath: filepath,
- oldString: "not in file",
- newString: "replacement",
- },
- ctx,
- ),
- ),
- ).rejects.toThrow()
- },
- })
- })
- test("throws error when file was not read first (FileTime)", async () => {
- await using tmp = await tmpdir()
- const filepath = path.join(tmp.path, "file.txt")
- await fs.writeFile(filepath, "content", "utf-8")
- await Instance.provide({
- directory: tmp.path,
- fn: async () => {
- const edit = await resolve()
- await expect(
- Effect.runPromise(
- edit.execute(
- {
- filePath: filepath,
- oldString: "content",
- newString: "modified",
- },
- ctx,
- ),
- ),
- ).rejects.toThrow("You must read file")
- },
- })
- })
- test("throws error when file has been modified since read", async () => {
- await using tmp = await tmpdir()
- const filepath = path.join(tmp.path, "file.txt")
- await fs.writeFile(filepath, "original content", "utf-8")
- await touch(filepath, 1_000)
- await Instance.provide({
- directory: tmp.path,
- fn: async () => {
- // Read first
- await readFileTime(ctx.sessionID, filepath)
- // Simulate external modification
- await fs.writeFile(filepath, "modified externally", "utf-8")
- await touch(filepath, 2_000)
- // Try to edit with the new content
- const edit = await resolve()
- await expect(
- Effect.runPromise(
- edit.execute(
- {
- filePath: filepath,
- oldString: "modified externally",
- newString: "edited",
- },
- ctx,
- ),
- ),
- ).rejects.toThrow("modified since it was last read")
- },
- })
- })
- test("replaces all occurrences with replaceAll option", async () => {
- await using tmp = await tmpdir()
- const filepath = path.join(tmp.path, "file.txt")
- await fs.writeFile(filepath, "foo bar foo baz foo", "utf-8")
- await Instance.provide({
- directory: tmp.path,
- fn: async () => {
- await readFileTime(ctx.sessionID, filepath)
- const edit = await resolve()
- await Effect.runPromise(
- edit.execute(
- {
- filePath: filepath,
- oldString: "foo",
- newString: "qux",
- replaceAll: true,
- },
- ctx,
- ),
- )
- const content = await fs.readFile(filepath, "utf-8")
- expect(content).toBe("qux bar qux baz qux")
- },
- })
- })
- test("emits change event for existing files", async () => {
- await using tmp = await tmpdir()
- const filepath = path.join(tmp.path, "file.txt")
- await fs.writeFile(filepath, "original", "utf-8")
- await Instance.provide({
- directory: tmp.path,
- fn: async () => {
- await readFileTime(ctx.sessionID, filepath)
- const { FileWatcher } = await import("../../src/file/watcher")
- const updated = await onceBus(FileWatcher.Event.Updated)
- try {
- const edit = await resolve()
- await Effect.runPromise(
- edit.execute(
- {
- filePath: filepath,
- oldString: "original",
- newString: "modified",
- },
- ctx,
- ),
- )
- await updated.wait
- } finally {
- updated.unsub()
- }
- },
- })
- })
- })
- describe("edge cases", () => {
- test("handles multiline replacements", async () => {
- await using tmp = await tmpdir()
- const filepath = path.join(tmp.path, "file.txt")
- await fs.writeFile(filepath, "line1\nline2\nline3", "utf-8")
- await Instance.provide({
- directory: tmp.path,
- fn: async () => {
- await readFileTime(ctx.sessionID, filepath)
- const edit = await resolve()
- await Effect.runPromise(
- edit.execute(
- {
- filePath: filepath,
- oldString: "line2",
- newString: "new line 2\nextra line",
- },
- ctx,
- ),
- )
- const content = await fs.readFile(filepath, "utf-8")
- expect(content).toBe("line1\nnew line 2\nextra line\nline3")
- },
- })
- })
- test("handles CRLF line endings", async () => {
- await using tmp = await tmpdir()
- const filepath = path.join(tmp.path, "file.txt")
- await fs.writeFile(filepath, "line1\r\nold\r\nline3", "utf-8")
- await Instance.provide({
- directory: tmp.path,
- fn: async () => {
- await readFileTime(ctx.sessionID, filepath)
- const edit = await resolve()
- await Effect.runPromise(
- edit.execute(
- {
- filePath: filepath,
- oldString: "old",
- newString: "new",
- },
- ctx,
- ),
- )
- const content = await fs.readFile(filepath, "utf-8")
- expect(content).toBe("line1\r\nnew\r\nline3")
- },
- })
- })
- test("throws error when oldString equals newString", async () => {
- await using tmp = await tmpdir()
- const filepath = path.join(tmp.path, "file.txt")
- await fs.writeFile(filepath, "content", "utf-8")
- await Instance.provide({
- directory: tmp.path,
- fn: async () => {
- const edit = await resolve()
- await expect(
- Effect.runPromise(
- edit.execute(
- {
- filePath: filepath,
- oldString: "",
- newString: "",
- },
- ctx,
- ),
- ),
- ).rejects.toThrow("identical")
- },
- })
- })
- test("throws error when path is directory", async () => {
- await using tmp = await tmpdir()
- const dirpath = path.join(tmp.path, "adir")
- await fs.mkdir(dirpath)
- await Instance.provide({
- directory: tmp.path,
- fn: async () => {
- await readFileTime(ctx.sessionID, dirpath)
- const edit = await resolve()
- await expect(
- Effect.runPromise(
- edit.execute(
- {
- filePath: dirpath,
- oldString: "old",
- newString: "new",
- },
- ctx,
- ),
- ),
- ).rejects.toThrow("directory")
- },
- })
- })
- test("tracks file diff statistics", async () => {
- await using tmp = await tmpdir()
- const filepath = path.join(tmp.path, "file.txt")
- await fs.writeFile(filepath, "line1\nline2\nline3", "utf-8")
- await Instance.provide({
- directory: tmp.path,
- fn: async () => {
- await readFileTime(ctx.sessionID, filepath)
- const edit = await resolve()
- const result = await Effect.runPromise(
- edit.execute(
- {
- filePath: filepath,
- oldString: "line2",
- newString: "new line a\nnew line b",
- },
- ctx,
- ),
- )
- 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 = async (input: Input) => {
- await using tmp = await tmpdir({
- init: async (dir) => {
- await Bun.write(path.join(dir, "test.txt"), input.content)
- },
- })
- return await Instance.provide({
- directory: tmp.path,
- fn: async () => {
- const edit = await resolve()
- const filePath = path.join(tmp.path, "test.txt")
- await readFileTime(ctx.sessionID, filePath)
- await Effect.runPromise(
- edit.execute(
- {
- filePath,
- oldString: input.oldString,
- newString: input.newString,
- replaceAll: input.replaceAll,
- },
- ctx,
- ),
- )
- return await Bun.file(filePath).text()
- },
- })
- }
- test("preserves LF with LF multi-line strings", async () => {
- const content = normalize(old + "\n", "\n")
- const output = await apply({
- content,
- oldString: normalize(old, "\n"),
- newString: normalize(next, "\n"),
- })
- expect(output).toBe(normalize(next + "\n", "\n"))
- expectLf(output)
- })
- test("preserves CRLF with CRLF multi-line strings", async () => {
- const content = normalize(old + "\n", "\r\n")
- const output = await apply({
- content,
- oldString: normalize(old, "\r\n"),
- newString: normalize(next, "\r\n"),
- })
- expect(output).toBe(normalize(next + "\n", "\r\n"))
- expectCrlf(output)
- })
- test("preserves LF when old/new use CRLF", async () => {
- const content = normalize(old + "\n", "\n")
- const output = await apply({
- content,
- oldString: normalize(old, "\r\n"),
- newString: normalize(next, "\r\n"),
- })
- expect(output).toBe(normalize(next + "\n", "\n"))
- expectLf(output)
- })
- test("preserves CRLF when old/new use LF", async () => {
- const content = normalize(old + "\n", "\r\n")
- const output = await apply({
- content,
- oldString: normalize(old, "\n"),
- newString: normalize(next, "\n"),
- })
- expect(output).toBe(normalize(next + "\n", "\r\n"))
- expectCrlf(output)
- })
- test("preserves LF when newString uses CRLF", async () => {
- const content = normalize(old + "\n", "\n")
- const output = await apply({
- content,
- oldString: normalize(old, "\n"),
- newString: normalize(next, "\r\n"),
- })
- expect(output).toBe(normalize(next + "\n", "\n"))
- expectLf(output)
- })
- test("preserves CRLF when newString uses LF", async () => {
- const content = normalize(old + "\n", "\r\n")
- const output = await apply({
- content,
- oldString: normalize(old, "\r\n"),
- newString: normalize(next, "\n"),
- })
- expect(output).toBe(normalize(next + "\n", "\r\n"))
- expectCrlf(output)
- })
- test("preserves LF with mixed old/new line endings", async () => {
- const content = normalize(old + "\n", "\n")
- const output = await apply({
- content,
- oldString: "alpha\nbeta\r\ngamma",
- newString: "alpha\r\nbeta\nomega",
- })
- expect(output).toBe(normalize(alt + "\n", "\n"))
- expectLf(output)
- })
- test("preserves CRLF with mixed old/new line endings", async () => {
- const content = normalize(old + "\n", "\r\n")
- const output = await apply({
- content,
- oldString: "alpha\r\nbeta\ngamma",
- newString: "alpha\nbeta\r\nomega",
- })
- expect(output).toBe(normalize(alt + "\n", "\r\n"))
- expectCrlf(output)
- })
- test("replaceAll preserves LF for multi-line blocks", async () => {
- const blockOld = "alpha\nbeta"
- const blockNew = "alpha\nbeta-updated"
- const content = normalize(blockOld + "\n" + blockOld + "\n", "\n")
- const output = await apply({
- content,
- oldString: normalize(blockOld, "\n"),
- newString: normalize(blockNew, "\n"),
- replaceAll: true,
- })
- expect(output).toBe(normalize(blockNew + "\n" + blockNew + "\n", "\n"))
- expectLf(output)
- })
- test("replaceAll preserves CRLF for multi-line blocks", async () => {
- const blockOld = "alpha\nbeta"
- const blockNew = "alpha\nbeta-updated"
- const content = normalize(blockOld + "\n" + blockOld + "\n", "\r\n")
- const output = await 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", () => {
- test("serializes concurrent edits to same file", async () => {
- await using tmp = await tmpdir()
- const filepath = path.join(tmp.path, "file.txt")
- await fs.writeFile(filepath, "0", "utf-8")
- await Instance.provide({
- directory: tmp.path,
- fn: async () => {
- await readFileTime(ctx.sessionID, filepath)
- const edit = await resolve()
- // Two concurrent edits
- const promise1 = Effect.runPromise(
- edit.execute(
- {
- filePath: filepath,
- oldString: "0",
- newString: "1",
- },
- ctx,
- ),
- )
- // Need to read again since FileTime tracks per-session
- await readFileTime(ctx.sessionID, filepath)
- const promise2 = Effect.runPromise(
- edit.execute(
- {
- filePath: filepath,
- oldString: "0",
- newString: "2",
- },
- ctx,
- ),
- )
- // Both should complete without error (though one might fail due to content mismatch)
- const results = await Promise.allSettled([promise1, promise2])
- expect(results.some((r) => r.status === "fulfilled")).toBe(true)
- },
- })
- })
- })
- })
|