2010-05-04 14:14:47 +08:00
|
|
|
import os
|
|
|
|
import sys
|
2013-07-01 20:22:27 +08:00
|
|
|
import unittest
|
2015-01-28 20:35:27 +08:00
|
|
|
from importlib import import_module
|
2010-05-04 14:14:47 +08:00
|
|
|
from zipimport import zipimporter
|
|
|
|
|
2018-11-27 03:05:02 +08:00
|
|
|
from django.test import SimpleTestCase, modify_settings
|
2014-12-22 04:19:05 +08:00
|
|
|
from django.test.utils import extend_sys_path
|
2015-01-28 20:35:27 +08:00
|
|
|
from django.utils.module_loading import (
|
|
|
|
autodiscover_modules,
|
|
|
|
import_string,
|
|
|
|
module_has_submodule,
|
|
|
|
)
|
2010-05-04 14:14:47 +08:00
|
|
|
|
2010-10-11 20:55:17 +08:00
|
|
|
|
2010-09-28 16:18:12 +08:00
|
|
|
class DefaultLoader(unittest.TestCase):
|
2010-05-04 14:14:47 +08:00
|
|
|
def test_loader(self):
|
|
|
|
"Normal module existence can be tested"
|
2013-02-26 20:19:18 +08:00
|
|
|
test_module = import_module("utils_tests.test_module")
|
2011-08-28 10:37:16 +08:00
|
|
|
test_no_submodule = import_module("utils_tests.test_no_submodule")
|
2010-05-04 14:14:47 +08:00
|
|
|
|
|
|
|
# An importable child
|
|
|
|
self.assertTrue(module_has_submodule(test_module, "good_module"))
|
2013-02-26 20:19:18 +08:00
|
|
|
mod = import_module("utils_tests.test_module.good_module")
|
2010-05-04 14:14:47 +08:00
|
|
|
self.assertEqual(mod.content, "Good Module")
|
|
|
|
|
|
|
|
# A child that exists, but will generate an import error if loaded
|
|
|
|
self.assertTrue(module_has_submodule(test_module, "bad_module"))
|
2016-01-17 19:26:39 +08:00
|
|
|
with self.assertRaises(ImportError):
|
|
|
|
import_module("utils_tests.test_module.bad_module")
|
2010-05-04 14:14:47 +08:00
|
|
|
|
|
|
|
# A child that doesn't exist
|
|
|
|
self.assertFalse(module_has_submodule(test_module, "no_such_module"))
|
2016-01-17 19:26:39 +08:00
|
|
|
with self.assertRaises(ImportError):
|
|
|
|
import_module("utils_tests.test_module.no_such_module")
|
2010-05-04 14:14:47 +08:00
|
|
|
|
2011-04-22 20:03:18 +08:00
|
|
|
# A child that doesn't exist, but is the name of a package on the path
|
|
|
|
self.assertFalse(module_has_submodule(test_module, "django"))
|
2016-01-17 19:26:39 +08:00
|
|
|
with self.assertRaises(ImportError):
|
|
|
|
import_module("utils_tests.test_module.django")
|
2011-04-22 20:03:18 +08:00
|
|
|
|
2011-01-30 21:10:47 +08:00
|
|
|
# Don't be confused by caching of import misses
|
2013-10-18 19:25:30 +08:00
|
|
|
import types # NOQA: causes attempted import of utils_tests.types
|
2022-02-04 03:24:19 +08:00
|
|
|
|
2013-02-26 20:19:18 +08:00
|
|
|
self.assertFalse(module_has_submodule(sys.modules["utils_tests"], "types"))
|
2011-01-30 21:10:47 +08:00
|
|
|
|
2011-08-28 10:37:16 +08:00
|
|
|
# A module which doesn't have a __path__ (so no submodules)
|
|
|
|
self.assertFalse(module_has_submodule(test_no_submodule, "anything"))
|
2016-01-17 19:26:39 +08:00
|
|
|
with self.assertRaises(ImportError):
|
|
|
|
import_module("utils_tests.test_no_submodule.anything")
|
2011-08-28 10:37:16 +08:00
|
|
|
|
2017-06-09 02:34:20 +08:00
|
|
|
def test_has_sumbodule_with_dotted_path(self):
|
|
|
|
"""Nested module existence can be tested."""
|
|
|
|
test_module = import_module("utils_tests.test_module")
|
|
|
|
# A grandchild that exists.
|
|
|
|
self.assertIs(
|
|
|
|
module_has_submodule(test_module, "child_module.grandchild_module"), True
|
|
|
|
)
|
|
|
|
# A grandchild that doesn't exist.
|
|
|
|
self.assertIs(
|
|
|
|
module_has_submodule(test_module, "child_module.no_such_module"), False
|
|
|
|
)
|
|
|
|
# A grandchild whose parent doesn't exist.
|
|
|
|
self.assertIs(
|
|
|
|
module_has_submodule(test_module, "no_such_module.grandchild_module"), False
|
|
|
|
)
|
|
|
|
# A grandchild whose parent is not a package.
|
|
|
|
self.assertIs(
|
|
|
|
module_has_submodule(test_module, "good_module.no_such_module"), False
|
|
|
|
)
|
|
|
|
|
2013-11-03 05:34:05 +08:00
|
|
|
|
2010-09-28 16:18:12 +08:00
|
|
|
class EggLoader(unittest.TestCase):
|
2010-05-04 14:14:47 +08:00
|
|
|
def setUp(self):
|
2017-01-20 21:01:02 +08:00
|
|
|
self.egg_dir = "%s/eggs" % os.path.dirname(__file__)
|
2010-05-04 14:14:47 +08:00
|
|
|
|
|
|
|
def tearDown(self):
|
2010-05-04 22:48:43 +08:00
|
|
|
sys.path_importer_cache.clear()
|
|
|
|
|
|
|
|
sys.modules.pop("egg_module.sub1.sub2.bad_module", None)
|
|
|
|
sys.modules.pop("egg_module.sub1.sub2.good_module", None)
|
|
|
|
sys.modules.pop("egg_module.sub1.sub2", None)
|
|
|
|
sys.modules.pop("egg_module.sub1", None)
|
|
|
|
sys.modules.pop("egg_module.bad_module", None)
|
|
|
|
sys.modules.pop("egg_module.good_module", None)
|
|
|
|
sys.modules.pop("egg_module", None)
|
2010-05-04 14:14:47 +08:00
|
|
|
|
|
|
|
def test_shallow_loader(self):
|
|
|
|
"Module existence can be tested inside eggs"
|
|
|
|
egg_name = "%s/test_egg.egg" % self.egg_dir
|
2014-01-26 13:50:40 +08:00
|
|
|
with extend_sys_path(egg_name):
|
|
|
|
egg_module = import_module("egg_module")
|
2010-05-04 14:14:47 +08:00
|
|
|
|
2014-01-26 13:50:40 +08:00
|
|
|
# An importable child
|
|
|
|
self.assertTrue(module_has_submodule(egg_module, "good_module"))
|
|
|
|
mod = import_module("egg_module.good_module")
|
|
|
|
self.assertEqual(mod.content, "Good Module")
|
2010-05-04 14:14:47 +08:00
|
|
|
|
2014-01-26 13:50:40 +08:00
|
|
|
# A child that exists, but will generate an import error if loaded
|
|
|
|
self.assertTrue(module_has_submodule(egg_module, "bad_module"))
|
2016-01-17 19:26:39 +08:00
|
|
|
with self.assertRaises(ImportError):
|
|
|
|
import_module("egg_module.bad_module")
|
2010-05-04 14:14:47 +08:00
|
|
|
|
2014-01-26 13:50:40 +08:00
|
|
|
# A child that doesn't exist
|
|
|
|
self.assertFalse(module_has_submodule(egg_module, "no_such_module"))
|
2016-01-17 19:26:39 +08:00
|
|
|
with self.assertRaises(ImportError):
|
|
|
|
import_module("egg_module.no_such_module")
|
2010-05-04 14:14:47 +08:00
|
|
|
|
|
|
|
def test_deep_loader(self):
|
|
|
|
"Modules deep inside an egg can still be tested for existence"
|
|
|
|
egg_name = "%s/test_egg.egg" % self.egg_dir
|
2014-01-26 13:50:40 +08:00
|
|
|
with extend_sys_path(egg_name):
|
|
|
|
egg_module = import_module("egg_module.sub1.sub2")
|
2010-05-04 14:14:47 +08:00
|
|
|
|
2014-01-26 13:50:40 +08:00
|
|
|
# An importable child
|
|
|
|
self.assertTrue(module_has_submodule(egg_module, "good_module"))
|
|
|
|
mod = import_module("egg_module.sub1.sub2.good_module")
|
|
|
|
self.assertEqual(mod.content, "Deep Good Module")
|
2010-05-04 14:14:47 +08:00
|
|
|
|
2014-01-26 13:50:40 +08:00
|
|
|
# A child that exists, but will generate an import error if loaded
|
|
|
|
self.assertTrue(module_has_submodule(egg_module, "bad_module"))
|
2016-01-17 19:26:39 +08:00
|
|
|
with self.assertRaises(ImportError):
|
|
|
|
import_module("egg_module.sub1.sub2.bad_module")
|
2010-05-04 14:14:47 +08:00
|
|
|
|
2014-01-26 13:50:40 +08:00
|
|
|
# A child that doesn't exist
|
|
|
|
self.assertFalse(module_has_submodule(egg_module, "no_such_module"))
|
2016-01-17 19:26:39 +08:00
|
|
|
with self.assertRaises(ImportError):
|
|
|
|
import_module("egg_module.sub1.sub2.no_such_module")
|
2010-05-04 14:14:47 +08:00
|
|
|
|
2013-02-03 05:58:02 +08:00
|
|
|
|
2018-11-27 03:05:02 +08:00
|
|
|
class ModuleImportTests(SimpleTestCase):
|
2014-01-21 04:15:14 +08:00
|
|
|
def test_import_string(self):
|
|
|
|
cls = import_string("django.utils.module_loading.import_string")
|
|
|
|
self.assertEqual(cls, import_string)
|
|
|
|
|
|
|
|
# Test exceptions raised
|
2016-01-17 19:26:39 +08:00
|
|
|
with self.assertRaises(ImportError):
|
|
|
|
import_string("no_dots_in_path")
|
2015-06-03 23:05:03 +08:00
|
|
|
msg = 'Module "utils_tests" does not define a "unexistent" attribute'
|
2016-01-18 16:45:45 +08:00
|
|
|
with self.assertRaisesMessage(ImportError, msg):
|
2015-06-03 23:05:03 +08:00
|
|
|
import_string("utils_tests.unexistent")
|
2014-01-21 04:15:14 +08:00
|
|
|
|
2013-11-03 05:34:05 +08:00
|
|
|
|
2013-12-23 17:37:34 +08:00
|
|
|
@modify_settings(INSTALLED_APPS={"append": "utils_tests.test_module"})
|
2013-09-07 07:23:25 +08:00
|
|
|
class AutodiscoverModulesTestCase(SimpleTestCase):
|
2014-10-31 17:22:15 +08:00
|
|
|
def tearDown(self):
|
|
|
|
sys.path_importer_cache.clear()
|
|
|
|
|
|
|
|
sys.modules.pop("utils_tests.test_module.another_bad_module", None)
|
|
|
|
sys.modules.pop("utils_tests.test_module.another_good_module", None)
|
|
|
|
sys.modules.pop("utils_tests.test_module.bad_module", None)
|
|
|
|
sys.modules.pop("utils_tests.test_module.good_module", None)
|
|
|
|
sys.modules.pop("utils_tests.test_module", None)
|
|
|
|
|
2013-09-07 07:23:25 +08:00
|
|
|
def test_autodiscover_modules_found(self):
|
|
|
|
autodiscover_modules("good_module")
|
|
|
|
|
|
|
|
def test_autodiscover_modules_not_found(self):
|
|
|
|
autodiscover_modules("missing_module")
|
|
|
|
|
|
|
|
def test_autodiscover_modules_found_but_bad_module(self):
|
2017-01-20 10:10:33 +08:00
|
|
|
with self.assertRaisesMessage(
|
|
|
|
ImportError, "No module named 'a_package_name_that_does_not_exist'"
|
|
|
|
):
|
2013-09-07 07:23:25 +08:00
|
|
|
autodiscover_modules("bad_module")
|
|
|
|
|
|
|
|
def test_autodiscover_modules_several_one_bad_module(self):
|
2017-01-20 10:10:33 +08:00
|
|
|
with self.assertRaisesMessage(
|
|
|
|
ImportError, "No module named 'a_package_name_that_does_not_exist'"
|
|
|
|
):
|
2013-09-07 07:23:25 +08:00
|
|
|
autodiscover_modules("good_module", "bad_module")
|
|
|
|
|
|
|
|
def test_autodiscover_modules_several_found(self):
|
|
|
|
autodiscover_modules("good_module", "another_good_module")
|
|
|
|
|
2014-10-31 17:22:15 +08:00
|
|
|
def test_autodiscover_modules_several_found_with_registry(self):
|
|
|
|
from .test_module import site
|
2022-02-04 03:24:19 +08:00
|
|
|
|
2014-10-31 17:22:15 +08:00
|
|
|
autodiscover_modules("good_module", "another_good_module", register_to=site)
|
|
|
|
self.assertEqual(site._registry, {"lorem": "ipsum"})
|
|
|
|
|
2013-09-07 07:23:25 +08:00
|
|
|
def test_validate_registry_keeps_intact(self):
|
|
|
|
from .test_module import site
|
2022-02-04 03:24:19 +08:00
|
|
|
|
2016-01-18 16:45:45 +08:00
|
|
|
with self.assertRaisesMessage(Exception, "Some random exception."):
|
2013-09-07 07:23:25 +08:00
|
|
|
autodiscover_modules("another_bad_module", register_to=site)
|
|
|
|
self.assertEqual(site._registry, {})
|
|
|
|
|
2014-10-31 17:22:15 +08:00
|
|
|
def test_validate_registry_resets_after_erroneous_module(self):
|
|
|
|
from .test_module import site
|
2022-02-04 03:24:19 +08:00
|
|
|
|
2016-01-18 16:45:45 +08:00
|
|
|
with self.assertRaisesMessage(Exception, "Some random exception."):
|
2014-10-31 17:22:15 +08:00
|
|
|
autodiscover_modules(
|
|
|
|
"another_good_module", "another_bad_module", register_to=site
|
|
|
|
)
|
|
|
|
self.assertEqual(site._registry, {"lorem": "ipsum"})
|
|
|
|
|
|
|
|
def test_validate_registry_resets_after_missing_module(self):
|
|
|
|
from .test_module import site
|
2022-02-04 03:24:19 +08:00
|
|
|
|
2014-10-31 17:22:15 +08:00
|
|
|
autodiscover_modules(
|
|
|
|
"does_not_exist", "another_good_module", "does_not_exist2", register_to=site
|
|
|
|
)
|
|
|
|
self.assertEqual(site._registry, {"lorem": "ipsum"})
|
|
|
|
|
2013-02-03 05:58:02 +08:00
|
|
|
|
2023-01-18 16:46:01 +08:00
|
|
|
class TestFinder:
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
self.importer = zipimporter(*args, **kwargs)
|
2022-02-04 03:24:19 +08:00
|
|
|
|
2023-01-18 16:46:01 +08:00
|
|
|
def find_spec(self, path, target=None):
|
|
|
|
return self.importer.find_spec(path, target)
|
2010-05-04 14:14:47 +08:00
|
|
|
|
2013-11-03 05:34:05 +08:00
|
|
|
|
2010-05-04 22:48:43 +08:00
|
|
|
class CustomLoader(EggLoader):
|
|
|
|
"""The Custom Loader test is exactly the same as the EggLoader, but
|
2021-09-17 02:20:54 +08:00
|
|
|
it uses a custom defined Loader class. Although the EggLoader combines both
|
|
|
|
functions into one class, this isn't required.
|
2010-05-04 22:48:43 +08:00
|
|
|
"""
|
2022-02-04 03:24:19 +08:00
|
|
|
|
2010-05-04 14:14:47 +08:00
|
|
|
def setUp(self):
|
2017-01-21 21:13:44 +08:00
|
|
|
super().setUp()
|
2010-05-04 14:14:47 +08:00
|
|
|
sys.path_hooks.insert(0, TestFinder)
|
|
|
|
sys.path_importer_cache.clear()
|
|
|
|
|
|
|
|
def tearDown(self):
|
2017-01-21 21:13:44 +08:00
|
|
|
super().tearDown()
|
2010-05-04 14:14:47 +08:00
|
|
|
sys.path_hooks.pop(0)
|