Generic Programming with Scala & Shapeless part 2
Last year I spent some time playing with, and writing about, Scala & Shapeless - walking through the simple example of generating random test data for a case class.Recently, I have played some more with Shapeless, this time with the goal of generating React (javascript) components for case classes. It was a very similar exercise, but this time I made use of the LabelledGeneric object, so I could access the field names - so I thought I'd re-visit here and talk a bit about some of the internals of what is going on.
Getting started
As before, I had to define implicits for the simple types I wanted to be able to handle, and the starting point is of course accepting a case class as input.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
implicit def caseClassToGenerator[A, Repr <: HList]( | |
implicit generic: LabelledGeneric.Aux[A, Repr], | |
gen: ComponentGenerator[Repr], | |
reactComponent: ReactComponentGenerator[A] | |
): ComponentGenerator[A] = | |
new ComponentGenerator[A] { | |
override def generate = reactComponent.generate(gen.generate) | |
} |
First of all, the method is parameterised with two types: caseClassToGenerator[A, Repr <: HList] A is simply going to be our case class type, and Repr is going to be a Shapeless HList.
Next up, we are expecting several implict method arguments (we will ignore the third implicit for now, that is an implicit that I am using purely for the react side of things - this can be skipped if the method handles everything itself):
implicit generic: LabelledGeneric.Aux[A, Repr], gen: ComponentGenerator[Repr],
Now, as this method's purpose is to handle the input of a case class, and as we are using shapeless, we want to make sure that from that starting input we can transform it into a HList so we can then start dealing with the fields one by one (in other words, this is the first step in converting a case class to a generic list that we can then handle element by element). In this setting, the second implicit argument is asking the compiler to check that we have also defined an appropriate ComponentGenerator (this is my custom type for generating React components) that can handle the generic HList representation (its no good being able to convert the case class to its generic representation, if we then have no means to actually process a generic HList).
Straight forward so far?
The first implicit argument is a bit more interesting. Functionally, all LabelledGeneric.Aux[A, Repr] is doing is asking the compiler to make sure we have an implicit LabelledGeneric instance that can handle converting between our parameter A (the case class input type) and Repr (the HList representation). This implicit means that if we try to pass some type A to this method, the compiler will check that we have a shapeless LabelledGeneric that can handle it - if not, we will get a compile error.
But things get more interesting if we look more at what the .Aux is doing!
Path dependent types & the Aux pattern
The best way to work out what is going on is to just jump into the Shapeless code and have a dig. I will proceed to use Generic as an example, as its a simpler case, but its the same for LabelledGeneric:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
trait Generic[T] extends Serializable { | |
/** The generic representation type for {T}, which will be composed of {Coproduct} and {HList} types */ | |
type Repr | |
/** Convert an instance of the concrete type to the generic value representation */ | |
def to(t : T) : Repr | |
/** Convert an instance of the generic representation to an instance of the concrete type */ | |
def from(r : Repr) : T | |
} |
The inner type, Repr, is what is called a path dependent type, in scala. That is, the type is dependent on the actual instance of the enclosing trait or class. This is a powerful mechanism in Scala (but one that can also catch you out, if you are in the habit of defining classes etc within other classes or traits). This is an important detail for our Generic here, as it could be given any parameter T, so the corresponding HList could be anything, but this makes sure it must match the given case class T - that is, the Repr is dependent on what T is.
To try and get our head around it, let's take a look at an example:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
case class Simple(number: Int, working: Boolean, someText: String) | |
val s = Simple(2, true, "hello") | |
// s: Simple = Simple(2,true,hello) | |
val sGen = Generic[Simple] | |
// sGen: shapeless.Generic[Simple]{type Repr = Int :: Boolean :: String :: shapeless.HNil} = anon$macro$4$1@215c6699 |
So, we still haven't got any closer to what this Aux method is doing, so let's dig into that..
The Aux Pattern
We can see from our code that the Aux method is taking two parameters, firstly A - which is the parameter that our Generic will take, but the Aux method also takes the parameter Repr - which we know (or at least pretend we can guess) corresponds to the path dependent type that is defined nested inside the Generic trait.The best way to work out what is going on from is to take a look at the shapeless code!
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
type Aux[T, Repr0] = Generic[T] { type Repr = Repr0 } |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/** Provides a representation of Generic[T], which has a nested Repr type, as a | |
* type with two type parameters instead. | |
* | |
* Here, we specify T, and we find a Generic.Aux[T,R] by implicit search. | |
* We then use R in the second argument. | |
* Generic.Aux[T, R] is exactly equivalent to Generic[T] { type Repr = R }, | |
* but Scala doesn't allow us to write it this way: | |
* | |
* {{{ | |
* def myMethod[T]()(eqGen: Generic[T] { Repr = R }, reqEq: Eq[egGen.Repr) = ??? | |
* }}} | |
* | |
* The reason is that we are not allowed to have dependencies between arguments | |
* in the same parameter group. So Aux neatly sidesteps this problem. | |
* | |
*/ |
That pretty nicely sums up the Aux pattern - the pattern allows us to essentially promote the result of a type-level computation to the higher level parameter. It can be used for a variety of things where we want to reason about the path dependent types, besides this, but this is a common use for the pattern.
So thats all I wanted to get into for now - you can see the code here, and hopefully with this overview, and the earlier shapeless overview, you can get an understanding of what the LabelledGeneric stuff is doing and how Shapeless is helping me generate React components.
0 comments: