1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40
|
{-# LANGUAGE AllowAmbiguousTypes #-}
{-# LANGUAGE ConstraintKinds #-}
module Spec.Distributive ( laws )
where
import Clothes (F, G, H, GH(..), NatTransf(..))
import Data.Functor.Identity (Identity(..))
import Data.Functor.Compose (Compose (..))
import Data.Functor.Barbie (FunctorB(..), DistributiveB(..))
import Data.Typeable (Typeable, typeRep, Proxy(..))
import Test.Tasty(testGroup, TestTree)
import Test.Tasty.QuickCheck(Arbitrary(..), testProperty, (===))
type IsDomain a = (Arbitrary a, Show a)
type IsRange a = (Eq a, Show a)
laws
:: forall b
. ( DistributiveB b
, IsDomain (b F)
, IsRange (b (Compose H F))
, IsRange (b (Compose Identity F))
, IsRange (b (Compose (Compose H G) F))
, Typeable b
)
=> TestTree
laws
= testGroup (show (typeRep (Proxy :: Proxy b)))
[ testProperty "naturality" $ \(GH (NatTransf h)) (fb :: G (b F)) ->
bdistribute (h fb) === bmap (Compose . h . getCompose) (bdistribute fb)
, testProperty "identity" $ \(b :: b F) ->
bdistribute (Identity b) === bmap (Compose . Identity) b
, testProperty "composition" $ \(fb :: H (G (b F))) ->
bdistribute (Compose fb) === bmap (Compose . Compose . fmap getCompose . getCompose) (bdistribute . fmap bdistribute $ fb)
]
|