from collections import namedtuple
import contextlib
+import io
import json
import os
import os.path
out, err = self.run_embedded_interpreter("test_repeated_init_exec", code)
self.assertEqual(out, '9\n' * INIT_LOOPS)
+ def test_static_types_inherited_slots(self):
+ slots = []
+ script = ['import sys']
+ from test.test_types import iter_builtin_types, iter_own_slot_wrappers
+ for cls in iter_builtin_types():
+ for slot in iter_own_slot_wrappers(cls):
+ slots.append((cls, slot))
+ attr = f'{cls.__name__}.{slot}'
+ script.append(f'print("{attr}:", {attr}, file=sys.stderr)')
+ script.append('')
+ script = os.linesep.join(script)
+
+ with contextlib.redirect_stderr(io.StringIO()) as stderr:
+ exec(script)
+ expected = stderr.getvalue().splitlines()
+
+ out, err = self.run_embedded_interpreter("test_repeated_init_exec", script)
+ results = err.split('--- Loop #')[1:]
+ results = [res.rpartition(' ---\n')[-1] for res in results]
+
+ self.maxDiff = None
+ for i, result in enumerate(results, start=1):
+ with self.subTest(loop=i):
+ self.assertEqual(result.splitlines(), expected)
+ self.assertEqual(out, '')
+
+
class InitConfigTests(EmbeddingTestsMixin, unittest.TestCase):
maxDiff = 4096
UTF8_MODE_ERRORS = ('surrogatepass' if MS_WINDOWS else 'surrogateescape')
# Python test set -- part 6, built-in types
from test.support import run_with_locale, cpython_only, MISSING_C_DOCSTRINGS
+from test.test_import import no_rerun
import collections.abc
from collections import namedtuple
import copy
f()
+def iter_builtin_types():
+ for obj in __builtins__.values():
+ if not isinstance(obj, type):
+ continue
+ cls = obj
+ if cls.__module__ != 'builtins':
+ continue
+ yield cls
+
+
+@cpython_only
+def iter_own_slot_wrappers(cls):
+ for name, value in vars(cls).items():
+ if not name.startswith('__') or not name.endswith('__'):
+ continue
+ if 'slot wrapper' not in str(value):
+ continue
+ yield name
+
+
class TypesTests(unittest.TestCase):
def test_truth_values(self):
raise unittest.SkipTest('subinterpreters required')
@cpython_only
+ @no_rerun('channels (and queues) might have a refleak; see gh-122199')
def test_slot_wrappers(self):
rch, sch = interpreters.create_channel()
- # For now it's sufficient to check int.__str__.
- # See https://github.com/python/cpython/issues/117482
- # and https://github.com/python/cpython/pull/117660.
- script = textwrap.dedent(f'''
- text = repr(int.__str__)
- sch = interpreters.SendChannel({sch.id})
- sch.send_nowait(text)
- ''')
+ slots = []
+ script = ''
+ for cls in iter_builtin_types():
+ for slot in iter_own_slot_wrappers(cls):
+ slots.append((cls, slot))
+ attr = f'{cls.__name__}.{slot}'
+ script += textwrap.dedent(f"""
+ sch.send_nowait('{attr}: ' + repr({attr}))
+ """)
exec(script)
- expected = rch.recv()
+ all_expected = []
+ for cls, slot in slots:
+ result = rch.recv()
+ assert result.startswith(f'{cls.__name__}.{slot}: '), (cls, slot, result)
+ all_expected.append(result)
interp = interpreters.create()
- interp.run('from test.support import interpreters')
+ interp.run(textwrap.dedent(f"""
+ from test.support import interpreters
+ sch = interpreters.SendChannel({sch.id})
+ """))
interp.run(script)
- results = rch.recv()
- self.assertEqual(results, expected)
+ for i, _ in enumerate(slots):
+ with self.subTest(cls=cls, slot=slot):
+ expected = all_expected[i]
+ result = rch.recv()
+ self.assertEqual(result, expected)
if __name__ == '__main__':