diff --git a/CHANGES/+select_plugins.feature b/CHANGES/+select_plugins.feature new file mode 100644 index 0000000000..ca6e2f966f --- /dev/null +++ b/CHANGES/+select_plugins.feature @@ -0,0 +1 @@ +Added ``ENABLED_PLUGINS`` option to allow selecting installed plugins to be enabled. diff --git a/pulpcore/app/settings.py b/pulpcore/app/settings.py index 23026460cc..56461b040f 100644 --- a/pulpcore/app/settings.py +++ b/pulpcore/app/settings.py @@ -12,9 +12,9 @@ from contextlib import suppress from importlib import import_module +from importlib.metadata import entry_points from logging import getLogger from pathlib import Path -from pkg_resources import iter_entry_points from cryptography.fernet import Fernet from django.core.exceptions import ImproperlyConfigured @@ -81,14 +81,6 @@ "pulpcore.app", ] -# Enumerate the installed Pulp plugins during the loading process for use in the status API -INSTALLED_PULP_PLUGINS = [] - -for entry_point in iter_entry_points("pulpcore.plugin"): - plugin_app_config = entry_point.load() - INSTALLED_PULP_PLUGINS.append(entry_point.module_name) - INSTALLED_APPS.append(plugin_app_config) - # Optional apps that help with development, or augment Pulp in some non-critical way OPTIONAL_APPS = [ "crispy_forms", @@ -314,9 +306,17 @@ # HERE STARTS DYNACONF EXTENSION LOAD (Keep at the very bottom of settings.py) # Read more at https://dynaconf.readthedocs.io/en/latest/guides/django.html -from dynaconf import DjangoDynaconf, Validator # noqa +from dynaconf import DjangoDynaconf, Dynaconf, Validator # noqa # Validators + +enabled_plugins_validator = Validator( + "ENABLED_PLUGINS", + is_type_of=list, + len_min=1, + when=Validator("ENABLED_PLUGINS", must_exist=True), +) + content_origin_validator = Validator( "CONTENT_ORIGIN", must_exist=True, @@ -380,19 +380,42 @@ ) +def load_plugin_config_hook(settings): + # Enumerate the installed Pulp plugins during the loading process for use in the status API + ENABLED_PLUGINS = getattr(settings, "ENABLED_PLUGINS", None) + installed_plugins = [] + installed_plugin_apps = [] + + for entry_point in entry_points()["pulpcore.plugin"]: + if ENABLED_PLUGINS is not None and entry_point.name not in ENABLED_PLUGINS: + continue + installed_plugins.append(entry_point.module) + installed_plugin_apps.append(entry_point.load()) + + plugin_settings = Dynaconf( + PRELOAD_FOR_DYNACONF=[f"{module}.app.settings" for module in installed_plugins] + ) + + data = {"dynaconf_merge": True} + data.update(plugin_settings.as_dict()) + data.update(settings.as_dict()) + data["INSTALLED_APPS"].extend(installed_plugin_apps) + data["INSTALLED_APPS"].append("dynaconf_merge_unique") + return data + + settings = DjangoDynaconf( __name__, ENVVAR_PREFIX_FOR_DYNACONF="PULP", ENV_SWITCHER_FOR_DYNACONF="PULP_ENV", - PRELOAD_FOR_DYNACONF=[ - "{}.app.settings".format(plugin_name) for plugin_name in INSTALLED_PULP_PLUGINS - ], ENVVAR_FOR_DYNACONF="PULP_SETTINGS", + post_hooks=(load_plugin_config_hook,), load_dotenv=False, validators=[ api_root_validator, cache_validator, content_origin_validator, + enabled_plugins_validator, sha256_validator, storage_validator, unknown_algs_validator, diff --git a/pulpcore/tests/functional/__init__.py b/pulpcore/tests/functional/__init__.py index 881907b1b6..9159067885 100644 --- a/pulpcore/tests/functional/__init__.py +++ b/pulpcore/tests/functional/__init__.py @@ -1033,11 +1033,7 @@ def _monitor_task_group(task_group_href, timeout=TASK_TIMEOUT): @pytest.fixture(scope="session") def pulp_settings(): - import django - - django.setup() - - from django.conf import settings + from dynaconf import settings return settings diff --git a/staging_docs/admin/learn/settings.md b/staging_docs/admin/learn/settings.md index 84efeae6eb..8562593672 100644 --- a/staging_docs/admin/learn/settings.md +++ b/staging_docs/admin/learn/settings.md @@ -157,6 +157,11 @@ The password for Redis. Pulp defines the following settings itself: +### ENABLED_PLUGINS + +An optional list of plugin names. +If provided, Pulp will limit loading plugins to this list. +If omitted, Pulp will load all installed plugins. ### API_ROOT