{-# LANGUAGE OverloadedStrings, TypeOperators #-} module Main (main) where import Data.String import qualified Text.Trifecta as Trifecta import Hedgehog hiding (Var) import Test.Tasty import Test.Tasty.Hedgehog import Test.Tasty.HUnit import Control.Effect.Sum import Data.File import qualified Generators as Gen import qualified Analysis.Eval as Eval import Data.Core import Data.Core.Pretty import Data.Core.Parser as Parse import Data.Name import Data.Term -- * Helpers true, false :: Term (Ann :+: Core) Name true = bool True false = bool False parseEither :: Trifecta.Parser a -> String -> Either String a parseEither p = Trifecta.foldResult (Left . show . Trifecta._errDoc) Right . Trifecta.parseString (p <* Trifecta.eof) mempty -- * Parser roundtripping properties. Note that parsing and prettyprinting is generally -- not a roundtrip, because the parser inserts 'Ann' nodes itself. prop_roundtrips :: Gen (Term (Ann :+: Core) Name) -> Property prop_roundtrips gen = property $ do input <- forAll gen tripping input (showCore . stripAnnotations) (parseEither (Parse.core <* Trifecta.eof)) parserProps :: TestTree parserProps = testGroup "Parsing: roundtripping" [ testProperty "literals" $ prop_roundtrips Gen.literal , testProperty "if/then/else" . prop_roundtrips . Gen.ifthenelse $ Gen.variable , testProperty "lambda" . prop_roundtrips $ Gen.lambda Gen.literal , testProperty "function application" . prop_roundtrips $ Gen.apply Gen.variable , testProperty "expressions" . prop_roundtrips $ Gen.expr ] -- * Parser specs parsesInto :: String -> Term (Ann :+: Core) Name -> Assertion parsesInto str res = case parseEither Parse.core str of Right x -> x @?= res Left m -> assertFailure m assert_booleans_parse :: Assertion assert_booleans_parse = do parseEither Parse.core "#true" @?= Right true parseEither Parse.core "#false" @?= Right false a, f, g, h :: Term (Ann :+: Core) Name (a, f, g, h) = (pure "a", pure "f", pure "g", pure "h") assert_ifthen_parse :: Assertion assert_ifthen_parse = "if #true then #true else #false" `parsesInto` (if' true true false) assert_application_parse :: Assertion assert_application_parse = "f g" `parsesInto` (f $$ g) assert_application_left_associative :: Assertion assert_application_left_associative = "f g h" `parsesInto` (f $$ g $$ h) assert_push_left_associative :: Assertion assert_push_left_associative = "f.g.h" `parsesInto` (f ... "g" ... "h") assert_ascii_lambda_parse :: Assertion assert_ascii_lambda_parse = "\\a -> a" `parsesInto` lam (named' "a") a assert_unicode_lambda_parse :: Assertion assert_unicode_lambda_parse = "λa → a" `parsesInto` lam (named' "a") a assert_quoted_name_parse :: Assertion assert_quoted_name_parse = "#{(NilClass)}" `parsesInto` pure "(NilClass)" parserSpecs :: TestTree parserSpecs = testGroup "Parsing: simple specs" [ testCase "true/false" assert_booleans_parse , testCase "if/then/else" assert_ifthen_parse , testCase "function application" assert_application_parse , testCase "application is left-associative" assert_application_left_associative , testCase "dotted push is left-associative" assert_push_left_associative , testCase "lambda with ASCII syntax" assert_ascii_lambda_parse , testCase "lambda with unicode syntax" assert_unicode_lambda_parse , testCase "quoted names" assert_quoted_name_parse ] assert_roundtrips :: File (Term (Ann :+: Core) Name) -> Assertion assert_roundtrips (File _ core) = case parseEither Parse.core (showCore (stripAnnotations core)) of Right v -> stripAnnotations v @?= stripAnnotations core Left e -> assertFailure e parserExamples :: TestTree parserExamples = testGroup "Parsing: Eval.hs examples" [ testCase "prog1" (assert_roundtrips Eval.prog1) , testCase "prog2" (assert_roundtrips Eval.prog2) , testCase "prog3" (assert_roundtrips Eval.prog3) , testCase "prog4" (assert_roundtrips Eval.prog4) , testCase "prog6.1" (assert_roundtrips (head Eval.prog6)) , testCase "prog6.2" (assert_roundtrips (last Eval.prog6)) , testCase "ruby" (assert_roundtrips Eval.ruby) ] tests :: TestTree tests = testGroup "semantic-core" [ parserSpecs , parserExamples , parserProps ] main :: IO () main = defaultMain tests