test_typing.py 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484
  1. # Copyright 2014-2016 OpenMarket Ltd
  2. #
  3. # Licensed under the Apache License, Version 2.0 (the "License");
  4. # you may not use this file except in compliance with the License.
  5. # You may obtain a copy of the License at
  6. #
  7. # http://www.apache.org/licenses/LICENSE-2.0
  8. #
  9. # Unless required by applicable law or agreed to in writing, software
  10. # distributed under the License is distributed on an "AS IS" BASIS,
  11. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. # See the License for the specific language governing permissions and
  13. # limitations under the License.
  14. import json
  15. from typing import Dict, List, Set
  16. from unittest.mock import ANY, AsyncMock, Mock, call
  17. from netaddr import IPSet
  18. from twisted.test.proto_helpers import MemoryReactor
  19. from twisted.web.resource import Resource
  20. from synapse.api.constants import EduTypes
  21. from synapse.api.errors import AuthError
  22. from synapse.federation.transport.server import TransportLayerServer
  23. from synapse.handlers.typing import TypingWriterHandler
  24. from synapse.http.federation.matrix_federation_agent import MatrixFederationAgent
  25. from synapse.server import HomeServer
  26. from synapse.types import JsonDict, Requester, UserID, create_requester
  27. from synapse.util import Clock
  28. from tests import unittest
  29. from tests.server import ThreadedMemoryReactorClock
  30. from tests.unittest import override_config
  31. # Some local users to test with
  32. U_APPLE = UserID.from_string("@apple:test")
  33. U_BANANA = UserID.from_string("@banana:test")
  34. # Remote user
  35. U_ONION = UserID.from_string("@onion:farm")
  36. # Test room id
  37. ROOM_ID = "a-room"
  38. # Room we're not in
  39. OTHER_ROOM_ID = "another-room"
  40. def _expect_edu_transaction(
  41. edu_type: str, content: JsonDict, origin: str = "test"
  42. ) -> JsonDict:
  43. return {
  44. "origin": origin,
  45. "origin_server_ts": 1000000,
  46. "pdus": [],
  47. "edus": [{"edu_type": edu_type, "content": content}],
  48. }
  49. def _make_edu_transaction_json(edu_type: str, content: JsonDict) -> bytes:
  50. return json.dumps(_expect_edu_transaction(edu_type, content)).encode("utf8")
  51. class TypingNotificationsTestCase(unittest.HomeserverTestCase):
  52. def make_homeserver(
  53. self,
  54. reactor: ThreadedMemoryReactorClock,
  55. clock: Clock,
  56. ) -> HomeServer:
  57. # we mock out the keyring so as to skip the authentication check on the
  58. # federation API call.
  59. mock_keyring = Mock(spec=["verify_json_for_server"])
  60. mock_keyring.verify_json_for_server = AsyncMock(return_value=True)
  61. # we mock out the federation client too
  62. self.mock_federation_client = AsyncMock(spec=["put_json"])
  63. self.mock_federation_client.put_json.return_value = (200, "OK")
  64. self.mock_federation_client.agent = MatrixFederationAgent(
  65. reactor,
  66. tls_client_options_factory=None,
  67. user_agent=b"SynapseInTrialTest/0.0.0",
  68. ip_allowlist=None,
  69. ip_blocklist=IPSet(),
  70. )
  71. # the tests assume that we are starting at unix time 1000
  72. reactor.pump((1000,))
  73. self.mock_hs_notifier = Mock()
  74. hs = self.setup_test_homeserver(
  75. notifier=self.mock_hs_notifier,
  76. federation_http_client=self.mock_federation_client,
  77. keyring=mock_keyring,
  78. replication_streams={},
  79. )
  80. return hs
  81. def create_resource_dict(self) -> Dict[str, Resource]:
  82. d = super().create_resource_dict()
  83. d["/_matrix/federation"] = TransportLayerServer(self.hs)
  84. return d
  85. def prepare(self, reactor: MemoryReactor, clock: Clock, hs: HomeServer) -> None:
  86. self.on_new_event = self.mock_hs_notifier.on_new_event
  87. # hs.get_typing_handler will return a TypingWriterHandler when calling it
  88. # from the main process, and a FollowerTypingHandler on workers.
  89. # We rely on methods only available on the former, so assert we have the
  90. # correct type here. We have to assign self.handler after the assert,
  91. # otherwise mypy will treat it as a FollowerTypingHandler
  92. handler = hs.get_typing_handler()
  93. assert isinstance(handler, TypingWriterHandler)
  94. self.handler = handler
  95. self.event_source = hs.get_event_sources().sources.typing
  96. self.datastore = hs.get_datastores().main
  97. self.datastore.get_device_updates_by_remote = AsyncMock( # type: ignore[method-assign]
  98. return_value=(0, [])
  99. )
  100. self.datastore.get_destination_last_successful_stream_ordering = AsyncMock( # type: ignore[method-assign]
  101. return_value=None
  102. )
  103. self.datastore.get_received_txn_response = AsyncMock( # type: ignore[method-assign]
  104. return_value=None
  105. )
  106. self.room_members: List[UserID] = []
  107. async def check_user_in_room(room_id: str, requester: Requester) -> None:
  108. if requester.user.to_string() not in [
  109. u.to_string() for u in self.room_members
  110. ]:
  111. raise AuthError(401, "User is not in the room")
  112. return None
  113. hs.get_auth().check_user_in_room = Mock( # type: ignore[method-assign]
  114. side_effect=check_user_in_room
  115. )
  116. async def check_host_in_room(room_id: str, server_name: str) -> bool:
  117. return room_id == ROOM_ID
  118. hs.get_event_auth_handler().is_host_in_room = Mock( # type: ignore[method-assign]
  119. side_effect=check_host_in_room
  120. )
  121. async def get_current_hosts_in_room(room_id: str) -> Set[str]:
  122. return {member.domain for member in self.room_members}
  123. hs.get_storage_controllers().state.get_current_hosts_in_room = Mock( # type: ignore[method-assign]
  124. side_effect=get_current_hosts_in_room
  125. )
  126. hs.get_storage_controllers().state.get_current_hosts_in_room_or_partial_state_approximation = Mock( # type: ignore[method-assign]
  127. side_effect=get_current_hosts_in_room
  128. )
  129. async def get_users_in_room(room_id: str) -> Set[str]:
  130. return {str(u) for u in self.room_members}
  131. self.datastore.get_users_in_room = Mock(side_effect=get_users_in_room)
  132. self.datastore.get_user_directory_stream_pos = AsyncMock( # type: ignore[method-assign]
  133. # we deliberately return a non-None stream pos to avoid
  134. # doing an initial_sync
  135. return_value=1
  136. )
  137. self.datastore.get_partial_current_state_deltas = Mock(return_value=(0, None)) # type: ignore[method-assign]
  138. self.datastore.get_to_device_stream_token = Mock( # type: ignore[method-assign]
  139. return_value=0
  140. )
  141. self.datastore.get_new_device_msgs_for_remote = AsyncMock( # type: ignore[method-assign]
  142. return_value=([], 0)
  143. )
  144. self.datastore.delete_device_msgs_for_remote = AsyncMock( # type: ignore[method-assign]
  145. return_value=None
  146. )
  147. self.datastore.set_received_txn_response = AsyncMock( # type: ignore[method-assign]
  148. return_value=None
  149. )
  150. def test_started_typing_local(self) -> None:
  151. self.room_members = [U_APPLE, U_BANANA]
  152. self.assertEqual(self.event_source.get_current_key(), 0)
  153. self.get_success(
  154. self.handler.started_typing(
  155. target_user=U_APPLE,
  156. requester=create_requester(U_APPLE),
  157. room_id=ROOM_ID,
  158. timeout=20000,
  159. )
  160. )
  161. self.on_new_event.assert_has_calls([call("typing_key", 1, rooms=[ROOM_ID])])
  162. self.assertEqual(self.event_source.get_current_key(), 1)
  163. events = self.get_success(
  164. self.event_source.get_new_events(
  165. user=U_APPLE, from_key=0, limit=0, room_ids=[ROOM_ID], is_guest=False
  166. )
  167. )
  168. self.assertEqual(
  169. events[0],
  170. [
  171. {
  172. "type": EduTypes.TYPING,
  173. "room_id": ROOM_ID,
  174. "content": {"user_ids": [U_APPLE.to_string()]},
  175. }
  176. ],
  177. )
  178. # Enable federation sending on the main process.
  179. @override_config({"federation_sender_instances": None})
  180. def test_started_typing_remote_send(self) -> None:
  181. self.room_members = [U_APPLE, U_ONION]
  182. self.get_success(
  183. self.handler.started_typing(
  184. target_user=U_APPLE,
  185. requester=create_requester(U_APPLE),
  186. room_id=ROOM_ID,
  187. timeout=20000,
  188. )
  189. )
  190. self.mock_federation_client.put_json.assert_called_once_with(
  191. "farm",
  192. path="/_matrix/federation/v1/send/1000000",
  193. data=_expect_edu_transaction(
  194. EduTypes.TYPING,
  195. content={
  196. "room_id": ROOM_ID,
  197. "user_id": U_APPLE.to_string(),
  198. "typing": True,
  199. },
  200. ),
  201. json_data_callback=ANY,
  202. long_retries=True,
  203. try_trailing_slash_on_400=True,
  204. backoff_on_all_error_codes=True,
  205. )
  206. def test_started_typing_remote_recv(self) -> None:
  207. self.room_members = [U_APPLE, U_ONION]
  208. self.assertEqual(self.event_source.get_current_key(), 0)
  209. channel = self.make_request(
  210. "PUT",
  211. "/_matrix/federation/v1/send/1000000",
  212. _make_edu_transaction_json(
  213. EduTypes.TYPING,
  214. content={
  215. "room_id": ROOM_ID,
  216. "user_id": U_ONION.to_string(),
  217. "typing": True,
  218. },
  219. ),
  220. federation_auth_origin=b"farm",
  221. )
  222. self.assertEqual(channel.code, 200)
  223. self.on_new_event.assert_has_calls([call("typing_key", 1, rooms=[ROOM_ID])])
  224. self.assertEqual(self.event_source.get_current_key(), 1)
  225. events = self.get_success(
  226. self.event_source.get_new_events(
  227. user=U_APPLE, from_key=0, limit=0, room_ids=[ROOM_ID], is_guest=False
  228. )
  229. )
  230. self.assertEqual(
  231. events[0],
  232. [
  233. {
  234. "type": EduTypes.TYPING,
  235. "room_id": ROOM_ID,
  236. "content": {"user_ids": [U_ONION.to_string()]},
  237. }
  238. ],
  239. )
  240. def test_started_typing_remote_recv_not_in_room(self) -> None:
  241. self.room_members = [U_APPLE, U_ONION]
  242. self.assertEqual(self.event_source.get_current_key(), 0)
  243. channel = self.make_request(
  244. "PUT",
  245. "/_matrix/federation/v1/send/1000000",
  246. _make_edu_transaction_json(
  247. EduTypes.TYPING,
  248. content={
  249. "room_id": OTHER_ROOM_ID,
  250. "user_id": U_ONION.to_string(),
  251. "typing": True,
  252. },
  253. ),
  254. federation_auth_origin=b"farm",
  255. )
  256. self.assertEqual(channel.code, 200)
  257. self.on_new_event.assert_not_called()
  258. self.assertEqual(self.event_source.get_current_key(), 0)
  259. events = self.get_success(
  260. self.event_source.get_new_events(
  261. user=U_APPLE,
  262. from_key=0,
  263. limit=0,
  264. room_ids=[OTHER_ROOM_ID],
  265. is_guest=False,
  266. )
  267. )
  268. self.assertEqual(events[0], [])
  269. self.assertEqual(events[1], 0)
  270. # Enable federation sending on the main process.
  271. @override_config({"federation_sender_instances": None})
  272. def test_stopped_typing(self) -> None:
  273. self.room_members = [U_APPLE, U_BANANA, U_ONION]
  274. # Gut-wrenching
  275. from synapse.handlers.typing import RoomMember
  276. member = RoomMember(ROOM_ID, U_APPLE.to_string())
  277. self.handler._member_typing_until[member] = 1002000
  278. self.handler._room_typing[ROOM_ID] = {U_APPLE.to_string()}
  279. self.assertEqual(self.event_source.get_current_key(), 0)
  280. self.get_success(
  281. self.handler.stopped_typing(
  282. target_user=U_APPLE,
  283. requester=create_requester(U_APPLE),
  284. room_id=ROOM_ID,
  285. )
  286. )
  287. self.on_new_event.assert_has_calls([call("typing_key", 1, rooms=[ROOM_ID])])
  288. self.mock_federation_client.put_json.assert_called_once_with(
  289. "farm",
  290. path="/_matrix/federation/v1/send/1000000",
  291. data=_expect_edu_transaction(
  292. EduTypes.TYPING,
  293. content={
  294. "room_id": ROOM_ID,
  295. "user_id": U_APPLE.to_string(),
  296. "typing": False,
  297. },
  298. ),
  299. json_data_callback=ANY,
  300. long_retries=True,
  301. backoff_on_all_error_codes=True,
  302. try_trailing_slash_on_400=True,
  303. )
  304. self.assertEqual(self.event_source.get_current_key(), 1)
  305. events = self.get_success(
  306. self.event_source.get_new_events(
  307. user=U_APPLE, from_key=0, limit=0, room_ids=[ROOM_ID], is_guest=False
  308. )
  309. )
  310. self.assertEqual(
  311. events[0],
  312. [
  313. {
  314. "type": EduTypes.TYPING,
  315. "room_id": ROOM_ID,
  316. "content": {"user_ids": []},
  317. }
  318. ],
  319. )
  320. def test_typing_timeout(self) -> None:
  321. self.room_members = [U_APPLE, U_BANANA]
  322. self.assertEqual(self.event_source.get_current_key(), 0)
  323. self.get_success(
  324. self.handler.started_typing(
  325. target_user=U_APPLE,
  326. requester=create_requester(U_APPLE),
  327. room_id=ROOM_ID,
  328. timeout=10000,
  329. )
  330. )
  331. self.on_new_event.assert_has_calls([call("typing_key", 1, rooms=[ROOM_ID])])
  332. self.on_new_event.reset_mock()
  333. self.assertEqual(self.event_source.get_current_key(), 1)
  334. events = self.get_success(
  335. self.event_source.get_new_events(
  336. user=U_APPLE,
  337. from_key=0,
  338. limit=0,
  339. room_ids=[ROOM_ID],
  340. is_guest=False,
  341. )
  342. )
  343. self.assertEqual(
  344. events[0],
  345. [
  346. {
  347. "type": EduTypes.TYPING,
  348. "room_id": ROOM_ID,
  349. "content": {"user_ids": [U_APPLE.to_string()]},
  350. }
  351. ],
  352. )
  353. self.reactor.pump([16])
  354. self.on_new_event.assert_has_calls([call("typing_key", 2, rooms=[ROOM_ID])])
  355. self.assertEqual(self.event_source.get_current_key(), 2)
  356. events = self.get_success(
  357. self.event_source.get_new_events(
  358. user=U_APPLE,
  359. from_key=1,
  360. limit=0,
  361. room_ids=[ROOM_ID],
  362. is_guest=False,
  363. )
  364. )
  365. self.assertEqual(
  366. events[0],
  367. [
  368. {
  369. "type": EduTypes.TYPING,
  370. "room_id": ROOM_ID,
  371. "content": {"user_ids": []},
  372. }
  373. ],
  374. )
  375. # SYN-230 - see if we can still set after timeout
  376. self.get_success(
  377. self.handler.started_typing(
  378. target_user=U_APPLE,
  379. requester=create_requester(U_APPLE),
  380. room_id=ROOM_ID,
  381. timeout=10000,
  382. )
  383. )
  384. self.on_new_event.assert_has_calls([call("typing_key", 3, rooms=[ROOM_ID])])
  385. self.on_new_event.reset_mock()
  386. self.assertEqual(self.event_source.get_current_key(), 3)
  387. events = self.get_success(
  388. self.event_source.get_new_events(
  389. user=U_APPLE,
  390. from_key=0,
  391. limit=0,
  392. room_ids=[ROOM_ID],
  393. is_guest=False,
  394. )
  395. )
  396. self.assertEqual(
  397. events[0],
  398. [
  399. {
  400. "type": EduTypes.TYPING,
  401. "room_id": ROOM_ID,
  402. "content": {"user_ids": [U_APPLE.to_string()]},
  403. }
  404. ],
  405. )