heftia/README.md

370 lines
23 KiB
Markdown
Raw Normal View History

# Heftia: higher-order algebraic effects done right
2023-09-18 10:07:42 +03:00
2023-09-27 10:44:34 +03:00
[![Hackage](https://img.shields.io/hackage/v/heftia.svg?logo=haskell&label=heftia)](https://hackage.haskell.org/package/heftia)
[![Hackage](https://img.shields.io/hackage/v/heftia-effects.svg?logo=haskell&label=heftia-effects)](https://hackage.haskell.org/package/heftia-effects)
2024-10-21 13:28:14 +03:00
[![Stackage](https://www.stackage.org/package/heftia-effects/badge/nightly?label=Stackage)](https://www.stackage.org/package/heftia-effects)
[![Build status](https://img.shields.io/github/actions/workflow/status/sayo-hs/heftia/haskell.yml?branch=develop)](https://github.com/sayo-hs/heftia/actions)
2023-09-18 10:07:42 +03:00
Heftia is an extensible effects library for Haskell that generalizes "Algebraic Effects and Handlers" to higher-order effects, providing users with maximum flexibility and delivering standard and reasonable speed.
2024-10-09 21:05:53 +03:00
In its generalization, the focus is on ensuring predictable results based on simple, consistent semantics, while preserving soundness.
2023-08-27 16:32:48 +03:00
2024-10-09 21:05:53 +03:00
Please refer to the [Haddock documentation](https://hackage.haskell.org/package/heftia-0.4.0.0/docs/Control-Monad-Hefty.html) for usage and semantics.
For information on performance, please refer to [performance.md](https://github.com/sayo-hs/heftia/blob/v0.5.0/benchmark/performance.md).
2024-10-11 22:05:12 +03:00
This library is inspired by the paper:
* Casper Bach Poulsen and Cas van der Rest. 2023. Hefty Algebras: Modular
Elaboration of Higher-Order Algebraic Effects. Proc. ACM Program. Lang. 7,
POPL, Article 62 (January 2023), 31 pages. <https://doi.org/10.1145/3571255>
The *elaboration* approach proposed in the above paper allows for a straightforward treatment of higher-order effects.
Heftia's data structure is an extension of the Freer monad, designed to be theoretically straightforward by eliminating ad-hoc elements.
## Why choose this library over others?
This library is based on algebraic effects. Currently, **none of the practical effect libraries other than this one are "algebraic."** So, why is being *algebraic* important?
For example, algebraic effects are essential for managing coroutines, generators, streaming, concurrency, non-deterministic computations, and more in a highly elegant and concise manner.
Algebraic effects provide a consistent and predictable framework for handling side effects, enhancing modularity and flexibility in your code.
Research in cutting-edge languages like [Koka](https://koka-lang.github.io/koka/doc/index.html), [Eff lang](https://www.eff-lang.org/), and [OCaml 5](https://ocaml.org/manual/effects.html) is advancing the understanding and implementation of algebraic effects, establishing them as **the programming paradigm of the future**.
Heftia extends this by supporting higher-order algebraic effects, allowing for more expressive and modular effect management.
This leads to more maintainable and extensible applications compared to non-algebraic effect libraries, positioning Heftia at **the forefront of modern effect handling techniques**.
2024-10-27 06:20:58 +03:00
Furthermore, **Heftia is functionally a superset of other effect libraries**, especially those based on `ReaderT` over `IO`.
2024-10-27 06:10:08 +03:00
In other words, anything that is possible with other libraries is also possible with this library.
This is because Heftia supports `MonadUnliftIO` in the form of higher-order effects.
2024-10-27 11:10:04 +03:00
**Heftia should be a good substitute for `mtl`, `polysemy`, `fused-effects`, and `freer-simple`.**
Additionally, if performance is not a top priority, it should also be a good alternative for `effectful`.
If performance is particularly important, [`effectful`](https://github.com/haskell-effectful/effectful) would be the best alternative to this library.
2024-10-11 22:05:12 +03:00
## Key Features
2024-10-09 21:05:53 +03:00
* **Correct Semantics for Higher-Order Effects & Continuations**
This library provides the following features simultaneously, which existing libraries could not support together:
* Higher-order effects
* Delimited continuations (algebraic effects)
2024-10-11 19:40:22 +03:00
* Coroutines (non-scoped resumptions)
2024-10-27 06:20:58 +03:00
* Coroutine-based, composable, and resumable concurrent streams
2024-10-11 19:40:22 +03:00
* Non-deterministic computations
* [`MonadUnliftIO`](https://hackage.haskell.org/package/unliftio)
2024-10-27 06:20:58 +03:00
* Examples
* [heftia-effects/Example/UnliftIO/Main.hs](https://github.com/sayo-hs/heftia/blob/v0.5.0/heftia-effects/Example/UnliftIO/Main.hs)
* [heftia-effects/Example/Stream/Main.hs](https://github.com/sayo-hs/heftia/blob/v0.5.0/heftia-effects/Example/Stream/Main.hs)
All of these interact through a simple, consistent, and predictable semantics based on algebraic effects.
* **Easy and Concise Implementation for Custom Effect Interpreters**
As you can see from the implementations of basic effect interpreters such as [State](https://hackage.haskell.org/package/heftia-effects-0.4.0.0/docs/src/Control.Monad.Hefty.State.html#runState), [Throw/Catch](https://hackage.haskell.org/package/heftia-effects-0.4.0.0/docs/src/Control.Monad.Hefty.Except.html#runThrow), [Writer](https://hackage.haskell.org/package/heftia-effects-0.4.0.0/docs/src/Control.Monad.Hefty.Writer.html#runTell), [NonDet](https://hackage.haskell.org/package/heftia-effects-0.4.0.0/docs/src/Control.Monad.Hefty.NonDet.html#runNonDet), and [Coroutine](https://hackage.haskell.org/package/heftia-effects-0.4.0.0/docs/src/Control.Monad.Hefty.Coroutine.html#runCoroutine), they can be implemented in just a few lines, or even a single line. Even for effects like NonDet and Coroutine, which involve continuations and might seem difficult to implement at first glance, this is exactly how simple it can be. This is the power of algebraic effects. Users can quickly define experimental and innovative custom effects using continuations.
* **Standard and Reasonable Performance**
It operates at a speed positioned roughly in the middle between faster libraries (like `effectful` or `eveff`) and relatively slower ones (like `polysemy` or `fused-effects`): [performance.md](https://github.com/sayo-hs/heftia/blob/v0.5.0/benchmark/performance.md).
* **Purity**
* Does not depend on the IO monad and can use any monad as the base monad.
* Semantics are isolated from the IO monad, meaning that aspects like asynchronous exceptions and threads do not affect the behavior of effects.
* The constructors of the `Eff` monad are [exposed](https://hackage.haskell.org/package/heftia-0.4.0.0/docs/Control-Monad-Hefty.html#t:Eff), and users can manipulate them directly without any safety concerns. Still, the semantics remain intact.
* These are in contrast to libraries like `effectful` and `eff`, making this library more **Haskell-ish and purely functional**.
2024-10-09 21:05:53 +03:00
2024-10-11 22:05:12 +03:00
## Downsides
2023-08-27 16:32:48 +03:00
2024-10-11 22:05:12 +03:00
This library has notable semantic differences, particularly compared to libraries like `effectful`, `polysemy`, and `fused-effects`.
The semantics of this library are almost equivalent to those of `freer-simple` and are also similar to Alexis King's `eff` library.
This type of semantics is often referred to as *continuation-based semantics*.
Additionally, unlike recent libraries such as `effectful`, which have an IO-fused effect system, the semantics of this library are separated from IO.
Due to these differences, people who are already familiar with the semantics of other major libraries may find it challenging to transition to this library due to the mental model differences.
2023-08-27 16:32:48 +03:00
2024-10-11 22:05:12 +03:00
For those who have not used an extensible effects library in Haskell before, this should not be a problem.
Particularly, if you are already somewhat familiar with the semantics of algebraic effects through languages like `koka` or `eff-lang`,
you likely already have the mental model needed for this library, and everything should go smoothly.
2023-08-25 07:23:49 +03:00
2024-07-06 16:45:56 +03:00
## Status
2024-07-17 10:29:56 +03:00
This library is currently in the beta stage.
There may be significant changes and potential bugs.
**We are looking forward to your feedback!**
2024-07-06 16:45:56 +03:00
2024-10-09 21:05:53 +03:00
## Getting Started
1.
```console
$ cabal update
```
2024-10-09 21:05:53 +03:00
2. Add `heftia-effects ^>= 0.4` and `ghc-typelits-knownnat ^>= 0.7` to the build dependencies. Enable the [ghc-typelits-knownnat](https://hackage.haskell.org/package/ghc-typelits-knownnat) plugin, `GHC2021`, and the following language extensions as needed:
* `LambdaCase`
* `DerivingStrategies`
* `DataKinds`
* `TypeFamilies`
* `BlockArguments`
* `FunctionalDependencies`
* `RecordWildCards`
* `DefaultSignatures`
* `PatternSynonyms`
Example .cabal:
```
...
build-depends:
...
2024-10-09 21:05:53 +03:00
heftia-effects ^>= 0.4,
ghc-typelits-knownnat ^>= 0.7,
default-language: GHC2021
default-extensions:
...
LambdaCase,
DerivingStrategies,
DataKinds,
TypeFamilies,
BlockArguments,
FunctionalDependencies,
RecordWildCards,
DefaultSignatures,
PatternSynonyms,
TemplateHaskell,
PartialTypeSignatures,
AllowAmbiguousTypes
...
```
If you encounter an error like the following, add the pragma:
```haskell
{-# OPTIONS_GHC -fplugin GHC.TypeLits.KnownNat.Solver #-}
```
to the header of your source file.
Could not deduce GHC.TypeNats.KnownNat (1 GHC.TypeNats.+ ...)
2024-10-10 02:25:30 +03:00
The supported versions are GHC 9.4.1 and later.
This library has been tested with GHC 9.8.2 and 9.4.1.
2024-07-06 16:45:56 +03:00
## Example
### Coroutine-based Composable Concurrent Stream (since v0.5)
Below is an example of using concurrent streams (pipes).
```haskell
{-# OPTIONS_GHC -fplugin GHC.TypeLits.KnownNat.Solver #-}
2024-07-09 19:26:05 +03:00
import Control.Monad.Hefty
import Control.Monad.Hefty.Concurrent.Stream
import Control.Monad.Hefty.Concurrent.Timer
import Control.Monad.Hefty.Except
import Control.Monad.Hefty.Unlift
import Control.Arrow ((>>>))
import Control.Monad (forever, void, when)
import Data.Foldable (for_)
import UnliftIO (bracket_)
-- | Generates a sequence of 1, 2, 3, 4 at 0.5-second intervals.
produce :: (Output Int <| ef, Timer <| ef) => Eff '[] ef ()
produce = void . runThrow @() $
for_ [1 ..] \(i :: Int) -> do
when (i == 5) $ throw ()
output i
sleep 0.5
-- | Receives the sequence at 0.5-second intervals and prints it.
consume :: (Input Int <| ef, Timer <| ef, IO <| ef) => Eff eh ef ()
consume = forever do
liftIO . print =<< input @Int
sleep 0.5
-- | Transforms by receiving the sequence as input at 0.5-second intervals,
-- adds 100, and outputs it.
plus100 :: (Input Int <| ef, Output Int <| ef, Timer <| ef, IO <| ef) => Eff eh ef ()
plus100 = forever do
i <- input @Int
let o = i + 100
liftIO $ putStrLn $ "Transform " <> show i <> " to " <> show o
output o
sleep 0.5
2024-07-09 19:26:05 +03:00
main :: IO ()
main = runUnliftIO . runTimerIO $ do
let produceWithBracket =
bracket_
(liftIO $ putStrLn "Start")
(liftIO $ putStrLn "End")
(raiseAllH produce)
runMachineryIO_ $
Unit @() @Int do
produceWithBracket
produceWithBracket
>>> Unit @Int @Int plus100
>>> Unit @Int @() consume
2024-07-09 19:26:05 +03:00
```
```
>>> main
Start
Transform 1 to 101
101
Transform 2 to 102
102
Transform 3 to 103
103
Transform 4 to 104
104
End
Start
Transform 1 to 101
101
Transform 2 to 102
102
Transform 3 to 103
103
Transform 4 to 104
104
End
2024-07-09 19:26:05 +03:00
```
* Each function (machine unit) `produce`, `consume`, and `plus100` operates with input/output at 0.5-second intervals, but note that the composed stream also maintains operation intervals at 0.5 seconds (not 1.5 seconds!). This means that each unit operates concurrently based on threads.
* `End` is displayed just after the first sequence ends and before the second sequence starts, even though `produceWithBracket` is executed twice in succession. This demonstrates that the `bracket_` function based on `MonadUnliftIO` for safe resource release works in such a way that resources are released immediately at the correct timing—even if the stream is still in progress—rather than waiting until the entire stream (including the second sequence) has completed. Existing stream libraries like [`pipes`](https://hackage.haskell.org/package/pipes) and [`conduit`](https://hackage.haskell.org/package/conduit) have the issue that immediate resource release like this is not possible. This problem was first addressed by the effect system library [`bluefin`](https://github.com/tomjaguarpaw/bluefin). For more details, please refer to [Bluefin streams finalize promptly](https://github.com/tomjaguarpaw/bluefin#bluefin-streams-finalize-promptly).
The complete code example can be found at [heftia-effects/Example/Stream/Main.hs](https://github.com/sayo-hs/heftia/blob/v0.5.0/heftia-effects/Example/Stream/Main.hs).
2023-09-18 05:38:40 +03:00
## Documentation
2024-10-09 21:05:53 +03:00
A detailed explanation of usage and semantics is available in [Haddock](https://hackage.haskell.org/package/heftia-0.4.0.0/docs/Control-Monad-Hefty.html).
2024-10-15 12:45:03 +03:00
The example codes are located in the [heftia-effects/Example/](https://github.com/sayo-hs/heftia/tree/v0.5.0/heftia-effects/Example) directory.
2024-10-09 22:08:56 +03:00
Also, the following *HeftWorld* example (outdated): https://github.com/sayo-hs/HeftWorld
2024-07-06 16:45:56 +03:00
2024-10-10 17:00:34 +03:00
About the internal *elaboration* mechanism: https://sayo-hs.github.io/jekyll/update/2024/09/04/how-the-heftia-extensible-effects-library-works.html
2024-07-06 16:54:45 +03:00
## Comparison
2024-07-14 06:04:00 +03:00
* Higher-Order Effects: Does it support higher-order effects?
* Delimited Continuation: The ability to manipulate delimited continuations.
2024-08-25 15:22:42 +03:00
* Effect System: For a term representing an effectful program, is it possible to statically decidable a type that enumerates all the effects the program may produce?
2024-07-14 06:04:00 +03:00
* Purely Monadic: Is an effectful program represented as a transparent data structure that is a monad, and can it be interpreted into other data types using only pure operations without side effects or `unsafePerformIO`?
2024-07-15 13:32:28 +03:00
* Dynamic Effect Rewriting: Can an effectful program have its internal effects altered afterwards (by functions typically referred to as `handle with`, `intercept`, `interpose`, `transform`, `translate`, or `rewrite`) ?
2024-09-05 00:46:37 +03:00
For example, would it be possible to apply `interpose` as many times as the number of values input by the user at runtime?
* Semantics: Classification of behaviors resulting from the interpretation of effects.
2024-10-09 21:05:53 +03:00
* Algebraic Effects: The same as Algebraic Effects and Handlers.
* IO-fused: IO + ReaderT pattern.
* Carrier dependent: The behavior depends on the specific type inference result of the monad. Tagless-final style.
| Library or Language | Higher-Order Effects | Delimited Continuation | Effect System | Purely Monadic | Dynamic Effect Rewriting | Semantics |
| ------------------- | -------------------- | ---------------------- | --------------| --------------------------------- | ------------------------ | -------------------------------- |
| `heftia` | Yes | Multi-shot | Yes | Yes | Yes | Algebraic Effects |
| `freer-simple` | No | Multi-shot | Yes | Yes | Yes | Algebraic Effects |
| `polysemy` | Yes | No | Yes | Yes | Yes | Weaving-based (functorial state) |
| `effectful` | Yes | No | Yes | No (based on the `IO` monad) | Yes | IO-fused |
2024-10-19 14:23:09 +03:00
| `bluefin` | Yes | No | Yes | No (based on the `IO` monad) | Yes | IO-fused |
2024-10-09 21:05:53 +03:00
| `eff` | Yes | Multi-shot | Yes | No (based on the `IO` monad) | Yes | Algebraic Effects & IO-fused [^6]|
| `speff` | Yes | Multi-shot (restriction: [^4]) | Yes | No (based on the `IO` monad) | Yes | Algebraic Effects & IO-fused |
| `in-other-words` | Yes | Multi-shot? | Yes | Yes | No? | Carrier dependent |
| `mtl` | Yes | Multi-shot (`ContT`) | Yes | Yes | No | Carrier dependent |
| `fused-effects` | Yes | No? | Yes | Yes | No | Carrier dependent & Weaving-based (functorial state) |
2024-10-09 22:08:56 +03:00
| Koka-lang | No | Multi-shot | Yes | No (language built-in) | Yes | Algebraic Effects |
2024-10-17 15:22:40 +03:00
| Eff-lang | No | Multi-shot | Yes | No (language built-in) | Yes | Algebraic Effects |
| OCaml-lang 5 | ? | One-shot | No [^3] | No (language built-in) | ? | Algebraic Effects |
2024-08-25 15:26:16 +03:00
[^3]: Effects do not appear in the type signature and can potentially cause unhandled errors at runtime
2024-09-13 18:23:36 +03:00
[^4]: Scoped Resumption only. e.g. Coroutines are not supported.
2024-09-24 20:43:10 +03:00
[^6]: https://github.com/hasura/eff/issues/12
Heftia can simply be described as a higher-order version of `freer-simple`.
This is indeed true in terms of its internal mechanisms as well.
2023-09-16 15:32:23 +03:00
2024-10-09 21:05:53 +03:00
Additionally, this library provides a consistent algebraic effects semantics that is independent of carriers and effects.
On the other hand, in libraries like `in-other-words`, `mtl`, and `fused-effects`, the semantics of the code depend on the effect and, in part, the carrier inferred by type inference.
Fixing the semantics to a algebraic effects model helps improve the predictability of the behavior (interpretation result) of the code without losing flexibility.
2024-09-05 00:46:37 +03:00
2024-09-05 00:57:44 +03:00
Carrier-dependent semantics can lead to unexpected behavior for code readers, particularly in situations where the types become implicit.
2024-09-05 00:46:37 +03:00
Particularly, attention should be given to the fact that due to type inference, semantic changes may propagate beyond the blocks enclosed by `interpret` or `interpose`.
In the case of carrier-independent semantics, especially with Freer-based effects, `interpret` and `interpose` do not alter the semantics by intervening in type inference or instance resolution of the carrier.
Instead, they function as traditional functions, simply transforming the content of the data structure.
This results in minimal surprise to the mental model of the code reader.
2024-10-09 21:05:53 +03:00
### Performance
2024-07-15 13:32:28 +03:00
2024-10-09 22:08:56 +03:00
Overall, the performance of this library is positioned roughly in the middle between the fast (`effectful`, `eveff`, etc.) and slow (`polysemy`, `fused-effects`, etc.) libraries, and can be considered average.
2024-10-09 21:05:53 +03:00
In all benchmarks, the speed is nearly equivalent to `freer-simple`, only slightly slower.
2024-07-15 13:32:28 +03:00
For more details, please refer to [performance.md](https://github.com/sayo-hs/heftia/blob/v0.5.0/benchmark/performance.md).
2024-07-15 13:32:28 +03:00
### Interoperability with other libraries
2024-07-15 13:32:28 +03:00
#### About `mtl`
2024-07-15 13:32:28 +03:00
* Since the representation of effectful programs in Heftia is simply a monad (`Eff`), it can be used as the base monad for transformers.
This means you can stack any transformer on top of it.
2024-10-03 00:33:23 +03:00
* The `Eff` monad is an instance of `MonadIO`, `MonadError`, `MonadRWS`, `MonadUnliftIO`, `Alternative`, etc., and these behave as the senders for the embedded `IO` or the effect GADTs defined in [data-effects](https://github.com/sayo-hs/data-effects).
2024-07-15 13:32:28 +03:00
#### About `effectful`
* In Heftia, since any monad can be used as the base monad of `Eff`, by setting the `Eff` monad from `effectful` as the base monad of Heftia, you can stack any effect in Heftia on top of `effectful`. In other words, the `Eff` of `Heftia` itself can be used like a monad transformer. This is not limited to `effectful`.
* By using `Control.Monad.Hefty.Unlift.runUnliftIO` instead of `Control.Monad.Hefty.runEff`, you can inherit and use the `MonadUnliftIO` functionality of `effectful`'s `Eff` as a higher-order `UnliftIO` effect within Heftia.
2024-10-09 21:05:53 +03:00
#### Representation of effects
* Heftia relies on [data-effects](https://hackage.haskell.org/package/data-effects) for the definitions of standard effects such as `Reader`, `Writer`, and `State`.
* It is generally recommended to use effects defined with automatic derivation provided by [data-effects-th](https://hackage.haskell.org/package/data-effects-th).
* The representation of first-order effects is compatible with `freer-simple`.
Therefore, effects defined for `freer-simple` can be used as is in this library.
2024-10-09 21:05:53 +03:00
However, to avoid confusion between redundantly defined effects,
it is recommended to use the effects defined in `data-effects`.
2024-10-17 13:38:20 +03:00
* GADTs for higher-order effects are formally similar to `polysemy` and `fused-effects`,
2024-10-09 21:05:53 +03:00
but they need to be instances of the [`HFunctor`](https://hackage.haskell.org/package/compdata-0.13.1/docs/Data-Comp-Multi-HFunctor.html#t:HFunctor) type class.
While it's not impossible to manually derive `HFunctor` for effect types based on these libraries and use them,
it's inconvenient, so it's better to use `data-effects`.
2024-10-17 13:38:20 +03:00
Also, it is not compatible with `effectful` and `eff`.
2024-10-09 21:05:53 +03:00
## Future Plans
2024-10-11 21:01:11 +03:00
* Increase effects and nurture the ecosystem
2024-10-16 10:38:06 +03:00
* ✅ concurrent/parallel programming, streaming, `co-log`: to be released in v0.5
* file system, Subprocesses, POSIX, and so on...
2024-10-11 21:01:11 +03:00
* Write practical software using Heftia
2024-10-10 02:25:30 +03:00
* Support for Applicative effects
2024-10-11 21:01:11 +03:00
* (Support for [Linear](https://hackage.haskell.org/package/linear-base) effects?)
## License
The license is MPL 2.0. Please refer to the [NOTICE](https://github.com/sayo-hs/heftia/blob/v0.5.0/NOTICE).
2024-10-10 14:19:58 +03:00
Additionally, the code from `freer-simple` has been modified and used internally within this library.
Therefore, some modules are licensed under both `MPL-2.0 AND BSD-3-Clause`.
For details on licenses and copyrights, please refer to the module's Haddock documentation.
2023-08-25 07:23:49 +03:00
## Your contributions are welcome!
Please see [CONTRIBUTING.md](https://github.com/sayo-hs/heftia/blob/v0.5.0/CONTRIBUTING.md).
2023-09-18 09:32:37 +03:00
2024-09-08 19:38:26 +03:00
## Acknowledgements, citations, and related work
The following is a non-exhaustive list of people and works that have had a significant impact, directly or indirectly, on Heftias design and implementation:
- Oleg Kiselyov, Amr Sabry, and Cameron Swords — [Extensible Effects: An alternative to monad transfomers][oleg:exteff]
- Oleg Kiselyov and Hiromi Ishii — [Freer Monads, More Extensible Effects][oleg:more]
- Rob Rix, Patrick Thomson, and other contributors — [`fused-effects`][gh:fused-effects]
- Sandy Maguire and other contributors — [`polysemy`][gh:polysemy]
- Alexis King and other contributors — [`freer-simple`][gh:freer-simple], [`eff`][gh:eff]
- Casper Bach Poulsen and Cas van der Rest — [Hefty Algebras: Modular Elaboration of Higher-Order Algebraic Effects][casper:hefty]
- Tom Ellis — [Bluefin streams finalize promptly][tom:bluefin-streams]
2024-09-08 19:38:26 +03:00
[gh:fused-effects]: https://github.com/fused-effects/fused-effects
[gh:polysemy]: https://github.com/polysemy-research/polysemy
[oleg:exteff]: http://okmij.org/ftp/Haskell/extensible/exteff.pdf
[oleg:more]: http://okmij.org/ftp/Haskell/extensible/more.pdf
[casper:hefty]: https://dl.acm.org/doi/10.1145/3571255
[gh:freer-simple]: https://github.com/lexi-lambda/freer-simple
[gh:eff]: https://github.com/lexi-lambda/eff
[tom:bluefin-streams]: https://h2.jaguarpaw.co.uk/posts/bluefin-streams-finalize-promptly/