From 7b0a11d57020235882600643421076d728988a72 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=C5=81ukasz=20Czajka?= <62751+lukaszcz@users.noreply.github.com> Date: Thu, 1 Feb 2024 13:58:38 +0100 Subject: [PATCH] JuvixTree negative evaluation tests (#2601) * Adds negative tests for the JuvixTree evaluator * Depends on #2600 * Depends on #2599 * Depends on #2598 * Depends on #2597 * Depends on #2596 * Depends on #2595 * Depends on #2594 * Depends on #2590 --- src/Juvix/Compiler/Tree/Evaluator.hs | 8 +++- test/Tree/Eval.hs | 3 +- test/Tree/Eval/Base.hs | 22 ++++++++++ test/Tree/Eval/Negative.hs | 65 ++++++++++++++++++++++++++++ tests/Tree/negative/test001.jvt | 9 ++++ tests/Tree/negative/test002.jvt | 5 +++ tests/Tree/negative/test003.jvt | 13 ++++++ tests/Tree/negative/test004.jvt | 8 ++++ tests/Tree/negative/test005.jvt | 12 +++++ tests/Tree/negative/test006.jvt | 5 +++ tests/Tree/negative/test007.jvt | 9 ++++ tests/Tree/negative/test008.jvt | 9 ++++ 12 files changed, 165 insertions(+), 3 deletions(-) create mode 100644 test/Tree/Eval/Negative.hs create mode 100644 tests/Tree/negative/test001.jvt create mode 100644 tests/Tree/negative/test002.jvt create mode 100644 tests/Tree/negative/test003.jvt create mode 100644 tests/Tree/negative/test004.jvt create mode 100644 tests/Tree/negative/test005.jvt create mode 100644 tests/Tree/negative/test006.jvt create mode 100644 tests/Tree/negative/test007.jvt create mode 100644 tests/Tree/negative/test008.jvt diff --git a/src/Juvix/Compiler/Tree/Evaluator.hs b/src/Juvix/Compiler/Tree/Evaluator.hs index 4711780fe..8279c6b4c 100644 --- a/src/Juvix/Compiler/Tree/Evaluator.hs +++ b/src/Juvix/Compiler/Tree/Evaluator.hs @@ -66,8 +66,12 @@ hEval hout tab = eval' [] mempty IntAdd -> goIntBinop (+) arg1 arg2 IntSub -> goIntBinop (-) arg1 arg2 IntMul -> goIntBinop (*) arg1 arg2 - IntDiv -> goIntBinop quot arg1 arg2 - IntMod -> goIntBinop rem arg1 arg2 + IntDiv + | arg2 == ValInteger 0 -> evalError "division by zero" + | otherwise -> goIntBinop quot arg1 arg2 + IntMod + | arg2 == ValInteger 0 -> evalError "division by zero" + | otherwise -> goIntBinop rem arg1 arg2 IntLe -> goIntCmpBinop (<=) arg1 arg2 IntLt -> goIntCmpBinop (<) arg1 arg2 ValEq diff --git a/test/Tree/Eval.hs b/test/Tree/Eval.hs index 82098e046..18cf26ed6 100644 --- a/test/Tree/Eval.hs +++ b/test/Tree/Eval.hs @@ -1,7 +1,8 @@ module Tree.Eval where import Base +import Tree.Eval.Negative qualified as N import Tree.Eval.Positive qualified as P allTests :: TestTree -allTests = testGroup "JuvixTree evaluation" [P.allTests] +allTests = testGroup "JuvixTree evaluation" [P.allTests, N.allTests] diff --git a/test/Tree/Eval/Base.hs b/test/Tree/Eval/Base.hs index ffa4f28d5..4c2d93cfa 100644 --- a/test/Tree/Eval/Base.hs +++ b/test/Tree/Eval/Base.hs @@ -73,3 +73,25 @@ doEval :: FunctionInfo -> IO (Either TreeError Value) doEval hout tab funInfo = catchEvalErrorIO (hEvalIO stdin hout tab funInfo) + +treeEvalErrorAssertion :: Path Abs File -> (String -> IO ()) -> Assertion +treeEvalErrorAssertion mainFile step = do + step "Parse" + s <- readFile (toFilePath mainFile) + case runParser (toFilePath mainFile) s of + Left err -> assertFailure (show (pretty err)) + Right tab -> + case tab ^. infoMainFunction of + Just sym -> do + withTempDir' + ( \dirPath -> do + let outputFile = dirPath $(mkRelFile "out.out") + hout <- openFile (toFilePath outputFile) WriteMode + step "Evaluate" + r' <- doEval hout tab (lookupFunInfo tab sym) + hClose hout + case r' of + Left _ -> assertBool "" True + Right _ -> assertFailure "no error" + ) + Nothing -> assertFailure "no main function" diff --git a/test/Tree/Eval/Negative.hs b/test/Tree/Eval/Negative.hs new file mode 100644 index 000000000..99e015ca3 --- /dev/null +++ b/test/Tree/Eval/Negative.hs @@ -0,0 +1,65 @@ +module Tree.Eval.Negative where + +import Base +import Tree.Eval.Base + +data NegTest = NegTest + { _name :: String, + _relDir :: Path Rel Dir, + _file :: Path Rel File + } + +root :: Path Abs Dir +root = relToProject $(mkRelDir "tests/Tree/negative") + +testDescr :: NegTest -> TestDescr +testDescr NegTest {..} = + let tRoot = root _relDir + file' = tRoot _file + in TestDescr + { _testName = _name, + _testRoot = tRoot, + _testAssertion = Steps $ treeEvalErrorAssertion file' + } + +allTests :: TestTree +allTests = + testGroup + "JuvixTree negative tests" + (map (mkTest . testDescr) tests) + +tests :: [NegTest] +tests = + [ NegTest + "Test001: Division by zero" + $(mkRelDir ".") + $(mkRelFile "test001.jvt"), + NegTest + "Test002: Arithmetic operations on non-numbers" + $(mkRelDir ".") + $(mkRelFile "test002.jvt"), + NegTest + "Test003: Case on non-data" + $(mkRelDir ".") + $(mkRelFile "test003.jvt"), + NegTest + "Test004: If on non-boolean" + $(mkRelDir ".") + $(mkRelFile "test004.jvt"), + NegTest + "Test005: No matching case branch" + $(mkRelDir ".") + $(mkRelFile "test005.jvt"), + NegTest + "Test006: Invalid closure call" + $(mkRelDir ".") + $(mkRelFile "test006.jvt"), + NegTest + "Test007: Call: wrong number of arguments" + $(mkRelDir ".") + $(mkRelFile "test007.jvt"), + NegTest + "Test008: Closure call: wrong number of arguments" + $(mkRelDir ".") + $(mkRelFile "test008.jvt") + ] diff --git a/tests/Tree/negative/test001.jvt b/tests/Tree/negative/test001.jvt new file mode 100644 index 000000000..41662baad --- /dev/null +++ b/tests/Tree/negative/test001.jvt @@ -0,0 +1,9 @@ +-- division by zero + +function f(x : integer) : integer { + div(2, x) +} + +function main() : * { + call[f](0) +} diff --git a/tests/Tree/negative/test002.jvt b/tests/Tree/negative/test002.jvt new file mode 100644 index 000000000..024646703 --- /dev/null +++ b/tests/Tree/negative/test002.jvt @@ -0,0 +1,5 @@ +-- arithmetic operations on non-numbers + +function main() : * { + add(2, calloc[main]()) +} diff --git a/tests/Tree/negative/test003.jvt b/tests/Tree/negative/test003.jvt new file mode 100644 index 000000000..189555fa2 --- /dev/null +++ b/tests/Tree/negative/test003.jvt @@ -0,0 +1,13 @@ +-- case on non-data + +type list { + nil : list; + cons : * -> list -> list; +} + +function main() : * { + case[list](3) { + nil: 0 + cons: 1 + } +} diff --git a/tests/Tree/negative/test004.jvt b/tests/Tree/negative/test004.jvt new file mode 100644 index 000000000..a92b2359c --- /dev/null +++ b/tests/Tree/negative/test004.jvt @@ -0,0 +1,8 @@ +-- if on non-boolean + +function main() : * { + br(3) { + true: 0 + false: 1 + } +} diff --git a/tests/Tree/negative/test005.jvt b/tests/Tree/negative/test005.jvt new file mode 100644 index 000000000..706be4775 --- /dev/null +++ b/tests/Tree/negative/test005.jvt @@ -0,0 +1,12 @@ +-- no matching case branch + +type list { + nil : list; + cons : * -> list -> list; +} + +function main() : * { + case[list](alloc[nil]()) { + cons: 1 + } +} diff --git a/tests/Tree/negative/test006.jvt b/tests/Tree/negative/test006.jvt new file mode 100644 index 000000000..2555db520 --- /dev/null +++ b/tests/Tree/negative/test006.jvt @@ -0,0 +1,5 @@ +-- invalid closure call + +function main() : * { + call(2, 1) +} diff --git a/tests/Tree/negative/test007.jvt b/tests/Tree/negative/test007.jvt new file mode 100644 index 000000000..762d1ef9d --- /dev/null +++ b/tests/Tree/negative/test007.jvt @@ -0,0 +1,9 @@ +-- call: wrong number of arguments + +function f(x : *) : * { + x +} + +function main() : * { + call[f](10, 11) +} diff --git a/tests/Tree/negative/test008.jvt b/tests/Tree/negative/test008.jvt new file mode 100644 index 000000000..f9414ebf0 --- /dev/null +++ b/tests/Tree/negative/test008.jvt @@ -0,0 +1,9 @@ +-- closure call: wrong number of arguments + +function f(x : *) : * { + x +} + +function main() : * { + call(calloc[f](), 10, 11) +}