| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474 |
- 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 { 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()
- function run<A>(fn: (svc: Project.Interface) => Effect.Effect<A>, layer = Project.defaultLayer) {
- return Effect.runPromise(
- Effect.gen(function* () {
- const svc = yield* Project.Service
- return yield* fn(svc)
- }).pipe(Effect.provide(layer)),
- )
- }
- /**
- * 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,
- unref: Effect.succeed(Effect.void),
- })
- }
- return yield* real.spawn(command)
- }),
- )
- }),
- ).pipe(Layer.provide(CrossSpawnSpawner.defaultLayer))
- }
- 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 run((svc) => svc.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 run((svc) => svc.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 kiloFile = path.join(tmp.path, ".git", "kilo")
- expect(await Bun.file(kiloFile).exists()).toBe(true)
- })
- test("returns global for non-git directory", async () => {
- await using tmp = await tmpdir()
- const { project } = await run((svc) => svc.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 run((svc) => svc.fromDirectory(tmp.path))
- const { project: b } = await run((svc) => svc.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 run((svc) => svc.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 run((svc) => svc.fromDirectory(tmp.path), 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 run((svc) => svc.fromDirectory(tmp.path), 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 run((svc) => svc.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 run((svc) => svc.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 run((svc) => svc.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 run((svc) => svc.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", "kilo")
- 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 run((svc) => svc.fromDirectory(tmp.path))
- const { project: b } = await run((svc) => svc.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 run((svc) => svc.fromDirectory(worktree1))
- const { project } = await run((svc) => svc.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 run((svc) => svc.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 run((svc) => svc.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 run((svc) => svc.fromDirectory(tmp.path))
- await Bun.write(path.join(tmp.path, "favicon.txt"), "not an image")
- await run((svc) => svc.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 run((svc) => svc.fromDirectory(tmp.path))
- const updated = await run((svc) =>
- svc.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 run((svc) => svc.fromDirectory(tmp.path))
- const updated = await run((svc) =>
- svc.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 run((svc) => svc.fromDirectory(tmp.path))
- const updated = await run((svc) =>
- svc.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 run((svc) => svc.fromDirectory(tmp.path))
- const updated = await run((svc) =>
- svc.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(
- run((svc) =>
- svc.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 run((svc) => svc.fromDirectory(tmp.path))
- let eventPayload: any = null
- const on = (data: any) => {
- eventPayload = data
- }
- GlobalBus.on("event", on)
- try {
- await run((svc) => svc.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 run((svc) => svc.fromDirectory(tmp.path))
- const updated = await run((svc) =>
- svc.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 run((svc) => svc.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 run((svc) => svc.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 run((svc) => svc.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 run((svc) => svc.fromDirectory(tmp.path))
- const sandboxDir = path.join(tmp.path, "sandbox-test")
- await run((svc) => svc.addSandbox(project.id, sandboxDir))
- let found = Project.get(project.id)
- expect(found?.sandboxes).toContain(sandboxDir)
- await run((svc) => svc.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 run((svc) => svc.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 run((svc) => svc.addSandbox(project.id, sandboxDir))
- GlobalBus.off("event", on)
- expect(events.some((e) => e.payload.type === Project.Event.Updated.type)).toBe(true)
- })
- })
|