From 84160e3d8de6cc00124dde8379f14e849594403b Mon Sep 17 00:00:00 2001 From: pablohashescobar Date: Wed, 28 Feb 2024 16:54:46 +0530 Subject: [PATCH] dev: refactor pagination --- apiserver/plane/app/views/issue.py | 64 ++++++++----------- apiserver/plane/utils/grouper.py | 75 ++++++++++++++++++++-- apiserver/plane/utils/paginator.py | 99 ++++++++++++++++++++++++++++-- 3 files changed, 191 insertions(+), 47 deletions(-) diff --git a/apiserver/plane/app/views/issue.py b/apiserver/plane/app/views/issue.py index b0191581e..c69ef589c 100644 --- a/apiserver/plane/app/views/issue.py +++ b/apiserver/plane/app/views/issue.py @@ -72,7 +72,7 @@ from plane.db.models import ( IssueRelation, ) from plane.bgtasks.issue_activites_task import issue_activity -from plane.utils.grouper import issue_grouper +from plane.utils.grouper import issue_grouper, issue_queryset_grouper from plane.utils.issue_filters import issue_filters from plane.utils.order_queryset import order_issue_queryset from plane.utils.paginator import GroupedOffsetPaginator @@ -264,32 +264,6 @@ class IssueViewSet(WebhookMixin, BaseViewSet): .annotate(count=Func(F("id"), function="Count")) .values("count") ) - .annotate( - label_ids=Coalesce( - ArrayAgg( - "labels__id", - distinct=True, - filter=~Q(labels__id__isnull=True), - ), - Value([], output_field=ArrayField(UUIDField())), - ), - assignee_ids=Coalesce( - ArrayAgg( - "assignees__id", - distinct=True, - filter=~Q(assignees__id__isnull=True), - ), - Value([], output_field=ArrayField(UUIDField())), - ), - module_ids=Coalesce( - ArrayAgg( - "issue_module__module_id", - distinct=True, - filter=~Q(issue_module__module_id__isnull=True), - ), - Value([], output_field=ArrayField(UUIDField())), - ), - ) ).distinct() @method_decorator(gzip_page) @@ -307,12 +281,7 @@ class IssueViewSet(WebhookMixin, BaseViewSet): ) def on_results(issues): - if self.expand or self.fields: - return IssueSerializer( - issues, many=True, expand=self.expand, fields=self.fields - ).data - - return issues.values( + required_fields = [ "id", "name", "state_id", @@ -326,9 +295,6 @@ class IssueViewSet(WebhookMixin, BaseViewSet): "project_id", "parent_id", "cycle_id", - "module_ids", - "label_ids", - "assignee_ids", "sub_issues_count", "created_at", "updated_at", @@ -338,7 +304,20 @@ class IssueViewSet(WebhookMixin, BaseViewSet): "link_count", "is_draft", "archived_at", - ) + ] + if group_by == "assignees__id": + required_fields.extend( + ["label_ids", "module_ids", "assignees__id"] + ) + if group_by == "labels__id": + required_fields.extend( + ["assignee_ids", "module_ids", "labels__id"] + ) + if group_by == "modules__id": + required_fields.extend( + ["assignee_ids", "label_ids", "modules__id"] + ) + return issues.values(*required_fields) # Group by group_by = request.GET.get("group_by", False) @@ -349,6 +328,9 @@ class IssueViewSet(WebhookMixin, BaseViewSet): request=request, queryset=issue_queryset, on_results=on_results ) + issue_queryset = issue_queryset_grouper( + queryset=issue_queryset, field=group_by + ) # Group paginate return self.paginate( request=request, @@ -359,6 +341,14 @@ class IssueViewSet(WebhookMixin, BaseViewSet): group_by_fields=issue_grouper( field=group_by, slug=slug, project_id=project_id ), + count_filter=Q( + Q(issue_inbox__status=1) + | Q(issue_inbox__status=-1) + | Q(issue_inbox__status=2) + | Q(issue_inbox__isnull=True), + archived_at__isnull=False, + is_draft=True, + ), ) def create(self, request, slug, project_id): diff --git a/apiserver/plane/utils/grouper.py b/apiserver/plane/utils/grouper.py index eb5c1d537..e87ca05c5 100644 --- a/apiserver/plane/utils/grouper.py +++ b/apiserver/plane/utils/grouper.py @@ -1,5 +1,9 @@ # Django imports -from django.db.models import Q +from django.db.models import Q, F +from django.contrib.postgres.aggregates import ArrayAgg +from django.contrib.postgres.fields import ArrayField +from django.db.models import Value, UUIDField +from django.db.models.functions import Coalesce # Module imports from plane.db.models import State, Label, ProjectMember, Cycle, Module @@ -256,13 +260,13 @@ def issue_grouper(field, slug, project_id): project_id=project_id, ).values_list("id", flat=True) ) - if field == "label_ids": + if field == "labels__id": return list( Label.objects.filter( workspace__slug=slug, project_id=project_id ).values_list("id", flat=True) ) - if field == "assignee_ids": + if field == "assignees__id": return list( ProjectMember.objects.filter( workspace__slug=slug, project_id=project_id @@ -283,9 +287,72 @@ def issue_grouper(field, slug, project_id): workspace__slug=slug, project_id=project_id ).values_list("id", flat=True) ) - if field == "module_ids": + if field == "modules__id": return list( Module.objects.filter( workspace__slug=slug, project_id=project_id ).values_list("id", flat=True) ) + + +def issue_queryset_grouper(field, queryset): + if field == "assignees__id": + return queryset.annotate( + label_ids=Coalesce( + ArrayAgg( + "labels__id", + distinct=True, + filter=~Q(labels__id__isnull=True), + ), + Value([], output_field=ArrayField(UUIDField())), + ), + module_ids=Coalesce( + ArrayAgg( + "issue_module__module_id", + distinct=True, + filter=~Q(issue_module__module_id__isnull=True), + ), + Value([], output_field=ArrayField(UUIDField())), + ), + ) + + if field == "labels__id": + return queryset.annotate( + assignee_ids=Coalesce( + ArrayAgg( + "assignees__id", + distinct=True, + filter=~Q(assignees__id__isnull=True), + ), + Value([], output_field=ArrayField(UUIDField())), + ), + module_ids=Coalesce( + ArrayAgg( + "issue_module__module_id", + distinct=True, + filter=~Q(issue_module__module_id__isnull=True), + ), + Value([], output_field=ArrayField(UUIDField())), + ), + ) + + if field == "modules__id": + return queryset.annotate(modules__id=F("issue_module__module_id")).annotate( + label_ids=Coalesce( + ArrayAgg( + "labels__id", + distinct=True, + filter=~Q(labels__id__isnull=True), + ), + Value([], output_field=ArrayField(UUIDField())), + ), + assignee_ids=Coalesce( + ArrayAgg( + "assignees__id", + distinct=True, + filter=~Q(assignees__id__isnull=True), + ), + Value([], output_field=ArrayField(UUIDField())), + ), + ) + return queryset \ No newline at end of file diff --git a/apiserver/plane/utils/paginator.py b/apiserver/plane/utils/paginator.py index ba617762e..e3eb6a53b 100644 --- a/apiserver/plane/utils/paginator.py +++ b/apiserver/plane/utils/paginator.py @@ -1,6 +1,7 @@ # Python imports import math from collections.abc import Sequence +from collections import defaultdict # Django imports from django.db.models import Window, F, Count, Q @@ -154,12 +155,26 @@ class OffsetPaginator: class GroupedOffsetPaginator(OffsetPaginator): + + FIELD_MAPPER = { + "labels__id": "label_ids", + "assignees__id": "assignee_ids", + "modules__id": "module_ids", + } + def __init__( - self, queryset, group_by_field_name, group_by_fields, *args, **kwargs + self, + queryset, + group_by_field_name, + group_by_fields, + count_filter, + *args, + **kwargs, ): super().__init__(queryset, *args, **kwargs) self.group_by_field_name = group_by_field_name self.group_by_fields = group_by_fields + self.count_filter = count_filter def get_result(self, limit=100, cursor=None): # offset is page # @@ -189,10 +204,6 @@ class GroupedOffsetPaginator(OffsetPaginator): # Compute the results results = {} queryset = queryset.annotate( - # group_rank=Window( - # expression=DenseRank(), - # order_by=F(self.group_by_field_name).asc() - # ), row_number=Window( expression=RowNumber(), partition_by=[F(self.group_by_field_name)], @@ -237,17 +248,91 @@ class GroupedOffsetPaginator(OffsetPaginator): max_hits=max_hits, ) - def process_results(self, results): + def __get_total_queryset(self): + return self.queryset.values(self.group_by_field_name).annotate( + count=Count( + self.group_by_field_name, + filter=self.count_filter, + ) + ) + + def __get_total_dict(self): + total_group_dict = {} + for group in self.__get_total_queryset(): + total_group_dict[str(group.get(self.group_by_field_name))] = ( + total_group_dict.get( + str(group.get(self.group_by_field_name)), 0 + ) + + (1 if group.get("count") == 0 else group.get("count")) + ) + + return total_group_dict + + def __query_multi_grouper(self, results): + + total_group_dict = self.__get_total_dict() + + # Preparing a dict to keep track of group IDs associated with each label ID + result_group_mapping = defaultdict(set) + # Preparing a dict to group result by group ID + grouped_by_field_name = defaultdict(list) + + # Iterate over results to fill the above dictionaries + for result in results: + result_id = result["id"] + group_id = result[self.group_by_field_name] + result_group_mapping[str(result_id)].add(str(group_id)) + + def result_already_added(result, group): + for existing_issue in group: + if existing_issue["id"] == result["id"]: + return True + return False + + # Adding group_ids key to each issue and grouping by group_name + for result in results: + result_id = result["id"] + group_ids = list(result_group_mapping[str(result_id)]) + result[self.FIELD_MAPPER.get(self.group_by_field_name)] = ( + [] if "None" in group_ids else group_ids + ) + # If a result belongs to multiple groups, add it to each group + for group_id in group_ids: + if not result_already_added( + result, grouped_by_field_name[group_id] + ): + grouped_by_field_name[group_id].append(result) + + # Convert grouped_by_field_name back to a list for each group + processed_results = { + str(group_id): { + "results": issues, + "total_results": total_group_dict.get(str(group_id)), + } + for group_id, issues in grouped_by_field_name.items() + } + return processed_results + + def __query_grouper(self, results): + total_group_dict = self.__get_total_dict() processed_results = {} for result in results: group_value = str(result.get(self.group_by_field_name)) if group_value not in processed_results: processed_results[str(group_value)] = { "results": [], + "total_results": total_group_dict.get(group_value), } processed_results[str(group_value)]["results"].append(result) return processed_results + def process_results(self, results): + if self.group_by_field_name in self.FIELD_MAPPER: + processed_results = self.__query_multi_grouper(results=results) + else: + processed_results = self.__query_grouper(results=results) + return processed_results + class BasePaginator: """BasePaginator class can be inherited by any View to return a paginated view""" @@ -283,6 +368,7 @@ class BasePaginator: controller=None, group_by_fields=None, group_by_field_name=None, + count_filter=None, **paginator_kwargs, ): """Paginate the request""" @@ -301,6 +387,7 @@ class BasePaginator: if group_by_fields and group_by_field_name: paginator_kwargs["group_by_fields"] = group_by_fields paginator_kwargs["group_by_field_name"] = group_by_field_name + paginator_kwargs["count_filter"] = count_filter paginator = paginator_cls(**paginator_kwargs) try: