Merge pull request #6434 from matrix-org/erikj/msc2367_membership_reasons
Implement MSC 2367 - Membership Reasons
This commit is contained in:
commit
1c3a61529f
|
@ -0,0 +1 @@
|
||||||
|
Add support for MSC 2367, which allows specifying a reason on all membership events.
|
|
@ -1428,9 +1428,9 @@ class FederationHandler(BaseHandler):
|
||||||
return event
|
return event
|
||||||
|
|
||||||
@defer.inlineCallbacks
|
@defer.inlineCallbacks
|
||||||
def do_remotely_reject_invite(self, target_hosts, room_id, user_id):
|
def do_remotely_reject_invite(self, target_hosts, room_id, user_id, content):
|
||||||
origin, event, event_format_version = yield self._make_and_verify_event(
|
origin, event, event_format_version = yield self._make_and_verify_event(
|
||||||
target_hosts, room_id, user_id, "leave"
|
target_hosts, room_id, user_id, "leave", content=content,
|
||||||
)
|
)
|
||||||
# Mark as outlier as we don't have any state for this event; we're not
|
# Mark as outlier as we don't have any state for this event; we're not
|
||||||
# even in the room.
|
# even in the room.
|
||||||
|
|
|
@ -94,7 +94,9 @@ class RoomMemberHandler(object):
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
@abc.abstractmethod
|
@abc.abstractmethod
|
||||||
def _remote_reject_invite(self, requester, remote_room_hosts, room_id, target):
|
def _remote_reject_invite(
|
||||||
|
self, requester, remote_room_hosts, room_id, target, content
|
||||||
|
):
|
||||||
"""Attempt to reject an invite for a room this server is not in. If we
|
"""Attempt to reject an invite for a room this server is not in. If we
|
||||||
fail to do so we locally mark the invite as rejected.
|
fail to do so we locally mark the invite as rejected.
|
||||||
|
|
||||||
|
@ -104,6 +106,7 @@ class RoomMemberHandler(object):
|
||||||
reject invite
|
reject invite
|
||||||
room_id (str)
|
room_id (str)
|
||||||
target (UserID): The user rejecting the invite
|
target (UserID): The user rejecting the invite
|
||||||
|
content (dict): The content for the rejection event
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
Deferred[dict]: A dictionary to be returned to the client, may
|
Deferred[dict]: A dictionary to be returned to the client, may
|
||||||
|
@ -471,7 +474,7 @@ class RoomMemberHandler(object):
|
||||||
# send the rejection to the inviter's HS.
|
# send the rejection to the inviter's HS.
|
||||||
remote_room_hosts = remote_room_hosts + [inviter.domain]
|
remote_room_hosts = remote_room_hosts + [inviter.domain]
|
||||||
res = yield self._remote_reject_invite(
|
res = yield self._remote_reject_invite(
|
||||||
requester, remote_room_hosts, room_id, target
|
requester, remote_room_hosts, room_id, target, content,
|
||||||
)
|
)
|
||||||
return res
|
return res
|
||||||
|
|
||||||
|
@ -971,13 +974,15 @@ class RoomMemberMasterHandler(RoomMemberHandler):
|
||||||
)
|
)
|
||||||
|
|
||||||
@defer.inlineCallbacks
|
@defer.inlineCallbacks
|
||||||
def _remote_reject_invite(self, requester, remote_room_hosts, room_id, target):
|
def _remote_reject_invite(
|
||||||
|
self, requester, remote_room_hosts, room_id, target, content
|
||||||
|
):
|
||||||
"""Implements RoomMemberHandler._remote_reject_invite
|
"""Implements RoomMemberHandler._remote_reject_invite
|
||||||
"""
|
"""
|
||||||
fed_handler = self.federation_handler
|
fed_handler = self.federation_handler
|
||||||
try:
|
try:
|
||||||
ret = yield fed_handler.do_remotely_reject_invite(
|
ret = yield fed_handler.do_remotely_reject_invite(
|
||||||
remote_room_hosts, room_id, target.to_string()
|
remote_room_hosts, room_id, target.to_string(), content=content,
|
||||||
)
|
)
|
||||||
return ret
|
return ret
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
|
|
|
@ -55,7 +55,9 @@ class RoomMemberWorkerHandler(RoomMemberHandler):
|
||||||
|
|
||||||
return ret
|
return ret
|
||||||
|
|
||||||
def _remote_reject_invite(self, requester, remote_room_hosts, room_id, target):
|
def _remote_reject_invite(
|
||||||
|
self, requester, remote_room_hosts, room_id, target, content
|
||||||
|
):
|
||||||
"""Implements RoomMemberHandler._remote_reject_invite
|
"""Implements RoomMemberHandler._remote_reject_invite
|
||||||
"""
|
"""
|
||||||
return self._remote_reject_client(
|
return self._remote_reject_client(
|
||||||
|
@ -63,6 +65,7 @@ class RoomMemberWorkerHandler(RoomMemberHandler):
|
||||||
remote_room_hosts=remote_room_hosts,
|
remote_room_hosts=remote_room_hosts,
|
||||||
room_id=room_id,
|
room_id=room_id,
|
||||||
user_id=target.to_string(),
|
user_id=target.to_string(),
|
||||||
|
content=content,
|
||||||
)
|
)
|
||||||
|
|
||||||
def _user_joined_room(self, target, room_id):
|
def _user_joined_room(self, target, room_id):
|
||||||
|
|
|
@ -93,6 +93,7 @@ class ReplicationRemoteRejectInviteRestServlet(ReplicationEndpoint):
|
||||||
{
|
{
|
||||||
"requester": ...,
|
"requester": ...,
|
||||||
"remote_room_hosts": [...],
|
"remote_room_hosts": [...],
|
||||||
|
"content": { ... }
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
@ -107,7 +108,7 @@ class ReplicationRemoteRejectInviteRestServlet(ReplicationEndpoint):
|
||||||
self.clock = hs.get_clock()
|
self.clock = hs.get_clock()
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def _serialize_payload(requester, room_id, user_id, remote_room_hosts):
|
def _serialize_payload(requester, room_id, user_id, remote_room_hosts, content):
|
||||||
"""
|
"""
|
||||||
Args:
|
Args:
|
||||||
requester(Requester)
|
requester(Requester)
|
||||||
|
@ -118,12 +119,14 @@ class ReplicationRemoteRejectInviteRestServlet(ReplicationEndpoint):
|
||||||
return {
|
return {
|
||||||
"requester": requester.serialize(),
|
"requester": requester.serialize(),
|
||||||
"remote_room_hosts": remote_room_hosts,
|
"remote_room_hosts": remote_room_hosts,
|
||||||
|
"content": content,
|
||||||
}
|
}
|
||||||
|
|
||||||
async def _handle_request(self, request, room_id, user_id):
|
async def _handle_request(self, request, room_id, user_id):
|
||||||
content = parse_json_object_from_request(request)
|
content = parse_json_object_from_request(request)
|
||||||
|
|
||||||
remote_room_hosts = content["remote_room_hosts"]
|
remote_room_hosts = content["remote_room_hosts"]
|
||||||
|
event_content = content["content"]
|
||||||
|
|
||||||
requester = Requester.deserialize(self.store, content["requester"])
|
requester = Requester.deserialize(self.store, content["requester"])
|
||||||
|
|
||||||
|
@ -134,7 +137,7 @@ class ReplicationRemoteRejectInviteRestServlet(ReplicationEndpoint):
|
||||||
|
|
||||||
try:
|
try:
|
||||||
event = await self.federation_handler.do_remotely_reject_invite(
|
event = await self.federation_handler.do_remotely_reject_invite(
|
||||||
remote_room_hosts, room_id, user_id
|
remote_room_hosts, room_id, user_id, event_content,
|
||||||
)
|
)
|
||||||
ret = event.get_pdu_json()
|
ret = event.get_pdu_json()
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
|
|
|
@ -714,7 +714,7 @@ class RoomMembershipRestServlet(TransactionRestServlet):
|
||||||
target = UserID.from_string(content["user_id"])
|
target = UserID.from_string(content["user_id"])
|
||||||
|
|
||||||
event_content = None
|
event_content = None
|
||||||
if "reason" in content and membership_action in ["kick", "ban"]:
|
if "reason" in content:
|
||||||
event_content = {"reason": content["reason"]}
|
event_content = {"reason": content["reason"]}
|
||||||
|
|
||||||
await self.room_member_handler.update_membership(
|
await self.room_member_handler.update_membership(
|
||||||
|
|
|
@ -1180,3 +1180,143 @@ class PerRoomProfilesForbiddenTestCase(unittest.HomeserverTestCase):
|
||||||
|
|
||||||
res_displayname = channel.json_body["content"]["displayname"]
|
res_displayname = channel.json_body["content"]["displayname"]
|
||||||
self.assertEqual(res_displayname, self.displayname, channel.result)
|
self.assertEqual(res_displayname, self.displayname, channel.result)
|
||||||
|
|
||||||
|
|
||||||
|
class RoomMembershipReasonTestCase(unittest.HomeserverTestCase):
|
||||||
|
"""Tests that clients can add a "reason" field to membership events and
|
||||||
|
that they get correctly added to the generated events and propagated.
|
||||||
|
"""
|
||||||
|
|
||||||
|
servlets = [
|
||||||
|
synapse.rest.admin.register_servlets_for_client_rest_resource,
|
||||||
|
room.register_servlets,
|
||||||
|
login.register_servlets,
|
||||||
|
]
|
||||||
|
|
||||||
|
def prepare(self, reactor, clock, homeserver):
|
||||||
|
self.creator = self.register_user("creator", "test")
|
||||||
|
self.creator_tok = self.login("creator", "test")
|
||||||
|
|
||||||
|
self.second_user_id = self.register_user("second", "test")
|
||||||
|
self.second_tok = self.login("second", "test")
|
||||||
|
|
||||||
|
self.room_id = self.helper.create_room_as(self.creator, tok=self.creator_tok)
|
||||||
|
|
||||||
|
def test_join_reason(self):
|
||||||
|
reason = "hello"
|
||||||
|
request, channel = self.make_request(
|
||||||
|
"POST",
|
||||||
|
"/_matrix/client/r0/rooms/{}/join".format(self.room_id),
|
||||||
|
content={"reason": reason},
|
||||||
|
access_token=self.second_tok,
|
||||||
|
)
|
||||||
|
self.render(request)
|
||||||
|
self.assertEqual(channel.code, 200, channel.result)
|
||||||
|
|
||||||
|
self._check_for_reason(reason)
|
||||||
|
|
||||||
|
def test_leave_reason(self):
|
||||||
|
self.helper.join(self.room_id, user=self.second_user_id, tok=self.second_tok)
|
||||||
|
|
||||||
|
reason = "hello"
|
||||||
|
request, channel = self.make_request(
|
||||||
|
"POST",
|
||||||
|
"/_matrix/client/r0/rooms/{}/leave".format(self.room_id),
|
||||||
|
content={"reason": reason},
|
||||||
|
access_token=self.second_tok,
|
||||||
|
)
|
||||||
|
self.render(request)
|
||||||
|
self.assertEqual(channel.code, 200, channel.result)
|
||||||
|
|
||||||
|
self._check_for_reason(reason)
|
||||||
|
|
||||||
|
def test_kick_reason(self):
|
||||||
|
self.helper.join(self.room_id, user=self.second_user_id, tok=self.second_tok)
|
||||||
|
|
||||||
|
reason = "hello"
|
||||||
|
request, channel = self.make_request(
|
||||||
|
"POST",
|
||||||
|
"/_matrix/client/r0/rooms/{}/kick".format(self.room_id),
|
||||||
|
content={"reason": reason, "user_id": self.second_user_id},
|
||||||
|
access_token=self.second_tok,
|
||||||
|
)
|
||||||
|
self.render(request)
|
||||||
|
self.assertEqual(channel.code, 200, channel.result)
|
||||||
|
|
||||||
|
self._check_for_reason(reason)
|
||||||
|
|
||||||
|
def test_ban_reason(self):
|
||||||
|
self.helper.join(self.room_id, user=self.second_user_id, tok=self.second_tok)
|
||||||
|
|
||||||
|
reason = "hello"
|
||||||
|
request, channel = self.make_request(
|
||||||
|
"POST",
|
||||||
|
"/_matrix/client/r0/rooms/{}/ban".format(self.room_id),
|
||||||
|
content={"reason": reason, "user_id": self.second_user_id},
|
||||||
|
access_token=self.creator_tok,
|
||||||
|
)
|
||||||
|
self.render(request)
|
||||||
|
self.assertEqual(channel.code, 200, channel.result)
|
||||||
|
|
||||||
|
self._check_for_reason(reason)
|
||||||
|
|
||||||
|
def test_unban_reason(self):
|
||||||
|
reason = "hello"
|
||||||
|
request, channel = self.make_request(
|
||||||
|
"POST",
|
||||||
|
"/_matrix/client/r0/rooms/{}/unban".format(self.room_id),
|
||||||
|
content={"reason": reason, "user_id": self.second_user_id},
|
||||||
|
access_token=self.creator_tok,
|
||||||
|
)
|
||||||
|
self.render(request)
|
||||||
|
self.assertEqual(channel.code, 200, channel.result)
|
||||||
|
|
||||||
|
self._check_for_reason(reason)
|
||||||
|
|
||||||
|
def test_invite_reason(self):
|
||||||
|
reason = "hello"
|
||||||
|
request, channel = self.make_request(
|
||||||
|
"POST",
|
||||||
|
"/_matrix/client/r0/rooms/{}/invite".format(self.room_id),
|
||||||
|
content={"reason": reason, "user_id": self.second_user_id},
|
||||||
|
access_token=self.creator_tok,
|
||||||
|
)
|
||||||
|
self.render(request)
|
||||||
|
self.assertEqual(channel.code, 200, channel.result)
|
||||||
|
|
||||||
|
self._check_for_reason(reason)
|
||||||
|
|
||||||
|
def test_reject_invite_reason(self):
|
||||||
|
self.helper.invite(
|
||||||
|
self.room_id,
|
||||||
|
src=self.creator,
|
||||||
|
targ=self.second_user_id,
|
||||||
|
tok=self.creator_tok,
|
||||||
|
)
|
||||||
|
|
||||||
|
reason = "hello"
|
||||||
|
request, channel = self.make_request(
|
||||||
|
"POST",
|
||||||
|
"/_matrix/client/r0/rooms/{}/leave".format(self.room_id),
|
||||||
|
content={"reason": reason},
|
||||||
|
access_token=self.second_tok,
|
||||||
|
)
|
||||||
|
self.render(request)
|
||||||
|
self.assertEqual(channel.code, 200, channel.result)
|
||||||
|
|
||||||
|
self._check_for_reason(reason)
|
||||||
|
|
||||||
|
def _check_for_reason(self, reason):
|
||||||
|
request, channel = self.make_request(
|
||||||
|
"GET",
|
||||||
|
"/_matrix/client/r0/rooms/{}/state/m.room.member/{}".format(
|
||||||
|
self.room_id, self.second_user_id
|
||||||
|
),
|
||||||
|
access_token=self.creator_tok,
|
||||||
|
)
|
||||||
|
self.render(request)
|
||||||
|
self.assertEqual(channel.code, 200, channel.result)
|
||||||
|
|
||||||
|
event_content = channel.json_body
|
||||||
|
|
||||||
|
self.assertEqual(event_content.get("reason"), reason, channel.result)
|
||||||
|
|
Loading…
Reference in New Issue