mirror of
https://github.com/anomalyco/opencode.git
synced 2026-05-24 05:35:15 +00:00
220 lines
8.5 KiB
TypeScript
220 lines
8.5 KiB
TypeScript
import { describe, expect } from "bun:test"
|
|
import { $ } from "bun"
|
|
import fs from "fs/promises"
|
|
import path from "path"
|
|
import { Effect } from "effect"
|
|
import { Project } from "@opencode-ai/core/project"
|
|
import { AbsolutePath } from "@opencode-ai/core/schema"
|
|
import { Hash } from "@opencode-ai/core/util/hash"
|
|
import { tmpdir } from "./fixture/tmpdir"
|
|
import { testEffect } from "./lib/effect"
|
|
|
|
const it = testEffect(Project.defaultLayer)
|
|
|
|
function remoteID(remote: string) {
|
|
return Project.ID.make(Hash.fast(`git-remote:${remote}`))
|
|
}
|
|
|
|
function abs(value: string) {
|
|
return AbsolutePath.make(value)
|
|
}
|
|
|
|
function real(value: string) {
|
|
return Effect.promise(() => fs.realpath(value)).pipe(Effect.map((value) => AbsolutePath.make(value)))
|
|
}
|
|
|
|
async function initRepo(dir: string, opts?: { commit?: boolean; remote?: string }) {
|
|
await $`git init`.cwd(dir).quiet()
|
|
await $`git config core.fsmonitor false`.cwd(dir).quiet()
|
|
await $`git config commit.gpgsign false`.cwd(dir).quiet()
|
|
await $`git config user.email test@opencode.test`.cwd(dir).quiet()
|
|
await $`git config user.name Test`.cwd(dir).quiet()
|
|
if (opts?.commit) await $`git commit --allow-empty -m root`.cwd(dir).quiet()
|
|
if (opts?.remote) await $`git remote add origin ${opts.remote}`.cwd(dir).quiet()
|
|
}
|
|
|
|
async function rootCommit(dir: string) {
|
|
return (await $`git rev-list --max-parents=0 HEAD`.cwd(dir).text()).trim()
|
|
}
|
|
|
|
describe("ProjectV2.resolve", () => {
|
|
it.live("returns global for non-git directory", () =>
|
|
Effect.gen(function* () {
|
|
const tmp = yield* Effect.acquireRelease(
|
|
Effect.promise(() => tmpdir()),
|
|
(tmp) => Effect.promise(() => tmp[Symbol.asyncDispose]()),
|
|
)
|
|
const project = yield* Project.Service
|
|
|
|
const result = yield* project.resolve(abs(tmp.path))
|
|
|
|
expect(result.id).toBe(Project.ID.make("global"))
|
|
expect(path.resolve(result.directory)).toBe(path.resolve(tmp.path))
|
|
expect(result.previous).toBeUndefined()
|
|
expect(result.vcs).toBeUndefined()
|
|
}),
|
|
)
|
|
|
|
it.live("returns git global for repo with no commits and no remote", () =>
|
|
Effect.gen(function* () {
|
|
const tmp = yield* Effect.acquireRelease(
|
|
Effect.promise(() => tmpdir()),
|
|
(tmp) => Effect.promise(() => tmp[Symbol.asyncDispose]()),
|
|
)
|
|
yield* Effect.promise(() => initRepo(tmp.path))
|
|
const project = yield* Project.Service
|
|
|
|
const result = yield* project.resolve(abs(tmp.path))
|
|
|
|
expect(result.id).toBe(Project.ID.make("global"))
|
|
expect(result.directory).toBe(yield* real(tmp.path))
|
|
expect(result.previous).toBeUndefined()
|
|
expect(result.vcs?.type).toBe("git")
|
|
}),
|
|
)
|
|
|
|
it.live("falls back to root commit when origin is missing", () =>
|
|
Effect.gen(function* () {
|
|
const tmp = yield* Effect.acquireRelease(
|
|
Effect.promise(() => tmpdir()),
|
|
(tmp) => Effect.promise(() => tmp[Symbol.asyncDispose]()),
|
|
)
|
|
yield* Effect.promise(() => initRepo(tmp.path, { commit: true }))
|
|
const project = yield* Project.Service
|
|
|
|
const result = yield* project.resolve(abs(tmp.path))
|
|
|
|
expect(result.id).toBe(Project.ID.make(yield* Effect.promise(() => rootCommit(tmp.path))))
|
|
expect(result.directory).toBe(yield* real(tmp.path))
|
|
expect(result.previous).toBeUndefined()
|
|
expect(result.vcs?.type).toBe("git")
|
|
}),
|
|
)
|
|
|
|
it.live("prefers normalized origin over root commit", () =>
|
|
Effect.gen(function* () {
|
|
const tmp = yield* Effect.acquireRelease(
|
|
Effect.promise(() => tmpdir()),
|
|
(tmp) => Effect.promise(() => tmp[Symbol.asyncDispose]()),
|
|
)
|
|
yield* Effect.promise(() => initRepo(tmp.path, { commit: true, remote: "git@github.com:Acme/App.git" }))
|
|
const project = yield* Project.Service
|
|
|
|
const result = yield* project.resolve(abs(tmp.path))
|
|
|
|
expect(result.id).toBe(remoteID("github.com/Acme/App"))
|
|
expect(result.id).not.toBe(Project.ID.make(yield* Effect.promise(() => rootCommit(tmp.path))))
|
|
expect(result.directory).toBe(yield* real(tmp.path))
|
|
expect(result.vcs?.type).toBe("git")
|
|
}),
|
|
)
|
|
|
|
it.live("normalizes ssh and https remotes to the same id", () =>
|
|
Effect.gen(function* () {
|
|
const ssh = yield* Effect.acquireRelease(
|
|
Effect.promise(() => tmpdir()),
|
|
(tmp) => Effect.promise(() => tmp[Symbol.asyncDispose]()),
|
|
)
|
|
const https = yield* Effect.acquireRelease(
|
|
Effect.promise(() => tmpdir()),
|
|
(tmp) => Effect.promise(() => tmp[Symbol.asyncDispose]()),
|
|
)
|
|
yield* Effect.promise(() => initRepo(ssh.path, { commit: true, remote: "git@github.com:owner/repo.git" }))
|
|
yield* Effect.promise(() => initRepo(https.path, { commit: true, remote: "https://github.com/owner/repo.git" }))
|
|
const project = yield* Project.Service
|
|
|
|
const a = yield* project.resolve(abs(ssh.path))
|
|
const b = yield* project.resolve(abs(https.path))
|
|
|
|
expect(a.id).toBe(remoteID("github.com/owner/repo"))
|
|
expect(b.id).toBe(a.id)
|
|
}),
|
|
)
|
|
|
|
it.live("ignores file remotes and falls back to root commit", () =>
|
|
Effect.gen(function* () {
|
|
const tmp = yield* Effect.acquireRelease(
|
|
Effect.promise(() => tmpdir()),
|
|
(tmp) => Effect.promise(() => tmp[Symbol.asyncDispose]()),
|
|
)
|
|
yield* Effect.promise(() => initRepo(tmp.path, { commit: true, remote: `file://${tmp.path}` }))
|
|
const project = yield* Project.Service
|
|
|
|
const result = yield* project.resolve(abs(tmp.path))
|
|
|
|
expect(result.id).toBe(Project.ID.make(yield* Effect.promise(() => rootCommit(tmp.path))))
|
|
}),
|
|
)
|
|
|
|
it.live("returns previous cached id from common dir", () =>
|
|
Effect.gen(function* () {
|
|
const tmp = yield* Effect.acquireRelease(
|
|
Effect.promise(() => tmpdir()),
|
|
(tmp) => Effect.promise(() => tmp[Symbol.asyncDispose]()),
|
|
)
|
|
yield* Effect.promise(() => initRepo(tmp.path, { commit: true, remote: "git@github.com:owner/repo.git" }))
|
|
yield* Effect.promise(() => Bun.write(path.join(tmp.path, ".git", "opencode"), "old-id"))
|
|
const project = yield* Project.Service
|
|
|
|
const result = yield* project.resolve(abs(tmp.path))
|
|
|
|
expect(result.previous).toBe(Project.ID.make("old-id"))
|
|
expect(result.id).toBe(remoteID("github.com/owner/repo"))
|
|
}),
|
|
)
|
|
|
|
it.live("does not write the cache while resolving", () =>
|
|
Effect.gen(function* () {
|
|
const tmp = yield* Effect.acquireRelease(
|
|
Effect.promise(() => tmpdir()),
|
|
(tmp) => Effect.promise(() => tmp[Symbol.asyncDispose]()),
|
|
)
|
|
yield* Effect.promise(() => initRepo(tmp.path, { commit: true, remote: "git@github.com:owner/repo.git" }))
|
|
const project = yield* Project.Service
|
|
|
|
yield* project.resolve(abs(tmp.path))
|
|
|
|
expect(yield* Effect.promise(() => Bun.file(path.join(tmp.path, ".git", "opencode")).exists())).toBe(false)
|
|
}),
|
|
)
|
|
|
|
it.live("resolves from nested directories to repo root", () =>
|
|
Effect.gen(function* () {
|
|
const tmp = yield* Effect.acquireRelease(
|
|
Effect.promise(() => tmpdir()),
|
|
(tmp) => Effect.promise(() => tmp[Symbol.asyncDispose]()),
|
|
)
|
|
yield* Effect.promise(() => initRepo(tmp.path, { commit: true }))
|
|
yield* Effect.promise(() => fs.mkdir(path.join(tmp.path, "a", "b"), { recursive: true }))
|
|
const project = yield* Project.Service
|
|
|
|
const result = yield* project.resolve(abs(path.join(tmp.path, "a", "b")))
|
|
|
|
expect(result.directory).toBe(yield* real(tmp.path))
|
|
}),
|
|
)
|
|
|
|
it.live("linked worktree returns opened worktree directory and previous from common dir", () =>
|
|
Effect.gen(function* () {
|
|
const tmp = yield* Effect.acquireRelease(
|
|
Effect.promise(() => tmpdir()),
|
|
(tmp) => Effect.promise(() => tmp[Symbol.asyncDispose]()),
|
|
)
|
|
const worktree = `${tmp.path}-worktree`
|
|
yield* Effect.addFinalizer(() =>
|
|
Effect.promise(() => $`rm -rf ${worktree}`.quiet().nothrow()).pipe(Effect.ignore),
|
|
)
|
|
yield* Effect.promise(() => initRepo(tmp.path, { commit: true, remote: "git@github.com:owner/repo.git" }))
|
|
yield* Effect.promise(() => Bun.write(path.join(tmp.path, ".git", "opencode"), "old-id"))
|
|
yield* Effect.promise(() => $`git worktree add ${worktree} -b test-${Date.now()}`.cwd(tmp.path).quiet())
|
|
const project = yield* Project.Service
|
|
|
|
const result = yield* project.resolve(abs(worktree))
|
|
|
|
expect(result.directory).toBe(yield* real(worktree))
|
|
expect(result.previous).toBe(Project.ID.make("old-id"))
|
|
expect(result.id).toBe(remoteID("github.com/owner/repo"))
|
|
expect(result.vcs?.type).toBe("git")
|
|
}),
|
|
)
|
|
})
|