def _get_parent_path(self):
parent_module_name, path_attr_name = self._find_parent_path_names()
- return getattr(sys.modules[parent_module_name], path_attr_name)
+ try:
+ module = sys.modules[parent_module_name]
+ except KeyError as e:
+ raise ModuleNotFoundError(
+ f"{parent_module_name!r} must be imported before finding {self._name!r}.",
+ name=parent_module_name,
+ ) from e
+ else:
+ return getattr(module, path_attr_name)
def _recalculate(self):
# If the parent's path has changed, recalculate _path
--- /dev/null
+This directory should not be a package, but should share a name with an
+unrelated subpackage.
self.assertEqual(a_test.attr, 'in module')
+class NamespaceSubpackageSameName(NamespacePackageTest):
+ paths = ['']
+
+ def test_namespace_subpackage_shares_name_with_directory(self):
+ submodule_path = 'project4.foo'
+ with self.assertRaises(ModuleNotFoundError) as cm:
+ importlib.machinery.PathFinder.find_spec(submodule_path)
+
+ self.assertEqual(cm.exception.name, 'project4')
+
+
class ReloadTests(NamespacePackageTest):
paths = ['portion1']
test/test_importlib/namespace_pkgs \
test/test_importlib/namespace_pkgs/both_portions \
test/test_importlib/namespace_pkgs/both_portions/foo \
+ test/test_importlib/namespace_pkgs/foo \
test/test_importlib/namespace_pkgs/module_and_namespace_package \
test/test_importlib/namespace_pkgs/module_and_namespace_package/a_test \
test/test_importlib/namespace_pkgs/not_a_namespace_pkg \
--- /dev/null
+Reraise :exc:`KeyError` as :exc:`ModuleNotFoundError` when
+:meth:`importlib.machinery.PathFinder.find_spec` is called on a submodule
+without importing the parent (and without a ``path`` argument).