{-# LANGUAGE OverloadedStrings #-} module Main where import Control.Exception (throw) import Control.Monad (forM) import Data.Map (Map, foldrWithKey, singleton, unions) import Data.Void (Void) import Gauge (bench, bgroup, defaultMain, env, nf, whnf) import System.Directory import qualified Data.ByteString.Lazy import qualified Data.Text as T import qualified Data.Text.IO as TIO import qualified Dhall.Binary import qualified Dhall.Core as Dhall import qualified Dhall.Parser as Dhall import qualified Gauge type PreludeFiles = Map FilePath T.Text loadPreludeFiles :: IO PreludeFiles loadPreludeFiles = loadDirectory "./dhall-lang/Prelude" where loadDirectory :: FilePath -> IO PreludeFiles loadDirectory dir = withCurrentDirectory dir $ do files <- getCurrentDirectory >>= listDirectory results <- forM files $ \file -> do file' <- makeAbsolute file doesExist <- doesFileExist file' if doesExist then loadFile file' else loadDirectory file' pure $ unions results loadFile :: FilePath -> IO PreludeFiles loadFile path = singleton path <$> TIO.readFile path benchParser :: PreludeFiles -> Gauge.Benchmark benchParser = bgroup "exprFromText" . foldrWithKey (\name expr -> (benchExprFromText name expr :)) [] benchExprFromText :: String -> T.Text -> Gauge.Benchmark benchExprFromText name expr = bench name $ whnf (Dhall.exprFromText "(input)") expr benchExprFromBytes :: String -> Data.ByteString.Lazy.ByteString -> Gauge.Benchmark benchExprFromBytes name bs = bench name (nf f bs) where f bytes = case Dhall.Binary.decodeExpression bytes of Left exception -> error (show exception) Right expression -> expression :: Dhall.Expr Void Dhall.Import benchNfExprFromText :: String -> T.Text -> Gauge.Benchmark benchNfExprFromText name expr = bench name $ nf (either throw id . Dhall.exprFromText "(input)") expr main :: IO () main = do prelude <- loadPreludeFiles defaultMain [ env issues $ \ ~(it, ib) -> bgroup "Issue #108" [ benchExprFromText "Text" it , benchExprFromBytes "Binary" ib ] , env kubernetesExample $ benchExprFromBytes "Kubernetes/Binary" , benchExprFromText "Long variable names" (T.replicate 1000000 "x") , benchExprFromText "Large number of function arguments" (T.replicate 10000 "x ") , benchExprFromText "Long double-quoted strings" ("\"" <> T.replicate 1000000 "x" <> "\"") , benchExprFromText "Long single-quoted strings" ("''" <> T.replicate 1000000 "x" <> "''") , benchExprFromText "Whitespace" (T.replicate 1000000 " " <> "x") , benchExprFromText "Line comment" ("x -- " <> T.replicate 1000000 " ") , benchExprFromText "Block comment" ("x {- " <> T.replicate 1000000 " " <> "-}") , benchExprFromText "Deeply nested parentheses" "((((((((((((((((x))))))))))))))))" , benchParser prelude , env cpkgExample $ benchNfExprFromText "CPkg/Text" ] where cpkgExample = TIO.readFile "benchmark/examples/cpkg.dhall" issue108Text = TIO.readFile "benchmark/examples/issue108.dhall" issue108Bytes = Data.ByteString.Lazy.readFile "benchmark/examples/issue108.dhall.bin" issues = (,) <$> issue108Text <*> issue108Bytes kubernetesExample = Data.ByteString.Lazy.readFile "benchmark/examples/kubernetes.dhall.bin"