|
|
@@ -1,260 +1 @@
|
|
|
-import * as CrossSpawnSpawner from "@/effect/cross-spawn-spawner"
|
|
|
-import { Effect, Layer, Context, Stream } from "effect"
|
|
|
-import { ChildProcess, ChildProcessSpawner } from "effect/unstable/process"
|
|
|
-
|
|
|
-export namespace Git {
|
|
|
- const cfg = [
|
|
|
- "--no-optional-locks",
|
|
|
- "-c",
|
|
|
- "core.autocrlf=false",
|
|
|
- "-c",
|
|
|
- "core.fsmonitor=false",
|
|
|
- "-c",
|
|
|
- "core.longpaths=true",
|
|
|
- "-c",
|
|
|
- "core.symlinks=true",
|
|
|
- "-c",
|
|
|
- "core.quotepath=false",
|
|
|
- ] as const
|
|
|
-
|
|
|
- const out = (result: { text(): string }) => result.text().trim()
|
|
|
- const nuls = (text: string) => text.split("\0").filter(Boolean)
|
|
|
- const fail = (err: unknown) =>
|
|
|
- ({
|
|
|
- exitCode: 1,
|
|
|
- text: () => "",
|
|
|
- stdout: Buffer.alloc(0),
|
|
|
- stderr: Buffer.from(err instanceof Error ? err.message : String(err)),
|
|
|
- }) satisfies Result
|
|
|
-
|
|
|
- export type Kind = "added" | "deleted" | "modified"
|
|
|
-
|
|
|
- export type Base = {
|
|
|
- readonly name: string
|
|
|
- readonly ref: string
|
|
|
- }
|
|
|
-
|
|
|
- export type Item = {
|
|
|
- readonly file: string
|
|
|
- readonly code: string
|
|
|
- readonly status: Kind
|
|
|
- }
|
|
|
-
|
|
|
- export type Stat = {
|
|
|
- readonly file: string
|
|
|
- readonly additions: number
|
|
|
- readonly deletions: number
|
|
|
- }
|
|
|
-
|
|
|
- export interface Result {
|
|
|
- readonly exitCode: number
|
|
|
- readonly text: () => string
|
|
|
- readonly stdout: Buffer
|
|
|
- readonly stderr: Buffer
|
|
|
- }
|
|
|
-
|
|
|
- export interface Options {
|
|
|
- readonly cwd: string
|
|
|
- readonly env?: Record<string, string>
|
|
|
- }
|
|
|
-
|
|
|
- export interface Interface {
|
|
|
- readonly run: (args: string[], opts: Options) => Effect.Effect<Result>
|
|
|
- readonly branch: (cwd: string) => Effect.Effect<string | undefined>
|
|
|
- readonly prefix: (cwd: string) => Effect.Effect<string>
|
|
|
- readonly defaultBranch: (cwd: string) => Effect.Effect<Base | undefined>
|
|
|
- readonly hasHead: (cwd: string) => Effect.Effect<boolean>
|
|
|
- readonly mergeBase: (cwd: string, base: string, head?: string) => Effect.Effect<string | undefined>
|
|
|
- readonly show: (cwd: string, ref: string, file: string, prefix?: string) => Effect.Effect<string>
|
|
|
- readonly status: (cwd: string) => Effect.Effect<Item[]>
|
|
|
- readonly diff: (cwd: string, ref: string) => Effect.Effect<Item[]>
|
|
|
- readonly stats: (cwd: string, ref: string) => Effect.Effect<Stat[]>
|
|
|
- }
|
|
|
-
|
|
|
- const kind = (code: string): Kind => {
|
|
|
- if (code === "??") return "added"
|
|
|
- if (code.includes("U")) return "modified"
|
|
|
- if (code.includes("A") && !code.includes("D")) return "added"
|
|
|
- if (code.includes("D") && !code.includes("A")) return "deleted"
|
|
|
- return "modified"
|
|
|
- }
|
|
|
-
|
|
|
- export class Service extends Context.Service<Service, Interface>()("@opencode/Git") {}
|
|
|
-
|
|
|
- export const layer = Layer.effect(
|
|
|
- Service,
|
|
|
- Effect.gen(function* () {
|
|
|
- const spawner = yield* ChildProcessSpawner.ChildProcessSpawner
|
|
|
-
|
|
|
- const run = Effect.fn("Git.run")(
|
|
|
- function* (args: string[], opts: Options) {
|
|
|
- const proc = ChildProcess.make("git", [...cfg, ...args], {
|
|
|
- cwd: opts.cwd,
|
|
|
- env: opts.env,
|
|
|
- extendEnv: true,
|
|
|
- stdin: "ignore",
|
|
|
- stdout: "pipe",
|
|
|
- stderr: "pipe",
|
|
|
- })
|
|
|
- const handle = yield* spawner.spawn(proc)
|
|
|
- const [stdout, stderr] = yield* Effect.all(
|
|
|
- [Stream.mkString(Stream.decodeText(handle.stdout)), Stream.mkString(Stream.decodeText(handle.stderr))],
|
|
|
- { concurrency: 2 },
|
|
|
- )
|
|
|
- return {
|
|
|
- exitCode: yield* handle.exitCode,
|
|
|
- text: () => stdout,
|
|
|
- stdout: Buffer.from(stdout),
|
|
|
- stderr: Buffer.from(stderr),
|
|
|
- } satisfies Result
|
|
|
- },
|
|
|
- Effect.scoped,
|
|
|
- Effect.catch((err) => Effect.succeed(fail(err))),
|
|
|
- )
|
|
|
-
|
|
|
- const text = Effect.fn("Git.text")(function* (args: string[], opts: Options) {
|
|
|
- return (yield* run(args, opts)).text()
|
|
|
- })
|
|
|
-
|
|
|
- const lines = Effect.fn("Git.lines")(function* (args: string[], opts: Options) {
|
|
|
- return (yield* text(args, opts))
|
|
|
- .split(/\r?\n/)
|
|
|
- .map((item) => item.trim())
|
|
|
- .filter(Boolean)
|
|
|
- })
|
|
|
-
|
|
|
- const refs = Effect.fnUntraced(function* (cwd: string) {
|
|
|
- return yield* lines(["for-each-ref", "--format=%(refname:short)", "refs/heads"], { cwd })
|
|
|
- })
|
|
|
-
|
|
|
- const configured = Effect.fnUntraced(function* (cwd: string, list: string[]) {
|
|
|
- const result = yield* run(["config", "init.defaultBranch"], { cwd })
|
|
|
- const name = out(result)
|
|
|
- if (!name || !list.includes(name)) return
|
|
|
- return { name, ref: name } satisfies Base
|
|
|
- })
|
|
|
-
|
|
|
- const primary = Effect.fnUntraced(function* (cwd: string) {
|
|
|
- const list = yield* lines(["remote"], { cwd })
|
|
|
- if (list.includes("origin")) return "origin"
|
|
|
- if (list.length === 1) return list[0]
|
|
|
- if (list.includes("upstream")) return "upstream"
|
|
|
- return list[0]
|
|
|
- })
|
|
|
-
|
|
|
- const branch = Effect.fn("Git.branch")(function* (cwd: string) {
|
|
|
- const result = yield* run(["symbolic-ref", "--quiet", "--short", "HEAD"], { cwd })
|
|
|
- if (result.exitCode !== 0) return
|
|
|
- const text = out(result)
|
|
|
- return text || undefined
|
|
|
- })
|
|
|
-
|
|
|
- const prefix = Effect.fn("Git.prefix")(function* (cwd: string) {
|
|
|
- const result = yield* run(["rev-parse", "--show-prefix"], { cwd })
|
|
|
- if (result.exitCode !== 0) return ""
|
|
|
- return out(result)
|
|
|
- })
|
|
|
-
|
|
|
- const defaultBranch = Effect.fn("Git.defaultBranch")(function* (cwd: string) {
|
|
|
- const remote = yield* primary(cwd)
|
|
|
- if (remote) {
|
|
|
- const head = yield* run(["symbolic-ref", `refs/remotes/${remote}/HEAD`], { cwd })
|
|
|
- if (head.exitCode === 0) {
|
|
|
- const ref = out(head).replace(/^refs\/remotes\//, "")
|
|
|
- const name = ref.startsWith(`${remote}/`) ? ref.slice(`${remote}/`.length) : ""
|
|
|
- if (name) return { name, ref } satisfies Base
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- const list = yield* refs(cwd)
|
|
|
- const next = yield* configured(cwd, list)
|
|
|
- if (next) return next
|
|
|
- if (list.includes("main")) return { name: "main", ref: "main" } satisfies Base
|
|
|
- if (list.includes("master")) return { name: "master", ref: "master" } satisfies Base
|
|
|
- })
|
|
|
-
|
|
|
- const hasHead = Effect.fn("Git.hasHead")(function* (cwd: string) {
|
|
|
- const result = yield* run(["rev-parse", "--verify", "HEAD"], { cwd })
|
|
|
- return result.exitCode === 0
|
|
|
- })
|
|
|
-
|
|
|
- const mergeBase = Effect.fn("Git.mergeBase")(function* (cwd: string, base: string, head = "HEAD") {
|
|
|
- const result = yield* run(["merge-base", base, head], { cwd })
|
|
|
- if (result.exitCode !== 0) return
|
|
|
- const text = out(result)
|
|
|
- return text || undefined
|
|
|
- })
|
|
|
-
|
|
|
- const show = Effect.fn("Git.show")(function* (cwd: string, ref: string, file: string, prefix = "") {
|
|
|
- const target = prefix ? `${prefix}${file}` : file
|
|
|
- const result = yield* run(["show", `${ref}:${target}`], { cwd })
|
|
|
- if (result.exitCode !== 0) return ""
|
|
|
- if (result.stdout.includes(0)) return ""
|
|
|
- return result.text()
|
|
|
- })
|
|
|
-
|
|
|
- const status = Effect.fn("Git.status")(function* (cwd: string) {
|
|
|
- return nuls(
|
|
|
- yield* text(["status", "--porcelain=v1", "--untracked-files=all", "--no-renames", "-z", "--", "."], {
|
|
|
- cwd,
|
|
|
- }),
|
|
|
- ).flatMap((item) => {
|
|
|
- const file = item.slice(3)
|
|
|
- if (!file) return []
|
|
|
- const code = item.slice(0, 2)
|
|
|
- return [{ file, code, status: kind(code) } satisfies Item]
|
|
|
- })
|
|
|
- })
|
|
|
-
|
|
|
- const diff = Effect.fn("Git.diff")(function* (cwd: string, ref: string) {
|
|
|
- const list = nuls(
|
|
|
- yield* text(["diff", "--no-ext-diff", "--no-renames", "--name-status", "-z", ref, "--", "."], { cwd }),
|
|
|
- )
|
|
|
- return list.flatMap((code, idx) => {
|
|
|
- if (idx % 2 !== 0) return []
|
|
|
- const file = list[idx + 1]
|
|
|
- if (!code || !file) return []
|
|
|
- return [{ file, code, status: kind(code) } satisfies Item]
|
|
|
- })
|
|
|
- })
|
|
|
-
|
|
|
- const stats = Effect.fn("Git.stats")(function* (cwd: string, ref: string) {
|
|
|
- return nuls(
|
|
|
- yield* text(["diff", "--no-ext-diff", "--no-renames", "--numstat", "-z", ref, "--", "."], { cwd }),
|
|
|
- ).flatMap((item) => {
|
|
|
- const a = item.indexOf("\t")
|
|
|
- const b = item.indexOf("\t", a + 1)
|
|
|
- if (a === -1 || b === -1) return []
|
|
|
- const file = item.slice(b + 1)
|
|
|
- if (!file) return []
|
|
|
- const adds = item.slice(0, a)
|
|
|
- const dels = item.slice(a + 1, b)
|
|
|
- const additions = adds === "-" ? 0 : Number.parseInt(adds || "0", 10)
|
|
|
- const deletions = dels === "-" ? 0 : Number.parseInt(dels || "0", 10)
|
|
|
- return [
|
|
|
- {
|
|
|
- file,
|
|
|
- additions: Number.isFinite(additions) ? additions : 0,
|
|
|
- deletions: Number.isFinite(deletions) ? deletions : 0,
|
|
|
- } satisfies Stat,
|
|
|
- ]
|
|
|
- })
|
|
|
- })
|
|
|
-
|
|
|
- return Service.of({
|
|
|
- run,
|
|
|
- branch,
|
|
|
- prefix,
|
|
|
- defaultBranch,
|
|
|
- hasHead,
|
|
|
- mergeBase,
|
|
|
- show,
|
|
|
- status,
|
|
|
- diff,
|
|
|
- stats,
|
|
|
- })
|
|
|
- }),
|
|
|
- )
|
|
|
-
|
|
|
- export const defaultLayer = layer.pipe(Layer.provide(CrossSpawnSpawner.defaultLayer))
|
|
|
-}
|
|
|
+export * as Git from "./git"
|