Additional accompaniment to Functional and Reactive Domain Modeling code samples. The idea is to introduce some of the implementations which are currently not covered in the text or the code samples.
As with any other modeling exercise we can build domain models at various levels of abstractions. In some cases (possibly with the simplest implementations), we specialize implementations too early. This is ok when we have a very simple implementation at hand. But for any non-trivial implementations, generalization is the way to go. As Lars Hupel said once, premature specialization is the root of all evil.
In functional programming we like to program with values and values can be reasoned about. Instead of directly writing a piece of code that executes when run, we would like to design our program as a value. This is as much a value as an Int
or a List[Int]
. This value abstracts the what of a program - it does not have any idea about how it will be executed. This value may be submitted to the runtime that executes it immediately or it can be submitted for later execution in a completely asynchronous manner. But irrespective of how it is executed, the program as a value remains the same in all the cases. Hence it is also not surprising that programs as values can be composed with other compatible values - much like we can combine two lists by appending one to the other. This gives better modularity and reusability.
In this accompaniment we will consider effects as values. Note an effect is something expressed in the form of a type constructor F[A]
, where A
is the result that the effect computes and F
is the additional stuff modeling the computation. Some examples of effects are IO[A]
, Option[A]
, Either[A, B]
etc. Instead of directly executing a repository action within a domain service, we will model them as effects and construct a value out of our service. This value can then be composed with other services (which are also modeled as values) to yield larger services. This way we build up abstractions of higher level services incrementally from smaller ones and build our domain model. Finally when we have the complete service definition built we will submit it to the appropriate execution engine by assembling appropriate runtimes for execution.
The domain model we will start with is the one we discussed in Chapter 6 of Functional and Reactive Domain Modeling.
Chapter 6 of the book contains complete implementations of reactive domain models using the various flavors of non blocking paradigms - scala.concurrent.Future
, scalaz.concurrent.Task
etc. I take these implementations to the next step by making them more algebraic, more referentially transparent and more compositional.
The build in this project now contains 4 implementations:
-
cats-effect based: Implementation of a domain model using cats-effect IO. The domain model is almost similar to the one in the book. This is the simplest one specialized on cats-effect IO.
-
tagless-final based: Same domain model based on the tagless final approach, with the following concrete implementations:
- cats-effect IO
- Monix Task
This implementation of tagless final closely follows the approach that I discussed in one of the blog posts quite some time back.
-
mtl tagless-final: Same domain implemented with mtl style APIs from cats-mtl alongside the algebraic approach of tagless-final. Here are some features of this implementation:
- In order to have optimum performance it eschews the monad transformer stack and uses hand rolled instances of the typeclasses. An awesome presentation on this topic was given by Pawel Szulc (@rabbitonweb on Twitter). Also Luka Jacobowitz wrote a nice blogpost on this topic on the typelevel blog.
- Has a complete implementation of
AccountRepository
using skunk in addition to the in-memory implementation - Features a complete runnable application with proper modularity and configurations. This is based on the techniques discussed in the excellent book Practical FP in Scala by Gabriel Volpe
- Demonstrates the usage of several other functional libraries like ciris for functional configuration management, enumeratum for typesafe reflection free enumerations and refined for refinement types
-
zio based: The same domain model implemented with ZIO using the bifunctor based abstraction
ZIO[R, E, A]
, whereR
is the environment,E
is the exception that can arise out of the execution andA
is the result that the effect generates. Thanks to zio-todo-backend for a nice illustrative example that helped in the implementation. Here are some features of this implementation:- Features usage of
ZLayer
for modularity and dependency injection of the application - Has a complete implementation of
AccountRepository
using doobie in addition to the in-memory implementation
- Features usage of
Note: Each implementation is self complete and can be learnt independently. The domain model is almost identical with some minor differences.