diff --git a/.github/workflows/codeql.yml b/.github/workflows/codeql.yml
index a858bcc59..d7b94d245 100644
--- a/.github/workflows/codeql.yml
+++ b/.github/workflows/codeql.yml
@@ -3,7 +3,7 @@ name: "CodeQL"
on:
workflow_dispatch:
push:
- branches: ["master"]
+ branches: ["develop", "preview", "master"]
pull_request:
branches: ["develop", "preview", "master"]
schedule:
diff --git a/apiserver/plane/api/serializers/issue.py b/apiserver/plane/api/serializers/issue.py
index b8f194b32..c78b109ef 100644
--- a/apiserver/plane/api/serializers/issue.py
+++ b/apiserver/plane/api/serializers/issue.py
@@ -1,32 +1,33 @@
-from lxml import html
+from django.core.exceptions import ValidationError
+from django.core.validators import URLValidator
# Django imports
from django.utils import timezone
-from django.core.validators import URLValidator
-from django.core.exceptions import ValidationError
+from lxml import html
# Third party imports
from rest_framework import serializers
# Module imports
from plane.db.models import (
- User,
Issue,
- State,
+ IssueActivity,
IssueAssignee,
- Label,
+ IssueAttachment,
+ IssueComment,
IssueLabel,
IssueLink,
- IssueComment,
- IssueAttachment,
- IssueActivity,
+ Label,
ProjectMember,
+ State,
+ User,
)
+
from .base import BaseSerializer
-from .cycle import CycleSerializer, CycleLiteSerializer
-from .module import ModuleSerializer, ModuleLiteSerializer
-from .user import UserLiteSerializer
+from .cycle import CycleLiteSerializer, CycleSerializer
+from .module import ModuleLiteSerializer, ModuleSerializer
from .state import StateLiteSerializer
+from .user import UserLiteSerializer
class IssueSerializer(BaseSerializer):
@@ -79,7 +80,7 @@ class IssueSerializer(BaseSerializer):
data["description_html"] = parsed_str
except Exception as e:
- raise serializers.ValidationError(f"Invalid HTML: {str(e)}")
+ raise serializers.ValidationError("Invalid HTML passed")
# Validate assignees are from project
if data.get("assignees", []):
@@ -294,7 +295,7 @@ class IssueLinkSerializer(BaseSerializer):
raise serializers.ValidationError("Invalid URL format.")
# Check URL scheme
- if not value.startswith(('http://', 'https://')):
+ if not value.startswith(("http://", "https://")):
raise serializers.ValidationError("Invalid URL scheme.")
return value
@@ -366,7 +367,7 @@ class IssueCommentSerializer(BaseSerializer):
data["comment_html"] = parsed_str
except Exception as e:
- raise serializers.ValidationError(f"Invalid HTML: {str(e)}")
+ raise serializers.ValidationError("Invalid HTML passed")
return data
diff --git a/apiserver/plane/app/views/module/base.py b/apiserver/plane/app/views/module/base.py
index 390e1acf0..a00e12464 100644
--- a/apiserver/plane/app/views/module/base.py
+++ b/apiserver/plane/app/views/module/base.py
@@ -1,19 +1,6 @@
# Python imports
import json
-# Django Imports
-from django.utils import timezone
-from django.db.models import (
- Prefetch,
- F,
- OuterRef,
- Exists,
- Count,
- Q,
- Func,
- Subquery,
- IntegerField,
-)
from django.contrib.postgres.aggregates import ArrayAgg
from django.contrib.postgres.fields import ArrayField
from django.db.models import (
@@ -21,9 +8,11 @@ from django.db.models import (
Exists,
F,
Func,
+ IntegerField,
OuterRef,
Prefetch,
Q,
+ Subquery,
UUIDField,
Value,
)
@@ -31,8 +20,6 @@ from django.db.models.functions import Coalesce
# Django Imports
from django.utils import timezone
-from django.utils.decorators import method_decorator
-from django.views.decorators.gzip import gzip_page
# Third party imports
from rest_framework import status
@@ -45,7 +32,6 @@ from plane.app.permissions import (
from plane.app.serializers import (
ModuleDetailSerializer,
ModuleFavoriteSerializer,
- ModuleIssueSerializer,
ModuleLinkSerializer,
ModuleSerializer,
ModuleUserPropertiesSerializer,
@@ -54,8 +40,6 @@ from plane.app.serializers import (
from plane.bgtasks.issue_activites_task import issue_activity
from plane.db.models import (
Issue,
- IssueAttachment,
- IssueLink,
Module,
ModuleFavorite,
ModuleIssue,
@@ -64,17 +48,6 @@ from plane.db.models import (
Project,
)
from plane.utils.analytics_plot import burndown_plot
-from plane.utils.grouper import (
- issue_group_values,
- issue_on_results,
- 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,
- SubGroupedOffsetPaginator,
-)
# Module imports
from .. import BaseAPIView, BaseViewSet, WebhookMixin
@@ -422,9 +395,11 @@ class ModuleViewSet(WebhookMixin, BaseViewSet):
"completion_chart": {},
}
- if queryset.first().start_date and queryset.first().target_date:
+ # Fetch the modules
+ modules = queryset.first()
+ if modules and modules.start_date and modules.target_date:
data["distribution"]["completion_chart"] = burndown_plot(
- queryset=queryset.first(),
+ queryset=modules,
slug=slug,
project_id=project_id,
module_id=pk,
diff --git a/apiserver/plane/app/views/workspace/estimate.py b/apiserver/plane/app/views/workspace/estimate.py
index 8ca24efd7..59a23d867 100644
--- a/apiserver/plane/app/views/workspace/estimate.py
+++ b/apiserver/plane/app/views/workspace/estimate.py
@@ -3,15 +3,10 @@ from rest_framework import status
from rest_framework.response import Response
# Module imports
+from plane.app.permissions import WorkspaceEntityPermission
from plane.app.serializers import WorkspaceEstimateSerializer
from plane.app.views.base import BaseAPIView
-from plane.db.models import Project, Estimate
-from plane.app.permissions import WorkspaceEntityPermission
-
-# Django imports
-from django.db.models import (
- Prefetch,
-)
+from plane.db.models import Estimate, Project
from plane.utils.cache import cache_response
diff --git a/apiserver/templates/emails/invitations/project_invitation.html b/apiserver/templates/emails/invitations/project_invitation.html
index 630a5eab3..def576601 100644
--- a/apiserver/templates/emails/invitations/project_invitation.html
+++ b/apiserver/templates/emails/invitations/project_invitation.html
@@ -1,349 +1,1815 @@
-
-
-
-
-
-
- {{ first_name }} invited you to join {{ project_name }} on Plane
-
-
-
-
-
-
-
-
+
+
+
+
-
-
-
-
-
-
-
+
+
+
+
+
+
+
+
+
+
+
-
-
-
-
-
+
+
+
+
+
+
+ |
+
+
+
+
+ |
+
+
+
+
+
+
+
-
- |
-
-
- |
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- |
-
-
- |
-
-
- |
-
-
- |
-
-
-
-
-
- |
-
-
- {{first_name}} has invited you to join the
- {{project_name}} project on Plane
-
- |
- |
-
-
- |
-
-
-
-
- |
-
-
-
-
-
- |
- |
- |
-
-
- |
-
-
- Note: Plane is still in its early days, not everything will be perfect yet, and hiccups may happen. Please let us know of any suggestions, ideas, or bugs that you encounter on our Discord or GitHub, and we will use your feedback to improve on our upcoming releases.
-
- |
- |
-
-
- |
- |
- |
-
-
- |
-
-
- |
-
-
- |
-
-
- |
+
+
+
+
+
+
+
+
+ |
+
+
+
+
+ |
+
+
+
+
+ |
+
+
+ |
+
+
+
+
+
+
+
+ |
+
+
+
+ {{first_name}}
+ has invited you to
+ join the
+
+
+
+ {{project_name}}
+ project on Plane
+
+
+ |
+
+
+ |
+
+
+ |
+
+
+
+
+ |
+
+
+
+
+
+
+
+ |
+
+
+ |
+
+
+ |
+
+
+
+
+ |
+
+
+
+ Note: Plane is still
+ in its early days, not
+ everything will be
+ perfect yet, and
+ hiccups may happen.
+ Please let us know of
+ any suggestions,
+ ideas, or bugs that
+ you encounter on our Discord
+ or GitHub, and we will use
+ your feedback to
+ improve on our
+ upcoming
+ releases.
+
+
+ |
+
+
+ |
+
+
+
+
+ |
+
+
+ |
+
+
+ |
+
+
+ |
+
+
+ |
-
- |
-
-
-
- |
-
-
-
-
-
- |
-
-
-
-
- |
-
-
- |
-
-
- |
-
-
- |
+
+
+
+ |
+ |
+
+
+
+
+ |
+
+
+ |
-
- |
-
-
-
+ |
+
+
+
+
+
+
+ |
+
+
+
+
+
+
+
+
+
+
+
+ |
+
+
+ |
+
+
+ |
+
+
+ |
+
+
+ |
+
+
+ |
+
+
+ |
+
+
+
+
+
+ |
+
+
+