message.py 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726
  1. # -*- coding: utf-8 -*-
  2. # Copyright 2014 - 2016 OpenMarket Ltd
  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. from twisted.internet import defer
  16. from synapse.api.constants import EventTypes, Membership
  17. from synapse.api.errors import AuthError, Codes, SynapseError
  18. from synapse.streams.config import PaginationConfig
  19. from synapse.events.utils import serialize_event
  20. from synapse.events.validator import EventValidator
  21. from synapse.util import unwrapFirstError
  22. from synapse.util.async import concurrently_execute
  23. from synapse.util.caches.snapshot_cache import SnapshotCache
  24. from synapse.types import UserID, RoomStreamToken, StreamToken
  25. from ._base import BaseHandler
  26. from canonicaljson import encode_canonical_json
  27. import logging
  28. logger = logging.getLogger(__name__)
  29. class MessageHandler(BaseHandler):
  30. def __init__(self, hs):
  31. super(MessageHandler, self).__init__(hs)
  32. self.hs = hs
  33. self.state = hs.get_state_handler()
  34. self.clock = hs.get_clock()
  35. self.validator = EventValidator()
  36. self.snapshot_cache = SnapshotCache()
  37. @defer.inlineCallbacks
  38. def get_messages(self, requester, room_id=None, pagin_config=None,
  39. as_client_event=True):
  40. """Get messages in a room.
  41. Args:
  42. requester (Requester): The user requesting messages.
  43. room_id (str): The room they want messages from.
  44. pagin_config (synapse.api.streams.PaginationConfig): The pagination
  45. config rules to apply, if any.
  46. as_client_event (bool): True to get events in client-server format.
  47. Returns:
  48. dict: Pagination API results
  49. """
  50. user_id = requester.user.to_string()
  51. data_source = self.hs.get_event_sources().sources["room"]
  52. if pagin_config.from_token:
  53. room_token = pagin_config.from_token.room_key
  54. else:
  55. pagin_config.from_token = (
  56. yield self.hs.get_event_sources().get_current_token(
  57. direction='b'
  58. )
  59. )
  60. room_token = pagin_config.from_token.room_key
  61. room_token = RoomStreamToken.parse(room_token)
  62. pagin_config.from_token = pagin_config.from_token.copy_and_replace(
  63. "room_key", str(room_token)
  64. )
  65. source_config = pagin_config.get_source_config("room")
  66. membership, member_event_id = yield self._check_in_room_or_world_readable(
  67. room_id, user_id
  68. )
  69. if source_config.direction == 'b':
  70. # if we're going backwards, we might need to backfill. This
  71. # requires that we have a topo token.
  72. if room_token.topological:
  73. max_topo = room_token.topological
  74. else:
  75. max_topo = yield self.store.get_max_topological_token_for_stream_and_room(
  76. room_id, room_token.stream
  77. )
  78. if membership == Membership.LEAVE:
  79. # If they have left the room then clamp the token to be before
  80. # they left the room, to save the effort of loading from the
  81. # database.
  82. leave_token = yield self.store.get_topological_token_for_event(
  83. member_event_id
  84. )
  85. leave_token = RoomStreamToken.parse(leave_token)
  86. if leave_token.topological < max_topo:
  87. source_config.from_key = str(leave_token)
  88. yield self.hs.get_handlers().federation_handler.maybe_backfill(
  89. room_id, max_topo
  90. )
  91. events, next_key = yield data_source.get_pagination_rows(
  92. requester.user, source_config, room_id
  93. )
  94. next_token = pagin_config.from_token.copy_and_replace(
  95. "room_key", next_key
  96. )
  97. if not events:
  98. defer.returnValue({
  99. "chunk": [],
  100. "start": pagin_config.from_token.to_string(),
  101. "end": next_token.to_string(),
  102. })
  103. events = yield self._filter_events_for_client(
  104. user_id,
  105. events,
  106. is_peeking=(member_event_id is None),
  107. )
  108. time_now = self.clock.time_msec()
  109. chunk = {
  110. "chunk": [
  111. serialize_event(e, time_now, as_client_event)
  112. for e in events
  113. ],
  114. "start": pagin_config.from_token.to_string(),
  115. "end": next_token.to_string(),
  116. }
  117. defer.returnValue(chunk)
  118. @defer.inlineCallbacks
  119. def create_event(self, event_dict, token_id=None, txn_id=None, prev_event_ids=None):
  120. """
  121. Given a dict from a client, create a new event.
  122. Creates an FrozenEvent object, filling out auth_events, prev_events,
  123. etc.
  124. Adds display names to Join membership events.
  125. Args:
  126. event_dict (dict): An entire event
  127. token_id (str)
  128. txn_id (str)
  129. prev_event_ids (list): The prev event ids to use when creating the event
  130. Returns:
  131. Tuple of created event (FrozenEvent), Context
  132. """
  133. builder = self.event_builder_factory.new(event_dict)
  134. self.validator.validate_new(builder)
  135. if builder.type == EventTypes.Member:
  136. membership = builder.content.get("membership", None)
  137. target = UserID.from_string(builder.state_key)
  138. if membership in {Membership.JOIN, Membership.INVITE}:
  139. # If event doesn't include a display name, add one.
  140. profile = self.hs.get_handlers().profile_handler
  141. content = builder.content
  142. try:
  143. content["displayname"] = yield profile.get_displayname(target)
  144. content["avatar_url"] = yield profile.get_avatar_url(target)
  145. except Exception as e:
  146. logger.info(
  147. "Failed to get profile information for %r: %s",
  148. target, e
  149. )
  150. if token_id is not None:
  151. builder.internal_metadata.token_id = token_id
  152. if txn_id is not None:
  153. builder.internal_metadata.txn_id = txn_id
  154. event, context = yield self._create_new_client_event(
  155. builder=builder,
  156. prev_event_ids=prev_event_ids,
  157. )
  158. defer.returnValue((event, context))
  159. @defer.inlineCallbacks
  160. def send_nonmember_event(self, requester, event, context, ratelimit=True):
  161. """
  162. Persists and notifies local clients and federation of an event.
  163. Args:
  164. event (FrozenEvent) the event to send.
  165. context (Context) the context of the event.
  166. ratelimit (bool): Whether to rate limit this send.
  167. is_guest (bool): Whether the sender is a guest.
  168. """
  169. if event.type == EventTypes.Member:
  170. raise SynapseError(
  171. 500,
  172. "Tried to send member event through non-member codepath"
  173. )
  174. user = UserID.from_string(event.sender)
  175. assert self.hs.is_mine(user), "User must be our own: %s" % (user,)
  176. if event.is_state():
  177. prev_state = self.deduplicate_state_event(event, context)
  178. if prev_state is not None:
  179. defer.returnValue(prev_state)
  180. yield self.handle_new_client_event(
  181. requester=requester,
  182. event=event,
  183. context=context,
  184. ratelimit=ratelimit,
  185. )
  186. if event.type == EventTypes.Message:
  187. presence = self.hs.get_handlers().presence_handler
  188. yield presence.bump_presence_active_time(user)
  189. def deduplicate_state_event(self, event, context):
  190. """
  191. Checks whether event is in the latest resolved state in context.
  192. If so, returns the version of the event in context.
  193. Otherwise, returns None.
  194. """
  195. prev_event = context.current_state.get((event.type, event.state_key))
  196. if prev_event and event.user_id == prev_event.user_id:
  197. prev_content = encode_canonical_json(prev_event.content)
  198. next_content = encode_canonical_json(event.content)
  199. if prev_content == next_content:
  200. return prev_event
  201. return None
  202. @defer.inlineCallbacks
  203. def create_and_send_nonmember_event(
  204. self,
  205. requester,
  206. event_dict,
  207. ratelimit=True,
  208. txn_id=None
  209. ):
  210. """
  211. Creates an event, then sends it.
  212. See self.create_event and self.send_nonmember_event.
  213. """
  214. event, context = yield self.create_event(
  215. event_dict,
  216. token_id=requester.access_token_id,
  217. txn_id=txn_id
  218. )
  219. yield self.send_nonmember_event(
  220. requester,
  221. event,
  222. context,
  223. ratelimit=ratelimit,
  224. )
  225. defer.returnValue(event)
  226. @defer.inlineCallbacks
  227. def get_room_data(self, user_id=None, room_id=None,
  228. event_type=None, state_key="", is_guest=False):
  229. """ Get data from a room.
  230. Args:
  231. event : The room path event
  232. Returns:
  233. The path data content.
  234. Raises:
  235. SynapseError if something went wrong.
  236. """
  237. membership, membership_event_id = yield self._check_in_room_or_world_readable(
  238. room_id, user_id
  239. )
  240. if membership == Membership.JOIN:
  241. data = yield self.state_handler.get_current_state(
  242. room_id, event_type, state_key
  243. )
  244. elif membership == Membership.LEAVE:
  245. key = (event_type, state_key)
  246. room_state = yield self.store.get_state_for_events(
  247. [membership_event_id], [key]
  248. )
  249. data = room_state[membership_event_id].get(key)
  250. defer.returnValue(data)
  251. @defer.inlineCallbacks
  252. def _check_in_room_or_world_readable(self, room_id, user_id):
  253. try:
  254. # check_user_was_in_room will return the most recent membership
  255. # event for the user if:
  256. # * The user is a non-guest user, and was ever in the room
  257. # * The user is a guest user, and has joined the room
  258. # else it will throw.
  259. member_event = yield self.auth.check_user_was_in_room(room_id, user_id)
  260. defer.returnValue((member_event.membership, member_event.event_id))
  261. return
  262. except AuthError:
  263. visibility = yield self.state_handler.get_current_state(
  264. room_id, EventTypes.RoomHistoryVisibility, ""
  265. )
  266. if (
  267. visibility and
  268. visibility.content["history_visibility"] == "world_readable"
  269. ):
  270. defer.returnValue((Membership.JOIN, None))
  271. return
  272. raise AuthError(
  273. 403, "Guest access not allowed", errcode=Codes.GUEST_ACCESS_FORBIDDEN
  274. )
  275. @defer.inlineCallbacks
  276. def get_state_events(self, user_id, room_id, is_guest=False):
  277. """Retrieve all state events for a given room. If the user is
  278. joined to the room then return the current state. If the user has
  279. left the room return the state events from when they left.
  280. Args:
  281. user_id(str): The user requesting state events.
  282. room_id(str): The room ID to get all state events from.
  283. Returns:
  284. A list of dicts representing state events. [{}, {}, {}]
  285. """
  286. membership, membership_event_id = yield self._check_in_room_or_world_readable(
  287. room_id, user_id
  288. )
  289. if membership == Membership.JOIN:
  290. room_state = yield self.state_handler.get_current_state(room_id)
  291. elif membership == Membership.LEAVE:
  292. room_state = yield self.store.get_state_for_events(
  293. [membership_event_id], None
  294. )
  295. room_state = room_state[membership_event_id]
  296. now = self.clock.time_msec()
  297. defer.returnValue(
  298. [serialize_event(c, now) for c in room_state.values()]
  299. )
  300. def snapshot_all_rooms(self, user_id=None, pagin_config=None,
  301. as_client_event=True, include_archived=False):
  302. """Retrieve a snapshot of all rooms the user is invited or has joined.
  303. This snapshot may include messages for all rooms where the user is
  304. joined, depending on the pagination config.
  305. Args:
  306. user_id (str): The ID of the user making the request.
  307. pagin_config (synapse.api.streams.PaginationConfig): The pagination
  308. config used to determine how many messages *PER ROOM* to return.
  309. as_client_event (bool): True to get events in client-server format.
  310. include_archived (bool): True to get rooms that the user has left
  311. Returns:
  312. A list of dicts with "room_id" and "membership" keys for all rooms
  313. the user is currently invited or joined in on. Rooms where the user
  314. is joined on, may return a "messages" key with messages, depending
  315. on the specified PaginationConfig.
  316. """
  317. key = (
  318. user_id,
  319. pagin_config.from_token,
  320. pagin_config.to_token,
  321. pagin_config.direction,
  322. pagin_config.limit,
  323. as_client_event,
  324. include_archived,
  325. )
  326. now_ms = self.clock.time_msec()
  327. result = self.snapshot_cache.get(now_ms, key)
  328. if result is not None:
  329. return result
  330. return self.snapshot_cache.set(now_ms, key, self._snapshot_all_rooms(
  331. user_id, pagin_config, as_client_event, include_archived
  332. ))
  333. @defer.inlineCallbacks
  334. def _snapshot_all_rooms(self, user_id=None, pagin_config=None,
  335. as_client_event=True, include_archived=False):
  336. memberships = [Membership.INVITE, Membership.JOIN]
  337. if include_archived:
  338. memberships.append(Membership.LEAVE)
  339. room_list = yield self.store.get_rooms_for_user_where_membership_is(
  340. user_id=user_id, membership_list=memberships
  341. )
  342. user = UserID.from_string(user_id)
  343. rooms_ret = []
  344. now_token = yield self.hs.get_event_sources().get_current_token()
  345. presence_stream = self.hs.get_event_sources().sources["presence"]
  346. pagination_config = PaginationConfig(from_token=now_token)
  347. presence, _ = yield presence_stream.get_pagination_rows(
  348. user, pagination_config.get_source_config("presence"), None
  349. )
  350. receipt_stream = self.hs.get_event_sources().sources["receipt"]
  351. receipt, _ = yield receipt_stream.get_pagination_rows(
  352. user, pagination_config.get_source_config("receipt"), None
  353. )
  354. tags_by_room = yield self.store.get_tags_for_user(user_id)
  355. account_data, account_data_by_room = (
  356. yield self.store.get_account_data_for_user(user_id)
  357. )
  358. public_room_ids = yield self.store.get_public_room_ids()
  359. limit = pagin_config.limit
  360. if limit is None:
  361. limit = 10
  362. @defer.inlineCallbacks
  363. def handle_room(event):
  364. d = {
  365. "room_id": event.room_id,
  366. "membership": event.membership,
  367. "visibility": (
  368. "public" if event.room_id in public_room_ids
  369. else "private"
  370. ),
  371. }
  372. if event.membership == Membership.INVITE:
  373. time_now = self.clock.time_msec()
  374. d["inviter"] = event.sender
  375. invite_event = yield self.store.get_event(event.event_id)
  376. d["invite"] = serialize_event(invite_event, time_now, as_client_event)
  377. rooms_ret.append(d)
  378. if event.membership not in (Membership.JOIN, Membership.LEAVE):
  379. return
  380. try:
  381. if event.membership == Membership.JOIN:
  382. room_end_token = now_token.room_key
  383. deferred_room_state = self.state_handler.get_current_state(
  384. event.room_id
  385. )
  386. elif event.membership == Membership.LEAVE:
  387. room_end_token = "s%d" % (event.stream_ordering,)
  388. deferred_room_state = self.store.get_state_for_events(
  389. [event.event_id], None
  390. )
  391. deferred_room_state.addCallback(
  392. lambda states: states[event.event_id]
  393. )
  394. (messages, token), current_state = yield defer.gatherResults(
  395. [
  396. self.store.get_recent_events_for_room(
  397. event.room_id,
  398. limit=limit,
  399. end_token=room_end_token,
  400. ),
  401. deferred_room_state,
  402. ]
  403. ).addErrback(unwrapFirstError)
  404. messages = yield self._filter_events_for_client(
  405. user_id, messages
  406. )
  407. start_token = now_token.copy_and_replace("room_key", token[0])
  408. end_token = now_token.copy_and_replace("room_key", token[1])
  409. time_now = self.clock.time_msec()
  410. d["messages"] = {
  411. "chunk": [
  412. serialize_event(m, time_now, as_client_event)
  413. for m in messages
  414. ],
  415. "start": start_token.to_string(),
  416. "end": end_token.to_string(),
  417. }
  418. d["state"] = [
  419. serialize_event(c, time_now, as_client_event)
  420. for c in current_state.values()
  421. ]
  422. account_data_events = []
  423. tags = tags_by_room.get(event.room_id)
  424. if tags:
  425. account_data_events.append({
  426. "type": "m.tag",
  427. "content": {"tags": tags},
  428. })
  429. account_data = account_data_by_room.get(event.room_id, {})
  430. for account_data_type, content in account_data.items():
  431. account_data_events.append({
  432. "type": account_data_type,
  433. "content": content,
  434. })
  435. d["account_data"] = account_data_events
  436. except:
  437. logger.exception("Failed to get snapshot")
  438. yield concurrently_execute(handle_room, room_list, 10)
  439. account_data_events = []
  440. for account_data_type, content in account_data.items():
  441. account_data_events.append({
  442. "type": account_data_type,
  443. "content": content,
  444. })
  445. ret = {
  446. "rooms": rooms_ret,
  447. "presence": presence,
  448. "account_data": account_data_events,
  449. "receipts": receipt,
  450. "end": now_token.to_string(),
  451. }
  452. defer.returnValue(ret)
  453. @defer.inlineCallbacks
  454. def room_initial_sync(self, requester, room_id, pagin_config=None):
  455. """Capture the a snapshot of a room. If user is currently a member of
  456. the room this will be what is currently in the room. If the user left
  457. the room this will be what was in the room when they left.
  458. Args:
  459. requester(Requester): The user to get a snapshot for.
  460. room_id(str): The room to get a snapshot of.
  461. pagin_config(synapse.streams.config.PaginationConfig):
  462. The pagination config used to determine how many messages to
  463. return.
  464. Raises:
  465. AuthError if the user wasn't in the room.
  466. Returns:
  467. A JSON serialisable dict with the snapshot of the room.
  468. """
  469. user_id = requester.user.to_string()
  470. membership, member_event_id = yield self._check_in_room_or_world_readable(
  471. room_id, user_id,
  472. )
  473. is_peeking = member_event_id is None
  474. if membership == Membership.JOIN:
  475. result = yield self._room_initial_sync_joined(
  476. user_id, room_id, pagin_config, membership, is_peeking
  477. )
  478. elif membership == Membership.LEAVE:
  479. result = yield self._room_initial_sync_parted(
  480. user_id, room_id, pagin_config, membership, member_event_id, is_peeking
  481. )
  482. account_data_events = []
  483. tags = yield self.store.get_tags_for_room(user_id, room_id)
  484. if tags:
  485. account_data_events.append({
  486. "type": "m.tag",
  487. "content": {"tags": tags},
  488. })
  489. account_data = yield self.store.get_account_data_for_room(user_id, room_id)
  490. for account_data_type, content in account_data.items():
  491. account_data_events.append({
  492. "type": account_data_type,
  493. "content": content,
  494. })
  495. result["account_data"] = account_data_events
  496. defer.returnValue(result)
  497. @defer.inlineCallbacks
  498. def _room_initial_sync_parted(self, user_id, room_id, pagin_config,
  499. membership, member_event_id, is_peeking):
  500. room_state = yield self.store.get_state_for_events(
  501. [member_event_id], None
  502. )
  503. room_state = room_state[member_event_id]
  504. limit = pagin_config.limit if pagin_config else None
  505. if limit is None:
  506. limit = 10
  507. stream_token = yield self.store.get_stream_token_for_event(
  508. member_event_id
  509. )
  510. messages, token = yield self.store.get_recent_events_for_room(
  511. room_id,
  512. limit=limit,
  513. end_token=stream_token
  514. )
  515. messages = yield self._filter_events_for_client(
  516. user_id, messages, is_peeking=is_peeking
  517. )
  518. start_token = StreamToken.START.copy_and_replace("room_key", token[0])
  519. end_token = StreamToken.START.copy_and_replace("room_key", token[1])
  520. time_now = self.clock.time_msec()
  521. defer.returnValue({
  522. "membership": membership,
  523. "room_id": room_id,
  524. "messages": {
  525. "chunk": [serialize_event(m, time_now) for m in messages],
  526. "start": start_token.to_string(),
  527. "end": end_token.to_string(),
  528. },
  529. "state": [serialize_event(s, time_now) for s in room_state.values()],
  530. "presence": [],
  531. "receipts": [],
  532. })
  533. @defer.inlineCallbacks
  534. def _room_initial_sync_joined(self, user_id, room_id, pagin_config,
  535. membership, is_peeking):
  536. current_state = yield self.state.get_current_state(
  537. room_id=room_id,
  538. )
  539. # TODO: These concurrently
  540. time_now = self.clock.time_msec()
  541. state = [
  542. serialize_event(x, time_now)
  543. for x in current_state.values()
  544. ]
  545. now_token = yield self.hs.get_event_sources().get_current_token()
  546. limit = pagin_config.limit if pagin_config else None
  547. if limit is None:
  548. limit = 10
  549. room_members = [
  550. m for m in current_state.values()
  551. if m.type == EventTypes.Member
  552. and m.content["membership"] == Membership.JOIN
  553. ]
  554. presence_handler = self.hs.get_handlers().presence_handler
  555. @defer.inlineCallbacks
  556. def get_presence():
  557. states = yield presence_handler.get_states(
  558. [m.user_id for m in room_members],
  559. as_event=True,
  560. )
  561. defer.returnValue(states)
  562. @defer.inlineCallbacks
  563. def get_receipts():
  564. receipts_handler = self.hs.get_handlers().receipts_handler
  565. receipts = yield receipts_handler.get_receipts_for_room(
  566. room_id,
  567. now_token.receipt_key
  568. )
  569. defer.returnValue(receipts)
  570. presence, receipts, (messages, token) = yield defer.gatherResults(
  571. [
  572. get_presence(),
  573. get_receipts(),
  574. self.store.get_recent_events_for_room(
  575. room_id,
  576. limit=limit,
  577. end_token=now_token.room_key,
  578. )
  579. ],
  580. consumeErrors=True,
  581. ).addErrback(unwrapFirstError)
  582. messages = yield self._filter_events_for_client(
  583. user_id, messages, is_peeking=is_peeking,
  584. )
  585. start_token = now_token.copy_and_replace("room_key", token[0])
  586. end_token = now_token.copy_and_replace("room_key", token[1])
  587. time_now = self.clock.time_msec()
  588. ret = {
  589. "room_id": room_id,
  590. "messages": {
  591. "chunk": [serialize_event(m, time_now) for m in messages],
  592. "start": start_token.to_string(),
  593. "end": end_token.to_string(),
  594. },
  595. "state": state,
  596. "presence": presence,
  597. "receipts": receipts,
  598. }
  599. if not is_peeking:
  600. ret["membership"] = membership
  601. defer.returnValue(ret)