{-# LANGUAGE CPP #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE GeneralizedNewtypeDeriving #-}
{-# LANGUAGE QuantifiedConstraints #-}
{-# LANGUAGE RankNTypes #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE StandaloneDeriving #-}
{-# OPTIONS_GHC -fno-warn-orphans #-}
module Test.QuickCheck.Classes.FoldableOrd
(
foldableLaws
) where
import Control.Exception (ErrorCall, evaluate, try)
import Control.Monad.Trans.Class (lift)
import Test.QuickCheck hiding ((.&.))
import Test.QuickCheck.Monadic (monadicIO)
import Test.QuickCheck.Property (Property)
import qualified Data.Foldable as F
import qualified Data.Semigroup as SG
import Test.QuickCheck.Classes.Internal
newtype ApplyOrd f a = ApplyOrd { unApplyOrd :: f a }
deriving instance (forall x. Eq x => Eq (f x), Eq a) => Eq (ApplyOrd f a)
deriving instance (forall x. Show x => Show (f x), Show a) => Show (ApplyOrd f a)
deriving instance
( forall x. (Ord x, Arbitrary x) => Arbitrary (f x)
, Ord a
, Arbitrary a
) => Arbitrary (ApplyOrd f a)
deriving instance Ord (Bottom Integer)
deriving instance Ord (VerySmallList Integer)
foldableLaws :: forall proxy f.
(Foldable f, forall a. Show a => Show (f a), forall a. (Arbitrary a, Ord a) => Arbitrary (f a))
=> proxy f -> Laws
foldableLaws = foldableLawsInternal
foldableLawsInternal :: forall proxy f.
(Foldable f, forall a. Show a => Show (f a), forall a. (Arbitrary a, Ord a) => Arbitrary (f a))
=> proxy f -> Laws
foldableLawsInternal p = Laws "Foldable"
[ (,) "fold" $ property $ \(ApplyOrd (a :: f (VerySmallList Integer))) ->
F.fold a == F.foldMap id a
, (,) "foldMap" $ property $ \(ApplyOrd (a :: f Integer)) (e :: QuadraticEquation) ->
let f = VerySmallList . return . runQuadraticEquation e
in F.foldMap f a == F.foldr (mappend . f) mempty a
, (,) "foldr" $ property $ \(e :: LinearEquationTwo) (z :: Integer) (ApplyOrd (t :: f Integer)) ->
let f = runLinearEquationTwo e
in F.foldr f z t == SG.appEndo (foldMap (SG.Endo . f) t) z
, (,) "foldr'" (foldableFoldr' p)
, (,) "foldl" $ property $ \(e :: LinearEquationTwo) (z :: Integer) (ApplyOrd (t :: f Integer)) ->
let f = runLinearEquationTwo e
in F.foldl f z t == SG.appEndo (SG.getDual (F.foldMap (SG.Dual . SG.Endo . flip f) t)) z
, (,) "foldl'" (foldableFoldl' p)
, (,) "foldl1" $ property $ \(e :: LinearEquationTwo) (ApplyOrd (t :: f Integer)) ->
case compatToList t of
[] -> True
x : xs ->
let f = runLinearEquationTwo e
in F.foldl1 f t == F.foldl f x xs
, (,) "foldr1" $ property $ \(e :: LinearEquationTwo) (ApplyOrd (t :: f Integer)) ->
case unsnoc (compatToList t) of
Nothing -> True
Just (xs,x) ->
let f = runLinearEquationTwo e
in F.foldr1 f t == F.foldr f x xs
, (,) "toList" $ property $ \(ApplyOrd (t :: f Integer)) ->
eq1 (F.toList t) (F.foldr (:) [] t)
#if MIN_VERSION_base(4,8,0)
, (,) "null" $ property $ \(ApplyOrd (t :: f Integer)) ->
null t == F.foldr (const (const False)) True t
, (,) "length" $ property $ \(ApplyOrd (t :: f Integer)) ->
F.length t == SG.getSum (F.foldMap (const (SG.Sum 1)) t)
#endif
]
unsnoc :: [a] -> Maybe ([a],a)
unsnoc [] = Nothing
unsnoc [x] = Just ([],x)
unsnoc (x:y:xs) = fmap (\(bs,b) -> (x:bs,b)) (unsnoc (y : xs))
compatToList :: Foldable f => f a -> [a]
compatToList = foldMap (\x -> [x])
foldableFoldl' :: forall proxy f.
(Foldable f, forall a. Show a => Show (f a), forall a. (Arbitrary a, Ord a) => Arbitrary (f a))
=> proxy f -> Property
foldableFoldl' _ = property $ \(_ :: ChooseSecond) (_ :: LastNothing) (ApplyOrd (xs :: f (Bottom Integer))) ->
monadicIO $ do
let f :: Integer -> Bottom Integer -> Integer
f a b = case b of
BottomUndefined -> error "foldableFoldl' example"
BottomValue v -> if even v
then a
else v
z0 = 0
r1 <- lift $ do
let f' x k z = k $! f z x
e <- try (evaluate (F.foldr f' id xs z0))
case e of
Left (_ :: ErrorCall) -> return Nothing
Right i -> return (Just i)
r2 <- lift $ do
e <- try (evaluate (F.foldl' f z0 xs))
case e of
Left (_ :: ErrorCall) -> return Nothing
Right i -> return (Just i)
return (r1 == r2)
foldableFoldr' :: forall proxy f.
(Foldable f, forall a. Show a => Show (f a), forall a. (Arbitrary a, Ord a) => Arbitrary (f a))
=> proxy f -> Property
foldableFoldr' _ = property $ \(_ :: ChooseFirst) (_ :: LastNothing) (ApplyOrd (xs :: f (Bottom Integer))) ->
monadicIO $ do
let f :: Bottom Integer -> Integer -> Integer
f a b = case a of
BottomUndefined -> error "foldableFoldl' example"
BottomValue v -> if even v
then v
else b
z0 = 0
r1 <- lift $ do
let f' k x z = k $! f x z
e <- try (evaluate (F.foldl f' id xs z0))
case e of
Left (_ :: ErrorCall) -> return Nothing
Right i -> return (Just i)
r2 <- lift $ do
e <- try (evaluate (F.foldr' f z0 xs))
case e of
Left (_ :: ErrorCall) -> return Nothing
Right i -> return (Just i)
return (r1 == r2)
{-# ANN module ("HLint: ignore" :: String) #-}