Walter Dörwald | b1ded1e | 2003-04-15 11:10:33 +0000 | [diff] [blame] | 1 | import unittest |
Antoine Pitrou | 4de39cd | 2009-10-10 21:08:31 +0000 | [diff] [blame] | 2 | from test import support |
| 3 | import os |
Skip Montanaro | e99d5ea | 2001-01-20 19:54:20 +0000 | [diff] [blame] | 4 | import sys |
Tim Peters | b05cd49 | 2002-04-11 20:04:12 +0000 | [diff] [blame] | 5 | |
Skip Montanaro | e99d5ea | 2001-01-20 19:54:20 +0000 | [diff] [blame] | 6 | |
Antoine Pitrou | 4de39cd | 2009-10-10 21:08:31 +0000 | [diff] [blame] | 7 | class NoAll(RuntimeError): |
| 8 | pass |
| 9 | |
| 10 | class FailedImport(RuntimeError): |
| 11 | pass |
| 12 | |
| 13 | |
Walter Dörwald | b1ded1e | 2003-04-15 11:10:33 +0000 | [diff] [blame] | 14 | class AllTest(unittest.TestCase): |
| 15 | |
| 16 | def check_all(self, modname): |
| 17 | names = {} |
Antoine Pitrou | a0d2f4d | 2010-10-29 11:53:34 +0000 | [diff] [blame] | 18 | with support.check_warnings( |
| 19 | (".* (module|package)", DeprecationWarning), |
| 20 | ("", ResourceWarning), |
| 21 | quiet=True): |
Christian Heimes | 75ca4ea | 2008-05-06 23:48:04 +0000 | [diff] [blame] | 22 | try: |
| 23 | exec("import %s" % modname, names) |
Antoine Pitrou | 4de39cd | 2009-10-10 21:08:31 +0000 | [diff] [blame] | 24 | except: |
Christian Heimes | 75ca4ea | 2008-05-06 23:48:04 +0000 | [diff] [blame] | 25 | # Silent fail here seems the best route since some modules |
Antoine Pitrou | 4de39cd | 2009-10-10 21:08:31 +0000 | [diff] [blame] | 26 | # may not be available or not initialize properly in all |
| 27 | # environments. |
| 28 | raise FailedImport(modname) |
| 29 | if not hasattr(sys.modules[modname], "__all__"): |
| 30 | raise NoAll(modname) |
Walter Dörwald | b1ded1e | 2003-04-15 11:10:33 +0000 | [diff] [blame] | 31 | names = {} |
Antoine Pitrou | 4de39cd | 2009-10-10 21:08:31 +0000 | [diff] [blame] | 32 | try: |
| 33 | exec("from %s import *" % modname, names) |
| 34 | except Exception as e: |
| 35 | # Include the module name in the exception string |
| 36 | self.fail("__all__ failure in {}: {}: {}".format( |
| 37 | modname, e.__class__.__name__, e)) |
Guido van Rossum | e2b70bc | 2006-08-18 22:13:04 +0000 | [diff] [blame] | 38 | if "__builtins__" in names: |
Walter Dörwald | b1ded1e | 2003-04-15 11:10:33 +0000 | [diff] [blame] | 39 | del names["__builtins__"] |
Raymond Hettinger | a690a99 | 2003-11-16 16:17:49 +0000 | [diff] [blame] | 40 | keys = set(names) |
| 41 | all = set(sys.modules[modname].__all__) |
Guido van Rossum | d8faa36 | 2007-04-27 19:54:29 +0000 | [diff] [blame] | 42 | self.assertEqual(keys, all) |
Skip Montanaro | e99d5ea | 2001-01-20 19:54:20 +0000 | [diff] [blame] | 43 | |
Antoine Pitrou | 4de39cd | 2009-10-10 21:08:31 +0000 | [diff] [blame] | 44 | def walk_modules(self, basedir, modpath): |
| 45 | for fn in sorted(os.listdir(basedir)): |
| 46 | path = os.path.join(basedir, fn) |
| 47 | if os.path.isdir(path): |
| 48 | pkg_init = os.path.join(path, '__init__.py') |
| 49 | if os.path.exists(pkg_init): |
| 50 | yield pkg_init, modpath + fn |
| 51 | for p, m in self.walk_modules(path, modpath + fn + "."): |
| 52 | yield p, m |
| 53 | continue |
| 54 | if not fn.endswith('.py') or fn == '__init__.py': |
| 55 | continue |
| 56 | yield path, modpath + fn[:-3] |
| 57 | |
Walter Dörwald | b1ded1e | 2003-04-15 11:10:33 +0000 | [diff] [blame] | 58 | def test_all(self): |
Antoine Pitrou | 4de39cd | 2009-10-10 21:08:31 +0000 | [diff] [blame] | 59 | # Blacklisted modules and packages |
| 60 | blacklist = set([ |
| 61 | # Will raise a SyntaxError when compiling the exec statement |
| 62 | '__future__', |
| 63 | ]) |
| 64 | |
Walter Dörwald | b1ded1e | 2003-04-15 11:10:33 +0000 | [diff] [blame] | 65 | if not sys.platform.startswith('java'): |
| 66 | # In case _socket fails to build, make this test fail more gracefully |
| 67 | # than an AttributeError somewhere deep in CGIHTTPServer. |
| 68 | import _socket |
Tim Peters | ab9ba27 | 2001-08-09 21:40:30 +0000 | [diff] [blame] | 69 | |
Walter Dörwald | b1ded1e | 2003-04-15 11:10:33 +0000 | [diff] [blame] | 70 | # rlcompleter needs special consideration; it import readline which |
| 71 | # initializes GNU readline which calls setlocale(LC_CTYPE, "")... :-( |
| 72 | try: |
Antoine Pitrou | 4de39cd | 2009-10-10 21:08:31 +0000 | [diff] [blame] | 73 | import rlcompleter |
| 74 | import locale |
| 75 | except ImportError: |
| 76 | pass |
| 77 | else: |
| 78 | locale.setlocale(locale.LC_CTYPE, 'C') |
| 79 | |
| 80 | ignored = [] |
| 81 | failed_imports = [] |
| 82 | lib_dir = os.path.dirname(os.path.dirname(__file__)) |
| 83 | for path, modname in self.walk_modules(lib_dir, ""): |
| 84 | m = modname |
| 85 | blacklisted = False |
| 86 | while m: |
| 87 | if m in blacklist: |
| 88 | blacklisted = True |
| 89 | break |
| 90 | m = m.rpartition('.')[0] |
| 91 | if blacklisted: |
| 92 | continue |
| 93 | if support.verbose: |
| 94 | print(modname) |
Walter Dörwald | b1ded1e | 2003-04-15 11:10:33 +0000 | [diff] [blame] | 95 | try: |
Antoine Pitrou | 4de39cd | 2009-10-10 21:08:31 +0000 | [diff] [blame] | 96 | # This heuristic speeds up the process by removing, de facto, |
| 97 | # most test modules (and avoiding the auto-executing ones). |
| 98 | with open(path, "rb") as f: |
| 99 | if b"__all__" not in f.read(): |
| 100 | raise NoAll(modname) |
| 101 | self.check_all(modname) |
| 102 | except NoAll: |
| 103 | ignored.append(modname) |
| 104 | except FailedImport: |
| 105 | failed_imports.append(modname) |
| 106 | |
| 107 | if support.verbose: |
| 108 | print('Following modules have no __all__ and have been ignored:', |
| 109 | ignored) |
| 110 | print('Following modules failed to be imported:', failed_imports) |
Walter Dörwald | b1ded1e | 2003-04-15 11:10:33 +0000 | [diff] [blame] | 111 | |
| 112 | |
| 113 | def test_main(): |
Antoine Pitrou | 4de39cd | 2009-10-10 21:08:31 +0000 | [diff] [blame] | 114 | support.run_unittest(AllTest) |
Walter Dörwald | b1ded1e | 2003-04-15 11:10:33 +0000 | [diff] [blame] | 115 | |
| 116 | if __name__ == "__main__": |
| 117 | test_main() |