mirror of
https://github.com/PostHog/posthog.git
synced 2024-11-22 08:40:03 +01:00
8d01d5ef54
Co-authored-by: github-actions <41898282+github-actions[bot]@users.noreply.github.com>
146 lines
5.5 KiB
Python
146 lines
5.5 KiB
Python
from typing import cast
|
|
|
|
from django.db import IntegrityError
|
|
from rest_framework import mixins, serializers, viewsets
|
|
from rest_framework.permissions import SAFE_METHODS, BasePermission
|
|
|
|
from ee.models.feature_flag_role_access import FeatureFlagRoleAccess
|
|
from ee.models.rbac.organization_resource_access import OrganizationResourceAccess
|
|
from ee.models.rbac.role import Role, RoleMembership
|
|
from posthog.api.organization_member import OrganizationMemberSerializer
|
|
from posthog.api.routing import TeamAndOrgViewSetMixin
|
|
from posthog.api.shared import UserBasicSerializer
|
|
from posthog.models import OrganizationMembership
|
|
from posthog.models.feature_flag import FeatureFlag
|
|
from posthog.models.user import User
|
|
|
|
|
|
class RolePermissions(BasePermission):
|
|
"""
|
|
Requires organization admin level to change object, allows everyone read
|
|
"""
|
|
|
|
message = "You need to have admin level or higher."
|
|
|
|
def has_permission(self, request, view):
|
|
organization = request.user.organization
|
|
|
|
requesting_membership: OrganizationMembership = OrganizationMembership.objects.get(
|
|
user_id=cast(User, request.user).id,
|
|
organization=organization,
|
|
)
|
|
|
|
if request.method in SAFE_METHODS or requesting_membership.level >= OrganizationMembership.Level.ADMIN:
|
|
return True
|
|
return False
|
|
|
|
|
|
class RoleSerializer(serializers.ModelSerializer):
|
|
created_by = UserBasicSerializer(read_only=True)
|
|
members = serializers.SerializerMethodField()
|
|
associated_flags = serializers.SerializerMethodField()
|
|
|
|
class Meta:
|
|
model = Role
|
|
fields = [
|
|
"id",
|
|
"name",
|
|
"feature_flags_access_level",
|
|
"created_at",
|
|
"created_by",
|
|
"members",
|
|
"associated_flags",
|
|
]
|
|
read_only_fields = ["id", "created_at", "created_by"]
|
|
|
|
def validate_name(self, name):
|
|
if Role.objects.filter(name__iexact=name, organization=self.context["request"].user.organization).exists():
|
|
raise serializers.ValidationError("There is already a role with this name.", code="unique")
|
|
return name
|
|
|
|
def create(self, validated_data):
|
|
organization = self.context["request"].user.organization
|
|
validated_data["organization"] = organization
|
|
try:
|
|
default_flags_org_setting = OrganizationResourceAccess.objects.get(
|
|
organization=organization,
|
|
resource=OrganizationResourceAccess.Resources.FEATURE_FLAGS,
|
|
).access_level
|
|
except OrganizationResourceAccess.DoesNotExist:
|
|
default_flags_org_setting = OrganizationResourceAccess.AccessLevel.CAN_ALWAYS_EDIT
|
|
validated_data["feature_flags_access_level"] = default_flags_org_setting
|
|
return super().create(validated_data)
|
|
|
|
def get_members(self, role: Role):
|
|
members = RoleMembership.objects.filter(role=role)
|
|
return RoleMembershipSerializer(members, many=True).data
|
|
|
|
def get_associated_flags(self, role: Role):
|
|
associated_flags: list[dict] = []
|
|
|
|
role_access_objects = FeatureFlagRoleAccess.objects.filter(role=role).values_list("feature_flag_id")
|
|
flags = FeatureFlag.objects.filter(id__in=role_access_objects)
|
|
for flag in flags:
|
|
associated_flags.append({"id": flag.id, "key": flag.key})
|
|
return associated_flags
|
|
|
|
|
|
class RoleViewSet(
|
|
TeamAndOrgViewSetMixin,
|
|
mixins.ListModelMixin,
|
|
mixins.CreateModelMixin,
|
|
mixins.RetrieveModelMixin,
|
|
mixins.UpdateModelMixin,
|
|
mixins.DestroyModelMixin,
|
|
viewsets.GenericViewSet,
|
|
):
|
|
scope_object = "organization"
|
|
permission_classes = [RolePermissions]
|
|
serializer_class = RoleSerializer
|
|
queryset = Role.objects.all()
|
|
|
|
def safely_get_queryset(self, queryset):
|
|
return queryset.filter(**self.request.GET.dict())
|
|
|
|
|
|
class RoleMembershipSerializer(serializers.ModelSerializer):
|
|
user = UserBasicSerializer(read_only=True)
|
|
organization_member = OrganizationMemberSerializer(read_only=True)
|
|
role_id = serializers.UUIDField(read_only=True)
|
|
user_uuid = serializers.UUIDField(required=True, write_only=True)
|
|
|
|
class Meta:
|
|
model = RoleMembership
|
|
fields = ["id", "role_id", "organization_member", "user", "joined_at", "updated_at", "user_uuid"]
|
|
read_only_fields = ["id", "role_id", "organization_member", "user", "joined_at", "updated_at"]
|
|
|
|
def create(self, validated_data):
|
|
user_uuid = validated_data.pop("user_uuid")
|
|
try:
|
|
validated_data["organization_member"] = OrganizationMembership.objects.select_related("user").get(
|
|
organization_id=self.context["organization_id"], user__uuid=user_uuid, user__is_active=True
|
|
)
|
|
|
|
validated_data["user"] = validated_data["organization_member"].user
|
|
except OrganizationMembership.DoesNotExist:
|
|
raise serializers.ValidationError("User does not exist.")
|
|
validated_data["role_id"] = self.context["role_id"]
|
|
try:
|
|
return super().create(validated_data)
|
|
except IntegrityError:
|
|
raise serializers.ValidationError("User is already part of the role.")
|
|
|
|
|
|
class RoleMembershipViewSet(
|
|
TeamAndOrgViewSetMixin,
|
|
mixins.ListModelMixin,
|
|
mixins.CreateModelMixin,
|
|
mixins.DestroyModelMixin,
|
|
viewsets.GenericViewSet,
|
|
):
|
|
scope_object = "organization"
|
|
permission_classes = [RolePermissions]
|
|
serializer_class = RoleMembershipSerializer
|
|
queryset = RoleMembership.objects.select_related("role")
|
|
filter_rewrite_rules = {"organization_id": "role__organization_id"}
|