mirror of
https://github.com/anomalyco/opencode.git
synced 2026-05-24 13:54:07 +00:00
169 lines
6.4 KiB
TypeScript
169 lines
6.4 KiB
TypeScript
import { test, type TestOptions } from "bun:test"
|
|
import { Cause, Duration, Effect, Exit, Layer } from "effect"
|
|
import * as Scope from "effect/Scope"
|
|
import * as TestClock from "effect/testing/TestClock"
|
|
import * as TestConsole from "effect/testing/TestConsole"
|
|
import { memoMap } from "@opencode-ai/core/effect/memo-map"
|
|
import type { Config } from "@/config/config"
|
|
import { TestInstance, withTmpdirInstance } from "../fixture/fixture"
|
|
|
|
type Body<A, E, R> = Effect.Effect<A, E, R> | (() => Effect.Effect<A, E, R>)
|
|
type InstanceOptions = { git?: boolean; config?: Partial<Config.Info> | (() => Partial<Config.Info>) }
|
|
|
|
function isInstanceOptions(options: InstanceOptions | number | TestOptions | undefined): options is InstanceOptions {
|
|
return !!options && typeof options === "object" && ("git" in options || "config" in options)
|
|
}
|
|
|
|
function instanceArgs(
|
|
options?: InstanceOptions | number | TestOptions,
|
|
testOptions?: number | TestOptions,
|
|
): { instanceOptions: InstanceOptions | undefined; testOptions: number | TestOptions | undefined } {
|
|
if (typeof options === "number") return { instanceOptions: undefined, testOptions: options }
|
|
if (isInstanceOptions(options)) return { instanceOptions: options, testOptions }
|
|
return { instanceOptions: undefined, testOptions: options }
|
|
}
|
|
|
|
const body = <A, E, R>(value: Body<A, E, R>) => Effect.suspend(() => (typeof value === "function" ? value() : value))
|
|
|
|
type Runner = <A, E, R, E2>(value: Body<A, E, R | Scope.Scope>, layer: Layer.Layer<R, E2>) => Promise<A>
|
|
|
|
const isolatedRun: Runner = (value, layer) =>
|
|
Effect.gen(function* () {
|
|
const exit = yield* body(value).pipe(Effect.scoped, Effect.provide(layer), Effect.exit)
|
|
if (Exit.isFailure(exit)) {
|
|
for (const err of Cause.prettyErrors(exit.cause)) {
|
|
yield* Effect.logError(err)
|
|
}
|
|
}
|
|
return yield* exit
|
|
}).pipe(Effect.runPromise)
|
|
|
|
// Builds the test layer through the shared process-wide memoMap so cached
|
|
// services (Bus, Session, …) match Server.Default's instances. Use for tests
|
|
// that publish to an in-process HTTP server and need pub/sub identity with
|
|
// the server's handlers.
|
|
const sharedRun: Runner = (value, layer) =>
|
|
Effect.gen(function* () {
|
|
const scope = yield* Scope.make()
|
|
const ctx = yield* Layer.buildWithMemoMap(layer, memoMap, scope)
|
|
const exit = yield* body(value).pipe(Effect.scoped, Effect.provide(ctx), Effect.exit)
|
|
yield* Scope.close(scope, Exit.void)
|
|
if (Exit.isFailure(exit)) {
|
|
for (const err of Cause.prettyErrors(exit.cause)) {
|
|
yield* Effect.logError(err)
|
|
}
|
|
}
|
|
return yield* exit
|
|
}).pipe(Effect.runPromise)
|
|
|
|
const make = <R, E>(testLayer: Layer.Layer<R, E>, liveLayer: Layer.Layer<R, E>, run: Runner = isolatedRun) => {
|
|
const effect = <A, E2>(name: string, value: Body<A, E2, R | Scope.Scope>, opts?: number | TestOptions) =>
|
|
test(name, () => run(value, testLayer), opts)
|
|
|
|
effect.only = <A, E2>(name: string, value: Body<A, E2, R | Scope.Scope>, opts?: number | TestOptions) =>
|
|
test.only(name, () => run(value, testLayer), opts)
|
|
|
|
effect.skip = <A, E2>(name: string, value: Body<A, E2, R | Scope.Scope>, opts?: number | TestOptions) =>
|
|
test.skip(name, () => run(value, testLayer), opts)
|
|
|
|
const live = <A, E2>(name: string, value: Body<A, E2, R | Scope.Scope>, opts?: number | TestOptions) =>
|
|
test(name, () => run(value, liveLayer), opts)
|
|
|
|
live.only = <A, E2>(name: string, value: Body<A, E2, R | Scope.Scope>, opts?: number | TestOptions) =>
|
|
test.only(name, () => run(value, liveLayer), opts)
|
|
|
|
live.skip = <A, E2>(name: string, value: Body<A, E2, R | Scope.Scope>, opts?: number | TestOptions) =>
|
|
test.skip(name, () => run(value, liveLayer), opts)
|
|
|
|
const instance = <A, E2>(
|
|
name: string,
|
|
value: Body<A, E2, R | TestInstance | Scope.Scope>,
|
|
options?: InstanceOptions | number | TestOptions,
|
|
opts?: number | TestOptions,
|
|
) => {
|
|
const args = instanceArgs(options, opts)
|
|
return test(
|
|
name,
|
|
() => run(body(value).pipe(withTmpdirInstance(args.instanceOptions)), liveLayer),
|
|
args.testOptions,
|
|
)
|
|
}
|
|
|
|
instance.only = <A, E2>(
|
|
name: string,
|
|
value: Body<A, E2, R | TestInstance | Scope.Scope>,
|
|
options?: InstanceOptions | number | TestOptions,
|
|
opts?: number | TestOptions,
|
|
) => {
|
|
const args = instanceArgs(options, opts)
|
|
return test.only(
|
|
name,
|
|
() => run(body(value).pipe(withTmpdirInstance(args.instanceOptions)), liveLayer),
|
|
args.testOptions,
|
|
)
|
|
}
|
|
|
|
instance.skip = <A, E2>(
|
|
name: string,
|
|
value: Body<A, E2, R | TestInstance | Scope.Scope>,
|
|
options?: InstanceOptions | number | TestOptions,
|
|
opts?: number | TestOptions,
|
|
) => {
|
|
const args = instanceArgs(options, opts)
|
|
return test.skip(
|
|
name,
|
|
() => run(body(value).pipe(withTmpdirInstance(args.instanceOptions)), liveLayer),
|
|
args.testOptions,
|
|
)
|
|
}
|
|
|
|
return { effect, live, instance }
|
|
}
|
|
|
|
// Test environment with TestClock and TestConsole
|
|
const testEnv = Layer.mergeAll(TestConsole.layer, TestClock.layer())
|
|
|
|
// Live environment - uses real clock, but keeps TestConsole for output capture
|
|
const liveEnv = TestConsole.layer
|
|
|
|
export const it = make(testEnv, liveEnv)
|
|
|
|
export const testEffect = <R, E>(layer: Layer.Layer<R, E>) =>
|
|
make(Layer.provideMerge(layer, testEnv), Layer.provideMerge(layer, liveEnv))
|
|
|
|
// Variant of `testEffect` that builds the test layer through the shared
|
|
// process-wide memoMap so services like Bus/Session resolve to the same
|
|
// instances Server.Default uses. Use when a test needs pub/sub identity with
|
|
// an in-process HTTP server — most tests should stick with `testEffect`.
|
|
export const testEffectShared = <R, E>(layer: Layer.Layer<R, E>) =>
|
|
make(Layer.provideMerge(layer, testEnv), Layer.provideMerge(layer, liveEnv), sharedRun)
|
|
|
|
export const awaitWithTimeout = <A, E, R>(
|
|
self: Effect.Effect<A, E, R>,
|
|
message: string,
|
|
duration: Duration.Input = "2 seconds",
|
|
) =>
|
|
self.pipe(
|
|
Effect.timeoutOrElse({
|
|
duration,
|
|
orElse: () => Effect.fail(new Error(message)),
|
|
}),
|
|
)
|
|
|
|
export const pollWithTimeout = <A, E, R>(
|
|
self: Effect.Effect<A | undefined, E, R>,
|
|
message: string,
|
|
duration: Duration.Input = "5 seconds",
|
|
) =>
|
|
Effect.gen(function* () {
|
|
while (true) {
|
|
const result = yield* self
|
|
if (result !== undefined) return result
|
|
yield* Effect.sleep("20 millis")
|
|
}
|
|
}).pipe(
|
|
Effect.timeoutOrElse({
|
|
duration,
|
|
orElse: () => Effect.fail(new Error(message)),
|
|
}),
|
|
)
|