Safe Haskell | None |
---|---|
Language | Haskell2010 |
Lifted versions of the
, Functor
, Pointed
and Apply
classes, plus template-haskell magic to automatically
derive instances.
"Lifted" because these classes are about datatypes parameterized over a
constructor (i.e. of kind Traversable
(* -> *) -> *
). For example
fmap :: (a -> b) -> f a -> f b
becomes
cMap :: (forall a . f a -> g a) -> c f -> c g
.
For the lifted version of Applicative
, we focus on liftA2
instead of
\<*\>
as this is the only way to make the lifted version work. As a
consequence, the class and method are named after zipWith
because of
the similarity of the signatures and the semantics.
liftA2 :: Applicative f => (g -> h -> i ) -> f g -> f h -> f i zipWith :: (g -> h -> i ) -> [g] -> [h] -> [i] cZipWith :: CZipWith k => (forall a . g a -> h a -> i a) -> k g -> k h -> k i
Types of the corresponding kind occur for example when handling program configuration: When we define our an example configuration type like
data MyConfig f = MyConfig { flag_foo :: f Bool , flag_bar :: f Bool , flag_someLimit :: f Int }
then
MyConfig Maybe
can be used as the result-type of parsing the commandline or a configuration file; it includes the option that some field was not specified;MyConfig Identity
can be used to represent both the default configuration and the actual configuration derived from defaults and the user input;MyConfig (Const Text)
type to represent documentation for our config, to be displayed to the user.
This has the advantage that our configuration is defined in one place only,
so that changes are easy to make and we do not ever run into any internal
desynchonization of different datatypes. And once we obtained the final
config :: MyConfig Identity
, we don't have to think about Nothing
cases
anymore.
can initialize such polymorphic containers, and cPointed
further helps with this use-case, more specifically the merging of
input and default config: we can express the merging of user/default config
CZipWith
:: MyConfig Maybe -> MyConfig Identity -> MyConfig Identity
in terms of
. The instances are simple boilerplate and thus can be realized
using the provided template-haskell.cZipWith
As an example for such usage, the brittany package uses this approach together with using automatically-derived Semigroup-instances that allow merging of config values (for example when commandline args do not override, but are added to those settings read from config file). See the module containing the config type.
Synopsis
- class CFunctor c where
- cMap :: (forall a. f a -> g a) -> c f -> c g
- class CPointed c where
- cPoint :: (forall a. f a) -> c f
- class CZipWith (k :: (Type -> Type) -> Type) where
- cZipWith :: (forall a. g a -> h a -> i a) -> k g -> k h -> k i
- class CZipWith c => CZipWithM c where
- cTraverse :: Applicative m => (forall a. f a -> m (g a)) -> c f -> m (c g)
- cZipWithM :: Applicative m => (forall a. f a -> g a -> m (h a)) -> c f -> c g -> m (c h)
- cSequence :: Applicative m => CZipWithM c => c (Compose m f) -> m (c f)
- deriveCPointed :: Name -> DecsQ
- deriveCZipWith :: Name -> DecsQ
- deriveCZipWithM :: Name -> DecsQ
Documentation
class CZipWith (k :: (Type -> Type) -> Type) where Source #
laws:
This class seems to be some kind of "lifted" version of Applicative
(or rather: of
),
but it also seems to share an important property with the
Distributive
class from the
distributive package,
even when Apply
and Distributive
methods don't appear all that
similar. From the corresponding docs:CZipWith
To be distributable a container will need to have a way to consistently zip a potentially infinite number of copies of itself. This effectively means that the holes in all values of that type, must have the same cardinality, fixed sized vectors, infinite streams, functions, etc. and no extra information to try to merge together.
Especially "all values of that type must have the same cardinality" is
true for instances of CZipWith, the only difference being that the "holes"
are instantiations of the f :: * -> *
to some type, where they are simply
a :: *
for
.Distributive
For many
instances there are corresponding datatypes that
are instances of Distributive
(although they do not seem particularly
useful..), for example:CZipWith
newtype CUnit a f = CUnit (f a) -- corresponding to Identity
data CPair a b f = CPair (f a) (f b) -- corresponding to 'data MonoPair a = MonoPair a a'
-- (Pair being a trivial fixed-size vector example)
data CStream a f = CStream (f a) (CStream a f) -- corresponding to an infinite stream
class CZipWith c => CZipWithM c where Source #
Where CZipWith
is a "lifted Apply
", this is a "lifted Traversable
".
laws:
- naturality
t .
for every applicative transformationcTraverse
f =cTraverse
(t . f)t
- identity
cTraverse
Identity = Identity- composition
cTraverse
(Compose .fmap
g . f) = Compose .fmap
(cTraverse
g) .cTraverse
f
and cZipWithM f k l
must behave like
cTraverse getCompose (cZipWith (x y -> Compose (f x y)) k l)
cTraverse :: Applicative m => (forall a. f a -> m (g a)) -> c f -> m (c g) Source #
cZipWithM :: Applicative m => (forall a. f a -> g a -> m (h a)) -> c f -> c g -> m (c h) Source #
cSequence :: Applicative m => CZipWithM c => c (Compose m f) -> m (c f) Source #
The equivalent of
's Traversable
/sequence
sequenceA
deriveCPointed :: Name -> DecsQ Source #
Derives a cPointed
instance for a datatype of kind (* -> *) -> *
.
Requires that for this datatype (we shall call its argument f :: * -> *
here)
- there is exactly one constructor;
- all fields in the one constructor are either of the form
f x
for somex
or of the formX f
for some typeX
where there is aninstance cPointed X
.
For example, the following would be valid usage:
data A f = A { a_str :: f String , a_bool :: f Bool } data B f = B { b_int :: f Int , b_float :: f Float , b_a :: A f } derivecPointed ''A derivecPointed ''B
This produces the following instances:
instance cPointed A where cPoint f = A f f instance cPointed B where cPoint f = B f f (cPoint f f)
deriveCZipWith :: Name -> DecsQ Source #
Derives a CZipWith
instance for a datatype of kind (* -> *) -> *
.
Requires that for this datatype (we shall call its argument f :: * -> *
here)
- there is exactly one constructor;
- all fields in the one constructor are either of the form
f x
for somex
or of the formX f
for some typeX
where there is aninstance CZipWith X
.
For example, the following would be valid usage:
data A f = A { a_str :: f String , a_bool :: f Bool } data B f = B { b_int :: f Int , b_float :: f Float , b_a :: A f } deriveCZipWith ''A deriveCZipWith ''B
This produces the following instances:
instance CZipWith A where cZipWith f (A x1 x2) (A y1 y2) = A (f x1 y1) (f x2 y2) instance CZipWith B where cZipWith f (B x1 x2 x3) (B y1 y2 y3) = B (f x1 y1) (f x2 y2) (cZipWith f x3 y3)
deriveCZipWithM :: Name -> DecsQ Source #
Derives a CZipWithM
instance for a datatype of kind (* -> *) -> *
.
Requires that for this datatype (we shall call its argument f :: * -> *
here)
- there is exactly one constructor;
- all fields in the one constructor are either of the form
f x
for somex
or of the formX f
for some typeX
where there is aninstance CZipWithM X
.
For example, the following would be valid usage:
data A f = A { a_str :: f String , a_bool :: f Bool } data B f = B { b_int :: f Int , b_float :: f Float , b_a :: A f } deriveCZipWithM ''A deriveCZipWithM ''B
This produces the following instances:
instance CZipWithM A where cZipWithM f (A x1 x2) (A y1 y2) = A <$> f x1 y1 <*> f x2 y2 instance CZipWith B where cZipWithM f (B x1 x2 x3) (B y1 y2 y3) = B <$> f x1 y1 <*> f x2 y2 <*> cZipWithM f x3 y3