I recently read a nice blog post about the Reader monad written by Kyle Corbelli. The post uses Haskell for it’s example so I decided to translate it to Scala using the Cats library and put my own words to it.
Have you ever noticed a function that takes input parameters that it’s not using it self, the function only forwards them to other function that it calls.
This post is all about that and how to use the Reader monad to get well like, more readable code.
In the example provided by Kyle we are developing a web page using our own small little library. The core of the library looks like this:
type HTML = String
private def div(children: List[HTML]): HTML =
"<div>" + children.mkString("") + "</div>"
private def h1(children: List[HTML]): HTML =
"<h1>" + children.mkString("") + "</h1>"
private def p(children: List[HTML]): HTML =
"<p>" + children.mkString("") + "</p>"
Three simple small functions that we will use to build the more complex structure that forms our web page.
The page will have the following component hierarchy:
As we can see it’s a pretty standard page.
Notice that some of the components are marked with “needs email”? Since this is a dynamic page we will need to provide the email address each time it’s rendered (the address varies each time).
So let’s try and implement it:
def view(email: String) = div(List(page(email)))
private def page(email: String) =
div(List(
topNav(),
content(email)
))
private def topNav(): HTML =
h1(
List("OurSite.com")
)
private def content(email: String): HTML =
div(List(
h1(List("Custom content for " + email)),
left(),
right(email)
))
private def left(): HTML =
p(List("This is the left side"))
private def right(email: String) =
div(List(article(email)))
private def article(email: String) =
div(List(
p(List("This is an article")),
widget(email)
))
private def widget(email: String) =
div(List(p(List("Hey " + email + ", we've got a great offer for you!"))))
The result if we invoke the view with an email address is:
<div>
<div>
<h1>OurSite.com</h1>
<div>
<h1>Custom content for leopold.niklas@gmail.com</h1>
<p>This is the left side</p>
<div>
<div>
<p>This is an article</p>
<div>
<p>Hey leopold.niklas@gmail.com, we've got a great offer for you!</p>
</div>
</div>
</div>
</div>
</div>
</div>
The code will do the job, but don’t you find it irritating that we have to pass the email address all the way down the hierarchy? Take for example the article
function, it doesn’t use the mail address for anything, it just passes it on to widget
. The code feels a little bit dirty don’t you think?
In fact wouldn’t it be nicer if widget
somehow just could ask for the email address and let artice
go about it’s own business?
Something like this:
private def widget() = {
val email = ask
return div(
List(
p(List("Hey " + email + ", we've got a great offer for you!"))
)
)
}
However we still want the function to be pure, i.e. only depend on the input it is given, can we still do it? Can we use some magic?
Turn’s out that we can!
Let’s refactor widget
so that it’s using the Reader monad:
private def widget(): Reader[Context, HTML] = for (
context <- ask[Context]
) yield div(
List(
p(List("Hey " + context.email + ", we've got a great offer for you!"))
)
)
If we squint our eyes just like so, is this not almost the same as:
private def widget() = {
val email = ask
return div(
List(
p(List("Hey " + email + ", we've got a great offer for you!"))
)
)
}
The magic is provided by the Reader
monad and the ask
helper function.
private def ask[R]: Reader[R, R] = Reader(r => r)
As you can see we are no longer returning HTML, instead we wrap HTML in the Reader monad together with a Context that contains the email address.
case class Context(email: String)
To get the HTML we need to provide the context to the Reader by invoking it’s run method.
val widgetReader: Reader[Context, HTML] = widget();
val widgetResult = widgetReader.run(Context("leopold.niklas@gmail.com"))
assert(widgetResult == "<div><p>Hey leopold.niklas@gmail.com, we've got a great offer for you!</p></div>")
Have we succeeded? Can we just ask for it and still be pure? I say
So let’s apply the same pattern to the rest of the code:
def view: Reader[Context, HTML] = for (
p <- page
) yield div(List(p))
private def page = for (
c <- content
) yield div(List(topNav, c))
private def topNav: HTML =
h1(List("OurSite.com"))
private def content = for (
context <- ask[Context];
r <- right
) yield div(List(h1(List("Custom content for " + context.email)), left, r))
private def ask[R]: Reader[R, R] = Reader(r => r)
private def left: HTML =
p(List("This is the left side"))
private def right = for (
a <- article
) yield div(List(a))
private def article = for (
w <- widget
) yield div(List(p(List("This is an article")), w))
private def widget = for (
context <- ask[Context]
) yield div(List(p(List("Hey " + context.email + ", we've got a great offer for you!"))))
And that’s it.
If you find you’re self providing input arguments to functions that are only forwarded, the Reader
monad could be something for you.
Once again, here is the original blog post that I translated to Scala and put some other words to.
Happy coding!
All the code is available at github.