mirror of
https://github.com/jart/cosmopolitan.git
synced 2025-02-07 15:03:34 +00:00
8af197560e
Actually Portable Python is now outperforming the Python binaries that come bundled with Linux distros, at things like HTTP serving. You can now have a fully featured Python install in just one .com file that runs on six operating systems and is about 10mb in size. With tuning, the tiniest is ~1mb. We've got most of the libraries working, including pysqlite, and the repl now feels very pleasant. The things you can't do quite yet are: threads and shared objects but that can happen in the future, if the community falls in love with this project and wants to see it developed further. Changes: - Add siginterrupt() - Add sqlite3 to Python - Add issymlink() helper - Make GetZipCdir() faster - Add tgamma() and finite() - Add legacy function lutimes() - Add readlink() and realpath() - Use heap allocations when appropriate - Reorganize Python into two-stage build - Save Lua / Python shell history to dotfile - Integrate Python Lib embedding into linkage - Make isregularfile() and isdirectory() go faster - Make Python shell auto-completion work perfectly - Make crash reports work better if changed directory - Fix Python+NT open() / access() flag overflow error - Disable Python tests relating to \N{LONG NAME} syntax - Have Python REPL copyright() show all notice embeddings The biggest technical challenge at the moment is working around when Python tries to be too clever about filenames.
155 lines
5.6 KiB
Python
155 lines
5.6 KiB
Python
"Test InteractiveConsole and InteractiveInterpreter from code module"
|
|
import sys
|
|
import unittest
|
|
from textwrap import dedent
|
|
from contextlib import ExitStack
|
|
from unittest import mock
|
|
from test import support
|
|
|
|
code = support.import_module('code')
|
|
|
|
|
|
class TestInteractiveConsole(unittest.TestCase):
|
|
|
|
def setUp(self):
|
|
self.console = code.InteractiveConsole()
|
|
self.mock_sys()
|
|
|
|
def mock_sys(self):
|
|
"Mock system environment for InteractiveConsole"
|
|
# use exit stack to match patch context managers to addCleanup
|
|
stack = ExitStack()
|
|
self.addCleanup(stack.close)
|
|
self.infunc = stack.enter_context(mock.patch('code.input',
|
|
create=True))
|
|
self.stdout = stack.enter_context(mock.patch('code.sys.stdout'))
|
|
self.stderr = stack.enter_context(mock.patch('code.sys.stderr'))
|
|
prepatch = mock.patch('code.sys', wraps=code.sys, spec=code.sys)
|
|
self.sysmod = stack.enter_context(prepatch)
|
|
if sys.excepthook is sys.__excepthook__:
|
|
self.sysmod.excepthook = self.sysmod.__excepthook__
|
|
del self.sysmod.ps1
|
|
del self.sysmod.ps2
|
|
|
|
def test_ps1(self):
|
|
self.infunc.side_effect = EOFError('Finished')
|
|
self.console.interact()
|
|
self.assertEqual(self.sysmod.ps1, '>>> ')
|
|
self.sysmod.ps1 = 'custom1> '
|
|
self.console.interact()
|
|
self.assertEqual(self.sysmod.ps1, 'custom1> ')
|
|
|
|
def test_ps2(self):
|
|
self.infunc.side_effect = EOFError('Finished')
|
|
self.console.interact()
|
|
self.assertEqual(self.sysmod.ps2, '... ')
|
|
self.sysmod.ps1 = 'custom2> '
|
|
self.console.interact()
|
|
self.assertEqual(self.sysmod.ps1, 'custom2> ')
|
|
|
|
def test_console_stderr(self):
|
|
self.infunc.side_effect = ["'antioch'", "", EOFError('Finished')]
|
|
self.console.interact()
|
|
for call in list(self.stdout.method_calls):
|
|
if 'antioch' in ''.join(call[1]):
|
|
break
|
|
else:
|
|
raise AssertionError("no console stdout")
|
|
|
|
def test_syntax_error(self):
|
|
self.infunc.side_effect = ["undefined", EOFError('Finished')]
|
|
self.console.interact()
|
|
for call in self.stderr.method_calls:
|
|
if 'NameError' in ''.join(call[1]):
|
|
break
|
|
else:
|
|
raise AssertionError("No syntax error from console")
|
|
|
|
def test_sysexcepthook(self):
|
|
self.infunc.side_effect = ["raise ValueError('')",
|
|
EOFError('Finished')]
|
|
hook = mock.Mock()
|
|
self.sysmod.excepthook = hook
|
|
self.console.interact()
|
|
self.assertTrue(hook.called)
|
|
|
|
def test_banner(self):
|
|
# with banner
|
|
self.infunc.side_effect = EOFError('Finished')
|
|
self.console.interact(banner='Foo')
|
|
self.assertEqual(len(self.stderr.method_calls), 3)
|
|
banner_call = self.stderr.method_calls[0]
|
|
self.assertEqual(banner_call, ['write', ('Foo\n',), {}])
|
|
|
|
# no banner
|
|
self.stderr.reset_mock()
|
|
self.infunc.side_effect = EOFError('Finished')
|
|
self.console.interact(banner='')
|
|
self.assertEqual(len(self.stderr.method_calls), 2)
|
|
|
|
def test_exit_msg(self):
|
|
# default exit message
|
|
self.infunc.side_effect = EOFError('Finished')
|
|
self.console.interact(banner='')
|
|
self.assertEqual(len(self.stderr.method_calls), 2)
|
|
err_msg = self.stderr.method_calls[1]
|
|
expected = 'now exiting InteractiveConsole...\n'
|
|
self.assertEqual(err_msg, ['write', (expected,), {}])
|
|
|
|
# no exit message
|
|
self.stderr.reset_mock()
|
|
self.infunc.side_effect = EOFError('Finished')
|
|
self.console.interact(banner='', exitmsg='')
|
|
self.assertEqual(len(self.stderr.method_calls), 1)
|
|
|
|
# TODO(jart): pycomp.com needs \N thing
|
|
# # custom exit message
|
|
# self.stderr.reset_mock()
|
|
# message = (
|
|
# 'bye! \N{GREEK SMALL LETTER ZETA}\N{CYRILLIC SMALL LETTER ZHE}'
|
|
# )
|
|
# self.infunc.side_effect = EOFError('Finished')
|
|
# self.console.interact(banner='', exitmsg=message)
|
|
# self.assertEqual(len(self.stderr.method_calls), 2)
|
|
# err_msg = self.stderr.method_calls[1]
|
|
# expected = message + '\n'
|
|
# self.assertEqual(err_msg, ['write', (expected,), {}])
|
|
|
|
|
|
def test_cause_tb(self):
|
|
self.infunc.side_effect = ["raise ValueError('') from AttributeError",
|
|
EOFError('Finished')]
|
|
self.console.interact()
|
|
output = ''.join(''.join(call[1]) for call in self.stderr.method_calls)
|
|
expected = dedent("""
|
|
AttributeError
|
|
|
|
The above exception was the direct cause of the following exception:
|
|
|
|
Traceback (most recent call last):
|
|
File "<console>", line 1, in <module>
|
|
ValueError
|
|
""")
|
|
self.assertIn(expected, output)
|
|
|
|
def test_context_tb(self):
|
|
self.infunc.side_effect = ["try: ham\nexcept: eggs\n",
|
|
EOFError('Finished')]
|
|
self.console.interact()
|
|
output = ''.join(''.join(call[1]) for call in self.stderr.method_calls)
|
|
expected = dedent("""
|
|
Traceback (most recent call last):
|
|
File "<console>", line 1, in <module>
|
|
NameError: name 'ham' is not defined
|
|
|
|
During handling of the above exception, another exception occurred:
|
|
|
|
Traceback (most recent call last):
|
|
File "<console>", line 2, in <module>
|
|
NameError: name 'eggs' is not defined
|
|
""")
|
|
self.assertIn(expected, output)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
unittest.main()
|