2024-05-21 23:36:01 -03:00
|
|
|
import string
|
2024-05-21 13:44:09 -03:00
|
|
|
import unittest
|
|
|
|
|
2024-05-21 23:36:01 -03:00
|
|
|
from _pyrepl.keymap import _keynames, _escapes, parse_keys, compile_keymap, KeySpecError
|
2024-05-21 13:44:09 -03:00
|
|
|
|
|
|
|
|
|
|
|
class TestParseKeys(unittest.TestCase):
|
|
|
|
def test_single_character(self):
|
2024-05-21 23:36:01 -03:00
|
|
|
"""Ensure that single ascii characters or single digits are parsed as single characters."""
|
|
|
|
test_cases = [(key, [key]) for key in string.ascii_letters + string.digits]
|
|
|
|
for test_key, expected_keys in test_cases:
|
|
|
|
with self.subTest(f"{test_key} should be parsed as {expected_keys}"):
|
|
|
|
self.assertEqual(parse_keys(test_key), expected_keys)
|
|
|
|
|
|
|
|
def test_keynames(self):
|
|
|
|
"""Ensure that keynames are parsed to their corresponding mapping.
|
|
|
|
|
|
|
|
A keyname is expected to be of the following form: \\<keyname> such as \\<left>
|
|
|
|
which would get parsed as "left".
|
|
|
|
"""
|
|
|
|
test_cases = [(f"\\<{keyname}>", [parsed_keyname]) for keyname, parsed_keyname in _keynames.items()]
|
|
|
|
for test_key, expected_keys in test_cases:
|
|
|
|
with self.subTest(f"{test_key} should be parsed as {expected_keys}"):
|
|
|
|
self.assertEqual(parse_keys(test_key), expected_keys)
|
2024-05-21 13:44:09 -03:00
|
|
|
|
|
|
|
def test_escape_sequences(self):
|
2024-05-21 23:36:01 -03:00
|
|
|
"""Ensure that escaping sequences are parsed to their corresponding mapping."""
|
|
|
|
test_cases = [(f"\\{escape}", [parsed_escape]) for escape, parsed_escape in _escapes.items()]
|
|
|
|
for test_key, expected_keys in test_cases:
|
|
|
|
with self.subTest(f"{test_key} should be parsed as {expected_keys}"):
|
|
|
|
self.assertEqual(parse_keys(test_key), expected_keys)
|
2024-05-21 13:44:09 -03:00
|
|
|
|
|
|
|
def test_control_sequences(self):
|
2024-05-21 23:36:01 -03:00
|
|
|
"""Ensure that supported control sequences are parsed successfully."""
|
|
|
|
keys = ["@", "[", "]", "\\", "^", "_", "\\<space>", "\\<delete>"]
|
|
|
|
keys.extend(string.ascii_letters)
|
|
|
|
test_cases = [(f"\\C-{key}", chr(ord(key) & 0x1F)) for key in []]
|
|
|
|
for test_key, expected_keys in test_cases:
|
|
|
|
with self.subTest(f"{test_key} should be parsed as {expected_keys}"):
|
|
|
|
self.assertEqual(parse_keys(test_key), expected_keys)
|
2024-05-21 13:44:09 -03:00
|
|
|
|
|
|
|
def test_meta_sequences(self):
|
|
|
|
self.assertEqual(parse_keys("\\M-a"), ["\033", "a"])
|
|
|
|
self.assertEqual(parse_keys("\\M-b"), ["\033", "b"])
|
|
|
|
self.assertEqual(parse_keys("\\M-c"), ["\033", "c"])
|
|
|
|
|
|
|
|
def test_combinations(self):
|
|
|
|
self.assertEqual(parse_keys("\\C-a\\n\\<up>"), ["\x01", "\n", "up"])
|
|
|
|
self.assertEqual(parse_keys("\\M-a\\t\\<down>"), ["\033", "a", "\t", "down"])
|
|
|
|
|
2024-05-21 23:36:01 -03:00
|
|
|
def test_keyspec_errors(self):
|
|
|
|
cases = [
|
|
|
|
("\\Ca", "\\C must be followed by `-'"),
|
|
|
|
("\\ca", "\\C must be followed by `-'"),
|
|
|
|
("\\C-\\C-", "doubled \\C-"),
|
|
|
|
("\\Ma", "\\M must be followed by `-'"),
|
|
|
|
("\\ma", "\\M must be followed by `-'"),
|
|
|
|
("\\M-\\M-", "doubled \\M-"),
|
|
|
|
("\\<left", "unterminated \\<"),
|
|
|
|
("\\<unsupported>", "unrecognised keyname"),
|
|
|
|
("\\大", "unknown backslash escape"),
|
|
|
|
("\\C-\\<backspace>", "\\C- followed by invalid key")
|
|
|
|
]
|
|
|
|
for test_keys, expected_err in cases:
|
|
|
|
with self.subTest(f"{test_keys} should give error {expected_err}"):
|
|
|
|
with self.assertRaises(KeySpecError) as e:
|
|
|
|
parse_keys(test_keys)
|
|
|
|
self.assertIn(expected_err, str(e.exception))
|
|
|
|
|
|
|
|
def test_index_errors(self):
|
|
|
|
test_cases = ["\\", "\\C", "\\C-\\C"]
|
|
|
|
for test_keys in test_cases:
|
|
|
|
with self.assertRaises(IndexError):
|
|
|
|
parse_keys(test_keys)
|
|
|
|
|
2024-05-21 13:44:09 -03:00
|
|
|
|
|
|
|
class TestCompileKeymap(unittest.TestCase):
|
|
|
|
def test_empty_keymap(self):
|
|
|
|
keymap = {}
|
|
|
|
result = compile_keymap(keymap)
|
|
|
|
self.assertEqual(result, {})
|
|
|
|
|
|
|
|
def test_single_keymap(self):
|
|
|
|
keymap = {b"a": "action"}
|
|
|
|
result = compile_keymap(keymap)
|
|
|
|
self.assertEqual(result, {b"a": "action"})
|
|
|
|
|
|
|
|
def test_nested_keymap(self):
|
|
|
|
keymap = {b"a": {b"b": "action"}}
|
|
|
|
result = compile_keymap(keymap)
|
|
|
|
self.assertEqual(result, {b"a": {b"b": "action"}})
|
|
|
|
|
|
|
|
def test_empty_value(self):
|
|
|
|
keymap = {b"a": {b"": "action"}}
|
|
|
|
result = compile_keymap(keymap)
|
|
|
|
self.assertEqual(result, {b"a": {b"": "action"}})
|
|
|
|
|
|
|
|
def test_multiple_empty_values(self):
|
|
|
|
keymap = {b"a": {b"": "action1", b"b": "action2"}}
|
|
|
|
result = compile_keymap(keymap)
|
|
|
|
self.assertEqual(result, {b"a": {b"": "action1", b"b": "action2"}})
|
|
|
|
|
|
|
|
def test_multiple_keymaps(self):
|
|
|
|
keymap = {b"a": {b"b": "action1", b"c": "action2"}}
|
|
|
|
result = compile_keymap(keymap)
|
|
|
|
self.assertEqual(result, {b"a": {b"b": "action1", b"c": "action2"}})
|
|
|
|
|
|
|
|
def test_nested_multiple_keymaps(self):
|
|
|
|
keymap = {b"a": {b"b": {b"c": "action"}}}
|
|
|
|
result = compile_keymap(keymap)
|
|
|
|
self.assertEqual(result, {b"a": {b"b": {b"c": "action"}}})
|
2024-05-21 23:36:01 -03:00
|
|
|
|
|
|
|
def test_clashing_definitions(self):
|
|
|
|
km = {b'a': 'c', b'a' + b'b': 'd'}
|
|
|
|
with self.assertRaises(KeySpecError):
|
|
|
|
compile_keymap(km)
|
|
|
|
|
|
|
|
def test_non_bytes_key(self):
|
|
|
|
with self.assertRaises(TypeError):
|
|
|
|
compile_keymap({123: 'a'})
|