* WIP * WIP * WIP * WIP * Create home preference if not exist * chore: handled the unique state name validation (#6299) * fix: changed the response structure (#6301) * [WEB-1964]chore: cycles actions restructuring (#6298) * chore: cycles quick actions restructuring * chore: added additional actions to cycle list actions * chore: cycle quick action structure * chore: added additional actions to cycle list actions * chore: added end cycle hook * fix: updated end cycle export --------- Co-authored-by: gurusinath <gurusainath007@gmail.com> * fix: active cycle graph tooltip and endpoint validation (#6306) * [WEB-2870]feat: language support (#6215) * fix: adding language support package * fix: language support implementation using mobx * fix: adding more languages for support * fix: profile settings translations * feat: added language support for sidebar and user settings * feat: added language support for deactivation modal * fix: added project sync after transfer issues (#6200) * code refactor and improvement (#6203) * chore: package code refactoring * chore: component restructuring and refactor * chore: comment create improvement * refactor: enhance workspace and project wrapper modularity (#6207) * [WEB-2678]feat: added functionality to add labels directly from dropdown (#6211) * enhancement:added functionality to add features directly from dropdown * fix: fixed import order * fix: fixed lint errors * chore: added common component for project activity (#6212) * chore: added common component for project activity * fix: added enum * fix: added enum for initiatives * - Do not clear temp files that are locked. (#6214) - Handle edge cases in sync workspace * fix: labels empty state for drop down (#6216) * refactor: remove cn helper function from the editor package (#6217) * * feat: added language support to issue create modal in sidebar * fix: project activity type * * fix: added missing translations * fix: modified translation for plurals * fix: fixed spanish translation * dev: language type error in space user profile types * fix: type fixes * chore: added alpha tag --------- Co-authored-by: sriram veeraghanta <veeraghanta.sriram@gmail.com> Co-authored-by: Anmol Singh Bhatia <121005188+anmolsinghbhatia@users.noreply.github.com> Co-authored-by: Prateek Shourya <prateekshourya29@gmail.com> Co-authored-by: Akshita Goyal <36129505+gakshita@users.noreply.github.com> Co-authored-by: Satish Gandham <satish.iitg@gmail.com> Co-authored-by: Aaryan Khandelwal <65252264+aaryan610@users.noreply.github.com> Co-authored-by: gurusinath <gurusainath007@gmail.com> * feat: introduced stacked bar chart and tree map chart. (#6305) * feat: add issue attachment external endpoint (#6307) * [PE-97] chore: re-order pages options (#6303) * chore: re-order pages dropdown options * chore: re-order pages dropdown options * fix: remove localdb tracing * [WEB-2937] feat: home recent activies list endpoint (#6295) * Crud for wuick links * Validate quick link existence * Add custom method for destroy and retrieve * Add List method * Remove print statements * List all the workspace quick links * feat: endpoint to get recently active items * Resolve conflicts * Resolve conflicts * Add filter to only list required entities * Return required fields * Add filter * Add filter * fix: remove emoji edit for uneditable pages (#6304) * Removed duplicate imports * feat: patch api * Enable sort order to be updatable * Return key name only insert missing keys use serializer to return data * Remove random generation of sort_order * Remove name field Remove random generation of sort_order --------- Co-authored-by: Bavisetti Narayan <72156168+NarayanBavisetti@users.noreply.github.com> Co-authored-by: Vamsi Krishna <46787868+mathalav55@users.noreply.github.com> Co-authored-by: gurusinath <gurusainath007@gmail.com> Co-authored-by: Anmol Singh Bhatia <121005188+anmolsinghbhatia@users.noreply.github.com> Co-authored-by: sriram veeraghanta <veeraghanta.sriram@gmail.com> Co-authored-by: Prateek Shourya <prateekshourya29@gmail.com> Co-authored-by: Akshita Goyal <36129505+gakshita@users.noreply.github.com> Co-authored-by: Satish Gandham <satish.iitg@gmail.com> Co-authored-by: Aaryan Khandelwal <65252264+aaryan610@users.noreply.github.com> Co-authored-by: Nikhil <118773738+pablohashescobar@users.noreply.github.com>
259 lines
7 KiB
Python
259 lines
7 KiB
Python
# Third party imports
|
|
from rest_framework import serializers
|
|
from rest_framework import status
|
|
from rest_framework.response import Response
|
|
|
|
# Module imports
|
|
from .base import BaseSerializer, DynamicBaseSerializer
|
|
from .user import UserLiteSerializer, UserAdminLiteSerializer
|
|
|
|
|
|
from plane.db.models import (
|
|
Workspace,
|
|
WorkspaceMember,
|
|
WorkspaceMemberInvite,
|
|
WorkspaceTheme,
|
|
WorkspaceUserProperties,
|
|
WorkspaceUserLink,
|
|
UserRecentVisit,
|
|
Issue,
|
|
Page,
|
|
Project,
|
|
ProjectMember,
|
|
WorkspaceHomePreference,
|
|
)
|
|
from plane.utils.constants import RESTRICTED_WORKSPACE_SLUGS
|
|
|
|
# Django imports
|
|
from django.core.validators import URLValidator
|
|
from django.core.exceptions import ValidationError
|
|
|
|
|
|
class WorkSpaceSerializer(DynamicBaseSerializer):
|
|
owner = UserLiteSerializer(read_only=True)
|
|
total_members = serializers.IntegerField(read_only=True)
|
|
total_issues = serializers.IntegerField(read_only=True)
|
|
logo_url = serializers.CharField(read_only=True)
|
|
|
|
def validate_slug(self, value):
|
|
# Check if the slug is restricted
|
|
if value in RESTRICTED_WORKSPACE_SLUGS:
|
|
raise serializers.ValidationError("Slug is not valid")
|
|
return value
|
|
|
|
class Meta:
|
|
model = Workspace
|
|
fields = "__all__"
|
|
read_only_fields = [
|
|
"id",
|
|
"created_by",
|
|
"updated_by",
|
|
"created_at",
|
|
"updated_at",
|
|
"owner",
|
|
"logo_url",
|
|
]
|
|
|
|
|
|
class WorkspaceLiteSerializer(BaseSerializer):
|
|
class Meta:
|
|
model = Workspace
|
|
fields = ["name", "slug", "id"]
|
|
read_only_fields = fields
|
|
|
|
|
|
class WorkSpaceMemberSerializer(DynamicBaseSerializer):
|
|
member = UserLiteSerializer(read_only=True)
|
|
workspace = WorkspaceLiteSerializer(read_only=True)
|
|
|
|
class Meta:
|
|
model = WorkspaceMember
|
|
fields = "__all__"
|
|
|
|
|
|
class WorkspaceMemberMeSerializer(BaseSerializer):
|
|
draft_issue_count = serializers.IntegerField(read_only=True)
|
|
|
|
class Meta:
|
|
model = WorkspaceMember
|
|
fields = "__all__"
|
|
|
|
|
|
class WorkspaceMemberAdminSerializer(DynamicBaseSerializer):
|
|
member = UserAdminLiteSerializer(read_only=True)
|
|
workspace = WorkspaceLiteSerializer(read_only=True)
|
|
|
|
class Meta:
|
|
model = WorkspaceMember
|
|
fields = "__all__"
|
|
|
|
|
|
class WorkSpaceMemberInviteSerializer(BaseSerializer):
|
|
workspace = WorkSpaceSerializer(read_only=True)
|
|
total_members = serializers.IntegerField(read_only=True)
|
|
created_by_detail = UserLiteSerializer(read_only=True, source="created_by")
|
|
|
|
class Meta:
|
|
model = WorkspaceMemberInvite
|
|
fields = "__all__"
|
|
read_only_fields = [
|
|
"id",
|
|
"email",
|
|
"token",
|
|
"workspace",
|
|
"message",
|
|
"responded_at",
|
|
"created_at",
|
|
"updated_at",
|
|
]
|
|
|
|
|
|
class WorkspaceThemeSerializer(BaseSerializer):
|
|
class Meta:
|
|
model = WorkspaceTheme
|
|
fields = "__all__"
|
|
read_only_fields = ["workspace", "actor"]
|
|
|
|
|
|
class WorkspaceUserPropertiesSerializer(BaseSerializer):
|
|
class Meta:
|
|
model = WorkspaceUserProperties
|
|
fields = "__all__"
|
|
read_only_fields = ["workspace", "user"]
|
|
|
|
|
|
class WorkspaceUserLinkSerializer(BaseSerializer):
|
|
class Meta:
|
|
model = WorkspaceUserLink
|
|
fields = "__all__"
|
|
read_only_fields = ["workspace", "owner"]
|
|
|
|
def to_internal_value(self, data):
|
|
url = data.get("url", "")
|
|
if url and not url.startswith(("http://", "https://")):
|
|
data["url"] = "http://" + url
|
|
|
|
return super().to_internal_value(data)
|
|
|
|
def validate_url(self, value):
|
|
url_validator = URLValidator()
|
|
try:
|
|
url_validator(value)
|
|
except ValidationError:
|
|
raise serializers.ValidationError({"error": "Invalid URL format."})
|
|
|
|
return value
|
|
|
|
|
|
class IssueRecentVisitSerializer(serializers.ModelSerializer):
|
|
project_identifier = serializers.SerializerMethodField()
|
|
|
|
class Meta:
|
|
model = Issue
|
|
fields = [
|
|
"name",
|
|
"state",
|
|
"priority",
|
|
"assignees",
|
|
"type",
|
|
"sequence_id",
|
|
"project_id",
|
|
"project_identifier",
|
|
]
|
|
|
|
def get_project_identifier(self, obj):
|
|
project = obj.project
|
|
|
|
return project.identifier if project else None
|
|
|
|
|
|
class ProjectMemberSerializer(BaseSerializer):
|
|
member = UserLiteSerializer(read_only=True)
|
|
|
|
class Meta:
|
|
model = ProjectMember
|
|
fields = ["member"]
|
|
|
|
|
|
class ProjectRecentVisitSerializer(serializers.ModelSerializer):
|
|
project_members = serializers.SerializerMethodField()
|
|
|
|
class Meta:
|
|
model = Project
|
|
fields = ["id", "name", "logo_props", "project_members", "identifier"]
|
|
|
|
def get_project_members(self, obj):
|
|
members = ProjectMember.objects.filter(project_id=obj.id).select_related(
|
|
"member"
|
|
)
|
|
|
|
serializer = ProjectMemberSerializer(members, many=True)
|
|
return serializer.data
|
|
|
|
|
|
class PageRecentVisitSerializer(serializers.ModelSerializer):
|
|
project_id = serializers.SerializerMethodField()
|
|
project_identifier = serializers.SerializerMethodField()
|
|
|
|
class Meta:
|
|
model = Page
|
|
fields = [
|
|
"id",
|
|
"name",
|
|
"logo_props",
|
|
"project_id",
|
|
"owned_by",
|
|
"project_identifier",
|
|
]
|
|
|
|
def get_project_id(self, obj):
|
|
return (
|
|
obj.project_id
|
|
if hasattr(obj, "project_id")
|
|
else obj.projects.values_list("id", flat=True).first()
|
|
)
|
|
|
|
def get_project_identifier(self, obj):
|
|
project = obj.projects.first()
|
|
|
|
return project.identifier if project else None
|
|
|
|
|
|
def get_entity_model_and_serializer(entity_type):
|
|
entity_map = {
|
|
"issue": (Issue, IssueRecentVisitSerializer),
|
|
"page": (Page, PageRecentVisitSerializer),
|
|
"project": (Project, ProjectRecentVisitSerializer),
|
|
}
|
|
return entity_map.get(entity_type, (None, None))
|
|
|
|
|
|
class WorkspaceRecentVisitSerializer(BaseSerializer):
|
|
entity_data = serializers.SerializerMethodField()
|
|
|
|
class Meta:
|
|
model = UserRecentVisit
|
|
fields = ["id", "entity_name", "entity_identifier", "entity_data", "visited_at"]
|
|
read_only_fields = ["workspace", "owner", "created_by", "updated_by"]
|
|
|
|
def get_entity_data(self, obj):
|
|
entity_name = obj.entity_name
|
|
entity_identifier = obj.entity_identifier
|
|
|
|
entity_model, entity_serializer = get_entity_model_and_serializer(entity_name)
|
|
|
|
if entity_model and entity_serializer:
|
|
try:
|
|
entity = entity_model.objects.get(pk=entity_identifier)
|
|
|
|
return entity_serializer(entity).data
|
|
except entity_model.DoesNotExist:
|
|
return None
|
|
return None
|
|
|
|
|
|
class WorkspaceHomePreferenceSerializer(BaseSerializer):
|
|
class Meta:
|
|
model = WorkspaceHomePreference
|
|
fields = ["key", "is_enabled", "sort_order"]
|
|
read_only_fields = ["worspace", "created_by", "update_by"]
|