Commit 881535b7 authored by Brett Cannon's avatar Brett Cannon

Issue #14582: Import returns the module returned by a loader instead

of sys.modules when possible.

This is being done for two reasons. One is to gain a little bit of
performance by skipping an unnecessary dict lookup in sys.modules. But
the other (and main) reason is to be a little bit more clear in how
things should work from the perspective of import's interactions with
loaders. Otherwise loaders can easily forget to return the module even
though PEP 302 explicitly states they are expected to return the module
they loaded.
parent 27fc5287
...@@ -974,12 +974,12 @@ def _find_and_load(name, import_): ...@@ -974,12 +974,12 @@ def _find_and_load(name, import_):
loader = _find_module(name, path) loader = _find_module(name, path)
if loader is None: if loader is None:
raise ImportError(_ERR_MSG.format(name), name=name) raise ImportError(_ERR_MSG.format(name), name=name)
elif name not in sys.modules: elif name in sys.modules:
# The parent import may have already imported this module. # The parent module already imported this module.
loader.load_module(name) module = sys.modules[name]
else:
module = loader.load_module(name)
verbose_message('import {!r} # {!r}', name, loader) verbose_message('import {!r} # {!r}', name, loader)
# Backwards-compatibility; be nicer to skip the dict lookup.
module = sys.modules[name]
if parent: if parent:
# Set the module as an attribute on its parent. # Set the module as an attribute on its parent.
parent_module = sys.modules[parent] parent_module = sys.modules[parent]
...@@ -1078,7 +1078,11 @@ def __import__(name, globals={}, locals={}, fromlist=[], level=0): ...@@ -1078,7 +1078,11 @@ def __import__(name, globals={}, locals={}, fromlist=[], level=0):
# Return up to the first dot in 'name'. This is complicated by the fact # Return up to the first dot in 'name'. This is complicated by the fact
# that 'name' may be relative. # that 'name' may be relative.
if level == 0: if level == 0:
return sys.modules[name.partition('.')[0]] index = name.find('.')
if index == -1:
return module
else:
return sys.modules[name[:index]]
elif not name: elif not name:
return module return module
else: else:
......
...@@ -47,36 +47,12 @@ class UseCache(unittest.TestCase): ...@@ -47,36 +47,12 @@ class UseCache(unittest.TestCase):
mock.load_module = MethodType(load_module, mock) mock.load_module = MethodType(load_module, mock)
return mock return mock
# __import__ inconsistent between loaders and built-in import when it comes def test_using_loader_return(self):
# to when to use the module in sys.modules and when not to. loader_return = 'hi there!'
@import_util.importlib_only with self.create_mock('module', return_=loader_return) as mock:
def test_using_cache_after_loader(self):
# [from cache on return]
with self.create_mock('module') as mock:
with util.import_state(meta_path=[mock]): with util.import_state(meta_path=[mock]):
module = import_util.import_('module') module = import_util.import_('module')
self.assertEqual(id(module), id(sys.modules['module'])) self.assertEqual(module, loader_return)
# See test_using_cache_after_loader() for reasoning.
@import_util.importlib_only
def test_using_cache_for_assigning_to_attribute(self):
# [from cache to attribute]
with self.create_mock('pkg.__init__', 'pkg.module') as importer:
with util.import_state(meta_path=[importer]):
module = import_util.import_('pkg.module')
self.assertTrue(hasattr(module, 'module'))
self.assertTrue(id(module.module), id(sys.modules['pkg.module']))
# See test_using_cache_after_loader() for reasoning.
@import_util.importlib_only
def test_using_cache_for_fromlist(self):
# [from cache for fromlist]
with self.create_mock('pkg.__init__', 'pkg.module') as importer:
with util.import_state(meta_path=[importer]):
module = import_util.import_('pkg', fromlist=['module'])
self.assertTrue(hasattr(module, 'module'))
self.assertEqual(id(module.module),
id(sys.modules['pkg.module']))
def test_main(): def test_main():
......
...@@ -10,6 +10,9 @@ What's New in Python 3.3.0 Alpha 3? ...@@ -10,6 +10,9 @@ What's New in Python 3.3.0 Alpha 3?
Core and Builtins Core and Builtins
----------------- -----------------
- Issue #14582: Import directly returns the module as returned by a loader when
possible instead of fetching it from sys.modules.
- Issue #13889: Check and (if necessary) set FPU control word before calling - Issue #13889: Check and (if necessary) set FPU control word before calling
any of the dtoa.c string <-> float conversion functions, on MSVC builds of any of the dtoa.c string <-> float conversion functions, on MSVC builds of
Python. This fixes issues when embedding Python in a Delphi app. Python. This fixes issues when embedding Python in a Delphi app.
......
...@@ -3019,15 +3019,22 @@ PyImport_ImportModuleLevelObject(PyObject *name, PyObject *given_globals, ...@@ -3019,15 +3019,22 @@ PyImport_ImportModuleLevelObject(PyObject *name, PyObject *given_globals,
Py_DECREF(partition); Py_DECREF(partition);
if (level == 0) { if (level == 0) {
final_mod = PyDict_GetItem(interp->modules, front); if (PyUnicode_GET_LENGTH(name) ==
Py_DECREF(front); PyUnicode_GET_LENGTH(front)) {
if (final_mod == NULL) { final_mod = mod;
PyErr_Format(PyExc_KeyError,
"%R not in sys.modules as expected", front);
} }
else { else {
Py_INCREF(final_mod); final_mod = PyDict_GetItem(interp->modules, front);
if (final_mod == NULL) {
PyErr_Format(PyExc_KeyError,
"%R not in sys.modules as expected", front);
}
}
Py_DECREF(front);
if (final_mod == NULL) {
goto error_with_unlock;
} }
Py_INCREF(final_mod);
} }
else { else {
Py_ssize_t cut_off = PyUnicode_GetLength(name) - Py_ssize_t cut_off = PyUnicode_GetLength(name) -
......
This diff is collapsed.
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment