2003-04-15 08:10:33 -03:00
|
|
|
import unittest
|
2009-10-10 18:08:31 -03:00
|
|
|
from test import support
|
2020-08-07 12:18:38 -03:00
|
|
|
from test.support import warnings_helper
|
2009-10-10 18:08:31 -03:00
|
|
|
import os
|
2001-01-20 15:54:20 -04:00
|
|
|
import sys
|
2022-01-20 13:56:33 -04:00
|
|
|
|
2001-01-20 15:54:20 -04:00
|
|
|
|
2022-03-01 10:44:08 -04:00
|
|
|
if support.check_sanitizer(address=True, memory=True):
|
2023-08-22 10:52:32 -03:00
|
|
|
SKIP_MODULES = frozenset((
|
|
|
|
# gh-90791: Tests involving libX11 can SEGFAULT on ASAN/MSAN builds.
|
|
|
|
# Skip modules, packages and tests using '_tkinter'.
|
|
|
|
'_tkinter',
|
|
|
|
'tkinter',
|
|
|
|
'test_tkinter',
|
|
|
|
'test_ttk',
|
|
|
|
'test_ttk_textonly',
|
|
|
|
'idlelib',
|
|
|
|
'test_idle',
|
|
|
|
))
|
|
|
|
else:
|
|
|
|
SKIP_MODULES = ()
|
2022-03-01 10:44:08 -04:00
|
|
|
|
|
|
|
|
2009-10-10 18:08:31 -03:00
|
|
|
class NoAll(RuntimeError):
|
|
|
|
pass
|
|
|
|
|
|
|
|
class FailedImport(RuntimeError):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2003-04-15 08:10:33 -03:00
|
|
|
class AllTest(unittest.TestCase):
|
|
|
|
|
|
|
|
def check_all(self, modname):
|
|
|
|
names = {}
|
2020-08-07 12:18:38 -03:00
|
|
|
with warnings_helper.check_warnings(
|
2022-04-05 16:05:48 -03:00
|
|
|
(f".*{modname}", DeprecationWarning),
|
2010-10-29 08:53:34 -03:00
|
|
|
(".* (module|package)", DeprecationWarning),
|
2020-04-24 15:19:46 -03:00
|
|
|
(".* (module|package)", PendingDeprecationWarning),
|
2010-10-29 08:53:34 -03:00
|
|
|
("", ResourceWarning),
|
|
|
|
quiet=True):
|
2008-05-06 20:48:04 -03:00
|
|
|
try:
|
|
|
|
exec("import %s" % modname, names)
|
2009-10-10 18:08:31 -03:00
|
|
|
except:
|
2008-05-06 20:48:04 -03:00
|
|
|
# Silent fail here seems the best route since some modules
|
2009-10-10 18:08:31 -03:00
|
|
|
# may not be available or not initialize properly in all
|
|
|
|
# environments.
|
|
|
|
raise FailedImport(modname)
|
|
|
|
if not hasattr(sys.modules[modname], "__all__"):
|
|
|
|
raise NoAll(modname)
|
2003-04-15 08:10:33 -03:00
|
|
|
names = {}
|
2013-05-01 08:58:09 -03:00
|
|
|
with self.subTest(module=modname):
|
2020-08-07 12:18:38 -03:00
|
|
|
with warnings_helper.check_warnings(
|
2019-05-27 16:56:22 -03:00
|
|
|
("", DeprecationWarning),
|
|
|
|
("", ResourceWarning),
|
|
|
|
quiet=True):
|
|
|
|
try:
|
|
|
|
exec("from %s import *" % modname, names)
|
|
|
|
except Exception as e:
|
|
|
|
# Include the module name in the exception string
|
|
|
|
self.fail("__all__ failure in {}: {}: {}".format(
|
|
|
|
modname, e.__class__.__name__, e))
|
|
|
|
if "__builtins__" in names:
|
|
|
|
del names["__builtins__"]
|
|
|
|
if '__annotations__' in names:
|
|
|
|
del names['__annotations__']
|
|
|
|
if "__warningregistry__" in names:
|
|
|
|
del names["__warningregistry__"]
|
|
|
|
keys = set(names)
|
|
|
|
all_list = sys.modules[modname].__all__
|
|
|
|
all_set = set(all_list)
|
|
|
|
self.assertCountEqual(all_set, all_list, "in module {}".format(modname))
|
|
|
|
self.assertEqual(keys, all_set, "in module {}".format(modname))
|
2003-04-15 08:10:33 -03:00
|
|
|
|
2009-10-10 18:08:31 -03:00
|
|
|
def walk_modules(self, basedir, modpath):
|
|
|
|
for fn in sorted(os.listdir(basedir)):
|
|
|
|
path = os.path.join(basedir, fn)
|
|
|
|
if os.path.isdir(path):
|
2023-08-22 10:52:32 -03:00
|
|
|
if fn in SKIP_MODULES:
|
|
|
|
continue
|
2009-10-10 18:08:31 -03:00
|
|
|
pkg_init = os.path.join(path, '__init__.py')
|
|
|
|
if os.path.exists(pkg_init):
|
|
|
|
yield pkg_init, modpath + fn
|
|
|
|
for p, m in self.walk_modules(path, modpath + fn + "."):
|
|
|
|
yield p, m
|
|
|
|
continue
|
2023-08-22 10:52:32 -03:00
|
|
|
|
|
|
|
if fn == '__init__.py':
|
|
|
|
continue
|
|
|
|
if not fn.endswith('.py'):
|
|
|
|
continue
|
|
|
|
modname = fn.removesuffix('.py')
|
|
|
|
if modname in SKIP_MODULES:
|
2009-10-10 18:08:31 -03:00
|
|
|
continue
|
2023-08-22 10:52:32 -03:00
|
|
|
yield path, modpath + modname
|
2009-10-10 18:08:31 -03:00
|
|
|
|
2003-04-15 08:10:33 -03:00
|
|
|
def test_all(self):
|
2020-08-11 10:26:59 -03:00
|
|
|
# List of denied modules and packages
|
|
|
|
denylist = set([
|
2009-10-10 18:08:31 -03:00
|
|
|
# Will raise a SyntaxError when compiling the exec statement
|
|
|
|
'__future__',
|
|
|
|
])
|
|
|
|
|
2022-12-23 16:17:24 -04:00
|
|
|
# In case _socket fails to build, make this test fail more gracefully
|
2023-06-09 09:50:31 -03:00
|
|
|
# than an AttributeError somewhere deep in concurrent.futures, email
|
|
|
|
# or unittest.
|
2024-06-18 12:51:47 -03:00
|
|
|
import _socket # noqa: F401
|
2003-04-15 08:10:33 -03:00
|
|
|
|
2009-10-10 18:08:31 -03:00
|
|
|
ignored = []
|
|
|
|
failed_imports = []
|
|
|
|
lib_dir = os.path.dirname(os.path.dirname(__file__))
|
|
|
|
for path, modname in self.walk_modules(lib_dir, ""):
|
|
|
|
m = modname
|
2020-08-13 14:20:28 -03:00
|
|
|
denied = False
|
2009-10-10 18:08:31 -03:00
|
|
|
while m:
|
2020-08-11 10:26:59 -03:00
|
|
|
if m in denylist:
|
2020-08-13 14:20:28 -03:00
|
|
|
denied = True
|
2009-10-10 18:08:31 -03:00
|
|
|
break
|
|
|
|
m = m.rpartition('.')[0]
|
2020-08-13 14:20:28 -03:00
|
|
|
if denied:
|
2009-10-10 18:08:31 -03:00
|
|
|
continue
|
|
|
|
if support.verbose:
|
2023-08-22 10:52:32 -03:00
|
|
|
print(f"Check {modname}", flush=True)
|
2003-04-15 08:10:33 -03:00
|
|
|
try:
|
2009-10-10 18:08:31 -03:00
|
|
|
# This heuristic speeds up the process by removing, de facto,
|
|
|
|
# most test modules (and avoiding the auto-executing ones).
|
|
|
|
with open(path, "rb") as f:
|
|
|
|
if b"__all__" not in f.read():
|
|
|
|
raise NoAll(modname)
|
2023-08-22 10:52:32 -03:00
|
|
|
self.check_all(modname)
|
2009-10-10 18:08:31 -03:00
|
|
|
except NoAll:
|
|
|
|
ignored.append(modname)
|
|
|
|
except FailedImport:
|
|
|
|
failed_imports.append(modname)
|
|
|
|
|
|
|
|
if support.verbose:
|
|
|
|
print('Following modules have no __all__ and have been ignored:',
|
|
|
|
ignored)
|
|
|
|
print('Following modules failed to be imported:', failed_imports)
|
2001-01-20 15:54:20 -04:00
|
|
|
|
2001-08-09 18:40:30 -03:00
|
|
|
|
2003-04-15 08:10:33 -03:00
|
|
|
if __name__ == "__main__":
|
2013-06-12 21:12:30 -03:00
|
|
|
unittest.main()
|