The new DRF version (3.16.0) adds a check on unique together and needs more fields to be loaded. To prevent an extra query, we select the owner value in the DB query.
666 lines
22 KiB
Python
666 lines
22 KiB
Python
"""API endpoints"""
|
|
|
|
import datetime
|
|
import operator
|
|
from functools import reduce
|
|
|
|
from django.conf import settings
|
|
from django.db.models import OuterRef, Q, Subquery, Value
|
|
from django.db.models.functions import Coalesce
|
|
from django.utils import timezone
|
|
from django.utils.decorators import method_decorator
|
|
from django.views.decorators.cache import cache_page
|
|
|
|
from rest_framework import (
|
|
decorators,
|
|
exceptions,
|
|
filters,
|
|
mixins,
|
|
pagination,
|
|
response,
|
|
throttling,
|
|
views,
|
|
viewsets,
|
|
)
|
|
from rest_framework.permissions import AllowAny
|
|
|
|
from core import models
|
|
from core.api import permissions
|
|
from core.api.client import serializers
|
|
from core.utils.raw_sql import gen_sql_filter_json_array
|
|
|
|
from mailbox_manager import enums
|
|
from mailbox_manager import models as domains_models
|
|
|
|
|
|
class NestedGenericViewSet(viewsets.GenericViewSet):
|
|
"""
|
|
A generic Viewset aims to be used in a nested route context.
|
|
e.g: `/api/v1.0/resource_1/<resource_1_pk>/resource_2/<resource_2_pk>/`
|
|
|
|
It allows to define all url kwargs and lookup fields to perform the lookup.
|
|
"""
|
|
|
|
lookup_fields: list[str] = ["pk"]
|
|
lookup_url_kwargs: list[str] = []
|
|
|
|
def __getattribute__(self, item):
|
|
"""
|
|
This method is overridden to allow to get the last lookup field or lookup url kwarg
|
|
when accessing the `lookup_field` or `lookup_url_kwarg` attribute. This is useful
|
|
to keep compatibility with all methods used by the parent class `GenericViewSet`.
|
|
"""
|
|
if item in ["lookup_field", "lookup_url_kwarg"]:
|
|
return getattr(self, item + "s", [None])[-1]
|
|
|
|
return super().__getattribute__(item)
|
|
|
|
def get_queryset(self):
|
|
"""
|
|
Get the list of items for this view.
|
|
|
|
`lookup_fields` attribute is enumerated here to perform the nested lookup.
|
|
"""
|
|
queryset = super().get_queryset()
|
|
|
|
# The last lookup field is removed to perform the nested lookup as it corresponds
|
|
# to the object pk, it is used within get_object method.
|
|
lookup_url_kwargs = (
|
|
self.lookup_url_kwargs[:-1]
|
|
if self.lookup_url_kwargs
|
|
else self.lookup_fields[:-1]
|
|
)
|
|
|
|
filter_kwargs = {}
|
|
for index, lookup_url_kwarg in enumerate(lookup_url_kwargs):
|
|
if lookup_url_kwarg not in self.kwargs:
|
|
raise KeyError(
|
|
f"Expected view {self.__class__.__name__} to be called with a URL "
|
|
f'keyword argument named "{lookup_url_kwarg}". Fix your URL conf, or '
|
|
"set the `.lookup_fields` attribute on the view correctly."
|
|
)
|
|
|
|
filter_kwargs.update(
|
|
{self.lookup_fields[index]: self.kwargs[lookup_url_kwarg]}
|
|
)
|
|
|
|
return queryset.filter(**filter_kwargs)
|
|
|
|
|
|
class SerializerPerActionMixin:
|
|
"""
|
|
A mixin to allow to define serializer classes for each action.
|
|
|
|
This mixin is useful to avoid to define a serializer class for each action in the
|
|
`get_serializer_class` method.
|
|
|
|
Example:
|
|
```
|
|
class MyViewSet(SerializerPerActionMixin, viewsets.GenericViewSet):
|
|
serializer_class = MySerializer
|
|
list_serializer_class = MyListSerializer
|
|
retrieve_serializer_class = MyRetrieveSerializer
|
|
```
|
|
"""
|
|
|
|
def get_serializer_class(self):
|
|
"""
|
|
Return the serializer class to use depending on the action.
|
|
"""
|
|
if serializer_class := getattr(self, f"{self.action}_serializer_class", None):
|
|
return serializer_class
|
|
return super().get_serializer_class()
|
|
|
|
|
|
class Pagination(pagination.PageNumberPagination):
|
|
"""Pagination to display no more than 100 objects per page sorted by creation date."""
|
|
|
|
max_page_size = 100
|
|
page_size_query_param = "page_size"
|
|
|
|
|
|
class BurstRateThrottle(throttling.UserRateThrottle):
|
|
"""
|
|
Throttle rate for minutes. See DRF section in settings for default value.
|
|
"""
|
|
|
|
scope = "burst"
|
|
|
|
|
|
class SustainedRateThrottle(throttling.UserRateThrottle):
|
|
"""
|
|
Throttle rate for hours. See DRF section in settings for default value.
|
|
"""
|
|
|
|
scope = "sustained"
|
|
|
|
|
|
# pylint: disable=too-many-ancestors
|
|
class ContactViewSet(
|
|
mixins.CreateModelMixin,
|
|
mixins.DestroyModelMixin,
|
|
mixins.RetrieveModelMixin,
|
|
mixins.UpdateModelMixin,
|
|
viewsets.GenericViewSet,
|
|
):
|
|
"""Contact ViewSet"""
|
|
|
|
permission_classes = [permissions.AccessPermission]
|
|
queryset = models.Contact.objects.select_related("user", "owner").all()
|
|
serializer_class = serializers.ContactSerializer
|
|
throttle_classes = [BurstRateThrottle, SustainedRateThrottle]
|
|
ordering_fields = ["full_name", "short_name", "created_at"]
|
|
ordering = ["full_name"]
|
|
|
|
def list(self, request, *args, **kwargs):
|
|
"""Limit listed users by a query with throttle protection."""
|
|
user = self.request.user
|
|
queryset = self.filter_queryset(self.get_queryset())
|
|
|
|
# List only contacts that:
|
|
queryset = queryset.filter(
|
|
# - is public (owner is None)
|
|
Q(owner__isnull=True)
|
|
# - are owned by the user
|
|
| Q(owner=user)
|
|
# - are profile contacts for a user from the same organization
|
|
| Q(user__organization_id=user.organization_id),
|
|
# - are not overriden by another contact
|
|
overridden_by__isnull=True,
|
|
)
|
|
|
|
# Search by case-insensitive and accent-insensitive on:
|
|
# - full name
|
|
# - short name
|
|
# - email (from data `emails` field)
|
|
if query := self.request.GET.get("q", ""):
|
|
queryset = queryset.filter(
|
|
Q(full_name__unaccent__icontains=query)
|
|
| Q(short_name__unaccent__icontains=query)
|
|
| Q(
|
|
id__in=gen_sql_filter_json_array(
|
|
queryset.model,
|
|
"data->'emails'",
|
|
"value",
|
|
query,
|
|
)
|
|
)
|
|
)
|
|
|
|
serializer = self.get_serializer(queryset, many=True)
|
|
return response.Response(serializer.data)
|
|
|
|
def perform_create(self, serializer):
|
|
"""Set the current user as owner of the newly created contact."""
|
|
user = self.request.user
|
|
serializer.validated_data["owner"] = user
|
|
return super().perform_create(serializer)
|
|
|
|
|
|
class OrganizationViewSet(
|
|
mixins.RetrieveModelMixin,
|
|
mixins.UpdateModelMixin,
|
|
viewsets.GenericViewSet,
|
|
):
|
|
"""
|
|
Organization ViewSet
|
|
|
|
GET /api/organizations/<organization_id>/
|
|
Return the organization details for the given id.
|
|
|
|
PUT /api/organizations/<organization_id>/
|
|
Update the organization details for the given id.
|
|
|
|
PATCH /api/organizations/<organization_id>/
|
|
Partially update the organization details for the given id.
|
|
"""
|
|
|
|
permission_classes = [permissions.AccessPermission]
|
|
queryset = models.Organization.objects.all()
|
|
serializer_class = serializers.OrganizationSerializer
|
|
throttle_classes = [BurstRateThrottle, SustainedRateThrottle]
|
|
|
|
def get_queryset(self):
|
|
"""Limit listed organizations to the one the user belongs to."""
|
|
return (
|
|
super()
|
|
.get_queryset()
|
|
.filter(pk=self.request.user.organization_id)
|
|
.annotate(
|
|
user_role=Subquery(
|
|
models.OrganizationAccess.objects.filter(
|
|
user=self.request.user, organization=OuterRef("pk")
|
|
).values("role")[:1]
|
|
)
|
|
)
|
|
)
|
|
|
|
|
|
class UserViewSet(
|
|
SerializerPerActionMixin,
|
|
mixins.UpdateModelMixin,
|
|
viewsets.GenericViewSet,
|
|
mixins.ListModelMixin,
|
|
):
|
|
"""
|
|
User viewset for all interactions with user infos and teams.
|
|
|
|
GET /api/users/&q=query
|
|
Return a list of users whose email or name matches the query.
|
|
"""
|
|
|
|
permission_classes = [permissions.IsSelf]
|
|
queryset = (
|
|
models.User.objects.select_related("organization").all().order_by("-created_at")
|
|
)
|
|
serializer_class = serializers.UserSerializer
|
|
get_me_serializer_class = serializers.UserMeSerializer
|
|
throttle_classes = [BurstRateThrottle, SustainedRateThrottle]
|
|
pagination_class = Pagination
|
|
|
|
def get_queryset(self):
|
|
"""Limit listed users by a query. Pagination and throttle protection apply."""
|
|
queryset = self.queryset
|
|
|
|
if self.action == "list":
|
|
# Exclude inactive contacts
|
|
queryset = queryset.filter(
|
|
is_active=True,
|
|
)
|
|
|
|
# Exclude all users already in the given team
|
|
if team_id := self.request.GET.get("team_id", ""):
|
|
queryset = queryset.exclude(teams__id=team_id)
|
|
|
|
# Search by case-insensitive and accent-insensitive
|
|
if query := self.request.GET.get("q", ""):
|
|
queryset = queryset.filter(
|
|
Q(name__unaccent__icontains=query)
|
|
| Q(email__unaccent__icontains=query)
|
|
)
|
|
|
|
return queryset
|
|
|
|
@decorators.action(
|
|
detail=False,
|
|
methods=["get"],
|
|
url_name="me",
|
|
url_path="me",
|
|
)
|
|
def get_me(self, request):
|
|
"""
|
|
Return information on currently logged user
|
|
"""
|
|
user = request.user
|
|
return response.Response(self.get_serializer(user).data)
|
|
|
|
|
|
class TeamViewSet(
|
|
mixins.CreateModelMixin,
|
|
mixins.DestroyModelMixin,
|
|
mixins.ListModelMixin,
|
|
mixins.RetrieveModelMixin,
|
|
mixins.UpdateModelMixin,
|
|
viewsets.GenericViewSet,
|
|
):
|
|
"""Team ViewSet"""
|
|
|
|
permission_classes = [permissions.TeamPermission, permissions.AccessPermission]
|
|
serializer_class = serializers.TeamSerializer
|
|
filter_backends = [filters.OrderingFilter]
|
|
ordering_fields = ["created_at", "name", "path"]
|
|
ordering = ["-created_at"]
|
|
queryset = models.Team.objects.all()
|
|
pagination_class = None
|
|
|
|
def get_queryset(self):
|
|
"""Custom queryset to get user related teams."""
|
|
teams_queryset = models.Team.objects.filter(
|
|
accesses__user=self.request.user,
|
|
)
|
|
depth_path = teams_queryset.values("depth", "path")
|
|
|
|
if not depth_path:
|
|
return models.Team.objects.none()
|
|
|
|
user_role_query = models.TeamAccess.objects.filter(
|
|
user=self.request.user, team=OuterRef("pk")
|
|
).values("role")[:1]
|
|
|
|
return (
|
|
models.Team.objects.prefetch_related("accesses", "service_providers")
|
|
.filter(
|
|
reduce(
|
|
operator.or_,
|
|
(
|
|
Q(
|
|
# The team the user has access to
|
|
depth=d["depth"],
|
|
path=d["path"],
|
|
)
|
|
| Q(
|
|
# The parent team the user has access to
|
|
depth__lt=d["depth"],
|
|
path__startswith=d["path"][: models.Team.steplen],
|
|
organization_id=self.request.user.organization_id,
|
|
)
|
|
for d in depth_path
|
|
),
|
|
),
|
|
)
|
|
# Abilities are computed based on logged-in user's role for the team
|
|
# and if the user does not have access, it's ok to consider them as a member
|
|
# because it's a parent team.
|
|
.annotate(
|
|
user_role=Coalesce(
|
|
Subquery(user_role_query), Value(models.RoleChoices.MEMBER.value)
|
|
)
|
|
)
|
|
)
|
|
|
|
def perform_create(self, serializer):
|
|
"""Set the current user as owner of the newly created team."""
|
|
team = serializer.save()
|
|
models.TeamAccess.objects.create(
|
|
team=team,
|
|
user=self.request.user,
|
|
role=models.RoleChoices.OWNER,
|
|
)
|
|
|
|
|
|
class TeamAccessViewSet(
|
|
mixins.CreateModelMixin,
|
|
mixins.DestroyModelMixin,
|
|
mixins.ListModelMixin,
|
|
mixins.RetrieveModelMixin,
|
|
mixins.UpdateModelMixin,
|
|
viewsets.GenericViewSet,
|
|
):
|
|
"""
|
|
API ViewSet for all interactions with team accesses.
|
|
|
|
GET /api/v1.0/teams/<team_id>/accesses/:<team_access_id>
|
|
Return list of all team accesses related to the logged-in user or one
|
|
team access if an id is provided.
|
|
|
|
POST /api/v1.0/teams/<team_id>/accesses/ with expected data:
|
|
- user: str
|
|
- role: str [owner|admin|member]
|
|
Return newly created team access
|
|
|
|
PUT /api/v1.0/teams/<team_id>/accesses/<team_access_id>/ with expected data:
|
|
- role: str [owner|admin|member]
|
|
Return updated team access
|
|
|
|
PATCH /api/v1.0/teams/<team_id>/accesses/<team_access_id>/ with expected data:
|
|
- role: str [owner|admin|member]
|
|
Return partially updated team access
|
|
|
|
DELETE /api/v1.0/teams/<team_id>/accesses/<team_access_id>/
|
|
Delete targeted team access
|
|
"""
|
|
|
|
lookup_field = "pk"
|
|
pagination_class = Pagination
|
|
permission_classes = [permissions.AccessPermission]
|
|
queryset = (
|
|
models.TeamAccess.objects.all().select_related("user").order_by("-created_at")
|
|
)
|
|
list_serializer_class = serializers.TeamAccessReadOnlySerializer
|
|
detail_serializer_class = serializers.TeamAccessSerializer
|
|
|
|
filter_backends = [filters.OrderingFilter]
|
|
ordering = ["role"]
|
|
ordering_fields = ["role", "user__email", "user__name"]
|
|
|
|
def get_permissions(self):
|
|
"""User only needs to be authenticated to list team accesses"""
|
|
if self.action == "list":
|
|
permission_classes = [permissions.IsAuthenticated]
|
|
else:
|
|
return super().get_permissions()
|
|
|
|
return [permission() for permission in permission_classes]
|
|
|
|
def get_serializer_context(self):
|
|
"""Extra context provided to the serializer class."""
|
|
context = super().get_serializer_context()
|
|
context["team_id"] = self.kwargs["team_id"]
|
|
return context
|
|
|
|
def get_serializer_class(self):
|
|
"""Chooses list or detail serializer according to the action."""
|
|
if self.action in {"list", "retrieve"}:
|
|
return self.list_serializer_class
|
|
return self.detail_serializer_class
|
|
|
|
def get_queryset(self):
|
|
"""Return the queryset according to the action."""
|
|
queryset = super().get_queryset()
|
|
queryset = queryset.filter(team=self.kwargs["team_id"])
|
|
|
|
if self.action in {"list", "retrieve"}:
|
|
if query := self.request.GET.get("q", ""):
|
|
queryset = queryset.filter(
|
|
Q(user__email__unaccent__icontains=query)
|
|
| Q(user__name__unaccent__icontains=query)
|
|
)
|
|
|
|
# Determine which role the logged-in user has in the team
|
|
user_role_query = models.TeamAccess.objects.filter(
|
|
user=self.request.user, team=self.kwargs["team_id"]
|
|
).values("role")[:1]
|
|
|
|
queryset = (
|
|
# The logged-in user should be part of a team to see its accesses
|
|
queryset.filter(
|
|
team__accesses__user=self.request.user,
|
|
)
|
|
# Abilities are computed based on logged-in user's role and
|
|
# the user role on each team access
|
|
.annotate(
|
|
user_role=Subquery(user_role_query),
|
|
)
|
|
.select_related("user")
|
|
.distinct()
|
|
)
|
|
return queryset
|
|
|
|
def destroy(self, request, *args, **kwargs):
|
|
"""Forbid deleting the last owner access"""
|
|
instance = self.get_object()
|
|
team = instance.team
|
|
|
|
# Check if the access being deleted is the last owner access for the team
|
|
if instance.role == "owner" and team.accesses.filter(role="owner").count() == 1:
|
|
return response.Response(
|
|
{"detail": "Cannot delete the last owner access for the team."},
|
|
status=400,
|
|
)
|
|
|
|
return super().destroy(request, *args, **kwargs)
|
|
|
|
def perform_update(self, serializer):
|
|
"""Check that we don't change the role if it leads to losing the last owner."""
|
|
instance = serializer.instance
|
|
|
|
# Check if the role is being updated and the new role is not "owner"
|
|
if (
|
|
"role" in self.request.data
|
|
and self.request.data["role"] != models.RoleChoices.OWNER
|
|
):
|
|
team = instance.team
|
|
# Check if the access being updated is the last owner access for the team
|
|
if (
|
|
instance.role == models.RoleChoices.OWNER
|
|
and team.accesses.filter(role=models.RoleChoices.OWNER).count() == 1
|
|
):
|
|
message = "Cannot change the role to a non-owner role for the last owner access."
|
|
raise exceptions.ValidationError({"role": message})
|
|
|
|
serializer.save()
|
|
|
|
|
|
class InvitationViewset(
|
|
mixins.CreateModelMixin,
|
|
mixins.ListModelMixin,
|
|
mixins.RetrieveModelMixin,
|
|
mixins.DestroyModelMixin,
|
|
viewsets.GenericViewSet,
|
|
):
|
|
"""API ViewSet for user invitations to team.
|
|
|
|
GET /api/v1.0/teams/<team_id>/invitations/:<invitation_id>/
|
|
Return list of invitations related to that team or or one
|
|
team access if an id is provided.
|
|
|
|
POST /api/v1.0/teams/<team_id>/invitations/ with expected data:
|
|
- email: str
|
|
- role: str [owner|admin|member]
|
|
- issuer : User, automatically added from user making query, if allowed
|
|
- team : Team, automatically added from requested URI
|
|
Return newly created invitation
|
|
|
|
PUT / PATCH : Not permitted. Instead of updating your invitation,
|
|
delete and create a new one.
|
|
|
|
DELETE /api/v1.0/teams/<team_id>/invitations/<invitation_id>/
|
|
Delete targeted invitation
|
|
"""
|
|
|
|
lookup_field = "id"
|
|
pagination_class = Pagination
|
|
permission_classes = [permissions.AccessPermission]
|
|
queryset = (
|
|
models.Invitation.objects.all().select_related("team").order_by("-created_at")
|
|
)
|
|
serializer_class = serializers.InvitationSerializer
|
|
|
|
def get_permissions(self):
|
|
"""User only needs to be authenticated to list invitations"""
|
|
if self.action == "list":
|
|
permission_classes = [permissions.IsAuthenticated]
|
|
else:
|
|
return super().get_permissions()
|
|
|
|
return [permission() for permission in permission_classes]
|
|
|
|
def get_serializer_context(self):
|
|
"""Extra context provided to the serializer class."""
|
|
context = super().get_serializer_context()
|
|
context["team_id"] = self.kwargs["team_id"]
|
|
return context
|
|
|
|
def get_queryset(self):
|
|
"""Return the queryset according to the action."""
|
|
queryset = super().get_queryset()
|
|
queryset = queryset.filter(team=self.kwargs["team_id"])
|
|
|
|
if self.action == "list":
|
|
# Determine which role the logged-in user has in the team
|
|
user_role_query = models.TeamAccess.objects.filter(
|
|
user=self.request.user, team=self.kwargs["team_id"]
|
|
).values("role")[:1]
|
|
|
|
queryset = (
|
|
# The logged-in user should be part of a team to see its accesses
|
|
queryset.filter(
|
|
team__accesses__user=self.request.user,
|
|
)
|
|
# Abilities are computed based on logged-in user's role and
|
|
# the user role on each team access
|
|
.annotate(user_role=Subquery(user_role_query))
|
|
.distinct()
|
|
)
|
|
return queryset
|
|
|
|
|
|
class ConfigView(views.APIView):
|
|
"""API ViewSet for sharing some public settings."""
|
|
|
|
permission_classes = [AllowAny]
|
|
|
|
def get(self, request):
|
|
"""
|
|
GET /api/v1.0/config/
|
|
Return a dictionary of public settings.
|
|
"""
|
|
array_settings = ["LANGUAGES", "FEATURES", "RELEASE", "COMMIT"]
|
|
dict_settings = {}
|
|
for setting in array_settings:
|
|
dict_settings[setting] = getattr(settings, setting)
|
|
|
|
return response.Response(dict_settings)
|
|
|
|
|
|
class StatView(views.APIView):
|
|
"""API ViewSet for sharing some public metrics."""
|
|
|
|
permission_classes = [AllowAny]
|
|
|
|
@method_decorator(cache_page(3600))
|
|
def get(self, request):
|
|
"""
|
|
GET /api/v1.0/stats/
|
|
Return a dictionary of public metrics.
|
|
"""
|
|
context = {
|
|
"total_users": models.User.objects.count(),
|
|
"mau": models.User.objects.filter(
|
|
last_login__gte=timezone.now() - datetime.timedelta(30)
|
|
).count(),
|
|
"teams": models.Team.objects.count(),
|
|
"domains": domains_models.MailDomain.objects.filter(
|
|
status=enums.MailDomainStatusChoices.ENABLED
|
|
).count(),
|
|
"mailboxes": domains_models.Mailbox.objects.count(),
|
|
}
|
|
return response.Response(context)
|
|
|
|
|
|
class ServiceProviderFilter(filters.BaseFilterBackend):
|
|
"""
|
|
Filter service providers.
|
|
"""
|
|
|
|
def filter_queryset(self, request, queryset, view):
|
|
"""
|
|
Filter service providers by audience or name.
|
|
"""
|
|
if name := request.GET.get("name"):
|
|
queryset = queryset.filter(name__icontains=name)
|
|
if audience_id := request.GET.get("audience_id"):
|
|
queryset = queryset.filter(audience_id=audience_id)
|
|
return queryset
|
|
|
|
|
|
class ServiceProviderViewSet(
|
|
mixins.ListModelMixin,
|
|
mixins.RetrieveModelMixin,
|
|
viewsets.GenericViewSet,
|
|
):
|
|
"""
|
|
API ViewSet for all interactions with service providers.
|
|
|
|
GET /api/v1.0/service-providers/
|
|
Return a list of service providers.
|
|
|
|
GET /api/v1.0/service-providers/<service_provider_id>/
|
|
Return a service provider.
|
|
"""
|
|
|
|
permission_classes = [permissions.IsAuthenticated]
|
|
queryset = models.ServiceProvider.objects.all()
|
|
serializer_class = serializers.ServiceProviderSerializer
|
|
throttle_classes = [BurstRateThrottle, SustainedRateThrottle]
|
|
pagination_class = Pagination
|
|
filter_backends = [filters.OrderingFilter, ServiceProviderFilter]
|
|
ordering = ["name"]
|
|
ordering_fields = ["name", "created_at"]
|
|
|
|
def get_queryset(self):
|
|
"""Filter the queryset to limit results to user's organization."""
|
|
queryset = super().get_queryset()
|
|
queryset = queryset.filter(organizations__id=self.request.user.organization_id)
|
|
return queryset
|