diff --git a/pulp_file/app/viewsets.py b/pulp_file/app/viewsets.py index 214d310f819..05d8dd95d28 100644 --- a/pulp_file/app/viewsets.py +++ b/pulp_file/app/viewsets.py @@ -82,7 +82,7 @@ class FileContentViewSet(SingleArtifactContentUploadViewSet): "effect": "allow", }, { - "action": ["create"], + "action": ["create", "set_label", "unset_label"], "principal": "authenticated", "effect": "allow", "condition": [ diff --git a/pulpcore/app/migrations/0123_content_pulp_labels.py b/pulpcore/app/migrations/0123_content_pulp_labels.py new file mode 100644 index 00000000000..de9d4de0daa --- /dev/null +++ b/pulpcore/app/migrations/0123_content_pulp_labels.py @@ -0,0 +1,19 @@ +# Generated by Django 4.2.15 on 2024-09-08 23:15 + +import django.contrib.postgres.fields.hstore +from django.db import migrations + + +class Migration(migrations.Migration): + + dependencies = [ + ('core', '0122_record_last_replication_timestamp'), + ] + + operations = [ + migrations.AddField( + model_name='content', + name='pulp_labels', + field=django.contrib.postgres.fields.hstore.HStoreField(default=dict), + ), + ] diff --git a/pulpcore/app/models/content.py b/pulpcore/app/models/content.py index 5205715cab4..2d4c8b46f81 100644 --- a/pulpcore/app/models/content.py +++ b/pulpcore/app/models/content.py @@ -16,6 +16,7 @@ from itertools import chain from django.conf import settings +from django.contrib.postgres.fields import HStoreField from django.core import validators from django.db import IntegrityError, models, transaction from django.forms.models import model_to_dict @@ -515,6 +516,7 @@ class Content(MasterModel, QueryMixin): Fields: upstream_id (models.UUIDField) : identifier of content imported from an 'upstream' Pulp timestamp_of_interest (models.DateTimeField): timestamp that prevents orphan cleanup + pulp_labels (HStoreField): Dictionary of string values. Relations: @@ -528,6 +530,7 @@ class Content(MasterModel, QueryMixin): TYPE = "content" repo_key_fields = () # Used by pulpcore.plugin.repo_version_utils.remove_duplicates upstream_id = models.UUIDField(null=True) # Used by PulpImport/Export processing + pulp_labels = HStoreField(default=dict) _artifacts = models.ManyToManyField(Artifact, through="ContentArtifact") timestamp_of_interest = models.DateTimeField(auto_now=True) diff --git a/pulpcore/app/serializers/content.py b/pulpcore/app/serializers/content.py index a7ec3ee312a..62fefaf08e3 100644 --- a/pulpcore/app/serializers/content.py +++ b/pulpcore/app/serializers/content.py @@ -5,12 +5,14 @@ from rest_framework.validators import UniqueValidator from pulpcore.app import models -from pulpcore.app.serializers import base, fields, DetailRelatedField +from pulpcore.app.serializers import base, fields, pulp_labels_validator, DetailRelatedField from pulpcore.app.util import get_domain class NoArtifactContentSerializer(base.ModelSerializer): pulp_href = base.DetailIdentityField(view_name_pattern=r"contents(-.*/.*)-detail") + pulp_labels = serializers.HStoreField(required=False, validators=[pulp_labels_validator]) + repository = DetailRelatedField( help_text=_("A URI of a repository the new content unit should be associated with."), required=False, @@ -104,7 +106,10 @@ def create(self, validated_data): class Meta: model = models.Content - fields = base.ModelSerializer.Meta.fields + ("repository",) + fields = base.ModelSerializer.Meta.fields + ( + "repository", + "pulp_labels", + ) class SingleArtifactContentSerializer(NoArtifactContentSerializer): diff --git a/pulpcore/app/serializers/repository.py b/pulpcore/app/serializers/repository.py index 6d9d2c6f5b7..17f0cbacc65 100644 --- a/pulpcore/app/serializers/repository.py +++ b/pulpcore/app/serializers/repository.py @@ -444,8 +444,9 @@ class RepositoryAddRemoveContentSerializer(ModelSerializer, NestedHyperlinkedMod remove_content_units = serializers.ListField( help_text=_( "A list of content units to remove from the latest repository version. " - "You may also specify '*' as an entry to remove all content. This content is " - "removed before add_content_units are added." + "You may specify '*' as an entry to remove all content, or 'key=value' pairs " + "to remove content by-label. This content is removed before add_content_units " + "are added." ), child=serializers.CharField(error_messages={"invalid": "Not a valid URI of a resource."}), required=False, @@ -474,19 +475,29 @@ def validate_add_content_units(self, value): def validate_remove_content_units(self, value): remove_content_units = {} - + # "* must be alone, and means "all-content" if "*" in value: if len(value) > 1: raise serializers.ValidationError("Cannot supply content units and '*'.") else: return ["*"] - else: + + # Q: just one label? can label/hrefs be mixed? accept full label-operand-set? + # Current: multiple labels, mixed with hrefs, only '=' + # "x=y" can happen multiple times, and means "all content with label x=y, pass k=v pair" + labels_specified = [s for s in value if "=" in s] + if labels_specified: + value = set(value) - set(labels_specified) + + # Anything remaining means "content HREFs, pass UUIDs" + if value: for url in value: remove_content_units[extract_pk(url)] = url content_units_pks = set(remove_content_units.keys()) existing_content_units = models.Content.objects.filter(pk__in=content_units_pks) raise_for_unknown_content_units(existing_content_units, remove_content_units) - return list(remove_content_units.keys()) + + return list(remove_content_units.keys()) + labels_specified class Meta: model = models.RepositoryVersion diff --git a/pulpcore/app/tasks/repository.py b/pulpcore/app/tasks/repository.py index 8b535107eea..812ca9bcc87 100644 --- a/pulpcore/app/tasks/repository.py +++ b/pulpcore/app/tasks/repository.py @@ -232,6 +232,24 @@ def add_and_remove(repository_pk, add_content_units, remove_content_units, base_ remove_content_units = latest.content.values_list("pk", flat=True) else: remove_content_units = [] + else: + # Deal with labels in remove_content_units + # If there are none - then we can "just use" remove_content_units as-is + labels_specified = [s for s in remove_content_units if "=" in s] # Ew. + if labels_specified: + # "x=y" can happen multiple times, and means "all content with label x=y, pass UUIDS" + latest = repository.latest_version() + if latest: + # First, remove labels from set-to-be-removed, r_c_u is now a set + remove_content_units = set(remove_content_units) - set(labels_specified) + # For each label, find all content w/ that label and add to be-removed + for label in labels_specified: + kv = label.split("=") + labeled_content_pks = latest.content.filter( + **{f"pulp_labels__{kv[0]}": kv[1]} + ).values_list("pk", flat=True) + remove_content_units.update(labeled_content_pks) + remove_content_units = list(remove_content_units) with repository.new_version(base_version=base_version) as new_version: new_version.remove_content(models.Content.objects.filter(pk__in=remove_content_units)) diff --git a/pulpcore/app/viewsets/content.py b/pulpcore/app/viewsets/content.py index 8e49e487a27..4ed712a719c 100644 --- a/pulpcore/app/viewsets/content.py +++ b/pulpcore/app/viewsets/content.py @@ -15,7 +15,8 @@ SigningServiceSerializer, ) from pulpcore.app.util import get_viewset_for_model -from pulpcore.app.viewsets.base import NamedModelViewSet +from pulpcore.app.viewsets.base import NamedModelViewSet, LabelsMixin +from pulpcore.app.viewsets.custom_filters import LabelFilter from .custom_filters import ( ArtifactRepositoryVersionFilter, @@ -127,6 +128,8 @@ class ContentFilter(BaseFilterSet): orphaned_for: Return Content which has been orphaned for a given number of minutes; -1 uses ORPHAN_PROTECTION_TIME value. + pulp_label_select: + Return Content which has has the specified label """ repository_version = ContentRepositoryVersionFilter() @@ -135,6 +138,7 @@ class ContentFilter(BaseFilterSet): orphaned_for = OrphanedFilter( help_text="Minutes Content has been orphaned for. -1 uses ORPHAN_PROTECTION_TIME." ) + pulp_label_select = LabelFilter() class BaseContentViewSet(NamedModelViewSet): @@ -199,7 +203,11 @@ def routable(cls): class ContentViewSet( - BaseContentViewSet, mixins.CreateModelMixin, mixins.RetrieveModelMixin, mixins.ListModelMixin + BaseContentViewSet, + mixins.CreateModelMixin, + mixins.RetrieveModelMixin, + mixins.ListModelMixin, + LabelsMixin, ): """ Content viewset that supports POST and GET by default.