mirror of
				https://github.com/python/cpython.git
				synced 2025-11-03 23:21:29 +00:00 
			
		
		
		
	(cherry picked from commit 73ab83b27f)
Co-authored-by: Eugene Triguba <eugenetriguba@gmail.com>
Co-authored-by: Łukasz Langa <lukasz@langa.pl>
		
	
			
		
			
				
	
	
		
			120 lines
		
	
	
	
		
			5.1 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			120 lines
		
	
	
	
		
			5.1 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
import string
 | 
						|
import unittest
 | 
						|
 | 
						|
from _pyrepl.keymap import _keynames, _escapes, parse_keys, compile_keymap, KeySpecError
 | 
						|
 | 
						|
 | 
						|
class TestParseKeys(unittest.TestCase):
 | 
						|
    def test_single_character(self):
 | 
						|
        """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)
 | 
						|
 | 
						|
    def test_escape_sequences(self):
 | 
						|
        """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)
 | 
						|
 | 
						|
    def test_control_sequences(self):
 | 
						|
        """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)
 | 
						|
 | 
						|
    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"])
 | 
						|
 | 
						|
    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)
 | 
						|
 | 
						|
 | 
						|
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"}}})
 | 
						|
 | 
						|
    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'})
 |