2014-01-25 19:37:05 -07:00
|
|
|
import os
|
2023-06-01 16:39:52 +02:00
|
|
|
from unittest.mock import patch
|
2014-01-25 19:37:05 -07:00
|
|
|
|
2023-06-01 16:39:52 +02:00
|
|
|
import django
|
2015-01-28 07:35:27 -05:00
|
|
|
from django.apps import AppConfig, apps
|
2013-12-24 12:25:17 +01:00
|
|
|
from django.apps.registry import Apps
|
2014-01-26 12:46:28 +01:00
|
|
|
from django.contrib.admin.models import LogEntry
|
2015-02-08 13:38:09 -05:00
|
|
|
from django.core.exceptions import AppRegistryNotReady, ImproperlyConfigured
|
2023-06-01 16:39:52 +02:00
|
|
|
from django.db import connections, models
|
|
|
|
from django.test import (
|
|
|
|
SimpleTestCase,
|
|
|
|
TransactionTestCase,
|
|
|
|
override_settings,
|
|
|
|
skipUnlessDBFeature,
|
|
|
|
)
|
2015-11-17 00:39:28 -05:00
|
|
|
from django.test.utils import extend_sys_path, isolate_apps
|
2021-09-16 09:06:40 +02:00
|
|
|
|
2015-01-28 07:35:27 -05:00
|
|
|
from .models import SoAlternative, TotallyNormal, new_apps
|
2020-07-21 10:35:12 +02:00
|
|
|
from .one_config_app.apps import OneConfig
|
|
|
|
from .two_configs_one_default_app.apps import TwoConfig
|
2013-12-24 12:25:17 +01:00
|
|
|
|
2013-12-26 18:03:17 +01:00
|
|
|
# Small list with a variety of cases for tests that iterate on installed apps.
|
|
|
|
# Intentionally not in alphabetical order to check if the order is preserved.
|
|
|
|
|
|
|
|
SOME_INSTALLED_APPS = [
|
2022-02-03 20:24:19 +01:00
|
|
|
"apps.apps.MyAdmin",
|
|
|
|
"apps.apps.MyAuth",
|
|
|
|
"django.contrib.contenttypes",
|
|
|
|
"django.contrib.sessions",
|
|
|
|
"django.contrib.messages",
|
|
|
|
"django.contrib.staticfiles",
|
2013-12-26 18:03:17 +01:00
|
|
|
]
|
|
|
|
|
|
|
|
SOME_INSTALLED_APPS_NAMES = [
|
2022-02-03 20:24:19 +01:00
|
|
|
"django.contrib.admin",
|
|
|
|
"django.contrib.auth",
|
2013-12-26 18:03:17 +01:00
|
|
|
] + SOME_INSTALLED_APPS[2:]
|
|
|
|
|
2017-01-20 08:01:02 -05:00
|
|
|
HERE = os.path.dirname(__file__)
|
2014-01-25 19:37:05 -07:00
|
|
|
|
2013-12-26 19:20:38 +01:00
|
|
|
|
2015-04-17 17:38:20 -04:00
|
|
|
class AppsTests(SimpleTestCase):
|
2021-11-22 10:47:38 +00:00
|
|
|
def test_singleton_main(self):
|
2013-12-26 15:04:58 +01:00
|
|
|
"""
|
2021-11-22 10:47:38 +00:00
|
|
|
Only one main registry can exist.
|
2013-12-26 15:04:58 +01:00
|
|
|
"""
|
|
|
|
with self.assertRaises(RuntimeError):
|
2013-12-30 16:03:06 +01:00
|
|
|
Apps(installed_apps=None)
|
2013-12-26 15:04:58 +01:00
|
|
|
|
|
|
|
def test_ready(self):
|
|
|
|
"""
|
2021-11-22 10:47:38 +00:00
|
|
|
Tests the ready property of the main registry.
|
2013-12-26 15:04:58 +01:00
|
|
|
"""
|
2021-11-22 10:47:38 +00:00
|
|
|
# The main app registry is always ready when the tests run.
|
2018-01-30 16:44:58 +01:00
|
|
|
self.assertIs(apps.ready, True)
|
2021-11-22 10:47:38 +00:00
|
|
|
# Non-main app registries are populated in __init__.
|
2018-01-30 16:44:58 +01:00
|
|
|
self.assertIs(Apps().ready, True)
|
2019-01-14 01:33:47 +00:00
|
|
|
# The condition is set when apps are ready
|
|
|
|
self.assertIs(apps.ready_event.is_set(), True)
|
|
|
|
self.assertIs(Apps().ready_event.is_set(), True)
|
2013-12-26 15:04:58 +01:00
|
|
|
|
2013-12-26 19:20:38 +01:00
|
|
|
def test_bad_app_config(self):
|
|
|
|
"""
|
|
|
|
Tests when INSTALLED_APPS contains an incorrect app config.
|
|
|
|
"""
|
2017-05-28 21:37:21 +02:00
|
|
|
msg = "'apps.apps.BadConfig' must supply a name attribute."
|
|
|
|
with self.assertRaisesMessage(ImproperlyConfigured, msg):
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.settings(INSTALLED_APPS=["apps.apps.BadConfig"]):
|
2013-12-26 19:20:38 +01:00
|
|
|
pass
|
|
|
|
|
|
|
|
def test_not_an_app_config(self):
|
|
|
|
"""
|
|
|
|
Tests when INSTALLED_APPS contains a class that isn't an app config.
|
|
|
|
"""
|
2017-05-28 21:37:21 +02:00
|
|
|
msg = "'apps.apps.NotAConfig' isn't a subclass of AppConfig."
|
|
|
|
with self.assertRaisesMessage(ImproperlyConfigured, msg):
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.settings(INSTALLED_APPS=["apps.apps.NotAConfig"]):
|
2013-12-26 19:20:38 +01:00
|
|
|
pass
|
|
|
|
|
|
|
|
def test_no_such_app(self):
|
|
|
|
"""
|
2013-12-26 19:29:32 +01:00
|
|
|
Tests when INSTALLED_APPS contains an app that doesn't exist, either
|
|
|
|
directly or via an app config.
|
2013-12-26 19:20:38 +01:00
|
|
|
"""
|
2013-12-26 19:29:32 +01:00
|
|
|
with self.assertRaises(ImportError):
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.settings(INSTALLED_APPS=["there is no such app"]):
|
2013-12-26 19:29:32 +01:00
|
|
|
pass
|
2022-02-04 08:08:27 +01:00
|
|
|
msg = (
|
|
|
|
"Cannot import 'there is no such app'. Check that "
|
|
|
|
"'apps.apps.NoSuchApp.name' is correct."
|
|
|
|
)
|
2016-06-01 13:08:59 -07:00
|
|
|
with self.assertRaisesMessage(ImproperlyConfigured, msg):
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.settings(INSTALLED_APPS=["apps.apps.NoSuchApp"]):
|
2013-12-26 19:20:38 +01:00
|
|
|
pass
|
|
|
|
|
|
|
|
def test_no_such_app_config(self):
|
2020-07-21 10:35:12 +02:00
|
|
|
msg = "Module 'apps' does not contain a 'NoSuchConfig' class."
|
2018-09-18 18:19:18 +02:00
|
|
|
with self.assertRaisesMessage(ImportError, msg):
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.settings(INSTALLED_APPS=["apps.NoSuchConfig"]):
|
2018-09-18 18:19:18 +02:00
|
|
|
pass
|
|
|
|
|
|
|
|
def test_no_such_app_config_with_choices(self):
|
|
|
|
msg = (
|
2020-07-21 10:35:12 +02:00
|
|
|
"Module 'apps.apps' does not contain a 'NoSuchConfig' class. "
|
2020-07-12 13:59:57 +01:00
|
|
|
"Choices are: 'BadConfig', 'ModelPKAppsConfig', 'MyAdmin', "
|
|
|
|
"'MyAuth', 'NoSuchApp', 'PlainAppsConfig', 'RelabeledAppsConfig'."
|
2018-09-18 18:19:18 +02:00
|
|
|
)
|
2020-07-21 10:35:12 +02:00
|
|
|
with self.assertRaisesMessage(ImportError, msg):
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.settings(INSTALLED_APPS=["apps.apps.NoSuchConfig"]):
|
2013-12-26 19:20:38 +01:00
|
|
|
pass
|
|
|
|
|
2020-07-21 10:35:12 +02:00
|
|
|
def test_no_config_app(self):
|
|
|
|
"""Load an app that doesn't provide an AppConfig class."""
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.settings(INSTALLED_APPS=["apps.no_config_app"]):
|
|
|
|
config = apps.get_app_config("no_config_app")
|
2020-07-21 10:35:12 +02:00
|
|
|
self.assertIsInstance(config, AppConfig)
|
|
|
|
|
|
|
|
def test_one_config_app(self):
|
|
|
|
"""Load an app that provides an AppConfig class."""
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.settings(INSTALLED_APPS=["apps.one_config_app"]):
|
|
|
|
config = apps.get_app_config("one_config_app")
|
2020-07-21 10:35:12 +02:00
|
|
|
self.assertIsInstance(config, OneConfig)
|
|
|
|
|
|
|
|
def test_two_configs_app(self):
|
|
|
|
"""Load an app that provides two AppConfig classes."""
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.settings(INSTALLED_APPS=["apps.two_configs_app"]):
|
|
|
|
config = apps.get_app_config("two_configs_app")
|
2020-07-21 10:35:12 +02:00
|
|
|
self.assertIsInstance(config, AppConfig)
|
|
|
|
|
|
|
|
def test_two_default_configs_app(self):
|
|
|
|
"""Load an app that provides two default AppConfig classes."""
|
|
|
|
msg = (
|
|
|
|
"'apps.two_default_configs_app.apps' declares more than one "
|
|
|
|
"default AppConfig: 'TwoConfig', 'TwoConfigBis'."
|
|
|
|
)
|
|
|
|
with self.assertRaisesMessage(RuntimeError, msg):
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.settings(INSTALLED_APPS=["apps.two_default_configs_app"]):
|
2020-07-21 10:35:12 +02:00
|
|
|
pass
|
|
|
|
|
|
|
|
def test_two_configs_one_default_app(self):
|
|
|
|
"""
|
|
|
|
Load an app that provides two AppConfig classes, one being the default.
|
|
|
|
"""
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.settings(INSTALLED_APPS=["apps.two_configs_one_default_app"]):
|
|
|
|
config = apps.get_app_config("two_configs_one_default_app")
|
2020-07-21 10:35:12 +02:00
|
|
|
self.assertIsInstance(config, TwoConfig)
|
2014-01-24 22:43:00 +01:00
|
|
|
|
2013-12-26 18:03:17 +01:00
|
|
|
@override_settings(INSTALLED_APPS=SOME_INSTALLED_APPS)
|
|
|
|
def test_get_app_configs(self):
|
|
|
|
"""
|
2014-01-26 12:46:28 +01:00
|
|
|
Tests apps.get_app_configs().
|
2013-12-26 18:03:17 +01:00
|
|
|
"""
|
|
|
|
app_configs = apps.get_app_configs()
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertEqual(
|
|
|
|
[app_config.name for app_config in app_configs], SOME_INSTALLED_APPS_NAMES
|
|
|
|
)
|
2013-12-26 18:03:17 +01:00
|
|
|
|
|
|
|
@override_settings(INSTALLED_APPS=SOME_INSTALLED_APPS)
|
|
|
|
def test_get_app_config(self):
|
|
|
|
"""
|
2014-01-26 12:46:28 +01:00
|
|
|
Tests apps.get_app_config().
|
2013-12-26 18:03:17 +01:00
|
|
|
"""
|
2022-02-03 20:24:19 +01:00
|
|
|
app_config = apps.get_app_config("admin")
|
|
|
|
self.assertEqual(app_config.name, "django.contrib.admin")
|
2013-12-26 18:03:17 +01:00
|
|
|
|
2022-02-03 20:24:19 +01:00
|
|
|
app_config = apps.get_app_config("staticfiles")
|
|
|
|
self.assertEqual(app_config.name, "django.contrib.staticfiles")
|
2013-12-26 18:03:17 +01:00
|
|
|
|
|
|
|
with self.assertRaises(LookupError):
|
2022-02-03 20:24:19 +01:00
|
|
|
apps.get_app_config("admindocs")
|
2013-12-26 18:03:17 +01:00
|
|
|
|
2015-05-12 10:02:23 +01:00
|
|
|
msg = "No installed app with label 'django.contrib.auth'. Did you mean 'myauth'"
|
|
|
|
with self.assertRaisesMessage(LookupError, msg):
|
2022-02-03 20:24:19 +01:00
|
|
|
apps.get_app_config("django.contrib.auth")
|
2015-05-12 10:02:23 +01:00
|
|
|
|
2013-12-26 20:04:06 +01:00
|
|
|
@override_settings(INSTALLED_APPS=SOME_INSTALLED_APPS)
|
2014-01-06 22:48:41 +01:00
|
|
|
def test_is_installed(self):
|
2014-01-26 12:46:28 +01:00
|
|
|
"""
|
|
|
|
Tests apps.is_installed().
|
|
|
|
"""
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertIs(apps.is_installed("django.contrib.admin"), True)
|
|
|
|
self.assertIs(apps.is_installed("django.contrib.auth"), True)
|
|
|
|
self.assertIs(apps.is_installed("django.contrib.staticfiles"), True)
|
|
|
|
self.assertIs(apps.is_installed("django.contrib.admindocs"), False)
|
2013-12-26 20:04:06 +01:00
|
|
|
|
2014-01-26 12:46:28 +01:00
|
|
|
@override_settings(INSTALLED_APPS=SOME_INSTALLED_APPS)
|
|
|
|
def test_get_model(self):
|
|
|
|
"""
|
|
|
|
Tests apps.get_model().
|
|
|
|
"""
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertEqual(apps.get_model("admin", "LogEntry"), LogEntry)
|
2014-01-26 12:46:28 +01:00
|
|
|
with self.assertRaises(LookupError):
|
2022-02-03 20:24:19 +01:00
|
|
|
apps.get_model("admin", "LogExit")
|
2014-01-26 12:46:28 +01:00
|
|
|
|
|
|
|
# App label is case-sensitive, Model name is case-insensitive.
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertEqual(apps.get_model("admin", "loGentrY"), LogEntry)
|
2014-01-26 12:46:28 +01:00
|
|
|
with self.assertRaises(LookupError):
|
2022-02-03 20:24:19 +01:00
|
|
|
apps.get_model("Admin", "LogEntry")
|
2014-01-26 12:46:28 +01:00
|
|
|
|
|
|
|
# A single argument is accepted.
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertEqual(apps.get_model("admin.LogEntry"), LogEntry)
|
2014-01-26 12:46:28 +01:00
|
|
|
with self.assertRaises(LookupError):
|
2022-02-03 20:24:19 +01:00
|
|
|
apps.get_model("admin.LogExit")
|
2014-01-26 12:46:28 +01:00
|
|
|
with self.assertRaises(ValueError):
|
2022-02-03 20:24:19 +01:00
|
|
|
apps.get_model("admin_LogEntry")
|
2014-01-26 12:46:28 +01:00
|
|
|
|
2023-06-29 16:13:14 +03:00
|
|
|
@override_settings(INSTALLED_APPS=SOME_INSTALLED_APPS)
|
|
|
|
def test_clear_cache(self):
|
|
|
|
# Set cache.
|
|
|
|
self.assertIsNone(apps.get_swappable_settings_name("admin.LogEntry"))
|
|
|
|
apps.get_models()
|
|
|
|
|
|
|
|
apps.clear_cache()
|
|
|
|
|
|
|
|
self.assertEqual(apps.get_swappable_settings_name.cache_info().currsize, 0)
|
|
|
|
self.assertEqual(apps.get_models.cache_info().currsize, 0)
|
|
|
|
|
2022-02-03 20:24:19 +01:00
|
|
|
@override_settings(INSTALLED_APPS=["apps.apps.RelabeledAppsConfig"])
|
2013-12-31 16:23:42 +01:00
|
|
|
def test_relabeling(self):
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertEqual(apps.get_app_config("relabeled").name, "apps")
|
2013-12-31 16:23:42 +01:00
|
|
|
|
2013-12-31 17:25:57 +01:00
|
|
|
def test_duplicate_labels(self):
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.assertRaisesMessage(
|
|
|
|
ImproperlyConfigured, "Application labels aren't unique"
|
|
|
|
):
|
|
|
|
with self.settings(INSTALLED_APPS=["apps.apps.PlainAppsConfig", "apps"]):
|
2013-12-31 17:25:57 +01:00
|
|
|
pass
|
|
|
|
|
|
|
|
def test_duplicate_names(self):
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.assertRaisesMessage(
|
|
|
|
ImproperlyConfigured, "Application names aren't unique"
|
|
|
|
):
|
|
|
|
with self.settings(
|
|
|
|
INSTALLED_APPS=["apps.apps.RelabeledAppsConfig", "apps"]
|
|
|
|
):
|
2013-12-31 17:25:57 +01:00
|
|
|
pass
|
|
|
|
|
2014-08-31 18:51:55 +02:00
|
|
|
def test_import_exception_is_not_masked(self):
|
|
|
|
"""
|
|
|
|
App discovery should preserve stack traces. Regression test for #22920.
|
|
|
|
"""
|
2016-01-18 12:15:45 +03:30
|
|
|
with self.assertRaisesMessage(ImportError, "Oops"):
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.settings(INSTALLED_APPS=["import_error_package"]):
|
2014-08-31 18:51:55 +02:00
|
|
|
pass
|
|
|
|
|
2013-12-24 12:25:17 +01:00
|
|
|
def test_models_py(self):
|
|
|
|
"""
|
2016-10-27 14:53:39 +07:00
|
|
|
The models in the models.py file were loaded correctly.
|
2013-12-24 12:25:17 +01:00
|
|
|
"""
|
|
|
|
self.assertEqual(apps.get_model("apps", "TotallyNormal"), TotallyNormal)
|
2013-12-28 14:55:54 +01:00
|
|
|
with self.assertRaises(LookupError):
|
|
|
|
apps.get_model("apps", "SoAlternative")
|
2013-12-24 12:25:17 +01:00
|
|
|
|
2013-12-28 14:55:54 +01:00
|
|
|
with self.assertRaises(LookupError):
|
|
|
|
new_apps.get_model("apps", "TotallyNormal")
|
2013-12-24 12:25:17 +01:00
|
|
|
self.assertEqual(new_apps.get_model("apps", "SoAlternative"), SoAlternative)
|
|
|
|
|
2018-01-30 16:44:58 +01:00
|
|
|
def test_models_not_loaded(self):
|
|
|
|
"""
|
|
|
|
apps.get_models() raises an exception if apps.models_ready isn't True.
|
|
|
|
"""
|
|
|
|
apps.models_ready = False
|
|
|
|
try:
|
|
|
|
# The cache must be cleared to trigger the exception.
|
|
|
|
apps.get_models.cache_clear()
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.assertRaisesMessage(
|
|
|
|
AppRegistryNotReady, "Models aren't loaded yet."
|
|
|
|
):
|
2018-01-30 16:44:58 +01:00
|
|
|
apps.get_models()
|
|
|
|
finally:
|
|
|
|
apps.models_ready = True
|
|
|
|
|
2013-12-24 12:25:17 +01:00
|
|
|
def test_dynamic_load(self):
|
|
|
|
"""
|
|
|
|
Makes a new model at runtime and ensures it goes into the right place.
|
|
|
|
"""
|
2013-12-29 21:47:55 +01:00
|
|
|
old_models = list(apps.get_app_config("apps").get_models())
|
2013-12-24 12:25:17 +01:00
|
|
|
# Construct a new model in a new app registry
|
|
|
|
body = {}
|
2013-12-30 16:03:06 +01:00
|
|
|
new_apps = Apps(["apps"])
|
2013-12-24 12:25:17 +01:00
|
|
|
meta_contents = {
|
2022-02-03 20:24:19 +01:00
|
|
|
"app_label": "apps",
|
|
|
|
"apps": new_apps,
|
2013-12-24 12:25:17 +01:00
|
|
|
}
|
2017-06-01 16:08:59 -07:00
|
|
|
meta = type("Meta", (), meta_contents)
|
2022-02-03 20:24:19 +01:00
|
|
|
body["Meta"] = meta
|
|
|
|
body["__module__"] = TotallyNormal.__module__
|
2017-01-19 09:48:01 -05:00
|
|
|
temp_model = type("SouthPonies", (models.Model,), body)
|
2013-12-24 12:25:17 +01:00
|
|
|
# Make sure it appeared in the right place!
|
2017-03-17 07:51:48 -04:00
|
|
|
self.assertEqual(list(apps.get_app_config("apps").get_models()), old_models)
|
2013-12-28 14:55:54 +01:00
|
|
|
with self.assertRaises(LookupError):
|
|
|
|
apps.get_model("apps", "SouthPonies")
|
2013-12-24 12:25:17 +01:00
|
|
|
self.assertEqual(new_apps.get_model("apps", "SouthPonies"), temp_model)
|
2014-01-25 19:37:05 -07:00
|
|
|
|
2014-10-22 23:16:32 +07:00
|
|
|
def test_model_clash(self):
|
|
|
|
"""
|
|
|
|
Test for behavior when two models clash in the app registry.
|
|
|
|
"""
|
|
|
|
new_apps = Apps(["apps"])
|
|
|
|
meta_contents = {
|
2022-02-03 20:24:19 +01:00
|
|
|
"app_label": "apps",
|
|
|
|
"apps": new_apps,
|
2014-10-22 23:16:32 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
body = {}
|
2022-02-03 20:24:19 +01:00
|
|
|
body["Meta"] = type("Meta", (), meta_contents)
|
|
|
|
body["__module__"] = TotallyNormal.__module__
|
2017-01-19 09:48:01 -05:00
|
|
|
type("SouthPonies", (models.Model,), body)
|
2014-10-22 23:16:32 +07:00
|
|
|
|
|
|
|
# When __name__ and __module__ match we assume the module
|
|
|
|
# was reloaded and issue a warning. This use-case is
|
|
|
|
# useful for REPL. Refs #23621.
|
|
|
|
body = {}
|
2022-02-03 20:24:19 +01:00
|
|
|
body["Meta"] = type("Meta", (), meta_contents)
|
|
|
|
body["__module__"] = TotallyNormal.__module__
|
2015-11-13 15:54:05 -05:00
|
|
|
msg = (
|
|
|
|
"Model 'apps.southponies' was already registered. "
|
|
|
|
"Reloading models is not advised as it can lead to inconsistencies, "
|
|
|
|
"most notably with related models."
|
|
|
|
)
|
|
|
|
with self.assertRaisesMessage(RuntimeWarning, msg):
|
2017-01-19 09:48:01 -05:00
|
|
|
type("SouthPonies", (models.Model,), body)
|
2014-10-22 23:16:32 +07:00
|
|
|
|
|
|
|
# If it doesn't appear to be a reloaded module then we expect
|
|
|
|
# a RuntimeError.
|
|
|
|
body = {}
|
2022-02-03 20:24:19 +01:00
|
|
|
body["Meta"] = type("Meta", (), meta_contents)
|
|
|
|
body["__module__"] = TotallyNormal.__module__ + ".whatever"
|
|
|
|
with self.assertRaisesMessage(
|
|
|
|
RuntimeError, "Conflicting 'southponies' models in application 'apps':"
|
|
|
|
):
|
2017-01-19 09:48:01 -05:00
|
|
|
type("SouthPonies", (models.Model,), body)
|
2014-01-25 19:37:05 -07:00
|
|
|
|
2015-02-08 13:38:09 -05:00
|
|
|
def test_get_containing_app_config_apps_not_ready(self):
|
|
|
|
"""
|
|
|
|
apps.get_containing_app_config() should raise an exception if
|
|
|
|
apps.apps_ready isn't True.
|
|
|
|
"""
|
|
|
|
apps.apps_ready = False
|
|
|
|
try:
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.assertRaisesMessage(
|
|
|
|
AppRegistryNotReady, "Apps aren't loaded yet"
|
|
|
|
):
|
|
|
|
apps.get_containing_app_config("foo")
|
2015-02-08 13:38:09 -05:00
|
|
|
finally:
|
|
|
|
apps.apps_ready = True
|
|
|
|
|
2022-02-03 20:24:19 +01:00
|
|
|
@isolate_apps("apps", kwarg_name="apps")
|
2015-11-17 00:39:28 -05:00
|
|
|
def test_lazy_model_operation(self, apps):
|
2015-03-03 16:43:56 +08:00
|
|
|
"""
|
|
|
|
Tests apps.lazy_model_operation().
|
|
|
|
"""
|
|
|
|
model_classes = []
|
|
|
|
initial_pending = set(apps._pending_operations)
|
|
|
|
|
|
|
|
def test_func(*models):
|
|
|
|
model_classes[:] = models
|
|
|
|
|
|
|
|
class LazyA(models.Model):
|
|
|
|
pass
|
|
|
|
|
|
|
|
# Test models appearing twice, and models appearing consecutively
|
2022-02-03 20:24:19 +01:00
|
|
|
model_keys = [
|
|
|
|
("apps", model_name)
|
|
|
|
for model_name in ["lazya", "lazyb", "lazyb", "lazyc", "lazya"]
|
|
|
|
]
|
2015-03-03 16:43:56 +08:00
|
|
|
apps.lazy_model_operation(test_func, *model_keys)
|
|
|
|
|
|
|
|
# LazyModelA shouldn't be waited on since it's already registered,
|
|
|
|
# and LazyModelC shouldn't be waited on until LazyModelB exists.
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertEqual(
|
|
|
|
set(apps._pending_operations) - initial_pending, {("apps", "lazyb")}
|
|
|
|
)
|
2015-03-03 16:43:56 +08:00
|
|
|
|
2016-10-27 14:53:39 +07:00
|
|
|
# Multiple operations can wait on the same model
|
2022-02-03 20:24:19 +01:00
|
|
|
apps.lazy_model_operation(test_func, ("apps", "lazyb"))
|
2015-03-03 16:43:56 +08:00
|
|
|
|
|
|
|
class LazyB(models.Model):
|
|
|
|
pass
|
|
|
|
|
2017-03-17 07:51:48 -04:00
|
|
|
self.assertEqual(model_classes, [LazyB])
|
2015-03-03 16:43:56 +08:00
|
|
|
|
|
|
|
# Now we are just waiting on LazyModelC.
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertEqual(
|
|
|
|
set(apps._pending_operations) - initial_pending, {("apps", "lazyc")}
|
|
|
|
)
|
2015-03-03 16:43:56 +08:00
|
|
|
|
|
|
|
class LazyC(models.Model):
|
|
|
|
pass
|
|
|
|
|
|
|
|
# Everything should be loaded - make sure the callback was executed properly.
|
2017-03-17 07:51:48 -04:00
|
|
|
self.assertEqual(model_classes, [LazyA, LazyB, LazyB, LazyC, LazyA])
|
2015-03-03 16:43:56 +08:00
|
|
|
|
2014-10-22 21:21:02 -04:00
|
|
|
|
2017-01-19 02:39:46 -05:00
|
|
|
class Stub:
|
2014-01-27 13:28:53 -07:00
|
|
|
def __init__(self, **kwargs):
|
|
|
|
self.__dict__.update(kwargs)
|
|
|
|
|
|
|
|
|
2015-04-17 17:38:20 -04:00
|
|
|
class AppConfigTests(SimpleTestCase):
|
2014-01-27 13:28:53 -07:00
|
|
|
"""Unit tests for AppConfig class."""
|
2022-02-03 20:24:19 +01:00
|
|
|
|
2014-01-27 13:28:53 -07:00
|
|
|
def test_path_set_explicitly(self):
|
|
|
|
"""If subclass sets path as class attr, no module attributes needed."""
|
2022-02-03 20:24:19 +01:00
|
|
|
|
2014-01-27 13:28:53 -07:00
|
|
|
class MyAppConfig(AppConfig):
|
2022-02-03 20:24:19 +01:00
|
|
|
path = "foo"
|
2014-01-27 13:28:53 -07:00
|
|
|
|
2022-02-03 20:24:19 +01:00
|
|
|
ac = MyAppConfig("label", Stub())
|
2014-01-27 13:28:53 -07:00
|
|
|
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertEqual(ac.path, "foo")
|
2014-01-27 13:28:53 -07:00
|
|
|
|
|
|
|
def test_explicit_path_overrides(self):
|
|
|
|
"""If path set as class attr, overrides __path__ and __file__."""
|
2022-02-03 20:24:19 +01:00
|
|
|
|
2014-01-27 13:28:53 -07:00
|
|
|
class MyAppConfig(AppConfig):
|
2022-02-03 20:24:19 +01:00
|
|
|
path = "foo"
|
2014-01-27 13:28:53 -07:00
|
|
|
|
2022-02-03 20:24:19 +01:00
|
|
|
ac = MyAppConfig("label", Stub(__path__=["a"], __file__="b/__init__.py"))
|
2014-01-27 13:28:53 -07:00
|
|
|
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertEqual(ac.path, "foo")
|
2014-01-27 13:28:53 -07:00
|
|
|
|
|
|
|
def test_dunder_path(self):
|
|
|
|
"""If single element in __path__, use it (in preference to __file__)."""
|
2022-02-03 20:24:19 +01:00
|
|
|
ac = AppConfig("label", Stub(__path__=["a"], __file__="b/__init__.py"))
|
2014-01-27 13:28:53 -07:00
|
|
|
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertEqual(ac.path, "a")
|
2014-01-27 13:28:53 -07:00
|
|
|
|
|
|
|
def test_no_dunder_path_fallback_to_dunder_file(self):
|
|
|
|
"""If there is no __path__ attr, use __file__."""
|
2022-02-03 20:24:19 +01:00
|
|
|
ac = AppConfig("label", Stub(__file__="b/__init__.py"))
|
2014-01-27 13:28:53 -07:00
|
|
|
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertEqual(ac.path, "b")
|
2014-01-27 13:28:53 -07:00
|
|
|
|
|
|
|
def test_empty_dunder_path_fallback_to_dunder_file(self):
|
|
|
|
"""If the __path__ attr is empty, use __file__ if set."""
|
2022-02-03 20:24:19 +01:00
|
|
|
ac = AppConfig("label", Stub(__path__=[], __file__="b/__init__.py"))
|
2014-01-27 13:28:53 -07:00
|
|
|
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertEqual(ac.path, "b")
|
2014-01-27 13:28:53 -07:00
|
|
|
|
|
|
|
def test_multiple_dunder_path_fallback_to_dunder_file(self):
|
|
|
|
"""If the __path__ attr is length>1, use __file__ if set."""
|
2022-02-03 20:24:19 +01:00
|
|
|
ac = AppConfig("label", Stub(__path__=["a", "b"], __file__="c/__init__.py"))
|
2014-01-27 13:28:53 -07:00
|
|
|
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertEqual(ac.path, "c")
|
2014-01-27 13:28:53 -07:00
|
|
|
|
|
|
|
def test_no_dunder_path_or_dunder_file(self):
|
|
|
|
"""If there is no __path__ or __file__, raise ImproperlyConfigured."""
|
|
|
|
with self.assertRaises(ImproperlyConfigured):
|
2022-02-03 20:24:19 +01:00
|
|
|
AppConfig("label", Stub())
|
2014-01-27 13:28:53 -07:00
|
|
|
|
|
|
|
def test_empty_dunder_path_no_dunder_file(self):
|
|
|
|
"""If the __path__ attr is empty and there is no __file__, raise."""
|
|
|
|
with self.assertRaises(ImproperlyConfigured):
|
2022-02-03 20:24:19 +01:00
|
|
|
AppConfig("label", Stub(__path__=[]))
|
2014-01-27 13:28:53 -07:00
|
|
|
|
|
|
|
def test_multiple_dunder_path_no_dunder_file(self):
|
|
|
|
"""If the __path__ attr is length>1 and there is no __file__, raise."""
|
|
|
|
with self.assertRaises(ImproperlyConfigured):
|
2022-02-03 20:24:19 +01:00
|
|
|
AppConfig("label", Stub(__path__=["a", "b"]))
|
2014-01-27 13:28:53 -07:00
|
|
|
|
2015-08-23 21:07:00 -03:00
|
|
|
def test_duplicate_dunder_path_no_dunder_file(self):
|
|
|
|
"""
|
|
|
|
If the __path__ attr contains duplicate paths and there is no
|
|
|
|
__file__, they duplicates should be deduplicated (#25246).
|
|
|
|
"""
|
2022-02-03 20:24:19 +01:00
|
|
|
ac = AppConfig("label", Stub(__path__=["a", "a"]))
|
|
|
|
self.assertEqual(ac.path, "a")
|
2015-08-23 21:07:00 -03:00
|
|
|
|
2017-01-26 17:16:13 -05:00
|
|
|
def test_repr(self):
|
2022-02-03 20:24:19 +01:00
|
|
|
ac = AppConfig("label", Stub(__path__=["a"]))
|
|
|
|
self.assertEqual(repr(ac), "<AppConfig: label>")
|
2017-01-26 17:16:13 -05:00
|
|
|
|
2020-12-21 14:21:25 +01:00
|
|
|
def test_invalid_label(self):
|
|
|
|
class MyAppConfig(AppConfig):
|
2022-02-03 20:24:19 +01:00
|
|
|
label = "invalid.label"
|
2020-12-21 14:21:25 +01:00
|
|
|
|
|
|
|
msg = "The app label 'invalid.label' is not a valid Python identifier."
|
|
|
|
with self.assertRaisesMessage(ImproperlyConfigured, msg):
|
2022-02-03 20:24:19 +01:00
|
|
|
MyAppConfig("test_app", Stub())
|
2020-12-21 14:21:25 +01:00
|
|
|
|
2020-07-12 13:59:57 +01:00
|
|
|
@override_settings(
|
2022-02-03 20:24:19 +01:00
|
|
|
INSTALLED_APPS=["apps.apps.ModelPKAppsConfig"],
|
|
|
|
DEFAULT_AUTO_FIELD="django.db.models.SmallAutoField",
|
2020-07-12 13:59:57 +01:00
|
|
|
)
|
|
|
|
def test_app_default_auto_field(self):
|
2022-02-03 20:24:19 +01:00
|
|
|
apps_config = apps.get_app_config("apps")
|
2020-07-12 13:59:57 +01:00
|
|
|
self.assertEqual(
|
|
|
|
apps_config.default_auto_field,
|
2022-02-03 20:24:19 +01:00
|
|
|
"django.db.models.BigAutoField",
|
2020-07-12 13:59:57 +01:00
|
|
|
)
|
|
|
|
self.assertIs(apps_config._is_default_auto_field_overridden, True)
|
|
|
|
|
|
|
|
@override_settings(
|
2022-02-03 20:24:19 +01:00
|
|
|
INSTALLED_APPS=["apps.apps.PlainAppsConfig"],
|
|
|
|
DEFAULT_AUTO_FIELD="django.db.models.SmallAutoField",
|
2020-07-12 13:59:57 +01:00
|
|
|
)
|
|
|
|
def test_default_auto_field_setting(self):
|
2022-02-03 20:24:19 +01:00
|
|
|
apps_config = apps.get_app_config("apps")
|
2020-07-12 13:59:57 +01:00
|
|
|
self.assertEqual(
|
|
|
|
apps_config.default_auto_field,
|
2022-02-03 20:24:19 +01:00
|
|
|
"django.db.models.SmallAutoField",
|
2020-07-12 13:59:57 +01:00
|
|
|
)
|
|
|
|
self.assertIs(apps_config._is_default_auto_field_overridden, False)
|
|
|
|
|
2014-01-27 13:28:53 -07:00
|
|
|
|
2015-04-17 17:38:20 -04:00
|
|
|
class NamespacePackageAppTests(SimpleTestCase):
|
2014-01-25 19:37:05 -07:00
|
|
|
# We need nsapp to be top-level so our multiple-paths tests can add another
|
|
|
|
# location for it (if its inside a normal package with an __init__.py that
|
|
|
|
# isn't possible). In order to avoid cluttering the already-full tests/ dir
|
|
|
|
# (which is on sys.path), we add these new entries to sys.path temporarily.
|
2022-02-03 20:24:19 +01:00
|
|
|
base_location = os.path.join(HERE, "namespace_package_base")
|
|
|
|
other_location = os.path.join(HERE, "namespace_package_other_base")
|
|
|
|
app_path = os.path.join(base_location, "nsapp")
|
2014-01-25 19:37:05 -07:00
|
|
|
|
|
|
|
def test_single_path(self):
|
|
|
|
"""
|
|
|
|
A Py3.3+ namespace package can be an app if it has only one path.
|
|
|
|
"""
|
2014-01-25 22:50:40 -07:00
|
|
|
with extend_sys_path(self.base_location):
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.settings(INSTALLED_APPS=["nsapp"]):
|
|
|
|
app_config = apps.get_app_config("nsapp")
|
2017-01-20 08:01:02 -05:00
|
|
|
self.assertEqual(app_config.path, self.app_path)
|
2014-01-25 19:37:05 -07:00
|
|
|
|
|
|
|
def test_multiple_paths(self):
|
|
|
|
"""
|
|
|
|
A Py3.3+ namespace package with multiple locations cannot be an app.
|
|
|
|
|
|
|
|
(Because then we wouldn't know where to load its templates, static
|
2016-01-10 17:48:16 +01:00
|
|
|
assets, etc. from.)
|
2014-01-25 19:37:05 -07:00
|
|
|
"""
|
|
|
|
# Temporarily add two directories to sys.path that both contain
|
|
|
|
# components of the "nsapp" package.
|
2014-01-25 22:50:40 -07:00
|
|
|
with extend_sys_path(self.base_location, self.other_location):
|
2014-01-25 19:37:05 -07:00
|
|
|
with self.assertRaises(ImproperlyConfigured):
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.settings(INSTALLED_APPS=["nsapp"]):
|
2014-01-25 19:37:05 -07:00
|
|
|
pass
|
|
|
|
|
|
|
|
def test_multiple_paths_explicit_path(self):
|
|
|
|
"""
|
|
|
|
Multiple locations are ok only if app-config has explicit path.
|
|
|
|
"""
|
|
|
|
# Temporarily add two directories to sys.path that both contain
|
|
|
|
# components of the "nsapp" package.
|
2014-01-25 22:50:40 -07:00
|
|
|
with extend_sys_path(self.base_location, self.other_location):
|
2022-02-03 20:24:19 +01:00
|
|
|
with self.settings(INSTALLED_APPS=["nsapp.apps.NSAppConfig"]):
|
|
|
|
app_config = apps.get_app_config("nsapp")
|
2017-01-20 08:01:02 -05:00
|
|
|
self.assertEqual(app_config.path, self.app_path)
|
2023-06-01 16:39:52 +02:00
|
|
|
|
|
|
|
|
|
|
|
class QueryPerformingAppTests(TransactionTestCase):
|
|
|
|
available_apps = ["apps"]
|
|
|
|
databases = {"default", "other"}
|
|
|
|
expected_msg = (
|
|
|
|
"Accessing the database during app initialization is discouraged. To fix this "
|
|
|
|
"warning, avoid executing queries in AppConfig.ready() or when your app "
|
|
|
|
"modules are imported."
|
|
|
|
)
|
|
|
|
|
|
|
|
def test_query_default_database_using_model(self):
|
|
|
|
query_results = self.run_setup("QueryDefaultDatabaseModelAppConfig")
|
|
|
|
self.assertSequenceEqual(query_results, [("new name",)])
|
|
|
|
|
|
|
|
def test_query_other_database_using_model(self):
|
|
|
|
query_results = self.run_setup("QueryOtherDatabaseModelAppConfig")
|
|
|
|
self.assertSequenceEqual(query_results, [("new name",)])
|
|
|
|
|
|
|
|
def test_query_default_database_using_cursor(self):
|
|
|
|
query_results = self.run_setup("QueryDefaultDatabaseCursorAppConfig")
|
|
|
|
self.assertSequenceEqual(query_results, [(42,)])
|
|
|
|
|
|
|
|
def test_query_other_database_using_cursor(self):
|
|
|
|
query_results = self.run_setup("QueryOtherDatabaseCursorAppConfig")
|
|
|
|
self.assertSequenceEqual(query_results, [(42,)])
|
|
|
|
|
|
|
|
def test_query_many_default_database_using_cursor(self):
|
|
|
|
self.run_setup("QueryDefaultDatabaseCursorManyAppConfig")
|
|
|
|
|
|
|
|
def test_query_many_other_database_using_cursor(self):
|
|
|
|
self.run_setup("QueryOtherDatabaseCursorManyAppConfig")
|
|
|
|
|
|
|
|
@skipUnlessDBFeature("create_test_procedure_without_params_sql")
|
|
|
|
def test_query_default_database_using_stored_procedure(self):
|
|
|
|
connection = connections["default"]
|
|
|
|
with connection.cursor() as cursor:
|
|
|
|
cursor.execute(connection.features.create_test_procedure_without_params_sql)
|
|
|
|
|
|
|
|
try:
|
|
|
|
self.run_setup("QueryDefaultDatabaseStoredProcedureAppConfig")
|
|
|
|
finally:
|
|
|
|
with connection.schema_editor() as editor:
|
|
|
|
editor.remove_procedure("test_procedure")
|
|
|
|
|
|
|
|
@skipUnlessDBFeature("create_test_procedure_without_params_sql")
|
|
|
|
def test_query_other_database_using_stored_procedure(self):
|
|
|
|
connection = connections["other"]
|
|
|
|
with connection.cursor() as cursor:
|
|
|
|
cursor.execute(connection.features.create_test_procedure_without_params_sql)
|
|
|
|
|
|
|
|
try:
|
|
|
|
self.run_setup("QueryOtherDatabaseStoredProcedureAppConfig")
|
|
|
|
finally:
|
|
|
|
with connection.schema_editor() as editor:
|
|
|
|
editor.remove_procedure("test_procedure")
|
|
|
|
|
|
|
|
def run_setup(self, app_config_name):
|
|
|
|
custom_settings = override_settings(
|
|
|
|
INSTALLED_APPS=[f"apps.query_performing_app.apps.{app_config_name}"]
|
|
|
|
)
|
|
|
|
# Ignore the RuntimeWarning, as override_settings.enable() calls
|
|
|
|
# AppConfig.ready() which will trigger the warning.
|
|
|
|
with self.assertWarnsMessage(RuntimeWarning, self.expected_msg):
|
|
|
|
custom_settings.enable()
|
|
|
|
try:
|
|
|
|
with patch.multiple(apps, ready=False, loading=False, app_configs={}):
|
|
|
|
with self.assertWarnsMessage(RuntimeWarning, self.expected_msg):
|
|
|
|
django.setup()
|
|
|
|
|
|
|
|
app_config = apps.get_app_config("query_performing_app")
|
|
|
|
return app_config.query_results
|
|
|
|
finally:
|
|
|
|
custom_settings.disable()
|