2024-12-18 18:55:51 -06:00
|
|
|
module Main where
|
|
|
|
|
|
|
|
import Eval
|
|
|
|
import Lexer
|
|
|
|
import Parser
|
|
|
|
import Research
|
2024-12-19 19:53:32 -06:00
|
|
|
import Control.Exception (evaluate, try, SomeException)
|
2024-12-18 18:55:51 -06:00
|
|
|
import Test.Tasty
|
|
|
|
import Test.Tasty.HUnit
|
|
|
|
import Test.Tasty.QuickCheck
|
|
|
|
import Text.Megaparsec (runParser)
|
|
|
|
|
2024-12-27 12:27:00 -06:00
|
|
|
import qualified Data.Map as Map
|
|
|
|
import qualified Data.Set as Set
|
|
|
|
|
2024-12-18 18:55:51 -06:00
|
|
|
main :: IO ()
|
|
|
|
main = defaultMain tests
|
|
|
|
|
2024-12-27 12:27:00 -06:00
|
|
|
runSapling :: String -> String
|
|
|
|
runSapling s = show $ result (evalSapling Map.empty $ parseSapling s)
|
|
|
|
|
2024-12-18 18:55:51 -06:00
|
|
|
tests :: TestTree
|
|
|
|
tests = testGroup "Sapling Tests"
|
2024-12-27 12:27:00 -06:00
|
|
|
[ lexerTests
|
|
|
|
, parserTests
|
|
|
|
, integrationTests
|
|
|
|
, evaluationTests
|
|
|
|
, propertyTests
|
|
|
|
, lambdaEvalTests
|
|
|
|
]
|
2024-12-18 18:55:51 -06:00
|
|
|
|
|
|
|
lexerTests :: TestTree
|
|
|
|
lexerTests = testGroup "Lexer Tests"
|
2024-12-27 12:27:00 -06:00
|
|
|
[ testCase "Lex simple identifiers" $ do
|
|
|
|
let input = "x a b = a"
|
|
|
|
expect = Right [LIdentifier "x", LIdentifier "a", LIdentifier "b", LAssign, LIdentifier "a"]
|
|
|
|
runParser saplingLexer "" input @?= expect
|
|
|
|
, testCase "Lex Tree Calculus terms" $ do
|
|
|
|
let input = "t t t"
|
|
|
|
expect = Right [LKeywordT, LKeywordT, LKeywordT]
|
|
|
|
runParser saplingLexer "" input @?= expect
|
|
|
|
, testCase "Lex escaped characters in strings" $ do
|
|
|
|
let input = "\"hello\\nworld\""
|
|
|
|
expect = Right [LStringLiteral "hello\\nworld"]
|
|
|
|
runParser saplingLexer "" input @?= expect
|
|
|
|
, testCase "Lex mixed literals" $ do
|
|
|
|
let input = "t \"string\" 42"
|
|
|
|
expect = Right [LKeywordT, LStringLiteral "string", LIntegerLiteral 42]
|
|
|
|
runParser saplingLexer "" input @?= expect
|
|
|
|
, testCase "Lex invalid token" $ do
|
|
|
|
let input = "$invalid"
|
|
|
|
case runParser saplingLexer "" input of
|
|
|
|
Left _ -> return ()
|
|
|
|
Right _ -> assertFailure "Expected lexer to fail on invalid token"
|
|
|
|
, testCase "Drop trailing whitespace in definitions" $ do
|
|
|
|
let input = "x = 5 "
|
|
|
|
expect = [LIdentifier "x",LAssign,LIntegerLiteral 5]
|
|
|
|
case (runParser saplingLexer "" input) of
|
|
|
|
Left _ -> assertFailure "Failed to lex input"
|
|
|
|
Right i -> i @?= expect
|
|
|
|
, testCase "Error when using invalid characters in identifiers" $ do
|
2024-12-27 08:17:06 -06:00
|
|
|
case (runParser saplingLexer "" "__result = 5") of
|
2024-12-27 12:27:00 -06:00
|
|
|
Left _ -> return ()
|
|
|
|
Right _ -> assertFailure "Expected failure when trying to assign the value of __result"
|
|
|
|
]
|
2024-12-18 18:55:51 -06:00
|
|
|
|
|
|
|
parserTests :: TestTree
|
|
|
|
parserTests = testGroup "Parser Tests"
|
2024-12-27 12:27:00 -06:00
|
|
|
[ testCase "Error when parsing incomplete definitions" $ do
|
|
|
|
let input = lexSapling "x = "
|
|
|
|
case (runParser parseExpression "" input) of
|
|
|
|
Left _ -> return ()
|
|
|
|
Right _ -> assertFailure "Expected failure on invalid input"
|
|
|
|
, testCase "Error when assigning a value to T" $ do
|
|
|
|
let input = lexSapling "t = x"
|
|
|
|
case (runParser parseExpression "" input) of
|
|
|
|
Left _ -> return ()
|
|
|
|
Right _ -> assertFailure "Expected failure when trying to assign the value of T"
|
|
|
|
, testCase "Parse function definitions" $ do
|
|
|
|
let input = "x = (\\a b c : a)"
|
|
|
|
expect = SFunc "x" [] (SLambda ["a"] (SLambda ["b"] (SLambda ["c"] (SVar "a"))))
|
|
|
|
parseSingle input @?= expect
|
|
|
|
, testCase "Parse nested Tree Calculus terms" $ do
|
|
|
|
let input = "t (t t) t"
|
|
|
|
expect = SApp (SApp TLeaf (SApp TLeaf TLeaf)) TLeaf
|
|
|
|
parseSingle input @?= expect
|
|
|
|
, testCase "Parse sequential Tree Calculus terms" $ do
|
|
|
|
let input = "t t t"
|
|
|
|
expect = SApp (SApp TLeaf TLeaf) TLeaf
|
|
|
|
parseSingle input @?= expect
|
|
|
|
, testCase "Parse mixed list literals" $ do
|
|
|
|
let input = "[t (\"hello\") t]"
|
|
|
|
expect = SList [TLeaf, SStr "hello", TLeaf]
|
|
|
|
parseSingle input @?= expect
|
|
|
|
, testCase "Parse function with applications" $ do
|
|
|
|
let input = "f = (\\x : t x)"
|
|
|
|
expect = SFunc "f" [] (SLambda ["x"] (SApp TLeaf (SVar "x")))
|
|
|
|
parseSingle input @?= expect
|
|
|
|
, testCase "Parse nested lists" $ do
|
|
|
|
let input = "[t [(t t)]]"
|
|
|
|
expect = SList [TLeaf,SList [SApp TLeaf TLeaf]]
|
|
|
|
parseSingle input @?= expect
|
|
|
|
, testCase "Parse complex parentheses" $ do
|
|
|
|
let input = "t (t t (t t))"
|
|
|
|
expect = SApp TLeaf (SApp (SApp TLeaf TLeaf) (SApp TLeaf TLeaf))
|
|
|
|
parseSingle input @?= expect
|
|
|
|
, testCase "Parse empty list" $ do
|
|
|
|
let input = "[]"
|
|
|
|
expect = SList []
|
|
|
|
parseSingle input @?= expect
|
|
|
|
, testCase "Parse multiple nested lists" $ do
|
|
|
|
let input = "[[t t] [t (t t)]]"
|
|
|
|
expect = SList [SList [TLeaf,TLeaf],SList [TLeaf,SApp TLeaf TLeaf]]
|
|
|
|
parseSingle input @?= expect
|
|
|
|
, testCase "Parse whitespace variance" $ do
|
|
|
|
let input1 = "[t t]"
|
|
|
|
let input2 = "[ t t ]"
|
|
|
|
expect = SList [TLeaf, TLeaf]
|
|
|
|
parseSingle input1 @?= expect
|
|
|
|
parseSingle input2 @?= expect
|
|
|
|
, testCase "Parse string in list" $ do
|
|
|
|
let input = "[(\"hello\")]"
|
|
|
|
expect = SList [SStr "hello"]
|
|
|
|
parseSingle input @?= expect
|
|
|
|
, testCase "Parse parentheses inside list" $ do
|
|
|
|
let input = "[t (t t)]"
|
|
|
|
expect = SList [TLeaf,SApp TLeaf TLeaf]
|
|
|
|
parseSingle input @?= expect
|
|
|
|
, testCase "Parse nested parentheses in function body" $ do
|
|
|
|
let input = "f = (\\x : t (t (t t)))"
|
|
|
|
expect = SFunc "f" [] (SLambda ["x"] (SApp TLeaf (SApp TLeaf (SApp TLeaf TLeaf))))
|
|
|
|
parseSingle input @?= expect
|
|
|
|
, testCase "Parse lambda abstractions" $ do
|
|
|
|
let input = "(\\a : a)"
|
|
|
|
expect = (SLambda ["a"] (SVar "a"))
|
|
|
|
parseSingle input @?= expect
|
|
|
|
, testCase "Parse multiple arguments to lambda abstractions" $ do
|
|
|
|
let input = "x = (\\a b : a)"
|
|
|
|
expect = SFunc "x" [] (SLambda ["a"] (SLambda ["b"] (SVar "a")))
|
|
|
|
parseSingle input @?= expect
|
|
|
|
, testCase "Grouping T terms with parentheses in function application" $ do
|
|
|
|
let input = "x = (\\a : a)\nx (t)"
|
|
|
|
expect = [SFunc "x" [] (SLambda ["a"] (SVar "a")),SApp (SVar "x") TLeaf]
|
|
|
|
parseSapling input @?= expect
|
|
|
|
]
|
2024-12-18 18:55:51 -06:00
|
|
|
|
|
|
|
integrationTests :: TestTree
|
|
|
|
integrationTests = testGroup "Integration Tests"
|
2024-12-27 12:27:00 -06:00
|
|
|
[ testCase "Combine lexer and parser" $ do
|
|
|
|
let input = "x = t t t"
|
|
|
|
expect = SApp (SVar "x") (SApp (SApp TLeaf TLeaf) TLeaf)
|
|
|
|
parseSingle input @?= expect
|
|
|
|
, testCase "Complex Tree Calculus expression" $ do
|
|
|
|
let input = "t (t t t) t"
|
|
|
|
expect = SApp (SApp TLeaf (SApp (SApp TLeaf TLeaf) TLeaf)) TLeaf
|
|
|
|
parseSingle input @?= expect
|
|
|
|
]
|
2024-12-18 18:55:51 -06:00
|
|
|
|
|
|
|
evaluationTests :: TestTree
|
|
|
|
evaluationTests = testGroup "Evaluation Tests"
|
2024-12-27 12:27:00 -06:00
|
|
|
[ testCase "Evaluate single Leaf" $ do
|
|
|
|
let input = "t"
|
|
|
|
let ast = parseSingle input
|
|
|
|
(result $ evalSingle Map.empty ast) @?= Leaf
|
|
|
|
, testCase "Evaluate single Stem" $ do
|
|
|
|
let input = "t t"
|
|
|
|
let ast = parseSingle input
|
|
|
|
(result $ evalSingle Map.empty ast) @?= Stem Leaf
|
|
|
|
, testCase "Evaluate single Fork" $ do
|
|
|
|
let input = "t t t"
|
|
|
|
let ast = parseSingle input
|
|
|
|
(result $ evalSingle Map.empty ast) @?= Fork Leaf Leaf
|
|
|
|
, testCase "Evaluate nested Fork and Stem" $ do
|
|
|
|
let input = "t (t t) t"
|
|
|
|
let ast = parseSingle input
|
|
|
|
(result $ evalSingle Map.empty ast) @?= Fork (Stem Leaf) Leaf
|
|
|
|
, testCase "Evaluate `not` function" $ do
|
|
|
|
let input = "t (t (t t) (t t t)) t"
|
|
|
|
let ast = parseSingle input
|
|
|
|
(result $ evalSingle Map.empty ast) @?=
|
|
|
|
Fork (Fork (Stem Leaf) (Fork Leaf Leaf)) Leaf
|
|
|
|
, testCase "Environment updates with definitions" $ do
|
|
|
|
let input = "x = t\ny = x"
|
|
|
|
env = evalSapling Map.empty (parseSapling input)
|
|
|
|
Map.lookup "x" env @?= Just Leaf
|
|
|
|
Map.lookup "y" env @?= Just Leaf
|
|
|
|
, testCase "Variable substitution" $ do
|
|
|
|
let input = "x = t t\ny = t x\ny"
|
|
|
|
env = evalSapling Map.empty (parseSapling input)
|
|
|
|
(result env) @?= Stem (Stem Leaf)
|
|
|
|
, testCase "Multiline input evaluation" $ do
|
|
|
|
let input = "x = t\ny = t t\nx"
|
|
|
|
env = evalSapling Map.empty (parseSapling input)
|
|
|
|
(result env) @?= Leaf
|
|
|
|
, testCase "Evaluate string literal" $ do
|
|
|
|
let input = "\"hello\""
|
|
|
|
let ast = parseSingle input
|
|
|
|
(result $ evalSingle Map.empty ast) @?= toString "hello"
|
|
|
|
, testCase "Evaluate list literal" $ do
|
|
|
|
let input = "[t (t t)]"
|
|
|
|
let ast = parseSingle input
|
|
|
|
(result $ evalSingle Map.empty ast) @?= toList [Leaf, Stem Leaf]
|
|
|
|
, testCase "Evaluate empty list" $ do
|
|
|
|
let input = "[]"
|
|
|
|
let ast = parseSingle input
|
|
|
|
(result $ evalSingle Map.empty ast) @?= toList []
|
|
|
|
, testCase "Evaluate variable dependency chain" $ do
|
|
|
|
let input = "x = t (t t)\n \
|
|
|
|
\ y = x\n \
|
|
|
|
\ z = y\n \
|
|
|
|
\ variablewithamuchlongername = z\n \
|
|
|
|
\ variablewithamuchlongername"
|
|
|
|
env = evalSapling Map.empty (parseSapling input)
|
|
|
|
(result env) @?= (Stem (Stem Leaf))
|
|
|
|
, testCase "Evaluate variable shadowing" $ do
|
|
|
|
let input = "x = t t\nx = t\nx"
|
|
|
|
env = evalSapling Map.empty (parseSapling input)
|
|
|
|
(result env) @?= Leaf
|
2024-12-27 08:17:06 -06:00
|
|
|
, testCase "Apply identity to Boolean Not" $ do
|
2024-12-27 12:27:00 -06:00
|
|
|
let not = "(t (t (t t) (t t t)) t)"
|
|
|
|
let input = "x = (\\a : a)\nx " ++ not
|
|
|
|
env = evalSapling Map.empty (parseSapling input)
|
|
|
|
result env @?= Fork (Fork (Stem Leaf) (Fork Leaf Leaf)) Leaf
|
2024-12-27 08:17:06 -06:00
|
|
|
, testCase "Constant function matches" $ do
|
2024-12-27 12:27:00 -06:00
|
|
|
let input = "k = (\\a b : a)\nk (t t) t"
|
|
|
|
env = evalSapling Map.empty (parseSapling input)
|
|
|
|
result env @?= Stem Leaf
|
|
|
|
, testCase "Boolean AND_ TF" $ do
|
|
|
|
let input = "and (t t) (t)"
|
|
|
|
env = evalSapling boolEnv (parseSapling input)
|
|
|
|
result env @?= Leaf
|
|
|
|
, testCase "Boolean AND_ FT" $ do
|
|
|
|
let input = "and (t) (t t)"
|
|
|
|
env = evalSapling boolEnv (parseSapling input)
|
|
|
|
result env @?= Leaf
|
|
|
|
, testCase "Boolean AND_ FF" $ do
|
|
|
|
let input = "and (t) (t)"
|
|
|
|
env = evalSapling boolEnv (parseSapling input)
|
|
|
|
result env @?= Leaf
|
|
|
|
, testCase "Boolean AND_ TT" $ do
|
|
|
|
let input = "and (t t) (t t)"
|
|
|
|
env = evalSapling boolEnv (parseSapling input)
|
|
|
|
result env @?= Stem Leaf
|
|
|
|
, testCase "Verifying Equality" $ do
|
|
|
|
let input = "equal (t t t) (t t t)"
|
|
|
|
env = evalSapling boolEnv (parseSapling input)
|
|
|
|
result env @?= Stem Leaf
|
|
|
|
]
|
|
|
|
where
|
|
|
|
boolEnv = evalSapling Map.empty $ parseSapling
|
|
|
|
"false = t\n \
|
|
|
|
\ true = t t\n \
|
|
|
|
\ falseL = (\\z : false)\n \
|
|
|
|
\ id = (\\a : a)\n \
|
|
|
|
\ triage = (\\a b c : (t (t a b) c))\n \
|
|
|
|
\ match_bool = (\\ot of : triage of (\\z : ot) t)\n \
|
|
|
|
\ and = match_bool id falseL\n \
|
|
|
|
\ fix = (\\m wait f : wait m (\\x : f (wait m x))) (\\x : x x) (\\a b c : (t (t a) (t t c) b))\n \
|
|
|
|
\ equal = fix ((\\self : triage (triage true (\\z : false) (\\z x : false)) (\\ax : triage false (self ax) (\\z x : false)) (\\ax ay : triage false (\\z : false) (\\bx by : and (self ax bx) (self ay by)))))\
|
|
|
|
\ "
|
|
|
|
|
2024-12-18 18:55:51 -06:00
|
|
|
|
|
|
|
propertyTests :: TestTree
|
|
|
|
propertyTests = testGroup "Property Tests"
|
2024-12-27 12:27:00 -06:00
|
|
|
[ testProperty "Lexing and parsing round-trip" $ \input ->
|
|
|
|
case runParser saplingLexer "" input of
|
|
|
|
Left _ -> property True
|
|
|
|
Right tokens -> case runParser parseExpression "" tokens of
|
|
|
|
Left _ -> property True
|
|
|
|
Right ast -> parseSingle input === ast
|
|
|
|
]
|
|
|
|
|
|
|
|
lambdaEvalTests :: TestTree
|
|
|
|
lambdaEvalTests = testGroup "Lambda Evaluation Tests"
|
|
|
|
[ testCase "Lambda Identity Function" $ do
|
|
|
|
let input = "id = (\\x : x)\nid t"
|
|
|
|
runSapling input @?= "Leaf"
|
|
|
|
|
|
|
|
, testCase "Lambda Constant Function (K combinator)" $ do
|
|
|
|
let input = "k = (\\x y : x)\nk t (t t)"
|
|
|
|
runSapling input @?= "Leaf"
|
|
|
|
|
|
|
|
, testCase "Lambda Application with Variable" $ do
|
|
|
|
let input = "id = (\\x : x)\nval = t t\nid val"
|
|
|
|
runSapling input @?= "Stem Leaf"
|
|
|
|
|
|
|
|
, testCase "Lambda Application with Multiple Arguments" $ do
|
|
|
|
let input = "apply = (\\f x y : f x y)\nk = (\\a b : a)\napply k t (t t)"
|
|
|
|
runSapling input @?= "Leaf"
|
|
|
|
|
|
|
|
, testCase "Nested Lambda Application" $ do
|
|
|
|
let input = "apply = (\\f x y : f x y)\nid = (\\x : x)\napply (\\f x : f x) id t"
|
|
|
|
runSapling input @?= "Leaf"
|
|
|
|
|
|
|
|
, testCase "Lambda with a complex body" $ do
|
|
|
|
let input = "f = (\\x : t (t x))\nf t"
|
|
|
|
runSapling input @?= "Stem (Stem Leaf)"
|
|
|
|
|
|
|
|
, testCase "Lambda returning a function" $ do
|
|
|
|
let input = "f = (\\x : (\\y : x))\ng = f t\ng (t t)"
|
|
|
|
runSapling input @?= "Leaf"
|
|
|
|
|
|
|
|
, testCase "Lambda with Shadowing" $ do
|
|
|
|
let input = "f = (\\x : (\\x : x))\nf t (t t)"
|
|
|
|
runSapling input @?= "Stem Leaf"
|
|
|
|
|
|
|
|
, testCase "Lambda returning another lambda" $ do
|
|
|
|
let input = "k = (\\x : (\\y : x))\nk_app = k t\nk_app (t t)"
|
|
|
|
runSapling input @?= "Leaf"
|
|
|
|
|
|
|
|
, testCase "Lambda with free variables" $ do
|
|
|
|
let input = "y = t t\nf = (\\x : y)\nf t"
|
|
|
|
runSapling input @?= "Stem Leaf"
|
|
|
|
|
|
|
|
, testCase "SKI Composition" $ do
|
|
|
|
let input = "s = (\\x y z : x z (y z))\nk = (\\x y : x)\ni = (\\x : x)\ncomp = s k i\ncomp t (t t)"
|
|
|
|
runSapling input @?= "Leaf"
|
|
|
|
, testCase "Lambda with multiple parameters and application" $ do
|
|
|
|
let input = "f = (\\a b c : t a b c)\nf t (t t) (t t t)"
|
|
|
|
runSapling input @?= "Fork (Fork Leaf Leaf) Leaf"
|
|
|
|
|
|
|
|
, testCase "Lambda with nested application in the body" $ do
|
|
|
|
let input = "f = (\\x : t (t (t x)))\nf t"
|
|
|
|
runSapling input @?= "Stem (Stem (Stem Leaf))"
|
|
|
|
, testCase "Lambda returning a function and applying it" $ do
|
|
|
|
let input = "f = (\\x : (\\y : t x y))\ng = f t\ng (t t)"
|
|
|
|
runSapling input @?= "Fork Leaf (Stem Leaf)"
|
|
|
|
, testCase "Lambda applying a variable" $ do
|
|
|
|
let input = "id = (\\x : x)\na = t t\nid a"
|
|
|
|
runSapling input @?= "Stem Leaf"
|
|
|
|
, testCase "Multiple lambda abstractions in the same expression" $ do
|
|
|
|
let input = "f = (\\x : (\\y : x y))\ng = (\\z : z)\nf g t"
|
|
|
|
runSapling input @?= "Stem Leaf"
|
|
|
|
, testCase "Lambda with a string literal" $ do
|
|
|
|
let input = "f = (\\x : x)\nf \"hello\""
|
|
|
|
runSapling input @?= "Fork (Fork (Fork (Stem Leaf) (Fork Leaf Leaf)) (Fork (Stem Leaf) (Fork Leaf Leaf))) (Fork (Fork (Stem Leaf) (Fork Leaf Leaf)) (Fork (Stem Leaf) (Fork Leaf Leaf)))"
|
|
|
|
, testCase "Lambda with an integer literal" $ do
|
|
|
|
let input = "f = (\\x : x)\nf 42"
|
|
|
|
runSapling input @?= "Fork (Leaf) (Fork (Stem Leaf) (Fork Leaf Leaf))"
|
|
|
|
, testCase "Lambda with a list literal" $ do
|
|
|
|
let input = "f = (\\x : x)\nf [t (t t)]"
|
|
|
|
runSapling input @?= "Fork Leaf (Fork (Stem Leaf) Leaf)"
|
|
|
|
]
|