Skip to content

Commit

Permalink
fix: Cannot use an API Key to add users to a group (#4362)
Browse files Browse the repository at this point in the history
  • Loading branch information
matthewelwell authored Jul 23, 2024
1 parent ad99f45 commit 0390075
Show file tree
Hide file tree
Showing 4 changed files with 75 additions and 33 deletions.
5 changes: 5 additions & 0 deletions api/api_keys/user.py
Original file line number Diff line number Diff line change
Expand Up @@ -64,6 +64,11 @@ def is_project_admin(self, project: "Project") -> bool:
def is_environment_admin(self, environment: "Environment") -> bool:
return is_master_api_key_environment_admin(self.key, environment)

def is_group_admin(self, group_id: int) -> bool:
# This is added to match the interface of the FFAdminUser model,
# but a MasterAPIKey cannot be a group admin.
return False

def has_project_permission(
self, permission: str, project: "Project", tag_ids: typing.List[int] = None
) -> bool:
Expand Down
83 changes: 54 additions & 29 deletions api/tests/unit/users/test_unit_users_views.py
Original file line number Diff line number Diff line change
Expand Up @@ -142,8 +142,7 @@ def test_can_join_organisation_as_admin_if_invite_role_is_admin(


def test_admin_can_update_role_for_a_user_in_organisation(
admin_user: FFAdminUser,
admin_client: APIClient,
admin_client_new: APIClient,
organisation: Organisation,
):
# Given
Expand All @@ -156,7 +155,7 @@ def test_admin_can_update_role_for_a_user_in_organisation(
data = {"role": OrganisationRole.ADMIN.name}

# When
response = admin_client.post(url, data=data)
response = admin_client_new.post(url, data=data)

# Then
assert response.status_code == status.HTTP_200_OK
Expand All @@ -170,7 +169,7 @@ def test_admin_can_update_role_for_a_user_in_organisation(

def test_admin_can_get_users_in_organisation(
admin_user: FFAdminUser,
admin_client: APIClient,
admin_client_new: APIClient,
staff_user: FFAdminUser,
organisation: Organisation,
django_assert_num_queries: DjangoAssertNumQueries,
Expand All @@ -190,7 +189,7 @@ def test_admin_can_get_users_in_organisation(

# When
with django_assert_num_queries(5):
response = admin_client.get(url)
response = admin_client_new.get(url)

# Then
assert response.status_code == status.HTTP_200_OK
Expand Down Expand Up @@ -250,7 +249,7 @@ def test_org_user_can_exclude_themself_when_getting_users_in_organisation(

def test_organisation_admin_can_interact_with_groups(
organisation: Organisation,
admin_client: APIClient,
admin_client_new: APIClient,
) -> None:
# Given
# Create a group
Expand All @@ -260,13 +259,13 @@ def test_organisation_admin_can_interact_with_groups(
)

# When / Then
response = admin_client.post(url, data=create_data)
response = admin_client_new.post(url, data=create_data)
assert response.status_code == status.HTTP_201_CREATED
assert UserPermissionGroup.objects.filter(name=create_data["name"]).exists()
group_id = response.json()["id"]

# Group appears in the groups list
response = admin_client.get(url)
response = admin_client_new.get(url)
assert response.status_code == status.HTTP_200_OK
assert response.json()["results"][0]["name"] == "Test Group"

Expand All @@ -277,16 +276,16 @@ def test_organisation_admin_can_interact_with_groups(
args=[organisation.id, group_id],
)

response = admin_client.patch(url, data=update_data)
response = admin_client_new.patch(url, data=update_data)
assert response.status_code == status.HTTP_200_OK

# Update is reflected when getting the group
response = admin_client.get(url)
response = admin_client_new.get(url)
assert response.status_code == status.HTTP_200_OK
assert response.data["name"] == update_data["name"]

# Delete the group
response = admin_client.delete(url)
response = admin_client_new.delete(url)
assert response.status_code == status.HTTP_204_NO_CONTENT
assert not UserPermissionGroup.objects.filter(name=update_data["name"]).exists()

Expand Down Expand Up @@ -362,7 +361,33 @@ def test_can_add_multiple_users_including_current_user(
staff_user: FFAdminUser,
organisation: Organisation,
admin_user: FFAdminUser,
admin_client: APIClient,
admin_client_new: APIClient,
) -> None:
# Given
group = UserPermissionGroup.objects.create(
name="Test Group", organisation=organisation
)
url = reverse(
"api-v1:organisations:organisation-groups-add-users",
args=[organisation.id, group.id],
)
data = {"user_ids": [admin_user.id, staff_user.id]}

# When
response = admin_client_new.post(
url, data=json.dumps(data), content_type="application/json"
)

# Then
assert response.status_code == status.HTTP_200_OK
assert all(user in group.users.all() for user in [admin_user, staff_user])


def test_can_add_users_with_master_api_key(
staff_user: FFAdminUser,
organisation: Organisation,
admin_user: FFAdminUser,
admin_master_api_key_client: APIClient,
) -> None:
# Given
group = UserPermissionGroup.objects.create(
Expand All @@ -375,7 +400,7 @@ def test_can_add_multiple_users_including_current_user(
data = {"user_ids": [admin_user.id, staff_user.id]}

# When
response = admin_client.post(
response = admin_master_api_key_client.post(
url, data=json.dumps(data), content_type="application/json"
)

Expand All @@ -385,7 +410,7 @@ def test_can_add_multiple_users_including_current_user(


def test_cannot_add_user_from_another_organisation(
admin_client: APIClient,
admin_client_new: APIClient,
organisation: Organisation,
):
# Given
Expand All @@ -402,7 +427,7 @@ def test_cannot_add_user_from_another_organisation(
data = {"user_ids": [another_user.id]}

# When
response = admin_client.post(
response = admin_client_new.post(
url, data=json.dumps(data), content_type="application/json"
)

Expand All @@ -413,7 +438,7 @@ def test_cannot_add_user_from_another_organisation(
def test_cannot_add_same_user_twice(
staff_user: FFAdminUser,
organisation: Organisation,
admin_client: APIClient,
admin_client_new: APIClient,
) -> None:
# Given
group = UserPermissionGroup.objects.create(
Expand All @@ -427,7 +452,7 @@ def test_cannot_add_same_user_twice(
data = {"user_ids": [staff_user.id]}

# When
admin_client.post(url, data=json.dumps(data), content_type="application/json")
admin_client_new.post(url, data=json.dumps(data), content_type="application/json")

# Then
assert staff_user in group.users.all() and group.users.count() == 1
Expand All @@ -437,7 +462,7 @@ def test_remove_users_from_group(
staff_user: FFAdminUser,
organisation: Organisation,
admin_user: FFAdminUser,
admin_client: APIClient,
admin_client_new: APIClient,
) -> None:
# Given
group = UserPermissionGroup.objects.create(
Expand All @@ -452,7 +477,7 @@ def test_remove_users_from_group(
data = {"user_ids": [staff_user.id]}

# When
admin_client.post(url, data=json.dumps(data), content_type="application/json")
admin_client_new.post(url, data=json.dumps(data), content_type="application/json")

# Then
# staff user has been removed
Expand All @@ -465,7 +490,7 @@ def test_remove_users_from_group(
def test_remove_users_silently_fails_if_user_not_in_group(
staff_user: FFAdminUser,
organisation: Organisation,
admin_client: APIClient,
admin_client_new: APIClient,
admin_user: FFAdminUser,
) -> None:
# Given
Expand All @@ -480,7 +505,7 @@ def test_remove_users_silently_fails_if_user_not_in_group(
data = {"user_ids": [staff_user.id]}

# When
response = admin_client.post(
response = admin_client_new.post(
url, data=json.dumps(data), content_type="application/json"
)

Expand All @@ -492,7 +517,7 @@ def test_remove_users_silently_fails_if_user_not_in_group(


def test_user_permission_group_can_update_is_default(
admin_client, organisation, user_permission_group
admin_client_new, organisation, user_permission_group
):
# Given
args = [organisation.id, user_permission_group.id]
Expand All @@ -501,7 +526,7 @@ def test_user_permission_group_can_update_is_default(
data = {"is_default": True, "name": user_permission_group.name}

# When
response = admin_client.put(url, data=data)
response = admin_client_new.put(url, data=data)

# Then
assert response.status_code == status.HTTP_200_OK
Expand All @@ -513,7 +538,7 @@ def test_user_permission_group_can_update_is_default(


def test_user_permission_group_can_update_external_id(
admin_client, organisation, user_permission_group
admin_client_new, organisation, user_permission_group
):
# Given
args = [organisation.id, user_permission_group.id]
Expand All @@ -523,15 +548,15 @@ def test_user_permission_group_can_update_external_id(
data = {"external_id": external_id, "name": user_permission_group.name}

# When
response = admin_client.put(url, data=data)
response = admin_client_new.put(url, data=data)

# Then
assert response.status_code == status.HTTP_200_OK
assert response.json()["external_id"] == external_id


def test_users_in_organisation_have_last_login(
admin_client, organisation, rf, mocker, admin_user
admin_client_new, organisation, rf, mocker, admin_user
):
# Given
req = rf.get("/")
Expand All @@ -544,15 +569,15 @@ def test_users_in_organisation_have_last_login(
)

# When
res = admin_client.get(url)
res = admin_client_new.get(url)

# Then
assert res.json()[0]["last_login"] is not None
assert res.status_code == status.HTTP_200_OK


def test_retrieve_user_permission_group_includes_group_admin(
admin_client, admin_user, organisation, user_permission_group
admin_client_new, admin_user, organisation, user_permission_group
):
# Given
group_admin_user = FFAdminUser.objects.create(email="[email protected]")
Expand All @@ -565,7 +590,7 @@ def test_retrieve_user_permission_group_includes_group_admin(
)

# When
response = admin_client.get(url)
response = admin_client_new.get(url)

# Then
assert response.status_code == status.HTTP_200_OK
Expand Down
4 changes: 4 additions & 0 deletions api/users/abc.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,10 @@ def is_project_admin(self, project: "Project") -> bool:
def is_environment_admin(self, environment: "Environment") -> bool:
raise NotImplementedError()

@abstractmethod
def is_group_admin(self, group_id: int) -> bool:
raise NotImplementedError()

@abstractmethod
def has_project_permission(self, permission: str, project: "Project") -> bool:
raise NotImplementedError()
Expand Down
16 changes: 12 additions & 4 deletions api/users/views.py
Original file line number Diff line number Diff line change
Expand Up @@ -227,8 +227,12 @@ def add_users(self, request, organisation_pk, pk):
group = self.get_object()
user_ids = request.data["user_ids"]

if request.user.id in user_ids and not request.user.is_organisation_admin(
Organisation.objects.get(pk=organisation_pk)
if (
isinstance(request.user, FFAdminUser)
and request.user.id in user_ids
and not request.user.is_organisation_admin(
Organisation.objects.get(pk=organisation_pk)
)
):
raise PermissionDenied("Non-admin users cannot add themselves to a group.")

Expand All @@ -248,8 +252,12 @@ def remove_users(self, request, organisation_pk, pk):
group = self.get_object()
user_ids = request.data["user_ids"]

if request.user.id in user_ids and not request.user.is_organisation_admin(
Organisation.objects.get(pk=organisation_pk)
if (
isinstance(request.user, FFAdminUser)
and request.user.id in user_ids
and not request.user.is_organisation_admin(
Organisation.objects.get(pk=organisation_pk)
)
):
raise PermissionDenied(
"Non-admin users cannot remove themselves from a group."
Expand Down

0 comments on commit 0390075

Please sign in to comment.