| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348 |
- 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<string, string> }) => {
- 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<void>) {
- 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")
- })
- })
|