knock.py 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108
  1. # Copyright 2020 Sorunome
  2. # Copyright 2020 The Matrix.org Foundation C.I.C.
  3. #
  4. # Licensed under the Apache License, Version 2.0 (the "License");
  5. # you may not use this file except in compliance with the License.
  6. # You may obtain a copy of the License at
  7. #
  8. # http://www.apache.org/licenses/LICENSE-2.0
  9. #
  10. # Unless required by applicable law or agreed to in writing, software
  11. # distributed under the License is distributed on an "AS IS" BASIS,
  12. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. # See the License for the specific language governing permissions and
  14. # limitations under the License.
  15. import logging
  16. from typing import TYPE_CHECKING, Awaitable, Dict, List, Optional, Tuple
  17. from synapse.api.constants import Membership
  18. from synapse.api.errors import SynapseError
  19. from synapse.http.server import HttpServer
  20. from synapse.http.servlet import (
  21. RestServlet,
  22. parse_json_object_from_request,
  23. parse_strings_from_args,
  24. )
  25. from synapse.http.site import SynapseRequest
  26. from synapse.logging.opentracing import set_tag
  27. from synapse.rest.client.transactions import HttpTransactionCache
  28. from synapse.types import JsonDict, RoomAlias, RoomID
  29. if TYPE_CHECKING:
  30. from synapse.app.homeserver import HomeServer
  31. from ._base import client_patterns
  32. logger = logging.getLogger(__name__)
  33. class KnockRoomAliasServlet(RestServlet):
  34. """
  35. POST /knock/{roomIdOrAlias}
  36. """
  37. PATTERNS = client_patterns("/knock/(?P<room_identifier>[^/]*)")
  38. def __init__(self, hs: "HomeServer"):
  39. super().__init__()
  40. self.txns = HttpTransactionCache(hs)
  41. self.room_member_handler = hs.get_room_member_handler()
  42. self.auth = hs.get_auth()
  43. async def on_POST(
  44. self,
  45. request: SynapseRequest,
  46. room_identifier: str,
  47. txn_id: Optional[str] = None,
  48. ) -> Tuple[int, JsonDict]:
  49. requester = await self.auth.get_user_by_req(request)
  50. content = parse_json_object_from_request(request)
  51. event_content = None
  52. if "reason" in content:
  53. event_content = {"reason": content["reason"]}
  54. if RoomID.is_valid(room_identifier):
  55. room_id = room_identifier
  56. # twisted.web.server.Request.args is incorrectly defined as Optional[Any]
  57. args: Dict[bytes, List[bytes]] = request.args # type: ignore
  58. remote_room_hosts = parse_strings_from_args(
  59. args, "server_name", required=False
  60. )
  61. elif RoomAlias.is_valid(room_identifier):
  62. handler = self.room_member_handler
  63. room_alias = RoomAlias.from_string(room_identifier)
  64. room_id_obj, remote_room_hosts = await handler.lookup_room_alias(room_alias)
  65. room_id = room_id_obj.to_string()
  66. else:
  67. raise SynapseError(
  68. 400, "%s was not legal room ID or room alias" % (room_identifier,)
  69. )
  70. await self.room_member_handler.update_membership(
  71. requester=requester,
  72. target=requester.user,
  73. room_id=room_id,
  74. action=Membership.KNOCK,
  75. txn_id=txn_id,
  76. third_party_signed=None,
  77. remote_room_hosts=remote_room_hosts,
  78. content=event_content,
  79. )
  80. return 200, {"room_id": room_id}
  81. def on_PUT(
  82. self, request: SynapseRequest, room_identifier: str, txn_id: str
  83. ) -> Awaitable[Tuple[int, JsonDict]]:
  84. set_tag("txn_id", txn_id)
  85. return self.txns.fetch_or_execute_request(
  86. request, self.on_POST, request, room_identifier, txn_id
  87. )
  88. def register_servlets(hs: "HomeServer", http_server: HttpServer) -> None:
  89. KnockRoomAliasServlet(hs).register(http_server)