{-# LANGUAGE DeriveFunctor #-} {-# LANGUAGE GeneralizedNewtypeDeriving #-} module Test.QuickCheck.Special ( Special(..), SpecialValues(..), ) where import Data.Int import Data.Word import Numeric.Natural (Natural) import Test.QuickCheck import qualified Data.ByteString as BS import qualified Data.ByteString.Lazy as BL import qualified Data.Scientific as Scientific import qualified Data.Text as TS import qualified Data.Text.Lazy as TL -- | Additionally to the standard Arbitrary instances, -- this generates special values with a small probability. newtype Special a = Special { getSpecial :: a } deriving (Show, Read, Functor, Bounded, Enum, Eq, Ord, Num, Real, Integral) -- | Provides a list of special values or edge cases class SpecialValues a where -- | Finite list of special values specialValues :: [a] instance SpecialValues Int where specialValues = specialInt instance SpecialValues Int8 where specialValues = specialInt instance SpecialValues Int16 where specialValues = specialInt instance SpecialValues Int32 where specialValues = specialInt instance SpecialValues Int64 where specialValues = specialInt instance SpecialValues Word where specialValues = specialBoundedEnum instance SpecialValues Word8 where specialValues = specialBoundedEnum instance SpecialValues Word16 where specialValues = specialBoundedEnum instance SpecialValues Word32 where specialValues = specialBoundedEnum instance SpecialValues Word64 where specialValues = specialBoundedEnum instance SpecialValues Bool where specialValues = [True, False] instance SpecialValues Float where specialValues = specialIEEE instance SpecialValues Double where specialValues = specialIEEE instance SpecialValues Integer where specialValues = [ 0, 1, -1 , fromIntegral (minBound :: Int64) - 1 , fromIntegral (maxBound :: Int64) + 1 , fromIntegral (maxBound :: Word64) + 1 ] instance SpecialValues Natural where specialValues = [ 0, 1, fromIntegral (maxBound :: Word64) + 1 ] instance SpecialValues Char where specialValues = specialBoundedEnum ++ "\0\a\b\f\n\r\t\v\'\"\\aƤ " instance SpecialValues TS.Text where specialValues = fmap TS.pack specialValues instance SpecialValues TL.Text where specialValues = fmap TL.pack specialValues instance SpecialValues BS.ByteString where specialValues = fmap BS.pack specialValues instance SpecialValues BL.ByteString where specialValues = fmap BL.pack specialValues instance SpecialValues Scientific.Scientific where specialValues = [ fromInteger 0 , fromInteger 1, negate $ fromInteger 1 , Scientific.scientific 1 (-1000), negate $ Scientific.scientific 1 (-1000) , Scientific.scientific 1 (1000), negate $ Scientific.scientific 1 (1000) ] instance SpecialValues a => SpecialValues [a] where specialValues = [[], specialValues] instance SpecialValues a => SpecialValues (Maybe a) where specialValues = Nothing : fmap Just specialValues instance (SpecialValues a, SpecialValues b) => SpecialValues (Either a b) where specialValues = fmap Left specialValues ++ fmap Right specialValues instance (SpecialValues a, SpecialValues b) => SpecialValues (a, b) where specialValues = zip specialValues specialValues class RealFloat a => FloatIEEE a where nan :: a infinity :: a epsilon :: a minDenormal :: a minNormal :: a maxFinite :: a -- echo | gcc -E -dM - | grep _FLT_ instance FloatIEEE Float where nan = 0/0 infinity = 1/0 epsilon = 1.19209289550781250000e-7 minDenormal = 1.40129846432481707092e-45 minNormal = 1.17549435082228750797e-38 maxFinite = 3.40282346638528859812e+38 -- echo | gcc -E -dM - | grep _DBL_ instance FloatIEEE Double where nan = 0/0 infinity = 1/0 epsilon = 2.22044604925031308085e-16 minDenormal = 4.94065645841246544177e-324 minNormal = 2.22507385850720138309e-308 maxFinite = 1.79769313486231570815e+308 instance (Arbitrary a, SpecialValues a) => Arbitrary (Special a) where shrink = fmap Special . shrink . getSpecial arbitrary = fmap Special $ frequency $ list specialValues where list s = (10 * length s, arbitrary) : (fmap (\t -> (1, return t)) s) instance CoArbitrary a => CoArbitrary (Special a) where coarbitrary = coarbitrary . getSpecial specialIEEE :: FloatIEEE a => [a] specialIEEE = [0, 1, -1 , nan , epsilon, -epsilon , infinity, -infinity , minDenormal, -minDenormal , minNormal, -minNormal , maxFinite, -maxFinite ] specialInt :: (Num a, Bounded a) => [a] specialInt = [0, 1, -1, minBound, maxBound, minBound + 1, maxBound - 1] specialBoundedEnum :: (Enum a, Bounded a) => [a] specialBoundedEnum = [minBound, maxBound, succ minBound, pred maxBound]