Skip to content

Commit

Permalink
Allow users to call the add-membership API
Browse files Browse the repository at this point in the history
Allow users (not just authclients) to call the add-membership API.
  • Loading branch information
seanh committed Dec 10, 2024
1 parent 19731f3 commit c522c82
Show file tree
Hide file tree
Showing 8 changed files with 388 additions and 22 deletions.
5 changes: 4 additions & 1 deletion h/security/permission_map.py
Original file line number Diff line number Diff line change
Expand Up @@ -66,7 +66,10 @@
[p.group_has_user_as_admin],
[p.group_has_user_as_moderator],
],
Permission.Group.MEMBER_ADD: [[p.group_matches_authenticated_client_authority]],
Permission.Group.MEMBER_ADD: [
[p.group_member_add],
[p.group_matches_authenticated_client_authority],
],
Permission.Group.MEMBER_REMOVE: [[p.group_member_remove]],
Permission.Group.MEMBER_EDIT: [[p.group_member_edit]],
Permission.Group.MODERATE: GROUP_MODERATE_PREDICATES.values(),
Expand Down
32 changes: 31 additions & 1 deletion h/security/predicates.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,11 @@
from itertools import chain

from h.models.group import GroupMembershipRoles, JoinableBy, ReadableBy, WriteableBy
from h.traversal import EditGroupMembershipContext, GroupMembershipContext
from h.traversal import (
AddGroupMembershipContext,
EditGroupMembershipContext,
GroupMembershipContext,
)


def requires(*parent_predicates):
Expand Down Expand Up @@ -217,6 +221,32 @@ def get_authenticated_users_membership():
)


@requires(authenticated_user, group_found)
def group_member_add(identity, context: AddGroupMembershipContext):
def get_authenticated_users_roles():
"""Return the authenticated users roles in the target group."""
for membership in identity.user.memberships:
if membership.group.id == context.group.id:
return membership.roles

return []

authenticated_users_roles = get_authenticated_users_roles()

if GroupMembershipRoles.OWNER in authenticated_users_roles:
# Owners can add members with any roles.
return True

if GroupMembershipRoles.ADMIN in authenticated_users_roles:
# Admins can add members with any role below admin.
return (
GroupMembershipRoles.OWNER not in context.roles
and GroupMembershipRoles.ADMIN not in context.roles
)

return False


