mirror of
				https://github.com/python/cpython.git
				synced 2025-10-31 13:41:24 +00:00 
			
		
		
		
	 a65366ed87
			
		
	
	
		a65366ed87
		
			
		
	
	
	
	
		
			
			To support virtual terminal mode in Windows PYREPL, we need a scanner to read over the supported escaped VT sequences. Windows REPL input was using virtual key mode, which does not support terminal escape sequences. This patch calls `SetConsoleMode` properly when initializing and send sequences to enable bracketed-paste modes to support verbatim copy-and-paste. Signed-off-by: y5c4l3 <y5c4l3@proton.me> Co-authored-by: Petr Viktorin <encukou@gmail.com> Co-authored-by: Pablo Galindo Salgado <Pablogsal@gmail.com> Co-authored-by: Dustin L. Howett <dustin@howett.net> Co-authored-by: wheeheee <104880306+wheeheee@users.noreply.github.com>
		
			
				
	
	
		
			143 lines
		
	
	
	
		
			4.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			143 lines
		
	
	
	
		
			4.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import tempfile
 | |
| import unittest
 | |
| import sys
 | |
| from unittest.mock import patch
 | |
| from test import support
 | |
| 
 | |
| try:
 | |
|     from _pyrepl.console import Event
 | |
|     from _pyrepl import base_eventqueue
 | |
| except ImportError:
 | |
|     pass
 | |
| 
 | |
| try:
 | |
|     from _pyrepl import unix_eventqueue
 | |
| except ImportError:
 | |
|     pass
 | |
| 
 | |
| try:
 | |
|     from _pyrepl import windows_eventqueue
 | |
| except ImportError:
 | |
|     pass
 | |
| 
 | |
| class EventQueueTestBase:
 | |
|     """OS-independent mixin"""
 | |
|     def make_eventqueue(self) -> base_eventqueue.BaseEventQueue:
 | |
|         raise NotImplementedError()
 | |
| 
 | |
|     def test_get(self):
 | |
|         eq = self.make_eventqueue()
 | |
|         event = Event("key", "a", b"a")
 | |
|         eq.insert(event)
 | |
|         self.assertEqual(eq.get(), event)
 | |
| 
 | |
|     def test_empty(self):
 | |
|         eq = self.make_eventqueue()
 | |
|         self.assertTrue(eq.empty())
 | |
|         eq.insert(Event("key", "a", b"a"))
 | |
|         self.assertFalse(eq.empty())
 | |
| 
 | |
|     def test_flush_buf(self):
 | |
|         eq = self.make_eventqueue()
 | |
|         eq.buf.extend(b"test")
 | |
|         self.assertEqual(eq.flush_buf(), b"test")
 | |
|         self.assertEqual(eq.buf, bytearray())
 | |
| 
 | |
|     def test_insert(self):
 | |
|         eq = self.make_eventqueue()
 | |
|         event = Event("key", "a", b"a")
 | |
|         eq.insert(event)
 | |
|         self.assertEqual(eq.events[0], event)
 | |
| 
 | |
|     @patch("_pyrepl.base_eventqueue.keymap")
 | |
|     def test_push_with_key_in_keymap(self, mock_keymap):
 | |
|         mock_keymap.compile_keymap.return_value = {"a": "b"}
 | |
|         eq = self.make_eventqueue()
 | |
|         eq.keymap = {b"a": "b"}
 | |
|         eq.push("a")
 | |
|         mock_keymap.compile_keymap.assert_called()
 | |
|         self.assertEqual(eq.events[0].evt, "key")
 | |
|         self.assertEqual(eq.events[0].data, "b")
 | |
| 
 | |
|     @patch("_pyrepl.base_eventqueue.keymap")
 | |
|     def test_push_without_key_in_keymap(self, mock_keymap):
 | |
|         mock_keymap.compile_keymap.return_value = {"a": "b"}
 | |
|         eq = self.make_eventqueue()
 | |
|         eq.keymap = {b"c": "d"}
 | |
|         eq.push("a")
 | |
|         mock_keymap.compile_keymap.assert_called()
 | |
|         self.assertEqual(eq.events[0].evt, "key")
 | |
