{-# OPTIONS_HADDOCK hide #-}
{-# LANGUAGE CPP #-}
#ifndef NO_SAFE_HASKELL
{-# LANGUAGE Trustworthy #-}
#endif
module Test.QuickCheck.Test where
import Test.QuickCheck.Gen
import Test.QuickCheck.Property hiding ( Result( reason, theException, labels, classes, tables ), (.&.) )
import qualified Test.QuickCheck.Property as P
import Test.QuickCheck.Text
import Test.QuickCheck.State hiding (labels, classes, tables, requiredCoverage)
import qualified Test.QuickCheck.State as S
import Test.QuickCheck.Exception
import Test.QuickCheck.Random
import Data.Number.Erf(invnormcdf)
import System.Random(split)
#if defined(MIN_VERSION_containers)
#if MIN_VERSION_containers(0,5,0)
import qualified Data.Map.Strict as Map
#else
import qualified Data.Map as Map
#endif
#else
import qualified Data.Map as Map
#endif
import qualified Data.Set as Set
import Data.Set(Set)
import Data.Map(Map)
import Data.Char
( isSpace
)
import Data.List
( sort
, sortBy
, group
, intersperse
, intercalate
)
import Data.Maybe(fromMaybe, isNothing, catMaybes)
import Data.Ord(comparing)
import Text.Printf(printf)
import Data.Either(lefts, rights)
import Control.Monad
import Data.Bits
data Args
= Args
{ replay :: Maybe (QCGen,Int)
, maxSuccess :: Int
, maxDiscardRatio :: Int
, maxSize :: Int
, chatty :: Bool
, maxShrinks :: Int
}
deriving ( Show, Read )
data Result
= Success
{ numTests :: Int
, numDiscarded :: Int
, labels :: !(Map [String] Int)
, classes :: !(Map String Int)
, tables :: !(Map String (Map String Int))
, output :: String
}
| GaveUp
{ numTests :: Int
, numDiscarded :: Int
, labels :: !(Map [String] Int)
, classes :: !(Map String Int)
, tables :: !(Map String (Map String Int))
, output :: String
}
| Failure
{ numTests :: Int
, numDiscarded :: Int
, numShrinks :: Int
, numShrinkTries :: Int
, numShrinkFinal :: Int
, usedSeed :: QCGen
, usedSize :: Int
, reason :: String
, theException :: Maybe AnException
, output :: String
, failingTestCase :: [String]
, failingLabels :: [String]
, failingClasses :: Set String
}
| NoExpectedFailure
{ numTests :: Int
, numDiscarded :: Int
, labels :: !(Map [String] Int)
, classes :: !(Map String Int)
, tables :: !(Map String (Map String Int))
, output :: String
}
deriving ( Show )
isSuccess :: Result -> Bool
isSuccess Success{} = True
isSuccess _ = False
stdArgs :: Args
stdArgs = Args
{ replay = Nothing
, maxSuccess = 100
, maxDiscardRatio = 10
, maxSize = 100
, chatty = True
, maxShrinks = maxBound
}
quickCheck :: Testable prop => prop -> IO ()
quickCheck p = quickCheckWith stdArgs p
quickCheckWith :: Testable prop => Args -> prop -> IO ()
quickCheckWith args p = quickCheckWithResult args p >> return ()
quickCheckResult :: Testable prop => prop -> IO Result
quickCheckResult p = quickCheckWithResult stdArgs p
quickCheckWithResult :: Testable prop => Args -> prop -> IO Result
quickCheckWithResult a p =
withState a (\s -> test s (property p))
withState :: Args -> (State -> IO a) -> IO a
withState a test = (if chatty a then withStdioTerminal else withNullTerminal) $ \tm -> do
rnd <- case replay a of
Nothing -> newQCGen
Just (rnd,_) -> return rnd
test MkState{ terminal = tm
, maxSuccessTests = maxSuccess a
, coverageConfidence = Nothing
, maxDiscardedRatio = maxDiscardRatio a
, computeSize = case replay a of
Nothing -> computeSize'
Just (_,s) -> computeSize' `at0` s
, numTotMaxShrinks = maxShrinks a
, numSuccessTests = 0
, numDiscardedTests = 0
, numRecentlyDiscardedTests = 0
, S.labels = Map.empty
, S.classes = Map.empty
, S.tables = Map.empty
, S.requiredCoverage = Map.empty
, expected = True
, randomSeed = rnd
, numSuccessShrinks = 0
, numTryShrinks = 0
, numTotTryShrinks = 0
}
where computeSize' n d
| n `roundTo` maxSize a + maxSize a <= maxSuccess a ||
n >= maxSuccess a ||
maxSuccess a `mod` maxSize a == 0 = (n `mod` maxSize a + d `div` 10) `min` maxSize a
| otherwise =
((n `mod` maxSize a) * maxSize a `div` (maxSuccess a `mod` maxSize a) + d `div` 10) `min` maxSize a
n `roundTo` m = (n `div` m) * m
at0 f s 0 0 = s
at0 f s n d = f n d
verboseCheck :: Testable prop => prop -> IO ()
verboseCheck p = quickCheck (verbose p)
verboseCheckWith :: Testable prop => Args -> prop -> IO ()
verboseCheckWith args p = quickCheckWith args (verbose p)
verboseCheckResult :: Testable prop => prop -> IO Result
verboseCheckResult p = quickCheckResult (verbose p)
verboseCheckWithResult :: Testable prop => Args -> prop -> IO Result
verboseCheckWithResult a p = quickCheckWithResult a (verbose p)
test :: State -> Property -> IO Result
test st f
| numSuccessTests st >= maxSuccessTests st && isNothing (coverageConfidence st) =
doneTesting st f
| numDiscardedTests st >= maxDiscardedRatio st * max (numSuccessTests st) (maxSuccessTests st) =
giveUp st f
| otherwise =
runATest st f
doneTesting :: State -> Property -> IO Result
doneTesting st _f
| expected st == False = do
putPart (terminal st)
( bold ("*** Failed!")
++ " Passed "
++ showTestCount st
++ " (expected failure)"
)
finished NoExpectedFailure
| otherwise = do
putPart (terminal st)
( "+++ OK, passed "
++ showTestCount st
)
finished Success
where
finished k = do
success st
theOutput <- terminalOutput (terminal st)
return (k (numSuccessTests st) (numDiscardedTests st) (S.labels st) (S.classes st) (S.tables st) theOutput)
giveUp :: State -> Property -> IO Result
giveUp st _f =
do
putPart (terminal st)
( bold ("*** Gave up!")
++ " Passed only "
++ showTestCount st
++ " tests"
)
success st
theOutput <- terminalOutput (terminal st)
return GaveUp{ numTests = numSuccessTests st
, numDiscarded = numDiscardedTests st
, labels = S.labels st
, classes = S.classes st
, tables = S.tables st
, output = theOutput
}
showTestCount :: State -> String
showTestCount st =
number (numSuccessTests st) "test"
++ concat [ "; " ++ show (numDiscardedTests st) ++ " discarded"
| numDiscardedTests st > 0
]
runATest :: State -> Property -> IO Result
runATest st f =
do
putTemp (terminal st)
( "("
++ showTestCount st
++ ")"
)
let powerOfTwo n = n .&. (n - 1) == 0
let f_or_cov =
case coverageConfidence st of
Just confidence | (1 + numSuccessTests st) `mod` 100 == 0 && powerOfTwo ((1 + numSuccessTests st) `div` 100) ->
addCoverageCheck confidence st f
_ -> f
let size = computeSize st (numSuccessTests st) (numRecentlyDiscardedTests st)
MkRose res ts <- protectRose (reduceRose (unProp (unGen (unProperty f_or_cov) rnd1 size)))
res <- callbackPostTest st res
let continue break st' | abort res = break st'
| otherwise = test st'
let inc x = Map.insertWith (+) x 1
let st' = st{ coverageConfidence = maybeCheckCoverage res `mplus` coverageConfidence st
, maxSuccessTests = fromMaybe (maxSuccessTests st) (maybeNumTests res)
, S.labels = inc (P.labels res) (S.labels st)
, S.classes = foldr inc (S.classes st) (P.classes res)
, S.tables =
foldr (\(tab, x) -> Map.insertWith (Map.unionWith (+)) tab (Map.singleton x 1))
(S.tables st) (P.tables res)
, S.requiredCoverage =
foldr (\(key, value, p) -> Map.insertWith max (key, value) p)
(S.requiredCoverage st) (P.requiredCoverage res)
, expected = expect res }
case res of
MkResult{ok = Just True} ->
do continue doneTesting
st'{ numSuccessTests = numSuccessTests st' + 1
, numRecentlyDiscardedTests = 0
, randomSeed = rnd2
} f
MkResult{ok = Nothing, expect = expect, maybeNumTests = mnt, maybeCheckCoverage = mcc} ->
do continue giveUp
st{ numDiscardedTests = numDiscardedTests st' + 1
, numRecentlyDiscardedTests = numRecentlyDiscardedTests st' + 1
, randomSeed = rnd2
} f
MkResult{ok = Just False} ->
do (numShrinks, totFailed, lastFailed, res) <- foundFailure st' res ts
theOutput <- terminalOutput (terminal st')
if not (expect res) then
return Success{ labels = S.labels st',
classes = S.classes st',
tables = S.tables st',
numTests = numSuccessTests st'+1,
numDiscarded = numDiscardedTests st',
output = theOutput }
else do
testCase <- mapM showCounterexample (P.testCase res)
return Failure{ usedSeed = randomSeed st'
, usedSize = size
, numTests = numSuccessTests st'+1
, numDiscarded = numDiscardedTests st'
, numShrinks = numShrinks
, numShrinkTries = totFailed
, numShrinkFinal = lastFailed
, output = theOutput
, reason = P.reason res
, theException = P.theException res
, failingTestCase = testCase
, failingLabels = P.labels res
, failingClasses = Set.fromList (P.classes res)
}
where
(rnd1,rnd2) = split (randomSeed st)
failureSummary :: State -> P.Result -> String
failureSummary st res = fst (failureSummaryAndReason st res)
failureReason :: State -> P.Result -> [String]
failureReason st res = snd (failureSummaryAndReason st res)
failureSummaryAndReason :: State -> P.Result -> (String, [String])
failureSummaryAndReason st res = (summary, full)
where
summary =
header ++
short 26 (oneLine reason ++ " ") ++
count True ++ "..."
full =
(header ++
(if isOneLine reason then reason ++ " " else "") ++
count False ++ ":"):
if isOneLine reason then [] else lines reason
reason = P.reason res
header =
if expect res then
bold "*** Failed! "
else "+++ OK, failed as expected. "
count full =
"(after " ++ number (numSuccessTests st+1) "test" ++
concat [
" and " ++
show (numSuccessShrinks st) ++
concat [ "." ++ show (numTryShrinks st) | showNumTryShrinks ] ++
" shrink" ++
(if numSuccessShrinks st == 1 && not showNumTryShrinks then "" else "s")
| numSuccessShrinks st > 0 || showNumTryShrinks ] ++
")"
where
showNumTryShrinks = full && numTryShrinks st > 0
success :: State -> IO ()
success st = do
mapM_ (putLine $ terminal st) (paragraphs [short, long])
where
(short, long) =
case labelsAndTables st of
([msg], long) ->
([" (" ++ dropWhile isSpace msg ++ ")."], long)
([], long) ->
(["."], long)
(short, long) ->
(":":short, long)
labelsAndTables :: State -> ([String], [String])
labelsAndTables st = (labels, tables)
where
labels :: [String]
labels =
paragraphs $
[ showTable (numSuccessTests st) Nothing m
| m <- S.classes st:Map.elems numberedLabels ]
numberedLabels :: Map Int (Map String Int)
numberedLabels =
Map.fromListWith (Map.unionWith (+)) $
[ (i, Map.singleton l n)
| (labels, n) <- Map.toList (S.labels st),
(i, l) <- zip [0..] labels ]
tables :: [String]
tables =
paragraphs $
[ showTable (sum (Map.elems m)) (Just table) m
| (table, m) <- Map.toList (S.tables st) ] ++
[[ (case mtable of Nothing -> "Only "; Just table -> "Table '" ++ table ++ "' had only ")
++ lpercent n tot ++ " " ++ label ++ ", but expected " ++ lpercentage p tot
| (mtable, label, tot, n, p) <- allCoverage st,
insufficientlyCovered (fmap certainty (coverageConfidence st)) tot n p ]]
showTable :: Int -> Maybe String -> Map String Int -> [String]
showTable k mtable m =
[table ++ " " ++ total ++ ":" | Just table <- [mtable]] ++
(map format .
reverse . sortBy (comparing snd) .
reverse . sortBy (comparing fst) $ Map.toList m)
where
format (key, v) =
rpercent v k ++ " " ++ key
total = printf "(%d in total)" k
foundFailure :: State -> P.Result -> [Rose P.Result] -> IO (Int, Int, Int, P.Result)
foundFailure st res ts =
do localMin st{ numTryShrinks = 0 } res ts
localMin :: State -> P.Result -> [Rose P.Result] -> IO (Int, Int, Int, P.Result)
localMin st res ts
| numSuccessShrinks st + numTotTryShrinks st >= numTotMaxShrinks st =
localMinFound st res
localMin st res ts = do
r <- tryEvaluateIO $
putTemp (terminal st) (failureSummary st res)
case r of
Left err ->
localMinFound st (exception "Exception while printing status message" err) { callbacks = callbacks res }
Right () -> do
r <- tryEvaluate ts
case r of
Left err ->
localMinFound st
(exception "Exception while generating shrink-list" err) { callbacks = callbacks res }
Right ts' -> localMin' st res ts'
localMin' :: State -> P.Result -> [Rose P.Result] -> IO (Int, Int, Int, P.Result)
localMin' st res [] = localMinFound st res
localMin' st res (t:ts) =
do
MkRose res' ts' <- protectRose (reduceRose t)
res' <- callbackPostTest st res'
if ok res' == Just False
then localMin st{ numSuccessShrinks = numSuccessShrinks st + 1,
numTryShrinks = 0 } res' ts'
else localMin st{ numTryShrinks = numTryShrinks st + 1,
numTotTryShrinks = numTotTryShrinks st + 1 } res ts
localMinFound :: State -> P.Result -> IO (Int, Int, Int, P.Result)
localMinFound st res =
do sequence_ [ putLine (terminal st) msg | msg <- failureReason st res ]
callbackPostFinalFailure st res
return (numSuccessShrinks st, numTotTryShrinks st - numTryShrinks st, numTryShrinks st, res)
callbackPostTest :: State -> P.Result -> IO P.Result
callbackPostTest st res = protect (exception "Exception running callback") $ do
sequence_ [ f st res | PostTest _ f <- callbacks res ]
return res
callbackPostFinalFailure :: State -> P.Result -> IO ()
callbackPostFinalFailure st res = do
x <- tryEvaluateIO $ sequence_ [ f st res | PostFinalFailure _ f <- callbacks res ]
case x of
Left err -> do
putLine (terminal st) "*** Exception running callback: "
tryEvaluateIO $ putLine (terminal st) (show err)
return ()
Right () -> return ()
sufficientlyCovered :: Confidence -> Int -> Int -> Double -> Bool
sufficientlyCovered confidence n k p =
wilsonLow (fromIntegral k) (fromIntegral n) (1 / fromIntegral err) >= tol * p
where
err = certainty confidence
tol = tolerance confidence
insufficientlyCovered :: Maybe Integer -> Int -> Int -> Double -> Bool
insufficientlyCovered Nothing n k p =
fromIntegral k < p * fromIntegral n
insufficientlyCovered (Just err) n k p =
wilsonHigh (fromIntegral k) (fromIntegral n) (1 / fromIntegral err) < p
wilson :: Integer -> Integer -> Double -> Double
wilson k n z =
(p + z*z/(2*nf) + z*sqrt (p*(1-p)/nf + z*z/(4*nf*nf)))/(1 + z*z/nf)
where
nf = fromIntegral n
p = fromIntegral k / fromIntegral n
wilsonLow :: Integer -> Integer -> Double -> Double
wilsonLow k n a = wilson k n (invnormcdf (a/2))
wilsonHigh :: Integer -> Integer -> Double -> Double
wilsonHigh k n a = wilson k n (invnormcdf (1-a/2))
addCoverageCheck :: Confidence -> State -> Property -> Property
addCoverageCheck confidence st prop
| and [ sufficientlyCovered confidence tot n p
| (_, _, tot, n, p) <- allCoverage st ] =
once prop
| or [ insufficientlyCovered (Just (certainty confidence)) tot n p
| (_, _, tot, n, p) <- allCoverage st ] =
let (labels, tables) = labelsAndTables st in
foldr counterexample (property failed{P.reason = "Insufficient coverage"})
(paragraphs [labels, tables])
| otherwise = prop
allCoverage :: State -> [(Maybe String, String, Int, Int, Double)]
allCoverage st =
[ (key, value, tot, n, p)
| ((key, value), p) <- Map.toList (S.requiredCoverage st),
let tot =
case key of
Just key -> Map.findWithDefault 0 key totals
Nothing -> numSuccessTests st,
let n = Map.findWithDefault 0 value (Map.findWithDefault Map.empty key combinedCounts) ]
where
combinedCounts :: Map (Maybe String) (Map String Int)
combinedCounts =
Map.insert Nothing (S.classes st)
(Map.mapKeys Just (S.tables st))
totals :: Map String Int
totals = fmap (sum . Map.elems) (S.tables st)