diff --git a/packages/opencode/test/file/ripgrep.test.ts b/packages/opencode/test/file/ripgrep.test.ts
index a76c7ebe26..d71ce205ea 100644
--- a/packages/opencode/test/file/ripgrep.test.ts
+++ b/packages/opencode/test/file/ripgrep.test.ts
@@ -1,214 +1,220 @@
-import { describe, expect, test } from "bun:test"
+import { describe, expect } from "bun:test"
import { Effect } from "effect"
import * as Stream from "effect/Stream"
import fs from "fs/promises"
+import os from "os"
import path from "path"
-import { tmpdir } from "../fixture/fixture"
import { Ripgrep } from "../../src/file/ripgrep"
+import { testEffect } from "../lib/effect"
-const run = (effect: Effect.Effect) =>
- effect.pipe(Effect.provide(Ripgrep.defaultLayer), Effect.runPromise)
+const it = testEffect(Ripgrep.defaultLayer)
+
+const tmpdir = (init?: (dir: string) => Effect.Effect) =>
+ Effect.acquireRelease(
+ Effect.promise(async () => fs.realpath(await fs.mkdtemp(path.join(os.tmpdir(), "opencode-test-")))),
+ (dir) =>
+ Effect.promise(() =>
+ fs.rm(dir, {
+ recursive: true,
+ force: true,
+ maxRetries: 5,
+ retryDelay: 100,
+ }),
+ ).pipe(Effect.ignore),
+ ).pipe(Effect.tap((dir) => init?.(dir) ?? Effect.void))
+
+const write = (file: string, data: string) => Effect.promise(() => Bun.write(file, data))
+const mkdir = (dir: string) => Effect.promise(() => fs.mkdir(dir, { recursive: true }))
+const collectFiles = (input: Ripgrep.FilesInput) =>
+ Ripgrep.Service.use((rg) =>
+ rg.files(input).pipe(
+ Stream.runCollect,
+ Effect.map((c) => [...c]),
+ ),
+ )
+
+const withRipgrepConfig = (value: string, effect: Effect.Effect) =>
+ Effect.acquireUseRelease(
+ Effect.sync(() => {
+ const prev = process.env["RIPGREP_CONFIG_PATH"]
+ process.env["RIPGREP_CONFIG_PATH"] = value
+ return prev
+ }),
+ () => effect,
+ (prev) =>
+ Effect.sync(() => {
+ if (prev === undefined) delete process.env["RIPGREP_CONFIG_PATH"]
+ else process.env["RIPGREP_CONFIG_PATH"] = prev
+ }),
+ )
describe("file.ripgrep", () => {
- test("defaults to include hidden", async () => {
- await using tmp = await tmpdir({
- init: async (dir) => {
- await Bun.write(path.join(dir, "visible.txt"), "hello")
- await fs.mkdir(path.join(dir, ".opencode"), { recursive: true })
- await Bun.write(path.join(dir, ".opencode", "thing.json"), "{}")
- },
- })
+ it.live("defaults to include hidden", () =>
+ Effect.gen(function* () {
+ const dir = yield* tmpdir((dir) =>
+ Effect.gen(function* () {
+ yield* write(path.join(dir, "visible.txt"), "hello")
+ yield* mkdir(path.join(dir, ".opencode"))
+ yield* write(path.join(dir, ".opencode", "thing.json"), "{}")
+ }),
+ )
- const files = await run(
- Ripgrep.Service.use((rg) =>
- rg.files({ cwd: tmp.path }).pipe(
- Stream.runCollect,
- Effect.map((c) => [...c]),
- ),
- ),
- )
- expect(files.includes("visible.txt")).toBe(true)
- expect(files.includes(path.join(".opencode", "thing.json"))).toBe(true)
- })
+ const files = yield* collectFiles({ cwd: dir })
+ expect(files.includes("visible.txt")).toBe(true)
+ expect(files.includes(path.join(".opencode", "thing.json"))).toBe(true)
+ }),
+ )
- test("hidden false excludes hidden", async () => {
- await using tmp = await tmpdir({
- init: async (dir) => {
- await Bun.write(path.join(dir, "visible.txt"), "hello")
- await fs.mkdir(path.join(dir, ".opencode"), { recursive: true })
- await Bun.write(path.join(dir, ".opencode", "thing.json"), "{}")
- },
- })
+ it.live("hidden false excludes hidden", () =>
+ Effect.gen(function* () {
+ const dir = yield* tmpdir((dir) =>
+ Effect.gen(function* () {
+ yield* write(path.join(dir, "visible.txt"), "hello")
+ yield* mkdir(path.join(dir, ".opencode"))
+ yield* write(path.join(dir, ".opencode", "thing.json"), "{}")
+ }),
+ )
- const files = await run(
- Ripgrep.Service.use((rg) =>
- rg.files({ cwd: tmp.path, hidden: false }).pipe(
- Stream.runCollect,
- Effect.map((c) => [...c]),
- ),
- ),
- )
- expect(files.includes("visible.txt")).toBe(true)
- expect(files.includes(path.join(".opencode", "thing.json"))).toBe(false)
- })
+ const files = yield* collectFiles({ cwd: dir, hidden: false })
+ expect(files.includes("visible.txt")).toBe(true)
+ expect(files.includes(path.join(".opencode", "thing.json"))).toBe(false)
+ }),
+ )
- test("search returns empty when nothing matches", async () => {
- await using tmp = await tmpdir({
- init: async (dir) => {
- await Bun.write(path.join(dir, "match.ts"), "const value = 'other'\n")
- },
- })
+ it.live("search returns empty when nothing matches", () =>
+ Effect.gen(function* () {
+ const dir = yield* tmpdir((dir) => write(path.join(dir, "match.ts"), "const value = 'other'\n"))
- const result = await run(Ripgrep.Service.use((rg) => rg.search({ cwd: tmp.path, pattern: "needle" })))
- expect(result.partial).toBe(false)
- expect(result.items).toEqual([])
- })
+ const result = yield* Ripgrep.Service.use((rg) => rg.search({ cwd: dir, pattern: "needle" }))
+ expect(result.partial).toBe(false)
+ expect(result.items).toEqual([])
+ }),
+ )
- test("search returns match metadata with normalized path", async () => {
- await using tmp = await tmpdir({
- init: async (dir) => {
- await fs.mkdir(path.join(dir, "src"), { recursive: true })
- await Bun.write(path.join(dir, "src", "match.ts"), "const needle = 1\n")
- },
- })
+ it.live("search returns match metadata with normalized path", () =>
+ Effect.gen(function* () {
+ const dir = yield* tmpdir((dir) =>
+ Effect.gen(function* () {
+ yield* mkdir(path.join(dir, "src"))
+ yield* write(path.join(dir, "src", "match.ts"), "const needle = 1\n")
+ }),
+ )
- const result = await run(Ripgrep.Service.use((rg) => rg.search({ cwd: tmp.path, pattern: "needle" })))
- expect(result.partial).toBe(false)
- expect(result.items).toHaveLength(1)
- expect(result.items[0]?.path.text).toBe(path.join("src", "match.ts"))
- expect(result.items[0]?.line_number).toBe(1)
- expect(result.items[0]?.lines.text).toContain("needle")
- })
-
- test("search returns matched rows with glob filter", async () => {
- await using tmp = await tmpdir({
- init: async (dir) => {
- await Bun.write(path.join(dir, "match.ts"), "const value = 'needle'\n")
- await Bun.write(path.join(dir, "skip.txt"), "const value = 'other'\n")
- },
- })
-
- const result = await run(
- Ripgrep.Service.use((rg) => rg.search({ cwd: tmp.path, pattern: "needle", glob: ["*.ts"] })),
- )
- expect(result.partial).toBe(false)
- expect(result.items).toHaveLength(1)
- expect(result.items[0]?.path.text).toContain("match.ts")
- expect(result.items[0]?.lines.text).toContain("needle")
- })
-
- test("search supports explicit file targets", async () => {
- await using tmp = await tmpdir({
- init: async (dir) => {
- await Bun.write(path.join(dir, "match.ts"), "const value = 'needle'\n")
- await Bun.write(path.join(dir, "skip.ts"), "const value = 'needle'\n")
- },
- })
-
- const file = path.join(tmp.path, "match.ts")
- const result = await run(Ripgrep.Service.use((rg) => rg.search({ cwd: tmp.path, pattern: "needle", file: [file] })))
- expect(result.partial).toBe(false)
- expect(result.items).toHaveLength(1)
- expect(result.items[0]?.path.text).toBe(file)
- })
-
- test("files returns empty when glob matches no files", async () => {
- await using tmp = await tmpdir({
- init: async (dir) => {
- await fs.mkdir(path.join(dir, "packages", "console"), { recursive: true })
- await Bun.write(path.join(dir, "packages", "console", "package.json"), "{}")
- },
- })
-
- const files = await run(
- Ripgrep.Service.use((rg) =>
- rg.files({ cwd: tmp.path, glob: ["packages/*"] }).pipe(
- Stream.runCollect,
- Effect.map((c) => [...c]),
- ),
- ),
- )
- expect(files).toEqual([])
- })
-
- test("files returns stream of filenames", async () => {
- await using tmp = await tmpdir({
- init: async (dir) => {
- await Bun.write(path.join(dir, "a.txt"), "hello")
- await Bun.write(path.join(dir, "b.txt"), "world")
- },
- })
-
- const files = await run(
- Ripgrep.Service.use((rg) =>
- rg.files({ cwd: tmp.path }).pipe(
- Stream.runCollect,
- Effect.map((c) => [...c].sort()),
- ),
- ),
- )
- expect(files).toEqual(["a.txt", "b.txt"])
- })
-
- test("files respects glob filter", async () => {
- await using tmp = await tmpdir({
- init: async (dir) => {
- await Bun.write(path.join(dir, "keep.ts"), "yes")
- await Bun.write(path.join(dir, "skip.txt"), "no")
- },
- })
-
- const files = await run(
- Ripgrep.Service.use((rg) =>
- rg.files({ cwd: tmp.path, glob: ["*.ts"] }).pipe(
- Stream.runCollect,
- Effect.map((c) => [...c]),
- ),
- ),
- )
- expect(files).toEqual(["keep.ts"])
- })
-
- test("files dies on nonexistent directory", async () => {
- const exit = await Ripgrep.Service.use((rg) =>
- rg.files({ cwd: "/tmp/nonexistent-dir-12345" }).pipe(Stream.runCollect),
- ).pipe(Effect.provide(Ripgrep.defaultLayer), Effect.runPromiseExit)
- expect(exit._tag).toBe("Failure")
- })
-
- test("ignores RIPGREP_CONFIG_PATH in direct mode", async () => {
- await using tmp = await tmpdir({
- init: async (dir) => {
- await Bun.write(path.join(dir, "match.ts"), "const needle = 1\n")
- },
- })
-
- const prev = process.env["RIPGREP_CONFIG_PATH"]
- process.env["RIPGREP_CONFIG_PATH"] = path.join(tmp.path, "missing-ripgreprc")
- try {
- const result = await run(Ripgrep.Service.use((rg) => rg.search({ cwd: tmp.path, pattern: "needle" })))
+ const result = yield* Ripgrep.Service.use((rg) => rg.search({ cwd: dir, pattern: "needle" }))
+ expect(result.partial).toBe(false)
expect(result.items).toHaveLength(1)
- } finally {
- if (prev === undefined) delete process.env["RIPGREP_CONFIG_PATH"]
- else process.env["RIPGREP_CONFIG_PATH"] = prev
- }
- })
+ expect(result.items[0]?.path.text).toBe(path.join("src", "match.ts"))
+ expect(result.items[0]?.line_number).toBe(1)
+ expect(result.items[0]?.lines.text).toContain("needle")
+ }),
+ )
- test("ignores RIPGREP_CONFIG_PATH in worker mode", async () => {
- await using tmp = await tmpdir({
- init: async (dir) => {
- await Bun.write(path.join(dir, "match.ts"), "const needle = 1\n")
- },
- })
+ it.live("search returns matched rows with glob filter", () =>
+ Effect.gen(function* () {
+ const dir = yield* tmpdir((dir) =>
+ Effect.gen(function* () {
+ yield* write(path.join(dir, "match.ts"), "const value = 'needle'\n")
+ yield* write(path.join(dir, "skip.txt"), "const value = 'other'\n")
+ }),
+ )
- const prev = process.env["RIPGREP_CONFIG_PATH"]
- process.env["RIPGREP_CONFIG_PATH"] = path.join(tmp.path, "missing-ripgreprc")
- try {
- const result = await run(Ripgrep.Service.use((rg) => rg.search({ cwd: tmp.path, pattern: "needle" })))
+ const result = yield* Ripgrep.Service.use((rg) => rg.search({ cwd: dir, pattern: "needle", glob: ["*.ts"] }))
+ expect(result.partial).toBe(false)
expect(result.items).toHaveLength(1)
- } finally {
- if (prev === undefined) delete process.env["RIPGREP_CONFIG_PATH"]
- else process.env["RIPGREP_CONFIG_PATH"] = prev
- }
- })
+ expect(result.items[0]?.path.text).toContain("match.ts")
+ expect(result.items[0]?.lines.text).toContain("needle")
+ }),
+ )
+
+ it.live("search supports explicit file targets", () =>
+ Effect.gen(function* () {
+ const dir = yield* tmpdir((dir) =>
+ Effect.gen(function* () {
+ yield* write(path.join(dir, "match.ts"), "const value = 'needle'\n")
+ yield* write(path.join(dir, "skip.ts"), "const value = 'needle'\n")
+ }),
+ )
+
+ const file = path.join(dir, "match.ts")
+ const result = yield* Ripgrep.Service.use((rg) => rg.search({ cwd: dir, pattern: "needle", file: [file] }))
+ expect(result.partial).toBe(false)
+ expect(result.items).toHaveLength(1)
+ expect(result.items[0]?.path.text).toBe(file)
+ }),
+ )
+
+ it.live("files returns empty when glob matches no files", () =>
+ Effect.gen(function* () {
+ const dir = yield* tmpdir((dir) =>
+ Effect.gen(function* () {
+ yield* mkdir(path.join(dir, "packages", "console"))
+ yield* write(path.join(dir, "packages", "console", "package.json"), "{}")
+ }),
+ )
+
+ const files = yield* collectFiles({ cwd: dir, glob: ["packages/*"] })
+ expect(files).toEqual([])
+ }),
+ )
+
+ it.live("files returns stream of filenames", () =>
+ Effect.gen(function* () {
+ const dir = yield* tmpdir((dir) =>
+ Effect.gen(function* () {
+ yield* write(path.join(dir, "a.txt"), "hello")
+ yield* write(path.join(dir, "b.txt"), "world")
+ }),
+ )
+
+ const files = yield* collectFiles({ cwd: dir }).pipe(Effect.map((files) => files.sort()))
+ expect(files).toEqual(["a.txt", "b.txt"])
+ }),
+ )
+
+ it.live("files respects glob filter", () =>
+ Effect.gen(function* () {
+ const dir = yield* tmpdir((dir) =>
+ Effect.gen(function* () {
+ yield* write(path.join(dir, "keep.ts"), "yes")
+ yield* write(path.join(dir, "skip.txt"), "no")
+ }),
+ )
+
+ const files = yield* collectFiles({ cwd: dir, glob: ["*.ts"] })
+ expect(files).toEqual(["keep.ts"])
+ }),
+ )
+
+ it.live("files dies on nonexistent directory", () =>
+ Effect.gen(function* () {
+ const exit = yield* Ripgrep.Service.use((rg) =>
+ rg.files({ cwd: "/tmp/nonexistent-dir-12345" }).pipe(Stream.runCollect),
+ ).pipe(Effect.exit)
+ expect(exit._tag).toBe("Failure")
+ }),
+ )
+
+ it.live("ignores RIPGREP_CONFIG_PATH in direct mode", () =>
+ Effect.gen(function* () {
+ const dir = yield* tmpdir((dir) => write(path.join(dir, "match.ts"), "const needle = 1\n"))
+
+ const result = yield* withRipgrepConfig(
+ path.join(dir, "missing-ripgreprc"),
+ Ripgrep.Service.use((rg) => rg.search({ cwd: dir, pattern: "needle" })),
+ )
+ expect(result.items).toHaveLength(1)
+ }),
+ )
+
+ it.live("ignores RIPGREP_CONFIG_PATH in worker mode", () =>
+ Effect.gen(function* () {
+ const dir = yield* tmpdir((dir) => write(path.join(dir, "match.ts"), "const needle = 1\n"))
+
+ const result = yield* withRipgrepConfig(
+ path.join(dir, "missing-ripgreprc"),
+ Ripgrep.Service.use((rg) => rg.search({ cwd: dir, pattern: "needle" })),
+ )
+ expect(result.items).toHaveLength(1)
+ }),
+ )
})