349 lines
9.6 KiB
TypeScript
349 lines
9.6 KiB
TypeScript
import { describe, test, expect, beforeEach, afterEach } from "bun:test"
|
|
import { Patch } from "../../src/patch"
|
|
import * as fs from "fs/promises"
|
|
import * as path from "path"
|
|
import { tmpdir } from "os"
|
|
|
|
describe("Patch namespace", () => {
|
|
let tempDir: string
|
|
|
|
beforeEach(async () => {
|
|
tempDir = await fs.mkdtemp(path.join(tmpdir(), "patch-test-"))
|
|
})
|
|
|
|
afterEach(async () => {
|
|
// Clean up temp directory
|
|
await fs.rm(tempDir, { recursive: true, force: true })
|
|
})
|
|
|
|
describe("parsePatch", () => {
|
|
test("should parse simple add file patch", () => {
|
|
const patchText = `*** Begin Patch
|
|
*** Add File: test.txt
|
|
+Hello World
|
|
*** End Patch`
|
|
|
|
const result = Patch.parsePatch(patchText)
|
|
expect(result.hunks).toHaveLength(1)
|
|
expect(result.hunks[0]).toEqual({
|
|
type: "add",
|
|
path: "test.txt",
|
|
contents: "Hello World",
|
|
})
|
|
})
|
|
|
|
test("should parse delete file patch", () => {
|
|
const patchText = `*** Begin Patch
|
|
*** Delete File: old.txt
|
|
*** End Patch`
|
|
|
|
const result = Patch.parsePatch(patchText)
|
|
expect(result.hunks).toHaveLength(1)
|
|
const hunk = result.hunks[0]
|
|
expect(hunk.type).toBe("delete")
|
|
expect(hunk.path).toBe("old.txt")
|
|
})
|
|
|
|
test("should parse patch with multiple hunks", () => {
|
|
const patchText = `*** Begin Patch
|
|
*** Add File: new.txt
|
|
+This is a new file
|
|
*** Update File: existing.txt
|
|
@@
|
|
old line
|
|
-new line
|
|
+updated line
|
|
*** End Patch`
|
|
|
|
const result = Patch.parsePatch(patchText)
|
|
expect(result.hunks).toHaveLength(2)
|
|
expect(result.hunks[0].type).toBe("add")
|
|
expect(result.hunks[1].type).toBe("update")
|
|
})
|
|
|
|
test("should parse file move operation", () => {
|
|
const patchText = `*** Begin Patch
|
|
*** Update File: old-name.txt
|
|
*** Move to: new-name.txt
|
|
@@
|
|
-Old content
|
|
+New content
|
|
*** End Patch`
|
|
|
|
const result = Patch.parsePatch(patchText)
|
|
expect(result.hunks).toHaveLength(1)
|
|
const hunk = result.hunks[0]
|
|
expect(hunk.type).toBe("update")
|
|
expect(hunk.path).toBe("old-name.txt")
|
|
if (hunk.type === "update") {
|
|
expect(hunk.move_path).toBe("new-name.txt")
|
|
}
|
|
})
|
|
|
|
test("should throw error for invalid patch format", () => {
|
|
const invalidPatch = `This is not a valid patch`
|
|
|
|
expect(() => Patch.parsePatch(invalidPatch)).toThrow("Invalid patch format")
|
|
})
|
|
})
|
|
|
|
describe("maybeParseApplyPatch", () => {
|
|
test("should parse direct apply_patch command", () => {
|
|
const patchText = `*** Begin Patch
|
|
*** Add File: test.txt
|
|
+Content
|
|
*** End Patch`
|
|
|
|
const result = Patch.maybeParseApplyPatch(["apply_patch", patchText])
|
|
expect(result.type).toBe(Patch.MaybeApplyPatch.Body)
|
|
if (result.type === Patch.MaybeApplyPatch.Body) {
|
|
expect(result.args.patch).toBe(patchText)
|
|
expect(result.args.hunks).toHaveLength(1)
|
|
}
|
|
})
|
|
|
|
test("should parse applypatch command", () => {
|
|
const patchText = `*** Begin Patch
|
|
*** Add File: test.txt
|
|
+Content
|
|
*** End Patch`
|
|
|
|
const result = Patch.maybeParseApplyPatch(["applypatch", patchText])
|
|
expect(result.type).toBe(Patch.MaybeApplyPatch.Body)
|
|
})
|
|
|
|
test("should handle bash heredoc format", () => {
|
|
const script = `apply_patch <<'PATCH'
|
|
*** Begin Patch
|
|
*** Add File: test.txt
|
|
+Content
|
|
*** End Patch
|
|
PATCH`
|
|
|
|
const result = Patch.maybeParseApplyPatch(["bash", "-lc", script])
|
|
expect(result.type).toBe(Patch.MaybeApplyPatch.Body)
|
|
if (result.type === Patch.MaybeApplyPatch.Body) {
|
|
expect(result.args.hunks).toHaveLength(1)
|
|
}
|
|
})
|
|
|
|
test("should return NotApplyPatch for non-patch commands", () => {
|
|
const result = Patch.maybeParseApplyPatch(["echo", "hello"])
|
|
expect(result.type).toBe(Patch.MaybeApplyPatch.NotApplyPatch)
|
|
})
|
|
})
|
|
|
|
describe("applyPatch", () => {
|
|
test("should add a new file", async () => {
|
|
const patchText = `*** Begin Patch
|
|
*** Add File: ${tempDir}/new-file.txt
|
|
+Hello World
|
|
+This is a new file
|
|
*** End Patch`
|
|
|
|
const result = await Patch.applyPatch(patchText)
|
|
expect(result.added).toHaveLength(1)
|
|
expect(result.modified).toHaveLength(0)
|
|
expect(result.deleted).toHaveLength(0)
|
|
|
|
const content = await fs.readFile(result.added[0], "utf-8")
|
|
expect(content).toBe("Hello World\nThis is a new file")
|
|
})
|
|
|
|
test("should delete an existing file", async () => {
|
|
const filePath = path.join(tempDir, "to-delete.txt")
|
|
await fs.writeFile(filePath, "This file will be deleted")
|
|
|
|
const patchText = `*** Begin Patch
|
|
*** Delete File: ${filePath}
|
|
*** End Patch`
|
|
|
|
const result = await Patch.applyPatch(patchText)
|
|
expect(result.deleted).toHaveLength(1)
|
|
expect(result.deleted[0]).toBe(filePath)
|
|
|
|
const exists = await fs
|
|
.access(filePath)
|
|
.then(() => true)
|
|
.catch(() => false)
|
|
expect(exists).toBe(false)
|
|
})
|
|
|
|
test("should update an existing file", async () => {
|
|
const filePath = path.join(tempDir, "to-update.txt")
|
|
await fs.writeFile(filePath, "line 1\nline 2\nline 3\n")
|
|
|
|
const patchText = `*** Begin Patch
|
|
*** Update File: ${filePath}
|
|
@@
|
|
line 1
|
|
-line 2
|
|
+line 2 updated
|
|
line 3
|
|
*** End Patch`
|
|
|
|
const result = await Patch.applyPatch(patchText)
|
|
expect(result.modified).toHaveLength(1)
|
|
expect(result.modified[0]).toBe(filePath)
|
|
|
|
const content = await fs.readFile(filePath, "utf-8")
|
|
expect(content).toBe("line 1\nline 2 updated\nline 3\n")
|
|
})
|
|
|
|
test("should move and update a file", async () => {
|
|
const oldPath = path.join(tempDir, "old-name.txt")
|
|
const newPath = path.join(tempDir, "new-name.txt")
|
|
await fs.writeFile(oldPath, "old content\n")
|
|
|
|
const patchText = `*** Begin Patch
|
|
*** Update File: ${oldPath}
|
|
*** Move to: ${newPath}
|
|
@@
|
|
-old content
|
|
+new content
|
|
*** End Patch`
|
|
|
|
const result = await Patch.applyPatch(patchText)
|
|
expect(result.modified).toHaveLength(1)
|
|
expect(result.modified[0]).toBe(newPath)
|
|
|
|
const oldExists = await fs
|
|
.access(oldPath)
|
|
.then(() => true)
|
|
.catch(() => false)
|
|
expect(oldExists).toBe(false)
|
|
|
|
const newContent = await fs.readFile(newPath, "utf-8")
|
|
expect(newContent).toBe("new content\n")
|
|
})
|
|
|
|
test("should handle multiple operations in one patch", async () => {
|
|
const file1 = path.join(tempDir, "file1.txt")
|
|
const file2 = path.join(tempDir, "file2.txt")
|
|
const file3 = path.join(tempDir, "file3.txt")
|
|
|
|
await fs.writeFile(file1, "content 1")
|
|
await fs.writeFile(file2, "content 2")
|
|
|
|
const patchText = `*** Begin Patch
|
|
*** Add File: ${file3}
|
|
+new file content
|
|
*** Update File: ${file1}
|
|
@@
|
|
-content 1
|
|
+updated content 1
|
|
*** Delete File: ${file2}
|
|
*** End Patch`
|
|
|
|
const result = await Patch.applyPatch(patchText)
|
|
expect(result.added).toHaveLength(1)
|
|
expect(result.modified).toHaveLength(1)
|
|
expect(result.deleted).toHaveLength(1)
|
|
})
|
|
|
|
test("should create parent directories when adding files", async () => {
|
|
const nestedPath = path.join(tempDir, "deep", "nested", "file.txt")
|
|
|
|
const patchText = `*** Begin Patch
|
|
*** Add File: ${nestedPath}
|
|
+Deep nested content
|
|
*** End Patch`
|
|
|
|
const result = await Patch.applyPatch(patchText)
|
|
expect(result.added).toHaveLength(1)
|
|
expect(result.added[0]).toBe(nestedPath)
|
|
|
|
const exists = await fs
|
|
.access(nestedPath)
|
|
.then(() => true)
|
|
.catch(() => false)
|
|
expect(exists).toBe(true)
|
|
})
|
|
})
|
|
|
|
describe("error handling", () => {
|
|
test("should throw error when updating non-existent file", async () => {
|
|
const nonExistent = path.join(tempDir, "does-not-exist.txt")
|
|
|
|
const patchText = `*** Begin Patch
|
|
*** Update File: ${nonExistent}
|
|
@@
|
|
-old line
|
|
+new line
|
|
*** End Patch`
|
|
|
|
await expect(Patch.applyPatch(patchText)).rejects.toThrow()
|
|
})
|
|
|
|
test("should throw error when deleting non-existent file", async () => {
|
|
const nonExistent = path.join(tempDir, "does-not-exist.txt")
|
|
|
|
const patchText = `*** Begin Patch
|
|
*** Delete File: ${nonExistent}
|
|
*** End Patch`
|
|
|
|
await expect(Patch.applyPatch(patchText)).rejects.toThrow()
|
|
})
|
|
})
|
|
|
|
describe("edge cases", () => {
|
|
test("should handle empty files", async () => {
|
|
const emptyFile = path.join(tempDir, "empty.txt")
|
|
await fs.writeFile(emptyFile, "")
|
|
|
|
const patchText = `*** Begin Patch
|
|
*** Update File: ${emptyFile}
|
|
@@
|
|
+First line
|
|
*** End Patch`
|
|
|
|
const result = await Patch.applyPatch(patchText)
|
|
expect(result.modified).toHaveLength(1)
|
|
|
|
const content = await fs.readFile(emptyFile, "utf-8")
|
|
expect(content).toBe("First line\n")
|
|
})
|
|
|
|
test("should handle files with no trailing newline", async () => {
|
|
const filePath = path.join(tempDir, "no-newline.txt")
|
|
await fs.writeFile(filePath, "no newline")
|
|
|
|
const patchText = `*** Begin Patch
|
|
*** Update File: ${filePath}
|
|
@@
|
|
-no newline
|
|
+has newline now
|
|
*** End Patch`
|
|
|
|
const result = await Patch.applyPatch(patchText)
|
|
expect(result.modified).toHaveLength(1)
|
|
|
|
const content = await fs.readFile(filePath, "utf-8")
|
|
expect(content).toBe("has newline now\n")
|
|
})
|
|
|
|
test("should handle multiple update chunks in single file", async () => {
|
|
const filePath = path.join(tempDir, "multi-chunk.txt")
|
|
await fs.writeFile(filePath, "line 1\nline 2\nline 3\nline 4\n")
|
|
|
|
const patchText = `*** Begin Patch
|
|
*** Update File: ${filePath}
|
|
@@
|
|
line 1
|
|
-line 2
|
|
+LINE 2
|
|
@@
|
|
line 3
|
|
-line 4
|
|
+LINE 4
|
|
*** End Patch`
|
|
|
|
const result = await Patch.applyPatch(patchText)
|
|
expect(result.modified).toHaveLength(1)
|
|
|
|
const content = await fs.readFile(filePath, "utf-8")
|
|
expect(content).toBe("line 1\nLINE 2\nline 3\nLINE 4\n")
|
|
})
|
|
})
|
|
})
|