-
Notifications
You must be signed in to change notification settings - Fork 274
- What are goals of this project?
- How do you determine what falls into the scope of
lens
? - Will you be splitting out a separate
lens-core
package? - Wasn’t
fclabels
ordata-lens
good enough for you?
Q: What are the goals of this project? Why does this project exist?
A: The lens
library exists to provide more composable versions of the abstractions you already know how to use in Haskell. Every Haskell programmer already knows how to work with functors and functions or Foldable
and Traversable
containers.
We simply provide you with a vocabulary for composing them and working with their compositions.
One goal of lens
has been to provide a consistent vocabulary that lets you access and work with pure data of any sort, while retaining the ability to be able to reason about your code with laws.
Q: How do you determine what falls into the scope of lens
?
As a rule lens
incurs no package dependency that is not either in the Haskell Platform or required to implement its own functionality.
That said, we’ve tried to provide a “Batteries Included” API that provides useful tools for operating with anything that does fall into its scope.
Q: Will you be splitting out a separate lens-core
package? The build-depends:
list has a lot of stuff I don’t use.
A: This is on the surface a very reasonable request, but it doesn’t work very well in practice. To implement even basic lens
functionality requires a number of language extensions.
Consider the extensions needed to break out the types and combinators for lenses, traversals, etc. separately from the rest of the package. We’d need Rank2Types
to even write Lens
. Working with indexed lenses needs TypeFamilies
, because without type equality coercions type inference for them is unusable. By the time we get done with Projection
and Iso
we’ve brought in a whole pile of extensions and already tied ourselves to the Glorious Glasgow Haskell Compiler.
Even with just this functionality, implementing these combinators already dragged in the mtl
and a large number of dependencies. We had to define a large number of internal types along the way, types we actually expose elsewhere to the user in the API, like Context
and Bazaar
, which have useful Comonad
instances. This forces us to implement them correctly, orphan those instances or remove functionality.
Since we’re already tied to GHC, and the Template Haskell code generator for makeLenses
and makeClassy
is key to making the library usable, it makes sense to incorporate that directly into the base package. Implementing that brings with it dependencies on containers
.
The combinators in Control.Lens.Plated
are generally useful when working with any Traversal
and we use Plated
internally.
One part we could splinter out into a separate package are the combinators in Control.Lens.Zipper
, but they are sort of the “killer app” for lenses and having them brought into scope by default with the rest of Control.Lens
minimizes confusion and encourages their adoption.
Q: Wasn’t fclabels
or data-lens
good enough for you?
Most of the power of lens
comes from working with generalizations of the notion of a van Laarhoven lens.
None of fclabels
, data-lens
, data-accessor
, lenses
, yall
, etc. provided this style of lens and most had attempted to generalize the idea of a Lens
by shoe-horning a Monad
or some other notion of partiality into the middle of it. This came at the expense of the laws that made working with lenses worth doing.
Providing lenses for any of these libraries required picking up a dependency on a package, which means that it is really impractical or impossible for a reasonably “core” package on hackage to reasonably provide lenses for them.
However, the style of lenses used by lens
can be defined using functions from the Prelude
. No dependencies need be incurred to supply lenses, merely to use them!
There really wasn’t a good library for working with van Laarhoven lens families when lens
was started. lens-family
had tried to be that library, but it required 3 separate packages to work with and used the same names between its Haskell 98 lens-family-core
package and the main lens-family
package. Moreover, it is shackled by Haskell 98. That said, the combinators for working with lenses from lens-family-core
are mostly compatible with the lenses provided by or for use by this package.
By adopting and generalizing van Laarhoven lenses we are able to both provide rigorous laws for each of our lens variants and provide a better user experience, because almost any lens, projection, traversal, isomorphism, etc. that the user goes to reach for can be used with any combinator and it just “does the right thing”. Unlike earlier lens libraries explicit conversions are almost entirely eliminated and the combinatorial explosion of combinators is eliminated along at least one axis.
Q: Where can I learn more about lenses in general?
A: There are a number of resources online. Here are a few:
- The first answer to “lenses, fclabels, data-accessor – which library for structure access and mutation is better” on Stack Overflow goes into some depth about how you can think about lenses and the different design trade-offs between the lens libraries that were then extant in Haskell.
- The author, Edward Kmett, has video on youtube from his presentation on “Lenses: A Functional Imperative” covering the approach originally used in scalaz library for Scala. Slides are available online.
-
Seth Tisue gave an excellent introduction to lenses as provided by Miles Sabin’s shapeless library in Scala. The turtle example from his talk is available in the
examples/
folder.
- If you prefer to learn by example, the
examples/
folder contains a number of fully worked examples, including a playable game of pong and a brainfuck interpreter that were written by nandykins to learn his way aroundlens
.
- The Derivation page of this wiki covers some of the motivation about how and why van Laarhoven lenses compose so well.
- Russell O’Connor wrote a blog post on Polymorphic Update with van Laarhoven Lenses, which inspired Edward Kmett to post on his blog, the Comonad.Reader, Mirrored Lenses the immediate precursor to the approach taken here.
- The idea of a “van Laarhoven” lens goes back farther to Twan van Laarhoven. A number of posts about lenses or “functional references” can be found on
- The term “van Laarhoven” lens was coined by Russell O’Connor in “Functor is to Lens as Applicative is to Biplate: Introducing Multiplate”. The concept now known as a Traversal can be viewed as a “biplate family” mixing the vocabulary from the Mirrored Lenses post and Russell’s paper.
- The
Bazaar
comonad is an indexed version of the Kleene Store comonad from Russell’s “Functor is to Lens as Applicative is to Biplate: Introducing Multiplate”. That type goes back farther to Twan van Laarhoven, who called it aFunList
in “A non-regular data type challenge”.
Q: When should I define or use a Getter
rather than a function?
A: In general you shouldn’t bother defining values that are just a Getter
. It is almost always
a better idea to just supply a function, and then drop it into the chain of lenses or traversals with to
,
or by simply applying it to the final result.