import { describe, expect, test } from "bun:test" import { Project } from "../../src/project/project" import { Log } from "../../src/util/log" import { $ } from "bun" import path from "path" import { tmpdir } from "../fixture/fixture" import { GlobalBus } from "../../src/bus/global" import { ProjectID } from "../../src/project/schema" import { Effect, Layer, Stream } from "effect" import { ChildProcess, ChildProcessSpawner } from "effect/unstable/process" import { NodeFileSystem, NodePath } from "@effect/platform-node" import { AppFileSystem } from "../../src/filesystem" import * as CrossSpawnSpawner from "../../src/effect/cross-spawn-spawner" Log.init({ print: false }) const encoder = new TextEncoder() /** * Creates a mock ChildProcessSpawner layer that intercepts git subcommands * matching `failArg` and returns exit code 128, while delegating everything * else to the real CrossSpawnSpawner. */ function mockGitFailure(failArg: string) { return Layer.effect( ChildProcessSpawner.ChildProcessSpawner, Effect.gen(function* () { const real = yield* ChildProcessSpawner.ChildProcessSpawner return ChildProcessSpawner.make( Effect.fnUntraced(function* (command) { const std = ChildProcess.isStandardCommand(command) ? command : undefined if (std?.command === "git" && std.args.some((a) => a === failArg)) { return ChildProcessSpawner.makeHandle({ pid: ChildProcessSpawner.ProcessId(0), exitCode: Effect.succeed(ChildProcessSpawner.ExitCode(128)), isRunning: Effect.succeed(false), kill: () => Effect.void, stdin: { [Symbol.for("effect/Sink/TypeId")]: Symbol.for("effect/Sink/TypeId") } as any, stdout: Stream.empty, stderr: Stream.make(encoder.encode("fatal: simulated failure\n")), all: Stream.empty, getInputFd: () => ({ [Symbol.for("effect/Sink/TypeId")]: Symbol.for("effect/Sink/TypeId") }) as any, getOutputFd: () => Stream.empty, }) } return yield* real.spawn(command) }), ) }), ).pipe(Layer.provide(CrossSpawnSpawner.layer), Layer.provide(NodeFileSystem.layer), Layer.provide(NodePath.layer)) } function projectLayerWithFailure(failArg: string) { return Project.layer.pipe( Layer.provide(mockGitFailure(failArg)), Layer.provide(AppFileSystem.defaultLayer), Layer.provide(NodePath.layer), ) } describe("Project.fromDirectory", () => { test("should handle git repository with no commits", async () => { await using tmp = await tmpdir() await $`git init`.cwd(tmp.path).quiet() const { project } = await Project.fromDirectory(tmp.path) expect(project).toBeDefined() expect(project.id).toBe(ProjectID.global) expect(project.vcs).toBe("git") expect(project.worktree).toBe(tmp.path) const opencodeFile = path.join(tmp.path, ".git", "opencode") expect(await Bun.file(opencodeFile).exists()).toBe(false) }) test("should handle git repository with commits", async () => { await using tmp = await tmpdir({ git: true }) const { project } = await Project.fromDirectory(tmp.path) expect(project).toBeDefined() expect(project.id).not.toBe(ProjectID.global) expect(project.vcs).toBe("git") expect(project.worktree).toBe(tmp.path) const opencodeFile = path.join(tmp.path, ".git", "opencode") expect(await Bun.file(opencodeFile).exists()).toBe(true) }) test("returns global for non-git directory", async () => { await using tmp = await tmpdir() const { project } = await Project.fromDirectory(tmp.path) expect(project.id).toBe(ProjectID.global) }) test("derives stable project ID from root commit", async () => { await using tmp = await tmpdir({ git: true }) const { project: a } = await Project.fromDirectory(tmp.path) const { project: b } = await Project.fromDirectory(tmp.path) expect(b.id).toBe(a.id) }) }) describe("Project.fromDirectory git failure paths", () => { test("keeps vcs when rev-list exits non-zero (no commits)", async () => { await using tmp = await tmpdir() await $`git init`.cwd(tmp.path).quiet() // rev-list fails because HEAD doesn't exist yet — this is the natural scenario const { project } = await Project.fromDirectory(tmp.path) expect(project.vcs).toBe("git") expect(project.id).toBe(ProjectID.global) expect(project.worktree).toBe(tmp.path) }) test("handles show-toplevel failure gracefully", async () => { await using tmp = await tmpdir({ git: true }) const layer = projectLayerWithFailure("--show-toplevel") const { project, sandbox } = await Effect.runPromise( Project.Service.use((svc) => svc.fromDirectory(tmp.path)).pipe(Effect.provide(layer)), ) expect(project.worktree).toBe(tmp.path) expect(sandbox).toBe(tmp.path) }) test("handles git-common-dir failure gracefully", async () => { await using tmp = await tmpdir({ git: true }) const layer = projectLayerWithFailure("--git-common-dir") const { project, sandbox } = await Effect.runPromise( Project.Service.use((svc) => svc.fromDirectory(tmp.path)).pipe(Effect.provide(layer)), ) expect(project.worktree).toBe(tmp.path) expect(sandbox).toBe(tmp.path) }) }) describe("Project.fromDirectory with worktrees", () => { test("should set worktree to root when called from root", async () => { await using tmp = await tmpdir({ git: true }) const { project, sandbox } = await Project.fromDirectory(tmp.path) expect(project.worktree).toBe(tmp.path) expect(sandbox).toBe(tmp.path) expect(project.sandboxes).not.toContain(tmp.path) }) test("should set worktree to root when called from a worktree", async () => { await using tmp = await tmpdir({ git: true }) const worktreePath = path.join(tmp.path, "..", path.basename(tmp.path) + "-worktree") try { await $`git worktree add ${worktreePath} -b test-branch-${Date.now()}`.cwd(tmp.path).quiet() const { project, sandbox } = await Project.fromDirectory(worktreePath) expect(project.worktree).toBe(tmp.path) expect(sandbox).toBe(worktreePath) expect(project.sandboxes).toContain(worktreePath) expect(project.sandboxes).not.toContain(tmp.path) } finally { await $`git worktree remove ${worktreePath}` .cwd(tmp.path) .quiet() .catch(() => {}) } }) test("worktree should share project ID with main repo", async () => { await using tmp = await tmpdir({ git: true }) const { project: main } = await Project.fromDirectory(tmp.path) const worktreePath = path.join(tmp.path, "..", path.basename(tmp.path) + "-wt-shared") try { await $`git worktree add ${worktreePath} -b shared-${Date.now()}`.cwd(tmp.path).quiet() const { project: wt } = await Project.fromDirectory(worktreePath) expect(wt.id).toBe(main.id) // Cache should live in the common .git dir, not the worktree's .git file const cache = path.join(tmp.path, ".git", "opencode") const exists = await Bun.file(cache).exists() expect(exists).toBe(true) } finally { await $`git worktree remove ${worktreePath}` .cwd(tmp.path) .quiet() .catch(() => {}) } }) test("separate clones of the same repo should share project ID", async () => { await using tmp = await tmpdir({ git: true }) // Create a bare remote, push, then clone into a second directory const bare = tmp.path + "-bare" const clone = tmp.path + "-clone" try { await $`git clone --bare ${tmp.path} ${bare}`.quiet() await $`git clone ${bare} ${clone}`.quiet() const { project: a } = await Project.fromDirectory(tmp.path) const { project: b } = await Project.fromDirectory(clone) expect(b.id).toBe(a.id) } finally { await $`rm -rf ${bare} ${clone}`.quiet().nothrow() } }) test("should accumulate multiple worktrees in sandboxes", async () => { await using tmp = await tmpdir({ git: true }) const worktree1 = path.join(tmp.path, "..", path.basename(tmp.path) + "-wt1") const worktree2 = path.join(tmp.path, "..", path.basename(tmp.path) + "-wt2") try { await $`git worktree add ${worktree1} -b branch-${Date.now()}`.cwd(tmp.path).quiet() await $`git worktree add ${worktree2} -b branch-${Date.now() + 1}`.cwd(tmp.path).quiet() await Project.fromDirectory(worktree1) const { project } = await Project.fromDirectory(worktree2) expect(project.worktree).toBe(tmp.path) expect(project.sandboxes).toContain(worktree1) expect(project.sandboxes).toContain(worktree2) expect(project.sandboxes).not.toContain(tmp.path) } finally { await $`git worktree remove ${worktree1}` .cwd(tmp.path) .quiet() .catch(() => {}) await $`git worktree remove ${worktree2}` .cwd(tmp.path) .quiet() .catch(() => {}) } }) }) describe("Project.discover", () => { test("should discover favicon.png in root", async () => { await using tmp = await tmpdir({ git: true }) const { project } = await Project.fromDirectory(tmp.path) const pngData = Buffer.from([0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a]) await Bun.write(path.join(tmp.path, "favicon.png"), pngData) await Project.discover(project) const updated = Project.get(project.id) expect(updated).toBeDefined() expect(updated!.icon).toBeDefined() expect(updated!.icon?.url).toStartWith("data:") expect(updated!.icon?.url).toContain("base64") expect(updated!.icon?.color).toBeUndefined() }) test("should not discover non-image files", async () => { await using tmp = await tmpdir({ git: true }) const { project } = await Project.fromDirectory(tmp.path) await Bun.write(path.join(tmp.path, "favicon.txt"), "not an image") await Project.discover(project) const updated = Project.get(project.id) expect(updated).toBeDefined() expect(updated!.icon).toBeUndefined() }) }) describe("Project.update", () => { test("should update name", async () => { await using tmp = await tmpdir({ git: true }) const { project } = await Project.fromDirectory(tmp.path) const updated = await Project.update({ projectID: project.id, name: "New Project Name", }) expect(updated.name).toBe("New Project Name") const fromDb = Project.get(project.id) expect(fromDb?.name).toBe("New Project Name") }) test("should update icon url", async () => { await using tmp = await tmpdir({ git: true }) const { project } = await Project.fromDirectory(tmp.path) const updated = await Project.update({ projectID: project.id, icon: { url: "https://example.com/icon.png" }, }) expect(updated.icon?.url).toBe("https://example.com/icon.png") const fromDb = Project.get(project.id) expect(fromDb?.icon?.url).toBe("https://example.com/icon.png") }) test("should update icon color", async () => { await using tmp = await tmpdir({ git: true }) const { project } = await Project.fromDirectory(tmp.path) const updated = await Project.update({ projectID: project.id, icon: { color: "#ff0000" }, }) expect(updated.icon?.color).toBe("#ff0000") const fromDb = Project.get(project.id) expect(fromDb?.icon?.color).toBe("#ff0000") }) test("should update commands", async () => { await using tmp = await tmpdir({ git: true }) const { project } = await Project.fromDirectory(tmp.path) const updated = await Project.update({ projectID: project.id, commands: { start: "npm run dev" }, }) expect(updated.commands?.start).toBe("npm run dev") const fromDb = Project.get(project.id) expect(fromDb?.commands?.start).toBe("npm run dev") }) test("should throw error when project not found", async () => { await expect( Project.update({ projectID: ProjectID.make("nonexistent-project-id"), name: "Should Fail", }), ).rejects.toThrow("Project not found: nonexistent-project-id") }) test("should emit GlobalBus event on update", async () => { await using tmp = await tmpdir({ git: true }) const { project } = await Project.fromDirectory(tmp.path) let eventPayload: any = null const on = (data: any) => { eventPayload = data } GlobalBus.on("event", on) try { await Project.update({ projectID: project.id, name: "Updated Name", }) expect(eventPayload).not.toBeNull() expect(eventPayload.payload.type).toBe("project.updated") expect(eventPayload.payload.properties.name).toBe("Updated Name") } finally { GlobalBus.off("event", on) } }) test("should update multiple fields at once", async () => { await using tmp = await tmpdir({ git: true }) const { project } = await Project.fromDirectory(tmp.path) const updated = await Project.update({ projectID: project.id, name: "Multi Update", icon: { url: "https://example.com/favicon.ico", color: "#00ff00" }, commands: { start: "make start" }, }) expect(updated.name).toBe("Multi Update") expect(updated.icon?.url).toBe("https://example.com/favicon.ico") expect(updated.icon?.color).toBe("#00ff00") expect(updated.commands?.start).toBe("make start") }) }) describe("Project.list and Project.get", () => { test("list returns all projects", async () => { await using tmp = await tmpdir({ git: true }) const { project } = await Project.fromDirectory(tmp.path) const all = Project.list() expect(all.length).toBeGreaterThan(0) expect(all.find((p) => p.id === project.id)).toBeDefined() }) test("get returns project by id", async () => { await using tmp = await tmpdir({ git: true }) const { project } = await Project.fromDirectory(tmp.path) const found = Project.get(project.id) expect(found).toBeDefined() expect(found!.id).toBe(project.id) }) test("get returns undefined for unknown id", () => { const found = Project.get(ProjectID.make("nonexistent")) expect(found).toBeUndefined() }) }) describe("Project.setInitialized", () => { test("sets time_initialized on project", async () => { await using tmp = await tmpdir({ git: true }) const { project } = await Project.fromDirectory(tmp.path) expect(project.time.initialized).toBeUndefined() Project.setInitialized(project.id) const updated = Project.get(project.id) expect(updated?.time.initialized).toBeDefined() }) }) describe("Project.addSandbox and Project.removeSandbox", () => { test("addSandbox adds directory and removeSandbox removes it", async () => { await using tmp = await tmpdir({ git: true }) const { project } = await Project.fromDirectory(tmp.path) const sandboxDir = path.join(tmp.path, "sandbox-test") await Project.addSandbox(project.id, sandboxDir) let found = Project.get(project.id) expect(found?.sandboxes).toContain(sandboxDir) await Project.removeSandbox(project.id, sandboxDir) found = Project.get(project.id) expect(found?.sandboxes).not.toContain(sandboxDir) }) test("addSandbox emits GlobalBus event", async () => { await using tmp = await tmpdir({ git: true }) const { project } = await Project.fromDirectory(tmp.path) const sandboxDir = path.join(tmp.path, "sandbox-event") const events: any[] = [] const on = (evt: any) => events.push(evt) GlobalBus.on("event", on) await Project.addSandbox(project.id, sandboxDir) GlobalBus.off("event", on) expect(events.some((e) => e.payload.type === Project.Event.Updated.type)).toBe(true) }) })