Let’s imagine for a moment that we are in the presence of the following data type:
data Coffee = MkCoffee { coffeeBeans :: String
, coffeeOriginCountry :: Country
, coffeeBrewMethod :: BrewMethod
}
deriving (Generic)
However, we have only been given the type itself - and not a constructor for this value. That is, we don’t have access to this function:
MkCoffee :: String -> Country -> BrewMethod -> Coffee
Is it still possible to make Coffee
? As we’ll see, by making use of the Generics
type class it is indeed possible to make Coffee
, and with a little bit of functor tricky, we can derive an implementation of the MkCoffee
data value constructor ourselves.
Before we get going, it’s worth discussing briefly what is meant be the idea of a generic representation, specifically GHC generics. The term “generic programming” is fairly overloaded, but for this post we’re interested in the idea of shape generics. This (and the other types of generic programming) are discussed in José Pedro Magalhães’ thesis “Less is More”, but loosely speaking we’re interested in abstracting over the “shape” of data types. This means we find another value that isomorphic to our, but is made out of a smaller (and importantly, closed) set of primitives.
GHC Generics is one such approach to shape genericity - Rep
type is the type of isomorphic data, and we have to
and from
to move between both sides of the isomorphism. Let’s see what this all means for Coffee
:
> :kind! (Rep Coffee)
Rep Coffee :: * -> *
= M1
D
Main.D1Coffee
(M1
C
Main.C1_0Coffee
(M1 S Main.S1_0_0Coffee (K1 R [Char])
:*: (M1 S Main.S1_0_1Coffee (K1 R Country)
:*: M1 S Main.S1_0_2Coffee (K1 R BrewMethod))))
This is neither pretty nor succinct, but thankfully a lot of this is noise (for our purposes). Here’s another look at just the essential structure of Rep Coffee
:
K1 [Char] :*: (K1 R Country :*: K1 R BrewMethod)
Now we can see that Coffee
is isomorphic to the product of a string, a country and a BrewMethod
. These correspond to the three fields in the above Coffee
data type.
Using these generic representations, we can construct new Coffee
s:
> to (M1 (M1 (M1 (K1 "Single Origin") :*: (M1 (K1 (Country "Rwanda")) :*: M1 (K1 V60))))) :: Coffee
MkCoffee { coffeeBeans = "Single Origin"
, coffeeOriginCountry = Country "Rwanda"
, coffeeBrewMethod = V60
}
That’s pretty cool, no? It’s this idea that is at the heart of generic programming. We could imagine doing a similar thing by building these values by reading a binary serialisation from a file, or walking a JSON AST.
However, for our purposes we’ve not yet reached our goal. To recap, we really want to be able to build a function like MkCoffee
, keeping the generic representation behind the scenes. To do this, we’ll need to interpret the generic representation into a function.
The standard way to work with a generic representation is to walk the tree using instances of a type class. We’ll do the same, and walk the generic representation to reach a functor that will contain Rep Coffee
. Later, we’ll be able to use fmap to
, turning Rep Coffee
into real Coffee
.
Our workhorse is the following type class.
class Functor f => Mk rep f | rep -> f where
mk :: f (rep a)
As you can see, it’s a multi-parameter type class, taking the generic Rep
type, and also indicating which functor can “build” this Rep
. Each Rep
uniquely determines the constructing functor, which we indicate with a functional dependency. This is essential for having sane type inference.
Starting “at the bottom”, we can begin by constructing a single field. In our Coffee
example, we need to construct String
s, Country
s, and BrewMethod
s. In GHC Generics, each of these is represented with K1
. To actually construct K1
we need a value, so our constructing functor will be a one-argument function:
-- Remember that ((->) c) means (c ->)
instance Mk (K1 i c) ((->) c) where
mk = \x -> K1 x
A Coffee
is more than just a single field though, and we need a way to combine individual fields together. This is done by the use of the :*:
constructor, which we can think of as having “fields on the left” and “fields on the right”. To construct :*:
we need to compose a builder on the left with the builder on the right, so we use Compose
to join the left and right functors into one. The definition of mk
itself is a little cumbersome, but does the job:
instance (Mk l fl, Mk r fr) => Mk (l :*: r) (Compose fl fr) where
mk = Compose (fmap (\l -> fmap (\r -> l :*: r) mk) mk)
Finally, we just need to construct the M1
layers, which just hold meta-data. These don’t have any effect on what we’re trying to do, so their type class instance simply proxy through to other instances:
instance (Mk f f') => Mk (M1 i c f) f' where
mk = M1 <$> mk
Believe it or not, we’re now a good step closer to getting what we want. Let’s have a look and see what we get if we try and mk
some Coffee
:
fmap (to :: Rep Coffee a -> Coffee) mk
:: Compose
((->) String) (Compose ((->) Country) ((->) BrewMethod)) Coffee
Hey, that’s actually pretty close! If we squint, this is some sort of function that takes a String
, a Country
and a BrewMethod
and yields some Coffee
. All that we have to do is somehow get rid of all the Compose
noise.
At this point, I was originally stuck, but then I remembered smart people have already come before me to work on very similar problems. Specifically, Ralf Hinze published a Functional Pearl a while ago called “Formatting: A Class Act”, which uses an identical construction (which I can honestly say was a happy accident!). Hinze then goes a step further with this magic little type class:
class Functor f => Apply f a b | f a -> b where
apply :: f a -> b
instance Apply ((->) a) b (a -> b) where
apply = id
instance (Apply g a b, Apply f b c) => Apply (Compose f g) a c where
apply (Compose x) = apply (fmap apply x)
This time we have a type class of three types (!), where the first two types determine the third. However, it’s not so bad - f
is the functor we need to expand, a
is the type of data under the functor and b
is the final type after expansion. If you look at the specific instances, it should be clear how this all plays out.
Now we can use apply
with mk
to generate a function!
> apply (fmap to mk) :: String -> Country -> BrewMethod -> Coffee
<interactive>:1:2:
No instance for (Apply
f0 a0 (String -> Country -> BrewMethod -> Coffee))
arising from a use of ‘apply’
The type variables ‘f0’, ‘a0’ are ambiguous
Note: there are several potential instances:
instance (Apply g a b, Apply f b c) => Apply (Compose f g) a c
-- Defined at 2014-04-26-coffee.hs:39:10
instance Apply ((->) a) b (a -> b)
-- Defined at 2014-04-26-coffee.hs:36:10
Damn! What went wrong?
Unfortunately, if we use apply
with mk
we lose parametricity on f
itself. There could be many different ways to reach the type we desire (especially as type classes are open), and it turns out that the crucial ingredient is having rep
available.
However, from our perspective it should be possible to infer all of this from the return type of the function we are building. Of course, GHC can’t guess, so we will need to encode this information somehow. With GHC 7.8 we can easily express this with a closed type family:
type family Returns (f :: *) :: * where
Returns (a -> b) = Returns b
Returns r = r
Returns
lets us figure out what the final type of a function is, and now we can complete the loop and build our final make
function:
make :: forall b f z. (Generic (Returns b), Apply f (Returns b) b, Mk (Rep (Returns b)) f) => b
make = apply (fmap (to :: Rep (Returns b) z -> (Returns b)) (mk :: f (Rep (Returns b) z)))
We need to use ScopedTypeVariables
to carry a bit of extra type information around, but luckily this is all behind the scenes.
Finally, we can now write mkCoffee
:
mkCoffee :: String -> Country -> BrewMethod -> Coffee
mkCoffee = make
Does it work?
> mkCoffee "Single Origin" (Country "Rwanda") V60
Coffee { coffeeBeans = "Single Origin"
, coffeeOriginCountry = Country "Rwanda"
, coffeeBrewMethod = V60
}
Tada!
You can contact me via email at ollie@ocharles.org.uk or tweet to me @acid2. I share almost all of my work at GitHub. This post is licensed under a Creative Commons Attribution-NonCommercial-NoDerivs 3.0 Unported License.
I accept Bitcoin donations: 14SsYeM3dmcUxj3cLz7JBQnhNdhg7dUiJn
. Alternatively, please consider leaving a tip on