Files
python/exercises/practice/forth/forth_test.py

204 lines
7.0 KiB
Python
Raw Normal View History

import unittest
2017-10-23 15:06:44 -05:00
from forth import evaluate, StackUnderflowError
2020-10-15 12:46:24 -04:00
# Tests adapted from `problem-specifications//canonical-data.json`
class ForthTest(unittest.TestCase):
# Utility functions
def assertRaisesWithMessage(self, exception):
return self.assertRaisesRegex(exception, r".+")
class ParsingAndNumbersTest(ForthTest):
def test_numbers_just_get_pushed_onto_the_stack(self):
self.assertEqual(evaluate(["1 2 3 4 5"]), [1, 2, 3, 4, 5])
2017-10-09 09:14:56 -05:00
class AdditionTest(ForthTest):
2017-10-06 15:34:38 -05:00
def test_can_add_two_numbers(self):
self.assertEqual(evaluate(["1 2 +"]), [3])
2017-10-06 15:34:38 -05:00
def test_errors_if_there_is_nothing_on_the_stack(self):
with self.assertRaisesWithMessage(StackUnderflowError):
evaluate(["+"])
2017-10-06 15:34:38 -05:00
def test_errors_if_there_is_only_one_value_on_the_stack(self):
with self.assertRaisesWithMessage(StackUnderflowError):
evaluate(["1 +"])
2017-10-06 15:34:38 -05:00
class SubtractionTest(ForthTest):
2017-10-06 15:34:38 -05:00
def test_can_subtract_two_numbers(self):
self.assertEqual(evaluate(["3 4 -"]), [-1])
2017-10-06 15:34:38 -05:00
def test_errors_if_there_is_nothing_on_the_stack(self):
with self.assertRaisesWithMessage(StackUnderflowError):
evaluate(["-"])
2017-10-06 15:34:38 -05:00
def test_errors_if_there_is_only_one_value_on_the_stack(self):
with self.assertRaisesWithMessage(StackUnderflowError):
evaluate(["1 -"])
2017-10-06 15:34:38 -05:00
class MultiplicationTest(ForthTest):
2017-10-06 15:34:38 -05:00
def test_can_multiply_two_numbers(self):
self.assertEqual(evaluate(["2 4 *"]), [8])
2017-10-06 15:34:38 -05:00
def test_errors_if_there_is_nothing_on_the_stack(self):
with self.assertRaisesWithMessage(StackUnderflowError):
evaluate(["*"])
2017-10-06 15:34:38 -05:00
def test_errors_if_there_is_only_one_value_on_the_stack(self):
with self.assertRaisesWithMessage(StackUnderflowError):
evaluate(["1 *"])
2017-10-06 15:34:38 -05:00
class DivisionTest(ForthTest):
2017-10-06 15:34:38 -05:00
def test_can_divide_two_numbers(self):
self.assertEqual(evaluate(["12 3 /"]), [4])
2017-10-06 15:34:38 -05:00
def test_performs_integer_division(self):
self.assertEqual(evaluate(["8 3 /"]), [2])
2017-10-06 15:34:38 -05:00
def test_errors_if_dividing_by_zero(self):
# divide by zero
with self.assertRaisesWithMessage(ZeroDivisionError):
evaluate(["4 0 /"])
2017-10-06 15:34:38 -05:00
def test_errors_if_there_is_nothing_on_the_stack(self):
with self.assertRaisesWithMessage(StackUnderflowError):
evaluate(["/"])
2017-10-06 15:34:38 -05:00
def test_errors_if_there_is_only_one_value_on_the_stack(self):
with self.assertRaisesWithMessage(StackUnderflowError):
evaluate(["1 /"])
2017-10-06 15:34:38 -05:00
class CombinedArithmeticTest(ForthTest):
2017-10-06 15:34:38 -05:00
def test_addition_and_subtraction(self):
self.assertEqual(evaluate(["1 2 + 4 -"]), [-1])
2017-10-06 15:34:38 -05:00
def test_multiplication_and_division(self):
self.assertEqual(evaluate(["2 4 * 3 /"]), [2])
2017-10-06 15:34:38 -05:00
class DupTest(ForthTest):
def test_copies_a_value_on_the_stack(self):
self.assertEqual(evaluate(["1 dup"]), [1, 1])
2017-10-06 15:34:38 -05:00
def test_copies_the_top_value_on_the_stack(self):
self.assertEqual(evaluate(["1 2 dup"]), [1, 2, 2])
2017-10-06 15:34:38 -05:00
def test_errors_if_there_is_nothing_on_the_stack(self):
with self.assertRaisesWithMessage(StackUnderflowError):
evaluate(["dup"])
2017-10-06 15:34:38 -05:00
class DropTest(ForthTest):
2017-10-06 15:34:38 -05:00
def test_removes_the_top_value_on_the_stack_if_it_is_the_only_one(self):
self.assertEqual(evaluate(["1 drop"]), [])
2017-10-06 15:34:38 -05:00
def test_removes_the_top_value_on_the_stack_if_it_is_not_the_only_one(self):
self.assertEqual(evaluate(["1 2 drop"]), [1])
2017-10-06 15:34:38 -05:00
def test_errors_if_there_is_nothing_on_the_stack(self):
with self.assertRaisesWithMessage(StackUnderflowError):
evaluate(["drop"])
2017-10-06 15:34:38 -05:00
class SwapTest(ForthTest):
def test_swaps_the_top_two_values_on_the_stack_if_they_are_the_only_ones(self):
self.assertEqual(evaluate(["1 2 swap"]), [2, 1])
2017-10-06 15:34:38 -05:00
def test_swaps_the_top_two_values_on_the_stack_if_they_are_not_the_only_ones(self):
self.assertEqual(evaluate(["1 2 3 swap"]), [1, 3, 2])
2017-10-06 15:34:38 -05:00
def test_errors_if_there_is_nothing_on_the_stack(self):
with self.assertRaisesWithMessage(StackUnderflowError):
evaluate(["swap"])
2017-10-06 15:34:38 -05:00
def test_errors_if_there_is_only_one_value_on_the_stack(self):
with self.assertRaisesWithMessage(StackUnderflowError):
evaluate(["1 swap"])
2017-10-06 15:34:38 -05:00
class OverTest(ForthTest):
2017-10-06 15:34:38 -05:00
def test_copies_the_second_element_if_there_are_only_two(self):
self.assertEqual(evaluate(["1 2 over"]), [1, 2, 1])
2017-10-06 15:34:38 -05:00
def test_copies_the_second_element_if_there_are_more_than_two(self):
self.assertEqual(evaluate(["1 2 3 over"]), [1, 2, 3, 2])
2017-10-06 15:34:38 -05:00
def test_errors_if_there_is_nothing_on_the_stack(self):
with self.assertRaisesWithMessage(StackUnderflowError):
evaluate(["over"])
2017-10-06 15:34:38 -05:00
def test_errors_if_there_is_only_one_value_on_the_stack(self):
with self.assertRaisesWithMessage(StackUnderflowError):
evaluate(["1 over"])
2017-10-06 15:34:38 -05:00
class UserDefinedWordsTest(ForthTest):
2017-10-06 15:34:38 -05:00
def test_can_consist_of_built_in_words(self):
self.assertEqual(evaluate([": dup-twice dup dup ;", "1 dup-twice"]), [1, 1, 1])
2017-10-06 15:34:38 -05:00
def test_execute_in_the_right_order(self):
self.assertEqual(evaluate([": countup 1 2 3 ;", "countup"]), [1, 2, 3])
2017-10-06 15:34:38 -05:00
def test_can_override_other_user_defined_words(self):
self.assertEqual(
evaluate([": foo dup ;", ": foo dup dup ;", "1 foo"]), [1, 1, 1]
)
2017-10-06 15:34:38 -05:00
def test_can_override_built_in_words(self):
self.assertEqual(evaluate([": swap dup ;", "1 swap"]), [1, 1])
2017-10-06 15:34:38 -05:00
def test_can_override_built_in_operators(self):
self.assertEqual(evaluate([": + * ;", "3 4 +"]), [12])
def test_can_use_different_words_with_the_same_name(self):
self.assertEqual(
evaluate([": foo 5 ;", ": bar foo ;", ": foo 6 ;", "bar foo"]), [5, 6]
)
def test_can_define_word_that_uses_word_with_the_same_name(self):
self.assertEqual(evaluate([": foo 10 ;", ": foo foo 1 + ;", "foo"]), [11])
2018-07-20 14:33:17 -04:00
2017-10-06 15:34:38 -05:00
def test_cannot_redefine_numbers(self):
with self.assertRaisesWithMessage(ValueError):
evaluate([": 1 2 ;"])
2017-10-06 15:34:38 -05:00
def test_errors_if_executing_a_non_existent_word(self):
with self.assertRaisesWithMessage(ValueError):
evaluate(["foo"])
2017-10-06 15:34:38 -05:00
class CaseInsensitivityTest(ForthTest):
def test_dup_is_case_insensitive(self):
self.assertEqual(evaluate(["1 DUP Dup dup"]), [1, 1, 1, 1])
def test_drop_is_case_insensitive(self):
self.assertEqual(evaluate(["1 2 3 4 DROP Drop drop"]), [1])
def test_swap_is_case_insensitive(self):
self.assertEqual(evaluate(["1 2 SWAP 3 Swap 4 swap"]), [2, 3, 4, 1])
def test_over_is_case_insensitive(self):
self.assertEqual(evaluate(["1 2 OVER Over over"]), [1, 2, 1, 2, 1])
def test_user_defined_words_are_case_insensitive(self):
self.assertEqual(evaluate([": foo dup ;", "1 FOO Foo foo"]), [1, 1, 1, 1])
def test_definitions_are_case_insensitive(self):
self.assertEqual(evaluate([": SWAP DUP Dup dup ;", "1 swap"]), [1, 1, 1, 1])
# Utility functions
def assertRaisesWithMessage(self, exception):
return self.assertRaisesRegex(exception, r".+")
if __name__ == "__main__":
2017-10-06 15:34:38 -05:00
unittest.main()