{-# LANGUAGE AllowAmbiguousTypes #-}
{-# LANGUAGE RankNTypes #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TypeApplications #-}
module Test.Validity.Optics
( lensSpecOnValid
, lensSpec
, lensSpecOnArbitrary
, lensSpecOnGen
, lensLaw1
, lensLaw2
, lensLaw3
, lensGettingProducesValidOnValid
, lensGettingProducesValid
, lensGettingProducesValidOnArbitrary
, lensGettingProducesValidOnGen
, lensSettingProducesValidOnValid
, lensSettingProducesValid
, lensSettingProducesValidOnArbitrary
, lensSettingProducesValidOnGen
) where
import Lens.Micro
import Lens.Micro.Extras
import Data.GenValidity
import Test.Hspec
import Test.QuickCheck
import Test.Validity.Utils
lensSpecOnValid ::
forall s b. (Show b, Eq b, GenValid b, Show s, Eq s, GenValid s)
=> Lens s s b b
-> Spec
lensSpecOnValid l =
lensSpecOnGen
l
(genValid @b)
"valid values"
shrinkValid
(genValid @s)
"valid values"
shrinkValid
lensSpec ::
forall s b.
( Show b
, Eq b
, GenUnchecked b
, Validity b
, Show s
, Eq s
, GenUnchecked s
, Validity s
)
=> Lens s s b b
-> Spec
lensSpec l =
lensSpecOnGen
l
(genUnchecked @b)
"unchecked values"
shrinkUnchecked
(genUnchecked @s)
"unchecked values"
shrinkUnchecked
lensSpecOnArbitrary ::
forall s b.
( Show b
, Eq b
, Arbitrary b
, Validity b
, Show s
, Eq s
, Arbitrary s
, Validity s
)
=> Lens s s b b
-> Spec
lensSpecOnArbitrary l =
lensSpecOnGen
l
(arbitrary @b)
"arbitrary values"
shrink
(arbitrary @s)
"arbitrary values"
shrink
lensSpecOnGen ::
(Show b, Eq b, Validity b, Show s, Eq s, Validity s)
=> Lens s s b b
-> Gen b
-> String
-> (b -> [b])
-> Gen s
-> String
-> (s -> [s])
-> Spec
lensSpecOnGen l genB genBName shrinkB genS genSName shrinkS = do
parallel $ do
it
(unwords
["satisfies the first lens law for", genBName, "and", genSName]) $
lensLaw1 l genB shrinkB genS shrinkS
it (unwords ["satisfies the second lens law for", genSName]) $
lensLaw2 l genS shrinkS
it
(unwords
["satisfies the third lens law for", genBName, "and", genSName]) $
lensLaw3 l genB shrinkB genS shrinkS
it (unwords ["gets valid values from", genSName, "values"]) $
lensGettingProducesValidOnGen l genS shrinkS
it
(unwords
[ "produces valid values when it is used to set"
, genBName
, "values on"
, genSName
, "values"
]) $
lensSettingProducesValidOnGen l genB shrinkB genS shrinkS
lensLaw1 ::
(Show b, Eq b, Show s)
=> Lens s s b b
-> Gen b
-> (b -> [b])
-> Gen s
-> (s -> [s])
-> Property
lensLaw1 l genB shrinkB genS shrinkS =
forAllShrink genB shrinkB $ \b ->
forAllShrink genS shrinkS $ \s -> view l (set l b s) `shouldBe` b
lensLaw2 :: (Show s, Eq s) => Lens s s b b -> Gen s -> (s -> [s]) -> Property
lensLaw2 l genS shrinkS =
forAllShrink genS shrinkS $ \s -> set l (view l s) s `shouldBe` s
lensLaw3 ::
(Show b, Eq b, Show s, Eq s)
=> Lens s s a b
-> Gen b
-> (b -> [b])
-> Gen s
-> (s -> [s])
-> Property
lensLaw3 l genB shrinkB genS shrinkS =
forAllShrink genB shrinkB $ \b ->
forAllShrink genB shrinkB $ \b' ->
forAllShrink genS shrinkS $ \s ->
set l b' (set l b s) `shouldBe` set l b' s
lensGettingProducesValidOnValid ::
(Show s, GenValid s, Show b, GenValid b) => Lens s s b b -> Property
lensGettingProducesValidOnValid l =
lensGettingProducesValidOnGen l genValid shrinkValid
lensGettingProducesValid ::
(Show s, GenUnchecked s, Show b, Validity b) => Lens s s b b -> Property
lensGettingProducesValid l =
lensGettingProducesValidOnGen l genUnchecked shrinkUnchecked
lensGettingProducesValidOnArbitrary ::
(Show s, Arbitrary s, Show b, Validity b)
=> Lens s s b b
-> Property
lensGettingProducesValidOnArbitrary l =
lensGettingProducesValidOnGen l arbitrary shrink
lensGettingProducesValidOnGen ::
(Validity b, Show b, Show s)
=> Lens s s b b
-> Gen s
-> (s -> [s])
-> Property
lensGettingProducesValidOnGen l genS shrinkS =
forAllShrink genS shrinkS $ \s -> shouldBeValid $ view l s
lensSettingProducesValidOnValid ::
(Show s, GenValid s, Show b, GenValid b, Show t, Validity t)
=> Lens s t a b
-> Property
lensSettingProducesValidOnValid l =
lensSettingProducesValidOnGen l genValid shrinkValid genValid shrinkValid
lensSettingProducesValid ::
(Show s, GenUnchecked s, Show b, GenUnchecked b, Show t, Validity t)
=> Lens s t a b
-> Property
lensSettingProducesValid l =
lensSettingProducesValidOnGen
l
genUnchecked
shrinkUnchecked
genUnchecked
shrinkUnchecked
lensSettingProducesValidOnArbitrary ::
(Show s, Arbitrary s, Show b, Arbitrary b, Show t, Validity t)
=> Lens s t a b
-> Property
lensSettingProducesValidOnArbitrary l =
lensSettingProducesValidOnGen l arbitrary shrink arbitrary shrink
lensSettingProducesValidOnGen ::
(Show s, Show b, Show t, Validity t)
=> Lens s t a b
-> Gen b
-> (b -> [b])
-> Gen s
-> (s -> [s])
-> Property
lensSettingProducesValidOnGen l genB shrinkB genS shrinkS =
forAllShrink genS shrinkS $ \s ->
forAllShrink genB shrinkB $ \b -> shouldBeValid $ set l b s