dev: endpoints for my, other, recent and favorite pages, add sort order and color for pages (#499)

* dev: endpoints for my, other, recent and favorite pages, add sort order and color for pages

* dev: fix state attribute error while saving page blocks
This commit is contained in:
pablohashescobar 2023-03-24 00:13:26 +05:30 committed by GitHub
parent 053ae2063e
commit 3056727190
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 190 additions and 6 deletions

View File

@ -106,7 +106,10 @@ from plane.api.views import (
PageBlockViewSet, PageBlockViewSet,
PageFavoriteViewSet, PageFavoriteViewSet,
CreateIssueFromPageBlockEndpoint, CreateIssueFromPageBlockEndpoint,
RecentPagesEndpoint,
FavoritePagesEndpoint,
MyPagesEndpoint, MyPagesEndpoint,
CreatedbyOtherPagesEndpoint,
## End Pages ## End Pages
# Api Tokens # Api Tokens
ApiTokenEndpoint, ApiTokenEndpoint,
@ -980,9 +983,24 @@ urlpatterns = [
name="page-block-issues", name="page-block-issues",
), ),
path( path(
"workspaces/<str:slug>/projects/<uuid:project_id>/user/pages/", "workspaces/<str:slug>/projects/<uuid:project_id>/pages/recent-pages/",
RecentPagesEndpoint.as_view(),
name="recent-pages",
),
path(
"workspaces/<str:slug>/projects/<uuid:project_id>/pages/favorite-pages/",
FavoritePagesEndpoint.as_view(),
name="recent-pages",
),
path(
"workspaces/<str:slug>/projects/<uuid:project_id>/pages/my-pages/",
MyPagesEndpoint.as_view(), MyPagesEndpoint.as_view(),
name="my-pages", name="user-pages",
),
path(
"workspaces/<str:slug>/projects/<uuid:project_id>/pages/created-by-other-pages/",
CreatedbyOtherPagesEndpoint.as_view(),
name="created-by-other-pages",
), ),
## End Pages ## End Pages
# API Tokens # API Tokens

View File

@ -116,5 +116,8 @@ from .page import (
PageBlockViewSet, PageBlockViewSet,
PageFavoriteViewSet, PageFavoriteViewSet,
CreateIssueFromPageBlockEndpoint, CreateIssueFromPageBlockEndpoint,
RecentPagesEndpoint,
FavoritePagesEndpoint,
MyPagesEndpoint, MyPagesEndpoint,
CreatedbyOtherPagesEndpoint,
) )

View File

