module Test.Framework.Providers.QuickCheck2 (
testProperty
) where
import Test.Framework.Providers.API
import Test.QuickCheck.Gen
import Test.QuickCheck.Property hiding ( Property, Result( reason, interrupted ) )
import qualified Test.QuickCheck.Property as P
import Test.QuickCheck.Test
import Test.QuickCheck.Text
import Test.QuickCheck.State
import qualified Control.Exception.Extensible as E
import System.Random
testProperty :: Testable a => TestName -> a -> Test
testProperty name = Test name . Property
instance TestResultlike PropertyTestCount PropertyResult where
testSucceeded = propertySucceeded
type PropertyTestCount = Int
data PropertyResult = PropertyResult {
pr_status :: PropertyStatus,
pr_used_seed :: Int,
pr_tests_run :: Maybe PropertyTestCount
}
data PropertyStatus = PropertyOK
| PropertyArgumentsExhausted
| PropertyFalsifiable String
| PropertyNoExpectedFailure
| PropertyTimedOut
| PropertyException String
instance Show PropertyResult where
show (PropertyResult { pr_status = status, pr_used_seed = used_seed, pr_tests_run = mb_tests_run })
= case status of
PropertyOK -> "OK, passed " ++ tests_run_str ++ " tests"
PropertyArgumentsExhausted -> "Arguments exhausted after " ++ tests_run_str ++ " tests"
PropertyFalsifiable fail_reason -> "Falsifiable with seed " ++ show used_seed ++ ", after " ++ tests_run_str ++ " tests. Reason: " ++ fail_reason
PropertyNoExpectedFailure -> "No expected failure with seed " ++ show used_seed ++ ", after " ++ tests_run_str ++ " tests"
PropertyTimedOut -> "Timed out after " ++ tests_run_str ++ " tests"
PropertyException text -> "Got an exception: " ++ text
where
tests_run_str = fmap show mb_tests_run `orElse` "an unknown number of"
propertySucceeded :: PropertyResult -> Bool
propertySucceeded property_result = propertyStatusIsSuccess (pr_status property_result)
propertyStatusIsSuccess :: PropertyStatus -> Bool
propertyStatusIsSuccess PropertyOK = True
propertyStatusIsSuccess PropertyArgumentsExhausted = True
propertyStatusIsSuccess _ = False
data Property = forall a. Testable a => Property a
instance Testlike PropertyTestCount PropertyResult Property where
runTest topts (Property testable) = runProperty topts testable
testTypeName _ = "Properties"
runProperty :: Testable a => CompleteTestOptions -> a -> IO (PropertyTestCount :~> PropertyResult, IO ())
runProperty topts testable = do
(seed, state) <- initialState topts
runImprovingIO $ do
mb_result <- maybeTimeoutImprovingIO (unK (topt_timeout topts)) $ myTest state (unGen (property testable))
return $ toPropertyResult seed $ case mb_result of
Nothing -> (PropertyTimedOut, Nothing)
Just (status, tests_run) -> (status, Just tests_run)
where
toPropertyResult seed (status, mb_tests_run) = PropertyResult {
pr_status = status,
pr_used_seed = seed,
pr_tests_run = mb_tests_run
}
initialState :: CompleteTestOptions -> IO (Int, State)
initialState topts = do
(gen, seed) <- newSeededStdGen (unK $ topt_seed topts)
tm <- newStdioTerminal
let max_success = unK $ topt_maximum_generated_tests topts
max_size = maxSize stdArgs
return $ (seed, MkState {
terminal = tm
, maxSuccessTests = unK $ topt_maximum_generated_tests topts
, maxDiscardedTests = unK $ topt_maximum_unsuitable_generated_tests topts
, computeSize = \n d -> (n * max_size) `div` max_success + (d `div` 10)
, numSuccessTests = 0
, numDiscardedTests = 0
, collected = []
, expectedFailure = False
, randomSeed = gen
, numSuccessShrinks = 0
, numTryShrinks = 0 })
myTest :: State -> (StdGen -> Int -> Prop) -> ImprovingIO PropertyTestCount f (PropertyStatus, PropertyTestCount)
myTest st f
| ntest >= maxSuccessTests st = return (if expectedFailure st then PropertyOK else PropertyNoExpectedFailure, ntest)
| numDiscardedTests st >= maxDiscardedTests st = return (PropertyArgumentsExhausted, ntest)
| otherwise = yieldImprovement ntest >> myRunATest st f
where ntest = numSuccessTests st
myRunATest :: State -> (StdGen -> Int -> Prop) -> ImprovingIO PropertyTestCount f (PropertyStatus, PropertyTestCount)
myRunATest st f = do
let size = computeSize st (numSuccessTests st) (numDiscardedTests st)
ei_st_res <- liftIO $ flip E.catch (\e -> return $ Left $ show (e :: E.SomeException)) $ do
MkRose res ts <- protectRose (reduceRose (unProp (f rnd1 size)))
return (Right (res, ts))
case ei_st_res of
Left text -> return (PropertyException text, numSuccessTests st + 1)
Right (res, ts) -> do
liftIO $ callbackPostTest st res
case res of
MkResult{ok = Just True, stamp = stamp, expect = expect} ->
do myTest st{ numSuccessTests = numSuccessTests st + 1
, randomSeed = rnd2
, collected = stamp : collected st
, expectedFailure = expect
} f
MkResult{ok = Nothing, expect = expect} ->
do myTest st{ numDiscardedTests = numDiscardedTests st + 1
, randomSeed = rnd2
, expectedFailure = expect
} f
MkResult{ok = Just False} ->
do if expect res
then liftIO $ myFoundFailure st res ts
else return (PropertyOK, numSuccessTests st + 1)
where
(rnd1,rnd2) = split (randomSeed st)
myFoundFailure :: State -> P.Result -> [Rose P.Result] -> IO (PropertyStatus, PropertyTestCount)
myFoundFailure st res ts =
do myLocalMin st{ numTryShrinks = 0 } res ts
myLocalMin :: State -> P.Result -> [Rose P.Result] -> IO (PropertyStatus, PropertyTestCount)
myLocalMin st res _ | P.interrupted res = myLocalMinFound st res
myLocalMin st res ts = do
r <- tryEvaluate ts
case r of
Left err ->
myLocalMinFound st
(exception "Exception while generating shrink-list" err) { callbacks = callbacks res }
Right ts' -> myLocalMin' st res ts'
myLocalMin' :: State -> P.Result -> [Rose P.Result] -> IO (PropertyStatus, PropertyTestCount)
myLocalMin' st res [] = myLocalMinFound st res
myLocalMin' st res (t:ts) =
do
MkRose res' ts' <- protectRose (reduceRose t)
callbackPostTest st res'
if ok res' == Just False
then myFoundFailure st{ numSuccessShrinks = numSuccessShrinks st + 1 } res' ts'
else myLocalMin st{ numTryShrinks = numTryShrinks st + 1 } res ts
myLocalMinFound :: State -> P.Result -> IO (PropertyStatus, PropertyTestCount)
myLocalMinFound st res =
do callbackPostFinalFailure st res
return (PropertyFalsifiable (P.reason res), numSuccessTests st + 1)
tryEvaluate :: a -> IO (Either E.SomeException a)
tryEvaluate x = tryEvaluateIO (return x)
tryEvaluateIO :: IO a -> IO (Either E.SomeException a)
tryEvaluateIO m = E.try (m >>= E.evaluate)