module Test.Hspec.Runner (
  Specs, hspec, hspecX, hspecB, hHspec, hHspecWithFormat, describe, it, toExitCode
) where
import Test.Hspec.Core
import Test.Hspec.Formatters
import System.IO
import System.CPUTime (getCPUTime)
import Control.Monad (when)
import System.Exit
type Specs = [IO Spec]
runFormatter :: Formatter -> Handle -> String -> [String] -> Specs -> IO [Spec]
runFormatter formatter h _     errors []     = do
  errorsFormatter formatter h (reverse errors)
  return []
runFormatter formatter h group errors (iospec:ioss) = do
  spec <- iospec
  when (group /= name spec) (exampleGroupStarted formatter h spec)
  case result spec of
    (Success  ) -> examplePassed formatter h spec errors
    (Fail _   ) -> exampleFailed formatter h spec errors
    (Pending _) -> examplePending formatter h spec errors
  let errors' = if isFailure (result spec)
                then errorDetails spec (length errors) : errors
                else errors
  specs <- runFormatter formatter h (name spec) errors' ioss
  return $ spec : specs
errorDetails :: Spec -> Int -> String
errorDetails spec i = case result spec of
  (Fail s   ) -> concat [ show (i + 1), ") ", name spec, " ",  requirement spec, " FAILED", if null s then "" else "\n" ++ s ]
  _           -> ""
hspecX :: IO Specs -> IO a
hspecX ss = hspecB ss >>= exitWith . toExitCode
hspecB :: IO Specs -> IO Bool
hspecB ss = hspec ss >>= return . success
hspec :: IO Specs -> IO [Spec]
hspec ss = hHspec stdout ss
hHspec :: Handle -> IO Specs -> IO [Spec]
hHspec h = hHspecWithFormat (specdoc $ h == stdout) h
hHspecWithFormat :: Formatter -> Handle -> IO Specs -> IO [Spec]
hHspecWithFormat formatter h ss = do
  t0 <- getCPUTime
  ioSpecList <- ss
  specList <- (runFormatter formatter) h "" [] ioSpecList
  t1 <- getCPUTime
  let runTime = ((fromIntegral $ t1  t0) / (10.0^(12::Integer)) :: Double)
  (footerFormatter formatter) h specList runTime
  return specList
toExitCode :: Bool -> ExitCode
toExitCode True  = ExitSuccess
toExitCode False = ExitFailure 1