__init__.py 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. # -*- coding: utf-8 -*-
  2. # Copyright 2014-2016 OpenMarket Ltd
  3. # Copyright 2018-2019 New Vector Ltd
  4. #
  5. # Licensed under the Apache License, Version 2.0 (the "License");
  6. # you may not use this file except in compliance with the License.
  7. # You may obtain a copy of the License at
  8. #
  9. # http://www.apache.org/licenses/LICENSE-2.0
  10. #
  11. # Unless required by applicable law or agreed to in writing, software
  12. # distributed under the License is distributed on an "AS IS" BASIS,
  13. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. # See the License for the specific language governing permissions and
  15. # limitations under the License.
  16. import logging
  17. import platform
  18. import re
  19. import synapse
  20. from synapse.api.errors import Codes, NotFoundError, SynapseError
  21. from synapse.http.server import JsonResource
  22. from synapse.http.servlet import RestServlet, parse_json_object_from_request
  23. from synapse.rest.admin._base import (
  24. assert_requester_is_admin,
  25. historical_admin_path_patterns,
  26. )
  27. from synapse.rest.admin.groups import DeleteGroupAdminRestServlet
  28. from synapse.rest.admin.media import ListMediaInRoom, register_servlets_for_media_repo
  29. from synapse.rest.admin.purge_room_servlet import PurgeRoomServlet
  30. from synapse.rest.admin.rooms import ListRoomRestServlet, ShutdownRoomRestServlet
  31. from synapse.rest.admin.server_notice_servlet import SendServerNoticeServlet
  32. from synapse.rest.admin.users import (
  33. AccountValidityRenewServlet,
  34. DeactivateAccountRestServlet,
  35. ResetPasswordRestServlet,
  36. SearchUsersRestServlet,
  37. UserAdminServlet,
  38. UserRegisterServlet,
  39. UserRestServletV2,
  40. UsersRestServlet,
  41. UsersRestServletV2,
  42. WhoisRestServlet,
  43. )
  44. from synapse.util.versionstring import get_version_string
  45. logger = logging.getLogger(__name__)
  46. class VersionServlet(RestServlet):
  47. PATTERNS = (re.compile("^/_synapse/admin/v1/server_version$"),)
  48. def __init__(self, hs):
  49. self.res = {
  50. "server_version": get_version_string(synapse),
  51. "python_version": platform.python_version(),
  52. }
  53. def on_GET(self, request):
  54. return 200, self.res
  55. class PurgeHistoryRestServlet(RestServlet):
  56. PATTERNS = historical_admin_path_patterns(
  57. "/purge_history/(?P<room_id>[^/]*)(/(?P<event_id>[^/]+))?"
  58. )
  59. def __init__(self, hs):
  60. """
  61. Args:
  62. hs (synapse.server.HomeServer)
  63. """
  64. self.pagination_handler = hs.get_pagination_handler()
  65. self.store = hs.get_datastore()
  66. self.auth = hs.get_auth()
  67. async def on_POST(self, request, room_id, event_id):
  68. await assert_requester_is_admin(self.auth, request)
  69. body = parse_json_object_from_request(request, allow_empty_body=True)
  70. delete_local_events = bool(body.get("delete_local_events", False))
  71. # establish the topological ordering we should keep events from. The
  72. # user can provide an event_id in the URL or the request body, or can
  73. # provide a timestamp in the request body.
  74. if event_id is None:
  75. event_id = body.get("purge_up_to_event_id")
  76. if event_id is not None:
  77. event = await self.store.get_event(event_id)
  78. if event.room_id != room_id:
  79. raise SynapseError(400, "Event is for wrong room.")
  80. token = await self.store.get_topological_token_for_event(event_id)
  81. logger.info("[purge] purging up to token %s (event_id %s)", token, event_id)
  82. elif "purge_up_to_ts" in body:
  83. ts = body["purge_up_to_ts"]
  84. if not isinstance(ts, int):
  85. raise SynapseError(
  86. 400, "purge_up_to_ts must be an int", errcode=Codes.BAD_JSON
  87. )
  88. stream_ordering = await self.store.find_first_stream_ordering_after_ts(ts)
  89. r = await self.store.get_room_event_before_stream_ordering(
  90. room_id, stream_ordering
  91. )
  92. if not r:
  93. logger.warning(
  94. "[purge] purging events not possible: No event found "
  95. "(received_ts %i => stream_ordering %i)",
  96. ts,
  97. stream_ordering,
  98. )
  99. raise SynapseError(
  100. 404, "there is no event to be purged", errcode=Codes.NOT_FOUND
  101. )
  102. (stream, topo, _event_id) = r
  103. token = "t%d-%d" % (topo, stream)
  104. logger.info(
  105. "[purge] purging up to token %s (received_ts %i => "
  106. "stream_ordering %i)",
  107. token,
  108. ts,
  109. stream_ordering,
  110. )
  111. else:
  112. raise SynapseError(
  113. 400,
  114. "must specify purge_up_to_event_id or purge_up_to_ts",
  115. errcode=Codes.BAD_JSON,
  116. )
  117. purge_id = self.pagination_handler.start_purge_history(
  118. room_id, token, delete_local_events=delete_local_events
  119. )
  120. return 200, {"purge_id": purge_id}
  121. class PurgeHistoryStatusRestServlet(RestServlet):
  122. PATTERNS = historical_admin_path_patterns(
  123. "/purge_history_status/(?P<purge_id>[^/]+)"
  124. )
  125. def __init__(self, hs):
  126. """
  127. Args:
  128. hs (synapse.server.HomeServer)
  129. """
  130. self.pagination_handler = hs.get_pagination_handler()
  131. self.auth = hs.get_auth()
  132. async def on_GET(self, request, purge_id):
  133. await assert_requester_is_admin(self.auth, request)
  134. purge_status = self.pagination_handler.get_purge_status(purge_id)
  135. if purge_status is None:
  136. raise NotFoundError("purge id '%s' not found" % purge_id)
  137. return 200, purge_status.asdict()
  138. ########################################################################################
  139. #
  140. # please don't add more servlets here: this file is already long and unwieldy. Put
  141. # them in separate files within the 'admin' package.
  142. #
  143. ########################################################################################
  144. class AdminRestResource(JsonResource):
  145. """The REST resource which gets mounted at /_synapse/admin"""
  146. def __init__(self, hs):
  147. JsonResource.__init__(self, hs, canonical_json=False)
  148. register_servlets(hs, self)
  149. def register_servlets(hs, http_server):
  150. """
  151. Register all the admin servlets.
  152. """
  153. register_servlets_for_client_rest_resource(hs, http_server)
  154. ListRoomRestServlet(hs).register(http_server)
  155. PurgeRoomServlet(hs).register(http_server)
  156. SendServerNoticeServlet(hs).register(http_server)
  157. VersionServlet(hs).register(http_server)
  158. UserAdminServlet(hs).register(http_server)
  159. UserRestServletV2(hs).register(http_server)
  160. UsersRestServletV2(hs).register(http_server)
  161. def register_servlets_for_client_rest_resource(hs, http_server):
  162. """Register only the servlets which need to be exposed on /_matrix/client/xxx"""
  163. WhoisRestServlet(hs).register(http_server)
  164. PurgeHistoryStatusRestServlet(hs).register(http_server)
  165. DeactivateAccountRestServlet(hs).register(http_server)
  166. PurgeHistoryRestServlet(hs).register(http_server)
  167. UsersRestServlet(hs).register(http_server)
  168. ResetPasswordRestServlet(hs).register(http_server)
  169. SearchUsersRestServlet(hs).register(http_server)
  170. ShutdownRoomRestServlet(hs).register(http_server)
  171. UserRegisterServlet(hs).register(http_server)
  172. DeleteGroupAdminRestServlet(hs).register(http_server)
  173. AccountValidityRenewServlet(hs).register(http_server)
  174. # Load the media repo ones if we're using them. Otherwise load the servlets which
  175. # don't need a media repo (typically readonly admin APIs).
  176. if hs.config.can_load_media_repo:
  177. register_servlets_for_media_repo(hs, http_server)
  178. else:
  179. ListMediaInRoom(hs).register(http_server)
  180. # don't add more things here: new servlets should only be exposed on
  181. # /_synapse/admin so should not go here. Instead register them in AdminRestResource.