You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

484 lines
17 KiB

  1. # Copyright 2016 OpenMarket Ltd
  2. # Copyright 2018 New Vector Ltd
  3. # Copyright 2020 The Matrix.org Foundation C.I.C.
  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. from typing import Optional
  17. from unittest import mock
  18. from twisted.test.proto_helpers import MemoryReactor
  19. from synapse.api.constants import RoomEncryptionAlgorithms
  20. from synapse.api.errors import NotFoundError, SynapseError
  21. from synapse.appservice import ApplicationService
  22. from synapse.handlers.device import MAX_DEVICE_DISPLAY_NAME_LEN, DeviceHandler
  23. from synapse.server import HomeServer
  24. from synapse.storage.databases.main.appservice import _make_exclusive_regex
  25. from synapse.types import JsonDict
  26. from synapse.util import Clock
  27. from tests import unittest
  28. from tests.test_utils import make_awaitable
  29. from tests.unittest import override_config
  30. user1 = "@boris:aaa"
  31. user2 = "@theresa:bbb"
  32. class DeviceTestCase(unittest.HomeserverTestCase):
  33. def make_homeserver(self, reactor: MemoryReactor, clock: Clock) -> HomeServer:
  34. self.appservice_api = mock.Mock()
  35. hs = self.setup_test_homeserver(
  36. "server",
  37. application_service_api=self.appservice_api,
  38. )
  39. handler = hs.get_device_handler()
  40. assert isinstance(handler, DeviceHandler)
  41. self.handler = handler
  42. self.store = hs.get_datastores().main
  43. return hs
  44. def prepare(self, reactor: MemoryReactor, clock: Clock, hs: HomeServer) -> None:
  45. # These tests assume that it starts 1000 seconds in.
  46. self.reactor.advance(1000)
  47. def test_device_is_created_with_invalid_name(self) -> None:
  48. self.get_failure(
  49. self.handler.check_device_registered(
  50. user_id="@boris:foo",
  51. device_id="foo",
  52. initial_device_display_name="a" * (MAX_DEVICE_DISPLAY_NAME_LEN + 1),
  53. ),
  54. SynapseError,
  55. )
  56. def test_device_is_created_if_doesnt_exist(self) -> None:
  57. res = self.get_success(
  58. self.handler.check_device_registered(
  59. user_id="@boris:foo",
  60. device_id="fco",
  61. initial_device_display_name="display name",
  62. )
  63. )
  64. self.assertEqual(res, "fco")
  65. dev = self.get_success(self.handler.store.get_device("@boris:foo", "fco"))
  66. assert dev is not None
  67. self.assertEqual(dev["display_name"], "display name")
  68. def test_device_is_preserved_if_exists(self) -> None:
  69. res1 = self.get_success(
  70. self.handler.check_device_registered(
  71. user_id="@boris:foo",
  72. device_id="fco",
  73. initial_device_display_name="display name",
  74. )
  75. )
  76. self.assertEqual(res1, "fco")
  77. res2 = self.get_success(
  78. self.handler.check_device_registered(
  79. user_id="@boris:foo",
  80. device_id="fco",
  81. initial_device_display_name="new display name",
  82. )
  83. )
  84. self.assertEqual(res2, "fco")
  85. dev = self.get_success(self.handler.store.get_device("@boris:foo", "fco"))
  86. assert dev is not None
  87. self.assertEqual(dev["display_name"], "display name")
  88. def test_device_id_is_made_up_if_unspecified(self) -> None:
  89. device_id = self.get_success(
  90. self.handler.check_device_registered(
  91. user_id="@theresa:foo",
  92. device_id=None,
  93. initial_device_display_name="display",
  94. )
  95. )
  96. dev = self.get_success(self.handler.store.get_device("@theresa:foo", device_id))
  97. assert dev is not None
  98. self.assertEqual(dev["display_name"], "display")
  99. def test_get_devices_by_user(self) -> None:
  100. self._record_users()
  101. res = self.get_success(self.handler.get_devices_by_user(user1))
  102. self.assertEqual(3, len(res))
  103. device_map = {d["device_id"]: d for d in res}
  104. self.assertDictContainsSubset(
  105. {
  106. "user_id": user1,
  107. "device_id": "xyz",
  108. "display_name": "display 0",
  109. "last_seen_ip": None,
  110. "last_seen_ts": None,
  111. },
  112. device_map["xyz"],
  113. )
  114. self.assertDictContainsSubset(
  115. {
  116. "user_id": user1,
  117. "device_id": "fco",
  118. "display_name": "display 1",
  119. "last_seen_ip": "ip1",
  120. "last_seen_ts": 1000000,
  121. },
  122. device_map["fco"],
  123. )
  124. self.assertDictContainsSubset(
  125. {
  126. "user_id": user1,
  127. "device_id": "abc",
  128. "display_name": "display 2",
  129. "last_seen_ip": "ip3",
  130. "last_seen_ts": 3000000,
  131. },
  132. device_map["abc"],
  133. )
  134. def test_get_device(self) -> None:
  135. self._record_users()
  136. res = self.get_success(self.handler.get_device(user1, "abc"))
  137. self.assertDictContainsSubset(
  138. {
  139. "user_id": user1,
  140. "device_id": "abc",
  141. "display_name": "display 2",
  142. "last_seen_ip": "ip3",
  143. "last_seen_ts": 3000000,
  144. },
  145. res,
  146. )
  147. def test_delete_device(self) -> None:
  148. self._record_users()
  149. # delete the device
  150. self.get_success(self.handler.delete_devices(user1, ["abc"]))
  151. # check the device was deleted
  152. self.get_failure(self.handler.get_device(user1, "abc"), NotFoundError)
  153. # we'd like to check the access token was invalidated, but that's a
  154. # bit of a PITA.
  155. def test_delete_device_and_device_inbox(self) -> None:
  156. self._record_users()
  157. # add an device_inbox
  158. self.get_success(
  159. self.store.db_pool.simple_insert(
  160. "device_inbox",
  161. {
  162. "user_id": user1,
  163. "device_id": "abc",
  164. "stream_id": 1,
  165. "message_json": "{}",
  166. },
  167. )
  168. )
  169. # delete the device
  170. self.get_success(self.handler.delete_devices(user1, ["abc"]))
  171. # check that the device_inbox was deleted
  172. res = self.get_success(
  173. self.store.db_pool.simple_select_one(
  174. table="device_inbox",
  175. keyvalues={"user_id": user1, "device_id": "abc"},
  176. retcols=("user_id", "device_id"),
  177. allow_none=True,
  178. desc="get_device_id_from_device_inbox",
  179. )
  180. )
  181. self.assertIsNone(res)
  182. def test_update_device(self) -> None:
  183. self._record_users()
  184. update = {"display_name": "new display"}
  185. self.get_success(self.handler.update_device(user1, "abc", update))
  186. res = self.get_success(self.handler.get_device(user1, "abc"))
  187. self.assertEqual(res["display_name"], "new display")
  188. def test_update_device_too_long_display_name(self) -> None:
  189. """Update a device with a display name that is invalid (too long)."""
  190. self._record_users()
  191. # Request to update a device display name with a new value that is longer than allowed.
  192. update = {"display_name": "a" * (MAX_DEVICE_DISPLAY_NAME_LEN + 1)}
  193. self.get_failure(
  194. self.handler.update_device(user1, "abc", update),
  195. SynapseError,
  196. )
  197. # Ensure the display name was not updated.
  198. res = self.get_success(self.handler.get_device(user1, "abc"))
  199. self.assertEqual(res["display_name"], "display 2")
  200. def test_update_unknown_device(self) -> None:
  201. update = {"display_name": "new_display"}
  202. self.get_failure(
  203. self.handler.update_device("user_id", "unknown_device_id", update),
  204. NotFoundError,
  205. )
  206. def _record_users(self) -> None:
  207. # check this works for both devices which have a recorded client_ip,
  208. # and those which don't.
  209. self._record_user(user1, "xyz", "display 0")
  210. self._record_user(user1, "fco", "display 1", "token1", "ip1")
  211. self._record_user(user1, "abc", "display 2", "token2", "ip2")
  212. self._record_user(user1, "abc", "display 2", "token3", "ip3")
  213. self._record_user(user2, "def", "dispkay", "token4", "ip4")
  214. self.reactor.advance(10000)
  215. def _record_user(
  216. self,
  217. user_id: str,
  218. device_id: str,
  219. display_name: str,
  220. access_token: Optional[str] = None,
  221. ip: Optional[str] = None,
  222. ) -> None:
  223. device_id = self.get_success(
  224. self.handler.check_device_registered(
  225. user_id=user_id,
  226. device_id=device_id,
  227. initial_device_display_name=display_name,
  228. )
  229. )
  230. if access_token is not None and ip is not None:
  231. self.get_success(
  232. self.store.insert_client_ip(
  233. user_id, access_token, ip, "user_agent", device_id
  234. )
  235. )
  236. self.reactor.advance(1000)
  237. @override_config({"experimental_features": {"msc3984_appservice_key_query": True}})
  238. def test_on_federation_query_user_devices_appservice(self) -> None:
  239. """Test that querying of appservices for keys overrides responses from the database."""
  240. local_user = "@boris:" + self.hs.hostname
  241. device_1 = "abc"
  242. device_2 = "def"
  243. device_3 = "ghi"
  244. # There are 3 devices:
  245. #
  246. # 1. One which is uploaded to the homeserver.
  247. # 2. One which is uploaded to the homeserver, but a newer copy is returned
  248. # by the appservice.
  249. # 3. One which is only returned by the appservice.
  250. device_key_1: JsonDict = {
  251. "user_id": local_user,
  252. "device_id": device_1,
  253. "algorithms": [
  254. "m.olm.curve25519-aes-sha2",
  255. RoomEncryptionAlgorithms.MEGOLM_V1_AES_SHA2,
  256. ],
  257. "keys": {
  258. "ed25519:abc": "base64+ed25519+key",
  259. "curve25519:abc": "base64+curve25519+key",
  260. },
  261. "signatures": {local_user: {"ed25519:abc": "base64+signature"}},
  262. }
  263. device_key_2a: JsonDict = {
  264. "user_id": local_user,
  265. "device_id": device_2,
  266. "algorithms": [
  267. "m.olm.curve25519-aes-sha2",
  268. RoomEncryptionAlgorithms.MEGOLM_V1_AES_SHA2,
  269. ],
  270. "keys": {
  271. "ed25519:def": "base64+ed25519+key",
  272. "curve25519:def": "base64+curve25519+key",
  273. },
  274. "signatures": {local_user: {"ed25519:def": "base64+signature"}},
  275. }
  276. device_key_2b: JsonDict = {
  277. "user_id": local_user,
  278. "device_id": device_2,
  279. "algorithms": [
  280. "m.olm.curve25519-aes-sha2",
  281. RoomEncryptionAlgorithms.MEGOLM_V1_AES_SHA2,
  282. ],
  283. # The device ID is the same (above), but the keys are different.
  284. "keys": {
  285. "ed25519:xyz": "base64+ed25519+key",
  286. "curve25519:xyz": "base64+curve25519+key",
  287. },
  288. "signatures": {local_user: {"ed25519:xyz": "base64+signature"}},
  289. }
  290. device_key_3: JsonDict = {
  291. "user_id": local_user,
  292. "device_id": device_3,
  293. "algorithms": [
  294. "m.olm.curve25519-aes-sha2",
  295. RoomEncryptionAlgorithms.MEGOLM_V1_AES_SHA2,
  296. ],
  297. "keys": {
  298. "ed25519:jkl": "base64+ed25519+key",
  299. "curve25519:jkl": "base64+curve25519+key",
  300. },
  301. "signatures": {local_user: {"ed25519:jkl": "base64+signature"}},
  302. }
  303. # Upload keys for devices 1 & 2a.
  304. e2e_keys_handler = self.hs.get_e2e_keys_handler()
  305. self.get_success(
  306. e2e_keys_handler.upload_keys_for_user(
  307. local_user, device_1, {"device_keys": device_key_1}
  308. )
  309. )
  310. self.get_success(
  311. e2e_keys_handler.upload_keys_for_user(
  312. local_user, device_2, {"device_keys": device_key_2a}
  313. )
  314. )
  315. # Inject an appservice interested in this user.
  316. appservice = ApplicationService(
  317. token="i_am_an_app_service",
  318. id="1234",
  319. namespaces={"users": [{"regex": r"@boris:.+", "exclusive": True}]},
  320. # Note: this user does not have to match the regex above
  321. sender="@as_main:test",
  322. )
  323. self.hs.get_datastores().main.services_cache = [appservice]
  324. self.hs.get_datastores().main.exclusive_user_regex = _make_exclusive_regex(
  325. [appservice]
  326. )
  327. # Setup a response.
  328. self.appservice_api.query_keys.return_value = make_awaitable(
  329. {
  330. "device_keys": {
  331. local_user: {device_2: device_key_2b, device_3: device_key_3}
  332. }
  333. }
  334. )
  335. # Request all devices.
  336. res = self.get_success(
  337. self.handler.on_federation_query_user_devices(local_user)
  338. )
  339. self.assertIn("devices", res)
  340. res_devices = res["devices"]
  341. for device in res_devices:
  342. device["keys"].pop("unsigned", None)
  343. self.assertEqual(
  344. res_devices,
  345. [
  346. {"device_id": device_1, "keys": device_key_1},
  347. {"device_id": device_2, "keys": device_key_2b},
  348. {"device_id": device_3, "keys": device_key_3},
  349. ],
  350. )
  351. class DehydrationTestCase(unittest.HomeserverTestCase):
  352. def make_homeserver(self, reactor: MemoryReactor, clock: Clock) -> HomeServer:
  353. hs = self.setup_test_homeserver("server")
  354. handler = hs.get_device_handler()
  355. assert isinstance(handler, DeviceHandler)
  356. self.handler = handler
  357. self.registration = hs.get_registration_handler()
  358. self.auth = hs.get_auth()
  359. self.store = hs.get_datastores().main
  360. return hs
  361. def test_dehydrate_and_rehydrate_device(self) -> None:
  362. user_id = "@boris:dehydration"
  363. self.get_success(self.store.register_user(user_id, "foobar"))
  364. # First check if we can store and fetch a dehydrated device
  365. stored_dehydrated_device_id = self.get_success(
  366. self.handler.store_dehydrated_device(
  367. user_id=user_id,
  368. device_data={"device_data": {"foo": "bar"}},
  369. initial_device_display_name="dehydrated device",
  370. )
  371. )
  372. result = self.get_success(self.handler.get_dehydrated_device(user_id=user_id))
  373. assert result is not None
  374. retrieved_device_id, device_data = result
  375. self.assertEqual(retrieved_device_id, stored_dehydrated_device_id)
  376. self.assertEqual(device_data, {"device_data": {"foo": "bar"}})
  377. # Create a new login for the user and dehydrated the device
  378. device_id, access_token, _expiration_time, _refresh_token = self.get_success(
  379. self.registration.register_device(
  380. user_id=user_id,
  381. device_id=None,
  382. initial_display_name="new device",
  383. )
  384. )
  385. # Trying to claim a nonexistent device should throw an error
  386. self.get_failure(
  387. self.handler.rehydrate_device(
  388. user_id=user_id,
  389. access_token=access_token,
  390. device_id="not the right device ID",
  391. ),
  392. NotFoundError,
  393. )
  394. # dehydrating the right devices should succeed and change our device ID
  395. # to the dehydrated device's ID
  396. res = self.get_success(
  397. self.handler.rehydrate_device(
  398. user_id=user_id,
  399. access_token=access_token,
  400. device_id=retrieved_device_id,
  401. )
  402. )
  403. self.assertEqual(res, {"success": True})
  404. # make sure that our device ID has changed
  405. user_info = self.get_success(self.auth.get_user_by_access_token(access_token))
  406. self.assertEqual(user_info.device_id, retrieved_device_id)
  407. # make sure the device has the display name that was set from the login
  408. res = self.get_success(self.handler.get_device(user_id, retrieved_device_id))
  409. self.assertEqual(res["display_name"], "new device")
  410. # make sure that the device ID that we were initially assigned no longer exists
  411. self.get_failure(
  412. self.handler.get_device(user_id, device_id),
  413. NotFoundError,
  414. )
  415. # make sure that there's no device available for dehydrating now
  416. ret = self.get_success(self.handler.get_dehydrated_device(user_id=user_id))
  417. self.assertIsNone(ret)