----------------------------------------------------------------------------- -- | -- Module : Control.Comonad -- Copyright : (C) 2008-2011 Edward Kmett, -- (C) 2004 Dave Menendez -- License : BSD-style (see the file LICENSE) -- -- Maintainer : Edward Kmett -- Stability : provisional -- Portability : portable -- -- A 'Comonad' is the categorical dual of a 'Monad'. ---------------------------------------------------------------------------- module Control.Comonad ( -- * Functor and Comonad Functor(..) , Comonad(..) -- * Functions -- ** Naming conventions -- $naming -- ** Operators , (=>=) -- :: Comonad w => (w a -> b) -> (w b -> c) -> w a -> c , (=<=) -- :: Comonad w => (w b -> c) -> (w a -> b) -> w a -> c , (=>>) -- :: Comonad w => w a -> (w a -> b) -> w b , (<<=) -- :: Comonad w => (w a -> b) -> w a -> w b -- * Fixed points and folds , wfix -- :: Comonad w => w (w a -> a) -> a , unfoldW -- :: Comonad w => (w b -> (a,b)) -> w b -> [a] -- ** Comonadic lifting , liftW -- :: Comonad w => (a -> b) -> w a -> w b -- * Comonads with Zipping , ComonadZip(..) , (<..>) -- :: ComonadZip w => w a -> w (a -> b) -> w b , liftW2 -- :: ComonadZip w => (a -> b -> c) -> w a -> w b -> w c , liftW3 -- :: ComonadZip w => (a -> b -> c -> d) -> w a -> w b -> w c -> w d -- * Cokleisli Arrows , Cokleisli(..) ) where import Prelude hiding (id, (.)) import Control.Applicative import Control.Arrow import Control.Category import Control.Monad.Trans.Identity import Data.Functor import Data.Functor.Identity import Data.Monoid infixl 1 =>> infixr 1 <<=, =<=, =>= infixl 4 <.>, <., .>, <..> {-| There are two ways to define a comonad: I. Provide definitions for 'extract' and 'extend' satisfying these laws: > extend extract = id > extract . extend f = f > extend f . extend g = extend (f . extend g) In this case, you may simply set 'fmap' = 'liftW'. These laws are directly analogous to the laws for monads and perhaps can be made clearer by viewing them as laws stating that Cokleisli composition must be associative, and has extract for a unit: > f =>= extract = f > extract =>= f = f > (f =>= g) =>= h = f =>= (g =>= h) II. Alternately, you may choose to provide definitions for 'fmap', 'extract', and 'duplicate' satisfying these laws: > extract . duplicate = id > fmap extract . duplicate = id > duplicate . duplicate = fmap duplicate . duplicate In this case you may not rely on the ability to define 'fmap' in terms of 'liftW'. You may of course, choose to define both 'duplicate' /and/ 'extend'. In that case you must also satisfy these laws: > extend f = fmap f . duplicate > duplicate = extend id > fmap f = extend (f . extract) These are the default definitions of 'extend' and'duplicate' and the 'default' definition of 'liftW' respectively. -} class Functor w => Comonad w where -- | aka coreturn extract:: w a -> a -- | aka cojoin duplicate :: w a -> w (w a) -- | aka cobind extend :: (w a -> b) -> w a -> w b extend f = fmap f . duplicate duplicate = extend id -- | A suitable default definition for 'fmap' for a 'Comonad'. -- Promotes a function to a comonad. liftW :: Comonad w => (a -> b) -> w a -> w b liftW f = extend (f . extract) {-# INLINE liftW #-} -- | 'extend' with the arguments swapped. Dual to '>>=' for a 'Monad'. (=>>) :: Comonad w => w a -> (w a -> b) -> w b (=>>) = flip extend {-# INLINE (=>>) #-} -- | 'extend' in operator form (<<=) :: Comonad w => (w a -> b) -> w a -> w b (<<=) = extend {-# INLINE (<<=) #-} -- | Right-to-left Cokleisli composition (=<=) :: Comonad w => (w b -> c) -> (w a -> b) -> w a -> c f =<= g = f . extend g {-# INLINE (=<=) #-} -- | Left-to-right Cokleisli composition (=>=) :: Comonad w => (w a -> b) -> (w b -> c) -> w a -> c f =>= g = g . extend f {-# INLINE (=>=) #-} -- | A generalized comonadic list anamorphism unfoldW :: Comonad w => (w b -> (a,b)) -> w b -> [a] unfoldW f w = fst (f w) : unfoldW f (w =>> snd . f) -- | Comonadic fixed point wfix :: Comonad w => w (w a -> a) -> a wfix w = extract w (extend wfix w) -- * Comonads for Prelude types: -- Instances: While Control.Comonad.Instances would be more symmetric with the definition of -- Control.Monad.Instances in base, the reason the latter exists is because of Haskell 98 specifying -- the types Either a, ((,)m) and ((->)e) and the class Monad without having the foresight to require -- or allow instances between them. Here Haskell 98 says nothing about Comonads, so we can include the -- instances directly avoiding the wart of orphan instances. instance Comonad ((,)e) where extract = snd duplicate ~(e,a) = (e,(e,a)) instance Monoid m => Comonad ((->)m) where extract f = f mempty duplicate f m = f . mappend m -- * Comonads for types from 'transformers'. -- This isn't really a transformer, so i have no compunction about including the instance here. -- TODO: Petition to move Data.Functor.Identity into base instance Comonad Identity where extract = runIdentity extend f = Identity . f duplicate = Identity -- Provided to avoid an orphan instance. Not proposed to standardize. -- If Comonad moved to base, consider moving instance into transformers? instance Comonad w => Comonad (IdentityT w) where extract = extract . runIdentityT extend f (IdentityT m) = IdentityT (extend (f . IdentityT) m) {- | As a symmetric semi-monoidal comonad, an instance of ComonadZip is required to satisfy: > extract (a <.> b) = extract a (extract b) Minimal definition: '<.>' Based on the ComonadZip from \"The Essence of Dataflow Programming\" by Tarmo Uustalu and Varmo Vene, but adapted to fit the programming style of Control.Applicative. -} class Comonad w => ComonadZip w where (<.>) :: w (a -> b) -> w a -> w b (.>) :: w a -> w b -> w b (<.) :: w a -> w b -> w a a .> b = const id <$> a <.> b a <. b = const <$> a <.> b instance Monoid m => ComonadZip ((,)m) where (<.>) = (<*>) instance Monoid m => ComonadZip ((->)m) where (<.>) = (<*>) instance ComonadZip Identity where (<.>) = (<*>) instance ComonadZip w => ComonadZip (IdentityT w) where IdentityT wa <.> IdentityT wb = IdentityT (wa <.> wb) -- | A variant of '<.>' with the arguments reversed. (<..>) :: ComonadZip w => w a -> w (a -> b) -> w b (<..>) = liftW2 (flip id) {-# INLINE (<..>) #-} -- | Lift a binary function into a comonad with zipping liftW2 :: ComonadZip w => (a -> b -> c) -> w a -> w b -> w c liftW2 f a b = f <$> a <.> b {-# INLINE liftW2 #-} -- | Lift a ternary function into a comonad with zipping liftW3 :: ComonadZip w => (a -> b -> c -> d) -> w a -> w b -> w c -> w d liftW3 f a b c = f <$> a <.> b <.> c {-# INLINE liftW3 #-} -- | The 'Cokleisli' 'Arrow's of a given 'Comonad' newtype Cokleisli w a b = Cokleisli { runCokleisli :: w a -> b } instance Comonad w => Arrow (Cokleisli w) where arr f = Cokleisli (f . extract) first f = f *** id second f = id *** f Cokleisli f *** Cokleisli g = Cokleisli (f . fmap fst &&& g . fmap snd) Cokleisli f &&& Cokleisli g = Cokleisli (f &&& g) instance Comonad w => Category (Cokleisli w) where id = Cokleisli extract Cokleisli f . Cokleisli g = Cokleisli (f =<= g) instance Comonad w => ArrowApply (Cokleisli w) where app = Cokleisli $ \w -> runCokleisli (fst (extract w)) (snd <$> w) instance Comonad w => ArrowChoice (Cokleisli w) where left = leftApp instance ComonadZip d => ArrowLoop (Cokleisli d) where loop (Cokleisli f) = Cokleisli (fst . wfix . extend f') where f' wa wb = f ((,) <$> wa <.> (snd <$> wb)) instance Functor (Cokleisli w a) where fmap f (Cokleisli g) = Cokleisli (f . g) instance Monad (Cokleisli w a) where return a = Cokleisli (const a) Cokleisli k >>= f = Cokleisli $ \w -> runCokleisli (f (k w)) w {- $naming The functions in this library use the following naming conventions, based on those of Control.Monad. * A postfix \'@W@\' always stands for a function in the Cokleisli category: The monad type constructor @w@ is added to function results (modulo currying) and nowhere else. So, for example, > filter :: (a -> Bool) -> [a] -> [a] > filterW :: Comonad w => (w a -> Bool) -> w [a] -> [a] * A prefix \'@w@\' generalizes an existing function to a comonadic form. Thus, for example: > fix :: (a -> a) -> a > wfix :: w (w a -> a) -> a When ambiguous, consistency with existing Control.Monad combinator naming supercedes these rules (e.g. 'liftW') -}