| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202 |
- '''Tests for WindowsConsoleIO
- '''
- import io
- import os
- import sys
- import tempfile
- import unittest
- from test.support import os_helper
- if sys.platform != 'win32':
- raise unittest.SkipTest("test only relevant on win32")
- from _testconsole import write_input
- ConIO = io._WindowsConsoleIO
- class WindowsConsoleIOTests(unittest.TestCase):
- def test_abc(self):
- self.assertTrue(issubclass(ConIO, io.RawIOBase))
- self.assertFalse(issubclass(ConIO, io.BufferedIOBase))
- self.assertFalse(issubclass(ConIO, io.TextIOBase))
- def test_open_fd(self):
- self.assertRaisesRegex(ValueError,
- "negative file descriptor", ConIO, -1)
- with tempfile.TemporaryFile() as tmpfile:
- fd = tmpfile.fileno()
- # Windows 10: "Cannot open non-console file"
- # Earlier: "Cannot open console output buffer for reading"
- self.assertRaisesRegex(ValueError,
- "Cannot open (console|non-console file)", ConIO, fd)
- try:
- f = ConIO(0)
- except ValueError:
- # cannot open console because it's not a real console
- pass
- else:
- self.assertTrue(f.readable())
- self.assertFalse(f.writable())
- self.assertEqual(0, f.fileno())
- f.close() # multiple close should not crash
- f.close()
- try:
- f = ConIO(1, 'w')
- except ValueError:
- # cannot open console because it's not a real console
- pass
- else:
- self.assertFalse(f.readable())
- self.assertTrue(f.writable())
- self.assertEqual(1, f.fileno())
- f.close()
- f.close()
- try:
- f = ConIO(2, 'w')
- except ValueError:
- # cannot open console because it's not a real console
- pass
- else:
- self.assertFalse(f.readable())
- self.assertTrue(f.writable())
- self.assertEqual(2, f.fileno())
- f.close()
- f.close()
- def test_open_name(self):
- self.assertRaises(ValueError, ConIO, sys.executable)
- f = ConIO("CON")
- self.assertTrue(f.readable())
- self.assertFalse(f.writable())
- self.assertIsNotNone(f.fileno())
- f.close() # multiple close should not crash
- f.close()
- f = ConIO('CONIN$')
- self.assertTrue(f.readable())
- self.assertFalse(f.writable())
- self.assertIsNotNone(f.fileno())
- f.close()
- f.close()
- f = ConIO('CONOUT$', 'w')
- self.assertFalse(f.readable())
- self.assertTrue(f.writable())
- self.assertIsNotNone(f.fileno())
- f.close()
- f.close()
- # bpo-45354: Windows 11 changed MS-DOS device name handling
- if sys.getwindowsversion()[:3] < (10, 0, 22000):
- f = open('C:/con', 'rb', buffering=0)
- self.assertIsInstance(f, ConIO)
- f.close()
- @unittest.skipIf(sys.getwindowsversion()[:2] <= (6, 1),
- "test does not work on Windows 7 and earlier")
- def test_conin_conout_names(self):
- f = open(r'\\.\conin$', 'rb', buffering=0)
- self.assertIsInstance(f, ConIO)
- f.close()
- f = open('//?/conout$', 'wb', buffering=0)
- self.assertIsInstance(f, ConIO)
- f.close()
- def test_conout_path(self):
- temp_path = tempfile.mkdtemp()
- self.addCleanup(os_helper.rmtree, temp_path)
- conout_path = os.path.join(temp_path, 'CONOUT$')
- with open(conout_path, 'wb', buffering=0) as f:
- # bpo-45354: Windows 11 changed MS-DOS device name handling
- if (6, 1) < sys.getwindowsversion()[:3] < (10, 0, 22000):
- self.assertIsInstance(f, ConIO)
- else:
- self.assertNotIsInstance(f, ConIO)
- def test_write_empty_data(self):
- with ConIO('CONOUT$', 'w') as f:
- self.assertEqual(f.write(b''), 0)
- def assertStdinRoundTrip(self, text):
- stdin = open('CONIN$', 'r')
- old_stdin = sys.stdin
- try:
- sys.stdin = stdin
- write_input(
- stdin.buffer.raw,
- (text + '\r\n').encode('utf-16-le', 'surrogatepass')
- )
- actual = input()
- finally:
- sys.stdin = old_stdin
- self.assertEqual(actual, text)
- def test_input(self):
- # ASCII
- self.assertStdinRoundTrip('abc123')
- # Non-ASCII
- self.assertStdinRoundTrip('ϼўТλФЙ')
- # Combining characters
- self.assertStdinRoundTrip('A͏B ﬖ̳AA̝')
- # bpo-38325
- @unittest.skipIf(True, "Handling Non-BMP characters is broken")
- def test_input_nonbmp(self):
- # Non-BMP
- self.assertStdinRoundTrip('\U00100000\U0010ffff\U0010fffd')
- def test_partial_reads(self):
- # Test that reading less than 1 full character works when stdin
- # contains multibyte UTF-8 sequences
- source = 'ϼўТλФЙ\r\n'.encode('utf-16-le')
- expected = 'ϼўТλФЙ\r\n'.encode('utf-8')
- for read_count in range(1, 16):
- with open('CONIN$', 'rb', buffering=0) as stdin:
- write_input(stdin, source)
- actual = b''
- while not actual.endswith(b'\n'):
- b = stdin.read(read_count)
- actual += b
- self.assertEqual(actual, expected, 'stdin.read({})'.format(read_count))
- # bpo-38325
- @unittest.skipIf(True, "Handling Non-BMP characters is broken")
- def test_partial_surrogate_reads(self):
- # Test that reading less than 1 full character works when stdin
- # contains surrogate pairs that cannot be decoded to UTF-8 without
- # reading an extra character.
- source = '\U00101FFF\U00101001\r\n'.encode('utf-16-le')
- expected = '\U00101FFF\U00101001\r\n'.encode('utf-8')
- for read_count in range(1, 16):
- with open('CONIN$', 'rb', buffering=0) as stdin:
- write_input(stdin, source)
- actual = b''
- while not actual.endswith(b'\n'):
- b = stdin.read(read_count)
- actual += b
- self.assertEqual(actual, expected, 'stdin.read({})'.format(read_count))
- def test_ctrl_z(self):
- with open('CONIN$', 'rb', buffering=0) as stdin:
- source = '\xC4\x1A\r\n'.encode('utf-16-le')
- expected = '\xC4'.encode('utf-8')
- write_input(stdin, source)
- a, b = stdin.read(1), stdin.readall()
- self.assertEqual(expected[0:1], a)
- self.assertEqual(expected[1:], b)
- if __name__ == "__main__":
- unittest.main()
|