From 630c2bd1c27e9f7ae1d09b94774211f1f23e79e8 Mon Sep 17 00:00:00 2001 From: Pedro Brochado Date: Wed, 11 Dec 2024 16:27:08 -0300 Subject: [PATCH 1/3] Replace DEFAULT_FILE_STORAGE with STORAGES in codebase Django deprecated the use of DEFAULT_FILE_STORAGES and STATICFILES_STORAGE in v4.2 and removed it in v5.0. It provides a compat layer in 4.2, so even if the user set this value using the legacy key, the value will be available in django.conf.settings as: * settings.STORAGES['default']['BACKEND'] # default storage * settings.STORAGES['staticfiles']['BACKEND'] # staticfiles storage This commit updates the codebase to use the new value. --- CHANGES/plugin_api/5404.removal | 9 +++++++++ .../tests/functional/api/test_download_policies.py | 8 ++++---- pulpcore/app/apps.py | 6 +++--- pulpcore/app/checks.py | 2 +- pulpcore/app/importexport.py | 2 +- pulpcore/app/tasks/export.py | 2 +- pulpcore/app/util.py | 6 ++++-- pulpcore/pytest_plugin.py | 2 +- .../functional/api/test_artifact_distribution.py | 7 +++---- .../tests/functional/api/test_crd_artifacts.py | 6 +++--- pulpcore/tests/functional/api/test_crud_domains.py | 14 +++++++++----- pulpcore/tests/functional/api/test_status.py | 13 ++++++------- .../functional/api/using_plugin/test_orphans.py | 10 ++++++---- pulpcore/tests/unit/models/test_content.py | 2 +- 14 files changed, 52 insertions(+), 37 deletions(-) create mode 100644 CHANGES/plugin_api/5404.removal diff --git a/CHANGES/plugin_api/5404.removal b/CHANGES/plugin_api/5404.removal new file mode 100644 index 0000000000..1bfef4f8b2 --- /dev/null +++ b/CHANGES/plugin_api/5404.removal @@ -0,0 +1,9 @@ +Started using `settings.STORAGES` internally instead of `settings.DEFAULT_FILE_STORAGE` and `settings.STATICFILES_STORAGE`, +which was deprecated in Django 4.2. + +For compatibility, plugins must replace access to: + +* `settings.DEFAULT_FILE_STORAGE` with `settings.STORAGES["default"]["BACKEND"]` +* `settings.STATICFILES_STORAGE` with `settings.STORAGES["staticfiles"]["BACKEND"]` + +See the new storage structure in [Django docs](https://docs.djangoproject.com/en/4.2/ref/settings/#std-setting-STORAGES). diff --git a/pulp_file/tests/functional/api/test_download_policies.py b/pulp_file/tests/functional/api/test_download_policies.py index eeaa5a1fb7..c6880da8bc 100644 --- a/pulp_file/tests/functional/api/test_download_policies.py +++ b/pulp_file/tests/functional/api/test_download_policies.py @@ -11,7 +11,6 @@ from pulpcore.tests.functional.utils import get_files_in_manifest, download_file -from pulpcore.app import settings from pulpcore.client.pulp_file import FileFilePublication, RepositorySyncURL @@ -54,7 +53,8 @@ def test_download_policy( download_policy, ): """Test that "on_demand" and "streamed" download policies work as expected.""" - if download_policy == "on_demand" and "SFTP" in pulp_settings.DEFAULT_FILE_STORAGE: + settings = pulp_settings + if download_policy == "on_demand" and "SFTP" in settings.STORAGES["default"]["BACKEND"]: pytest.skip("This storage technology is not properly supported.") remote = file_remote_ssl_factory( @@ -148,7 +148,7 @@ def test_download_policy( assert expected_checksum == actual_checksum if ( download_policy == "immediate" - and settings.DEFAULT_FILE_STORAGE != "pulpcore.app.models.storage.FileSystem" + and settings.STORAGES["default"]["BACKEND"] != "pulpcore.app.models.storage.FileSystem" and settings.REDIRECT_TO_OBJECT_STORAGE ): content_disposition = downloaded_file.response_obj.headers.get("Content-Disposition") @@ -187,7 +187,7 @@ def test_download_policy( content_unit = expected_files_list[4] content_unit_url = urljoin(distribution.base_url, content_unit[0]) # The S3 test API project doesn't handle invalid Range values correctly - if settings.DEFAULT_FILE_STORAGE == "pulpcore.app.models.storage.FileSystem": + if settings.STORAGES["default"]["BACKEND"] == "pulpcore.app.models.storage.FileSystem": with pytest.raises(ClientResponseError) as exc: range_header = {"Range": "bytes=-1-11"} download_file(content_unit_url, headers=range_header) diff --git a/pulpcore/app/apps.py b/pulpcore/app/apps.py index 78572f3b36..07b1938533 100644 --- a/pulpcore/app/apps.py +++ b/pulpcore/app/apps.py @@ -323,11 +323,11 @@ def _ensure_default_domain(sender, **kwargs): if ( settings.HIDE_GUARDED_DISTRIBUTIONS != default.hide_guarded_distributions or settings.REDIRECT_TO_OBJECT_STORAGE != default.redirect_to_object_storage - or settings.DEFAULT_FILE_STORAGE != default.storage_class + or settings.STORAGES["default"]["BACKEND"] != default.storage_class ): default.hide_guarded_distributions = settings.HIDE_GUARDED_DISTRIBUTIONS default.redirect_to_object_storage = settings.REDIRECT_TO_OBJECT_STORAGE - default.storage_class = settings.DEFAULT_FILE_STORAGE + default.storage_class = settings.STORAGES["default"]["BACKEND"] default.save(skip_hooks=True) @@ -394,7 +394,7 @@ def _get_permission(perm): def _populate_artifact_serving_distribution(sender, apps, verbosity, **kwargs): if ( - settings.DEFAULT_FILE_STORAGE == "pulpcore.app.models.storage.FileSystem" + settings.STORAGES["default"]["BACKEND"] == "pulpcore.app.models.storage.FileSystem" or not settings.REDIRECT_TO_OBJECT_STORAGE ): try: diff --git a/pulpcore/app/checks.py b/pulpcore/app/checks.py index a0129bf56b..a8d4a26b64 100644 --- a/pulpcore/app/checks.py +++ b/pulpcore/app/checks.py @@ -38,7 +38,7 @@ def secret_key_check(app_configs, **kwargs): def storage_paths(app_configs, **kwargs): warnings = [] - if settings.DEFAULT_FILE_STORAGE == "pulpcore.app.models.storage.FileSystem": + if settings.STORAGES["default"]["BACKEND"] == "pulpcore.app.models.storage.FileSystem": try: media_root_dev = Path(settings.MEDIA_ROOT).stat().st_dev except OSError: diff --git a/pulpcore/app/importexport.py b/pulpcore/app/importexport.py index 3dab3aa6ee..c8ade37077 100644 --- a/pulpcore/app/importexport.py +++ b/pulpcore/app/importexport.py @@ -114,7 +114,7 @@ def export_artifacts(export, artifact_pks): with ProgressReport(**data) as pb: pb.BATCH_INTERVAL = 5000 - if settings.DEFAULT_FILE_STORAGE != "pulpcore.app.models.storage.FileSystem": + if settings.STORAGES["default"]["BACKEND"] != "pulpcore.app.models.storage.FileSystem": with tempfile.TemporaryDirectory(dir=".") as temp_dir: for offset in range(0, len(artifact_pks), EXPORT_BATCH_SIZE): batch = artifact_pks[offset : offset + EXPORT_BATCH_SIZE] diff --git a/pulpcore/app/tasks/export.py b/pulpcore/app/tasks/export.py index 71956d2359..e7ef5f164a 100644 --- a/pulpcore/app/tasks/export.py +++ b/pulpcore/app/tasks/export.py @@ -70,7 +70,7 @@ def _export_to_file_system(path, relative_paths_to_artifacts, method=FS_EXPORT_M ValidationError: When path is not in the ALLOWED_EXPORT_PATHS setting """ using_filesystem_storage = ( - settings.DEFAULT_FILE_STORAGE == "pulpcore.app.models.storage.FileSystem" + settings.STORAGES["default"]["BACKEND"] == "pulpcore.app.models.storage.FileSystem" ) if method != FS_EXPORT_METHODS.WRITE and not using_filesystem_storage: diff --git a/pulpcore/app/util.py b/pulpcore/app/util.py index dd3119737e..ce80c9ff65 100644 --- a/pulpcore/app/util.py +++ b/pulpcore/app/util.py @@ -536,7 +536,7 @@ def get_artifact_url(artifact, headers=None, http_method=None): if settings.DOMAIN_ENABLED: loc = f"domain {artifact_domain.name}.storage_class" else: - loc = "settings.DEFAULT_FILE_STORAGE" + loc = "settings.STORAGES['default']['BACKEND']" raise NotImplementedError( f"The value {loc}={artifact_domain.storage_class} does not allow redirecting." @@ -582,7 +582,9 @@ def get_default_domain(): try: default_domain = Domain.objects.get(name="default") except Domain.DoesNotExist: - default_domain = Domain(name="default", storage_class=settings.DEFAULT_FILE_STORAGE) + default_domain = Domain( + name="default", storage_class=settings.STORAGES["default"]["BACKEND"] + ) default_domain.save(skip_hooks=True) return default_domain diff --git a/pulpcore/pytest_plugin.py b/pulpcore/pytest_plugin.py index e49b34858c..f05628cba0 100644 --- a/pulpcore/pytest_plugin.py +++ b/pulpcore/pytest_plugin.py @@ -621,7 +621,7 @@ def _settings_factory(storage_class=None, storage_settings=None): "AZURE_CONNECTION_STRING", ] settings = storage_settings or dict() - backend = storage_class or pulp_settings.DEFAULT_FILE_STORAGE + backend = storage_class or pulp_settings.STORAGES["default"]["BACKEND"] for key in keys[backend]: if key not in settings: settings[key] = getattr(pulp_settings, key, None) diff --git a/pulpcore/tests/functional/api/test_artifact_distribution.py b/pulpcore/tests/functional/api/test_artifact_distribution.py index d6735347c2..82f79c3fad 100644 --- a/pulpcore/tests/functional/api/test_artifact_distribution.py +++ b/pulpcore/tests/functional/api/test_artifact_distribution.py @@ -2,8 +2,6 @@ import subprocess from hashlib import sha256 -from django.conf import settings - OBJECT_STORAGES = ( "storages.backends.s3boto3.S3Boto3Storage", @@ -12,7 +10,8 @@ ) -def test_artifact_distribution(random_artifact): +def test_artifact_distribution(random_artifact, pulp_settings): + settings = pulp_settings artifact_uuid = random_artifact.pulp_href.split("/")[-2] commands = ( @@ -29,7 +28,7 @@ def test_artifact_distribution(random_artifact): hasher = sha256() hasher.update(response.content) assert hasher.hexdigest() == random_artifact.sha256 - if settings.DEFAULT_FILE_STORAGE in OBJECT_STORAGES: + if settings.STORAGES["default"]["BACKEND"] in OBJECT_STORAGES: content_disposition = response.headers.get("Content-Disposition") assert content_disposition is not None filename = artifact_uuid diff --git a/pulpcore/tests/functional/api/test_crd_artifacts.py b/pulpcore/tests/functional/api/test_crd_artifacts.py index 143907b1c1..74de2a38ae 100644 --- a/pulpcore/tests/functional/api/test_crd_artifacts.py +++ b/pulpcore/tests/functional/api/test_crd_artifacts.py @@ -6,7 +6,6 @@ import uuid import pytest -from django.conf import settings @pytest.fixture @@ -142,10 +141,11 @@ def test_upload_mixed_attrs(pulpcore_bindings, pulpcore_random_file): @pytest.mark.parallel -def test_delete_artifact(pulpcore_bindings, pulpcore_random_file, gen_user): +def test_delete_artifact(pulpcore_bindings, pulpcore_random_file, gen_user, pulp_settings): """Verify that the deletion of artifacts is prohibited for both regular users and administrators.""" - if settings.DEFAULT_FILE_STORAGE != "pulpcore.app.models.storage.FileSystem": + settings = pulp_settings + if settings.STORAGES["default"]["BACKEND"] != "pulpcore.app.models.storage.FileSystem": pytest.skip("this test only works for filesystem storage") media_root = settings.MEDIA_ROOT diff --git a/pulpcore/tests/functional/api/test_crud_domains.py b/pulpcore/tests/functional/api/test_crud_domains.py index 3741e07ca7..e645240613 100644 --- a/pulpcore/tests/functional/api/test_crud_domains.py +++ b/pulpcore/tests/functional/api/test_crud_domains.py @@ -5,7 +5,6 @@ import string import json from pulpcore.client.pulpcore import ApiException -from pulpcore.app import settings from pulpcore.tests.functional.utils import PulpTaskError @@ -51,15 +50,16 @@ def test_crud_domains(pulpcore_bindings, monitor_task): @pytest.mark.parallel -def test_default_domain(pulpcore_bindings): +def test_default_domain(pulpcore_bindings, pulp_settings): """Test properties around the default domain.""" + settings = pulp_settings domains = pulpcore_bindings.DomainsApi.list(name="default") assert domains.count == 1 # Read the default domain, ensure storage is set to default default_domain = domains.results[0] assert default_domain.name == "default" - assert default_domain.storage_class == settings.DEFAULT_FILE_STORAGE + assert default_domain.storage_class == settings.STORAGES["default"]["BACKEND"] assert default_domain.redirect_to_object_storage == settings.REDIRECT_TO_OBJECT_STORAGE assert default_domain.hide_guarded_distributions == settings.HIDE_GUARDED_DISTRIBUTIONS @@ -92,8 +92,9 @@ def test_default_domain(pulpcore_bindings): @pytest.mark.parallel -def test_active_domain_deletion(pulpcore_bindings, monitor_task): +def test_active_domain_deletion(pulpcore_bindings, monitor_task, pulp_settings): """Test trying to delete a domain that is in use, has objects in it.""" + settings = pulp_settings if not settings.DOMAIN_ENABLED: pytest.skip("Domains not enabled") name = str(uuid.uuid4()) @@ -133,8 +134,10 @@ def test_orphan_domain_deletion( gen_object_with_cleanup, monitor_task, tmp_path, + pulp_settings, ): """Test trying to delete a domain that is in use, has objects in it.""" + settings = pulp_settings if not settings.DOMAIN_ENABLED: pytest.skip("Domains not enabled") body = { @@ -177,8 +180,9 @@ def test_orphan_domain_deletion( @pytest.mark.parallel -def test_special_domain_creation(pulpcore_bindings, gen_object_with_cleanup): +def test_special_domain_creation(pulpcore_bindings, gen_object_with_cleanup, pulp_settings): """Test many possible domain creation scenarios.""" + settings = pulp_settings if not settings.DOMAIN_ENABLED: pytest.skip("Domains not enabled") # This test needs to account for which environment it is running in diff --git a/pulpcore/tests/functional/api/test_status.py b/pulpcore/tests/functional/api/test_status.py index 6997e3cbfb..ff4ae7ee2f 100644 --- a/pulpcore/tests/functional/api/test_status.py +++ b/pulpcore/tests/functional/api/test_status.py @@ -2,7 +2,6 @@ import pytest -from django.conf import settings from jsonschema import validate @@ -58,24 +57,24 @@ @pytest.mark.parallel -def test_get_authenticated(test_path, pulpcore_bindings): +def test_get_authenticated(test_path, pulpcore_bindings, pulp_settings): """GET the status path with valid credentials. Verify the response with :meth:`verify_get_response`. """ response = pulpcore_bindings.StatusApi.status_read() - verify_get_response(response.to_dict(), STATUS) + verify_get_response(response.to_dict(), STATUS, pulp_settings) @pytest.mark.parallel -def test_get_unauthenticated(test_path, pulpcore_bindings, anonymous_user): +def test_get_unauthenticated(test_path, pulpcore_bindings, anonymous_user, pulp_settings): """GET the status path with no credentials. Verify the response with :meth:`verify_get_response`. """ with anonymous_user: response = pulpcore_bindings.StatusApi.status_read() - verify_get_response(response.to_dict(), STATUS) + verify_get_response(response.to_dict(), STATUS, pulp_settings) @pytest.mark.parallel @@ -130,7 +129,7 @@ def test_storage_per_domain( assert default_status.storage != domain_status.storage -def verify_get_response(status, expected_schema): +def verify_get_response(status, expected_schema, settings): """Verify the response to an HTTP GET call. Verify that several attributes and have the correct type or value. @@ -145,7 +144,7 @@ def verify_get_response(status, expected_schema): assert status["content_settings"]["content_path_prefix"] is not None assert status["storage"]["used"] is not None - if settings.DEFAULT_FILE_STORAGE != "pulpcore.app.models.storage.FileSystem": + if settings.STORAGES["default"]["BACKEND"] != "pulpcore.app.models.storage.FileSystem": assert status["storage"]["free"] is None assert status["storage"]["total"] is None else: diff --git a/pulpcore/tests/functional/api/using_plugin/test_orphans.py b/pulpcore/tests/functional/api/using_plugin/test_orphans.py index a25b0d8328..a1e25c65d4 100644 --- a/pulpcore/tests/functional/api/using_plugin/test_orphans.py +++ b/pulpcore/tests/functional/api/using_plugin/test_orphans.py @@ -68,11 +68,12 @@ def test_orphans_delete( monitor_task, pulp_settings, ): + settings = pulp_settings # Verify that the system contains the orphan content unit and the orphan artifact. content_unit = file_bindings.ContentFilesApi.read(file_random_content_unit.pulp_href) artifact = pulpcore_bindings.ArtifactsApi.read(random_artifact.pulp_href) - if pulp_settings.DEFAULT_FILE_STORAGE == "pulpcore.app.models.storage.FileSystem": + if settings.STORAGES["default"]["BACKEND"] == "pulpcore.app.models.storage.FileSystem": # Verify that the artifacts are on disk relative_path = pulpcore_bindings.ArtifactsApi.read(content_unit.artifact).file artifact_path1 = os.path.join(pulp_settings.MEDIA_ROOT, relative_path) @@ -88,7 +89,7 @@ def test_orphans_delete( with pytest.raises(file_bindings.ApiException) as exc: file_bindings.ContentFilesApi.read(file_random_content_unit.pulp_href) assert exc.value.status == 404 - if pulp_settings.DEFAULT_FILE_STORAGE == "pulpcore.app.models.storage.FileSystem": + if settings.STORAGES["default"]["BACKEND"] == "pulpcore.app.models.storage.FileSystem": assert os.path.exists(artifact_path1) is False assert os.path.exists(artifact_path2) is False @@ -101,6 +102,7 @@ def test_orphans_cleanup( monitor_task, pulp_settings, ): + settings = pulp_settings # Cleanup orphans with a nonzero orphan_protection_time monitor_task(pulpcore_bindings.OrphansCleanupApi.cleanup({"orphan_protection_time": 10}).task) @@ -108,7 +110,7 @@ def test_orphans_cleanup( content_unit = file_bindings.ContentFilesApi.read(file_random_content_unit.pulp_href) artifact = pulpcore_bindings.ArtifactsApi.read(random_artifact.pulp_href) - if pulp_settings.DEFAULT_FILE_STORAGE == "pulpcore.app.models.storage.FileSystem": + if settings.STORAGES["default"]["BACKEND"] == "pulpcore.app.models.storage.FileSystem": # Verify that the artifacts are on disk relative_path = pulpcore_bindings.ArtifactsApi.read(content_unit.artifact).file artifact_path1 = os.path.join(pulp_settings.MEDIA_ROOT, relative_path) @@ -123,7 +125,7 @@ def test_orphans_cleanup( with pytest.raises(file_bindings.ApiException) as exc: file_bindings.ContentFilesApi.read(file_random_content_unit.pulp_href) assert exc.value.status == 404 - if pulp_settings.DEFAULT_FILE_STORAGE == "pulpcore.app.models.storage.FileSystem": + if settings.STORAGES["default"]["BACKEND"] == "pulpcore.app.models.storage.FileSystem": assert os.path.exists(artifact_path1) is False assert os.path.exists(artifact_path2) is False diff --git a/pulpcore/tests/unit/models/test_content.py b/pulpcore/tests/unit/models/test_content.py index 319180d64c..04eb424b8d 100644 --- a/pulpcore/tests/unit/models/test_content.py +++ b/pulpcore/tests/unit/models/test_content.py @@ -43,7 +43,7 @@ def test_create_read_delete_content(tmp_path): @pytest.mark.django_db def test_storage_location(tmp_path, settings): - if settings.DEFAULT_FILE_STORAGE != "pulpcore.app.models.storage.FileSystem": + if settings.STORAGES["default"]["BACKEND"] != "pulpcore.app.models.storage.FileSystem": pytest.skip("Skipping test for nonlocal storage.") tf = tmp_path / "ab" From 3e3492a00d12d751cebae6016f79370f3d73999d Mon Sep 17 00:00:00 2001 From: Pedro Brochado Date: Thu, 12 Dec 2024 14:16:59 -0300 Subject: [PATCH 2/3] Add compat layer to enable using STORAGES Co-authored-by: Matthias Dellweg <2500@gmx.de> --- CHANGES/5404.removal | 11 +++++++++ pulpcore/app/apps.py | 12 +++++++++- pulpcore/app/settings.py | 48 +++++++++++++++++++++++++++++++++------- 3 files changed, 62 insertions(+), 9 deletions(-) create mode 100644 CHANGES/5404.removal diff --git a/CHANGES/5404.removal b/CHANGES/5404.removal new file mode 100644 index 0000000000..97c80c5a69 --- /dev/null +++ b/CHANGES/5404.removal @@ -0,0 +1,11 @@ +Marked django's `DEFAULT_FILE_STORAGE` and `STATIC_FILE_STORAGE` settings to be +removed in pulpcore 3.85. Users should upgrade to use the +[`STORAGES`](https://docs.djangoproject.com/en/4.2/ref/settings/#std-setting-STORAGES) +setting instead. + +The [django-upgrade](https://github.com/adamchainz/django-upgrade?tab=readme-ov-file#django-42) +tool can handle simple cases. If cloud storages are being used, refer to django-storages +to adapt their specific storage options. E.g: + +* +* diff --git a/pulpcore/app/apps.py b/pulpcore/app/apps.py index 07b1938533..ec60d59afb 100644 --- a/pulpcore/app/apps.py +++ b/pulpcore/app/apps.py @@ -5,7 +5,6 @@ from importlib import import_module from django import apps -from django.conf import settings from django.core.exceptions import ImproperlyConfigured from django.db import connection, transaction from django.db.models.signals import post_migrate @@ -68,6 +67,15 @@ class PulpPluginAppConfig(apps.AppConfig): def __init__(self, app_name, app_module): super().__init__(app_name, app_module) + # begin Compatilibity layer for DEFAULT_FILE_STORAGE deprecation + # Remove on pulpcore=3.85 or pulpcore=4.0 + # * Workaround for getting the up-to-date settings instance, otherwise is doesnt + # get the patch in settings.py + # * Update code in signal handlers to use module-level imports again + from django.conf import settings + + self.settings = settings + # end try: self.version @@ -314,6 +322,7 @@ def _populate_system_id(sender, apps, verbosity, **kwargs): def _ensure_default_domain(sender, **kwargs): + settings = sender.settings table_names = connection.introspection.table_names() if "core_domain" in table_names: from pulpcore.app.util import get_default_domain @@ -393,6 +402,7 @@ def _get_permission(perm): def _populate_artifact_serving_distribution(sender, apps, verbosity, **kwargs): + settings = sender.settings if ( settings.STORAGES["default"]["BACKEND"] == "pulpcore.app.models.storage.FileSystem" or not settings.REDIRECT_TO_OBJECT_STORAGE diff --git a/pulpcore/app/settings.py b/pulpcore/app/settings.py index 465ef0a045..aa07cb2e07 100644 --- a/pulpcore/app/settings.py +++ b/pulpcore/app/settings.py @@ -16,6 +16,8 @@ from pathlib import Path from cryptography.fernet import Fernet +from django.core.files.storage import storages +from django.conf import global_settings from django.core.exceptions import ImproperlyConfigured from django.db import connection @@ -56,7 +58,28 @@ STATIC_URL = "/assets/" STATIC_ROOT = DEPLOY_ROOT / STATIC_URL.strip("/") -DEFAULT_FILE_STORAGE = "pulpcore.app.models.storage.FileSystem" +# begin compatilibity layer for DEFAULT_FILE_STORAGE +# Remove on pulpcore=3.85 or pulpcore=4.0 + +# - What is this? +# We shouldnt use STORAGES or DEFAULT_FILE_STORAGE directly because those are +# mutually exclusive by django, which constraints users to use whatever we use. +# This is a hack/workaround to set Pulp's default while still enabling users to choose +# the legacy or the new storage setting. +_DEFAULT_FILE_STORAGE = "pulpcore.app.models.storage.FileSystem" +_STORAGES = { + "default": { + "BACKEND": "pulpcore.app.models.storage.FileSystem", + }, + "staticfiles": { + "BACKEND": "django.contrib.staticfiles.storage.StaticFilesStorage", + }, +} + +setattr(global_settings, "DEFAULT_FILE_STORAGE", _DEFAULT_FILE_STORAGE) +setattr(global_settings, "STORAGES", _STORAGES) +# end DEFAULT_FILE_STORAGE deprecation layer + REDIRECT_TO_OBJECT_STORAGE = True WORKING_DIRECTORY = DEPLOY_ROOT / "tmp" @@ -371,16 +394,18 @@ from dynaconf import DjangoDynaconf, Validator # noqa # Validators +storage_keys = ("STORAGES.default.BACKEND", "DEFAULT_FILE_STORAGE") storage_validator = ( Validator("REDIRECT_TO_OBJECT_STORAGE", eq=False) - | Validator("DEFAULT_FILE_STORAGE", eq="pulpcore.app.models.storage.FileSystem") - | Validator("DEFAULT_FILE_STORAGE", eq="storages.backends.azure_storage.AzureStorage") - | Validator("DEFAULT_FILE_STORAGE", eq="storages.backends.s3boto3.S3Boto3Storage") - | Validator("DEFAULT_FILE_STORAGE", eq="storages.backends.gcloud.GoogleCloudStorage") + | Validator(*storage_keys, eq="pulpcore.app.models.storage.FileSystem") + | Validator(*storage_keys, eq="storages.backends.azure_storage.AzureStorage") + | Validator(*storage_keys, eq="storages.backends.s3boto3.S3Boto3Storage") + | Validator(*storage_keys, eq="storages.backends.gcloud.GoogleCloudStorage") ) storage_validator.messages["combined"] = ( - "'REDIRECT_TO_OBJECT_STORAGE=True' is only supported with the local file, S3, GCP or Azure" - "storage backend configured in DEFAULT_FILE_STORAGE." + "'REDIRECT_TO_OBJECT_STORAGE=True' is only supported with the local file, S3, GCP or Azure " + "storage backend configured in STORAGES['default']['BACKEND'] " + "(deprecated DEFAULT_FILE_STORAGE)." ) cache_enabled_validator = Validator("CACHE_ENABLED", eq=True) @@ -485,7 +510,14 @@ def otel_middleware_hook(settings): ], post_hooks=otel_middleware_hook, ) -# HERE ENDS DYNACONF EXTENSION LOAD (No more code below this line) + +# begin compatilibity layer for DEFAULT_FILE_STORAGE +# Remove on pulpcore=3.85 or pulpcore=4.0 + +# Ensures the cached property storage.backends uses the the right value +storages._backends = settings.STORAGES.copy() +storages.backends +# end compatibility layer _logger = getLogger(__name__) From d771a25f5108f7b8251be3525ad0c719f66588c6 Mon Sep 17 00:00:00 2001 From: Pedro Brochado Date: Fri, 13 Dec 2024 17:28:04 -0300 Subject: [PATCH 3/3] Update CI settings to partially use new storage settings --- .ci/ansible/settings.py.j2 | 31 +++++++++++++------ pulpcore/app/serializers/domain.py | 5 ++- pulpcore/pytest_plugin.py | 26 ++++++++++------ .../functional/api/test_crd_artifacts.py | 1 - 4 files changed, 42 insertions(+), 21 deletions(-) diff --git a/.ci/ansible/settings.py.j2 b/.ci/ansible/settings.py.j2 index 4ed28f42ad..9b5f5f4be4 100644 --- a/.ci/ansible/settings.py.j2 +++ b/.ci/ansible/settings.py.j2 @@ -27,19 +27,30 @@ API_ROOT = {{ api_root | repr }} {% endif %} {% if s3_test | default(false) %} -DEFAULT_FILE_STORAGE = "storages.backends.s3boto3.S3Boto3Storage" -MEDIA_ROOT = "" -AWS_ACCESS_KEY_ID = "{{ minio_access_key }}" -AWS_SECRET_ACCESS_KEY = "{{ minio_secret_key }}" -AWS_S3_REGION_NAME = "eu-central-1" -AWS_S3_ADDRESSING_STYLE = "path" +MEDIA_ROOT: "" S3_USE_SIGV4 = True -AWS_S3_SIGNATURE_VERSION = "s3v4" -AWS_STORAGE_BUCKET_NAME = "pulp3" -AWS_S3_ENDPOINT_URL = "http://minio:9000" -AWS_DEFAULT_ACL = "@none None" +STORAGES = { + "default": { + "BACKEND": "storages.backends.s3boto3.S3Boto3Storage", + "OPTIONS": { + "access_key": "{{ minio_access_key }}", + "secret_key": "{{ minio_secret_key }}", + "region_name": "eu-central-1", + "addressing_style": "path", + "signature_version": "s3v4", + "bucket_name": "pulp3", + "endpoint_url": "http://minio:9000", + "default_acl": "@none None", + }, + }, + "staticfiles": { + "BACKEND": "django.contrib.staticfiles.storage.StaticFilesStorage", + }, +} {% endif %} +# This is using DEFAULT_FILE_STORAGE to test both usages. +# Remove when DEFAULT_FILE_STORAGE is completely removed. {% if azure_test | default(false) %} DEFAULT_FILE_STORAGE = "storages.backends.azure_storage.AzureStorage" MEDIA_ROOT = "" diff --git a/pulpcore/app/serializers/domain.py b/pulpcore/app/serializers/domain.py index 54a21b5044..cd38598a8e 100644 --- a/pulpcore/app/serializers/domain.py +++ b/pulpcore/app/serializers/domain.py @@ -42,7 +42,10 @@ def to_representation(self, instance): # Should I convert back the saved settings to their Setting names for to_representation? if getattr(self.context.get("domain", None), "name", None) == "default": for setting_name, field in self.SETTING_MAPPING.items(): - if value := getattr(settings, setting_name.upper(), None): + value = getattr(settings, setting_name, None) or settings.STORAGES["default"].get( + "OPTIONS", {} + ).get(field) + if value: instance[field] = value return super().to_representation(instance) diff --git a/pulpcore/pytest_plugin.py b/pulpcore/pytest_plugin.py index f05628cba0..df89d87ab5 100644 --- a/pulpcore/pytest_plugin.py +++ b/pulpcore/pytest_plugin.py @@ -603,13 +603,13 @@ def _settings_factory(storage_class=None, storage_settings=None): keys = dict() keys["pulpcore.app.models.storage.FileSystem"] = ["MEDIA_ROOT", "MEDIA_URL"] keys["storages.backends.s3boto3.S3Boto3Storage"] = [ - "AWS_ACCESS_KEY_ID", - "AWS_SECRET_ACCESS_KEY", - "AWS_S3_ENDPOINT_URL", - "AWS_S3_ADDRESSING_STYLE", - "AWS_S3_SIGNATURE_VERSION", - "AWS_S3_REGION_NAME", - "AWS_STORAGE_BUCKET_NAME", + "access_key", + "secret_key", + "endpoint_url", + "addressing_style", + "signature_version", + "region_name", + "bucket_name", ] keys["storages.backends.azure_storage.AzureStorage"] = [ "AZURE_ACCOUNT_NAME", @@ -622,8 +622,16 @@ def _settings_factory(storage_class=None, storage_settings=None): ] settings = storage_settings or dict() backend = storage_class or pulp_settings.STORAGES["default"]["BACKEND"] - for key in keys[backend]: - if key not in settings: + not_defined_settings = (k for k in keys[backend] if k not in settings) + # The CI configures s3 with STORAGES and Azure with legacy + # Move all to STORAGES structure on DEFAULT_FILE_STORAGE removal + if backend == "storages.backends.s3boto3.S3Boto3Storage": + storages_dict = getattr(pulp_settings, "STORAGES", {}) + storage_options = storages_dict.get("default", {}).get("OPTIONS", {}) + for key in not_defined_settings: + settings[key] = storage_options.get(key) + else: + for key in not_defined_settings: settings[key] = getattr(pulp_settings, key, None) return backend, settings diff --git a/pulpcore/tests/functional/api/test_crd_artifacts.py b/pulpcore/tests/functional/api/test_crd_artifacts.py index 74de2a38ae..d5a8a1da5c 100644 --- a/pulpcore/tests/functional/api/test_crd_artifacts.py +++ b/pulpcore/tests/functional/api/test_crd_artifacts.py @@ -7,7 +7,6 @@ import pytest - @pytest.fixture def pulpcore_random_file(tmp_path): name = tmp_path / str(uuid.uuid4())