Skip to content

Commit

Permalink
Merge pull request #2092 from openedx/asheehan-edx/fixing-group-membe…
Browse files Browse the repository at this point in the history
…rships-hard-delete

fix: hard deleting expired group memberships
  • Loading branch information
alex-sheehan-edx authored May 8, 2024
2 parents 3f139ee + d49f762 commit a69d8a1
Show file tree
Hide file tree
Showing 7 changed files with 29 additions and 13 deletions.
4 changes: 4 additions & 0 deletions CHANGELOG.rst
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,10 @@ Unreleased
----------
* nothing unreleased

[4.17.5]
--------
* fix: hard deleting expired group memberships

[4.17.4]
--------
* fix: update language cookie if langauge cookie is not same as user's language preference
Expand Down
2 changes: 1 addition & 1 deletion enterprise/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,4 +2,4 @@
Your project description goes here.
"""

__version__ = "4.17.4"
__version__ = "4.17.5"
7 changes: 7 additions & 0 deletions enterprise/api/v1/serializers.py
Original file line number Diff line number Diff line change
Expand Up @@ -1732,3 +1732,10 @@ class EnterpriseGroupLearnersRequestQuerySerializer(serializers.Serializer):
required=False,
)
pending_users_only = serializers.BooleanField(required=False, default=False)
show_removed = serializers.BooleanField(required=False, default=False)
is_reversed = serializers.BooleanField(required=False, default=False)
page = serializers.IntegerField(required=False)
lms_users = serializers.ListField(
child=serializers.IntegerField(required=True),
required=False,
)
14 changes: 8 additions & 6 deletions enterprise/api/v1/views/enterprise_group.py
Original file line number Diff line number Diff line change
Expand Up @@ -107,12 +107,14 @@ def get_learners(self, request, *args, **kwargs):
Optional query params:
- ``pending_users_only`` (string, optional): Specify that results should only contain pending learners
- ``q`` (string, optional): Filter the returned members by user email and name with a provided sub-string
- ``user_query`` (string, optional): Filter the returned members by user email and name with a provided
sub-string
- ``sort_by`` (string, optional): Specify how the returned members should be ordered. Supported sorting values
are `memberDetails`, `memberStatus`, and `recentAction`. Ordering can be reversed by supplying a `-` at the
beginning of the sorting value ie `-memberStatus`.
- ``page`` (int, optional): Which page of paginated data to return.
- ``show_removed`` (bool, optional): Include removed learners in the response.
- ``is_reversed`` (bool, optional): Include to reverse the order of returned records.
Returns: Paginated list of learners that are associated with the enterprise group uuid::
Expand All @@ -136,10 +138,7 @@ def get_learners(self, request, *args, **kwargs):
}
"""
query_params = self.request.query_params.copy()
is_reversed = bool(query_params.get('is_reversed', False))
show_removed = bool(query_params.get('show_removed', False))

query_params = self.request.query_params
param_serializers = serializers.EnterpriseGroupLearnersRequestQuerySerializer(
data=query_params
)
Expand All @@ -148,8 +147,11 @@ def get_learners(self, request, *args, **kwargs):
return Response(param_serializers.errors, status=400)

user_query = param_serializers.validated_data.get('user_query')
show_removed = param_serializers.validated_data.get('show_removed', False)
is_reversed = param_serializers.validated_data.get('is_reversed', False)

sort_by = param_serializers.validated_data.get('sort_by')
pending_users_only = param_serializers.validated_data.get('pending_users_only')
pending_users_only = param_serializers.validated_data.get('pending_users_only', False)

group_uuid = kwargs.get('group_uuid')
try:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@ def add_arguments(self, parser):
parser.add_argument("-e", "--enterprise_customer")

def handle(self, *args, **options):
queryset = EnterpriseGroupMembership.objects.all()
queryset = EnterpriseGroupMembership.all_objects.all()
if enterprise_arg := options.get("enterprise_customer"):
try:
enterprise_customer = EnterpriseCustomer.objects.get(uuid=enterprise_arg)
Expand All @@ -43,5 +43,8 @@ def handle(self, *args, **options):
created__lte=localized_utcnow() - timedelta(days=90)
)
for membership in expired_memberships:
membership.pending_enterprise_customer_user.delete()
expired_memberships.delete()
pecu_to_delete = membership.pending_enterprise_customer_user
pecu_to_delete.delete()
membership.refresh_from_db()
# https://github.com/jazzband/django-model-utils/blob/master/model_utils/models.py#L133-L158
membership.delete(soft=False)
4 changes: 2 additions & 2 deletions enterprise/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -4402,14 +4402,14 @@ class EnterpriseGroupMembership(TimeStampedModel, SoftDeletableModel):
blank=True,
null=True,
related_name='memberships',
on_delete=models.deletion.CASCADE,
on_delete=models.deletion.SET_NULL,
)
pending_enterprise_customer_user = models.ForeignKey(
PendingEnterpriseCustomerUser,
blank=True,
null=True,
related_name='memberships',
on_delete=models.deletion.CASCADE,
on_delete=models.deletion.SET_NULL,
)
activated_at = models.DateTimeField(
default=None,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -93,7 +93,7 @@ def test_removing_old_records(self):

call_command(self.command)

# Assert that memberships and pending customers are removed
# Assert that pending memberships and pending customers are removed
assert not models.EnterpriseGroupMembership.all_objects.filter(pk=membership_to_remove.pk)
assert not models.PendingEnterpriseCustomerUser.objects.filter(
pk=membership_to_remove.pending_enterprise_customer_user.pk
Expand Down

0 comments on commit a69d8a1

Please sign in to comment.