Skip to content

Commit 543d36d

Browse files
authoredFeb 11, 2025··
Schedule: fix unsafe tapOutput signature (#4439)
1 parent 5808fbc commit 543d36d

File tree

5 files changed

+105
-18
lines changed

5 files changed

+105
-18
lines changed
 

‎.changeset/gold-jobs-love.md

+33
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
---
2+
"effect": patch
3+
---
4+
5+
Schedule: fix unsafe `tapOutput` signature.
6+
7+
Previously, `tapOutput` allowed using an output type that wasn't properly inferred, leading to potential runtime errors. Now, TypeScript correctly detects mismatches at compile time, preventing unexpected crashes.
8+
9+
**Before (Unsafe, Causes Runtime Error)**
10+
11+
```ts
12+
import { Effect, Schedule, Console } from "effect"
13+
14+
const schedule = Schedule.once.pipe(
15+
Schedule.as<number | string>(1),
16+
Schedule.tapOutput((s: string) => Console.log(s.trim())) // ❌ Runtime error
17+
)
18+
19+
Effect.runPromise(Effect.void.pipe(Effect.schedule(schedule)))
20+
// throws: TypeError: s.trim is not a function
21+
```
22+
23+
**After (Safe, Catches Type Error at Compile Time)**
24+
25+
```ts
26+
import { Console, Schedule } from "effect"
27+
28+
const schedule = Schedule.once.pipe(
29+
Schedule.as<number | string>(1),
30+
// ✅ Type Error: Type 'number' is not assignable to type 'string'
31+
Schedule.tapOutput((s: string) => Console.log(s.trim()))
32+
)
33+
```

‎packages/effect/dtslint/Schedule.ts

+33
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
import { Console, Schedule } from "effect"
2+
3+
// -------------------------------------------------------------------------------------
4+
// tapOutput
5+
// -------------------------------------------------------------------------------------
6+
7+
// $ExpectType Schedule<string | number, unknown, never>
8+
Schedule.once.pipe(
9+
Schedule.as<number | string>(1),
10+
Schedule.tapOutput((
11+
x // $ExpectType string | number
12+
) => Console.log(x))
13+
)
14+
15+
// The callback should not affect the type of the output (`number`)
16+
// $ExpectType Schedule<number, unknown, never>
17+
Schedule.once.pipe(
18+
Schedule.as(1),
19+
Schedule.tapOutput((x: string | number) => Console.log(x))
20+
)
21+
// $ExpectType Schedule<number, unknown, never>
22+
Schedule.tapOutput(
23+
Schedule.once.pipe(
24+
Schedule.as(1)
25+
),
26+
(x: string | number) => Console.log(x)
27+
)
28+
29+
Schedule.once.pipe(
30+
Schedule.as<number | string>(1),
31+
// @ts-expect-error
32+
Schedule.tapOutput((s: string) => Console.log(s.trim()))
33+
)

‎packages/effect/src/Schedule.ts

+8-7
Original file line numberDiff line numberDiff line change
@@ -1763,16 +1763,17 @@ export const spaced: (duration: Duration.DurationInput) => Schedule<number> = in
17631763
export const stop: Schedule<void> = internal.stop
17641764

17651765
/**
1766-
* Returns a schedule that runs once and produces the specified constant value.
1766+
* Returns a schedule that recurs indefinitely, always producing the specified
1767+
* constant value.
17671768
*
17681769
* @since 2.0.0
17691770
* @category Constructors
17701771
*/
17711772
export const succeed: <A>(value: A) => Schedule<A> = internal.succeed
17721773

17731774
/**
1774-
* Returns a schedule that runs once, evaluating the given function to produce a
1775-
* constant value.
1775+
* Returns a schedule that recurs indefinitely, evaluating the given function to
1776+
* produce a constant value.
17761777
*
17771778
* @category Constructors
17781779
* @since 2.0.0
@@ -1816,12 +1817,12 @@ export const tapInput: {
18161817
* @category Tapping
18171818
*/
18181819
export const tapOutput: {
1819-
<XO extends Out, X, R2, Out>(
1820-
f: (out: XO) => Effect.Effect<X, never, R2>
1820+
<X, R2, Out>(
1821+
f: (out: Types.NoInfer<Out>) => Effect.Effect<X, never, R2>
18211822
): <In, R>(self: Schedule<Out, In, R>) => Schedule<Out, In, R2 | R>
1822-
<Out, In, R, XO extends Out, X, R2>(
1823+
<Out, In, R, X, R2>(
18231824
self: Schedule<Out, In, R>,
1824-
f: (out: XO) => Effect.Effect<X, never, R2>
1825+
f: (out: Out) => Effect.Effect<X, never, R2>
18251826
): Schedule<Out, In, R | R2>
18261827
} = internal.tapOutput
18271828

‎packages/effect/src/internal/schedule.ts

+17-11
Original file line numberDiff line numberDiff line change
@@ -1362,19 +1362,25 @@ export const tapInput = dual<
13621362

13631363
/** @internal */
13641364
export const tapOutput = dual<
1365-
<XO extends Out, X, R2, Out>(
1366-
f: (out: XO) => Effect.Effect<X, never, R2>
1367-
) => <In, R>(self: Schedule.Schedule<Out, In, R>) => Schedule.Schedule<Out, In, R | R2>,
1368-
<Out, In, R, XO extends Out, X, R2>(
1365+
<X, R2, Out>(
1366+
f: (out: Types.NoInfer<Out>) => Effect.Effect<X, never, R2>
1367+
) => <In, R>(self: Schedule.Schedule<Out, In, R>) => Schedule.Schedule<Out, In, R2 | R>,
1368+
<Out, In, R, X, R2>(
13691369
self: Schedule.Schedule<Out, In, R>,
1370-
f: (out: XO) => Effect.Effect<X, never, R2>
1370+
f: (out: Out) => Effect.Effect<X, never, R2>
13711371
) => Schedule.Schedule<Out, In, R | R2>
1372-
>(2, (self, f) =>
1373-
makeWithState(self.initial, (now, input, state) =>
1374-
core.tap(
1375-
self.step(now, input, state),
1376-
([, out]) => f(out as any)
1377-
)))
1372+
>(
1373+
2,
1374+
<Out, In, R, X, R2>(
1375+
self: Schedule.Schedule<Out, In, R>,
1376+
f: (out: Out) => Effect.Effect<X, never, R2>
1377+
): Schedule.Schedule<Out, In, R | R2> =>
1378+
makeWithState(self.initial, (now, input, state) =>
1379+
core.tap(
1380+
self.step(now, input, state),
1381+
([, out]) => f(out)
1382+
))
1383+
)
13781384

13791385
/** @internal */
13801386
export const unfold = <A>(initial: A, f: (a: A) => A): Schedule.Schedule<A> =>

‎packages/effect/test/Schedule.test.ts

+14
Original file line numberDiff line numberDiff line change
@@ -814,6 +814,20 @@ describe("Schedule", () => {
814814
)
815815
deepStrictEqual(exit, Exit.die(exception))
816816
}))
817+
it.effect("tapOutput", () =>
818+
Effect.gen(function*() {
819+
const log: Array<number | string> = []
820+
const schedule = Schedule.once.pipe(
821+
Schedule.as<number | string>(1),
822+
Schedule.tapOutput((x) =>
823+
Effect.sync(() => {
824+
log.push(x)
825+
})
826+
)
827+
)
828+
yield* Effect.void.pipe(Effect.schedule(schedule))
829+
deepStrictEqual(log, [1, 1])
830+
}))
817831
})
818832
})
819833

0 commit comments

Comments
 (0)
Please sign in to comment.