2018-01-29 21:23:44 -04:00
|
|
|
import contextlib
|
2018-05-16 16:04:57 -03:00
|
|
|
import itertools
|
2018-01-29 21:23:44 -04:00
|
|
|
import os
|
|
|
|
import pickle
|
2018-05-16 16:04:57 -03:00
|
|
|
import sys
|
2019-07-01 13:28:25 -03:00
|
|
|
from textwrap import dedent
|
2018-01-29 21:23:44 -04:00
|
|
|
import threading
|
|
|
|
import unittest
|
|
|
|
|
|
|
|
from test import support
|
2020-06-25 07:38:51 -03:00
|
|
|
from test.support import import_helper
|
2023-11-22 17:48:45 -04:00
|
|
|
from test.support import os_helper
|
2018-01-29 21:23:44 -04:00
|
|
|
from test.support import script_helper
|
|
|
|
|
2018-06-07 21:28:28 -03:00
|
|
|
|
2020-06-25 07:38:51 -03:00
|
|
|
interpreters = import_helper.import_module('_xxsubinterpreters')
|
2024-03-09 09:28:13 -04:00
|
|
|
_testinternalcapi = import_helper.import_module('_testinternalcapi')
|
2023-12-12 11:24:31 -04:00
|
|
|
from _xxsubinterpreters import InterpreterNotFoundError
|
2018-01-29 21:23:44 -04:00
|
|
|
|
|
|
|
|
2018-05-16 16:04:57 -03:00
|
|
|
##################################
|
|
|
|
# helpers
|
|
|
|
|
2018-01-29 21:23:44 -04:00
|
|
|
def _captured_script(script):
|
|
|
|
r, w = os.pipe()
|
|
|
|
indented = script.replace('\n', '\n ')
|
|
|
|
wrapped = dedent(f"""
|
|
|
|
import contextlib
|
2021-04-02 00:53:46 -03:00
|
|
|
with open({w}, 'w', encoding="utf-8") as spipe:
|
2018-05-16 16:04:57 -03:00
|
|
|
with contextlib.redirect_stdout(spipe):
|
2018-01-29 21:23:44 -04:00
|
|
|
{indented}
|
|
|
|
""")
|
2021-04-02 00:53:46 -03:00
|
|
|
return wrapped, open(r, encoding="utf-8")
|
2018-01-29 21:23:44 -04:00
|
|
|
|
|
|
|
|
2023-12-12 14:06:06 -04:00
|
|
|
def _run_output(interp, request):
|
2018-05-16 16:04:57 -03:00
|
|
|
script, rpipe = _captured_script(request)
|
|
|
|
with rpipe:
|
2023-12-12 14:06:06 -04:00
|
|
|
interpreters.run_string(interp, script)
|
2018-05-16 16:04:57 -03:00
|
|
|
return rpipe.read()
|
2018-01-29 21:23:44 -04:00
|
|
|
|
|
|
|
|
2021-07-29 14:05:49 -03:00
|
|
|
def _wait_for_interp_to_run(interp, timeout=None):
|
|
|
|
# bpo-37224: Running this test file in multiprocesses will fail randomly.
|
|
|
|
# The failure reason is that the thread can't acquire the cpu to
|
|
|
|
# run subinterpreter eariler than the main thread in multiprocess.
|
|
|
|
if timeout is None:
|
|
|
|
timeout = support.SHORT_TIMEOUT
|
2022-06-15 06:42:10 -03:00
|
|
|
for _ in support.sleeping_retry(timeout, error=False):
|
|
|
|
if interpreters.is_running(interp):
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
raise RuntimeError('interp is not running')
|
2021-07-29 14:05:49 -03:00
|
|
|
|
|
|
|
|
2018-01-29 21:23:44 -04:00
|
|
|
@contextlib.contextmanager
|
|
|
|
def _running(interp):
|
|
|
|
r, w = os.pipe()
|
|
|
|
def run():
|
|
|
|
interpreters.run_string(interp, dedent(f"""
|
|
|
|
# wait for "signal"
|
2021-04-02 00:53:46 -03:00
|
|
|
with open({r}, encoding="utf-8") as rpipe:
|
2018-05-16 16:04:57 -03:00
|
|
|
rpipe.read()
|
2018-01-29 21:23:44 -04:00
|
|
|
"""))
|
|
|
|
|
|
|
|
t = threading.Thread(target=run)
|
|
|
|
t.start()
|
2021-07-29 14:05:49 -03:00
|
|
|
_wait_for_interp_to_run(interp)
|
2018-01-29 21:23:44 -04:00
|
|
|
|
|
|
|
yield
|
|
|
|
|
2021-04-02 00:53:46 -03:00
|
|
|
with open(w, 'w', encoding="utf-8") as spipe:
|
2018-05-16 16:04:57 -03:00
|
|
|
spipe.write('done')
|
2018-01-29 21:23:44 -04:00
|
|
|
t.join()
|
|
|
|
|
|
|
|
|
2018-05-16 16:04:57 -03:00
|
|
|
def clean_up_interpreters():
|
|
|
|
for id in interpreters.list_all():
|
|
|
|
if id == 0: # main
|
|
|
|
continue
|
|
|
|
try:
|
|
|
|
interpreters.destroy(id)
|
2024-04-03 13:58:39 -03:00
|
|
|
except interpreters.InterpreterError:
|
2018-05-16 16:04:57 -03:00
|
|
|
pass # already destroyed
|
|
|
|
|
|
|
|
|
|
|
|
class TestBase(unittest.TestCase):
|
|
|
|
|
|
|
|
def tearDown(self):
|
2022-12-05 16:40:20 -04:00
|
|
|
clean_up_interpreters()
|
2018-05-16 16:04:57 -03:00
|
|
|
|
|
|
|
|
|
|
|
##################################
|
|
|
|
# misc. tests
|
|
|
|
|
2018-01-29 21:23:44 -04:00
|
|
|
class IsShareableTests(unittest.TestCase):
|
|
|
|
|
|
|
|
def test_default_shareables(self):
|
|
|
|
shareables = [
|
|
|
|
# singletons
|
|
|
|
None,
|
|
|
|
# builtin objects
|
|
|
|
b'spam',
|
2018-05-16 16:04:57 -03:00
|
|
|
'spam',
|
|
|
|
10,
|
|
|
|
-10,
|
2023-11-01 21:09:01 -03:00
|
|
|
True,
|
|
|
|
False,
|
2023-10-31 11:17:20 -03:00
|
|
|
100.0,
|
2023-11-07 13:58:29 -04:00
|
|
|
(1, ('spam', 'eggs')),
|
2018-01-29 21:23:44 -04:00
|
|
|
]
|
|
|
|
for obj in shareables:
|
|
|
|
with self.subTest(obj):
|
|
|
|
self.assertTrue(
|
|
|
|
interpreters.is_shareable(obj))
|
|
|
|
|
|
|
|
def test_not_shareable(self):
|
|
|
|
class Cheese:
|
|
|
|
def __init__(self, name):
|
|
|
|
self.name = name
|
|
|
|
def __str__(self):
|
|
|
|
return self.name
|
|
|
|
|
|
|
|
class SubBytes(bytes):
|
|
|
|
"""A subclass of a shareable type."""
|
|
|
|
|
|
|
|
not_shareables = [
|
|
|
|
# singletons
|
|
|
|
NotImplemented,
|
|
|
|
...,
|
|
|
|
# builtin types and objects
|
|
|
|
type,
|
|
|
|
object,
|
|
|
|
object(),
|
|
|
|
Exception(),
|
|
|
|
# user-defined types and objects
|
|
|
|
Cheese,
|
|
|
|
Cheese('Wensleydale'),
|
|
|
|
SubBytes(b'spam'),
|
|
|
|
]
|
|
|
|
for obj in not_shareables:
|
2018-05-16 16:04:57 -03:00
|
|
|
with self.subTest(repr(obj)):
|
2018-01-29 21:23:44 -04:00
|
|
|
self.assertFalse(
|
|
|
|
interpreters.is_shareable(obj))
|
|
|
|
|
|
|
|
|
2018-05-16 16:04:57 -03:00
|
|
|
class ShareableTypeTests(unittest.TestCase):
|
|
|
|
|
|
|
|
def _assert_values(self, values):
|
|
|
|
for obj in values:
|
|
|
|
with self.subTest(obj):
|
2023-10-30 19:53:10 -03:00
|
|
|
xid = _testinternalcapi.get_crossinterp_data(obj)
|
|
|
|
got = _testinternalcapi.restore_crossinterp_data(xid)
|
2018-05-16 16:04:57 -03:00
|
|
|
|
|
|
|
self.assertEqual(got, obj)
|
|
|
|
self.assertIs(type(got), type(obj))
|
|
|
|
|
|
|
|
def test_singletons(self):
|
|
|
|
for obj in [None]:
|
|
|
|
with self.subTest(obj):
|
2023-10-30 19:53:10 -03:00
|
|
|
xid = _testinternalcapi.get_crossinterp_data(obj)
|
|
|
|
got = _testinternalcapi.restore_crossinterp_data(xid)
|
2018-05-16 16:04:57 -03:00
|
|
|
|
|
|
|
# XXX What about between interpreters?
|
|
|
|
self.assertIs(got, obj)
|
|
|
|
|
|
|
|
def test_types(self):
|
|
|
|
self._assert_values([
|
|
|
|
b'spam',
|
|
|
|
9999,
|
|
|
|
])
|
|
|
|
|
|
|
|
def test_bytes(self):
|
|
|
|
self._assert_values(i.to_bytes(2, 'little', signed=True)
|
|
|
|
for i in range(-1, 258))
|
2018-01-29 21:23:44 -04:00
|
|
|
|
2020-06-13 09:26:01 -03:00
|
|
|
def test_strs(self):
|
|
|
|
self._assert_values(['hello world', '你好世界', ''])
|
|
|
|
|
2018-05-16 16:04:57 -03:00
|
|
|
def test_int(self):
|
2019-02-12 12:06:43 -04:00
|
|
|
self._assert_values(itertools.chain(range(-1, 258),
|
|
|
|
[sys.maxsize, -sys.maxsize - 1]))
|
|
|
|
|
|
|
|
def test_non_shareable_int(self):
|
|
|
|
ints = [
|
|
|
|
sys.maxsize + 1,
|
|
|
|
-sys.maxsize - 2,
|
|
|
|
2**1000,
|
|
|
|
]
|
|
|
|
for i in ints:
|
|
|
|
with self.subTest(i):
|
|
|
|
with self.assertRaises(OverflowError):
|
2023-10-30 19:53:10 -03:00
|
|
|
_testinternalcapi.get_crossinterp_data(i)
|
2018-05-16 16:04:57 -03:00
|
|
|
|
2023-11-01 21:09:01 -03:00
|
|
|
def test_bool(self):
|
|
|
|
self._assert_values([True, False])
|
|
|
|
|
2023-10-31 11:17:20 -03:00
|
|
|
def test_float(self):
|
|
|
|
self._assert_values([0.0, 1.1, -1.0, 0.12345678, -0.12345678])
|
|
|
|
|
2023-11-07 13:58:29 -04:00
|
|
|
def test_tuple(self):
|
|
|
|
self._assert_values([(), (1,), ("hello", "world", ), (1, True, "hello")])
|
|
|
|
# Test nesting
|
|
|
|
self._assert_values([
|
|
|
|
((1,),),
|
|
|
|
((1, 2), (3, 4)),
|
|
|
|
((1, 2), (3, 4), (5, 6)),
|
|
|
|
])
|
|
|
|
|
|
|
|
def test_tuples_containing_non_shareable_types(self):
|
|
|
|
non_shareables = [
|
|
|
|
Exception(),
|
|
|
|
object(),
|
|
|
|
]
|
|
|
|
for s in non_shareables:
|
|
|
|
value = tuple([0, 1.0, s])
|
|
|
|
with self.subTest(repr(value)):
|
|
|
|
# XXX Assert the NotShareableError when it is exported
|
|
|
|
with self.assertRaises(ValueError):
|
|
|
|
_testinternalcapi.get_crossinterp_data(value)
|
|
|
|
# Check nested as well
|
|
|
|
value = tuple([0, 1., (s,)])
|
|
|
|
with self.subTest("nested " + repr(value)):
|
|
|
|
# XXX Assert the NotShareableError when it is exported
|
|
|
|
with self.assertRaises(ValueError):
|
|
|
|
_testinternalcapi.get_crossinterp_data(value)
|
|
|
|
|
2018-05-16 16:04:57 -03:00
|
|
|
|
2022-12-05 16:40:20 -04:00
|
|
|
class ModuleTests(TestBase):
|
|
|
|
|
|
|
|
def test_import_in_interpreter(self):
|
|
|
|
_run_output(
|
|
|
|
interpreters.create(),
|
|
|
|
'import _xxsubinterpreters as _interpreters',
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2018-05-16 16:04:57 -03:00
|
|
|
##################################
|
|
|
|
# interpreter tests
|
2018-01-29 21:23:44 -04:00
|
|
|
|
|
|
|
class ListAllTests(TestBase):
|
|
|
|
|
|
|
|
def test_initial(self):
|
|
|
|
main = interpreters.get_main()
|
|
|
|
ids = interpreters.list_all()
|
|
|
|
self.assertEqual(ids, [main])
|
|
|
|
|
|
|
|
def test_after_creating(self):
|
|
|
|
main = interpreters.get_main()
|
|
|
|
first = interpreters.create()
|
|
|
|
second = interpreters.create()
|
|
|
|
ids = interpreters.list_all()
|
|
|
|
self.assertEqual(ids, [main, first, second])
|
|
|
|
|
|
|
|
def test_after_destroying(self):
|
|
|
|
main = interpreters.get_main()
|
|
|
|
first = interpreters.create()
|
|
|
|
second = interpreters.create()
|
|
|
|
interpreters.destroy(first)
|
|
|
|
ids = interpreters.list_all()
|
|
|
|
self.assertEqual(ids, [main, second])
|
|
|
|
|
|
|
|
|
|
|
|
class GetCurrentTests(TestBase):
|
|
|
|
|
|
|
|
def test_main(self):
|
|
|
|
main = interpreters.get_main()
|
|
|
|
cur = interpreters.get_current()
|
|
|
|
self.assertEqual(cur, main)
|
2023-12-12 11:24:31 -04:00
|
|
|
self.assertIsInstance(cur, int)
|
2018-01-29 21:23:44 -04:00
|
|
|
|
|
|
|
def test_subinterpreter(self):
|
|
|
|
main = interpreters.get_main()
|
|
|
|
interp = interpreters.create()
|
|
|
|
out = _run_output(interp, dedent("""
|
|
|
|
import _xxsubinterpreters as _interpreters
|
2018-05-16 16:04:57 -03:00
|
|
|
cur = _interpreters.get_current()
|
|
|
|
print(cur)
|
2023-12-12 11:24:31 -04:00
|
|
|
assert isinstance(cur, int)
|
2018-01-29 21:23:44 -04:00
|
|
|
"""))
|
|
|
|
cur = int(out.strip())
|
|
|
|
_, expected = interpreters.list_all()
|
|
|
|
self.assertEqual(cur, expected)
|
|
|
|
self.assertNotEqual(cur, main)
|
|
|
|
|
|
|
|
|
|
|
|
class GetMainTests(TestBase):
|
|
|
|
|
|
|
|
def test_from_main(self):
|
|
|
|
[expected] = interpreters.list_all()
|
|
|
|
main = interpreters.get_main()
|
|
|
|
self.assertEqual(main, expected)
|
2023-12-12 11:24:31 -04:00
|
|
|
self.assertIsInstance(main, int)
|
2018-01-29 21:23:44 -04:00
|
|
|
|
|
|
|
def test_from_subinterpreter(self):
|
|
|
|
[expected] = interpreters.list_all()
|
|
|
|
interp = interpreters.create()
|
|
|
|
out = _run_output(interp, dedent("""
|
|
|
|
import _xxsubinterpreters as _interpreters
|
2018-05-16 16:04:57 -03:00
|
|
|
main = _interpreters.get_main()
|
|
|
|
print(main)
|
2023-12-12 11:24:31 -04:00
|
|
|
assert isinstance(main, int)
|
2018-01-29 21:23:44 -04:00
|
|
|
"""))
|
|
|
|
main = int(out.strip())
|
|
|
|
self.assertEqual(main, expected)
|
|
|
|
|
|
|
|
|
|
|
|
class IsRunningTests(TestBase):
|
|
|
|
|
|
|
|
def test_main(self):
|
|
|
|
main = interpreters.get_main()
|
|
|
|
self.assertTrue(interpreters.is_running(main))
|
|
|
|
|
2020-06-21 20:59:43 -03:00
|
|
|
@unittest.skip('Fails on FreeBSD')
|
2018-01-29 21:23:44 -04:00
|
|
|
def test_subinterpreter(self):
|
|
|
|
interp = interpreters.create()
|
|
|
|
self.assertFalse(interpreters.is_running(interp))
|
|
|
|
|
|
|
|
with _running(interp):
|
|
|
|
self.assertTrue(interpreters.is_running(interp))
|
|
|
|
self.assertFalse(interpreters.is_running(interp))
|
|
|
|
|
|
|
|
def test_from_subinterpreter(self):
|
|
|
|
interp = interpreters.create()
|
|
|
|
out = _run_output(interp, dedent(f"""
|
|
|
|
import _xxsubinterpreters as _interpreters
|
2018-05-16 16:04:57 -03:00
|
|
|
if _interpreters.is_running({interp}):
|
2018-01-29 21:23:44 -04:00
|
|
|
print(True)
|
|
|
|
else:
|
|
|
|
print(False)
|
|
|
|
"""))
|
|
|
|
self.assertEqual(out.strip(), 'True')
|
|
|
|
|
|
|
|
def test_already_destroyed(self):
|
|
|
|
interp = interpreters.create()
|
|
|
|
interpreters.destroy(interp)
|
2023-12-12 11:24:31 -04:00
|
|
|
with self.assertRaises(InterpreterNotFoundError):
|
2018-01-29 21:23:44 -04:00
|
|
|
interpreters.is_running(interp)
|
|
|
|
|
|
|
|
def test_does_not_exist(self):
|
2023-12-12 11:24:31 -04:00
|
|
|
with self.assertRaises(InterpreterNotFoundError):
|
2018-01-29 21:23:44 -04:00
|
|
|
interpreters.is_running(1_000_000)
|
|
|
|
|
|
|
|
def test_bad_id(self):
|
2019-09-25 12:35:57 -03:00
|
|
|
with self.assertRaises(ValueError):
|
2018-01-29 21:23:44 -04:00
|
|
|
interpreters.is_running(-1)
|
|
|
|
|
|
|
|
|
|
|
|
class CreateTests(TestBase):
|
|
|
|
|
|
|
|
def test_in_main(self):
|
|
|
|
id = interpreters.create()
|
2023-12-12 11:24:31 -04:00
|
|
|
self.assertIsInstance(id, int)
|
2018-01-29 21:23:44 -04:00
|
|
|
|
|
|
|
self.assertIn(id, interpreters.list_all())
|
|
|
|
|
|
|
|
@unittest.skip('enable this test when working on pystate.c')
|
|
|
|
def test_unique_id(self):
|
|
|
|
seen = set()
|
|
|
|
for _ in range(100):
|
|
|
|
id = interpreters.create()
|
|
|
|
interpreters.destroy(id)
|
|
|
|
seen.add(id)
|
|
|
|
|
|
|
|
self.assertEqual(len(seen), 100)
|
|
|
|
|
|
|
|
def test_in_thread(self):
|
|
|
|
lock = threading.Lock()
|
|
|
|
id = None
|
|
|
|
def f():
|
|
|
|
nonlocal id
|
|
|
|
id = interpreters.create()
|
|
|
|
lock.acquire()
|
|
|
|
lock.release()
|
|
|
|
|
|
|
|
t = threading.Thread(target=f)
|
|
|
|
with lock:
|
|
|
|
t.start()
|
|
|
|
t.join()
|
|
|
|
self.assertIn(id, interpreters.list_all())
|
|
|
|
|
|
|
|
def test_in_subinterpreter(self):
|
|
|
|
main, = interpreters.list_all()
|
|
|
|
id1 = interpreters.create()
|
|
|
|
out = _run_output(id1, dedent("""
|
|
|
|
import _xxsubinterpreters as _interpreters
|
|
|
|
id = _interpreters.create()
|
2018-05-16 16:04:57 -03:00
|
|
|
print(id)
|
2023-12-12 11:24:31 -04:00
|
|
|
assert isinstance(id, int)
|
2018-01-29 21:23:44 -04:00
|
|
|
"""))
|
|
|
|
id2 = int(out.strip())
|
|
|
|
|
|
|
|
self.assertEqual(set(interpreters.list_all()), {main, id1, id2})
|
|
|
|
|
|
|
|
def test_in_threaded_subinterpreter(self):
|
|
|
|
main, = interpreters.list_all()
|
|
|
|
id1 = interpreters.create()
|
|
|
|
id2 = None
|
|
|
|
def f():
|
|
|
|
nonlocal id2
|
|
|
|
out = _run_output(id1, dedent("""
|
|
|
|
import _xxsubinterpreters as _interpreters
|
|
|
|
id = _interpreters.create()
|
2018-05-16 16:04:57 -03:00
|
|
|
print(id)
|
2018-01-29 21:23:44 -04:00
|
|
|
"""))
|
|
|
|
id2 = int(out.strip())
|
|
|
|
|
|
|
|
t = threading.Thread(target=f)
|
|
|
|
t.start()
|
|
|
|
t.join()
|
|
|
|
|
|
|
|
self.assertEqual(set(interpreters.list_all()), {main, id1, id2})
|
|
|
|
|
|
|
|
def test_after_destroy_all(self):
|
|
|
|
before = set(interpreters.list_all())
|
|
|
|
# Create 3 subinterpreters.
|
|
|
|
ids = []
|
|
|
|
for _ in range(3):
|
|
|
|
id = interpreters.create()
|
|
|
|
ids.append(id)
|
|
|
|
# Now destroy them.
|
|
|
|
for id in ids:
|
|
|
|
interpreters.destroy(id)
|
|
|
|
# Finally, create another.
|
|
|
|
id = interpreters.create()
|
|
|
|
self.assertEqual(set(interpreters.list_all()), before | {id})
|
|
|
|
|
|
|
|
def test_after_destroy_some(self):
|
|
|
|
before = set(interpreters.list_all())
|
|
|
|
# Create 3 subinterpreters.
|
|
|
|
id1 = interpreters.create()
|
|
|
|
id2 = interpreters.create()
|
|
|
|
id3 = interpreters.create()
|
|
|
|
# Now destroy 2 of them.
|
|
|
|
interpreters.destroy(id1)
|
|
|
|
interpreters.destroy(id3)
|
|
|
|
# Finally, create another.
|
|
|
|
id = interpreters.create()
|
|
|
|
self.assertEqual(set(interpreters.list_all()), before | {id, id2})
|
|
|
|
|
|
|
|
|
|
|
|
class DestroyTests(TestBase):
|
|
|
|
|
|
|
|
def test_one(self):
|
|
|
|
id1 = interpreters.create()
|
|
|
|
id2 = interpreters.create()
|
|
|
|
id3 = interpreters.create()
|
|
|
|
self.assertIn(id2, interpreters.list_all())
|
|
|
|
interpreters.destroy(id2)
|
|
|
|
self.assertNotIn(id2, interpreters.list_all())
|
|
|
|
self.assertIn(id1, interpreters.list_all())
|
|
|
|
self.assertIn(id3, interpreters.list_all())
|
|
|
|
|
|
|
|
def test_all(self):
|
|
|
|
before = set(interpreters.list_all())
|
|
|
|
ids = set()
|
|
|
|
for _ in range(3):
|
|
|
|
id = interpreters.create()
|
|
|
|
ids.add(id)
|
|
|
|
self.assertEqual(set(interpreters.list_all()), before | ids)
|
|
|
|
for id in ids:
|
|
|
|
interpreters.destroy(id)
|
|
|
|
self.assertEqual(set(interpreters.list_all()), before)
|
|
|
|
|
|
|
|
def test_main(self):
|
|
|
|
main, = interpreters.list_all()
|
2024-04-03 13:58:39 -03:00
|
|
|
with self.assertRaises(interpreters.InterpreterError):
|
2018-01-29 21:23:44 -04:00
|
|
|
interpreters.destroy(main)
|
|
|
|
|
|
|
|
def f():
|
2024-04-03 13:58:39 -03:00
|
|
|
with self.assertRaises(interpreters.InterpreterError):
|
2018-01-29 21:23:44 -04:00
|
|
|
interpreters.destroy(main)
|
|
|
|
|
|
|
|
t = threading.Thread(target=f)
|
|
|
|
t.start()
|
|
|
|
t.join()
|
|
|
|
|
|
|
|
def test_already_destroyed(self):
|
|
|
|
id = interpreters.create()
|
|
|
|
interpreters.destroy(id)
|
2023-12-12 11:24:31 -04:00
|
|
|
with self.assertRaises(InterpreterNotFoundError):
|
2018-01-29 21:23:44 -04:00
|
|
|
interpreters.destroy(id)
|
|
|
|
|
|
|
|
def test_does_not_exist(self):
|
2023-12-12 11:24:31 -04:00
|
|
|
with self.assertRaises(InterpreterNotFoundError):
|
2018-01-29 21:23:44 -04:00
|
|
|
interpreters.destroy(1_000_000)
|
|
|
|
|
|
|
|
def test_bad_id(self):
|
2019-09-25 12:35:57 -03:00
|
|
|
with self.assertRaises(ValueError):
|
2018-01-29 21:23:44 -04:00
|
|
|
interpreters.destroy(-1)
|
|
|
|
|
|
|
|
def test_from_current(self):
|
|
|
|
main, = interpreters.list_all()
|
|
|
|
id = interpreters.create()
|
2018-02-03 00:49:49 -04:00
|
|
|
script = dedent(f"""
|
2018-01-29 21:23:44 -04:00
|
|
|
import _xxsubinterpreters as _interpreters
|
2018-02-03 00:49:49 -04:00
|
|
|
try:
|
2018-05-16 16:04:57 -03:00
|
|
|
_interpreters.destroy({id})
|
2024-04-03 13:58:39 -03:00
|
|
|
except interpreters.InterpreterError:
|
2018-02-03 00:49:49 -04:00
|
|
|
pass
|
|
|
|
""")
|
2018-01-29 21:23:44 -04:00
|
|
|
|
2018-02-03 00:49:49 -04:00
|
|
|
interpreters.run_string(id, script)
|
2018-01-29 21:23:44 -04:00
|
|
|
self.assertEqual(set(interpreters.list_all()), {main, id})
|
|
|
|
|
|
|
|
def test_from_sibling(self):
|
|
|
|
main, = interpreters.list_all()
|
|
|
|
id1 = interpreters.create()
|
|
|
|
id2 = interpreters.create()
|
2018-02-16 21:53:40 -04:00
|
|
|
script = dedent(f"""
|
2018-01-29 21:23:44 -04:00
|
|
|
import _xxsubinterpreters as _interpreters
|
2018-05-16 16:04:57 -03:00
|
|
|
_interpreters.destroy({id2})
|
2018-02-16 21:53:40 -04:00
|
|
|
""")
|
2018-01-29 21:23:44 -04:00
|
|
|
interpreters.run_string(id1, script)
|
|
|
|
|
|
|
|
self.assertEqual(set(interpreters.list_all()), {main, id1})
|
|
|
|
|
|
|
|
def test_from_other_thread(self):
|
|
|
|
id = interpreters.create()
|
|
|
|
def f():
|
|
|
|
interpreters.destroy(id)
|
|
|
|
|
|
|
|
t = threading.Thread(target=f)
|
|
|
|
t.start()
|
|
|
|
t.join()
|
|
|
|
|
|
|
|
def test_still_running(self):
|
|
|
|
main, = interpreters.list_all()
|
|
|
|
interp = interpreters.create()
|
|
|
|
with _running(interp):
|
2020-01-31 16:07:09 -04:00
|
|
|
self.assertTrue(interpreters.is_running(interp),
|
|
|
|
msg=f"Interp {interp} should be running before destruction.")
|
|
|
|
|
2024-04-03 13:58:39 -03:00
|
|
|
with self.assertRaises(interpreters.InterpreterError,
|
2020-01-31 16:07:09 -04:00
|
|
|
msg=f"Should not be able to destroy interp {interp} while it's still running."):
|
2018-01-29 21:23:44 -04:00
|
|
|
interpreters.destroy(interp)
|
|
|
|
self.assertTrue(interpreters.is_running(interp))
|
|
|
|
|
|
|
|
|
|
|
|
class RunStringTests(TestBase):
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
super().setUp()
|
|
|
|
self.id = interpreters.create()
|
|
|
|
|
|
|
|
def test_success(self):
|
|
|
|
script, file = _captured_script('print("it worked!", end="")')
|
|
|
|
with file:
|
|
|
|
interpreters.run_string(self.id, script)
|
|
|
|
out = file.read()
|
|
|
|
|
|
|
|
self.assertEqual(out, 'it worked!')
|
|
|
|
|
|
|
|
def test_in_thread(self):
|
|
|
|
script, file = _captured_script('print("it worked!", end="")')
|
|
|
|
with file:
|
|
|
|
def f():
|
|
|
|
interpreters.run_string(self.id, script)
|
|
|
|
|
|
|
|
t = threading.Thread(target=f)
|
|
|
|
t.start()
|
|
|
|
t.join()
|
|
|
|
out = file.read()
|
|
|
|
|
|
|
|
self.assertEqual(out, 'it worked!')
|
|
|
|
|
|
|
|
def test_create_thread(self):
|
gh-98610: Adjust the Optional Restrictions on Subinterpreters (GH-98618)
Previously, the optional restrictions on subinterpreters were: disallow fork, subprocess, and threads. By default, we were disallowing all three for "isolated" interpreters. We always allowed all three for the main interpreter and those created through the legacy `Py_NewInterpreter()` API.
Those settings were a bit conservative, so here we've adjusted the optional restrictions to: fork, exec, threads, and daemon threads. The default for "isolated" interpreters disables fork, exec, and daemon threads. Regular threads are allowed by default. We continue always allowing everything For the main interpreter and the legacy API.
In the code, we add `_PyInterpreterConfig.allow_exec` and `_PyInterpreterConfig.allow_daemon_threads`. We also add `Py_RTFLAGS_DAEMON_THREADS` and `Py_RTFLAGS_EXEC`.
2022-10-31 16:35:54 -03:00
|
|
|
subinterp = interpreters.create()
|
2018-01-29 21:23:44 -04:00
|
|
|
script, file = _captured_script("""
|
|
|
|
import threading
|
|
|
|
def f():
|
|
|
|
print('it worked!', end='')
|
|
|
|
|
|
|
|
t = threading.Thread(target=f)
|
|
|
|
t.start()
|
|
|
|
t.join()
|
|
|
|
""")
|
|
|
|
with file:
|
2020-05-01 06:33:44 -03:00
|
|
|
interpreters.run_string(subinterp, script)
|
2018-01-29 21:23:44 -04:00
|
|
|
out = file.read()
|
|
|
|
|
|
|
|
self.assertEqual(out, 'it worked!')
|
|
|
|
|
gh-98610: Adjust the Optional Restrictions on Subinterpreters (GH-98618)
Previously, the optional restrictions on subinterpreters were: disallow fork, subprocess, and threads. By default, we were disallowing all three for "isolated" interpreters. We always allowed all three for the main interpreter and those created through the legacy `Py_NewInterpreter()` API.
Those settings were a bit conservative, so here we've adjusted the optional restrictions to: fork, exec, threads, and daemon threads. The default for "isolated" interpreters disables fork, exec, and daemon threads. Regular threads are allowed by default. We continue always allowing everything For the main interpreter and the legacy API.
In the code, we add `_PyInterpreterConfig.allow_exec` and `_PyInterpreterConfig.allow_daemon_threads`. We also add `Py_RTFLAGS_DAEMON_THREADS` and `Py_RTFLAGS_EXEC`.
2022-10-31 16:35:54 -03:00
|
|
|
def test_create_daemon_thread(self):
|
|
|
|
with self.subTest('isolated'):
|
|
|
|
expected = 'spam spam spam spam spam'
|
2024-04-02 20:16:50 -03:00
|
|
|
subinterp = interpreters.create('isolated')
|
gh-98610: Adjust the Optional Restrictions on Subinterpreters (GH-98618)
Previously, the optional restrictions on subinterpreters were: disallow fork, subprocess, and threads. By default, we were disallowing all three for "isolated" interpreters. We always allowed all three for the main interpreter and those created through the legacy `Py_NewInterpreter()` API.
Those settings were a bit conservative, so here we've adjusted the optional restrictions to: fork, exec, threads, and daemon threads. The default for "isolated" interpreters disables fork, exec, and daemon threads. Regular threads are allowed by default. We continue always allowing everything For the main interpreter and the legacy API.
In the code, we add `_PyInterpreterConfig.allow_exec` and `_PyInterpreterConfig.allow_daemon_threads`. We also add `Py_RTFLAGS_DAEMON_THREADS` and `Py_RTFLAGS_EXEC`.
2022-10-31 16:35:54 -03:00
|
|
|
script, file = _captured_script(f"""
|
|
|
|
import threading
|
|
|
|
def f():
|
|
|
|
print('it worked!', end='')
|
|
|
|
|
|
|
|
try:
|
|
|
|
t = threading.Thread(target=f, daemon=True)
|
|
|
|
t.start()
|
|
|
|
t.join()
|
|
|
|
except RuntimeError:
|
|
|
|
print('{expected}', end='')
|
|
|
|
""")
|
|
|
|
with file:
|
|
|
|
interpreters.run_string(subinterp, script)
|
|
|
|
out = file.read()
|
|
|
|
|
|
|
|
self.assertEqual(out, expected)
|
|
|
|
|
|
|
|
with self.subTest('not isolated'):
|
2024-04-02 20:16:50 -03:00
|
|
|
subinterp = interpreters.create('legacy')
|
gh-98610: Adjust the Optional Restrictions on Subinterpreters (GH-98618)
Previously, the optional restrictions on subinterpreters were: disallow fork, subprocess, and threads. By default, we were disallowing all three for "isolated" interpreters. We always allowed all three for the main interpreter and those created through the legacy `Py_NewInterpreter()` API.
Those settings were a bit conservative, so here we've adjusted the optional restrictions to: fork, exec, threads, and daemon threads. The default for "isolated" interpreters disables fork, exec, and daemon threads. Regular threads are allowed by default. We continue always allowing everything For the main interpreter and the legacy API.
In the code, we add `_PyInterpreterConfig.allow_exec` and `_PyInterpreterConfig.allow_daemon_threads`. We also add `Py_RTFLAGS_DAEMON_THREADS` and `Py_RTFLAGS_EXEC`.
2022-10-31 16:35:54 -03:00
|
|
|
script, file = _captured_script("""
|
|
|
|
import threading
|
|
|
|
def f():
|
|
|
|
print('it worked!', end='')
|
|
|
|
|
|
|
|
t = threading.Thread(target=f, daemon=True)
|
|
|
|
t.start()
|
|
|
|
t.join()
|
|
|
|
""")
|
|
|
|
with file:
|
|
|
|
interpreters.run_string(subinterp, script)
|
|
|
|
out = file.read()
|
|
|
|
|
|
|
|
self.assertEqual(out, 'it worked!')
|
|
|
|
|
2023-02-03 21:14:43 -04:00
|
|
|
def test_shareable_types(self):
|
|
|
|
interp = interpreters.create()
|
|
|
|
objects = [
|
|
|
|
None,
|
|
|
|
'spam',
|
|
|
|
b'spam',
|
|
|
|
42,
|
|
|
|
]
|
|
|
|
for obj in objects:
|
|
|
|
with self.subTest(obj):
|
2023-12-12 14:06:06 -04:00
|
|
|
interpreters.set___main___attrs(interp, dict(obj=obj))
|
2023-02-03 21:14:43 -04:00
|
|
|
interpreters.run_string(
|
|
|
|
interp,
|
|
|
|
f'assert(obj == {obj!r})',
|
|
|
|
)
|
|
|
|
|
gh-98610: Adjust the Optional Restrictions on Subinterpreters (GH-98618)
Previously, the optional restrictions on subinterpreters were: disallow fork, subprocess, and threads. By default, we were disallowing all three for "isolated" interpreters. We always allowed all three for the main interpreter and those created through the legacy `Py_NewInterpreter()` API.
Those settings were a bit conservative, so here we've adjusted the optional restrictions to: fork, exec, threads, and daemon threads. The default for "isolated" interpreters disables fork, exec, and daemon threads. Regular threads are allowed by default. We continue always allowing everything For the main interpreter and the legacy API.
In the code, we add `_PyInterpreterConfig.allow_exec` and `_PyInterpreterConfig.allow_daemon_threads`. We also add `Py_RTFLAGS_DAEMON_THREADS` and `Py_RTFLAGS_EXEC`.
2022-10-31 16:35:54 -03:00
|
|
|
def test_os_exec(self):
|
|
|
|
expected = 'spam spam spam spam spam'
|
|
|
|
subinterp = interpreters.create()
|
|
|
|
script, file = _captured_script(f"""
|
|
|
|
import os, sys
|
|
|
|
try:
|
|
|
|
os.execl(sys.executable)
|
|
|
|
except RuntimeError:
|
|
|
|
print('{expected}', end='')
|
|
|
|
""")
|
|
|
|
with file:
|
|
|
|
interpreters.run_string(subinterp, script)
|
|
|
|
out = file.read()
|
|
|
|
|
|
|
|
self.assertEqual(out, expected)
|
|
|
|
|
2022-03-17 08:09:57 -03:00
|
|
|
@support.requires_fork()
|
2018-01-29 21:23:44 -04:00
|
|
|
def test_fork(self):
|
|
|
|
import tempfile
|
2021-04-02 00:53:46 -03:00
|
|
|
with tempfile.NamedTemporaryFile('w+', encoding="utf-8") as file:
|
2018-01-29 21:23:44 -04:00
|
|
|
file.write('')
|
|
|
|
file.flush()
|
|
|
|
|
|
|
|
expected = 'spam spam spam spam spam'
|
|
|
|
script = dedent(f"""
|
|
|
|
import os
|
2018-09-14 18:17:20 -03:00
|
|
|
try:
|
|
|
|
os.fork()
|
|
|
|
except RuntimeError:
|
2021-04-02 00:53:46 -03:00
|
|
|
with open('{file.name}', 'w', encoding='utf-8') as out:
|
2018-01-29 21:23:44 -04:00
|
|
|
out.write('{expected}')
|
|
|
|
""")
|
|
|
|
interpreters.run_string(self.id, script)
|
|
|
|
|
|
|
|
file.seek(0)
|
|
|
|
content = file.read()
|
|
|
|
self.assertEqual(content, expected)
|
|
|
|
|
|
|
|
def test_already_running(self):
|
|
|
|
with _running(self.id):
|
2024-04-03 13:58:39 -03:00
|
|
|
with self.assertRaises(interpreters.InterpreterError):
|
2018-01-29 21:23:44 -04:00
|
|
|
interpreters.run_string(self.id, 'print("spam")')
|
|
|
|
|
|
|
|
def test_does_not_exist(self):
|
|
|
|
id = 0
|
|
|
|
while id in interpreters.list_all():
|
|
|
|
id += 1
|
2023-12-12 11:24:31 -04:00
|
|
|
with self.assertRaises(InterpreterNotFoundError):
|
2018-01-29 21:23:44 -04:00
|
|
|
interpreters.run_string(id, 'print("spam")')
|
|
|
|
|
|
|
|
def test_error_id(self):
|
2019-09-25 12:35:57 -03:00
|
|
|
with self.assertRaises(ValueError):
|
2018-01-29 21:23:44 -04:00
|
|
|
interpreters.run_string(-1, 'print("spam")')
|
|
|
|
|
|
|
|
def test_bad_id(self):
|
|
|
|
with self.assertRaises(TypeError):
|
|
|
|
interpreters.run_string('spam', 'print("spam")')
|
|
|
|
|
|
|
|
def test_bad_script(self):
|
|
|
|
with self.assertRaises(TypeError):
|
|
|
|
interpreters.run_string(self.id, 10)
|
|
|
|
|
|
|
|
def test_bytes_for_script(self):
|
|
|
|
with self.assertRaises(TypeError):
|
|
|
|
interpreters.run_string(self.id, b'print("spam")')
|
|
|
|
|
|
|
|
def test_with_shared(self):
|
|
|
|
r, w = os.pipe()
|
|
|
|
|
|
|
|
shared = {
|
|
|
|
'spam': b'ham',
|
|
|
|
'eggs': b'-1',
|
|
|
|
'cheddar': None,
|
|
|
|
}
|
|
|
|
script = dedent(f"""
|
|
|
|
eggs = int(eggs)
|
|
|
|
spam = 42
|
|
|
|
result = spam + eggs
|
|
|
|
|
|
|
|
ns = dict(vars())
|
|
|
|
del ns['__builtins__']
|
|
|
|
import pickle
|
|
|
|
with open({w}, 'wb') as chan:
|
|
|
|
pickle.dump(ns, chan)
|
|
|
|
""")
|
2023-12-12 14:06:06 -04:00
|
|
|
interpreters.set___main___attrs(self.id, shared)
|
|
|
|
interpreters.run_string(self.id, script)
|
2018-01-29 21:23:44 -04:00
|
|
|
with open(r, 'rb') as chan:
|
|
|
|
ns = pickle.load(chan)
|
|
|
|
|
|
|
|
self.assertEqual(ns['spam'], 42)
|
|
|
|
self.assertEqual(ns['eggs'], -1)
|
|
|
|
self.assertEqual(ns['result'], 41)
|
|
|
|
self.assertIsNone(ns['cheddar'])
|
|
|
|
|
|
|
|
def test_shared_overwrites(self):
|
|
|
|
interpreters.run_string(self.id, dedent("""
|
|
|
|
spam = 'eggs'
|
|
|
|
ns1 = dict(vars())
|
|
|
|
del ns1['__builtins__']
|
|
|
|
"""))
|
|
|
|
|
|
|
|
shared = {'spam': b'ham'}
|
2023-04-24 20:24:49 -03:00
|
|
|
script = dedent("""
|
2018-01-29 21:23:44 -04:00
|
|
|
ns2 = dict(vars())
|
|
|
|
del ns2['__builtins__']
|
|
|
|
""")
|
2023-12-12 14:06:06 -04:00
|
|
|
interpreters.set___main___attrs(self.id, shared)
|
|
|
|
interpreters.run_string(self.id, script)
|
2018-01-29 21:23:44 -04:00
|
|
|
|
|
|
|
r, w = os.pipe()
|
|
|
|
script = dedent(f"""
|
|
|
|
ns = dict(vars())
|
|
|
|
del ns['__builtins__']
|
|
|
|
import pickle
|
|
|
|
with open({w}, 'wb') as chan:
|
|
|
|
pickle.dump(ns, chan)
|
|
|
|
""")
|
|
|
|
interpreters.run_string(self.id, script)
|
|
|
|
with open(r, 'rb') as chan:
|
|
|
|
ns = pickle.load(chan)
|
|
|
|
|
|
|
|
self.assertEqual(ns['ns1']['spam'], 'eggs')
|
|
|
|
self.assertEqual(ns['ns2']['spam'], b'ham')
|
|
|
|
self.assertEqual(ns['spam'], b'ham')
|
|
|
|
|
|
|
|
def test_shared_overwrites_default_vars(self):
|
|
|
|
r, w = os.pipe()
|
|
|
|
|
|
|
|
shared = {'__name__': b'not __main__'}
|
|
|
|
script = dedent(f"""
|
|
|
|
spam = 42
|
|
|
|
|
|
|
|
ns = dict(vars())
|
|
|
|
del ns['__builtins__']
|
|
|
|
import pickle
|
|
|
|
with open({w}, 'wb') as chan:
|
|
|
|
pickle.dump(ns, chan)
|
|
|
|
""")
|
2023-12-12 14:06:06 -04:00
|
|
|
interpreters.set___main___attrs(self.id, shared)
|
|
|
|
interpreters.run_string(self.id, script)
|
2018-01-29 21:23:44 -04:00
|
|
|
with open(r, 'rb') as chan:
|
|
|
|
ns = pickle.load(chan)
|
|
|
|
|
|
|
|
self.assertEqual(ns['__name__'], b'not __main__')
|
|
|
|
|
|
|
|
def test_main_reused(self):
|
|
|
|
r, w = os.pipe()
|
|
|
|
interpreters.run_string(self.id, dedent(f"""
|
|
|
|
spam = True
|
|
|
|
|
|
|
|
ns = dict(vars())
|
|
|
|
del ns['__builtins__']
|
|
|
|
import pickle
|
|
|
|
with open({w}, 'wb') as chan:
|
|
|
|
pickle.dump(ns, chan)
|
|
|
|
del ns, pickle, chan
|
|
|
|
"""))
|
|
|
|
with open(r, 'rb') as chan:
|
|
|
|
ns1 = pickle.load(chan)
|
|
|
|
|
|
|
|
r, w = os.pipe()
|
|
|
|
interpreters.run_string(self.id, dedent(f"""
|
|
|
|
eggs = False
|
|
|
|
|
|
|
|
ns = dict(vars())
|
|
|
|
del ns['__builtins__']
|
|
|
|
import pickle
|
|
|
|
with open({w}, 'wb') as chan:
|
|
|
|
pickle.dump(ns, chan)
|
|
|
|
"""))
|
|
|
|
with open(r, 'rb') as chan:
|
|
|
|
ns2 = pickle.load(chan)
|
|
|
|
|
|
|
|
self.assertIn('spam', ns1)
|
|
|
|
self.assertNotIn('eggs', ns1)
|
|
|
|
self.assertIn('eggs', ns2)
|
|
|
|
self.assertIn('spam', ns2)
|
|
|
|
|
|
|
|
def test_execution_namespace_is_main(self):
|
|
|
|
r, w = os.pipe()
|
|
|
|
|
|
|
|
script = dedent(f"""
|
|
|
|
spam = 42
|
|
|
|
|
|
|
|
ns = dict(vars())
|
|
|
|
ns['__builtins__'] = str(ns['__builtins__'])
|
|
|
|
import pickle
|
|
|
|
with open({w}, 'wb') as chan:
|
|
|
|
pickle.dump(ns, chan)
|
|
|
|
""")
|
|
|
|
interpreters.run_string(self.id, script)
|
|
|
|
with open(r, 'rb') as chan:
|
|
|
|
ns = pickle.load(chan)
|
|
|
|
|
|
|
|
ns.pop('__builtins__')
|
|
|
|
ns.pop('__loader__')
|
|
|
|
self.assertEqual(ns, {
|
|
|
|
'__name__': '__main__',
|
|
|
|
'__annotations__': {},
|
|
|
|
'__doc__': None,
|
|
|
|
'__package__': None,
|
|
|
|
'__spec__': None,
|
|
|
|
'spam': 42,
|
|
|
|
})
|
|
|
|
|
2018-02-16 21:53:40 -04:00
|
|
|
# XXX Fix this test!
|
|
|
|
@unittest.skip('blocking forever')
|
2018-01-29 21:23:44 -04:00
|
|
|
def test_still_running_at_exit(self):
|
2023-04-24 20:24:49 -03:00
|
|
|
script = dedent("""
|
2018-01-29 21:23:44 -04:00
|
|
|
from textwrap import dedent
|
|
|
|
import threading
|
|
|
|
import _xxsubinterpreters as _interpreters
|
2018-02-16 21:53:40 -04:00
|
|
|
id = _interpreters.create()
|
2018-01-29 21:23:44 -04:00
|
|
|
def f():
|
|
|
|
_interpreters.run_string(id, dedent('''
|
|
|
|
import time
|
|
|
|
# Give plenty of time for the main interpreter to finish.
|
|
|
|
time.sleep(1_000_000)
|
|
|
|
'''))
|
|
|
|
|
|
|
|
t = threading.Thread(target=f)
|
|
|
|
t.start()
|
|
|
|
""")
|
|
|
|
with support.temp_dir() as dirname:
|
|
|
|
filename = script_helper.make_script(dirname, 'interp', script)
|
|
|
|
with script_helper.spawn_python(filename) as proc:
|
|
|
|
retcode = proc.wait()
|
|
|
|
|
|
|
|
self.assertEqual(retcode, 0)
|
|
|
|
|
|
|
|
|
2023-11-22 17:48:45 -04:00
|
|
|
class RunFailedTests(TestBase):
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
super().setUp()
|
|
|
|
self.id = interpreters.create()
|
|
|
|
|
|
|
|
def add_module(self, modname, text):
|
|
|
|
import tempfile
|
|
|
|
tempdir = tempfile.mkdtemp()
|
|
|
|
self.addCleanup(lambda: os_helper.rmtree(tempdir))
|
|
|
|
interpreters.run_string(self.id, dedent(f"""
|
|
|
|
import sys
|
|
|
|
sys.path.insert(0, {tempdir!r})
|
|
|
|
"""))
|
|
|
|
return script_helper.make_script(tempdir, modname, text)
|
|
|
|
|
|
|
|
def run_script(self, text, *, fails=False):
|
|
|
|
r, w = os.pipe()
|
|
|
|
try:
|
|
|
|
script = dedent(f"""
|
|
|
|
import os, sys
|
|
|
|
os.write({w}, b'0')
|
|
|
|
|
|
|
|
# This raises an exception:
|
|
|
|
{{}}
|
|
|
|
|
|
|
|
# Nothing from here down should ever run.
|
|
|
|
os.write({w}, b'1')
|
|
|
|
class NeverError(Exception): pass
|
|
|
|
raise NeverError # never raised
|
|
|
|
""").format(dedent(text))
|
|
|
|
if fails:
|
2023-11-22 20:55:00 -04:00
|
|
|
err = interpreters.run_string(self.id, script)
|
|
|
|
self.assertIsNot(err, None)
|
|
|
|
return err
|
2023-11-22 17:48:45 -04:00
|
|
|
else:
|
2023-11-22 20:55:00 -04:00
|
|
|
err = interpreters.run_string(self.id, script)
|
|
|
|
self.assertIs(err, None)
|
2023-11-22 17:48:45 -04:00
|
|
|
return None
|
|
|
|
except:
|
|
|
|
raise # re-raise
|
|
|
|
else:
|
|
|
|
msg = os.read(r, 100)
|
|
|
|
self.assertEqual(msg, b'0')
|
|
|
|
finally:
|
|
|
|
os.close(r)
|
|
|
|
os.close(w)
|
|
|
|
|
|
|
|
def _assert_run_failed(self, exctype, msg, script):
|
|
|
|
if isinstance(exctype, str):
|
|
|
|
exctype_name = exctype
|
|
|
|
exctype = None
|
|
|
|
else:
|
|
|
|
exctype_name = exctype.__name__
|
|
|
|
|
|
|
|
# Run the script.
|
2023-11-22 20:55:00 -04:00
|
|
|
excinfo = self.run_script(script, fails=True)
|
2023-11-22 17:48:45 -04:00
|
|
|
|
|
|
|
# Check the wrapper exception.
|
2023-11-22 20:55:00 -04:00
|
|
|
self.assertEqual(excinfo.type.__name__, exctype_name)
|
2023-11-22 17:48:45 -04:00
|
|
|
if msg is None:
|
2023-11-22 20:55:00 -04:00
|
|
|
self.assertEqual(excinfo.formatted.split(':')[0],
|
2023-11-22 17:48:45 -04:00
|
|
|
exctype_name)
|
|
|
|
else:
|
2023-11-22 20:55:00 -04:00
|
|
|
self.assertEqual(excinfo.formatted,
|
2023-11-22 17:48:45 -04:00
|
|
|
'{}: {}'.format(exctype_name, msg))
|
|
|
|
|
2023-11-22 20:55:00 -04:00
|
|
|
return excinfo
|
2023-11-22 17:48:45 -04:00
|
|
|
|
|
|
|
def assert_run_failed(self, exctype, script):
|
|
|
|
self._assert_run_failed(exctype, None, script)
|
|
|
|
|
|
|
|
def assert_run_failed_msg(self, exctype, msg, script):
|
|
|
|
self._assert_run_failed(exctype, msg, script)
|
|
|
|
|
|
|
|
def test_exit(self):
|
|
|
|
with self.subTest('sys.exit(0)'):
|
|
|
|
# XXX Should an unhandled SystemExit(0) be handled as not-an-error?
|
|
|
|
self.assert_run_failed(SystemExit, """
|
|
|
|
sys.exit(0)
|
|
|
|
""")
|
|
|
|
|
|
|
|
with self.subTest('sys.exit()'):
|
|
|
|
self.assert_run_failed(SystemExit, """
|
|
|
|
import sys
|
|
|
|
sys.exit()
|
|
|
|
""")
|
|
|
|
|
|
|
|
with self.subTest('sys.exit(42)'):
|
|
|
|
self.assert_run_failed_msg(SystemExit, '42', """
|
|
|
|
import sys
|
|
|
|
sys.exit(42)
|
|
|
|
""")
|
|
|
|
|
|
|
|
with self.subTest('SystemExit'):
|
|
|
|
self.assert_run_failed_msg(SystemExit, '42', """
|
|
|
|
raise SystemExit(42)
|
|
|
|
""")
|
|
|
|
|
|
|
|
# XXX Also check os._exit() (via a subprocess)?
|
|
|
|
|
|
|
|
def test_plain_exception(self):
|
|
|
|
self.assert_run_failed_msg(Exception, 'spam', """
|
|
|
|
raise Exception("spam")
|
|
|
|
""")
|
|
|
|
|
|
|
|
def test_invalid_syntax(self):
|
|
|
|
script = dedent("""
|
|
|
|
x = 1 + 2
|
|
|
|
y = 2 + 4
|
|
|
|
z = 4 + 8
|
|
|
|
|
|
|
|
# missing close paren
|
|
|
|
print("spam"
|
|
|
|
|
|
|
|
if x + y + z < 20:
|
|
|
|
...
|
|
|
|
""")
|
|
|
|
|
|
|
|
with self.subTest('script'):
|
|
|
|
self.assert_run_failed(SyntaxError, script)
|
|
|
|
|
|
|
|
with self.subTest('module'):
|
|
|
|
modname = 'spam_spam_spam'
|
|
|
|
filename = self.add_module(modname, script)
|
|
|
|
self.assert_run_failed(SyntaxError, f"""
|
|
|
|
import {modname}
|
|
|
|
""")
|
|
|
|
|
|
|
|
def test_NameError(self):
|
|
|
|
self.assert_run_failed(NameError, """
|
|
|
|
res = spam + eggs
|
|
|
|
""")
|
|
|
|
# XXX check preserved suggestions
|
|
|
|
|
|
|
|
def test_AttributeError(self):
|
|
|
|
self.assert_run_failed(AttributeError, """
|
|
|
|
object().spam
|
|
|
|
""")
|
|
|
|
# XXX check preserved suggestions
|
|
|
|
|
|
|
|
def test_ExceptionGroup(self):
|
|
|
|
self.assert_run_failed(ExceptionGroup, """
|
|
|
|
raise ExceptionGroup('exceptions', [
|
|
|
|
Exception('spam'),
|
|
|
|
ImportError('eggs'),
|
|
|
|
])
|
|
|
|
""")
|
|
|
|
|
|
|
|
def test_user_defined_exception(self):
|
|
|
|
self.assert_run_failed_msg('MyError', 'spam', """
|
|
|
|
class MyError(Exception):
|
|
|
|
pass
|
|
|
|
raise MyError('spam')
|
|
|
|
""")
|
|
|
|
|
|
|
|
|
2023-10-06 20:52:22 -03:00
|
|
|
class RunFuncTests(TestBase):
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
super().setUp()
|
|
|
|
self.id = interpreters.create()
|
|
|
|
|
|
|
|
def test_success(self):
|
|
|
|
r, w = os.pipe()
|
|
|
|
def script():
|
|
|
|
global w
|
|
|
|
import contextlib
|
|
|
|
with open(w, 'w', encoding="utf-8") as spipe:
|
|
|
|
with contextlib.redirect_stdout(spipe):
|
|
|
|
print('it worked!', end='')
|
2023-12-12 14:06:06 -04:00
|
|
|
interpreters.set___main___attrs(self.id, dict(w=w))
|
|
|
|
interpreters.run_func(self.id, script)
|
2023-10-06 20:52:22 -03:00
|
|
|
|
|
|
|
with open(r, encoding="utf-8") as outfile:
|
|
|
|
out = outfile.read()
|
|
|
|
|
|
|
|
self.assertEqual(out, 'it worked!')
|
|
|
|
|
|
|
|
def test_in_thread(self):
|
|
|
|
r, w = os.pipe()
|
|
|
|
def script():
|
|
|
|
global w
|
|
|
|
import contextlib
|
|
|
|
with open(w, 'w', encoding="utf-8") as spipe:
|
|
|
|
with contextlib.redirect_stdout(spipe):
|
|
|
|
print('it worked!', end='')
|
|
|
|
def f():
|
2023-12-12 14:06:06 -04:00
|
|
|
interpreters.set___main___attrs(self.id, dict(w=w))
|
|
|
|
interpreters.run_func(self.id, script)
|
2023-10-06 20:52:22 -03:00
|
|
|
t = threading.Thread(target=f)
|
|
|
|
t.start()
|
|
|
|
t.join()
|
|
|
|
|
|
|
|
with open(r, encoding="utf-8") as outfile:
|
|
|
|
out = outfile.read()
|
|
|
|
|
|
|
|
self.assertEqual(out, 'it worked!')
|
|
|
|
|
|
|
|
def test_code_object(self):
|
|
|
|
r, w = os.pipe()
|
|
|
|
|
|
|
|
def script():
|
|
|
|
global w
|
|
|
|
import contextlib
|
|
|
|
with open(w, 'w', encoding="utf-8") as spipe:
|
|
|
|
with contextlib.redirect_stdout(spipe):
|
|
|
|
print('it worked!', end='')
|
|
|
|
code = script.__code__
|
2023-12-12 14:06:06 -04:00
|
|
|
interpreters.set___main___attrs(self.id, dict(w=w))
|
|
|
|
interpreters.run_func(self.id, code)
|
2023-10-06 20:52:22 -03:00
|
|
|
|
|
|
|
with open(r, encoding="utf-8") as outfile:
|
|
|
|
out = outfile.read()
|
|
|
|
|
|
|
|
self.assertEqual(out, 'it worked!')
|
|
|
|
|
|
|
|
def test_closure(self):
|
|
|
|
spam = True
|
|
|
|
def script():
|
|
|
|
assert spam
|
|
|
|
|
|
|
|
with self.assertRaises(ValueError):
|
|
|
|
interpreters.run_func(self.id, script)
|
|
|
|
|
|
|
|
# XXX This hasn't been fixed yet.
|
|
|
|
@unittest.expectedFailure
|
|
|
|
def test_return_value(self):
|
|
|
|
def script():
|
|
|
|
return 'spam'
|
|
|
|
with self.assertRaises(ValueError):
|
|
|
|
interpreters.run_func(self.id, script)
|
|
|
|
|
|
|
|
def test_args(self):
|
|
|
|
with self.subTest('args'):
|
|
|
|
def script(a, b=0):
|
|
|
|
assert a == b
|
|
|
|
with self.assertRaises(ValueError):
|
|
|
|
interpreters.run_func(self.id, script)
|
|
|
|
|
|
|
|
with self.subTest('*args'):
|
|
|
|
def script(*args):
|
|
|
|
assert not args
|
|
|
|
with self.assertRaises(ValueError):
|
|
|
|
interpreters.run_func(self.id, script)
|
|
|
|
|
|
|
|
with self.subTest('**kwargs'):
|
|
|
|
def script(**kwargs):
|
|
|
|
assert not kwargs
|
|
|
|
with self.assertRaises(ValueError):
|
|
|
|
interpreters.run_func(self.id, script)
|
|
|
|
|
|
|
|
with self.subTest('kwonly'):
|
|
|
|
def script(*, spam=True):
|
|
|
|
assert spam
|
|
|
|
with self.assertRaises(ValueError):
|
|
|
|
interpreters.run_func(self.id, script)
|
|
|
|
|
|
|
|
with self.subTest('posonly'):
|
|
|
|
def script(spam, /):
|
|
|
|
assert spam
|
|
|
|
with self.assertRaises(ValueError):
|
|
|
|
interpreters.run_func(self.id, script)
|
|
|
|
|
|
|
|
|
2018-01-29 21:23:44 -04:00
|
|
|
if __name__ == '__main__':
|
|
|
|
unittest.main()
|