mirror of
https://github.com/anomalyco/opencode.git
synced 2026-05-22 03:45:23 +00:00
844 lines
31 KiB
TypeScript
844 lines
31 KiB
TypeScript
import { afterEach, describe, expect } from "bun:test"
|
|
import { AppFileSystem } from "@opencode-ai/core/filesystem"
|
|
import { $ } from "bun"
|
|
import { Cause, Effect, Exit, Layer } from "effect"
|
|
import path from "path"
|
|
import fs from "fs/promises"
|
|
import { File } from "../../src/file"
|
|
import { Filesystem } from "@/util/filesystem"
|
|
import { disposeAllInstances, TestInstance, withTmpdirInstance } from "../fixture/fixture"
|
|
import { testEffect } from "../lib/effect"
|
|
|
|
afterEach(async () => {
|
|
await disposeAllInstances()
|
|
})
|
|
|
|
const it = testEffect(Layer.mergeAll(File.defaultLayer, AppFileSystem.defaultLayer))
|
|
|
|
const init = Effect.fn("FileTest.init")(function* () {
|
|
const file = yield* File.Service
|
|
return yield* file.init()
|
|
})
|
|
|
|
const status = Effect.fn("FileTest.status")(function* () {
|
|
const file = yield* File.Service
|
|
return yield* file.status()
|
|
})
|
|
|
|
const read = Effect.fn("FileTest.read")(function* (input: string) {
|
|
const file = yield* File.Service
|
|
return yield* file.read(input)
|
|
})
|
|
|
|
const list = Effect.fn("FileTest.list")(function* (dir?: string) {
|
|
const file = yield* File.Service
|
|
return yield* file.list(dir)
|
|
})
|
|
|
|
const search = Effect.fn("FileTest.search")(function* (input: {
|
|
query: string
|
|
limit?: number
|
|
dirs?: boolean
|
|
type?: "file" | "directory"
|
|
}) {
|
|
const file = yield* File.Service
|
|
return yield* file.search(input)
|
|
})
|
|
|
|
const gitAddAll = (directory: string) => Effect.promise(() => $`git add .`.cwd(directory).quiet())
|
|
const gitCommit = (directory: string, message: string) =>
|
|
Effect.promise(() => $`git commit -m ${message}`.cwd(directory).quiet())
|
|
|
|
const failureMessage = <A, E, R>(self: Effect.Effect<A, E, R>) =>
|
|
Effect.gen(function* () {
|
|
const exit = yield* self.pipe(Effect.exit)
|
|
if (Exit.isFailure(exit)) {
|
|
const error = Cause.squash(exit.cause)
|
|
return error instanceof Error ? error.message : String(error)
|
|
}
|
|
throw new Error("expected effect to fail")
|
|
})
|
|
|
|
const setupSearchableRepo = Effect.fn("FileTest.setupSearchableRepo")(function* (directory: string) {
|
|
const fsys = yield* AppFileSystem.Service
|
|
yield* fsys.writeWithDirs(path.join(directory, "index.ts"), "code")
|
|
yield* fsys.writeWithDirs(path.join(directory, "utils.ts"), "utils")
|
|
yield* fsys.writeWithDirs(path.join(directory, "readme.md"), "readme")
|
|
yield* fsys.writeWithDirs(path.join(directory, "src", "main.ts"), "main")
|
|
yield* fsys.writeWithDirs(path.join(directory, ".hidden", "secret.ts"), "secret")
|
|
})
|
|
|
|
describe("file/index Filesystem patterns", () => {
|
|
describe("read() - text content", () => {
|
|
it.instance("reads text file via Filesystem.readText()", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "test.txt"), "Hello World", "utf-8"))
|
|
|
|
const result = yield* read("test.txt")
|
|
expect(result.type).toBe("text")
|
|
expect(result.content).toBe("Hello World")
|
|
}),
|
|
)
|
|
|
|
it.instance("reads with Filesystem.exists() check", () =>
|
|
Effect.gen(function* () {
|
|
const result = yield* read("nonexistent.txt")
|
|
expect(result.type).toBe("text")
|
|
expect(result.content).toBe("")
|
|
}),
|
|
)
|
|
|
|
it.instance("trims whitespace from text content", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() =>
|
|
fs.writeFile(path.join(test.directory, "test.txt"), " content with spaces \n\n", "utf-8"),
|
|
)
|
|
|
|
const result = yield* read("test.txt")
|
|
expect(result.content).toBe("content with spaces")
|
|
}),
|
|
)
|
|
|
|
it.instance("handles empty text file", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "empty.txt"), "", "utf-8"))
|
|
|
|
const result = yield* read("empty.txt")
|
|
expect(result.type).toBe("text")
|
|
expect(result.content).toBe("")
|
|
}),
|
|
)
|
|
|
|
it.instance("handles multi-line text files", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "multiline.txt"), "line1\nline2\nline3", "utf-8"))
|
|
|
|
const result = yield* read("multiline.txt")
|
|
expect(result.content).toBe("line1\nline2\nline3")
|
|
}),
|
|
)
|
|
})
|
|
|
|
describe("read() - binary content", () => {
|
|
it.instance("reads binary file via Filesystem.readArrayBuffer()", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const binaryContent = Buffer.from([0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a])
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "image.png"), binaryContent))
|
|
|
|
const result = yield* read("image.png")
|
|
expect(result.type).toBe("text")
|
|
expect(result.encoding).toBe("base64")
|
|
expect(result.mimeType).toBe("image/png")
|
|
expect(result.content).toBe(binaryContent.toString("base64"))
|
|
}),
|
|
)
|
|
|
|
it.instance("returns empty for binary non-image files", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "binary.so"), Buffer.from([0x7f, 0x45, 0x4c, 0x46])))
|
|
|
|
const result = yield* read("binary.so")
|
|
expect(result.type).toBe("binary")
|
|
expect(result.content).toBe("")
|
|
}),
|
|
)
|
|
})
|
|
|
|
describe("read() - Filesystem.mimeType()", () => {
|
|
it.instance("detects MIME type via Filesystem.mimeType()", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "test.json")
|
|
yield* Effect.promise(() => fs.writeFile(filepath, '{"key": "value"}', "utf-8"))
|
|
|
|
expect(yield* Effect.promise(() => Filesystem.mimeType(filepath))).toContain("application/json")
|
|
|
|
const result = yield* read("test.json")
|
|
expect(result.type).toBe("text")
|
|
}),
|
|
)
|
|
|
|
it.instance("handles various image MIME types", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const testCases = [
|
|
{ ext: "jpg", mime: "image/jpeg" },
|
|
{ ext: "png", mime: "image/png" },
|
|
{ ext: "gif", mime: "image/gif" },
|
|
{ ext: "webp", mime: "image/webp" },
|
|
]
|
|
|
|
for (const testCase of testCases) {
|
|
const filepath = path.join(test.directory, `test.${testCase.ext}`)
|
|
yield* Effect.promise(() => fs.writeFile(filepath, Buffer.from([0x00, 0x00, 0x00, 0x00])))
|
|
expect(yield* Effect.promise(() => Filesystem.mimeType(filepath))).toContain(testCase.mime)
|
|
}
|
|
}),
|
|
)
|
|
})
|
|
|
|
describe("list() - Filesystem.exists() and readText()", () => {
|
|
it.instance(
|
|
"reads .gitignore via Filesystem.exists() and readText()",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const gitignorePath = path.join(test.directory, ".gitignore")
|
|
yield* Effect.promise(() => fs.writeFile(gitignorePath, "node_modules\ndist\n", "utf-8"))
|
|
|
|
expect(yield* Effect.promise(() => Filesystem.exists(gitignorePath))).toBe(true)
|
|
expect(yield* Effect.promise(() => Filesystem.readText(gitignorePath))).toContain("node_modules")
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"reads .ignore file similarly",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const ignorePath = path.join(test.directory, ".ignore")
|
|
yield* Effect.promise(() => fs.writeFile(ignorePath, "*.log\n.env\n", "utf-8"))
|
|
|
|
expect(yield* Effect.promise(() => Filesystem.exists(ignorePath))).toBe(true)
|
|
expect(yield* Effect.promise(() => Filesystem.readText(ignorePath))).toContain("*.log")
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"handles missing .gitignore gracefully",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const gitignorePath = path.join(test.directory, ".gitignore")
|
|
expect(yield* Effect.promise(() => Filesystem.exists(gitignorePath))).toBe(false)
|
|
|
|
const nodes = yield* list()
|
|
expect(Array.isArray(nodes)).toBe(true)
|
|
}),
|
|
{ git: true },
|
|
)
|
|
})
|
|
|
|
describe("File.changed() - Filesystem.readText() for untracked files", () => {
|
|
it.instance(
|
|
"reads untracked files via Filesystem.readText()",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const untrackedPath = path.join(test.directory, "untracked.txt")
|
|
yield* Effect.promise(() => fs.writeFile(untrackedPath, "new content\nwith multiple lines", "utf-8"))
|
|
|
|
const content = yield* Effect.promise(() => Filesystem.readText(untrackedPath))
|
|
expect(content.split("\n").length).toBe(2)
|
|
}),
|
|
{ git: true },
|
|
)
|
|
})
|
|
|
|
describe("Error handling", () => {
|
|
it.instance("handles errors gracefully in Filesystem.readText()", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "readonly.txt"), "content", "utf-8"))
|
|
|
|
const nonExistentPath = path.join(test.directory, "does-not-exist.txt")
|
|
expect(Exit.isFailure(yield* Effect.promise(() => Filesystem.readText(nonExistentPath)).pipe(Effect.exit))).toBe(true)
|
|
|
|
const result = yield* read("does-not-exist.txt")
|
|
expect(result.content).toBe("")
|
|
}),
|
|
)
|
|
|
|
it.instance("handles errors in Filesystem.readArrayBuffer()", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const nonExistentPath = path.join(test.directory, "does-not-exist.bin")
|
|
const buffer = yield* Effect.promise(() => Filesystem.readArrayBuffer(nonExistentPath).catch(() => new ArrayBuffer(0)))
|
|
expect(buffer.byteLength).toBe(0)
|
|
}),
|
|
)
|
|
|
|
it.instance("returns empty array buffer on error for images", () =>
|
|
Effect.gen(function* () {
|
|
const result = yield* read("broken.png")
|
|
expect(result.type).toBe("text")
|
|
expect(result.content).toBe("")
|
|
}),
|
|
)
|
|
})
|
|
|
|
describe("shouldEncode() logic", () => {
|
|
it.instance("treats .ts files as text", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "test.ts"), "export const value = 1", "utf-8"))
|
|
|
|
const result = yield* read("test.ts")
|
|
expect(result.type).toBe("text")
|
|
expect(result.content).toBe("export const value = 1")
|
|
}),
|
|
)
|
|
|
|
it.instance("treats .mts files as text", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "test.mts"), "export const value = 1", "utf-8"))
|
|
|
|
const result = yield* read("test.mts")
|
|
expect(result.type).toBe("text")
|
|
expect(result.content).toBe("export const value = 1")
|
|
}),
|
|
)
|
|
|
|
it.instance("treats .sh files as text", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "test.sh"), "#!/usr/bin/env bash\necho hello", "utf-8"))
|
|
|
|
const result = yield* read("test.sh")
|
|
expect(result.type).toBe("text")
|
|
expect(result.content).toBe("#!/usr/bin/env bash\necho hello")
|
|
}),
|
|
)
|
|
|
|
it.instance("treats Dockerfile as text", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "Dockerfile"), "FROM alpine:3.20", "utf-8"))
|
|
|
|
const result = yield* read("Dockerfile")
|
|
expect(result.type).toBe("text")
|
|
expect(result.content).toBe("FROM alpine:3.20")
|
|
}),
|
|
)
|
|
|
|
it.instance("returns encoding info for text files", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "test.txt"), "simple text", "utf-8"))
|
|
|
|
const result = yield* read("test.txt")
|
|
expect(result.encoding).toBeUndefined()
|
|
expect(result.type).toBe("text")
|
|
}),
|
|
)
|
|
|
|
it.instance("returns base64 encoding for images", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "test.jpg"), Buffer.from([0xff, 0xd8, 0xff, 0xe0])))
|
|
|
|
const result = yield* read("test.jpg")
|
|
expect(result.encoding).toBe("base64")
|
|
expect(result.mimeType).toBe("image/jpeg")
|
|
}),
|
|
)
|
|
})
|
|
|
|
describe("Path security", () => {
|
|
it.instance("throws for paths outside project directory", () =>
|
|
Effect.gen(function* () {
|
|
expect(yield* failureMessage(read("../outside.txt"))).toContain("Access denied")
|
|
}),
|
|
)
|
|
|
|
it.instance("throws for paths outside project directory", () =>
|
|
Effect.gen(function* () {
|
|
expect(yield* failureMessage(read("../outside.txt"))).toContain("Access denied")
|
|
}),
|
|
)
|
|
})
|
|
|
|
describe("status()", () => {
|
|
it.instance(
|
|
"detects modified file",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "file.txt")
|
|
yield* Effect.promise(() => fs.writeFile(filepath, "original\n", "utf-8"))
|
|
yield* gitAddAll(test.directory)
|
|
yield* gitCommit(test.directory, "add file")
|
|
yield* Effect.promise(() => fs.writeFile(filepath, "modified\nextra line\n", "utf-8"))
|
|
|
|
const result = yield* status()
|
|
const entry = result.find((file) => file.path === "file.txt")
|
|
expect(entry).toBeDefined()
|
|
expect(entry!.status).toBe("modified")
|
|
expect(entry!.added).toBeGreaterThan(0)
|
|
expect(entry!.removed).toBeGreaterThan(0)
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"detects untracked file as added",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "new.txt"), "line1\nline2\nline3\n", "utf-8"))
|
|
|
|
const result = yield* status()
|
|
const entry = result.find((file) => file.path === "new.txt")
|
|
expect(entry).toBeDefined()
|
|
expect(entry!.status).toBe("added")
|
|
expect(entry!.added).toBe(4)
|
|
expect(entry!.removed).toBe(0)
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"detects deleted file",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "gone.txt")
|
|
yield* Effect.promise(() => fs.writeFile(filepath, "content\n", "utf-8"))
|
|
yield* gitAddAll(test.directory)
|
|
yield* gitCommit(test.directory, "add file")
|
|
yield* Effect.promise(() => fs.rm(filepath))
|
|
|
|
const result = yield* status()
|
|
const entries = result.filter((file) => file.path === "gone.txt")
|
|
expect(entries.some((entry) => entry.status === "deleted")).toBe(true)
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"detects mixed changes",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "keep.txt"), "keep\n", "utf-8"))
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "remove.txt"), "remove\n", "utf-8"))
|
|
yield* gitAddAll(test.directory)
|
|
yield* gitCommit(test.directory, "initial")
|
|
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "keep.txt"), "changed\n", "utf-8"))
|
|
yield* Effect.promise(() => fs.rm(path.join(test.directory, "remove.txt")))
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "brand-new.txt"), "hello\n", "utf-8"))
|
|
|
|
const result = yield* status()
|
|
expect(result.some((file) => file.path === "keep.txt" && file.status === "modified")).toBe(true)
|
|
expect(result.some((file) => file.path === "remove.txt" && file.status === "deleted")).toBe(true)
|
|
expect(result.some((file) => file.path === "brand-new.txt" && file.status === "added")).toBe(true)
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance("returns empty for non-git project", () =>
|
|
Effect.gen(function* () {
|
|
expect(yield* status()).toEqual([])
|
|
}),
|
|
)
|
|
|
|
it.instance(
|
|
"returns empty for clean repo",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
expect(yield* status()).toEqual([])
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"parses binary numstat as 0",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "data.bin")
|
|
yield* Effect.promise(() => fs.writeFile(filepath, Buffer.from(Array.from({ length: 256 }, (_, index) => index))))
|
|
yield* gitAddAll(test.directory)
|
|
yield* gitCommit(test.directory, "add binary")
|
|
yield* Effect.promise(() => fs.writeFile(filepath, Buffer.from(Array.from({ length: 512 }, (_, index) => index % 256))))
|
|
|
|
const result = yield* status()
|
|
const entry = result.find((file) => file.path === "data.bin")
|
|
expect(entry).toBeDefined()
|
|
expect(entry!.status).toBe("modified")
|
|
expect(entry!.added).toBe(0)
|
|
expect(entry!.removed).toBe(0)
|
|
}),
|
|
{ git: true },
|
|
)
|
|
})
|
|
|
|
describe("list()", () => {
|
|
it.instance(
|
|
"returns files and directories with correct shape",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.mkdir(path.join(test.directory, "subdir")))
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "file.txt"), "content", "utf-8"))
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "subdir", "nested.txt"), "nested", "utf-8"))
|
|
|
|
const nodes = yield* list()
|
|
expect(nodes.length).toBeGreaterThanOrEqual(2)
|
|
for (const node of nodes) {
|
|
expect(node).toHaveProperty("name")
|
|
expect(node).toHaveProperty("path")
|
|
expect(node).toHaveProperty("absolute")
|
|
expect(node).toHaveProperty("type")
|
|
expect(node).toHaveProperty("ignored")
|
|
expect(["file", "directory"]).toContain(node.type)
|
|
}
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"sorts directories before files, alphabetical within each",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.mkdir(path.join(test.directory, "beta")))
|
|
yield* Effect.promise(() => fs.mkdir(path.join(test.directory, "alpha")))
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "zz.txt"), "", "utf-8"))
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "aa.txt"), "", "utf-8"))
|
|
|
|
const nodes = yield* list()
|
|
const dirs = nodes.filter((node) => node.type === "directory")
|
|
const files = nodes.filter((node) => node.type === "file")
|
|
const firstFile = nodes.findIndex((node) => node.type === "file")
|
|
const lastDir = nodes.findLastIndex((node) => node.type === "directory")
|
|
if (lastDir >= 0 && firstFile >= 0) {
|
|
expect(lastDir).toBeLessThan(firstFile)
|
|
}
|
|
expect(dirs.map((dir) => dir.name)).toEqual(dirs.map((dir) => dir.name).toSorted())
|
|
expect(files.map((file) => file.name)).toEqual(files.map((file) => file.name).toSorted())
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"excludes .git and .DS_Store",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, ".DS_Store"), "", "utf-8"))
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "visible.txt"), "", "utf-8"))
|
|
|
|
const names = (yield* list()).map((node) => node.name)
|
|
expect(names).not.toContain(".git")
|
|
expect(names).not.toContain(".DS_Store")
|
|
expect(names).toContain("visible.txt")
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"marks gitignored files as ignored",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, ".gitignore"), "*.log\nbuild/\n", "utf-8"))
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "app.log"), "log data", "utf-8"))
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "main.ts"), "code", "utf-8"))
|
|
yield* Effect.promise(() => fs.mkdir(path.join(test.directory, "build")))
|
|
|
|
const nodes = yield* list()
|
|
expect(nodes.find((node) => node.name === "app.log")?.ignored).toBe(true)
|
|
expect(nodes.find((node) => node.name === "main.ts")?.ignored).toBe(false)
|
|
expect(nodes.find((node) => node.name === "build")?.ignored).toBe(true)
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"lists subdirectory contents",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.mkdir(path.join(test.directory, "sub")))
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "sub", "a.txt"), "", "utf-8"))
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "sub", "b.txt"), "", "utf-8"))
|
|
|
|
const nodes = yield* list("sub")
|
|
expect(nodes.length).toBe(2)
|
|
expect(nodes.map((node) => node.name).sort()).toEqual(["a.txt", "b.txt"])
|
|
expect(nodes[0].path.replaceAll("\\", "/").startsWith("sub/")).toBe(true)
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"throws for paths outside project directory",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
expect(yield* failureMessage(list("../outside"))).toContain("Access denied")
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance("works without git", () =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "file.txt"), "hi", "utf-8"))
|
|
|
|
const nodes = yield* list()
|
|
expect(nodes.length).toBeGreaterThanOrEqual(1)
|
|
for (const node of nodes) {
|
|
expect(node.ignored).toBe(false)
|
|
}
|
|
}),
|
|
)
|
|
})
|
|
|
|
describe("search()", () => {
|
|
it.instance(
|
|
"empty query returns files",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* setupSearchableRepo(test.directory)
|
|
yield* init()
|
|
|
|
const result = yield* search({ query: "", type: "file" })
|
|
expect(result.length).toBeGreaterThan(0)
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"search works before explicit init",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* setupSearchableRepo(test.directory)
|
|
|
|
const result = yield* search({ query: "main", type: "file" })
|
|
expect(result.some((file) => file.includes("main"))).toBe(true)
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"empty query returns dirs sorted with hidden last",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* setupSearchableRepo(test.directory)
|
|
yield* init()
|
|
|
|
const result = yield* search({ query: "", type: "directory" })
|
|
expect(result.length).toBeGreaterThan(0)
|
|
const firstHidden = result.findIndex((dir) => dir.split("/").some((part) => part.startsWith(".") && part.length > 1))
|
|
const lastVisible = result.findLastIndex((dir) => !dir.split("/").some((part) => part.startsWith(".") && part.length > 1))
|
|
if (firstHidden >= 0 && lastVisible >= 0) {
|
|
expect(firstHidden).toBeGreaterThan(lastVisible)
|
|
}
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"fuzzy matches file names",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* setupSearchableRepo(test.directory)
|
|
yield* init()
|
|
|
|
const result = yield* search({ query: "main", type: "file" })
|
|
expect(result.some((file) => file.includes("main"))).toBe(true)
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"type filter returns only files",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* setupSearchableRepo(test.directory)
|
|
yield* init()
|
|
|
|
const result = yield* search({ query: "", type: "file" })
|
|
for (const file of result) {
|
|
expect(file.endsWith("/")).toBe(false)
|
|
}
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"type filter returns only directories",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* setupSearchableRepo(test.directory)
|
|
yield* init()
|
|
|
|
const result = yield* search({ query: "", type: "directory" })
|
|
for (const dir of result) {
|
|
expect(dir.endsWith("/")).toBe(true)
|
|
}
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"respects limit",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* setupSearchableRepo(test.directory)
|
|
yield* init()
|
|
|
|
const result = yield* search({ query: "", type: "file", limit: 2 })
|
|
expect(result.length).toBeLessThanOrEqual(2)
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"query starting with dot prefers hidden files",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* setupSearchableRepo(test.directory)
|
|
yield* init()
|
|
|
|
const result = yield* search({ query: ".hidden", type: "directory" })
|
|
expect(result.length).toBeGreaterThan(0)
|
|
expect(result[0]).toContain(".hidden")
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"search refreshes after init when files change",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* setupSearchableRepo(test.directory)
|
|
yield* init()
|
|
expect(yield* search({ query: "fresh", type: "file" })).toEqual([])
|
|
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "fresh.ts"), "fresh", "utf-8"))
|
|
|
|
expect(yield* search({ query: "fresh", type: "file" })).toContain("fresh.ts")
|
|
}),
|
|
{ git: true },
|
|
)
|
|
})
|
|
|
|
describe("read() - diff/patch", () => {
|
|
it.instance(
|
|
"returns diff and patch for modified tracked file",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "file.txt")
|
|
yield* Effect.promise(() => fs.writeFile(filepath, "original content\n", "utf-8"))
|
|
yield* gitAddAll(test.directory)
|
|
yield* gitCommit(test.directory, "add file")
|
|
yield* Effect.promise(() => fs.writeFile(filepath, "modified content\n", "utf-8"))
|
|
|
|
const result = yield* read("file.txt")
|
|
expect(result.type).toBe("text")
|
|
expect(result.content).toBe("modified content")
|
|
expect(result.diff).toBeDefined()
|
|
expect(result.diff).toContain("original content")
|
|
expect(result.diff).toContain("modified content")
|
|
expect(result.patch).toBeDefined()
|
|
expect(result.patch!.hunks.length).toBeGreaterThan(0)
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"returns diff for staged changes",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "staged.txt")
|
|
yield* Effect.promise(() => fs.writeFile(filepath, "before\n", "utf-8"))
|
|
yield* gitAddAll(test.directory)
|
|
yield* gitCommit(test.directory, "add file")
|
|
yield* Effect.promise(() => fs.writeFile(filepath, "after\n", "utf-8"))
|
|
yield* gitAddAll(test.directory)
|
|
|
|
const result = yield* read("staged.txt")
|
|
expect(result.diff).toBeDefined()
|
|
expect(result.patch).toBeDefined()
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"returns no diff for unmodified file",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
const filepath = path.join(test.directory, "clean.txt")
|
|
yield* Effect.promise(() => fs.writeFile(filepath, "unchanged\n", "utf-8"))
|
|
yield* gitAddAll(test.directory)
|
|
yield* gitCommit(test.directory, "add file")
|
|
|
|
const result = yield* read("clean.txt")
|
|
expect(result.type).toBe("text")
|
|
expect(result.content).toBe("unchanged")
|
|
expect(result.diff).toBeUndefined()
|
|
expect(result.patch).toBeUndefined()
|
|
}),
|
|
{ git: true },
|
|
)
|
|
})
|
|
|
|
describe("InstanceState isolation", () => {
|
|
it.instance(
|
|
"two directories get independent file caches",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const one = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(one.directory, "a.ts"), "one", "utf-8"))
|
|
yield* init()
|
|
expect(yield* search({ query: "a.ts", type: "file" })).toContain("a.ts")
|
|
expect(yield* search({ query: "b.ts", type: "file" })).not.toContain("b.ts")
|
|
|
|
yield* Effect.gen(function* () {
|
|
const two = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(two.directory, "b.ts"), "two", "utf-8"))
|
|
yield* init()
|
|
expect(yield* search({ query: "b.ts", type: "file" })).toContain("b.ts")
|
|
expect(yield* search({ query: "a.ts", type: "file" })).not.toContain("a.ts")
|
|
}).pipe(withTmpdirInstance({ git: true }))
|
|
}),
|
|
{ git: true },
|
|
)
|
|
|
|
it.instance(
|
|
"disposal gives fresh state on next access",
|
|
() =>
|
|
Effect.gen(function* () {
|
|
const test = yield* TestInstance
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "before.ts"), "before", "utf-8"))
|
|
yield* init()
|
|
expect(yield* search({ query: "before", type: "file" })).toContain("before.ts")
|
|
|
|
yield* Effect.promise(() => disposeAllInstances())
|
|
|
|
yield* Effect.promise(() => fs.writeFile(path.join(test.directory, "after.ts"), "after", "utf-8"))
|
|
yield* Effect.promise(() => fs.rm(path.join(test.directory, "before.ts")))
|
|
|
|
yield* init()
|
|
expect(yield* search({ query: "after", type: "file" })).toContain("after.ts")
|
|
expect(yield* search({ query: "before", type: "file" })).not.toContain("before.ts")
|
|
}),
|
|
{ git: true },
|
|
)
|
|
})
|
|
})
|