module Test.Framework.Runners.Console.Run (
showRunTestsTop
) where
import Test.Framework.Core
import Test.Framework.Improving
import Test.Framework.Runners.Console.Colors
import Test.Framework.Runners.Console.ProgressBar
import Test.Framework.Runners.Console.Statistics
import Test.Framework.Runners.Console.Utilities
import Test.Framework.Runners.Core
import Test.Framework.Runners.Statistics
import Test.Framework.Runners.TimedConsumption
import Test.Framework.Utilities
import System.Console.ANSI
import System.IO
import Text.PrettyPrint.ANSI.Leijen
import Data.Monoid
import Control.Arrow (second)
showRunTestsTop :: [RunningTest] -> IO [FinishedTest]
showRunTestsTop running_tests = hideCursorDuring $ do
let test_statistics = initialTestStatistics (totalRunTestsList running_tests)
(test_statistics', finished_tests) <- showRunTests 0 test_statistics running_tests
putStrLn ""
putDoc $ showFinalTestStatistics test_statistics'
return finished_tests
showRunTest :: Int -> TestStatistics -> RunningTest -> IO (TestStatistics, FinishedTest)
showRunTest indent_level test_statistics (RunTest name test_type (SomeImproving improving_result)) = do
let progress_bar = testStatisticsProgressBar test_statistics
(property_text, property_suceeded) <- showImprovingTestResult (return ()) indent_level name progress_bar improving_result
return (updateTestStatistics (\count -> adjustTestCount test_type count mempty) property_suceeded test_statistics, RunTest name test_type (property_text, property_suceeded))
showRunTest indent_level test_statistics (RunTestGroup name tests) = do
putDoc $ (indent indent_level (text name <> char ':')) <> linebreak
fmap (second $ RunTestGroup name) $ showRunTests (indent_level + 2) test_statistics tests
showRunTests :: Int -> TestStatistics -> [RunningTest] -> IO (TestStatistics, [FinishedTest])
showRunTests indent_level = mapAccumLM (showRunTest indent_level)
testStatisticsProgressBar :: TestStatistics -> Doc
testStatisticsProgressBar test_statistics = progressBar (colorPassOrFail no_failures) terminal_width (Progress run_tests total_tests)
where
run_tests = testCountTotal (ts_run_tests test_statistics)
total_tests = testCountTotal (ts_total_tests test_statistics)
no_failures = ts_no_failures test_statistics
terminal_width = 79
consumeImprovingThing :: (a :~> b) -> [(a :~> b)]
consumeImprovingThing improving@(Finished _) = [improving]
consumeImprovingThing improving@(Improving _ rest) = improving : consumeImprovingThing rest
showImprovingTestResult :: TestResultlike i r => IO () -> Int -> String -> Doc -> (i :~> r) -> IO (String, Bool)
showImprovingTestResult erase indent_level test_name progress_bar improving = do
improving_list <- consumeListInInterval 200000 (consumeImprovingThing improving)
case listToMaybeLast improving_list of
Nothing -> do
showImprovingTestResult erase indent_level test_name progress_bar improving
Just improving' -> do
showImprovingTestResult' erase indent_level test_name progress_bar improving'
showImprovingTestResult' :: TestResultlike i r => IO () -> Int -> String -> Doc -> (i :~> r) -> IO (String, Bool)
showImprovingTestResult' erase indent_level test_name _ (Finished result) = do
erase
putTestHeader indent_level test_name result_doc
clearLine
putDoc extra_doc
return (show result, success)
where
success = testSucceeded result
(result_doc, extra_doc) | success = (brackets $ colorPass (text (show result)), empty)
| otherwise = (brackets (colorFail (text "Failed")), text (show result) <> linebreak)
showImprovingTestResult' erase indent_level test_name progress_bar (Improving intermediate rest) = do
erase
putTestHeader indent_level test_name (brackets (text intermediate_str))
putDoc progress_bar
hFlush stdout
showImprovingTestResult (cursorUpLine 1 >> clearLine) indent_level test_name progress_bar rest
where
intermediate_str = show intermediate
putTestHeader :: Int -> String -> Doc -> IO ()
putTestHeader indent_level test_name result = putDoc $ (indent indent_level (text test_name <> char ':' <+> result)) <> linebreak