import { describe, expect, mock, 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 { Filesystem } from "../../src/util/filesystem" import { GlobalBus } from "../../src/bus/global" Log.init({ print: false }) const gitModule = await import("../../src/util/git") const originalGit = gitModule.git type Mode = "none" | "rev-list-fail" | "top-fail" | "common-dir-fail" let mode: Mode = "none" mock.module("../../src/util/git", () => ({ git: (args: string[], opts: { cwd: string; env?: Record }) => { const cmd = ["git", ...args].join(" ") if ( mode === "rev-list-fail" && cmd.includes("git rev-list") && cmd.includes("--max-parents=0") && cmd.includes("--all") ) { return Promise.resolve({ exitCode: 128, text: () => Promise.resolve(""), stdout: Buffer.from(""), stderr: Buffer.from("fatal"), }) } if (mode === "top-fail" && cmd.includes("git rev-parse") && cmd.includes("--show-toplevel")) { return Promise.resolve({ exitCode: 128, text: () => Promise.resolve(""), stdout: Buffer.from(""), stderr: Buffer.from("fatal"), }) } if (mode === "common-dir-fail" && cmd.includes("git rev-parse") && cmd.includes("--git-common-dir")) { return Promise.resolve({ exitCode: 128, text: () => Promise.resolve(""), stdout: Buffer.from(""), stderr: Buffer.from("fatal"), }) } return originalGit(args, opts) }, })) async function withMode(next: Mode, run: () => Promise) { const prev = mode mode = next try { await run() } finally { mode = prev } } async function loadProject() { return (await import("../../src/project/project")).Project } describe("Project.fromDirectory", () => { test("should handle git repository with no commits", async () => { const p = await loadProject() await using tmp = await tmpdir() await $`git init`.cwd(tmp.path).quiet() const { project } = await p.fromDirectory(tmp.path) expect(project).toBeDefined() expect(project.id).toBe("global") expect(project.vcs).toBe("git") expect(project.worktree).toBe(tmp.path) const opencodeFile = path.join(tmp.path, ".git", "opencode") const fileExists = await Filesystem.exists(opencodeFile) expect(fileExists).toBe(false) }) test("should handle git repository with commits", async () => { const p = await loadProject() await using tmp = await tmpdir({ git: true }) const { project } = await p.fromDirectory(tmp.path) expect(project).toBeDefined() expect(project.id).not.toBe("global") expect(project.vcs).toBe("git") expect(project.worktree).toBe(tmp.path) const opencodeFile = path.join(tmp.path, ".git", "opencode") const fileExists = await Filesystem.exists(opencodeFile) expect(fileExists).toBe(true) }) test("keeps git vcs when rev-list exits non-zero with empty output", async () => { const p = await loadProject() await using tmp = await tmpdir() await $`git init`.cwd(tmp.path).quiet() await withMode("rev-list-fail", async () => { const { project } = await p.fromDirectory(tmp.path) expect(project.vcs).toBe("git") expect(project.id).toBe("global") expect(project.worktree).toBe(tmp.path) }) }) test("keeps git vcs when show-toplevel exits non-zero with empty output", async () => { const p = await loadProject() await using tmp = await tmpdir({ git: true }) await withMode("top-fail", async () => { const { project, sandbox } = await p.fromDirectory(tmp.path) expect(project.vcs).toBe("git") expect(project.worktree).toBe(tmp.path) expect(sandbox).toBe(tmp.path) }) }) test("keeps git vcs when git-common-dir exits non-zero with empty output", async () => { const p = await loadProject() await using tmp = await tmpdir({ git: true }) await withMode("common-dir-fail", async () => { const { project, sandbox } = await p.fromDirectory(tmp.path) expect(project.vcs).toBe("git") 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 () => { const p = await loadProject() await using tmp = await tmpdir({ git: true }) const { project, sandbox } = await p.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 () => { const p = await loadProject() 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 p.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("should accumulate multiple worktrees in sandboxes", async () => { const p = await loadProject() 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 p.fromDirectory(worktree1) const { project } = await p.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 () => { const p = await loadProject() await using tmp = await tmpdir({ git: true }) const { project } = await p.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 p.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 () => { const p = await loadProject() await using tmp = await tmpdir({ git: true }) const { project } = await p.fromDirectory(tmp.path) await Bun.write(path.join(tmp.path, "favicon.txt"), "not an image") await p.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 using tmp = await tmpdir({ git: true }) await expect( Project.update({ projectID: "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 eventFired = false let eventPayload: any = null GlobalBus.on("event", (data) => { eventFired = true eventPayload = data }) await Project.update({ projectID: project.id, name: "Updated Name", }) expect(eventFired).toBe(true) expect(eventPayload.payload.type).toBe("project.updated") expect(eventPayload.payload.properties.name).toBe("Updated Name") }) 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") }) })