@requires(authenticated_user, group_found)
def group_member_edit(
identity, context: EditGroupMembershipContext
Expand Down
2 changes: 2 additions & 0 deletions h/traversal/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -63,6 +63,7 @@
from h.traversal.annotation import AnnotationContext, AnnotationRoot
from h.traversal.group import GroupContext, GroupRequiredRoot, GroupRoot
from h.traversal.group_membership import (
AddGroupMembershipContext,
EditGroupMembershipContext,
GroupMembershipContext,
group_membership_api_factory,
Expand All @@ -82,6 +83,7 @@
"UserByIDRoot",
"UserRoot",
"GroupContext",
"AddGroupMembershipContext",
"EditGroupMembershipContext",
"GroupMembershipContext",
"group_membership_api_factory",
Expand Down
7 changes: 7 additions & 0 deletions h/traversal/group_membership.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,13 @@ class GroupMembershipContext:
membership: GroupMembership | None


@dataclass
class AddGroupMembershipContext:
group: Group
user: User
roles: list[GroupMembershipRoles]


@dataclass
class EditGroupMembershipContext:
group: Group
Expand Down
19 changes: 14 additions & 5 deletions h/views/api/group_members.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,13 +3,19 @@
from pyramid.config import not_
from pyramid.httpexceptions import HTTPConflict, HTTPNoContent, HTTPNotFound

from h.models import GroupMembershipRoles
from h.presenters import GroupMembershipJSONPresenter
from h.schemas.api.group_membership import EditGroupMembershipAPISchema
from h.schemas.pagination import PaginationQueryParamsSchema
from h.schemas.util import validate_query_params
from h.security import Permission
from h.services.group_members import ConflictError
from h.traversal import EditGroupMembershipContext, GroupContext, GroupMembershipContext
from h.traversal import (
AddGroupMembershipContext,
EditGroupMembershipContext,
GroupContext,
GroupMembershipContext,
)
from h.views.api.config import api_config
from h.views.api.helpers.json_payload import json_payload

Expand Down Expand Up @@ -94,7 +100,6 @@ def remove_member(context: GroupMembershipContext, request):
request_method="POST",
link_name="group.member.add",
description="Add a user to a group",
permission=Permission.Group.MEMBER_ADD,
)
def add_member(context: GroupMembershipContext, request):
if context.user.authority != context.group.authority:
Expand All @@ -104,9 +109,13 @@ def add_member(context: GroupMembershipContext, request):
appstruct = EditGroupMembershipAPISchema().validate(json_payload(request))
roles = appstruct["roles"]
else:
# This doesn't mean the membership will be created with no roles:
# default roles will be applied by the service.
roles = None
roles = [GroupMembershipRoles.MEMBER]

if not request.has_permission(
Permission.Group.MEMBER_ADD,
AddGroupMembershipContext(context.group, context.user, roles),
):
raise HTTPNotFound()

group_members_service = request.find_service(name="group_members")

Expand Down
249 changes: 236 additions & 13 deletions tests/functional/api/groups/members_test.py
Original file line number Diff line number Diff line change
Expand Up @@ -261,6 +261,242 @@ def test_it_returns_an_error_if_offset_and_limit_are_invalid(


class TestAddMember:
@pytest.mark.parametrize(
"authenticated_users_roles,json,expected_roles",
[
(
[GroupMembershipRoles.OWNER],
{"roles": [GroupMembershipRoles.OWNER]},
[GroupMembershipRoles.OWNER],
),
(
[GroupMembershipRoles.OWNER],
{"roles": [GroupMembershipRoles.ADMIN]},
[GroupMembershipRoles.ADMIN],
),
(
[GroupMembershipRoles.OWNER],
{"roles": [GroupMembershipRoles.MODERATOR]},
[GroupMembershipRoles.MODERATOR],
),
(
[GroupMembershipRoles.OWNER],
{"roles": [GroupMembershipRoles.MEMBER]},
[GroupMembershipRoles.MEMBER],
),
(
[GroupMembershipRoles.OWNER],
None,
[GroupMembershipRoles.MEMBER],
),
(
[GroupMembershipRoles.ADMIN],
{"roles": [GroupMembershipRoles.OWNER]},
None,
),
(
[GroupMembershipRoles.ADMIN],
{"roles": [GroupMembershipRoles.ADMIN]},
None,
),
(
[GroupMembershipRoles.ADMIN],
{"roles": [GroupMembershipRoles.MODERATOR]},
[GroupMembershipRoles.MODERATOR],
),
(
[GroupMembershipRoles.ADMIN],
{"roles": [GroupMembershipRoles.MEMBER]},
[GroupMembershipRoles.MEMBER],
),
(
[GroupMembershipRoles.ADMIN],
None,
[GroupMembershipRoles.MEMBER],
),
(
[GroupMembershipRoles.MODERATOR],
{"roles": [GroupMembershipRoles.OWNER]},
None,
),
(
[GroupMembershipRoles.MODERATOR],
{"roles": [GroupMembershipRoles.ADMIN]},
None,
),
(
[GroupMembershipRoles.MODERATOR],
{"roles": [GroupMembershipRoles.MODERATOR]},
None,
),
(
[GroupMembershipRoles.MODERATOR],
{"roles": [GroupMembershipRoles.MEMBER]},
None,
),
(
[GroupMembershipRoles.MODERATOR],
None,
None,
),
(
[GroupMembershipRoles.MEMBER],
{"roles": [GroupMembershipRoles.OWNER]},
None,
),
(
[GroupMembershipRoles.MEMBER],
{"roles": [GroupMembershipRoles.ADMIN]},
None,
),
(
[GroupMembershipRoles.MEMBER],
{"roles": [GroupMembershipRoles.MODERATOR]},
None,
),
(
[GroupMembershipRoles.MEMBER],
{"roles": [GroupMembershipRoles.MEMBER]},
None,
),
(
[GroupMembershipRoles.MEMBER],
None,
None,
),
],
)
def test_it(
self,
do_request,
group,
user,
authenticated_users_membership,
authenticated_users_roles,
json,
expected_roles,
):
authenticated_users_membership.roles = authenticated_users_roles

do_request(json=json, status=200 if expected_roles else 404)

roles = None
for membership in group.memberships:
if membership.user == user:
roles = membership.roles
break
assert roles == expected_roles

@pytest.mark.parametrize(
"roles,status",
[
([GroupMembershipRoles.OWNER], 200),
([GroupMembershipRoles.MODERATOR], 409),
],
)
def test_me_alias(self, roles, status, do_request, group, authenticated_user):
do_request(userid="me", json={"roles": roles}, status=status)

roles = None
for membership in group.memberships:
if membership.user == authenticated_user:
roles = membership.roles
assert roles == [GroupMembershipRoles.OWNER]

def test_it_does_nothing_if_the_user_is_already_a_member_of_the_group(
self, do_request, group, user
):
group.memberships.append(GroupMembership(user=user))

do_request()

assert user in group.members

def test_it_when_a_conflicting_membership_already_exists(
self, do_request, group, user
):
group.memberships.append(
GroupMembership(user=user, roles=[GroupMembershipRoles.MEMBER])
)

response = do_request(
json={"roles": [GroupMembershipRoles.MODERATOR]}, status=409
)

assert (
response.json["reason"]
== "The user is already a member of the group, with conflicting membership attributes"
)

def test_it_errors_if_the_pubid_is_unknown(self, do_request):
do_request(pubid="UNKNOWN_PUBID", status=404)

def test_it_errors_if_the_userid_is_unknown(self, do_request, group):
do_request(userid="acct:UNKOWN_USERNAME@{group.authority}", status=404)

def test_it_errors_if_the_userid_is_invalid(self, do_request):
do_request(userid="INVALID_USERID", status=404)

def test_it_errors_if_the_request_isnt_authenticated(self, do_request, headers):
del headers["Authorization"]

do_request(status=404)

def test_it_errors_if_the_authenticated_user_isnt_a_member_of_the_group(
self, do_request, factories, headers
):
headers.update(
token_authorization_header(factories.DeveloperToken(user=factories.User()))
)

do_request(status=404)

@pytest.fixture(autouse=True)
def group(self, factories):
return factories.Group()

@pytest.fixture(autouse=True)
def user(self, factories, group):
return factories.User(authority=group.authority)

@pytest.fixture(autouse=True)
def authenticated_user(self, db_session, factories, group):
return factories.User(authority=group.authority)

@pytest.fixture(autouse=True)
def authenticated_users_membership(self, db_session, authenticated_user, group):
membership = GroupMembership(
group=group, user=authenticated_user, roles=[GroupMembershipRoles.OWNER]
)
db_session.add(membership)
return membership

@pytest.fixture
def headers(self, factories, authenticated_user):
return token_authorization_header(
factories.DeveloperToken(user=authenticated_user)
)

@pytest.fixture
def do_request(self, app, db_session, group, user, headers):
def do_request(
pubid=group.pubid,
userid=user.userid,
json={"roles": ["member"]},
headers=headers,
status=200,
):
db_session.commit()
path = f"/api/groups/{pubid}/members/{userid}"
if json is None:
return app.post(path, headers=headers, status=status)

return app.post_json(path, json, headers=headers, status=status)

return do_request


class TestAddMemberWithAuthclientAuthentication:
@pytest.mark.parametrize(
"json,expected_roles",
[
Expand Down Expand Up @@ -312,19 +548,6 @@ def test_it_errors_if_the_userid_is_unknown(self, do_request, authclient):
def test_it_errors_if_the_userid_is_invalid(self, do_request):
do_request(userid="INVALID_USERID", status=404)

def test_it_errors_if_the_request_isnt_authenticated(self, do_request, headers):
del headers["Authorization"]

do_request(status=404)

def test_it_errors_if_the_request_has_token_authentication(
self, do_request, factories, user, headers
):
token = factories.DeveloperToken(user=user)
headers.update(token_authorization_header(token))

do_request(status=404)

def test_it_errors_if_the_groups_authority_doesnt_match(
self, do_request, factories
):
Expand Down
Loading

0 comments on commit c522c82

Please sign in to comment.