import type { Effect } from "../Effect.ts" import * as Api from "../ExecutionPlan.ts" import { dual } from "../Function.ts" import * as Result from "../Result.ts" import * as Schedule from "../Schedule.ts" import * as effect from "./effect.ts" import * as internalLayer from "./layer.ts" import * as internalSchedule from "./schedule.ts" /** @internal */ export const withExecutionPlan: { ( plan: Api.ExecutionPlan<{ provides: Provides input: Input error: PlanE requirements: PlanR }> ): (effect: Effect) => Effect< A, E | PlanE, Exclude | PlanR > ( effect: Effect, plan: Api.ExecutionPlan<{ provides: Provides input: Input error: PlanE requirements: PlanR }> ): Effect< A, E | PlanE, Exclude | PlanR > } = dual(2, ( self: Effect, plan: Api.ExecutionPlan<{ provides: Provides input: Input error: PlanE requirements: PlanR }> ) => effect.suspend(() => { let i = 0 let meta: Api.Metadata = { attempt: 0, stepIndex: 0 } const provideMeta = effect.provideServiceEffect( Api.CurrentMetadata, effect.sync(() => { meta = { attempt: meta.attempt + 1, stepIndex: i } return meta }) ) let result: Result.Result | undefined return effect.flatMap( effect.whileLoop({ while: () => i < plan.steps.length && (result === undefined || Result.isFailure(result)), body() { const step = plan.steps[i] let nextEffect: Effect = provideMeta(internalLayer.provide(self, step.provide as any)) if (result) { let attempted = false const wrapped = nextEffect // ensure the schedule is applied at least once nextEffect = effect.suspend(() => { if (attempted) return wrapped attempted = true return effect.fromResult(result!) }) nextEffect = internalSchedule.retry(nextEffect, scheduleFromStep(step, false)!) } else { const schedule = scheduleFromStep(step, true) nextEffect = schedule ? internalSchedule.retry(nextEffect, schedule) : nextEffect } return effect.result(nextEffect) }, step(result_) { result = result_ i++ } }), () => effect.fromResult(result!) ) })) /** @internal */ export const scheduleFromStep = ( step: Api.ExecutionPlan<{ provides: Provides input: In error: PlanE requirements: PlanR }>["steps"][number], first: boolean ) => { if (!first) { return internalSchedule.buildFromOptions({ schedule: step.schedule ? step.schedule : step.attempts ? undefined : scheduleOnce, times: step.attempts, while: step.while }) } else if (step.attempts === 1 || !(step.schedule || step.attempts)) { return undefined } return internalSchedule.buildFromOptions({ schedule: step.schedule, while: step.while, times: step.attempts ? step.attempts - 1 : undefined }) } const scheduleOnce = Schedule.recurs(1)