{-# LANGUAGE CPP #-} {-# LANGUAGE TypeApplications #-} {-# LANGUAGE NamedFieldPuns #-} module Test.DocTest.Internal.Runner where import Prelude hiding (putStr, putStrLn, error) import Control.Concurrent (Chan, writeChan, readChan, newChan, forkIO) import Control.Exception (SomeException, catch) import Control.Monad hiding (forM_) import Data.Foldable (forM_) import Data.Function (on) import Data.List (sortBy) import Data.Maybe (fromMaybe, maybeToList) import GHC.Conc (numCapabilities) import System.IO (hPutStrLn, hPutStr, stderr, hIsTerminalDevice) import System.Random (randoms, mkStdGen) import Text.Printf (printf) import Control.Monad.Trans.State import Control.Monad.IO.Class import Test.DocTest.Internal.Interpreter (Interpreter) import qualified Test.DocTest.Internal.Interpreter as Interpreter import Test.DocTest.Internal.Parse import Test.DocTest.Internal.Options ( ModuleName, ModuleConfig (cfgPreserveIt), cfgSeed, cfgPreserveIt , cfgRandomizeOrder, cfgImplicitModuleImport, parseLocatedModuleOptions) import Test.DocTest.Internal.Location import Test.DocTest.Internal.Property ( runProperty, PropertyResult(Failure, Success, Error) ) import Test.DocTest.Internal.Runner.Example import System.IO.CodePage (withCP65001) #if __GLASGOW_HASKELL__ < 804 import Data.Semigroup #endif -- | Whether an "example" is part of setup block data FromSetup = FromSetup | NotFromSetup -- | Summary of a test run. data Summary = Summary { sExamples :: Int -- ^ Total number of lines of examples (excluding setup) , sTried :: Int -- ^ Executed /sTried/ lines so far , sErrors :: Int -- ^ Couldn't execute /sErrors/ examples , sFailures :: Int -- ^ Got unexpected output for /sFailures/ examples } deriving Eq emptySummary :: Summary emptySummary = Summary 0 0 0 0 -- | Format a summary. instance Show Summary where show (Summary examples tried errors failures) = printf "Examples: %d Tried: %d Errors: %d Unexpected output: %d" examples tried errors failures -- | Sum up summaries. instance Monoid Summary where mempty = Summary 0 0 0 0 #if __GLASGOW_HASKELL__ < 804 mappend = (<>) #endif instance Semigroup Summary where (<>) (Summary x1 x2 x3 x4) (Summary y1 y2 y3 y4) = Summary (x1 + y1) (x2 + y2) (x3 + y3) (x4 + y4) -- | Run all examples from a list of modules. runModules :: ModuleConfig -- ^ Configuration options specific to module -> Maybe Int -- ^ Number of threads to use. Defaults to 'numCapabilities'. -> Bool -- ^ Verbose -> Bool -- ^ Implicit Prelude -> [String] -- ^ Arguments passed to the GHCi process. -> Bool -- ^ Quiet mode activated -> [Module [Located DocTest]] -- ^ Modules under test -> IO Summary runModules modConfig nThreads verbose implicitPrelude args quiet modules = do isInteractive <- hIsTerminalDevice stderr -- Start a thread pool. It sends status updates to this thread through 'output'. (input, output) <- makeThreadPool (fromMaybe numCapabilities nThreads) (runModule modConfig implicitPrelude args) -- Send instructions to threads liftIO (mapM_ (writeChan input) modules) let nExamples = (sum . map count) modules initState = ReportState { reportStateCount = 0 , reportStateInteractive = isInteractive , reportStateVerbose = verbose , reportStateQuiet = quiet , reportStateSummary = mempty{sExamples=nExamples} } ReportState{reportStateSummary} <- (`execStateT` initState) $ do consumeUpdates output (length modules) unless quiet $ do verboseReport "# Final summary:" gets (show . reportStateSummary) >>= report return reportStateSummary where consumeUpdates _output 0 = pure () consumeUpdates output modsLeft = do update <- liftIO (readChan output) consumeUpdates output =<< case update of UpdateInternalError fs loc e -> reportInternalError fs loc e >> pure (modsLeft - 1) UpdateImportError modName result -> reportImportError modName result >> pure (modsLeft - 1) UpdateSuccess fs loc -> reportSuccess fs loc >> reportProgress >> pure modsLeft UpdateFailure fs loc expr errs -> reportFailure fs loc expr errs >> pure modsLeft UpdateError fs loc expr err -> reportError fs loc expr err >> pure modsLeft UpdateOptionError loc err -> reportOptionError loc err >> pure modsLeft UpdateVerbose msg -> verboseReport msg >> pure modsLeft UpdateStart loc expr msg -> reportStart loc expr msg >> pure modsLeft UpdateModuleDone -> pure (modsLeft - 1) -- | Count number of expressions in given module. count :: Module [Located DocTest] -> Int count (Module _ _ tests _) = sum (map length tests) -- | A monad for generating test reports. type Report = StateT ReportState IO data ReportState = ReportState { reportStateCount :: Int -- ^ characters on the current line , reportStateInteractive :: Bool -- ^ should intermediate results be printed? , reportStateVerbose :: Bool , reportStateQuiet :: Bool , reportStateSummary :: Summary -- ^ test summary } -- | Add output to the report. report :: String -> Report () report msg = do overwrite msg -- add a newline, this makes the output permanent liftIO $ hPutStrLn stderr "" modify (\st -> st {reportStateCount = 0}) -- | Add intermediate output to the report. -- -- This will be overwritten by subsequent calls to `report`/`report_`. -- Intermediate out may not contain any newlines. report_ :: String -> Report () report_ msg = do f <- gets reportStateInteractive when f $ do overwrite msg modify (\st -> st {reportStateCount = length msg}) -- | Add output to the report, overwrite any intermediate out. overwrite :: String -> Report () overwrite msg = do n <- gets reportStateCount let str | 0 < n = "\r" ++ msg ++ replicate (n - length msg) ' ' | otherwise = msg liftIO (hPutStr stderr str) -- | Shuffle a list given a seed for an RNG shuffle :: Int -> [a] -> [a] shuffle seed xs = map snd $ sortBy (compare `on` fst) $ zip (randoms @Int (mkStdGen seed)) xs -- | Run all examples from given module. runModule :: ModuleConfig -> Bool -> [String] -> Chan ReportUpdate -> Module [Located DocTest] -> IO () runModule modConfig0 implicitPrelude ghciArgs output mod_ = do case modConfig2 of Left (loc, flag) -> writeChan output (UpdateOptionError loc flag) Right modConfig3 -> do let examples1 | cfgRandomizeOrder modConfig3 = shuffle seed examples0 | otherwise = examples0 importModule | cfgImplicitModuleImport modConfig3 = Just (":m +" ++ module_) | otherwise = Nothing preserveIt = cfgPreserveIt modConfig3 seed = fromMaybe 0 (cfgSeed modConfig3) -- Should have been set already reload repl = do void $ Interpreter.safeEval repl ":reload" mapM_ (Interpreter.safeEval repl) $ if implicitPrelude then ":m Prelude" : maybeToList importModule else maybeToList importModule when preserveIt $ -- Evaluate a dumb expression to populate the 'it' variable NOTE: This is -- one reason why we cannot have safeEval = safeEvalIt: 'it' isn't set in -- a fresh GHCi session. void $ Interpreter.safeEval repl $ "()" setup_ repl = do reload repl forM_ setup $ \l -> forM_ l $ \(Located _ x) -> case x of Property _ -> return () Example e _ -> void $ safeEvalWith preserveIt repl e Interpreter.withInterpreter ghciArgs $ \repl -> withCP65001 $ do -- Try to import this module, if it fails, something is off importResult <- case importModule of Nothing -> pure (Right "") Just i -> Interpreter.safeEval repl i case importResult of Right "" -> do -- Run setup group successes <- mapM (runTestGroup FromSetup preserveIt repl (reload repl) output) setup -- only run tests, if setup does not produce any errors/failures when (and successes) (mapM_ (runTestGroup NotFromSetup preserveIt repl (setup_ repl) output) examples1) _ -> writeChan output (UpdateImportError module_ importResult) -- Signal main thread a module has been tested writeChan output UpdateModuleDone pure () where Module module_ setup examples0 modArgs = mod_ modConfig2 = parseLocatedModuleOptions module_ modConfig0 modArgs data ReportUpdate = UpdateSuccess FromSetup Location -- ^ Test succeeded | UpdateFailure FromSetup Location Expression [String] -- ^ Test failed with unexpected result | UpdateError FromSetup Location Expression String -- ^ Test failed with an error | UpdateVerbose String -- ^ Message to send when verbose output is activated | UpdateModuleDone -- ^ All examples tested in module | UpdateStart Location Expression String -- ^ Indicate test has started executing (verbose output) | UpdateInternalError FromSetup (Module [Located DocTest]) SomeException -- ^ Exception caught while executing internal code | UpdateImportError ModuleName (Either String String) -- ^ Could not import module | UpdateOptionError Location String -- ^ Unrecognized flag in module specific option makeThreadPool :: Int -> (Chan ReportUpdate -> Module [Located DocTest] -> IO ()) -> IO (Chan (Module [Located DocTest]), Chan ReportUpdate) makeThreadPool nThreads mutator = do input <- newChan output <- newChan forM_ [1..nThreads] $ \_ -> forkIO $ forever $ do i <- readChan input catch (mutator output i) (\e -> writeChan output (UpdateInternalError NotFromSetup i e)) return (input, output) reportStart :: Location -> Expression -> String -> Report () reportStart loc expression testType = do quiet <- gets reportStateQuiet unless quiet $ verboseReport (printf "### Started execution at %s.\n### %s:\n%s" (show loc) testType expression) reportFailure :: FromSetup -> Location -> Expression -> [String] -> Report () reportFailure fromSetup loc expression err = do report (printf "%s: failure in expression `%s'" (show loc) expression) mapM_ report err report "" updateSummary fromSetup (Summary 0 1 0 1) reportError :: FromSetup -> Location -> Expression -> String -> Report () reportError fromSetup loc expression err = do report (printf "%s: error in expression `%s'" (show loc) expression) report err report "" updateSummary fromSetup (Summary 0 1 1 0) reportOptionError :: Location -> String -> Report () reportOptionError loc opt = do report (printf "%s: unrecognized option: %s. Try --help to see all options." (show loc) opt) report "" updateSummary FromSetup (Summary 0 1 1 0) reportInternalError :: FromSetup -> Module a -> SomeException -> Report () reportInternalError fs mod_ err = do report (printf "Internal error when executing tests in %s" (moduleName mod_)) report (show err) report "" updateSummary fs emptySummary{sErrors=1} reportImportError :: ModuleName -> Either String String -> Report () reportImportError modName importResult = do report ("Could not import module: " <> modName <> ". This can be caused by a number of issues: ") report "" report " 1. A module found by GHC contained tests, but was not in 'exposed-modules'. If you want" report " to test non-exposed modules follow the instructions here:" report " https://github.com/martijnbastiaan/doctest-parallel#test-non-exposed-modules" report "" report " 2. For Cabal users: Cabal did not generate a GHC environment file. Either:" report " * Run with '--write-ghc-environment-files=always'" report " * Add 'write-ghc-environment-files: always' to your cabal.project" report "" report " 3. For Cabal users: Cabal did not generate a GHC environment file in time. This" report " can happen if you use 'cabal test' instead of 'cabal run doctests'. See" report " https://github.com/martijnbastiaan/doctest-parallel/issues/22." report "" report " 4. The testsuite executable does not have a dependency on your project library. Please" report " add it to the 'build-depends' section of the testsuite executable." report "" report "See the example project at https://github.com/martijnbastiaan/doctest-parallel/blob/main/example/README.md for more information." report "" report "The original reason given by GHCi was:" report "" case importResult of Left out -> do report "Unexpected output:" report out Right err -> do report "Error:" report err updateSummary FromSetup emptySummary{sErrors=1} reportSuccess :: FromSetup -> Location -> Report () reportSuccess fromSetup loc = do quiet <- gets reportStateQuiet unless quiet $ verboseReport (printf "### Successful `%s'!\n" (show loc)) updateSummary fromSetup (Summary 0 1 0 0) verboseReport :: String -> Report () verboseReport xs = do verbose <- gets reportStateVerbose quiet <- gets reportStateQuiet unless quiet $ when verbose $ report xs updateSummary :: FromSetup -> Summary -> Report () updateSummary FromSetup summary = -- Suppress counts, except for errors and unexpected outputs updateSummary NotFromSetup summary{sExamples=0, sTried=0} updateSummary NotFromSetup summary = do ReportState n f v q s <- get put (ReportState n f v q $ s `mappend` summary) reportProgress :: Report () reportProgress = do verbose <- gets reportStateVerbose quiet <- gets reportStateQuiet unless quiet $ unless verbose $ gets (show . reportStateSummary) >>= report_ -- | Run given test group. -- -- The interpreter state is zeroed with @:reload@ first. This means that you -- can reuse the same 'Interpreter' for several test groups. runTestGroup :: FromSetup -> Bool -> Interpreter -> IO () -> Chan ReportUpdate -> [Located DocTest] -> IO Bool runTestGroup fromSetup preserveIt repl setup output tests = do setup successExamples <- runExampleGroup fromSetup preserveIt repl output examples successesProperties <- forM properties $ \(loc, expression) -> do r <- do setup writeChan output (UpdateStart loc expression "property") runProperty repl expression case r of Success -> do writeChan output (UpdateSuccess fromSetup loc) pure True Error err -> do writeChan output (UpdateError fromSetup loc expression err) pure False Failure msg -> do writeChan output (UpdateFailure fromSetup loc expression [msg]) pure False pure (successExamples && and successesProperties) where properties = [(loc, p) | Located loc (Property p) <- tests] examples :: [Located Interaction] examples = [Located loc (e, r) | Located loc (Example e r) <- tests] -- | -- Execute all expressions from given example in given 'Interpreter' and verify -- the output. runExampleGroup :: FromSetup -> Bool -> Interpreter -> Chan ReportUpdate -> [Located Interaction] -> IO Bool runExampleGroup fromSetup preserveIt repl output = go where go ((Located loc (expression, expected)) : xs) = do writeChan output (UpdateStart loc expression "example") r <- fmap lines <$> safeEvalWith preserveIt repl expression case r of Left err -> do writeChan output (UpdateError fromSetup loc expression err) pure False Right actual -> case mkResult expected actual of NotEqual err -> do writeChan output (UpdateFailure fromSetup loc expression err) pure False Equal -> do writeChan output (UpdateSuccess fromSetup loc) go xs go [] = pure True safeEvalWith :: Bool -> Interpreter -> String -> IO (Either String String) safeEvalWith preserveIt | preserveIt = Interpreter.safeEvalIt | otherwise = Interpreter.safeEval