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
|
|
|
|
|
2023-02-03 21:14:43 -04:00
|
|
|
import _testcapi
|
2018-01-29 21:23:44 -04:00
|
|
|
from test import support
|
2020-06-25 07:38:51 -03:00
|
|
|
from test.support import import_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')
|
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
|
|
|
|
|
|
|
|
|
|
|
def _run_output(interp, request, shared=None):
|
2018-05-16 16:04:57 -03:00
|
|
|
script, rpipe = _captured_script(request)
|
|
|
|
with rpipe:
|
2018-01-29 21:23:44 -04:00
|
|
|
interpreters.run_string(interp, script, shared)
|
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)
|
|
|
|
except RuntimeError:
|
|
|
|
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,
|
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
|
|
|
|
True,
|
|
|
|
False,
|
|
|
|
NotImplemented,
|
|
|
|
...,
|
|
|
|
# builtin types and objects
|
|
|
|
type,
|
|
|
|
object,
|
|
|
|
object(),
|
|
|
|
Exception(),
|
|
|
|
100.0,
|
|
|
|
# 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-02-03 21:14:43 -04:00
|
|
|
xid = _testcapi.get_crossinterp_data(obj)
|
|
|
|
got = _testcapi.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-02-03 21:14:43 -04:00
|
|
|
xid = _testcapi.get_crossinterp_data(obj)
|
|
|
|
got = _testcapi.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-02-03 21:14:43 -04:00
|
|
|
_testcapi.get_crossinterp_data(i)
|
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)
|
2018-05-16 16:04:57 -03:00
|
|
|
self.assertIsInstance(cur, interpreters.InterpreterID)
|
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)
|
|
|
|
assert isinstance(cur, _interpreters.InterpreterID)
|
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)
|
2018-05-16 16:04:57 -03:00
|
|
|
self.assertIsInstance(main, interpreters.InterpreterID)
|
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)
|
|
|
|
assert isinstance(main, _interpreters.InterpreterID)
|
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)
|
|
|
|
with self.assertRaises(RuntimeError):
|
|
|
|
interpreters.is_running(interp)
|
|
|
|
|
|
|
|
def test_does_not_exist(self):
|
|
|
|
with self.assertRaises(RuntimeError):
|
|
|
|
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)
|
|
|
|
|
|
|
|
|
2018-02-16 21:53:40 -04:00
|
|
|
class InterpreterIDTests(TestBase):
|
|
|
|
|
|
|
|
def test_with_int(self):
|
|
|
|
id = interpreters.InterpreterID(10, force=True)
|
|
|
|
|
|
|
|
self.assertEqual(int(id), 10)
|
|
|
|
|
|
|
|
def test_coerce_id(self):
|
|
|
|
class Int(str):
|
2019-09-13 16:50:27 -03:00
|
|
|
def __index__(self):
|
|
|
|
return 10
|
2018-02-16 21:53:40 -04:00
|
|
|
|
2019-09-25 12:35:57 -03:00
|
|
|
id = interpreters.InterpreterID(Int(), force=True)
|
|
|
|
self.assertEqual(int(id), 10)
|
2018-02-16 21:53:40 -04:00
|
|
|
|
|
|
|
def test_bad_id(self):
|
2019-09-13 16:50:27 -03:00
|
|
|
self.assertRaises(TypeError, interpreters.InterpreterID, object())
|
|
|
|
self.assertRaises(TypeError, interpreters.InterpreterID, 10.0)
|
2019-09-25 12:35:57 -03:00
|
|
|
self.assertRaises(TypeError, interpreters.InterpreterID, '10')
|
2019-09-13 16:50:27 -03:00
|
|
|
self.assertRaises(TypeError, interpreters.InterpreterID, b'10')
|
|
|
|
self.assertRaises(ValueError, interpreters.InterpreterID, -1)
|
|
|
|
self.assertRaises(OverflowError, interpreters.InterpreterID, 2**64)
|
2018-02-16 21:53:40 -04:00
|
|
|
|
|
|
|
def test_does_not_exist(self):
|
2023-02-03 21:14:43 -04:00
|
|
|
id = interpreters.create()
|
2018-02-16 21:53:40 -04:00
|
|
|
with self.assertRaises(RuntimeError):
|
|
|
|
interpreters.InterpreterID(int(id) + 1) # unforced
|
|
|
|
|
2018-05-16 16:04:57 -03:00
|
|
|
def test_str(self):
|
|
|
|
id = interpreters.InterpreterID(10, force=True)
|
|
|
|
self.assertEqual(str(id), '10')
|
|
|
|
|
2018-02-16 21:53:40 -04:00
|
|
|
def test_repr(self):
|
|
|
|
id = interpreters.InterpreterID(10, force=True)
|
|
|
|
self.assertEqual(repr(id), 'InterpreterID(10)')
|
|
|
|
|
|
|
|
def test_equality(self):
|
|
|
|
id1 = interpreters.create()
|
|
|
|
id2 = interpreters.InterpreterID(int(id1))
|
|
|
|
id3 = interpreters.create()
|
|
|
|
|
|
|
|
self.assertTrue(id1 == id1)
|
|
|
|
self.assertTrue(id1 == id2)
|
|
|
|
self.assertTrue(id1 == int(id1))
|
2019-09-13 16:50:27 -03:00
|
|
|
self.assertTrue(int(id1) == id1)
|
|
|
|
self.assertTrue(id1 == float(int(id1)))
|
|
|
|
self.assertTrue(float(int(id1)) == id1)
|
|
|
|
self.assertFalse(id1 == float(int(id1)) + 0.1)
|
|
|
|
self.assertFalse(id1 == str(int(id1)))
|
|
|
|
self.assertFalse(id1 == 2**1000)
|
|
|
|
self.assertFalse(id1 == float('inf'))
|
|
|
|
self.assertFalse(id1 == 'spam')
|
2018-02-16 21:53:40 -04:00
|
|
|
self.assertFalse(id1 == id3)
|
|
|
|
|
|
|
|
self.assertFalse(id1 != id1)
|
|
|
|
self.assertFalse(id1 != id2)
|
|
|
|
self.assertTrue(id1 != id3)
|
|
|
|
|
|
|
|
|
2018-01-29 21:23:44 -04:00
|
|
|
class CreateTests(TestBase):
|
|
|
|
|
|
|
|
def test_in_main(self):
|
|
|
|
id = interpreters.create()
|
2018-05-16 16:04:57 -03:00
|
|
|
self.assertIsInstance(id, interpreters.InterpreterID)
|
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)
|
|
|
|
assert isinstance(id, _interpreters.InterpreterID)
|
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()
|
|
|
|
with self.assertRaises(RuntimeError):
|
|
|
|
interpreters.destroy(main)
|
|
|
|
|
|
|
|
def f():
|
|
|
|
with self.assertRaises(RuntimeError):
|
|
|
|
interpreters.destroy(main)
|
|
|
|
|
|
|
|
t = threading.Thread(target=f)
|
|
|
|
t.start()
|
|
|
|
t.join()
|
|
|
|
|
|
|
|
def test_already_destroyed(self):
|
|
|
|
id = interpreters.create()
|
|
|
|
interpreters.destroy(id)
|
|
|
|
with self.assertRaises(RuntimeError):
|
|
|
|
interpreters.destroy(id)
|
|
|
|
|
|
|
|
def test_does_not_exist(self):
|
|
|
|
with self.assertRaises(RuntimeError):
|
|
|
|
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})
|
2018-02-03 00:49:49 -04:00
|
|
|
except RuntimeError:
|
|
|
|
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.")
|
|
|
|
|
|
|
|
with self.assertRaises(RuntimeError,
|
|
|
|
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'
|
|
|
|
subinterp = interpreters.create(isolated=True)
|
|
|
|
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'):
|
|
|
|
subinterp = interpreters.create(isolated=False)
|
|
|
|
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):
|
|
|
|
interpreters.run_string(
|
|
|
|
interp,
|
|
|
|
f'assert(obj == {obj!r})',
|
|
|
|
shared=dict(obj=obj),
|
|
|
|
)
|
|
|
|
|
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):
|
|
|
|
with self.assertRaises(RuntimeError):
|
|
|
|
interpreters.run_string(self.id, 'print("spam")')
|
|
|
|
|
|
|
|
def test_does_not_exist(self):
|
|
|
|
id = 0
|
|
|
|
while id in interpreters.list_all():
|
|
|
|
id += 1
|
|
|
|
with self.assertRaises(RuntimeError):
|
|
|
|
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")')
|
|
|
|
|
|
|
|
@contextlib.contextmanager
|
|
|
|
def assert_run_failed(self, exctype, msg=None):
|
|
|
|
with self.assertRaises(interpreters.RunFailedError) as caught:
|
|
|
|
yield
|
|
|
|
if msg is None:
|
|
|
|
self.assertEqual(str(caught.exception).split(':')[0],
|
2023-07-08 05:44:50 -03:00
|
|
|
exctype.__name__)
|
2018-01-29 21:23:44 -04:00
|
|
|
else:
|
|
|
|
self.assertEqual(str(caught.exception),
|
2023-07-08 05:44:50 -03:00
|
|
|
"{}: {}".format(exctype.__name__, msg))
|
2018-01-29 21:23:44 -04:00
|
|
|
|
|
|
|
def test_invalid_syntax(self):
|
|
|
|
with self.assert_run_failed(SyntaxError):
|
|
|
|
# missing close paren
|
|
|
|
interpreters.run_string(self.id, 'print("spam"')
|
|
|
|
|
|
|
|
def test_failure(self):
|
|
|
|
with self.assert_run_failed(Exception, 'spam'):
|
|
|
|
interpreters.run_string(self.id, 'raise Exception("spam")')
|
|
|
|
|
|
|
|
def test_SystemExit(self):
|
|
|
|
with self.assert_run_failed(SystemExit, '42'):
|
|
|
|
interpreters.run_string(self.id, 'raise SystemExit(42)')
|
|
|
|
|
|
|
|
def test_sys_exit(self):
|
|
|
|
with self.assert_run_failed(SystemExit):
|
|
|
|
interpreters.run_string(self.id, dedent("""
|
|
|
|
import sys
|
|
|
|
sys.exit()
|
|
|
|
"""))
|
|
|
|
|
|
|
|
with self.assert_run_failed(SystemExit, '42'):
|
|
|
|
interpreters.run_string(self.id, dedent("""
|
|
|
|
import sys
|
|
|
|
sys.exit(42)
|
|
|
|
"""))
|
|
|
|
|
|
|
|
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)
|
|
|
|
""")
|
|
|
|
interpreters.run_string(self.id, script, shared)
|
|
|
|
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__']
|
|
|
|
""")
|
|
|
|
interpreters.run_string(self.id, script, shared)
|
|
|
|
|
|
|
|
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)
|
|
|
|
""")
|
|
|
|
interpreters.run_string(self.id, script, shared)
|
|
|
|
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)
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
unittest.main()
|