import type * as Context from "../Context.js" import type * as Effect from "../Effect.js" import type * as Exit from "../Exit.js" import type * as Fiber from "../Fiber.js" import { pipe } from "../Function.js" import { globalValue } from "../GlobalValue.js" import * as MutableRef from "../MutableRef.js" import type * as Option from "../Option.js" import { hasProperty, isTagged } from "../Predicate.js" import * as SortedSet from "../SortedSet.js" import type * as Supervisor from "../Supervisor.js" import * as core from "./core.js" /** @internal */ const SupervisorSymbolKey = "effect/Supervisor" /** @internal */ export const SupervisorTypeId: Supervisor.SupervisorTypeId = Symbol.for( SupervisorSymbolKey ) as Supervisor.SupervisorTypeId /** @internal */ export const supervisorVariance = { /* c8 ignore next */ _T: (_: never) => _ } /** @internal */ export class ProxySupervisor implements Supervisor.Supervisor { readonly [SupervisorTypeId] = supervisorVariance constructor( readonly underlying: Supervisor.Supervisor, readonly value0: Effect.Effect ) { } get value(): Effect.Effect { return this.value0 } onStart( context: Context.Context, effect: Effect.Effect, parent: Option.Option>, fiber: Fiber.RuntimeFiber ): void { this.underlying.onStart(context, effect, parent, fiber) } onEnd(value: Exit.Exit, fiber: Fiber.RuntimeFiber): void { this.underlying.onEnd(value, fiber) } onEffect(fiber: Fiber.RuntimeFiber, effect: Effect.Effect): void { this.underlying.onEffect(fiber, effect) } onSuspend(fiber: Fiber.RuntimeFiber): void { this.underlying.onSuspend(fiber) } onResume(fiber: Fiber.RuntimeFiber): void { this.underlying.onResume(fiber) } map(f: (a: T) => B): Supervisor.Supervisor { return new ProxySupervisor(this, pipe(this.value, core.map(f))) } zip(right: Supervisor.Supervisor): Supervisor.Supervisor<[T, B]> { return new Zip(this, right) } } /** @internal */ export class Zip implements Supervisor.Supervisor { readonly _tag = "Zip" readonly [SupervisorTypeId] = supervisorVariance constructor( readonly left: Supervisor.Supervisor, readonly right: Supervisor.Supervisor ) { } get value(): Effect.Effect<[T0, T1]> { return core.zip(this.left.value, this.right.value) } onStart( context: Context.Context, effect: Effect.Effect, parent: Option.Option>, fiber: Fiber.RuntimeFiber ): void { this.left.onStart(context, effect, parent, fiber) this.right.onStart(context, effect, parent, fiber) } onEnd(value: Exit.Exit, fiber: Fiber.RuntimeFiber): void { this.left.onEnd(value, fiber) this.right.onEnd(value, fiber) } onEffect(fiber: Fiber.RuntimeFiber, effect: Effect.Effect): void { this.left.onEffect(fiber, effect) this.right.onEffect(fiber, effect) } onSuspend(fiber: Fiber.RuntimeFiber): void { this.left.onSuspend(fiber) this.right.onSuspend(fiber) } onResume(fiber: Fiber.RuntimeFiber): void { this.left.onResume(fiber) this.right.onResume(fiber) } map(f: (a: [T0, T1]) => B): Supervisor.Supervisor { return new ProxySupervisor(this, pipe(this.value, core.map(f))) } zip(right: Supervisor.Supervisor): Supervisor.Supervisor<[[T0, T1], A]> { return new Zip(this, right) } } /** @internal */ export const isZip = (self: unknown): self is Zip => hasProperty(self, SupervisorTypeId) && isTagged(self, "Zip") /** @internal */ export class Track implements Supervisor.Supervisor>> { readonly [SupervisorTypeId] = supervisorVariance readonly fibers: Set> = new Set() get value(): Effect.Effect>> { return core.sync(() => Array.from(this.fibers)) } onStart( _context: Context.Context, _effect: Effect.Effect, _parent: Option.Option>, fiber: Fiber.RuntimeFiber ): void { this.fibers.add(fiber) } onEnd(_value: Exit.Exit, fiber: Fiber.RuntimeFiber): void { this.fibers.delete(fiber) } onEffect(_fiber: Fiber.RuntimeFiber, _effect: Effect.Effect): void { // } onSuspend(_fiber: Fiber.RuntimeFiber): void { // } onResume(_fiber: Fiber.RuntimeFiber): void { // } map(f: (a: Array>) => B): Supervisor.Supervisor { return new ProxySupervisor(this, pipe(this.value, core.map(f))) } zip( right: Supervisor.Supervisor ): Supervisor.Supervisor<[Array>, A]> { return new Zip(this, right) } onRun(execution: () => X, _fiber: Fiber.RuntimeFiber): X { return execution() } } /** @internal */ export class Const implements Supervisor.Supervisor { readonly [SupervisorTypeId] = supervisorVariance constructor(readonly effect: Effect.Effect) { } get value(): Effect.Effect { return this.effect } onStart( _context: Context.Context, _effect: Effect.Effect, _parent: Option.Option>, _fiber: Fiber.RuntimeFiber ): void { // } onEnd(_value: Exit.Exit, _fiber: Fiber.RuntimeFiber): void { // } onEffect(_fiber: Fiber.RuntimeFiber, _effect: Effect.Effect): void { // } onSuspend(_fiber: Fiber.RuntimeFiber): void { // } onResume(_fiber: Fiber.RuntimeFiber): void { // } map(f: (a: T) => B): Supervisor.Supervisor { return new ProxySupervisor(this, pipe(this.value, core.map(f))) } zip(right: Supervisor.Supervisor): Supervisor.Supervisor<[T, A]> { return new Zip(this, right) } onRun(execution: () => X, _fiber: Fiber.RuntimeFiber): X { return execution() } } class FibersIn implements Supervisor.Supervisor>> { readonly [SupervisorTypeId] = supervisorVariance constructor(readonly ref: MutableRef.MutableRef>>) { } get value(): Effect.Effect>> { return core.sync(() => MutableRef.get(this.ref)) } onStart( _context: Context.Context, _effect: Effect.Effect, _parent: Option.Option>, fiber: Fiber.RuntimeFiber ): void { pipe(this.ref, MutableRef.set(pipe(MutableRef.get(this.ref), SortedSet.add(fiber)))) } onEnd(_value: Exit.Exit, fiber: Fiber.RuntimeFiber): void { pipe(this.ref, MutableRef.set(pipe(MutableRef.get(this.ref), SortedSet.remove(fiber)))) } onEffect(_fiber: Fiber.RuntimeFiber, _effect: Effect.Effect): void { // } onSuspend(_fiber: Fiber.RuntimeFiber): void { // } onResume(_fiber: Fiber.RuntimeFiber): void { // } map(f: (a: SortedSet.SortedSet>) => B): Supervisor.Supervisor { return new ProxySupervisor(this, pipe(this.value, core.map(f))) } zip( right: Supervisor.Supervisor ): Supervisor.Supervisor<[SortedSet.SortedSet>, A]> { return new Zip(this, right) } onRun(execution: () => X, _fiber: Fiber.RuntimeFiber): X { return execution() } } /** @internal */ export const unsafeTrack = (): Supervisor.Supervisor>> => { return new Track() } /** @internal */ export const track: Effect.Effect>>> = core.sync(unsafeTrack) /** @internal */ export const fromEffect = (effect: Effect.Effect): Supervisor.Supervisor => { return new Const(effect) } /** @internal */ export const none = globalValue("effect/Supervisor/none", () => fromEffect(core.void)) /** @internal */ export const fibersIn = ( ref: MutableRef.MutableRef>> ): Effect.Effect>>> => core.sync(() => new FibersIn(ref))