2024-11-27
Monads are structures commonly used to abstract over the explicit passing of context, thus making programs cleaner and easier to understand. However, their underlying implementation is often described overly complicated or is hidden in a mess of convoluted types and instantiations.
If you go to the roots of a monad, you will find surprisingly simple mechanisms. After experimenting with monads myself in bruijn – an untyped, pure language – I found some beauty in purely functional, yet untyped, implementations that I want to share1. As a side effect, the monads become so small that most definitions take only a few characters!
I’ve received some constructive criticism about the inaccessibility of my writing, so here I use a common JavaScript syntax instead of bruijn and try to explain myself better. You can find all the functions as a library in this repository.
(also, none of this should be used in production, this is solely for fun and education)
I start with a primer on functional data structures – we want to encode state without using structures like objects or arrays! We also don’t care about any potential type checks.
A data structure can do (at least) two things: Store data, and extract data. For storing, we need to make sure that the data itself, when interpreted as a function, never gets called – otherwise the data may not be recoverable (or it just throws an error). For extracting, we will use arguments that I like to call selectors. We use these selectors in such a way that the stored data gets applied to the respective selector, thus making our data easily extractable.
For example, let’s say we want a data structure that can hold a single item of two different “types” (or tags!) – a person and an animal. We want the structure to give us information on which type of item it currently stores and we want a method to extract the item of the specific type.
We need two different selectors, one for either type, which are
basically just functions passed as arguments that we call with the
stored value. To store a value as the type “person” or “animal”, we use
two different constructor functions with an argument
value
:
// Person constructor
= value => person => animal => person(value)
Person
// Animal constructor
= value => person => animal => animal(value)
Animal
// Examples
= Person("Lars") // result: person => animal => person("Lars")
examplePerson = Animal("Duck") // result: person => animal => animal("Duck") exampleAnimal
To find out whether the constructed data is a person or an animal, we can use the selector functions. Since the selector function gets applied to the value inside the structure, we ignore the argument and then return the boolean:
// Person check
= personOrAnimal => personOrAnimal(_ => true)(_ => false)
isPerson
// Animal check
= personOrAnimal => personOrAnimal(_ => false)(_ => true)
isAnimal
console.log(isPerson(examplePerson)) // true
console.log(isPerson(exampleAnimal)) // false
Extracting the data from the structure is done in a similar way:
// Person extraction
= person => person(value => value)(_ => _)
getPerson // ---------^^^^^^----------
// this argument is ignored!
// Animal extraction
= animal => animal(_ => _)(value => value)
getAnimal
console.log(getPerson(examplePerson)) // "Lars"
console.log(getAnimal(exampleAnimal)) // "Duck"
This is of course a very minimal example that can be extended arbitrarily. For example, let’s say that a person has multiple properties, like a name and an age. We could now also extend the animal constructor in order to maintain the symmetry, or keep it and modify the functions as follows:
// Person constructor
= name => age => person => animal => person(name)(age)
Person
// Person extraction
= personOrAnimal => personOrAnimal(_ => _ => true)(_ => false)
isPerson // --^- -^- -----^------
// name age animal value
// Person extraction
= person => person(name => age => name)(_ => _)
getPersonName = person => person(name => age => age)(_ => _)
getPersonAge
// And similarly, adapting the ignored argument count:
= value => person => animal => animal(value)
Animal = personOrAnimal => personOrAnimal(_ => _ => false)(_ => true)
isAnimal = animal => animal(_ => _ => _)(value => value) getAnimal
I hope you can see the elegancy and power of this encoding. In fact, many formally studied encodings of pure lambda calculus (which is basically the above – a bunch of anonymous functions) come down to this exact principle of using multiple arguments and a selector function!
For example:
// Church pair constructor
// Single tag: Selector (s)
= a => b => s => s(a)(b)
ChurchPair // ^----^ -^----^-
// values selector
// Church numeral
// Two tags: Successor (s) and Zero (z)
// The stored value is in z via a composition of selectors
= s => z => s(s(s(z)))
churchThree // -^----^-
// selector
// Scott numeral
// Two tags: Successor (s) and Zero (z)
// The stored value is another Scott numeral!
= s1 => z1 => s1(s2 => z2 => s2(s3 => z3 => s3(s4 => z4 => z4)))
scottThree // -^----^-
// selector
The Maybe monad is very common and appears in most modern languages
in some way or another, sometimes with the name Option
. It
can store either one or zero elements and supports checks for whether an
item is stored or not.
For example, if a division of two numbers typically returns another
number, you can modify its return type to a Maybe
, such
that it includes the special case of dividing by zero, where the
function should not return anything. These two different states are
typically called “Just” and “Nothing”, or “Some” and “None”:
// Returns a Maybe: Either "Nothing" or "Just(<value>)"
function divide(a, b) {
if (b === 0)
return Nothing
return Just(a / b)
}
We can encode this structure as tagged unions!
= nothing => just => nothing
Nothing = v => nothing => just => just(v)
Just // --^-- --^^^^--
// value selector
= maybe => maybe(true)(_ => false)
isNothing = maybe => maybe(false)(_ => true)
isJust // ------^------
// ignored value
= just => just()(v => v)
getValue
// or even:
= maybe => maybe("Nothing")(v => "Just " + v)
prettyMaybe
console.log(prettyMaybe(Nothing)) // "Nothing"
console.log(prettyMaybe(Just(42))) // "Just 42"
We could now already work with the division function from above:
= divide(42, 0)
result if (isNothing(result))
console.error("OH NO!")
else
console.log(getValue(result))
When using monads this way, you would need many, potentially nested, if statements. However, this is where the magic of monads would normally set in! Exactly this ugly, explicit tracking of state (here “is nothing”) is what monads try to eliminate.
If JavaScript had native support for monads, the syntax for consecutive actions may look like this:
= +prompt("Enter a number!")
input do {
<- divide(42, input)
a <- divide(42, a)
b <- divide(b, a)
c return(c)
// either Nothing or Just(c) }
Within the do
, the actions get chained together and
stored in variables. However, while divide returns a Maybe monad, the
individual variables are in fact numbers! The numbers are automatically
extracted from Just
and then put into the next statement.
If any division would return Nothing
, the entire chain
would break and return Nothing
.
This behavior inbetween actions such as divide
is
defined by the bind
operation. The final
return
(hereafter called unit
, because of name
conflicts) again wraps the value in a monad (here via
Just
).2
Specifically, the bind
function does two things: Extract
the value of the monad (if existing), and apply it to a given function.
In typed languages this function is then required to return a monad
again, such that the final result of the bind
is always a
monad.
= Just
unit
// maybe is the Maybe monad, f is the function
= maybe => f => {
bind if (isNothing(maybe)) return Nothing
else return f(getValue(maybe))
}
This is not functional enough! Instead, try to understand why the following definition is equivalent:
= maybe => f => maybe(maybe)(f) bind
The previous do {}
block can now be translated using
multiple binds, which is basically what functional programming languages
with such syntax desugar to as well:
= +prompt("Enter a number!")
input
bind(divide(42, input))(a =>
bind(divide(42, a))(b =>
bind(divide(b, a))(c =>
unit(c))))
The Either monad is very similar to the Maybe monad. Instead of
storing either nothing or a value, the Either monad stores two different
values with either the tag “Left” or “Right”. For example, following the
example from above, let’s say our divide
function should
actually return an error message in the division-by-zero case instead of
returning nothing:
// Returns an Either: Either "Left(<value>)" or "Just(<value>)"
function divide(a, b) {
if (b === 0)
return Left("Error: division by zero")
return Right(a / b)
}
The required functions should now be fairly obvious:
= v => left => right => left(v)
Left = v => left => right => right(v)
Right // --^-- -^^^^^--
// value selector
= either => either(_ => true)(_ => false)
isLeft = either => either(_ => false)(_ => true)
isRight // ^-----------^
// ignored value
= left => left(v => v)()
getLeft = right => right()(v => v)
getRight
// or even:
= either => either(v => "Left " + v)(v => "Right " + v)
prettyEither
console.log(prettyEither(Left("error"))) // "Left error"
console.log(prettyEither(Right(42))) // "Right 42"
Chaining the Eithers monadically should work by only passing values tagged as “Right”, while the “Left” case is returned immediately:
// returned values should be tagged as "Right"
= Right
unit
// either is the Either monad, f is the function
= either => f => {
bind if (isLeft(either)) return either
else return f(getRight(either))
}
// or, minified:
= either => f => either(Left)(f) bind
Nested binds will now work as intended:
= +prompt("Enter a number!")
input
bind(divide(42, input))(a =>
bind(divide(42, a))(b =>
bind(divide(b, a))(c =>
unit(c))))
// input=42 => Right(42)
// input=0 => Left("Error: division by zero")
Before we continue with more complex monads, I want to introduce a nicer syntax.
As you’ve probably noticed, writing bind
all the time
becomes annoying and unreadable. That’s why most languages with monads
have a syntax such as the do
notation shown above. Inspired
by Magnus
Tovslid’s work, I’ve implemented something similar using
JavaScript’s generators:
= (unit, bind) => f => {
DO const gen = f()
const next = acc => {
const {done, value} = gen.next(acc)
return done ? unit(value) : bind(value)(next)
}return next()
}
// specific DO instance for Either:
= DO(unit, bind) doEither
With this, we can use a slightly weird, but much more readable
do
notation:
= 42
input = doEither(function* () {
result = yield divide(42, input)
a = yield divide(42, a),
b = yield divide(b, a),
c return c;
})console.log(prettyEither(result)) // "Right 42"
Now onto the more complex monads!
A State monad is useful if you don’t want to use mutable state but still want an elegant way of attaching state to your functions.
Let’s say you have a seeded random number generator and you want to generate three consecutive random numbers. With mutable state, this could be solved like this:
= max => seed => (1103515245 * seed + 12345) % max
rng
= 161
seed = () => (seed = rng(1000)(seed), seed)
rand
console.log([rand(), rand(), rand()]) // [790, 895, 620]
Instead, we construct a bind
that connects the
rand
calls by “mutating” the state automatically.
The State consists of two values, the current state (of course), and another additional variable that can be used for things like accumulating data – otherwise, chained actions could only ever pass the (albeit modified) state. We store these two values in a Church pair, which is nothing more than a tag (selector) with two values.
= v => st => s => s(st)(v)
State // --^-- -^^-- -^----^-
// value state selector
In bind
, the function in the first argument (e.g. a
state transformer like rand
) is then applied to the current
state and produces a new state alongside a value. The value and new
state is then passed to the next function in the binding chain (the
second argument of bind
).
/*
* Pseudocode:
* bind = run => f => s0 => {
* (s1, v) = run(s0)
* return f(v)(s1)
* }
*/
= run => f => s0 => run(s0)(s1 => v => f(v)(s1))
bind // --^^^^^^^^^^--
// uncurry Church
= State
unit
// specific DO instance for State:
= DO(unit, bind) doState
We can now translate the previous example to use monadic state:
// initialize rand with a State tuple with two initial numbers
= seed => (g => State(g)(g))(rng(1000)(seed))
rand
=
threeNumbers bind(rand)(a =>
bind(rand)(b =>
bind(rand)(c =>
unit([a, b, c]))))
// or, simply:
= doState(function* () {
threeNumbers const a = yield rand
const b = yield rand
const c = yield rand
return [a, b, c]
})
console.log(threeNumbers(161)(st => v => v)) // [790, 895, 620]
// -^^^ ^^^^^^^^^^^^
// seed selector
State monads are not only useful for generating random numbers, though. For example, it’s easy to derive a kind of “Writer” monad, that can accumulate logs while working with other data in parallel:
= (a, str) => st => s => s(st + str)(a)
log
= doState(function* () {
deepthought const answer = yield log(42, "Finding answer... ")
const correct = yield log(answer == 42, "Checking answer... ")
if (correct) yield log(null, "Is correct!")
else yield log(null, "Is false!")
return answer
})
console.log(deepthought("")(log => answer => ({answer, log})))
// { answer: 42, log: 'Finding answer... Checking answer... Is correct!' }
You can find some more advanced examples using get
,
put
, modify
, ap
, and
fmap
on GitHub.
Pure languages often use monads for IO, since tracking the inputs and outputs purely would otherwise result in horribly confusing code. However, the internals of typical implementations are also described as “deeply magical”3, so I show a slightly simplified version.
The IO Monad doesn’t normally make sense in a language like
JavaScript, where side effects like prompt
,
alert
, or console.log
can be triggered from
anywhere. Still, having all impure code (the IO
magic) in a
single place allows us to change the entire essence of our side
effects with very small changes, for example if we want to support
several kinds of IO.
Let’s say our IO is based on being able to read and write single characters. We can model the monad as an extension to the State monad:
= doState
doIO
// here we could also add file opening, reading, writing, etc.
= st => s => s(st)(st.read())
read = ch => st => s => s(st)(st.write(ch))
write // ----^^--^^----
// obj of effects
The state here is an object of available IO effects. The value that’s carried along the bind is the result of calling either one of them. There are many different solutions of using this state. We could, for example, track all the IO calls purely, and then lazily evaluate them all in one go at the end. Since JavaScript isn’t lazily evaluated, it makes more sense to call the impure IO effects immediately.
Via recursion, we can then define actions for reading and writing entire lines:
= str => doIO(function* () {
writeLine const head = str[0]
const tail = str.slice(1)
yield write(head)
yield tail === "" ? write('\n') : writeLine(tail)
})
// I don't think doIO via yield is powerful enough for this(?)
= bind(read)(ch =>
readLine === '\r' ? unit("")
ch : bind(readLine)(line => unit(ch + line))
)
Example usage:
= name => age => person => person(name)(age)
Person
= doIO(function* () {
constructPerson yield writeLine("Please enter your name!")
const name = yield readLine
yield writeLine(`Hello ${name}! Now please enter your age.`)
const age = yield readLine
return Person(name)(age) // arbitrary data!
})
Finally, when executing IO actions, we need to pass the initial state that contains the impure IO effects:
// cli effects for nodeJS:
// Note the modularity and how you could swap effects arbitrarily!
= () => {
nodeEffects = require("fs")
fs process.stdin.setRawMode(true)
= Buffer.alloc(1)
buffer = fs.openSync("/dev/tty", "rs")
fd return {
write: process.stdout.write.bind(process.stdout),
read: () => {
.readSync(fd, buffer, 0, 1)
fsreturn buffer.toString("utf8")
}
}
}
console.log(constructPerson(nodeEffects())(st => v => v(
=> age => `Person(name: ${name}, age: ${age})`
name ;
)))
// output: "Please enter your name!"
// input: "Marvin"
// output: "Hello Marvin! Now please enter your age."
// input: "21"
// log: "Person(name: Marvin, age: 21)"
Parsing is another topic where monads come up a lot. If you want to parse a language or some other data, you will need to keep track of the already parsed data as well as the remaining unparsed data. Mutable state makes this seem trivial at first, but at the latest when you descend recursively into a structure with multiple paths, you will probably need to consider backtracking (which could involve rolling back all mutated state to a previous version).
Tracking such data immutably via a parser monad can lead to an elegant yet powerful coding style – for example using parser combinators, where the different parsing steps can be combined modularly via single (often infix) functions.
A minimal parser monad can be constructed as follows: Since our parser can fail, we store all data in an Either monad. Its left case contains an error (for us: a string), and the right case contains a (Church) pair of the already parsed data and the rest of the unparsed input.
= v => left => right => left(v)
Left = v => left => right => right(v)
Right = either => f => either(Left)(f)
eitherBind
= err => s => Left(err) fail
The monadic bind
then consists of applying the parser
p
to the input s
. If the result is
Right
, a parsing function f
should be applied
to the already parsed data and the unparsed rest of the input:
= p => f => s => eitherBind(p(s))(right => right(cur => rst => f(cur)(rst)))
bind
= cur => rst => Right(s => s(cur)(rst))
unit
= DO(unit, bind) doParse
For parsing something based on a specific predicate, we can define a
function satisfy
:
= pred => s => {
satisfy if (s === "") return Left("end of input")
const head = s[0]
const tail = s.slice(1)
return pred(head) ? Right(s => s(head)(tail))
: Left("unexpected " + head)
}
Using this, we can parse a single char as well as a complete string:
= ch => satisfy(c => c == ch)
char
= str => doParse(function* () {
string const head = str[0]
const tail = str.slice(1)
yield char(head)
return yield tail === "" ? unit(str)
: bind(string(tail))(_ => unit(str))
})
Then, let’s say the input to the parser is a string “Hello, World!”. We can run several parsers on it:
= either =>
prettyParser either(v => "Error: " + v)(v => v(cur => rst => ({ cur, rst })))
= "Hello, World!"
input
= char('H')
parser console.log(prettyParser(parser(input)));
// { cur: 'H', rst: 'ello, World!' }
= char('h')
parser console.log(prettyParser(parser(input)));
// Error: unexpected H
= doParse(function* () {
parser const p = yield string("Hello")
yield char(',')
yield char(' ')
return p
})console.log(prettyParser(parser(input)))
// { cur: 'Hello', rst: 'World!' }
console.log(prettyParser(parser("Hallo, Welt!")))
// Error: unexpected a
Now, of course this is really only scratching the surface of what parsers are actually supposed to do – it should still serve as a good starting point though. You can find some further definitions based on the same idea in bruijn’s standard library.
That’s everything for now, thanks for reading. Contact me via email. Support on Ko-fi. Subscribe on RSS. Follow on Mastodon. Program in bruijn.
Imprint · AI Statement · RSS