|         self.assertEqual(eq.events[0].data, "a")
 | |
| 
 | |
|     @patch("_pyrepl.base_eventqueue.keymap")
 | |
|     def test_push_with_keymap_in_keymap(self, mock_keymap):
 | |
|         mock_keymap.compile_keymap.return_value = {"a": "b"}
 | |
|         eq = self.make_eventqueue()
 | |
|         eq.keymap = {b"a": {b"b": "c"}}
 | |
|         eq.push("a")
 | |
|         mock_keymap.compile_keymap.assert_called()
 | |
|         self.assertTrue(eq.empty())
 | |
|         eq.push("b")
 | |
|         self.assertEqual(eq.events[0].evt, "key")
 | |
|         self.assertEqual(eq.events[0].data, "c")
 | |
|         eq.push("d")
 | |
|         self.assertEqual(eq.events[1].evt, "key")
 | |
|         self.assertEqual(eq.events[1].data, "d")
 | |
| 
 | |
|     @patch("_pyrepl.base_eventqueue.keymap")
 | |
|     def test_push_with_keymap_in_keymap_and_escape(self, mock_keymap):
 | |
|         mock_keymap.compile_keymap.return_value = {"a": "b"}
 | |
|         eq = self.make_eventqueue()
 | |
|         eq.keymap = {b"a": {b"b": "c"}}
 | |
|         eq.push("a")
 | |
|         mock_keymap.compile_keymap.assert_called()
 | |
|         self.assertTrue(eq.empty())
 | |
|         eq.flush_buf()
 | |
|         eq.push("\033")
 | |
|         self.assertEqual(eq.events[0].evt, "key")
 | |
|         self.assertEqual(eq.events[0].data, "\033")
 | |
|         eq.push("b")
 | |
|         self.assertEqual(eq.events[1].evt, "key")
 | |
|         self.assertEqual(eq.events[1].data, "b")
 | |
| 
 | |
|     def test_push_special_key(self):
 | |
|         eq = self.make_eventqueue()
 | |
|         eq.keymap = {}
 | |
|         eq.push("\x1b")
 | |
|         eq.push("[")
 | |
|         eq.push("A")
 | |
|         self.assertEqual(eq.events[0].evt, "key")
 | |
|         self.assertEqual(eq.events[0].data, "\x1b")
 | |
| 
 | |
|     def test_push_unrecognized_escape_sequence(self):
 | |
|         eq = self.make_eventqueue()
 | |
|         eq.keymap = {}
 | |
|         eq.push("\x1b")
 | |
|         eq.push("[")
 | |
|         eq.push("Z")
 | |
|         self.assertEqual(len(eq.events), 3)
 | |
|         self.assertEqual(eq.events[0].evt, "key")
 | |
|         self.assertEqual(eq.events[0].data, "\x1b")
 | |
|         self.assertEqual(eq.events[1].evt, "key")
 | |
|         self.assertEqual(eq.events[1].data, "[")
 | |
|         self.assertEqual(eq.events[2].evt, "key")
 | |
|         self.assertEqual(eq.events[2].data, "Z")
 | |
| 
 | |
| 
 | |
| @unittest.skipIf(support.MS_WINDOWS, "No Unix event queue on Windows")
 | |
| class TestUnixEventQueue(EventQueueTestBase, unittest.TestCase):
 | |
|     def setUp(self):
 | |
|         self.enterContext(patch("_pyrepl.curses.tigetstr", lambda x: b""))
 | |
|         self.file = tempfile.TemporaryFile()
 | |
| 
 | |
|     def tearDown(self) -> None:
 | |
|         self.file.close()
 | |
| 
 | |
|     def make_eventqueue(self) -> base_eventqueue.BaseEventQueue:
 | |
|         return unix_eventqueue.EventQueue(self.file.fileno(), "utf-8")
 | |
| 
 | |
| 
 | |
| @unittest.skipUnless(support.MS_WINDOWS, "No Windows event queue on Unix")
 | |
| class TestWindowsEventQueue(EventQueueTestBase, unittest.TestCase):
 | |
|     def make_eventqueue(self) -> base_eventqueue.BaseEventQueue:
 | |
|         return windows_eventqueue.EventQueue("utf-8")
 |