@ -1,6 +1,10 @@
# Python imports
from datetime import timedelta
# Django imports # Django imports
from django.db import IntegrityError from django.db import IntegrityError
from django.db.models import Exists, OuterRef, Q from django.db.models import Exists, OuterRef, Q
from django.utils import timezone
# Third party imports # Third party imports
from rest_framework import status from rest_framework import status
@ -31,6 +35,9 @@ class PageViewSet(BaseViewSet):
permission_classes = [ permission_classes = [
ProjectEntityPermission, ProjectEntityPermission,
] ]
search_fields = [
"name",
]
def get_queryset(self): def get_queryset(self):
subquery = PageFavorite.objects.filter( subquery = PageFavorite.objects.filter(
@ -50,6 +57,7 @@ class PageViewSet(BaseViewSet):
.select_related("workspace") .select_related("workspace")
.select_related("owned_by") .select_related("owned_by")
.annotate(is_favorite=Exists(subquery)) .annotate(is_favorite=Exists(subquery))
.order_by(self.request.GET.get("order_by", "-created_at"))
.prefetch_related("labels") .prefetch_related("labels")
.distinct() .distinct()
) )
@ -175,6 +183,10 @@ class PageFavoriteViewSet(BaseViewSet):
class CreateIssueFromPageBlockEndpoint(BaseAPIView): class CreateIssueFromPageBlockEndpoint(BaseAPIView):
permission_classes = [
ProjectEntityPermission,
]
def post(self, request, slug, project_id, page_id, page_block_id): def post(self, request, slug, project_id, page_id, page_block_id):
try: try:
page_block = PageBlock.objects.get( page_block = PageBlock.objects.get(
@ -183,7 +195,13 @@ class CreateIssueFromPageBlockEndpoint(BaseAPIView):
project_id=project_id, project_id=project_id,
page_id=page_id, page_id=page_id,
) )
issue = Issue.objects.create(name=page_block.name, project_id=project_id) issue = Issue.objects.create(
name=page_block.name,
project_id=project_id,
description=page_block.description,
description_html=page_block.description_html,
description_stripped=page_block.description_stripped,
)
_ = IssueAssignee.objects.create( _ = IssueAssignee.objects.create(
issue=issue, assignee=request.user, project_id=project_id issue=issue, assignee=request.user, project_id=project_id
) )
@ -203,12 +221,126 @@ class CreateIssueFromPageBlockEndpoint(BaseAPIView):
) )
class MyPagesEndpoint(BaseAPIView): class RecentPagesEndpoint(BaseAPIView):
permission_classes = [
ProjectEntityPermission,
]
def get(self, request, slug, project_id): def get(self, request, slug, project_id):
try: try:
pages = Page.objects.filter( subquery = PageFavorite.objects.filter(
user=request.user,
page_id=OuterRef("pk"),
project_id=project_id,
workspace__slug=slug,
)
pages = (
(
Page.objects.filter(
updated_at__gte=(timezone.now() - timedelta(days=7)),
workspace__slug=slug,
project_id=project_id,
)
.filter(project__project_projectmember__member=request.user)
.annotate(is_favorite=Exists(subquery))
.order_by("-updated_by")
)
.select_related("project")
.select_related("workspace")
.select_related("owned_by")
.prefetch_related("labels")
)
serializer = PageSerializer(pages, many=True)
return Response(serializer.data, status=status.HTTP_200_OK)
except Exception as e:
print(e)
return Response(
{"error": "Something went wrong please try again later"},
status=status.HTTP_400_BAD_REQUEST,
)
class FavoritePagesEndpoint(BaseAPIView):
permission_classes = [
ProjectEntityPermission,
]
def get(self, request, slug, project_id):
try:
subquery = PageFavorite.objects.filter(
user=request.user,
page_id=OuterRef("pk"),
project_id=project_id,
workspace__slug=slug,
)
pages = (
Page.objects.filter(
workspace__slug=slug,
project_id=project_id,
)
.annotate(is_favorite=Exists(subquery))
.filter(is_favorite=True)
.select_related("project")
.select_related("workspace")
.select_related("owned_by")
.prefetch_related("labels")
)
serializer = PageSerializer(pages, many=True)
return Response(serializer.data, status=status.HTTP_200_OK)
except Exception as e:
capture_exception(e)
return Response(
{"error": "Something went wrong please try again later"},
status=status.HTTP_400_BAD_REQUEST,
)
class MyPagesEndpoint(BaseAPIView):
permission_classes = [
ProjectEntityPermission,
]
def get(self, request, slug, project_id):
try:
pages = (
Page.objects.filter(
workspace__slug=slug, project_id=project_id, owned_by=request.user workspace__slug=slug, project_id=project_id, owned_by=request.user
) )
.select_related("project")
.select_related("workspace")
.select_related("owned_by")
.prefetch_related("labels")
)
serializer = PageSerializer(pages, many=True)
return Response(serializer.data, status=status.HTTP_200_OK)
except Exception as e:
capture_exception(e)
return Response(
{"error": "Something went wrong please try again later"},
status=status.HTTP_400_BAD_REQUEST,
)
class CreatedbyOtherPagesEndpoint(BaseAPIView):
permission_classes = [
ProjectEntityPermission,
]
def get(self, request, slug, project_id):
try:
pages = (
Page.objects.filter(
~Q(owned_by=request.user),
workspace__slug=slug,
project_id=project_id,
)
.select_related("project")
.select_related("workspace")
.select_related("owned_by")
.prefetch_related("labels")
)
serializer = PageSerializer(pages, many=True) serializer = PageSerializer(pages, many=True)
return Response(serializer.data, status=status.HTTP_200_OK) return Response(serializer.data, status=status.HTTP_200_OK)
except Exception as e: except Exception as e:

View File

@ -4,6 +4,7 @@ from django.conf import settings
# Module imports # Module imports
from . import ProjectBaseModel from . import ProjectBaseModel
from plane.utils.html_processor import strip_tags
class Page(ProjectBaseModel): class Page(ProjectBaseModel):
@ -17,6 +18,7 @@ class Page(ProjectBaseModel):
access = models.PositiveSmallIntegerField( access = models.PositiveSmallIntegerField(
choices=((0, "Public"), (1, "Private")), default=0 choices=((0, "Public"), (1, "Private")), default=0
) )
color = models.CharField(max_length=255, blank=True)
labels = models.ManyToManyField( labels = models.ManyToManyField(
"db.Label", blank=True, related_name="pages", through="db.PageLabel" "db.Label", blank=True, related_name="pages", through="db.PageLabel"
) )
@ -42,6 +44,35 @@ class PageBlock(ProjectBaseModel):
"db.Issue", on_delete=models.SET_NULL, related_name="blocks", null=True "db.Issue", on_delete=models.SET_NULL, related_name="blocks", null=True
) )
completed_at = models.DateTimeField(null=True) completed_at = models.DateTimeField(null=True)
sort_order = models.FloatField(default=65535)
def save(self, *args, **kwargs):
if self._state.adding:
largest_sort_order = PageBlock.objects.filter(
project=self.project, page=self.page
).aggregate(largest=models.Max("sort_order"))["largest"]
if largest_sort_order is not None:
self.sort_order = largest_sort_order + 10000
# Strip the html tags using html parser
self.description_stripped = (
None
if (self.description_html == "" or self.description_html is None)
else strip_tags(self.description_html)
)
if self.completed_at and self.issue:
try:
from plane.db.models import State, Issue
completed_state = State.objects.filter(
group="completed", project=self.project
).first()
if completed_state is not None:
Issue.objects.update(pk=self.issue_id, state=completed_state)
except ImportError:
pass
super(PageBlock, self).save(*args, **kwargs)
class Meta: class Meta:
verbose_name = "Page Block" verbose_name = "Page Block"