{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TypeApplications #-}
{-# LANGUAGE AllowAmbiguousTypes #-}
module Test.Validity.Persist
( persistSpecOnValid
, persistSpec
, persistSpecOnArbitrary
, persistSpecOnGen
, fromPersistValueAndToPersistValueAreInversesOnGen
) where
import Data.GenValidity
import Control.Monad
import qualified Data.Text as T
import Data.Typeable
import Database.Persist (PersistField(..))
import Test.Hspec
import Test.QuickCheck
import Test.Validity.Utils
persistSpecOnValid ::
forall a. (Show a, Eq a, Typeable a, GenValid a, PersistField a)
=> Spec
persistSpecOnValid = persistSpecOnGen (genValid @a) "valid" shrinkValid
persistSpec ::
forall a. (Show a, Eq a, Typeable a, GenUnchecked a, PersistField a)
=> Spec
persistSpec = persistSpecOnGen (genUnchecked @a) "unchecked" shrinkUnchecked
persistSpecOnArbitrary ::
forall a. (Show a, Eq a, Typeable a, Arbitrary a, PersistField a)
=> Spec
persistSpecOnArbitrary = persistSpecOnGen (arbitrary @a) "arbitrary" shrink
persistSpecOnGen ::
forall a. (Show a, Eq a, Typeable a, PersistField a)
=> Gen a
-> String
-> (a -> [a])
-> Spec
persistSpecOnGen gen genname s =
parallel $ do
let name = nameOf @a
describe ("PersistField " ++ name ++ " (" ++ genname ++ ")") $ do
describe ("fromPersistValue :: PersistValue -> Either Text " ++ name) $
it
(unwords
[ "ensures that toPersistValue and fromPersistValue are inverses for"
, "\"" ++ genname
, name ++ "\"" ++ "'s"
]) $
fromPersistValueAndToPersistValueAreInversesOnGen gen s
fromPersistValueAndToPersistValueAreInversesOnGen ::
(Show a, Eq a, PersistField a) => Gen a -> (a -> [a]) -> Property
fromPersistValueAndToPersistValueAreInversesOnGen gen s =
forAllShrink gen s $ \(a :: a) ->
let encoded = toPersistValue a
errOrDecoded = fromPersistValue encoded
in case errOrDecoded of
Left err ->
expectationFailure $
unlines
[ "Decoding failed with error"
, T.unpack err
, "instead of decoding to"
, show a
, "'encode' encoded it to the persist"
, show encoded
]
Right decoded ->
unless (decoded == a) $
expectationFailure $
unlines
[ "Decoding succeeded, but the decoded value"
, show decoded
, "differs from expected decoded value"
, show a
, "'encode' encoded it to the persist"
, show encoded
]