import { test, type TestOptions } from "bun:test"
import { Cause, Effect, Exit, Layer } from "effect"
import type * as Scope from "effect/Scope"
import * as TestConsole from "effect/testing/TestConsole"
type Body = Effect.Effect | (() => Effect.Effect)
const env = TestConsole.layer
const body = (value: Body) => Effect.suspend(() => (typeof value === "function" ? value() : value))
const run = (value: Body, layer: Layer.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)
const make = (layer: Layer.Layer) => {
const effect = (name: string, value: Body, opts?: number | TestOptions) =>
test(name, () => run(value, layer), opts)
effect.only = (name: string, value: Body, opts?: number | TestOptions) =>
test.only(name, () => run(value, layer), opts)
effect.skip = (name: string, value: Body, opts?: number | TestOptions) =>
test.skip(name, () => run(value, layer), opts)
return { effect }
}
export const it = make(env)
export const testEffect = (layer: Layer.Layer) => make(Layer.provideMerge(layer, env))