Monio ('mo-neo') is an async-capable IO Monad (including "do" style) for JS, with several companion monads thrown in.
Monio balances the power of monads -- often dismissed by the non-FP programmer as academic and convoluted -- while pragmatically embracing the reality of the vast majority of JS programs: paradigm mixture (some OO, some FP, and probably a lot of imperative procedural code).
The driving inspiration behind Monio is an IO monad -- useful for managing side-effects -- that additionally supports "do-style" syntax with JS-ergonomic asynchrony (based on promises) in the style of familiar async..await
code. IOs are lazy, so their operations are not triggered until the run(..)
method is called.
Monio's IO is a transformer over promises, which means that when promises are produced in an IO, they are automatically unwrapped; of course, that means subsequent IO operations are deferred. If any IO in a chain produces a promise, run(..)
's result will be "lifted" to a promise that resolves when the entire IO chain is complete. Otherwise, the IO instance and its run(..)
call will operate synchronously and immediately produce the result.
Monio intentionally chooses to model asynchrony over promises instead of Task monads, because of its goal of balancing FP with pragmatic and idomatic non-FP JS. However, there's nothing that should prevent you from using a Task monad with Monio if you prefer.
IO "do-style" syntax is specified with the do(..)
method (automatically lifts the IO to promise-producing asynchrony), which accepts JS generators (including "async generators": async function *whatever(){ .. }
). yield
is used for chaining IOs (which can produce promises to defer), whereas await
is for explicitly deferring on a promise that's not already wrapped in an IO. The resulting style of code should be more broadly approachable for JS developers, while still benefitting from monads.
IO's do(..)
is also JS-ergonomic for exception handling. Uncaught JS exceptions become promise rejections, and IO-produced promise rejections are try..catch
'able. Monio also supports modeling exception handling through Either monads: doEither(..)
transforms uncaught exceptions into Either:Left values, and recognizes IO-produced Either:Left values as try..catch
'able exceptions.
Monio's IO is also a Reader monad, which carries side-effect read environments alongside IO operations.
Monio also includes several supporting monads/helpers in addition to IO
:
-
Maybe
(includingJust
andNothing
) -
Either
-
Monio-specific
AsyncEither
(same promise-transforming behavior as IO) -
IOEventStream(..)
: creates an IO instance that produces an "event stream" -- an async-iterable consumable with afor await..of
loop -- from an event emitter (ie, a DOM element, or a Node EventEmitter instance)
A test suite is included in this repository, as well as the npm package distribution. The default test behavior runs the test suite using the files in src/
.
-
The tests are run with QUnit.
-
To run the test utility with npm:
npm test
-
To run the test utility directly without npm:
qunit
If you have NYC (Istanbul) already installed on your system (requires v14.1+), you can use it to check the test coverage:
npm run coverage
Then open up coverage/lcov-report/index.html
in a browser to view the report.
Note: The npm script coverage:report
is only intended for use by project maintainers. It sends coverage reports to Coveralls.
All code and documentation are (c) 2020 Kyle Simpson and released under the MIT License. A copy of the MIT License is also included.