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))