opencode/packages/opencode/test/server/project-init-git.test.ts

120 lines
4.2 KiB
TypeScript

import { afterEach, describe, expect } from "bun:test"
import { AppFileSystem } from "@opencode-ai/core/filesystem"
import { Effect, Layer } from "effect"
import path from "path"
import { InstanceRef } from "../../src/effect/instance-ref"
import { InstanceBootstrap } from "../../src/project/bootstrap-service"
import { InstanceStore } from "../../src/project/instance-store"
import { GlobalBus, type GlobalEvent } from "../../src/bus/global"
import { Snapshot } from "../../src/snapshot"
import { Server } from "../../src/server/server"
import * as Log from "@opencode-ai/core/util/log"
import { resetDatabase } from "../fixture/db"
import { disposeAllInstances, TestInstance } from "../fixture/fixture"
import { testEffect } from "../lib/effect"
void Log.init({ print: false })
afterEach(async () => {
await disposeAllInstances()
await resetDatabase()
})
const noopBootstrap = Layer.succeed(InstanceBootstrap.Service, InstanceBootstrap.Service.of({ run: Effect.void }))
const testInstanceStore = InstanceStore.defaultLayer.pipe(Layer.provide(noopBootstrap))
const it = testEffect(Layer.mergeAll(AppFileSystem.defaultLayer, Snapshot.defaultLayer, testInstanceStore))
function request(directory: string, url: string, init: RequestInit = {}) {
return Effect.promise(() => {
const headers = new Headers(init.headers)
headers.set("x-opencode-directory", directory)
return Promise.resolve(Server.Default().app.request(url, { ...init, headers }))
})
}
function json<T>(response: Response) {
return Effect.promise(() => response.json() as Promise<T>)
}
function collectGlobalEvents() {
return Effect.acquireRelease(
Effect.sync(() => {
const seen: GlobalEvent[] = []
const on = (event: GlobalEvent) => {
seen.push(event)
}
GlobalBus.on("event", on)
return { seen, on }
}),
({ on }) => Effect.sync(() => GlobalBus.off("event", on)),
)
}
const disposedEvents = (seen: GlobalEvent[], dir: string) =>
seen.filter((evt) => evt.directory === dir && evt.payload.type === "server.instance.disposed").length
describe("project.initGit endpoint", () => {
it.instance("initializes git and reloads immediately", () =>
Effect.gen(function* () {
const tmp = yield* TestInstance
const fs = yield* AppFileSystem.Service
const events = yield* collectGlobalEvents()
const init = yield* request(tmp.directory, "/project/git/init", {
method: "POST",
})
const body = yield* json(init)
expect(init.status).toBe(200)
expect(body).toMatchObject({
id: "global",
vcs: "git",
worktree: tmp.directory,
})
// Reload behavior: bus emits exactly one server.instance.disposed for the directory.
expect(disposedEvents(events.seen, tmp.directory)).toBe(1)
expect(yield* fs.exists(path.join(tmp.directory, ".git", "opencode"))).toBe(false)
const current = yield* request(tmp.directory, "/project/current")
expect(current.status).toBe(200)
expect(yield* json(current)).toMatchObject({
id: "global",
vcs: "git",
worktree: tmp.directory,
})
const ctx = yield* InstanceStore.Service.use((store) => store.reload({ directory: tmp.directory }))
const tracked = yield* Snapshot.Service.use((snapshot) => snapshot.track()).pipe(
Effect.provideService(InstanceRef, ctx),
)
expect(tracked).toBeTruthy()
}),
)
it.instance(
"does not reload when the project is already git",
() =>
Effect.gen(function* () {
const tmp = yield* TestInstance
const events = yield* collectGlobalEvents()
const init = yield* request(tmp.directory, "/project/git/init", {
method: "POST",
})
expect(init.status).toBe(200)
expect(yield* json(init)).toMatchObject({
vcs: "git",
worktree: tmp.directory,
})
expect(disposedEvents(events.seen, tmp.directory)).toBe(0)
const current = yield* request(tmp.directory, "/project/current")
expect(current.status).toBe(200)
expect(yield* json(current)).toMatchObject({
vcs: "git",
worktree: tmp.directory,
})
}),
{ git: true },
)
})