matrix-synapse 1.145.0rc2__cp310-abi3-manylinux_2_28_x86_64.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of matrix-synapse might be problematic. Click here for more details.

Files changed (1636) hide show
  1. AUTHORS.rst +51 -0
  2. CHANGES.md +1732 -0
  3. CONTRIBUTING.md +3 -0
  4. Cargo.lock +1923 -0
  5. Cargo.toml +6 -0
  6. INSTALL.md +7 -0
  7. LICENSE-AGPL-3.0 +661 -0
  8. LICENSE-COMMERCIAL +6 -0
  9. UPGRADE.rst +7 -0
  10. book.toml +47 -0
  11. changelog.d/.gitignore +1 -0
  12. demo/.gitignore +4 -0
  13. demo/clean.sh +22 -0
  14. demo/start.sh +173 -0
  15. demo/stop.sh +14 -0
  16. docs/.sample_config_header.yaml +12 -0
  17. docs/CAPTCHA_SETUP.md +37 -0
  18. docs/README.md +86 -0
  19. docs/SUMMARY.md +126 -0
  20. docs/admin_api/README.rst +13 -0
  21. docs/admin_api/account_validity.md +47 -0
  22. docs/admin_api/client_server_api_extensions.md +67 -0
  23. docs/admin_api/event_reports.md +185 -0
  24. docs/admin_api/experimental_features.md +52 -0
  25. docs/admin_api/fetch_event.md +53 -0
  26. docs/admin_api/media_admin_api.md +372 -0
  27. docs/admin_api/purge_history_api.md +77 -0
  28. docs/admin_api/register_api.md +92 -0
  29. docs/admin_api/room_membership.md +35 -0
  30. docs/admin_api/rooms.md +1190 -0
  31. docs/admin_api/scheduled_tasks.md +54 -0
  32. docs/admin_api/server_notices.md +48 -0
  33. docs/admin_api/statistics.md +132 -0
  34. docs/admin_api/user_admin_api.md +1602 -0
  35. docs/admin_api/version_api.md +23 -0
  36. docs/ancient_architecture_notes.md +81 -0
  37. docs/application_services.md +36 -0
  38. docs/architecture.md +65 -0
  39. docs/auth_chain_diff.dot +32 -0
  40. docs/auth_chain_diff.dot.png +0 -0
  41. docs/auth_chain_difference_algorithm.md +141 -0
  42. docs/changelogs/CHANGES-2019.md +1039 -0
  43. docs/changelogs/CHANGES-2020.md +2145 -0
  44. docs/changelogs/CHANGES-2021.md +2573 -0
  45. docs/changelogs/CHANGES-2022.md +2766 -0
  46. docs/changelogs/CHANGES-2023.md +2202 -0
  47. docs/changelogs/CHANGES-2024.md +1586 -0
  48. docs/changelogs/CHANGES-pre-1.0.md +3641 -0
  49. docs/changelogs/README.md +1 -0
  50. docs/code_style.md +119 -0
  51. docs/consent_tracking.md +197 -0
  52. docs/delegate.md +111 -0
  53. docs/deprecation_policy.md +89 -0
  54. docs/development/cas.md +64 -0
  55. docs/development/contributing_guide.md +554 -0
  56. docs/development/database_schema.md +404 -0
  57. docs/development/demo.md +42 -0
  58. docs/development/dependencies.md +326 -0
  59. docs/development/experimental_features.md +37 -0
  60. docs/development/git.md +148 -0
  61. docs/development/img/git/branches.jpg +0 -0
  62. docs/development/img/git/clean.png +0 -0
  63. docs/development/img/git/squash.png +0 -0
  64. docs/development/internal_documentation/README.md +12 -0
  65. docs/development/internal_documentation/release_notes_review_checklist.md +12 -0
  66. docs/development/releases.md +37 -0
  67. docs/development/reviews.md +41 -0
  68. docs/development/room-dag-concepts.md +115 -0
  69. docs/development/saml.md +40 -0
  70. docs/development/synapse_architecture/cancellation.md +392 -0
  71. docs/development/synapse_architecture/faster_joins.md +375 -0
  72. docs/development/synapse_architecture/streams.md +198 -0
  73. docs/element_logo_white_bg.svg +94 -0
  74. docs/favicon.png +0 -0
  75. docs/favicon.svg +58 -0
  76. docs/federate.md +67 -0
  77. docs/jwt.md +105 -0
  78. docs/log_contexts.md +566 -0
  79. docs/manhole.md +101 -0
  80. docs/media_repository.md +78 -0
  81. docs/message_retention_policies.md +207 -0
  82. docs/metrics-howto.md +315 -0
  83. docs/modules/account_data_callbacks.md +106 -0
  84. docs/modules/account_validity_callbacks.md +57 -0
  85. docs/modules/add_extra_fields_to_client_events_unsigned.md +32 -0
  86. docs/modules/background_update_controller_callbacks.md +71 -0
  87. docs/modules/index.md +53 -0
  88. docs/modules/media_repository_callbacks.md +131 -0
  89. docs/modules/password_auth_provider_callbacks.md +270 -0
  90. docs/modules/porting_legacy_module.md +20 -0
  91. docs/modules/presence_router_callbacks.md +112 -0
  92. docs/modules/ratelimit_callbacks.md +43 -0
  93. docs/modules/spam_checker_callbacks.md +517 -0
  94. docs/modules/third_party_rules_callbacks.md +350 -0
  95. docs/modules/writing_a_module.md +141 -0
  96. docs/openid.md +783 -0
  97. docs/opentracing.md +99 -0
  98. docs/other/running_synapse_on_single_board_computers.md +75 -0
  99. docs/password_auth_providers.md +129 -0
  100. docs/postgres.md +269 -0
  101. docs/presence_router_module.md +247 -0
  102. docs/privacy_policy_templates/en/1.0.html +26 -0
  103. docs/privacy_policy_templates/en/success.html +11 -0
  104. docs/replication.md +36 -0
  105. docs/reverse_proxy.md +327 -0
  106. docs/room_and_user_statistics.md +22 -0
  107. docs/sample_config.yaml +47 -0
  108. docs/sample_log_config.yaml +75 -0
  109. docs/server_notices.md +66 -0
  110. docs/setup/forward_proxy.md +89 -0
  111. docs/setup/installation.md +656 -0
  112. docs/setup/security.md +41 -0
  113. docs/setup/turn/coturn.md +197 -0
  114. docs/setup/turn/eturnal.md +170 -0
  115. docs/spam_checker.md +122 -0
  116. docs/sso_mapping_providers.md +208 -0
  117. docs/structured_logging.md +80 -0
  118. docs/synctl_workers.md +36 -0
  119. docs/systemd-with-workers/README.md +101 -0
  120. docs/systemd-with-workers/system/matrix-synapse-worker@.service +26 -0
  121. docs/systemd-with-workers/system/matrix-synapse.service +23 -0
  122. docs/systemd-with-workers/system/matrix-synapse.target +6 -0
  123. docs/systemd-with-workers/workers/background_worker.yaml +4 -0
  124. docs/systemd-with-workers/workers/event_persister.yaml +20 -0
  125. docs/systemd-with-workers/workers/federation_sender.yaml +4 -0
  126. docs/systemd-with-workers/workers/generic_worker.yaml +11 -0
  127. docs/systemd-with-workers/workers/media_worker.yaml +11 -0
  128. docs/systemd-with-workers/workers/pusher_worker.yaml +4 -0
  129. docs/tcp_replication.md +258 -0
  130. docs/templates.md +254 -0
  131. docs/turn-howto.md +168 -0
  132. docs/upgrade.md +2876 -0
  133. docs/usage/administration/README.md +7 -0
  134. docs/usage/administration/admin_api/README.md +47 -0
  135. docs/usage/administration/admin_api/background_updates.md +109 -0
  136. docs/usage/administration/admin_api/federation.md +212 -0
  137. docs/usage/administration/admin_api/registration_tokens.md +298 -0
  138. docs/usage/administration/admin_faq.md +294 -0
  139. docs/usage/administration/backups.md +125 -0
  140. docs/usage/administration/database_maintenance_tools.md +18 -0
  141. docs/usage/administration/monitoring/reporting_homeserver_usage_statistics.md +77 -0
  142. docs/usage/administration/monthly_active_users.md +84 -0
  143. docs/usage/administration/request_log.md +44 -0
  144. docs/usage/administration/state_groups.md +25 -0
  145. docs/usage/administration/understanding_synapse_through_grafana_graphs.md +83 -0
  146. docs/usage/administration/useful_sql_for_admins.md +216 -0
  147. docs/usage/configuration/README.md +4 -0
  148. docs/usage/configuration/config_documentation.md +4768 -0
  149. docs/usage/configuration/homeserver_sample_config.md +17 -0
  150. docs/usage/configuration/logging_sample_config.md +19 -0
  151. docs/usage/configuration/user_authentication/README.md +15 -0
  152. docs/usage/configuration/user_authentication/refresh_tokens.md +139 -0
  153. docs/usage/configuration/user_authentication/single_sign_on/README.md +5 -0
  154. docs/usage/configuration/user_authentication/single_sign_on/cas.md +8 -0
  155. docs/usage/configuration/user_authentication/single_sign_on/saml.md +8 -0
  156. docs/user_directory.md +130 -0
  157. docs/website_files/README.md +35 -0
  158. docs/website_files/indent-section-headers.css +7 -0
  159. docs/website_files/remove-nav-buttons.css +8 -0
  160. docs/website_files/table-of-contents.css +47 -0
  161. docs/website_files/table-of-contents.js +148 -0
  162. docs/website_files/theme/index.hbs +324 -0
  163. docs/website_files/version-picker.css +78 -0
  164. docs/website_files/version-picker.js +147 -0
  165. docs/website_files/version.js +1 -0
  166. docs/welcome_and_overview.md +67 -0
  167. docs/workers.md +897 -0
  168. matrix_synapse-1.145.0rc2.dist-info/METADATA +260 -0
  169. matrix_synapse-1.145.0rc2.dist-info/RECORD +1636 -0
  170. matrix_synapse-1.145.0rc2.dist-info/WHEEL +5 -0
  171. matrix_synapse-1.145.0rc2.dist-info/entry_points.txt +13 -0
  172. matrix_synapse-1.145.0rc2.dist-info/licenses/AUTHORS.rst +51 -0
  173. matrix_synapse-1.145.0rc2.dist-info/licenses/LICENSE-AGPL-3.0 +661 -0
  174. matrix_synapse-1.145.0rc2.dist-info/licenses/LICENSE-COMMERCIAL +6 -0
  175. mypy.ini +115 -0
  176. rust/Cargo.toml +63 -0
  177. rust/build.rs +45 -0
  178. scripts-dev/build_debian_packages.py +228 -0
  179. scripts-dev/check-newsfragment.sh +62 -0
  180. scripts-dev/check_line_terminators.sh +29 -0
  181. scripts-dev/check_locked_deps_have_sdists.py +64 -0
  182. scripts-dev/check_schema_delta.py +240 -0
  183. scripts-dev/complement.sh +332 -0
  184. scripts-dev/config-lint.sh +15 -0
  185. scripts-dev/database-save.sh +15 -0
  186. scripts-dev/docker_update_debian_changelog.sh +70 -0
  187. scripts-dev/dump_macaroon.py +25 -0
  188. scripts-dev/federation_client.py +428 -0
  189. scripts-dev/gen_config_documentation.py +510 -0
  190. scripts-dev/generate_sample_config.sh +28 -0
  191. scripts-dev/lint.sh +141 -0
  192. scripts-dev/make_full_schema.sh +310 -0
  193. scripts-dev/mypy_synapse_plugin.py +918 -0
  194. scripts-dev/next_github_number.sh +9 -0
  195. scripts-dev/release.py +986 -0
  196. scripts-dev/schema_versions.py +182 -0
  197. scripts-dev/sign_json.py +172 -0
  198. synapse/__init__.py +97 -0
  199. synapse/_scripts/__init__.py +0 -0
  200. synapse/_scripts/export_signing_key.py +109 -0
  201. synapse/_scripts/generate_config.py +83 -0
  202. synapse/_scripts/generate_log_config.py +56 -0
  203. synapse/_scripts/generate_signing_key.py +55 -0
  204. synapse/_scripts/generate_workers_map.py +318 -0
  205. synapse/_scripts/hash_password.py +95 -0
  206. synapse/_scripts/move_remote_media_to_new_store.py +128 -0
  207. synapse/_scripts/register_new_matrix_user.py +402 -0
  208. synapse/_scripts/review_recent_signups.py +212 -0
  209. synapse/_scripts/synapse_port_db.py +1604 -0
  210. synapse/_scripts/synctl.py +365 -0
  211. synapse/_scripts/update_synapse_database.py +130 -0
  212. synapse/api/__init__.py +20 -0
  213. synapse/api/auth/__init__.py +207 -0
  214. synapse/api/auth/base.py +406 -0
  215. synapse/api/auth/internal.py +299 -0
  216. synapse/api/auth/mas.py +436 -0
  217. synapse/api/auth/msc3861_delegated.py +617 -0
  218. synapse/api/auth_blocking.py +144 -0
  219. synapse/api/constants.py +379 -0
  220. synapse/api/errors.py +913 -0
  221. synapse/api/filtering.py +537 -0
  222. synapse/api/presence.py +102 -0
  223. synapse/api/ratelimiting.py +481 -0
  224. synapse/api/room_versions.py +535 -0
  225. synapse/api/urls.py +118 -0
  226. synapse/app/__init__.py +62 -0
  227. synapse/app/_base.py +906 -0
  228. synapse/app/admin_cmd.py +388 -0
  229. synapse/app/appservice.py +30 -0
  230. synapse/app/client_reader.py +30 -0
  231. synapse/app/complement_fork_starter.py +206 -0
  232. synapse/app/event_creator.py +29 -0
  233. synapse/app/federation_reader.py +30 -0
  234. synapse/app/federation_sender.py +30 -0
  235. synapse/app/frontend_proxy.py +30 -0
  236. synapse/app/generic_worker.py +475 -0
  237. synapse/app/homeserver.py +505 -0
  238. synapse/app/media_repository.py +30 -0
  239. synapse/app/phone_stats_home.py +292 -0
  240. synapse/app/pusher.py +30 -0
  241. synapse/app/synchrotron.py +30 -0
  242. synapse/app/user_dir.py +31 -0
  243. synapse/appservice/__init__.py +458 -0
  244. synapse/appservice/api.py +567 -0
  245. synapse/appservice/scheduler.py +565 -0
  246. synapse/config/__init__.py +27 -0
  247. synapse/config/__main__.py +62 -0
  248. synapse/config/_base.py +1154 -0
  249. synapse/config/_base.pyi +216 -0
  250. synapse/config/_util.py +99 -0
  251. synapse/config/account_validity.py +116 -0
  252. synapse/config/api.py +141 -0
  253. synapse/config/appservice.py +210 -0
  254. synapse/config/auth.py +80 -0
  255. synapse/config/auto_accept_invites.py +43 -0
  256. synapse/config/background_updates.py +44 -0
  257. synapse/config/cache.py +231 -0
  258. synapse/config/captcha.py +90 -0
  259. synapse/config/cas.py +116 -0
  260. synapse/config/consent.py +73 -0
  261. synapse/config/database.py +184 -0
  262. synapse/config/emailconfig.py +367 -0
  263. synapse/config/experimental.py +601 -0
  264. synapse/config/federation.py +114 -0
  265. synapse/config/homeserver.py +141 -0
  266. synapse/config/jwt.py +55 -0
  267. synapse/config/key.py +447 -0
  268. synapse/config/logger.py +390 -0
  269. synapse/config/mas.py +192 -0
  270. synapse/config/matrixrtc.py +66 -0
  271. synapse/config/metrics.py +93 -0
  272. synapse/config/modules.py +40 -0
  273. synapse/config/oembed.py +185 -0
  274. synapse/config/oidc.py +509 -0
  275. synapse/config/password_auth_providers.py +82 -0
  276. synapse/config/push.py +64 -0
  277. synapse/config/ratelimiting.py +260 -0
  278. synapse/config/redis.py +74 -0
  279. synapse/config/registration.py +296 -0
  280. synapse/config/repository.py +311 -0
  281. synapse/config/retention.py +162 -0
  282. synapse/config/room.py +88 -0
  283. synapse/config/room_directory.py +165 -0
  284. synapse/config/saml2.py +251 -0
  285. synapse/config/server.py +1139 -0
  286. synapse/config/server_notices.py +84 -0
  287. synapse/config/spam_checker.py +66 -0
  288. synapse/config/sso.py +121 -0
  289. synapse/config/stats.py +54 -0
  290. synapse/config/third_party_event_rules.py +40 -0
  291. synapse/config/tls.py +192 -0
  292. synapse/config/tracer.py +71 -0
  293. synapse/config/user_directory.py +47 -0
  294. synapse/config/user_types.py +42 -0
  295. synapse/config/voip.py +59 -0
  296. synapse/config/workers.py +642 -0
  297. synapse/crypto/__init__.py +20 -0
  298. synapse/crypto/context_factory.py +278 -0
  299. synapse/crypto/event_signing.py +194 -0
  300. synapse/crypto/keyring.py +980 -0
  301. synapse/event_auth.py +1266 -0
  302. synapse/events/__init__.py +667 -0
  303. synapse/events/auto_accept_invites.py +216 -0
  304. synapse/events/builder.py +387 -0
  305. synapse/events/presence_router.py +243 -0
  306. synapse/events/snapshot.py +559 -0
  307. synapse/events/utils.py +924 -0
  308. synapse/events/validator.py +305 -0
  309. synapse/federation/__init__.py +22 -0
  310. synapse/federation/federation_base.py +382 -0
  311. synapse/federation/federation_client.py +2133 -0
  312. synapse/federation/federation_server.py +1543 -0
  313. synapse/federation/persistence.py +70 -0
  314. synapse/federation/send_queue.py +532 -0
  315. synapse/federation/sender/__init__.py +1165 -0
  316. synapse/federation/sender/per_destination_queue.py +886 -0
  317. synapse/federation/sender/transaction_manager.py +210 -0
  318. synapse/federation/transport/__init__.py +28 -0
  319. synapse/federation/transport/client.py +1199 -0
  320. synapse/federation/transport/server/__init__.py +334 -0
  321. synapse/federation/transport/server/_base.py +429 -0
  322. synapse/federation/transport/server/federation.py +910 -0
  323. synapse/federation/units.py +133 -0
  324. synapse/handlers/__init__.py +20 -0
  325. synapse/handlers/account.py +162 -0
  326. synapse/handlers/account_data.py +360 -0
  327. synapse/handlers/account_validity.py +362 -0
  328. synapse/handlers/admin.py +615 -0
  329. synapse/handlers/appservice.py +989 -0
  330. synapse/handlers/auth.py +2482 -0
  331. synapse/handlers/cas.py +413 -0
  332. synapse/handlers/deactivate_account.py +363 -0
  333. synapse/handlers/delayed_events.py +607 -0
  334. synapse/handlers/device.py +1869 -0
  335. synapse/handlers/devicemessage.py +399 -0
  336. synapse/handlers/directory.py +545 -0
  337. synapse/handlers/e2e_keys.py +1835 -0
  338. synapse/handlers/e2e_room_keys.py +455 -0
  339. synapse/handlers/event_auth.py +390 -0
  340. synapse/handlers/events.py +203 -0
  341. synapse/handlers/federation.py +2042 -0
  342. synapse/handlers/federation_event.py +2420 -0
  343. synapse/handlers/identity.py +812 -0
  344. synapse/handlers/initial_sync.py +528 -0
  345. synapse/handlers/jwt.py +120 -0
  346. synapse/handlers/message.py +2357 -0
  347. synapse/handlers/oidc.py +1801 -0
  348. synapse/handlers/pagination.py +811 -0
  349. synapse/handlers/password_policy.py +102 -0
  350. synapse/handlers/presence.py +2634 -0
  351. synapse/handlers/profile.py +656 -0
  352. synapse/handlers/push_rules.py +164 -0
  353. synapse/handlers/read_marker.py +79 -0
  354. synapse/handlers/receipts.py +351 -0
  355. synapse/handlers/register.py +1059 -0
  356. synapse/handlers/relations.py +623 -0
  357. synapse/handlers/reports.py +98 -0
  358. synapse/handlers/room.py +2449 -0
  359. synapse/handlers/room_list.py +632 -0
  360. synapse/handlers/room_member.py +2366 -0
  361. synapse/handlers/room_member_worker.py +146 -0
  362. synapse/handlers/room_policy.py +186 -0
  363. synapse/handlers/room_summary.py +1057 -0
  364. synapse/handlers/saml.py +524 -0
  365. synapse/handlers/search.py +723 -0
  366. synapse/handlers/send_email.py +209 -0
  367. synapse/handlers/set_password.py +71 -0
  368. synapse/handlers/sliding_sync/__init__.py +1961 -0
  369. synapse/handlers/sliding_sync/extensions.py +969 -0
  370. synapse/handlers/sliding_sync/room_lists.py +2317 -0
  371. synapse/handlers/sliding_sync/store.py +126 -0
  372. synapse/handlers/sso.py +1291 -0
  373. synapse/handlers/state_deltas.py +82 -0
  374. synapse/handlers/stats.py +322 -0
  375. synapse/handlers/sync.py +3096 -0
  376. synapse/handlers/thread_subscriptions.py +190 -0
  377. synapse/handlers/typing.py +612 -0
  378. synapse/handlers/ui_auth/__init__.py +48 -0
  379. synapse/handlers/ui_auth/checkers.py +332 -0
  380. synapse/handlers/user_directory.py +786 -0
  381. synapse/handlers/worker_lock.py +371 -0
  382. synapse/http/__init__.py +105 -0
  383. synapse/http/additional_resource.py +62 -0
  384. synapse/http/client.py +1377 -0
  385. synapse/http/connectproxyclient.py +316 -0
  386. synapse/http/federation/__init__.py +19 -0
  387. synapse/http/federation/matrix_federation_agent.py +490 -0
  388. synapse/http/federation/srv_resolver.py +196 -0
  389. synapse/http/federation/well_known_resolver.py +368 -0
  390. synapse/http/matrixfederationclient.py +1874 -0
  391. synapse/http/proxy.py +290 -0
  392. synapse/http/proxyagent.py +497 -0
  393. synapse/http/replicationagent.py +203 -0
  394. synapse/http/request_metrics.py +309 -0
  395. synapse/http/server.py +1111 -0
  396. synapse/http/servlet.py +1018 -0
  397. synapse/http/site.py +952 -0
  398. synapse/http/types.py +27 -0
  399. synapse/logging/__init__.py +31 -0
  400. synapse/logging/_remote.py +261 -0
  401. synapse/logging/_terse_json.py +95 -0
  402. synapse/logging/context.py +1209 -0
  403. synapse/logging/formatter.py +62 -0
  404. synapse/logging/handlers.py +99 -0
  405. synapse/logging/loggers.py +25 -0
  406. synapse/logging/opentracing.py +1132 -0
  407. synapse/logging/scopecontextmanager.py +160 -0
  408. synapse/media/_base.py +831 -0
  409. synapse/media/filepath.py +417 -0
  410. synapse/media/media_repository.py +1595 -0
  411. synapse/media/media_storage.py +703 -0
  412. synapse/media/oembed.py +277 -0
  413. synapse/media/preview_html.py +556 -0
  414. synapse/media/storage_provider.py +195 -0
  415. synapse/media/thumbnailer.py +833 -0
  416. synapse/media/url_previewer.py +884 -0
  417. synapse/metrics/__init__.py +748 -0
  418. synapse/metrics/_gc.py +219 -0
  419. synapse/metrics/_reactor_metrics.py +171 -0
  420. synapse/metrics/_types.py +38 -0
  421. synapse/metrics/background_process_metrics.py +555 -0
  422. synapse/metrics/common_usage_metrics.py +95 -0
  423. synapse/metrics/jemalloc.py +248 -0
  424. synapse/module_api/__init__.py +2131 -0
  425. synapse/module_api/callbacks/__init__.py +50 -0
  426. synapse/module_api/callbacks/account_validity_callbacks.py +106 -0
  427. synapse/module_api/callbacks/media_repository_callbacks.py +157 -0
  428. synapse/module_api/callbacks/ratelimit_callbacks.py +78 -0
  429. synapse/module_api/callbacks/spamchecker_callbacks.py +991 -0
  430. synapse/module_api/callbacks/third_party_event_rules_callbacks.py +592 -0
  431. synapse/module_api/errors.py +42 -0
  432. synapse/notifier.py +969 -0
  433. synapse/push/__init__.py +212 -0
  434. synapse/push/bulk_push_rule_evaluator.py +635 -0
  435. synapse/push/clientformat.py +126 -0
  436. synapse/push/emailpusher.py +334 -0
  437. synapse/push/httppusher.py +565 -0
  438. synapse/push/mailer.py +1009 -0
  439. synapse/push/presentable_names.py +216 -0
  440. synapse/push/push_tools.py +114 -0
  441. synapse/push/push_types.py +141 -0
  442. synapse/push/pusher.py +87 -0
  443. synapse/push/pusherpool.py +501 -0
  444. synapse/push/rulekinds.py +33 -0
  445. synapse/py.typed +0 -0
  446. synapse/replication/__init__.py +20 -0
  447. synapse/replication/http/__init__.py +68 -0
  448. synapse/replication/http/_base.py +469 -0
  449. synapse/replication/http/account_data.py +297 -0
  450. synapse/replication/http/deactivate_account.py +81 -0
  451. synapse/replication/http/delayed_events.py +62 -0
  452. synapse/replication/http/devices.py +254 -0
  453. synapse/replication/http/federation.py +334 -0
  454. synapse/replication/http/login.py +106 -0
  455. synapse/replication/http/membership.py +364 -0
  456. synapse/replication/http/presence.py +133 -0
  457. synapse/replication/http/push.py +156 -0
  458. synapse/replication/http/register.py +172 -0
  459. synapse/replication/http/send_events.py +182 -0
  460. synapse/replication/http/state.py +82 -0
  461. synapse/replication/http/streams.py +101 -0
  462. synapse/replication/tcp/__init__.py +56 -0
  463. synapse/replication/tcp/client.py +553 -0
  464. synapse/replication/tcp/commands.py +569 -0
  465. synapse/replication/tcp/context.py +41 -0
  466. synapse/replication/tcp/external_cache.py +156 -0
  467. synapse/replication/tcp/handler.py +922 -0
  468. synapse/replication/tcp/protocol.py +611 -0
  469. synapse/replication/tcp/redis.py +510 -0
  470. synapse/replication/tcp/resource.py +349 -0
  471. synapse/replication/tcp/streams/__init__.py +96 -0
  472. synapse/replication/tcp/streams/_base.py +765 -0
  473. synapse/replication/tcp/streams/events.py +287 -0
  474. synapse/replication/tcp/streams/federation.py +92 -0
  475. synapse/replication/tcp/streams/partial_state.py +80 -0
  476. synapse/res/providers.json +29 -0
  477. synapse/res/templates/_base.html +29 -0
  478. synapse/res/templates/account_previously_renewed.html +6 -0
  479. synapse/res/templates/account_renewed.html +6 -0
  480. synapse/res/templates/add_threepid.html +8 -0
  481. synapse/res/templates/add_threepid.txt +6 -0
  482. synapse/res/templates/add_threepid_failure.html +7 -0
  483. synapse/res/templates/add_threepid_success.html +6 -0
  484. synapse/res/templates/already_in_use.html +12 -0
  485. synapse/res/templates/already_in_use.txt +10 -0
  486. synapse/res/templates/auth_success.html +21 -0
  487. synapse/res/templates/invalid_token.html +6 -0
  488. synapse/res/templates/mail-Element.css +7 -0
  489. synapse/res/templates/mail-Vector.css +7 -0
  490. synapse/res/templates/mail-expiry.css +4 -0
  491. synapse/res/templates/mail.css +156 -0
  492. synapse/res/templates/notice_expiry.html +46 -0
  493. synapse/res/templates/notice_expiry.txt +7 -0
  494. synapse/res/templates/notif.html +51 -0
  495. synapse/res/templates/notif.txt +22 -0
  496. synapse/res/templates/notif_mail.html +59 -0
  497. synapse/res/templates/notif_mail.txt +10 -0
  498. synapse/res/templates/password_reset.html +10 -0
  499. synapse/res/templates/password_reset.txt +7 -0
  500. synapse/res/templates/password_reset_confirmation.html +15 -0
  501. synapse/res/templates/password_reset_failure.html +7 -0
  502. synapse/res/templates/password_reset_success.html +6 -0
  503. synapse/res/templates/recaptcha.html +42 -0
  504. synapse/res/templates/registration.html +12 -0
  505. synapse/res/templates/registration.txt +10 -0
  506. synapse/res/templates/registration_failure.html +6 -0
  507. synapse/res/templates/registration_success.html +6 -0
  508. synapse/res/templates/registration_token.html +18 -0
  509. synapse/res/templates/room.html +33 -0
  510. synapse/res/templates/room.txt +9 -0
  511. synapse/res/templates/sso.css +129 -0
  512. synapse/res/templates/sso_account_deactivated.html +25 -0
  513. synapse/res/templates/sso_auth_account_details.html +186 -0
  514. synapse/res/templates/sso_auth_account_details.js +116 -0
  515. synapse/res/templates/sso_auth_bad_user.html +26 -0
  516. synapse/res/templates/sso_auth_confirm.html +27 -0
  517. synapse/res/templates/sso_auth_success.html +26 -0
  518. synapse/res/templates/sso_error.html +71 -0
  519. synapse/res/templates/sso_footer.html +19 -0
  520. synapse/res/templates/sso_login_idp_picker.html +60 -0
  521. synapse/res/templates/sso_new_user_consent.html +30 -0
  522. synapse/res/templates/sso_partial_profile.html +19 -0
  523. synapse/res/templates/sso_redirect_confirm.html +39 -0
  524. synapse/res/templates/style.css +33 -0
  525. synapse/res/templates/terms.html +27 -0
  526. synapse/rest/__init__.py +197 -0
  527. synapse/rest/admin/__init__.py +395 -0
  528. synapse/rest/admin/_base.py +72 -0
  529. synapse/rest/admin/background_updates.py +171 -0
  530. synapse/rest/admin/devices.py +221 -0
  531. synapse/rest/admin/event_reports.py +173 -0
  532. synapse/rest/admin/events.py +69 -0
  533. synapse/rest/admin/experimental_features.py +137 -0
  534. synapse/rest/admin/federation.py +243 -0
  535. synapse/rest/admin/media.py +540 -0
  536. synapse/rest/admin/registration_tokens.py +358 -0
  537. synapse/rest/admin/rooms.py +1092 -0
  538. synapse/rest/admin/scheduled_tasks.py +70 -0
  539. synapse/rest/admin/server_notice_servlet.py +132 -0
  540. synapse/rest/admin/statistics.py +132 -0
  541. synapse/rest/admin/username_available.py +58 -0
  542. synapse/rest/admin/users.py +1634 -0
  543. synapse/rest/client/__init__.py +20 -0
  544. synapse/rest/client/_base.py +113 -0
  545. synapse/rest/client/account.py +937 -0
  546. synapse/rest/client/account_data.py +319 -0
  547. synapse/rest/client/account_validity.py +103 -0
  548. synapse/rest/client/appservice_ping.py +125 -0
  549. synapse/rest/client/auth.py +218 -0
  550. synapse/rest/client/auth_metadata.py +122 -0
  551. synapse/rest/client/capabilities.py +121 -0
  552. synapse/rest/client/delayed_events.py +165 -0
  553. synapse/rest/client/devices.py +587 -0
  554. synapse/rest/client/directory.py +211 -0
  555. synapse/rest/client/events.py +116 -0
  556. synapse/rest/client/filter.py +112 -0
  557. synapse/rest/client/initial_sync.py +65 -0
  558. synapse/rest/client/keys.py +678 -0
  559. synapse/rest/client/knock.py +104 -0
  560. synapse/rest/client/login.py +750 -0
  561. synapse/rest/client/login_token_request.py +127 -0
  562. synapse/rest/client/logout.py +93 -0
  563. synapse/rest/client/matrixrtc.py +52 -0
  564. synapse/rest/client/media.py +307 -0
  565. synapse/rest/client/mutual_rooms.py +145 -0
  566. synapse/rest/client/notifications.py +137 -0
  567. synapse/rest/client/openid.py +109 -0
  568. synapse/rest/client/password_policy.py +69 -0
  569. synapse/rest/client/presence.py +131 -0
  570. synapse/rest/client/profile.py +291 -0
  571. synapse/rest/client/push_rule.py +331 -0
  572. synapse/rest/client/pusher.py +181 -0
  573. synapse/rest/client/read_marker.py +104 -0
  574. synapse/rest/client/receipts.py +165 -0
  575. synapse/rest/client/register.py +1072 -0
  576. synapse/rest/client/relations.py +138 -0
  577. synapse/rest/client/rendezvous.py +76 -0
  578. synapse/rest/client/reporting.py +207 -0
  579. synapse/rest/client/room.py +1763 -0
  580. synapse/rest/client/room_keys.py +426 -0
  581. synapse/rest/client/room_upgrade_rest_servlet.py +112 -0
  582. synapse/rest/client/sendtodevice.py +85 -0
  583. synapse/rest/client/sync.py +1131 -0
  584. synapse/rest/client/tags.py +129 -0
  585. synapse/rest/client/thirdparty.py +130 -0
  586. synapse/rest/client/thread_subscriptions.py +247 -0
  587. synapse/rest/client/tokenrefresh.py +52 -0
  588. synapse/rest/client/transactions.py +150 -0
  589. synapse/rest/client/user_directory.py +99 -0
  590. synapse/rest/client/versions.py +193 -0
  591. synapse/rest/client/voip.py +88 -0
  592. synapse/rest/consent/__init__.py +0 -0
  593. synapse/rest/consent/consent_resource.py +210 -0
  594. synapse/rest/health.py +38 -0
  595. synapse/rest/key/__init__.py +20 -0
  596. synapse/rest/key/v2/__init__.py +40 -0
  597. synapse/rest/key/v2/local_key_resource.py +125 -0
  598. synapse/rest/key/v2/remote_key_resource.py +302 -0
  599. synapse/rest/media/__init__.py +0 -0
  600. synapse/rest/media/config_resource.py +53 -0
  601. synapse/rest/media/create_resource.py +90 -0
  602. synapse/rest/media/download_resource.py +110 -0
  603. synapse/rest/media/media_repository_resource.py +113 -0
  604. synapse/rest/media/preview_url_resource.py +77 -0
  605. synapse/rest/media/thumbnail_resource.py +142 -0
  606. synapse/rest/media/upload_resource.py +187 -0
  607. synapse/rest/media/v1/__init__.py +39 -0
  608. synapse/rest/media/v1/_base.py +23 -0
  609. synapse/rest/media/v1/media_storage.py +23 -0
  610. synapse/rest/media/v1/storage_provider.py +23 -0
  611. synapse/rest/synapse/__init__.py +20 -0
  612. synapse/rest/synapse/client/__init__.py +93 -0
  613. synapse/rest/synapse/client/federation_whitelist.py +66 -0
  614. synapse/rest/synapse/client/jwks.py +77 -0
  615. synapse/rest/synapse/client/new_user_consent.py +115 -0
  616. synapse/rest/synapse/client/oidc/__init__.py +45 -0
  617. synapse/rest/synapse/client/oidc/backchannel_logout_resource.py +42 -0
  618. synapse/rest/synapse/client/oidc/callback_resource.py +48 -0
  619. synapse/rest/synapse/client/password_reset.py +129 -0
  620. synapse/rest/synapse/client/pick_idp.py +107 -0
  621. synapse/rest/synapse/client/pick_username.py +153 -0
  622. synapse/rest/synapse/client/rendezvous.py +58 -0
  623. synapse/rest/synapse/client/saml2/__init__.py +42 -0
  624. synapse/rest/synapse/client/saml2/metadata_resource.py +46 -0
  625. synapse/rest/synapse/client/saml2/response_resource.py +52 -0
  626. synapse/rest/synapse/client/sso_register.py +56 -0
  627. synapse/rest/synapse/client/unsubscribe.py +88 -0
  628. synapse/rest/synapse/mas/__init__.py +71 -0
  629. synapse/rest/synapse/mas/_base.py +55 -0
  630. synapse/rest/synapse/mas/devices.py +239 -0
  631. synapse/rest/synapse/mas/users.py +469 -0
  632. synapse/rest/well_known.py +148 -0
  633. synapse/server.py +1279 -0
  634. synapse/server_notices/__init__.py +0 -0
  635. synapse/server_notices/consent_server_notices.py +136 -0
  636. synapse/server_notices/resource_limits_server_notices.py +215 -0
  637. synapse/server_notices/server_notices_manager.py +388 -0
  638. synapse/server_notices/server_notices_sender.py +67 -0
  639. synapse/server_notices/worker_server_notices_sender.py +46 -0
  640. synapse/spam_checker_api/__init__.py +31 -0
  641. synapse/state/__init__.py +1023 -0
  642. synapse/state/v1.py +369 -0
  643. synapse/state/v2.py +985 -0
  644. synapse/static/client/login/index.html +47 -0
  645. synapse/static/client/login/js/jquery-3.4.1.min.js +2 -0
  646. synapse/static/client/login/js/login.js +291 -0
  647. synapse/static/client/login/spinner.gif +0 -0
  648. synapse/static/client/login/style.css +79 -0
  649. synapse/static/index.html +63 -0
  650. synapse/storage/__init__.py +43 -0
  651. synapse/storage/_base.py +245 -0
  652. synapse/storage/admin_client_config.py +25 -0
  653. synapse/storage/background_updates.py +1189 -0
  654. synapse/storage/controllers/__init__.py +57 -0
  655. synapse/storage/controllers/persist_events.py +1237 -0
  656. synapse/storage/controllers/purge_events.py +456 -0
  657. synapse/storage/controllers/state.py +950 -0
  658. synapse/storage/controllers/stats.py +119 -0
  659. synapse/storage/database.py +2720 -0
  660. synapse/storage/databases/__init__.py +175 -0
  661. synapse/storage/databases/main/__init__.py +420 -0
  662. synapse/storage/databases/main/account_data.py +1073 -0
  663. synapse/storage/databases/main/appservice.py +473 -0
  664. synapse/storage/databases/main/cache.py +912 -0
  665. synapse/storage/databases/main/censor_events.py +226 -0
  666. synapse/storage/databases/main/client_ips.py +816 -0
  667. synapse/storage/databases/main/delayed_events.py +577 -0
  668. synapse/storage/databases/main/deviceinbox.py +1272 -0
  669. synapse/storage/databases/main/devices.py +2579 -0
  670. synapse/storage/databases/main/directory.py +212 -0
  671. synapse/storage/databases/main/e2e_room_keys.py +689 -0
  672. synapse/storage/databases/main/end_to_end_keys.py +1894 -0
  673. synapse/storage/databases/main/event_federation.py +2511 -0
  674. synapse/storage/databases/main/event_push_actions.py +1936 -0
  675. synapse/storage/databases/main/events.py +3765 -0
  676. synapse/storage/databases/main/events_bg_updates.py +2910 -0
  677. synapse/storage/databases/main/events_forward_extremities.py +126 -0
  678. synapse/storage/databases/main/events_worker.py +2787 -0
  679. synapse/storage/databases/main/experimental_features.py +130 -0
  680. synapse/storage/databases/main/filtering.py +231 -0
  681. synapse/storage/databases/main/keys.py +291 -0
  682. synapse/storage/databases/main/lock.py +554 -0
  683. synapse/storage/databases/main/media_repository.py +1068 -0
  684. synapse/storage/databases/main/metrics.py +461 -0
  685. synapse/storage/databases/main/monthly_active_users.py +443 -0
  686. synapse/storage/databases/main/openid.py +60 -0
  687. synapse/storage/databases/main/presence.py +509 -0
  688. synapse/storage/databases/main/profile.py +539 -0
  689. synapse/storage/databases/main/purge_events.py +521 -0
  690. synapse/storage/databases/main/push_rule.py +970 -0
  691. synapse/storage/databases/main/pusher.py +793 -0
  692. synapse/storage/databases/main/receipts.py +1341 -0
  693. synapse/storage/databases/main/registration.py +3071 -0
  694. synapse/storage/databases/main/rejections.py +37 -0
  695. synapse/storage/databases/main/relations.py +1116 -0
  696. synapse/storage/databases/main/room.py +2779 -0
  697. synapse/storage/databases/main/roommember.py +2132 -0
  698. synapse/storage/databases/main/search.py +939 -0
  699. synapse/storage/databases/main/session.py +152 -0
  700. synapse/storage/databases/main/signatures.py +94 -0
  701. synapse/storage/databases/main/sliding_sync.py +842 -0
  702. synapse/storage/databases/main/state.py +1002 -0
  703. synapse/storage/databases/main/state_deltas.py +360 -0
  704. synapse/storage/databases/main/stats.py +789 -0
  705. synapse/storage/databases/main/stream.py +2589 -0
  706. synapse/storage/databases/main/tags.py +360 -0
  707. synapse/storage/databases/main/task_scheduler.py +225 -0
  708. synapse/storage/databases/main/thread_subscriptions.py +589 -0
  709. synapse/storage/databases/main/transactions.py +676 -0
  710. synapse/storage/databases/main/ui_auth.py +420 -0
  711. synapse/storage/databases/main/user_directory.py +1330 -0
  712. synapse/storage/databases/main/user_erasure_store.py +117 -0
  713. synapse/storage/databases/state/__init__.py +22 -0
  714. synapse/storage/databases/state/bg_updates.py +497 -0
  715. synapse/storage/databases/state/deletion.py +557 -0
  716. synapse/storage/databases/state/store.py +948 -0
  717. synapse/storage/engines/__init__.py +70 -0
  718. synapse/storage/engines/_base.py +154 -0
  719. synapse/storage/engines/postgres.py +261 -0
  720. synapse/storage/engines/sqlite.py +199 -0
  721. synapse/storage/invite_rule.py +152 -0
  722. synapse/storage/keys.py +40 -0
  723. synapse/storage/prepare_database.py +730 -0
  724. synapse/storage/push_rule.py +28 -0
  725. synapse/storage/roommember.py +88 -0
  726. synapse/storage/schema/README.md +4 -0
  727. synapse/storage/schema/__init__.py +186 -0
  728. synapse/storage/schema/common/delta/25/00background_updates.sql +40 -0
  729. synapse/storage/schema/common/delta/35/00background_updates_add_col.sql +36 -0
  730. synapse/storage/schema/common/delta/58/00background_update_ordering.sql +38 -0
  731. synapse/storage/schema/common/full_schemas/72/full.sql.postgres +8 -0
  732. synapse/storage/schema/common/full_schemas/72/full.sql.sqlite +6 -0
  733. synapse/storage/schema/common/schema_version.sql +60 -0
  734. synapse/storage/schema/main/delta/12/v12.sql +82 -0
  735. synapse/storage/schema/main/delta/13/v13.sql +38 -0
  736. synapse/storage/schema/main/delta/14/v14.sql +42 -0
  737. synapse/storage/schema/main/delta/15/appservice_txns.sql +50 -0
  738. synapse/storage/schema/main/delta/15/presence_indices.sql +2 -0
  739. synapse/storage/schema/main/delta/15/v15.sql +24 -0
  740. synapse/storage/schema/main/delta/16/events_order_index.sql +4 -0
  741. synapse/storage/schema/main/delta/16/remote_media_cache_index.sql +2 -0
  742. synapse/storage/schema/main/delta/16/remove_duplicates.sql +9 -0
  743. synapse/storage/schema/main/delta/16/room_alias_index.sql +3 -0
  744. synapse/storage/schema/main/delta/16/unique_constraints.sql +72 -0
  745. synapse/storage/schema/main/delta/16/users.sql +56 -0
  746. synapse/storage/schema/main/delta/17/drop_indexes.sql +37 -0
  747. synapse/storage/schema/main/delta/17/server_keys.sql +43 -0
  748. synapse/storage/schema/main/delta/17/user_threepids.sql +9 -0
  749. synapse/storage/schema/main/delta/18/server_keys_bigger_ints.sql +51 -0
  750. synapse/storage/schema/main/delta/19/event_index.sql +38 -0
  751. synapse/storage/schema/main/delta/20/dummy.sql +1 -0
  752. synapse/storage/schema/main/delta/20/pushers.py +93 -0
  753. synapse/storage/schema/main/delta/21/end_to_end_keys.sql +53 -0
  754. synapse/storage/schema/main/delta/21/receipts.sql +57 -0
  755. synapse/storage/schema/main/delta/22/receipts_index.sql +41 -0
  756. synapse/storage/schema/main/delta/22/user_threepids_unique.sql +19 -0
  757. synapse/storage/schema/main/delta/24/stats_reporting.sql +37 -0
  758. synapse/storage/schema/main/delta/25/fts.py +81 -0
  759. synapse/storage/schema/main/delta/25/guest_access.sql +44 -0
  760. synapse/storage/schema/main/delta/25/history_visibility.sql +44 -0
  761. synapse/storage/schema/main/delta/25/tags.sql +57 -0
  762. synapse/storage/schema/main/delta/26/account_data.sql +36 -0
  763. synapse/storage/schema/main/delta/27/account_data.sql +55 -0
  764. synapse/storage/schema/main/delta/27/forgotten_memberships.sql +45 -0
  765. synapse/storage/schema/main/delta/27/ts.py +61 -0
  766. synapse/storage/schema/main/delta/28/event_push_actions.sql +46 -0
  767. synapse/storage/schema/main/delta/28/events_room_stream.sql +39 -0
  768. synapse/storage/schema/main/delta/28/public_roms_index.sql +39 -0
  769. synapse/storage/schema/main/delta/28/receipts_user_id_index.sql +41 -0
  770. synapse/storage/schema/main/delta/28/upgrade_times.sql +40 -0
  771. synapse/storage/schema/main/delta/28/users_is_guest.sql +41 -0
  772. synapse/storage/schema/main/delta/29/push_actions.sql +54 -0
  773. synapse/storage/schema/main/delta/30/alias_creator.sql +35 -0
  774. synapse/storage/schema/main/delta/30/as_users.py +82 -0
  775. synapse/storage/schema/main/delta/30/deleted_pushers.sql +44 -0
  776. synapse/storage/schema/main/delta/30/presence_stream.sql +49 -0
  777. synapse/storage/schema/main/delta/30/public_rooms.sql +42 -0
  778. synapse/storage/schema/main/delta/30/push_rule_stream.sql +57 -0
  779. synapse/storage/schema/main/delta/30/threepid_guest_access_tokens.sql +43 -0
  780. synapse/storage/schema/main/delta/31/invites.sql +61 -0
  781. synapse/storage/schema/main/delta/31/local_media_repository_url_cache.sql +46 -0
  782. synapse/storage/schema/main/delta/31/pushers_0.py +92 -0
  783. synapse/storage/schema/main/delta/31/pushers_index.sql +41 -0
  784. synapse/storage/schema/main/delta/31/search_update.py +65 -0
  785. synapse/storage/schema/main/delta/32/events.sql +35 -0
  786. synapse/storage/schema/main/delta/32/openid.sql +9 -0
  787. synapse/storage/schema/main/delta/32/pusher_throttle.sql +42 -0
  788. synapse/storage/schema/main/delta/32/remove_indices.sql +52 -0
  789. synapse/storage/schema/main/delta/32/reports.sql +44 -0
  790. synapse/storage/schema/main/delta/33/access_tokens_device_index.sql +36 -0
  791. synapse/storage/schema/main/delta/33/devices.sql +40 -0
  792. synapse/storage/schema/main/delta/33/devices_for_e2e_keys.sql +38 -0
  793. synapse/storage/schema/main/delta/33/devices_for_e2e_keys_clear_unknown_device.sql +39 -0
  794. synapse/storage/schema/main/delta/33/event_fields.py +61 -0
  795. synapse/storage/schema/main/delta/33/remote_media_ts.py +43 -0
  796. synapse/storage/schema/main/delta/33/user_ips_index.sql +36 -0
  797. synapse/storage/schema/main/delta/34/appservice_stream.sql +42 -0
  798. synapse/storage/schema/main/delta/34/cache_stream.py +50 -0
  799. synapse/storage/schema/main/delta/34/device_inbox.sql +43 -0
  800. synapse/storage/schema/main/delta/34/push_display_name_rename.sql +39 -0
  801. synapse/storage/schema/main/delta/34/received_txn_purge.py +36 -0
  802. synapse/storage/schema/main/delta/35/contains_url.sql +36 -0
  803. synapse/storage/schema/main/delta/35/device_outbox.sql +58 -0
  804. synapse/storage/schema/main/delta/35/device_stream_id.sql +40 -0
  805. synapse/storage/schema/main/delta/35/event_push_actions_index.sql +36 -0
  806. synapse/storage/schema/main/delta/35/public_room_list_change_stream.sql +52 -0
  807. synapse/storage/schema/main/delta/35/stream_order_to_extrem.sql +56 -0
  808. synapse/storage/schema/main/delta/36/readd_public_rooms.sql +45 -0
  809. synapse/storage/schema/main/delta/37/remove_auth_idx.py +89 -0
  810. synapse/storage/schema/main/delta/37/user_threepids.sql +71 -0
  811. synapse/storage/schema/main/delta/38/postgres_fts_gist.sql +38 -0
  812. synapse/storage/schema/main/delta/39/appservice_room_list.sql +48 -0
  813. synapse/storage/schema/main/delta/39/device_federation_stream_idx.sql +35 -0
  814. synapse/storage/schema/main/delta/39/event_push_index.sql +36 -0
  815. synapse/storage/schema/main/delta/39/federation_out_position.sql +41 -0
  816. synapse/storage/schema/main/delta/39/membership_profile.sql +39 -0
  817. synapse/storage/schema/main/delta/40/current_state_idx.sql +36 -0
  818. synapse/storage/schema/main/delta/40/device_inbox.sql +40 -0
  819. synapse/storage/schema/main/delta/40/device_list_streams.sql +79 -0
  820. synapse/storage/schema/main/delta/40/event_push_summary.sql +57 -0
  821. synapse/storage/schema/main/delta/40/pushers.sql +58 -0
  822. synapse/storage/schema/main/delta/41/device_list_stream_idx.sql +36 -0
  823. synapse/storage/schema/main/delta/41/device_outbound_index.sql +35 -0
  824. synapse/storage/schema/main/delta/41/event_search_event_id_idx.sql +36 -0
  825. synapse/storage/schema/main/delta/41/ratelimit.sql +41 -0
  826. synapse/storage/schema/main/delta/42/current_state_delta.sql +48 -0
  827. synapse/storage/schema/main/delta/42/device_list_last_id.sql +52 -0
  828. synapse/storage/schema/main/delta/42/event_auth_state_only.sql +36 -0
  829. synapse/storage/schema/main/delta/42/user_dir.py +88 -0
  830. synapse/storage/schema/main/delta/43/blocked_rooms.sql +40 -0
  831. synapse/storage/schema/main/delta/43/quarantine_media.sql +36 -0
  832. synapse/storage/schema/main/delta/43/url_cache.sql +35 -0
  833. synapse/storage/schema/main/delta/43/user_share.sql +52 -0
  834. synapse/storage/schema/main/delta/44/expire_url_cache.sql +60 -0
  835. synapse/storage/schema/main/delta/45/group_server.sql +186 -0
  836. synapse/storage/schema/main/delta/45/profile_cache.sql +47 -0
  837. synapse/storage/schema/main/delta/46/drop_refresh_tokens.sql +36 -0
  838. synapse/storage/schema/main/delta/46/drop_unique_deleted_pushers.sql +54 -0
  839. synapse/storage/schema/main/delta/46/group_server.sql +51 -0
  840. synapse/storage/schema/main/delta/46/local_media_repository_url_idx.sql +43 -0
  841. synapse/storage/schema/main/delta/46/user_dir_null_room_ids.sql +54 -0
  842. synapse/storage/schema/main/delta/46/user_dir_typos.sql +43 -0
  843. synapse/storage/schema/main/delta/47/last_access_media.sql +35 -0
  844. synapse/storage/schema/main/delta/47/postgres_fts_gin.sql +36 -0
  845. synapse/storage/schema/main/delta/47/push_actions_staging.sql +47 -0
  846. synapse/storage/schema/main/delta/48/add_user_consent.sql +37 -0
  847. synapse/storage/schema/main/delta/48/add_user_ips_last_seen_index.sql +36 -0
  848. synapse/storage/schema/main/delta/48/deactivated_users.sql +44 -0
  849. synapse/storage/schema/main/delta/48/group_unique_indexes.py +67 -0
  850. synapse/storage/schema/main/delta/48/groups_joinable.sql +41 -0
  851. synapse/storage/schema/main/delta/49/add_user_consent_server_notice_sent.sql +39 -0
  852. synapse/storage/schema/main/delta/49/add_user_daily_visits.sql +40 -0
  853. synapse/storage/schema/main/delta/49/add_user_ips_last_seen_only_index.sql +36 -0
  854. synapse/storage/schema/main/delta/50/add_creation_ts_users_index.sql +38 -0
  855. synapse/storage/schema/main/delta/50/erasure_store.sql +40 -0
  856. synapse/storage/schema/main/delta/50/make_event_content_nullable.py +102 -0
  857. synapse/storage/schema/main/delta/51/e2e_room_keys.sql +58 -0
  858. synapse/storage/schema/main/delta/51/monthly_active_users.sql +46 -0
  859. synapse/storage/schema/main/delta/52/add_event_to_state_group_index.sql +38 -0
  860. synapse/storage/schema/main/delta/52/device_list_streams_unique_idx.sql +55 -0
  861. synapse/storage/schema/main/delta/52/e2e_room_keys.sql +72 -0
  862. synapse/storage/schema/main/delta/53/add_user_type_to_users.sql +38 -0
  863. synapse/storage/schema/main/delta/53/drop_sent_transactions.sql +35 -0
  864. synapse/storage/schema/main/delta/53/event_format_version.sql +35 -0
  865. synapse/storage/schema/main/delta/53/user_dir_populate.sql +49 -0
  866. synapse/storage/schema/main/delta/53/user_ips_index.sql +49 -0
  867. synapse/storage/schema/main/delta/53/user_share.sql +63 -0
  868. synapse/storage/schema/main/delta/53/user_threepid_id.sql +48 -0
  869. synapse/storage/schema/main/delta/53/users_in_public_rooms.sql +47 -0
  870. synapse/storage/schema/main/delta/54/account_validity_with_renewal.sql +49 -0
  871. synapse/storage/schema/main/delta/54/add_validity_to_server_keys.sql +42 -0
  872. synapse/storage/schema/main/delta/54/delete_forward_extremities.sql +42 -0
  873. synapse/storage/schema/main/delta/54/drop_legacy_tables.sql +49 -0
  874. synapse/storage/schema/main/delta/54/drop_presence_list.sql +35 -0
  875. synapse/storage/schema/main/delta/54/relations.sql +46 -0
  876. synapse/storage/schema/main/delta/54/stats.sql +99 -0
  877. synapse/storage/schema/main/delta/54/stats2.sql +47 -0
  878. synapse/storage/schema/main/delta/55/access_token_expiry.sql +37 -0
  879. synapse/storage/schema/main/delta/55/track_threepid_validations.sql +50 -0
  880. synapse/storage/schema/main/delta/55/users_alter_deactivated.sql +38 -0
  881. synapse/storage/schema/main/delta/56/add_spans_to_device_lists.sql +39 -0
  882. synapse/storage/schema/main/delta/56/current_state_events_membership.sql +41 -0
  883. synapse/storage/schema/main/delta/56/current_state_events_membership_mk2.sql +43 -0
  884. synapse/storage/schema/main/delta/56/delete_keys_from_deleted_backups.sql +44 -0
  885. synapse/storage/schema/main/delta/56/destinations_failure_ts.sql +44 -0
  886. synapse/storage/schema/main/delta/56/destinations_retry_interval_type.sql.postgres +18 -0
  887. synapse/storage/schema/main/delta/56/device_stream_id_insert.sql +39 -0
  888. synapse/storage/schema/main/delta/56/devices_last_seen.sql +43 -0
  889. synapse/storage/schema/main/delta/56/drop_unused_event_tables.sql +39 -0
  890. synapse/storage/schema/main/delta/56/event_expiry.sql +40 -0
  891. synapse/storage/schema/main/delta/56/event_labels.sql +49 -0
  892. synapse/storage/schema/main/delta/56/event_labels_background_update.sql +36 -0
  893. synapse/storage/schema/main/delta/56/fix_room_keys_index.sql +37 -0
  894. synapse/storage/schema/main/delta/56/hidden_devices.sql +37 -0
  895. synapse/storage/schema/main/delta/56/hidden_devices_fix.sql.sqlite +42 -0
  896. synapse/storage/schema/main/delta/56/nuke_empty_communities_from_db.sql +48 -0
  897. synapse/storage/schema/main/delta/56/public_room_list_idx.sql +35 -0
  898. synapse/storage/schema/main/delta/56/redaction_censor.sql +35 -0
  899. synapse/storage/schema/main/delta/56/redaction_censor2.sql +41 -0
  900. synapse/storage/schema/main/delta/56/redaction_censor3_fix_update.sql.postgres +25 -0
  901. synapse/storage/schema/main/delta/56/redaction_censor4.sql +35 -0
  902. synapse/storage/schema/main/delta/56/remove_tombstoned_rooms_from_directory.sql +38 -0
  903. synapse/storage/schema/main/delta/56/room_key_etag.sql +36 -0
  904. synapse/storage/schema/main/delta/56/room_membership_idx.sql +37 -0
  905. synapse/storage/schema/main/delta/56/room_retention.sql +52 -0
  906. synapse/storage/schema/main/delta/56/signing_keys.sql +75 -0
  907. synapse/storage/schema/main/delta/56/signing_keys_nonunique_signatures.sql +41 -0
  908. synapse/storage/schema/main/delta/56/stats_separated.sql +175 -0
  909. synapse/storage/schema/main/delta/56/unique_user_filter_index.py +46 -0
  910. synapse/storage/schema/main/delta/56/user_external_ids.sql +43 -0
  911. synapse/storage/schema/main/delta/56/users_in_public_rooms_idx.sql +36 -0
  912. synapse/storage/schema/main/delta/57/delete_old_current_state_events.sql +41 -0
  913. synapse/storage/schema/main/delta/57/device_list_remote_cache_stale.sql +44 -0
  914. synapse/storage/schema/main/delta/57/local_current_membership.py +111 -0
  915. synapse/storage/schema/main/delta/57/remove_sent_outbound_pokes.sql +40 -0
  916. synapse/storage/schema/main/delta/57/rooms_version_column.sql +43 -0
  917. synapse/storage/schema/main/delta/57/rooms_version_column_2.sql.postgres +35 -0
  918. synapse/storage/schema/main/delta/57/rooms_version_column_2.sql.sqlite +22 -0
  919. synapse/storage/schema/main/delta/57/rooms_version_column_3.sql.postgres +39 -0
  920. synapse/storage/schema/main/delta/57/rooms_version_column_3.sql.sqlite +23 -0
  921. synapse/storage/schema/main/delta/58/02remove_dup_outbound_pokes.sql +41 -0
  922. synapse/storage/schema/main/delta/58/03persist_ui_auth.sql +55 -0
  923. synapse/storage/schema/main/delta/58/05cache_instance.sql.postgres +30 -0
  924. synapse/storage/schema/main/delta/58/06dlols_unique_idx.py +83 -0
  925. synapse/storage/schema/main/delta/58/07add_method_to_thumbnail_constraint.sql.postgres +33 -0
  926. synapse/storage/schema/main/delta/58/07add_method_to_thumbnail_constraint.sql.sqlite +44 -0
  927. synapse/storage/schema/main/delta/58/07persist_ui_auth_ips.sql +44 -0
  928. synapse/storage/schema/main/delta/58/08_media_safe_from_quarantine.sql.postgres +18 -0
  929. synapse/storage/schema/main/delta/58/08_media_safe_from_quarantine.sql.sqlite +18 -0
  930. synapse/storage/schema/main/delta/58/09shadow_ban.sql +37 -0
  931. synapse/storage/schema/main/delta/58/10_pushrules_enabled_delete_obsolete.sql +47 -0
  932. synapse/storage/schema/main/delta/58/10drop_local_rejections_stream.sql +41 -0
  933. synapse/storage/schema/main/delta/58/10federation_pos_instance_name.sql +41 -0
  934. synapse/storage/schema/main/delta/58/11dehydration.sql +39 -0
  935. synapse/storage/schema/main/delta/58/11fallback.sql +43 -0
  936. synapse/storage/schema/main/delta/58/11user_id_seq.py +38 -0
  937. synapse/storage/schema/main/delta/58/12room_stats.sql +51 -0
  938. synapse/storage/schema/main/delta/58/13remove_presence_allow_inbound.sql +36 -0
  939. synapse/storage/schema/main/delta/58/14events_instance_name.sql +35 -0
  940. synapse/storage/schema/main/delta/58/14events_instance_name.sql.postgres +28 -0
  941. synapse/storage/schema/main/delta/58/15_catchup_destination_rooms.sql +61 -0
  942. synapse/storage/schema/main/delta/58/15unread_count.sql +45 -0
  943. synapse/storage/schema/main/delta/58/16populate_stats_process_rooms_fix.sql +41 -0
  944. synapse/storage/schema/main/delta/58/17_catchup_last_successful.sql +40 -0
  945. synapse/storage/schema/main/delta/58/18stream_positions.sql +41 -0
  946. synapse/storage/schema/main/delta/58/19instance_map.sql.postgres +25 -0
  947. synapse/storage/schema/main/delta/58/19txn_id.sql +59 -0
  948. synapse/storage/schema/main/delta/58/20instance_name_event_tables.sql +36 -0
  949. synapse/storage/schema/main/delta/58/20user_daily_visits.sql +37 -0
  950. synapse/storage/schema/main/delta/58/21as_device_stream.sql +36 -0
  951. synapse/storage/schema/main/delta/58/21drop_device_max_stream_id.sql +1 -0
  952. synapse/storage/schema/main/delta/58/22puppet_token.sql +36 -0
  953. synapse/storage/schema/main/delta/58/22users_have_local_media.sql +2 -0
  954. synapse/storage/schema/main/delta/58/23e2e_cross_signing_keys_idx.sql +36 -0
  955. synapse/storage/schema/main/delta/58/24drop_event_json_index.sql +38 -0
  956. synapse/storage/schema/main/delta/58/25user_external_ids_user_id_idx.sql +36 -0
  957. synapse/storage/schema/main/delta/58/26access_token_last_validated.sql +37 -0
  958. synapse/storage/schema/main/delta/58/27local_invites.sql +37 -0
  959. synapse/storage/schema/main/delta/58/28drop_last_used_column.sql.postgres +16 -0
  960. synapse/storage/schema/main/delta/58/28drop_last_used_column.sql.sqlite +62 -0
  961. synapse/storage/schema/main/delta/59/01ignored_user.py +85 -0
  962. synapse/storage/schema/main/delta/59/02shard_send_to_device.sql +37 -0
  963. synapse/storage/schema/main/delta/59/03shard_send_to_device_sequence.sql.postgres +25 -0
  964. synapse/storage/schema/main/delta/59/04_event_auth_chains.sql +71 -0
  965. synapse/storage/schema/main/delta/59/04_event_auth_chains.sql.postgres +16 -0
  966. synapse/storage/schema/main/delta/59/04drop_account_data.sql +36 -0
  967. synapse/storage/schema/main/delta/59/05cache_invalidation.sql +36 -0
  968. synapse/storage/schema/main/delta/59/06chain_cover_index.sql +36 -0
  969. synapse/storage/schema/main/delta/59/06shard_account_data.sql +39 -0
  970. synapse/storage/schema/main/delta/59/06shard_account_data.sql.postgres +32 -0
  971. synapse/storage/schema/main/delta/59/07shard_account_data_fix.sql +37 -0
  972. synapse/storage/schema/main/delta/59/08delete_pushers_for_deactivated_accounts.sql +39 -0
  973. synapse/storage/schema/main/delta/59/08delete_stale_pushers.sql +39 -0
  974. synapse/storage/schema/main/delta/59/09rejected_events_metadata.sql +45 -0
  975. synapse/storage/schema/main/delta/59/10delete_purged_chain_cover.sql +36 -0
  976. synapse/storage/schema/main/delta/59/11add_knock_members_to_stats.sql +39 -0
  977. synapse/storage/schema/main/delta/59/11drop_thumbnail_constraint.sql.postgres +22 -0
  978. synapse/storage/schema/main/delta/59/12account_validity_token_used_ts_ms.sql +37 -0
  979. synapse/storage/schema/main/delta/59/12presence_stream_instance.sql +37 -0
  980. synapse/storage/schema/main/delta/59/12presence_stream_instance_seq.sql.postgres +20 -0
  981. synapse/storage/schema/main/delta/59/13users_to_send_full_presence_to.sql +53 -0
  982. synapse/storage/schema/main/delta/59/14refresh_tokens.sql +53 -0
  983. synapse/storage/schema/main/delta/59/15locks.sql +56 -0
  984. synapse/storage/schema/main/delta/59/16federation_inbound_staging.sql +51 -0
  985. synapse/storage/schema/main/delta/60/01recreate_stream_ordering.sql.postgres +45 -0
  986. synapse/storage/schema/main/delta/60/02change_stream_ordering_columns.sql.postgres +30 -0
  987. synapse/storage/schema/main/delta/61/01change_appservices_txns.sql.postgres +23 -0
  988. synapse/storage/schema/main/delta/61/01insertion_event_lookups.sql +68 -0
  989. synapse/storage/schema/main/delta/61/02drop_redundant_room_depth_index.sql +37 -0
  990. synapse/storage/schema/main/delta/61/03recreate_min_depth.py +74 -0
  991. synapse/storage/schema/main/delta/62/01insertion_event_extremities.sql +43 -0
  992. synapse/storage/schema/main/delta/63/01create_registration_tokens.sql +42 -0
  993. synapse/storage/schema/main/delta/63/02delete_unlinked_email_pushers.sql +39 -0
  994. synapse/storage/schema/main/delta/63/02populate-rooms-creator.sql +36 -0
  995. synapse/storage/schema/main/delta/63/03session_store.sql +42 -0
  996. synapse/storage/schema/main/delta/63/04add_presence_stream_not_offline_index.sql +37 -0
  997. synapse/storage/schema/main/delta/64/01msc2716_chunk_to_batch_rename.sql.postgres +23 -0
  998. synapse/storage/schema/main/delta/64/01msc2716_chunk_to_batch_rename.sql.sqlite +37 -0
  999. synapse/storage/schema/main/delta/65/01msc2716_insertion_event_edges.sql +38 -0
  1000. synapse/storage/schema/main/delta/65/03remove_hidden_devices_from_device_inbox.sql +41 -0
  1001. synapse/storage/schema/main/delta/65/04_local_group_updates.sql +37 -0
  1002. synapse/storage/schema/main/delta/65/05_remove_room_stats_historical_and_user_stats_historical.sql +38 -0
  1003. synapse/storage/schema/main/delta/65/06remove_deleted_devices_from_device_inbox.sql +53 -0
  1004. synapse/storage/schema/main/delta/65/07_arbitrary_relations.sql +37 -0
  1005. synapse/storage/schema/main/delta/65/08_device_inbox_background_updates.sql +37 -0
  1006. synapse/storage/schema/main/delta/65/10_expirable_refresh_tokens.sql +47 -0
  1007. synapse/storage/schema/main/delta/65/11_devices_auth_provider_session.sql +46 -0
  1008. synapse/storage/schema/main/delta/67/01drop_public_room_list_stream.sql +37 -0
  1009. synapse/storage/schema/main/delta/68/01event_columns.sql +45 -0
  1010. synapse/storage/schema/main/delta/68/02_msc2409_add_device_id_appservice_stream_type.sql +40 -0
  1011. synapse/storage/schema/main/delta/68/03_delete_account_data_for_deactivated_accounts.sql +39 -0
  1012. synapse/storage/schema/main/delta/68/04_refresh_tokens_index_next_token_id.sql +47 -0
  1013. synapse/storage/schema/main/delta/68/04partial_state_rooms.sql +60 -0
  1014. synapse/storage/schema/main/delta/68/05_delete_non_strings_from_event_search.sql.sqlite +22 -0
  1015. synapse/storage/schema/main/delta/68/05partial_state_rooms_triggers.py +80 -0
  1016. synapse/storage/schema/main/delta/68/06_msc3202_add_device_list_appservice_stream_type.sql +42 -0
  1017. synapse/storage/schema/main/delta/69/01as_txn_seq.py +54 -0
  1018. synapse/storage/schema/main/delta/69/01device_list_oubound_by_room.sql +57 -0
  1019. synapse/storage/schema/main/delta/69/02cache_invalidation_index.sql +37 -0
  1020. synapse/storage/schema/main/delta/70/01clean_table_purged_rooms.sql +39 -0
  1021. synapse/storage/schema/main/delta/71/01rebuild_event_edges.sql.postgres +43 -0
  1022. synapse/storage/schema/main/delta/71/01rebuild_event_edges.sql.sqlite +47 -0
  1023. synapse/storage/schema/main/delta/71/01remove_noop_background_updates.sql +80 -0
  1024. synapse/storage/schema/main/delta/71/02event_push_summary_unique.sql +37 -0
  1025. synapse/storage/schema/main/delta/72/01add_room_type_to_state_stats.sql +38 -0
  1026. synapse/storage/schema/main/delta/72/01event_push_summary_receipt.sql +54 -0
  1027. synapse/storage/schema/main/delta/72/02event_push_actions_index.sql +38 -0
  1028. synapse/storage/schema/main/delta/72/03bg_populate_events_columns.py +57 -0
  1029. synapse/storage/schema/main/delta/72/03drop_event_reference_hashes.sql +36 -0
  1030. synapse/storage/schema/main/delta/72/03remove_groups.sql +50 -0
  1031. synapse/storage/schema/main/delta/72/04drop_column_application_services_state_last_txn.sql.postgres +17 -0
  1032. synapse/storage/schema/main/delta/72/04drop_column_application_services_state_last_txn.sql.sqlite +40 -0
  1033. synapse/storage/schema/main/delta/72/05receipts_event_stream_ordering.sql +38 -0
  1034. synapse/storage/schema/main/delta/72/05remove_unstable_private_read_receipts.sql +38 -0
  1035. synapse/storage/schema/main/delta/72/06add_consent_ts_to_users.sql +35 -0
  1036. synapse/storage/schema/main/delta/72/06thread_notifications.sql +49 -0
  1037. synapse/storage/schema/main/delta/72/07force_update_current_state_events_membership.py +67 -0
  1038. synapse/storage/schema/main/delta/72/07thread_receipts.sql.postgres +30 -0
  1039. synapse/storage/schema/main/delta/72/07thread_receipts.sql.sqlite +70 -0
  1040. synapse/storage/schema/main/delta/72/08begin_cache_invalidation_seq_at_2.sql.postgres +23 -0
  1041. synapse/storage/schema/main/delta/72/08thread_receipts.sql +39 -0
  1042. synapse/storage/schema/main/delta/72/09partial_indices.sql.sqlite +56 -0
  1043. synapse/storage/schema/main/delta/73/01event_failed_pull_attempts.sql +48 -0
  1044. synapse/storage/schema/main/delta/73/02add_pusher_enabled.sql +35 -0
  1045. synapse/storage/schema/main/delta/73/02room_id_indexes_for_purging.sql +41 -0
  1046. synapse/storage/schema/main/delta/73/03pusher_device_id.sql +39 -0
  1047. synapse/storage/schema/main/delta/73/03users_approved_column.sql +39 -0
  1048. synapse/storage/schema/main/delta/73/04partial_join_details.sql +42 -0
  1049. synapse/storage/schema/main/delta/73/04pending_device_list_updates.sql +47 -0
  1050. synapse/storage/schema/main/delta/73/05old_push_actions.sql.postgres +22 -0
  1051. synapse/storage/schema/main/delta/73/05old_push_actions.sql.sqlite +24 -0
  1052. synapse/storage/schema/main/delta/73/06thread_notifications_thread_id_idx.sql +42 -0
  1053. synapse/storage/schema/main/delta/73/08thread_receipts_non_null.sql.postgres +23 -0
  1054. synapse/storage/schema/main/delta/73/08thread_receipts_non_null.sql.sqlite +76 -0
  1055. synapse/storage/schema/main/delta/73/09partial_joined_via_destination.sql +37 -0
  1056. synapse/storage/schema/main/delta/73/09threads_table.sql +49 -0
  1057. synapse/storage/schema/main/delta/73/10_update_sqlite_fts4_tokenizer.py +71 -0
  1058. synapse/storage/schema/main/delta/73/10login_tokens.sql +54 -0
  1059. synapse/storage/schema/main/delta/73/11event_search_room_id_n_distinct.sql.postgres +33 -0
  1060. synapse/storage/schema/main/delta/73/12refactor_device_list_outbound_pokes.sql +72 -0
  1061. synapse/storage/schema/main/delta/73/13add_device_lists_index.sql +39 -0
  1062. synapse/storage/schema/main/delta/73/20_un_partial_stated_room_stream.sql +51 -0
  1063. synapse/storage/schema/main/delta/73/21_un_partial_stated_room_stream_seq.sql.postgres +20 -0
  1064. synapse/storage/schema/main/delta/73/22_rebuild_user_dir_stats.sql +48 -0
  1065. synapse/storage/schema/main/delta/73/22_un_partial_stated_event_stream.sql +53 -0
  1066. synapse/storage/schema/main/delta/73/23_fix_thread_index.sql +52 -0
  1067. synapse/storage/schema/main/delta/73/23_un_partial_stated_room_stream_seq.sql.postgres +20 -0
  1068. synapse/storage/schema/main/delta/73/24_events_jump_to_date_index.sql +36 -0
  1069. synapse/storage/schema/main/delta/73/25drop_presence.sql +36 -0
  1070. synapse/storage/schema/main/delta/74/01_user_directory_stale_remote_users.sql +58 -0
  1071. synapse/storage/schema/main/delta/74/02_set_device_id_for_pushers_bg_update.sql +38 -0
  1072. synapse/storage/schema/main/delta/74/03_membership_tables_event_stream_ordering.sql.postgres +29 -0
  1073. synapse/storage/schema/main/delta/74/03_membership_tables_event_stream_ordering.sql.sqlite +23 -0
  1074. synapse/storage/schema/main/delta/74/03_room_membership_index.sql +38 -0
  1075. synapse/storage/schema/main/delta/74/04_delete_e2e_backup_keys_for_deactivated_users.sql +36 -0
  1076. synapse/storage/schema/main/delta/74/04_membership_tables_event_stream_ordering_triggers.py +87 -0
  1077. synapse/storage/schema/main/delta/74/05_events_txn_id_device_id.sql +72 -0
  1078. synapse/storage/schema/main/delta/74/90COMMENTS_destinations.sql.postgres +52 -0
  1079. synapse/storage/schema/main/delta/76/01_add_profiles_full_user_id_column.sql +39 -0
  1080. synapse/storage/schema/main/delta/76/02_add_user_filters_full_user_id_column.sql +39 -0
  1081. synapse/storage/schema/main/delta/76/03_per_user_experimental_features.sql +46 -0
  1082. synapse/storage/schema/main/delta/76/04_add_room_forgetter.sql +43 -0
  1083. synapse/storage/schema/main/delta/77/01_add_profiles_not_valid_check.sql.postgres +16 -0
  1084. synapse/storage/schema/main/delta/77/02_add_user_filters_not_valid_check.sql.postgres +16 -0
  1085. synapse/storage/schema/main/delta/77/03bg_populate_full_user_id_profiles.sql +35 -0
  1086. synapse/storage/schema/main/delta/77/04bg_populate_full_user_id_user_filters.sql +35 -0
  1087. synapse/storage/schema/main/delta/77/05thread_notifications_backfill.sql +67 -0
  1088. synapse/storage/schema/main/delta/77/06thread_notifications_not_null.sql.sqlite +102 -0
  1089. synapse/storage/schema/main/delta/77/06thread_notifications_not_null_event_push_actions.sql.postgres +27 -0
  1090. synapse/storage/schema/main/delta/77/06thread_notifications_not_null_event_push_actions_staging.sql.postgres +27 -0
  1091. synapse/storage/schema/main/delta/77/06thread_notifications_not_null_event_push_summary.sql.postgres +29 -0
  1092. synapse/storage/schema/main/delta/77/14bg_indices_event_stream_ordering.sql +39 -0
  1093. synapse/storage/schema/main/delta/78/01_validate_and_update_profiles.py +99 -0
  1094. synapse/storage/schema/main/delta/78/02_validate_and_update_user_filters.py +100 -0
  1095. synapse/storage/schema/main/delta/78/03_remove_unused_indexes_user_filters.py +72 -0
  1096. synapse/storage/schema/main/delta/78/03event_extremities_constraints.py +65 -0
  1097. synapse/storage/schema/main/delta/78/04_add_full_user_id_index_user_filters.py +32 -0
  1098. synapse/storage/schema/main/delta/79/03_read_write_locks_triggers.sql.postgres +102 -0
  1099. synapse/storage/schema/main/delta/79/03_read_write_locks_triggers.sql.sqlite +72 -0
  1100. synapse/storage/schema/main/delta/79/04_mitigate_stream_ordering_update_race.py +70 -0
  1101. synapse/storage/schema/main/delta/79/05_read_write_locks_triggers.sql.postgres +69 -0
  1102. synapse/storage/schema/main/delta/79/05_read_write_locks_triggers.sql.sqlite +65 -0
  1103. synapse/storage/schema/main/delta/80/01_users_alter_locked.sql +35 -0
  1104. synapse/storage/schema/main/delta/80/02_read_write_locks_unlogged.sql.postgres +30 -0
  1105. synapse/storage/schema/main/delta/80/02_scheduled_tasks.sql +47 -0
  1106. synapse/storage/schema/main/delta/80/03_read_write_locks_triggers.sql.postgres +37 -0
  1107. synapse/storage/schema/main/delta/80/04_read_write_locks_deadlock.sql.postgres +71 -0
  1108. synapse/storage/schema/main/delta/82/02_scheduled_tasks_index.sql +35 -0
  1109. synapse/storage/schema/main/delta/82/04_add_indices_for_purging_rooms.sql +39 -0
  1110. synapse/storage/schema/main/delta/82/05gaps.sql +44 -0
  1111. synapse/storage/schema/main/delta/83/01_drop_old_tables.sql +43 -0
  1112. synapse/storage/schema/main/delta/83/03_instance_name_receipts.sql.sqlite +17 -0
  1113. synapse/storage/schema/main/delta/83/05_cross_signing_key_update_grant.sql +34 -0
  1114. synapse/storage/schema/main/delta/83/06_event_push_summary_room.sql +36 -0
  1115. synapse/storage/schema/main/delta/84/01_auth_links_stats.sql.postgres +20 -0
  1116. synapse/storage/schema/main/delta/84/02_auth_links_index.sql +16 -0
  1117. synapse/storage/schema/main/delta/84/03_auth_links_analyze.sql.postgres +16 -0
  1118. synapse/storage/schema/main/delta/84/04_access_token_index.sql +15 -0
  1119. synapse/storage/schema/main/delta/85/01_add_suspended.sql +14 -0
  1120. synapse/storage/schema/main/delta/85/02_add_instance_names.sql +27 -0
  1121. synapse/storage/schema/main/delta/85/03_new_sequences.sql.postgres +54 -0
  1122. synapse/storage/schema/main/delta/85/04_cleanup_device_federation_outbox.sql +15 -0
  1123. synapse/storage/schema/main/delta/85/05_add_instance_names_converted_pos.sql +16 -0
  1124. synapse/storage/schema/main/delta/85/06_add_room_reports.sql +20 -0
  1125. synapse/storage/schema/main/delta/86/01_authenticate_media.sql +15 -0
  1126. synapse/storage/schema/main/delta/86/02_receipts_event_id_index.sql +15 -0
  1127. synapse/storage/schema/main/delta/87/01_sliding_sync_memberships.sql +169 -0
  1128. synapse/storage/schema/main/delta/87/02_per_connection_state.sql +81 -0
  1129. synapse/storage/schema/main/delta/87/03_current_state_index.sql +19 -0
  1130. synapse/storage/schema/main/delta/88/01_add_delayed_events.sql +43 -0
  1131. synapse/storage/schema/main/delta/88/01_custom_profile_fields.sql +15 -0
  1132. synapse/storage/schema/main/delta/88/02_fix_sliding_sync_membership_snapshots_forgotten_column.sql +21 -0
  1133. synapse/storage/schema/main/delta/88/03_add_otk_ts_added_index.sql +18 -0
  1134. synapse/storage/schema/main/delta/88/04_current_state_delta_index.sql +18 -0
  1135. synapse/storage/schema/main/delta/88/05_drop_old_otks.sql.postgres +19 -0
  1136. synapse/storage/schema/main/delta/88/05_drop_old_otks.sql.sqlite +19 -0
  1137. synapse/storage/schema/main/delta/88/05_sliding_sync_room_config_index.sql +20 -0
  1138. synapse/storage/schema/main/delta/88/06_events_received_ts_index.sql +17 -0
  1139. synapse/storage/schema/main/delta/89/01_sliding_sync_membership_snapshot_index.sql +15 -0
  1140. synapse/storage/schema/main/delta/90/01_add_column_participant_room_memberships_table.sql +16 -0
  1141. synapse/storage/schema/main/delta/91/01_media_hash.sql +28 -0
  1142. synapse/storage/schema/main/delta/92/01_remove_trigger.sql.postgres +16 -0
  1143. synapse/storage/schema/main/delta/92/01_remove_trigger.sql.sqlite +16 -0
  1144. synapse/storage/schema/main/delta/92/02_remove_populate_participant_bg_update.sql +17 -0
  1145. synapse/storage/schema/main/delta/92/04_ss_membership_snapshot_idx.sql +16 -0
  1146. synapse/storage/schema/main/delta/92/04_thread_subscriptions.sql +59 -0
  1147. synapse/storage/schema/main/delta/92/04_thread_subscriptions_seq.sql.postgres +19 -0
  1148. synapse/storage/schema/main/delta/92/05_fixup_max_depth_cap.sql +17 -0
  1149. synapse/storage/schema/main/delta/92/05_thread_subscriptions_comments.sql.postgres +18 -0
  1150. synapse/storage/schema/main/delta/92/06_device_federation_inbox_index.sql +16 -0
  1151. synapse/storage/schema/main/delta/92/06_threads_last_sent_stream_ordering_comments.sql.postgres +24 -0
  1152. synapse/storage/schema/main/delta/92/07_add_user_reports.sql +22 -0
  1153. synapse/storage/schema/main/delta/92/07_event_txn_id_device_id_txn_id2.sql +15 -0
  1154. synapse/storage/schema/main/delta/92/08_room_ban_redactions.sql +21 -0
  1155. synapse/storage/schema/main/delta/92/08_thread_subscriptions_seq_fixup.sql.postgres +19 -0
  1156. synapse/storage/schema/main/delta/92/09_thread_subscriptions_update.sql +20 -0
  1157. synapse/storage/schema/main/delta/92/09_thread_subscriptions_update.sql.postgres +18 -0
  1158. synapse/storage/schema/main/delta/93/01_add_delayed_events.sql +15 -0
  1159. synapse/storage/schema/main/delta/93/02_sliding_sync_members.sql +60 -0
  1160. synapse/storage/schema/main/delta/93/03_sss_pos_last_used.sql +27 -0
  1161. synapse/storage/schema/main/full_schemas/72/full.sql.postgres +1344 -0
  1162. synapse/storage/schema/main/full_schemas/72/full.sql.sqlite +646 -0
  1163. synapse/storage/schema/state/delta/23/drop_state_index.sql +35 -0
  1164. synapse/storage/schema/state/delta/32/remove_state_indices.sql +38 -0
  1165. synapse/storage/schema/state/delta/35/add_state_index.sql +36 -0
  1166. synapse/storage/schema/state/delta/35/state.sql +41 -0
  1167. synapse/storage/schema/state/delta/35/state_dedupe.sql +36 -0
  1168. synapse/storage/schema/state/delta/47/state_group_seq.py +38 -0
  1169. synapse/storage/schema/state/delta/56/state_group_room_idx.sql +36 -0
  1170. synapse/storage/schema/state/delta/61/02state_groups_state_n_distinct.sql.postgres +34 -0
  1171. synapse/storage/schema/state/delta/70/08_state_group_edges_unique.sql +36 -0
  1172. synapse/storage/schema/state/delta/89/01_state_groups_deletion.sql +39 -0
  1173. synapse/storage/schema/state/delta/90/02_delete_unreferenced_state_groups.sql +16 -0
  1174. synapse/storage/schema/state/delta/90/03_remove_old_deletion_bg_update.sql +15 -0
  1175. synapse/storage/schema/state/full_schemas/72/full.sql.postgres +30 -0
  1176. synapse/storage/schema/state/full_schemas/72/full.sql.sqlite +20 -0
  1177. synapse/storage/types.py +183 -0
  1178. synapse/storage/util/__init__.py +20 -0
  1179. synapse/storage/util/id_generators.py +928 -0
  1180. synapse/storage/util/partial_state_events_tracker.py +194 -0
  1181. synapse/storage/util/sequence.py +315 -0
  1182. synapse/streams/__init__.py +43 -0
  1183. synapse/streams/config.py +91 -0
  1184. synapse/streams/events.py +203 -0
  1185. synapse/synapse_rust/__init__.pyi +3 -0
  1186. synapse/synapse_rust/acl.pyi +20 -0
  1187. synapse/synapse_rust/events.pyi +136 -0
  1188. synapse/synapse_rust/http_client.pyi +32 -0
  1189. synapse/synapse_rust/push.pyi +86 -0
  1190. synapse/synapse_rust/rendezvous.pyi +30 -0
  1191. synapse/synapse_rust/segmenter.pyi +1 -0
  1192. synapse/synapse_rust.abi3.so +0 -0
  1193. synapse/types/__init__.py +1600 -0
  1194. synapse/types/handlers/__init__.py +93 -0
  1195. synapse/types/handlers/policy_server.py +16 -0
  1196. synapse/types/handlers/sliding_sync.py +1004 -0
  1197. synapse/types/rest/__init__.py +25 -0
  1198. synapse/types/rest/client/__init__.py +413 -0
  1199. synapse/types/state.py +634 -0
  1200. synapse/types/storage/__init__.py +66 -0
  1201. synapse/util/__init__.py +160 -0
  1202. synapse/util/async_helpers.py +1048 -0
  1203. synapse/util/background_queue.py +142 -0
  1204. synapse/util/batching_queue.py +203 -0
  1205. synapse/util/caches/__init__.py +300 -0
  1206. synapse/util/caches/cached_call.py +143 -0
  1207. synapse/util/caches/deferred_cache.py +530 -0
  1208. synapse/util/caches/descriptors.py +692 -0
  1209. synapse/util/caches/dictionary_cache.py +346 -0
  1210. synapse/util/caches/expiringcache.py +250 -0
  1211. synapse/util/caches/lrucache.py +976 -0
  1212. synapse/util/caches/response_cache.py +323 -0
  1213. synapse/util/caches/stream_change_cache.py +370 -0
  1214. synapse/util/caches/treecache.py +189 -0
  1215. synapse/util/caches/ttlcache.py +197 -0
  1216. synapse/util/cancellation.py +63 -0
  1217. synapse/util/check_dependencies.py +335 -0
  1218. synapse/util/clock.py +592 -0
  1219. synapse/util/daemonize.py +165 -0
  1220. synapse/util/distributor.py +157 -0
  1221. synapse/util/duration.py +117 -0
  1222. synapse/util/events.py +134 -0
  1223. synapse/util/file_consumer.py +164 -0
  1224. synapse/util/frozenutils.py +57 -0
  1225. synapse/util/gai_resolver.py +178 -0
  1226. synapse/util/hash.py +38 -0
  1227. synapse/util/httpresourcetree.py +108 -0
  1228. synapse/util/iterutils.py +190 -0
  1229. synapse/util/json.py +56 -0
  1230. synapse/util/linked_list.py +156 -0
  1231. synapse/util/logcontext.py +46 -0
  1232. synapse/util/logformatter.py +28 -0
  1233. synapse/util/macaroons.py +325 -0
  1234. synapse/util/manhole.py +191 -0
  1235. synapse/util/metrics.py +339 -0
  1236. synapse/util/module_loader.py +116 -0
  1237. synapse/util/msisdn.py +51 -0
  1238. synapse/util/patch_inline_callbacks.py +250 -0
  1239. synapse/util/pydantic_models.py +63 -0
  1240. synapse/util/ratelimitutils.py +422 -0
  1241. synapse/util/retryutils.py +339 -0
  1242. synapse/util/rlimit.py +42 -0
  1243. synapse/util/rust.py +164 -0
  1244. synapse/util/sentinel.py +21 -0
  1245. synapse/util/stringutils.py +293 -0
  1246. synapse/util/task_scheduler.py +494 -0
  1247. synapse/util/templates.py +126 -0
  1248. synapse/util/threepids.py +123 -0
  1249. synapse/util/wheel_timer.py +112 -0
  1250. synapse/visibility.py +869 -0
  1251. synmark/__init__.py +47 -0
  1252. synmark/__main__.py +128 -0
  1253. synmark/suites/__init__.py +9 -0
  1254. synmark/suites/logging.py +154 -0
  1255. synmark/suites/lrucache.py +48 -0
  1256. synmark/suites/lrucache_evict.py +49 -0
  1257. sytest-blacklist +33 -0
  1258. tests/__init__.py +29 -0
  1259. tests/api/__init__.py +0 -0
  1260. tests/api/test_auth.py +548 -0
  1261. tests/api/test_errors.py +46 -0
  1262. tests/api/test_filtering.py +622 -0
  1263. tests/api/test_ratelimiting.py +505 -0
  1264. tests/api/test_urls.py +81 -0
  1265. tests/app/__init__.py +0 -0
  1266. tests/app/test_homeserver_shutdown.py +271 -0
  1267. tests/app/test_homeserver_start.py +49 -0
  1268. tests/app/test_openid_listener.py +139 -0
  1269. tests/app/test_phone_stats_home.py +248 -0
  1270. tests/appservice/__init__.py +20 -0
  1271. tests/appservice/test_api.py +253 -0
  1272. tests/appservice/test_appservice.py +259 -0
  1273. tests/appservice/test_scheduler.py +477 -0
  1274. tests/config/__init__.py +20 -0
  1275. tests/config/test___main__.py +38 -0
  1276. tests/config/test_api.py +146 -0
  1277. tests/config/test_appservice.py +48 -0
  1278. tests/config/test_background_update.py +65 -0
  1279. tests/config/test_base.py +151 -0
  1280. tests/config/test_cache.py +193 -0
  1281. tests/config/test_database.py +42 -0
  1282. tests/config/test_generate.py +72 -0
  1283. tests/config/test_load.py +322 -0
  1284. tests/config/test_oauth_delegation.py +454 -0
  1285. tests/config/test_ratelimiting.py +77 -0
  1286. tests/config/test_registration_config.py +203 -0
  1287. tests/config/test_room_directory.py +203 -0
  1288. tests/config/test_server.py +248 -0
  1289. tests/config/test_tls.py +209 -0
  1290. tests/config/test_util.py +59 -0
  1291. tests/config/test_workers.py +332 -0
  1292. tests/config/utils.py +66 -0
  1293. tests/crypto/__init__.py +20 -0
  1294. tests/crypto/test_event_signing.py +109 -0
  1295. tests/crypto/test_keyring.py +737 -0
  1296. tests/events/__init__.py +0 -0
  1297. tests/events/test_auto_accept_invites.py +827 -0
  1298. tests/events/test_presence_router.py +540 -0
  1299. tests/events/test_snapshot.py +121 -0
  1300. tests/events/test_utils.py +986 -0
  1301. tests/federation/__init__.py +0 -0
  1302. tests/federation/test_complexity.py +257 -0
  1303. tests/federation/test_federation_catch_up.py +585 -0
  1304. tests/federation/test_federation_client.py +316 -0
  1305. tests/federation/test_federation_devices.py +161 -0
  1306. tests/federation/test_federation_media.py +295 -0
  1307. tests/federation/test_federation_out_of_band_membership.py +671 -0
  1308. tests/federation/test_federation_sender.py +956 -0
  1309. tests/federation/test_federation_server.py +631 -0
  1310. tests/federation/transport/__init__.py +0 -0
  1311. tests/federation/transport/server/__init__.py +20 -0
  1312. tests/federation/transport/server/test__base.py +156 -0
  1313. tests/federation/transport/test_client.py +157 -0
  1314. tests/federation/transport/test_knocking.py +323 -0
  1315. tests/federation/transport/test_server.py +74 -0
  1316. tests/handlers/__init__.py +0 -0
  1317. tests/handlers/oidc_test_key.p8 +5 -0
  1318. tests/handlers/oidc_test_key.pub.pem +4 -0
  1319. tests/handlers/test_admin.py +361 -0
  1320. tests/handlers/test_appservice.py +1336 -0
  1321. tests/handlers/test_auth.py +248 -0
  1322. tests/handlers/test_cas.py +239 -0
  1323. tests/handlers/test_deactivate_account.py +485 -0
  1324. tests/handlers/test_device.py +665 -0
  1325. tests/handlers/test_directory.py +613 -0
  1326. tests/handlers/test_e2e_keys.py +2025 -0
  1327. tests/handlers/test_e2e_room_keys.py +569 -0
  1328. tests/handlers/test_federation.py +794 -0
  1329. tests/handlers/test_federation_event.py +1181 -0
  1330. tests/handlers/test_message.py +322 -0
  1331. tests/handlers/test_oauth_delegation.py +1314 -0
  1332. tests/handlers/test_oidc.py +1688 -0
  1333. tests/handlers/test_password_providers.py +987 -0
  1334. tests/handlers/test_presence.py +2144 -0
  1335. tests/handlers/test_profile.py +401 -0
  1336. tests/handlers/test_receipts.py +342 -0
  1337. tests/handlers/test_register.py +880 -0
  1338. tests/handlers/test_room.py +108 -0
  1339. tests/handlers/test_room_list.py +93 -0
  1340. tests/handlers/test_room_member.py +764 -0
  1341. tests/handlers/test_room_policy.py +468 -0
  1342. tests/handlers/test_room_summary.py +1248 -0
  1343. tests/handlers/test_saml.py +427 -0
  1344. tests/handlers/test_send_email.py +230 -0
  1345. tests/handlers/test_sliding_sync.py +5065 -0
  1346. tests/handlers/test_sso.py +152 -0
  1347. tests/handlers/test_stats.py +594 -0
  1348. tests/handlers/test_sync.py +1275 -0
  1349. tests/handlers/test_typing.py +557 -0
  1350. tests/handlers/test_user_directory.py +1435 -0
  1351. tests/handlers/test_worker_lock.py +126 -0
  1352. tests/http/__init__.py +196 -0
  1353. tests/http/ca.crt +19 -0
  1354. tests/http/ca.key +27 -0
  1355. tests/http/federation/__init__.py +19 -0
  1356. tests/http/federation/test_matrix_federation_agent.py +1855 -0
  1357. tests/http/federation/test_srv_resolver.py +220 -0
  1358. tests/http/server/__init__.py +20 -0
  1359. tests/http/server/_base.py +621 -0
  1360. tests/http/server.key +27 -0
  1361. tests/http/test_additional_resource.py +76 -0
  1362. tests/http/test_client.py +422 -0
  1363. tests/http/test_endpoint.py +62 -0
  1364. tests/http/test_matrixfederationclient.py +1092 -0
  1365. tests/http/test_proxy.py +75 -0
  1366. tests/http/test_proxyagent.py +1008 -0
  1367. tests/http/test_servlet.py +145 -0
  1368. tests/http/test_simple_client.py +188 -0
  1369. tests/http/test_site.py +247 -0
  1370. tests/logging/__init__.py +42 -0
  1371. tests/logging/test_loggers.py +127 -0
  1372. tests/logging/test_opentracing.py +524 -0
  1373. tests/logging/test_remote_handler.py +184 -0
  1374. tests/logging/test_terse_json.py +253 -0
  1375. tests/media/__init__.py +20 -0
  1376. tests/media/test_base.py +88 -0
  1377. tests/media/test_filepath.py +602 -0
  1378. tests/media/test_html_preview.py +565 -0
  1379. tests/media/test_media_retention.py +299 -0
  1380. tests/media/test_media_storage.py +1401 -0
  1381. tests/media/test_oembed.py +172 -0
  1382. tests/media/test_url_previewer.py +120 -0
  1383. tests/metrics/__init__.py +0 -0
  1384. tests/metrics/test_background_process_metrics.py +21 -0
  1385. tests/metrics/test_metrics.py +407 -0
  1386. tests/metrics/test_phone_home_stats.py +263 -0
  1387. tests/module_api/__init__.py +0 -0
  1388. tests/module_api/test_account_data_manager.py +171 -0
  1389. tests/module_api/test_api.py +1035 -0
  1390. tests/module_api/test_event_unsigned_addition.py +66 -0
  1391. tests/module_api/test_spamchecker.py +286 -0
  1392. tests/push/__init__.py +0 -0
  1393. tests/push/test_bulk_push_rule_evaluator.py +652 -0
  1394. tests/push/test_email.py +570 -0
  1395. tests/push/test_http.py +1247 -0
  1396. tests/push/test_presentable_names.py +238 -0
  1397. tests/push/test_push_rule_evaluator.py +1069 -0
  1398. tests/replication/__init__.py +20 -0
  1399. tests/replication/_base.py +619 -0
  1400. tests/replication/http/__init__.py +20 -0
  1401. tests/replication/http/test__base.py +113 -0
  1402. tests/replication/storage/__init__.py +20 -0
  1403. tests/replication/storage/_base.py +85 -0
  1404. tests/replication/storage/test_events.py +299 -0
  1405. tests/replication/tcp/__init__.py +19 -0
  1406. tests/replication/tcp/streams/__init__.py +19 -0
  1407. tests/replication/tcp/streams/test_account_data.py +133 -0
  1408. tests/replication/tcp/streams/test_events.py +565 -0
  1409. tests/replication/tcp/streams/test_federation.py +117 -0
  1410. tests/replication/tcp/streams/test_partial_state.py +72 -0
  1411. tests/replication/tcp/streams/test_receipts.py +110 -0
  1412. tests/replication/tcp/streams/test_thread_subscriptions.py +157 -0
  1413. tests/replication/tcp/streams/test_to_device.py +112 -0
  1414. tests/replication/tcp/streams/test_typing.py +223 -0
  1415. tests/replication/tcp/test_commands.py +50 -0
  1416. tests/replication/tcp/test_handler.py +211 -0
  1417. tests/replication/test_auth.py +120 -0
  1418. tests/replication/test_client_reader_shard.py +101 -0
  1419. tests/replication/test_federation_ack.py +88 -0
  1420. tests/replication/test_federation_sender_shard.py +352 -0
  1421. tests/replication/test_module_cache_invalidation.py +89 -0
  1422. tests/replication/test_multi_media_repo.py +496 -0
  1423. tests/replication/test_pusher_shard.py +192 -0
  1424. tests/replication/test_sharded_event_persister.py +332 -0
  1425. tests/replication/test_sharded_receipts.py +250 -0
  1426. tests/rest/__init__.py +20 -0
  1427. tests/rest/admin/__init__.py +19 -0
  1428. tests/rest/admin/test_admin.py +614 -0
  1429. tests/rest/admin/test_background_updates.py +375 -0
  1430. tests/rest/admin/test_device.py +600 -0
  1431. tests/rest/admin/test_event.py +74 -0
  1432. tests/rest/admin/test_event_reports.py +781 -0
  1433. tests/rest/admin/test_federation.py +863 -0
  1434. tests/rest/admin/test_jwks.py +106 -0
  1435. tests/rest/admin/test_media.py +1091 -0
  1436. tests/rest/admin/test_registration_tokens.py +729 -0
  1437. tests/rest/admin/test_room.py +3626 -0
  1438. tests/rest/admin/test_scheduled_tasks.py +192 -0
  1439. tests/rest/admin/test_server_notice.py +753 -0
  1440. tests/rest/admin/test_statistics.py +523 -0
  1441. tests/rest/admin/test_user.py +6061 -0
  1442. tests/rest/admin/test_username_available.py +82 -0
  1443. tests/rest/client/__init__.py +20 -0
  1444. tests/rest/client/sliding_sync/__init__.py +13 -0
  1445. tests/rest/client/sliding_sync/test_connection_tracking.py +505 -0
  1446. tests/rest/client/sliding_sync/test_extension_account_data.py +1056 -0
  1447. tests/rest/client/sliding_sync/test_extension_e2ee.py +459 -0
  1448. tests/rest/client/sliding_sync/test_extension_receipts.py +934 -0
  1449. tests/rest/client/sliding_sync/test_extension_thread_subscriptions.py +497 -0
  1450. tests/rest/client/sliding_sync/test_extension_to_device.py +294 -0
  1451. tests/rest/client/sliding_sync/test_extension_typing.py +500 -0
  1452. tests/rest/client/sliding_sync/test_extensions.py +306 -0
  1453. tests/rest/client/sliding_sync/test_lists_filters.py +1975 -0
  1454. tests/rest/client/sliding_sync/test_room_subscriptions.py +303 -0
  1455. tests/rest/client/sliding_sync/test_rooms_invites.py +528 -0
  1456. tests/rest/client/sliding_sync/test_rooms_meta.py +1338 -0
  1457. tests/rest/client/sliding_sync/test_rooms_required_state.py +2247 -0
  1458. tests/rest/client/sliding_sync/test_rooms_timeline.py +718 -0
  1459. tests/rest/client/sliding_sync/test_sliding_sync.py +1688 -0
  1460. tests/rest/client/test_account.py +1543 -0
  1461. tests/rest/client/test_account_data.py +81 -0
  1462. tests/rest/client/test_auth.py +1508 -0
  1463. tests/rest/client/test_auth_metadata.py +145 -0
  1464. tests/rest/client/test_capabilities.py +318 -0
  1465. tests/rest/client/test_consent.py +138 -0
  1466. tests/rest/client/test_delayed_events.py +553 -0
  1467. tests/rest/client/test_devices.py +634 -0
  1468. tests/rest/client/test_directory.py +249 -0
  1469. tests/rest/client/test_ephemeral_message.py +113 -0
  1470. tests/rest/client/test_events.py +165 -0
  1471. tests/rest/client/test_filter.py +124 -0
  1472. tests/rest/client/test_identity.py +67 -0
  1473. tests/rest/client/test_keys.py +516 -0
  1474. tests/rest/client/test_login.py +1881 -0
  1475. tests/rest/client/test_login_token_request.py +175 -0
  1476. tests/rest/client/test_matrixrtc.py +105 -0
  1477. tests/rest/client/test_media.py +3156 -0
  1478. tests/rest/client/test_models.py +83 -0
  1479. tests/rest/client/test_mutual_rooms.py +235 -0
  1480. tests/rest/client/test_notifications.py +231 -0
  1481. tests/rest/client/test_owned_state.py +308 -0
  1482. tests/rest/client/test_password_policy.py +186 -0
  1483. tests/rest/client/test_power_levels.py +295 -0
  1484. tests/rest/client/test_presence.py +149 -0
  1485. tests/rest/client/test_profile.py +925 -0
  1486. tests/rest/client/test_push_rule_attrs.py +510 -0
  1487. tests/rest/client/test_read_marker.py +151 -0
  1488. tests/rest/client/test_receipts.py +287 -0
  1489. tests/rest/client/test_redactions.py +657 -0
  1490. tests/rest/client/test_register.py +1314 -0
  1491. tests/rest/client/test_relations.py +1954 -0
  1492. tests/rest/client/test_rendezvous.py +468 -0
  1493. tests/rest/client/test_reporting.py +324 -0
  1494. tests/rest/client/test_retention.py +389 -0
  1495. tests/rest/client/test_rooms.py +5486 -0
  1496. tests/rest/client/test_sendtodevice.py +271 -0
  1497. tests/rest/client/test_shadow_banned.py +335 -0
  1498. tests/rest/client/test_sync.py +1147 -0
  1499. tests/rest/client/test_tags.py +161 -0
  1500. tests/rest/client/test_third_party_rules.py +1076 -0
  1501. tests/rest/client/test_thread_subscriptions.py +351 -0
  1502. tests/rest/client/test_transactions.py +204 -0
  1503. tests/rest/client/test_typing.py +114 -0
  1504. tests/rest/client/test_upgrade_room.py +433 -0
  1505. tests/rest/client/utils.py +985 -0
  1506. tests/rest/key/__init__.py +0 -0
  1507. tests/rest/key/v2/__init__.py +0 -0
  1508. tests/rest/key/v2/test_remote_key_resource.py +282 -0
  1509. tests/rest/media/__init__.py +19 -0
  1510. tests/rest/media/test_domain_blocking.py +148 -0
  1511. tests/rest/media/test_url_preview.py +1445 -0
  1512. tests/rest/synapse/__init__.py +12 -0
  1513. tests/rest/synapse/client/__init__.py +12 -0
  1514. tests/rest/synapse/client/test_federation_whitelist.py +118 -0
  1515. tests/rest/synapse/mas/__init__.py +12 -0
  1516. tests/rest/synapse/mas/_base.py +43 -0
  1517. tests/rest/synapse/mas/test_devices.py +693 -0
  1518. tests/rest/synapse/mas/test_users.py +1399 -0
  1519. tests/rest/test_health.py +35 -0
  1520. tests/rest/test_well_known.py +155 -0
  1521. tests/scripts/__init__.py +0 -0
  1522. tests/scripts/test_new_matrix_user.py +172 -0
  1523. tests/server.py +1374 -0
  1524. tests/server_notices/__init__.py +241 -0
  1525. tests/server_notices/test_consent.py +111 -0
  1526. tests/server_notices/test_resource_limits_server_notices.py +409 -0
  1527. tests/state/__init__.py +0 -0
  1528. tests/state/test_v2.py +1096 -0
  1529. tests/state/test_v21.py +506 -0
  1530. tests/storage/__init__.py +0 -0
  1531. tests/storage/databases/__init__.py +20 -0
  1532. tests/storage/databases/main/__init__.py +20 -0
  1533. tests/storage/databases/main/test_cache.py +124 -0
  1534. tests/storage/databases/main/test_deviceinbox.py +323 -0
  1535. tests/storage/databases/main/test_end_to_end_keys.py +127 -0
  1536. tests/storage/databases/main/test_events_worker.py +594 -0
  1537. tests/storage/databases/main/test_lock.py +499 -0
  1538. tests/storage/databases/main/test_metrics.py +88 -0
  1539. tests/storage/databases/main/test_receipts.py +218 -0
  1540. tests/storage/databases/main/test_room.py +192 -0
  1541. tests/storage/test__base.py +178 -0
  1542. tests/storage/test_account_data.py +186 -0
  1543. tests/storage/test_appservice.py +568 -0
  1544. tests/storage/test_background_update.py +671 -0
  1545. tests/storage/test_base.py +813 -0
  1546. tests/storage/test_cleanup_extrems.py +396 -0
  1547. tests/storage/test_client_ips.py +788 -0
  1548. tests/storage/test_database.py +288 -0
  1549. tests/storage/test_devices.py +353 -0
  1550. tests/storage/test_directory.py +74 -0
  1551. tests/storage/test_e2e_room_keys.py +87 -0
  1552. tests/storage/test_end_to_end_keys.py +120 -0
  1553. tests/storage/test_event_chain.py +826 -0
  1554. tests/storage/test_event_federation.py +1433 -0
  1555. tests/storage/test_event_push_actions.py +809 -0
  1556. tests/storage/test_events.py +591 -0
  1557. tests/storage/test_events_bg_updates.py +156 -0
  1558. tests/storage/test_id_generators.py +791 -0
  1559. tests/storage/test_invite_rule.py +171 -0
  1560. tests/storage/test_main.py +56 -0
  1561. tests/storage/test_monthly_active_users.py +500 -0
  1562. tests/storage/test_profile.py +134 -0
  1563. tests/storage/test_purge.py +459 -0
  1564. tests/storage/test_receipts.py +309 -0
  1565. tests/storage/test_redaction.py +462 -0
  1566. tests/storage/test_registration.py +277 -0
  1567. tests/storage/test_relations.py +118 -0
  1568. tests/storage/test_rollback_worker.py +132 -0
  1569. tests/storage/test_room.py +69 -0
  1570. tests/storage/test_room_search.py +383 -0
  1571. tests/storage/test_roommember.py +812 -0
  1572. tests/storage/test_sliding_sync_tables.py +5187 -0
  1573. tests/storage/test_state.py +959 -0
  1574. tests/storage/test_state_deletion.py +475 -0
  1575. tests/storage/test_stream.py +1533 -0
  1576. tests/storage/test_thread_subscriptions.py +369 -0
  1577. tests/storage/test_transactions.py +77 -0
  1578. tests/storage/test_txn_limit.py +49 -0
  1579. tests/storage/test_unsafe_locale.py +67 -0
  1580. tests/storage/test_user_directory.py +691 -0
  1581. tests/storage/test_user_filters.py +101 -0
  1582. tests/storage/util/__init__.py +20 -0
  1583. tests/storage/util/test_partial_state_events_tracker.py +181 -0
  1584. tests/synapse_rust/__init__.py +11 -0
  1585. tests/synapse_rust/test_http_client.py +225 -0
  1586. tests/test_distributor.py +74 -0
  1587. tests/test_event_auth.py +921 -0
  1588. tests/test_mau.py +347 -0
  1589. tests/test_phone_home.py +102 -0
  1590. tests/test_rust.py +11 -0
  1591. tests/test_server.py +557 -0
  1592. tests/test_state.py +902 -0
  1593. tests/test_terms_auth.py +128 -0
  1594. tests/test_types.py +201 -0
  1595. tests/test_utils/__init__.py +161 -0
  1596. tests/test_utils/event_injection.py +150 -0
  1597. tests/test_utils/html_parsers.py +59 -0
  1598. tests/test_utils/logging_setup.py +74 -0
  1599. tests/test_utils/oidc.py +370 -0
  1600. tests/test_visibility.py +712 -0
  1601. tests/types/__init__.py +0 -0
  1602. tests/types/test_init.py +51 -0
  1603. tests/types/test_state.py +627 -0
  1604. tests/unittest.py +1108 -0
  1605. tests/util/__init__.py +20 -0
  1606. tests/util/caches/__init__.py +20 -0
  1607. tests/util/caches/test_cached_call.py +168 -0
  1608. tests/util/caches/test_deferred_cache.py +317 -0
  1609. tests/util/caches/test_descriptors.py +1110 -0
  1610. tests/util/caches/test_response_cache.py +225 -0
  1611. tests/util/caches/test_ttlcache.py +90 -0
  1612. tests/util/test_async_helpers.py +808 -0
  1613. tests/util/test_background_queue.py +117 -0
  1614. tests/util/test_batching_queue.py +252 -0
  1615. tests/util/test_check_dependencies.py +243 -0
  1616. tests/util/test_dict_cache.py +130 -0
  1617. tests/util/test_events.py +118 -0
  1618. tests/util/test_expiring_cache.py +113 -0
  1619. tests/util/test_file_consumer.py +199 -0
  1620. tests/util/test_itertools.py +190 -0
  1621. tests/util/test_linearizer.py +264 -0
  1622. tests/util/test_logcontext.py +715 -0
  1623. tests/util/test_logformatter.py +44 -0
  1624. tests/util/test_lrucache.py +479 -0
  1625. tests/util/test_macaroons.py +126 -0
  1626. tests/util/test_mutable_overlay_mapping.py +189 -0
  1627. tests/util/test_ratelimitutils.py +146 -0
  1628. tests/util/test_retryutils.py +314 -0
  1629. tests/util/test_rwlock.py +401 -0
  1630. tests/util/test_stream_change_cache.py +304 -0
  1631. tests/util/test_stringutils.py +86 -0
  1632. tests/util/test_task_scheduler.py +227 -0
  1633. tests/util/test_threepids.py +55 -0
  1634. tests/util/test_treecache.py +93 -0
  1635. tests/util/test_wheel_timer.py +82 -0
  1636. tests/utils.py +342 -0
@@ -0,0 +1,1869 @@
1
+ #
2
+ # This file is licensed under the Affero General Public License (AGPL) version 3.
3
+ #
4
+ # Copyright 2019,2020 The Matrix.org Foundation C.I.C.
5
+ # Copyright 2016 OpenMarket Ltd
6
+ # Copyright (C) 2023 New Vector, Ltd
7
+ #
8
+ # This program is free software: you can redistribute it and/or modify
9
+ # it under the terms of the GNU Affero General Public License as
10
+ # published by the Free Software Foundation, either version 3 of the
11
+ # License, or (at your option) any later version.
12
+ #
13
+ # See the GNU Affero General Public License for more details:
14
+ # <https://www.gnu.org/licenses/agpl-3.0.html>.
15
+ #
16
+ # Originally licensed under the Apache License, Version 2.0:
17
+ # <http://www.apache.org/licenses/LICENSE-2.0>.
18
+ #
19
+ # [This file includes modifications made by New Vector Limited]
20
+ #
21
+ #
22
+ import logging
23
+ import random
24
+ from threading import Lock
25
+ from typing import (
26
+ TYPE_CHECKING,
27
+ AbstractSet,
28
+ Iterable,
29
+ Mapping,
30
+ cast,
31
+ )
32
+
33
+ from synapse.api import errors
34
+ from synapse.api.constants import EduTypes, EventTypes, Membership
35
+ from synapse.api.errors import (
36
+ Codes,
37
+ FederationDeniedError,
38
+ HttpResponseException,
39
+ InvalidAPICallError,
40
+ RequestSendFailed,
41
+ SynapseError,
42
+ )
43
+ from synapse.logging.opentracing import log_kv, set_tag, trace
44
+ from synapse.metrics.background_process_metrics import (
45
+ wrap_as_background_process,
46
+ )
47
+ from synapse.replication.http.devices import (
48
+ ReplicationDeviceHandleRoomUnPartialStated,
49
+ ReplicationHandleNewDeviceUpdateRestServlet,
50
+ ReplicationMultiUserDevicesResyncRestServlet,
51
+ ReplicationNotifyDeviceUpdateRestServlet,
52
+ ReplicationNotifyUserSignatureUpdateRestServlet,
53
+ )
54
+ from synapse.storage.databases.main.client_ips import DeviceLastConnectionInfo
55
+ from synapse.storage.databases.main.roommember import EventIdMembership
56
+ from synapse.storage.databases.main.state_deltas import StateDelta
57
+ from synapse.types import (
58
+ DeviceListUpdates,
59
+ JsonDict,
60
+ JsonMapping,
61
+ ScheduledTask,
62
+ StrCollection,
63
+ StreamKeyType,
64
+ StreamToken,
65
+ TaskStatus,
66
+ UserID,
67
+ get_domain_from_id,
68
+ get_verify_key_from_cross_signing_key,
69
+ )
70
+ from synapse.util import stringutils
71
+ from synapse.util.async_helpers import Linearizer
72
+ from synapse.util.caches.expiringcache import ExpiringCache
73
+ from synapse.util.cancellation import cancellable
74
+ from synapse.util.duration import Duration
75
+ from synapse.util.metrics import measure_func
76
+ from synapse.util.retryutils import (
77
+ NotRetryingDestination,
78
+ filter_destinations_by_retry_limiter,
79
+ )
80
+
81
+ if TYPE_CHECKING:
82
+ from synapse.app.generic_worker import GenericWorkerStore
83
+ from synapse.server import HomeServer
84
+
85
+ logger = logging.getLogger(__name__)
86
+
87
+ DELETE_DEVICE_MSGS_TASK_NAME = "delete_device_messages"
88
+ MAX_DEVICE_DISPLAY_NAME_LEN = 100
89
+ DELETE_STALE_DEVICES_INTERVAL = Duration(days=1)
90
+
91
+
92
+ def _check_device_name_length(name: str | None) -> None:
93
+ """
94
+ Checks whether a device name is longer than the maximum allowed length.
95
+
96
+ Args:
97
+ name: The name of the device.
98
+
99
+ Raises:
100
+ SynapseError: if the device name is too long.
101
+ """
102
+ if name and len(name) > MAX_DEVICE_DISPLAY_NAME_LEN:
103
+ raise SynapseError(
104
+ 400,
105
+ "Device display name is too long (max %i)" % (MAX_DEVICE_DISPLAY_NAME_LEN,),
106
+ errcode=Codes.TOO_LARGE,
107
+ )
108
+
109
+
110
+ class DeviceHandler:
111
+ """
112
+ Handles most things related to devices. This doesn't do any writing to the
113
+ device list stream on its own, and will call to device list writers through
114
+ replication when necessary (see DeviceWriterHandler).
115
+ """
116
+
117
+ device_list_updater: "DeviceListWorkerUpdater"
118
+ store: "GenericWorkerStore"
119
+
120
+ def __init__(self, hs: "HomeServer"):
121
+ self.server_name = hs.hostname # nb must be called this for @measure_func
122
+ self.clock = hs.get_clock() # nb must be called this for @measure_func
123
+ self.hs = hs # nb must be called this for @wrap_as_background_process
124
+ self.store = cast("GenericWorkerStore", hs.get_datastores().main)
125
+ self.notifier = hs.get_notifier()
126
+ self.state = hs.get_state_handler()
127
+ self._appservice_handler = hs.get_application_service_handler()
128
+ self._state_storage = hs.get_storage_controllers().state
129
+ self._auth_handler = hs.get_auth_handler()
130
+ self._account_data_handler = hs.get_account_data_handler()
131
+ self._event_sources = hs.get_event_sources()
132
+ self._msc3852_enabled = hs.config.experimental.msc3852_enabled
133
+ self._query_appservices_for_keys = (
134
+ hs.config.experimental.msc3984_appservice_key_query
135
+ )
136
+ self._task_scheduler = hs.get_task_scheduler()
137
+
138
+ self._dont_notify_new_devices_for = (
139
+ hs.config.registration.dont_notify_new_devices_for
140
+ )
141
+
142
+ self.device_list_updater = DeviceListWorkerUpdater(hs)
143
+
144
+ self._task_scheduler.register_action(
145
+ self._delete_device_messages, DELETE_DEVICE_MSGS_TASK_NAME
146
+ )
147
+
148
+ self._device_list_writers = hs.config.worker.writers.device_lists
149
+
150
+ # Ensure a few operations are only running on the first device list writer
151
+ #
152
+ # This is needed because of a few linearizers in the DeviceListUpdater,
153
+ # and avoid using cross-worker locks.
154
+ #
155
+ # The main logic update is that the DeviceListUpdater is now only
156
+ # instantiated on the first device list writer, and a few methods that
157
+ # were safe to move to any worker were moved to the DeviceListWorkerUpdater
158
+ # This must be kept in sync with DeviceListWorkerUpdater
159
+ self._main_device_list_writer = hs.config.worker.writers.device_lists[0]
160
+
161
+ self._notify_device_update_client = (
162
+ ReplicationNotifyDeviceUpdateRestServlet.make_client(hs)
163
+ )
164
+ self._notify_user_signature_update_client = (
165
+ ReplicationNotifyUserSignatureUpdateRestServlet.make_client(hs)
166
+ )
167
+ self._handle_new_device_update_client = (
168
+ ReplicationHandleNewDeviceUpdateRestServlet.make_client(hs)
169
+ )
170
+ self._handle_room_un_partial_stated_client = (
171
+ ReplicationDeviceHandleRoomUnPartialStated.make_client(hs)
172
+ )
173
+
174
+ # The EDUs are handled on a single writer, as it needs to acquire a
175
+ # per-user lock, for which it is cheaper to use in-memory linearizers
176
+ # than cross-worker locks.
177
+ hs.get_federation_registry().register_instances_for_edu(
178
+ EduTypes.DEVICE_LIST_UPDATE,
179
+ [self._main_device_list_writer],
180
+ )
181
+
182
+ self._delete_stale_devices_after = hs.config.server.delete_stale_devices_after
183
+
184
+ if (
185
+ hs.config.worker.run_background_tasks
186
+ and self._delete_stale_devices_after is not None
187
+ ):
188
+ self.clock.looping_call(
189
+ self.hs.run_as_background_process,
190
+ DELETE_STALE_DEVICES_INTERVAL,
191
+ desc="delete_stale_devices",
192
+ func=self._delete_stale_devices,
193
+ )
194
+
195
+ async def _delete_stale_devices(self) -> None:
196
+ """Background task that deletes devices which haven't been accessed for more than
197
+ a configured time period.
198
+ """
199
+ # We should only be running this job if the config option is defined.
200
+ assert self._delete_stale_devices_after is not None
201
+ now_ms = self.clock.time_msec()
202
+ since_ms = now_ms - self._delete_stale_devices_after
203
+ devices = await self.store.get_local_devices_not_accessed_since(since_ms)
204
+
205
+ for user_id, user_devices in devices.items():
206
+ await self.delete_devices(user_id, user_devices)
207
+
208
+ async def check_device_registered(
209
+ self,
210
+ user_id: str,
211
+ device_id: str | None,
212
+ initial_device_display_name: str | None = None,
213
+ auth_provider_id: str | None = None,
214
+ auth_provider_session_id: str | None = None,
215
+ ) -> str:
216
+ """
217
+ If the given device has not been registered, register it with the
218
+ supplied display name.
219
+
220
+ If no device_id is supplied, we make one up.
221
+
222
+ Args:
223
+ user_id: @user:id
224
+ device_id: device id supplied by client
225
+ initial_device_display_name: device display name from client
226
+ auth_provider_id: The SSO IdP the user used, if any.
227
+ auth_provider_session_id: The session ID (sid) got from the SSO IdP.
228
+ Returns:
229
+ device id (generated if none was supplied)
230
+ """
231
+
232
+ _check_device_name_length(initial_device_display_name)
233
+
234
+ # Check if we should send out device lists updates for this new device.
235
+ notify = user_id not in self._dont_notify_new_devices_for
236
+
237
+ if device_id is not None:
238
+ new_device = await self.store.store_device(
239
+ user_id=user_id,
240
+ device_id=device_id,
241
+ initial_device_display_name=initial_device_display_name,
242
+ auth_provider_id=auth_provider_id,
243
+ auth_provider_session_id=auth_provider_session_id,
244
+ )
245
+ if new_device:
246
+ if notify:
247
+ await self.notify_device_update(user_id, [device_id])
248
+ return device_id
249
+
250
+ # if the device id is not specified, we'll autogen one, but loop a few
251
+ # times in case of a clash.
252
+ attempts = 0
253
+ while attempts < 5:
254
+ new_device_id = stringutils.random_string(10).upper()
255
+ new_device = await self.store.store_device(
256
+ user_id=user_id,
257
+ device_id=new_device_id,
258
+ initial_device_display_name=initial_device_display_name,
259
+ auth_provider_id=auth_provider_id,
260
+ auth_provider_session_id=auth_provider_session_id,
261
+ )
262
+ if new_device:
263
+ if notify:
264
+ await self.notify_device_update(user_id, [new_device_id])
265
+ return new_device_id
266
+ attempts += 1
267
+
268
+ raise errors.StoreError(500, "Couldn't generate a device ID.")
269
+
270
+ @trace
271
+ async def delete_all_devices_for_user(
272
+ self, user_id: str, except_device_id: str | None = None
273
+ ) -> None:
274
+ """Delete all of the user's devices
275
+
276
+ Args:
277
+ user_id: The user to remove all devices from
278
+ except_device_id: optional device id which should not be deleted
279
+ """
280
+ device_map = await self.store.get_devices_by_user(user_id)
281
+ if except_device_id is not None:
282
+ device_map.pop(except_device_id, None)
283
+ user_device_ids = device_map.keys()
284
+ await self.delete_devices(user_id, user_device_ids)
285
+
286
+ async def delete_devices(self, user_id: str, device_ids: StrCollection) -> None:
287
+ """Delete several devices
288
+
289
+ Args:
290
+ user_id: The user to delete devices from.
291
+ device_ids: The list of device IDs to delete
292
+ """
293
+ to_device_stream_id = self._event_sources.get_current_token().to_device_key
294
+
295
+ try:
296
+ await self.store.delete_devices(user_id, device_ids)
297
+ except errors.StoreError as e:
298
+ if e.code == 404:
299
+ # no match
300
+ set_tag("error", True)
301
+ set_tag("reason", "User doesn't have that device id.")
302
+ else:
303
+ raise
304
+
305
+ # Delete data specific to each device. Not optimised as its an
306
+ # experimental MSC.
307
+ if self.hs.config.experimental.msc3890_enabled:
308
+ for device_id in device_ids:
309
+ # Remove any local notification settings for this device in accordance
310
+ # with MSC3890.
311
+ await self._account_data_handler.remove_account_data_for_user(
312
+ user_id,
313
+ f"org.matrix.msc3890.local_notification_settings.{device_id}",
314
+ )
315
+
316
+ # If we're deleting a lot of devices, a bunch of them may not have any
317
+ # to-device messages queued up. We filter those out to avoid scheduling
318
+ # unnecessary tasks.
319
+ devices_with_messages = await self.store.get_devices_with_messages(
320
+ user_id, device_ids
321
+ )
322
+ for device_id in devices_with_messages:
323
+ # Delete device messages asynchronously and in batches using the task scheduler
324
+ # We specify an upper stream id to avoid deleting non delivered messages
325
+ # if an user re-uses a device ID.
326
+ await self._task_scheduler.schedule_task(
327
+ DELETE_DEVICE_MSGS_TASK_NAME,
328
+ resource_id=device_id,
329
+ params={
330
+ "user_id": user_id,
331
+ "device_id": device_id,
332
+ "up_to_stream_id": to_device_stream_id,
333
+ },
334
+ )
335
+
336
+ await self._auth_handler.delete_access_tokens_for_devices(
337
+ user_id, device_ids=device_ids
338
+ )
339
+
340
+ # Pushers are deleted after `delete_access_tokens_for_user` is called so that
341
+ # modules using `on_logged_out` hook can use them if needed.
342
+ await self.hs.get_pusherpool().remove_pushers_by_devices(user_id, device_ids)
343
+
344
+ await self.notify_device_update(user_id, device_ids)
345
+
346
+ async def upsert_device(
347
+ self, user_id: str, device_id: str, display_name: str | None = None
348
+ ) -> bool:
349
+ """Create or update a device
350
+
351
+ Args:
352
+ user_id: The user to update devices of.
353
+ device_id: The device to update.
354
+ display_name: The new display name for this device.
355
+
356
+ Returns:
357
+ True if the device was created, False if it was updated.
358
+
359
+ """
360
+
361
+ # Reject a new displayname which is too long.
362
+ _check_device_name_length(display_name)
363
+
364
+ created = await self.store.store_device(
365
+ user_id,
366
+ device_id,
367
+ initial_device_display_name=display_name,
368
+ )
369
+
370
+ if not created:
371
+ await self.store.update_device(
372
+ user_id,
373
+ device_id,
374
+ new_display_name=display_name,
375
+ )
376
+
377
+ await self.notify_device_update(user_id, [device_id])
378
+ return created
379
+
380
+ async def update_device(self, user_id: str, device_id: str, content: dict) -> None:
381
+ """Update the given device
382
+
383
+ Args:
384
+ user_id: The user to update devices of.
385
+ device_id: The device to update.
386
+ content: body of update request
387
+ """
388
+
389
+ # Reject a new displayname which is too long.
390
+ new_display_name = content.get("display_name")
391
+
392
+ _check_device_name_length(new_display_name)
393
+
394
+ try:
395
+ await self.store.update_device(
396
+ user_id, device_id, new_display_name=new_display_name
397
+ )
398
+ await self.notify_device_update(user_id, [device_id])
399
+ except errors.StoreError as e:
400
+ if e.code == 404:
401
+ raise errors.NotFoundError()
402
+ else:
403
+ raise
404
+
405
+ @trace
406
+ async def get_devices_by_user(self, user_id: str) -> list[JsonDict]:
407
+ """
408
+ Retrieve the given user's devices
409
+
410
+ Args:
411
+ user_id: The user ID to query for devices.
412
+ Returns:
413
+ info on each device
414
+ """
415
+
416
+ set_tag("user_id", user_id)
417
+ device_map = await self.store.get_devices_by_user(user_id)
418
+
419
+ ips = await self.store.get_last_client_ip_by_device(user_id, device_id=None)
420
+
421
+ devices = list(device_map.values())
422
+ for device in devices:
423
+ _update_device_from_client_ips(device, ips)
424
+
425
+ log_kv(device_map)
426
+ return devices
427
+
428
+ async def get_dehydrated_device(self, user_id: str) -> tuple[str, JsonDict] | None:
429
+ """Retrieve the information for a dehydrated device.
430
+
431
+ Args:
432
+ user_id: the user whose dehydrated device we are looking for
433
+ Returns:
434
+ a tuple whose first item is the device ID, and the second item is
435
+ the dehydrated device information
436
+ """
437
+ return await self.store.get_dehydrated_device(user_id)
438
+
439
+ async def store_dehydrated_device(
440
+ self,
441
+ user_id: str,
442
+ device_id: str | None,
443
+ device_data: JsonDict,
444
+ initial_device_display_name: str | None = None,
445
+ keys_for_device: JsonDict | None = None,
446
+ ) -> str:
447
+ """Store a dehydrated device for a user, optionally storing the keys associated with
448
+ it as well. If the user had a previous dehydrated device, it is removed.
449
+
450
+ Args:
451
+ user_id: the user that we are storing the device for
452
+ device_id: device id supplied by client
453
+ device_data: the dehydrated device information
454
+ initial_device_display_name: The display name to use for the device
455
+ keys_for_device: keys for the dehydrated device
456
+ Returns:
457
+ device id of the dehydrated device
458
+ """
459
+ device_id = await self.check_device_registered(
460
+ user_id,
461
+ device_id,
462
+ initial_device_display_name,
463
+ )
464
+
465
+ time_now = self.clock.time_msec()
466
+
467
+ old_device_id = await self.store.store_dehydrated_device(
468
+ user_id, device_id, device_data, time_now, keys_for_device
469
+ )
470
+
471
+ if old_device_id is not None:
472
+ await self.delete_devices(user_id, [old_device_id])
473
+
474
+ return device_id
475
+
476
+ async def rehydrate_device(
477
+ self, user_id: str, access_token: str, device_id: str
478
+ ) -> dict:
479
+ """Process a rehydration request from the user.
480
+
481
+ Args:
482
+ user_id: the user who is rehydrating the device
483
+ access_token: the access token used for the request
484
+ device_id: the ID of the device that will be rehydrated
485
+ Returns:
486
+ a dict containing {"success": True}
487
+ """
488
+ success = await self.store.remove_dehydrated_device(user_id, device_id)
489
+
490
+ if not success:
491
+ raise errors.NotFoundError()
492
+
493
+ # If the dehydrated device was successfully deleted (the device ID
494
+ # matched the stored dehydrated device), then modify the access
495
+ # token and refresh token to use the dehydrated device's ID and
496
+ # copy the old device display name to the dehydrated device,
497
+ # and destroy the old device ID
498
+ old_device_id = await self.store.set_device_for_access_token(
499
+ access_token, device_id
500
+ )
501
+ await self.store.set_device_for_refresh_token(user_id, old_device_id, device_id)
502
+ old_device = await self.store.get_device(user_id, old_device_id)
503
+ if old_device is None:
504
+ raise errors.NotFoundError()
505
+ await self.store.update_device(user_id, device_id, old_device["display_name"])
506
+ # can't call self.delete_device because that will clobber the
507
+ # access token so call the storage layer directly
508
+ await self.store.delete_devices(user_id, [old_device_id])
509
+
510
+ # tell everyone that the old device is gone and that the dehydrated
511
+ # device has a new display name
512
+ await self.notify_device_update(user_id, [old_device_id, device_id])
513
+
514
+ return {"success": True}
515
+
516
+ async def delete_dehydrated_device(self, user_id: str, device_id: str) -> None:
517
+ """
518
+ Delete a stored dehydrated device.
519
+
520
+ Args:
521
+ user_id: the user_id to delete the device from
522
+ device_id: id of the dehydrated device to delete
523
+ """
524
+ success = await self.store.remove_dehydrated_device(user_id, device_id)
525
+
526
+ if not success:
527
+ raise errors.NotFoundError()
528
+
529
+ await self.delete_devices(user_id, [device_id])
530
+
531
+ @trace
532
+ async def get_device(self, user_id: str, device_id: str) -> JsonDict:
533
+ """Retrieve the given device
534
+
535
+ Args:
536
+ user_id: The user to get the device from
537
+ device_id: The device to fetch.
538
+
539
+ Returns:
540
+ info on the device
541
+ Raises:
542
+ errors.NotFoundError: if the device was not found
543
+ """
544
+ device = await self.store.get_device(user_id, device_id)
545
+ if device is None:
546
+ raise errors.NotFoundError()
547
+
548
+ ips = await self.store.get_last_client_ip_by_device(user_id, device_id)
549
+
550
+ device = dict(device)
551
+ _update_device_from_client_ips(device, ips)
552
+
553
+ set_tag("device", str(device))
554
+ set_tag("ips", str(ips))
555
+
556
+ return device
557
+
558
+ @cancellable
559
+ async def get_device_changes_in_shared_rooms(
560
+ self,
561
+ user_id: str,
562
+ room_ids: StrCollection,
563
+ from_token: StreamToken,
564
+ now_token: StreamToken | None = None,
565
+ ) -> set[str]:
566
+ """Get the set of users whose devices have changed who share a room with
567
+ the given user.
568
+ """
569
+ now_device_lists_key = self.store.get_device_stream_token()
570
+ if now_token:
571
+ now_device_lists_key = now_token.device_list_key
572
+
573
+ changed_users = await self.store.get_device_list_changes_in_rooms(
574
+ room_ids,
575
+ from_token.device_list_key,
576
+ now_device_lists_key,
577
+ )
578
+
579
+ if changed_users is not None:
580
+ # We also check if the given user has changed their device. If
581
+ # they're in no rooms then the above query won't include them.
582
+ changed = await self.store.get_users_whose_devices_changed(
583
+ from_token.device_list_key,
584
+ [user_id],
585
+ to_key=now_device_lists_key,
586
+ )
587
+ changed_users.update(changed)
588
+ return changed_users
589
+
590
+ # If the DB returned None then the `from_token` is too old, so we fall
591
+ # back on looking for device updates for all users.
592
+
593
+ users_who_share_room = await self.store.get_users_who_share_room_with_user(
594
+ user_id
595
+ )
596
+
597
+ tracked_users = set(users_who_share_room)
598
+
599
+ # Always tell the user about their own devices
600
+ tracked_users.add(user_id)
601
+
602
+ changed = await self.store.get_users_whose_devices_changed(
603
+ from_token.device_list_key,
604
+ tracked_users,
605
+ to_key=now_device_lists_key,
606
+ )
607
+
608
+ return changed
609
+
610
+ @trace
611
+ @cancellable
612
+ async def get_user_ids_changed(
613
+ self, user_id: str, from_token: StreamToken
614
+ ) -> DeviceListUpdates:
615
+ """Get list of users that have had the devices updated, or have newly
616
+ joined a room, that `user_id` may be interested in.
617
+ """
618
+
619
+ set_tag("user_id", user_id)
620
+ set_tag("from_token", str(from_token))
621
+
622
+ now_token = self._event_sources.get_current_token()
623
+
624
+ # We need to work out all the different membership changes for the user
625
+ # and user they share a room with, to pass to
626
+ # `generate_sync_entry_for_device_list`. See its docstring for details
627
+ # on the data required.
628
+
629
+ joined_room_ids = await self.store.get_rooms_for_user(user_id)
630
+
631
+ # Get the set of rooms that the user has joined/left
632
+ membership_changes = (
633
+ await self.store.get_current_state_delta_membership_changes_for_user(
634
+ user_id, from_key=from_token.room_key, to_key=now_token.room_key
635
+ )
636
+ )
637
+
638
+ # Check for newly joined or left rooms. We need to make sure that we add
639
+ # to newly joined in the case membership goes from join -> leave -> join
640
+ # again.
641
+ newly_joined_rooms: set[str] = set()
642
+ newly_left_rooms: set[str] = set()
643
+ for change in membership_changes:
644
+ # We check for changes in "joinedness", i.e. if the membership has
645
+ # changed to or from JOIN.
646
+ if change.membership == Membership.JOIN:
647
+ if change.prev_membership != Membership.JOIN:
648
+ newly_joined_rooms.add(change.room_id)
649
+ newly_left_rooms.discard(change.room_id)
650
+ elif change.prev_membership == Membership.JOIN:
651
+ newly_joined_rooms.discard(change.room_id)
652
+ newly_left_rooms.add(change.room_id)
653
+
654
+ # We now work out if any other users have since joined or left the rooms
655
+ # the user is currently in.
656
+
657
+ # List of membership changes per room
658
+ room_to_deltas: dict[str, list[StateDelta]] = {}
659
+ # The set of event IDs of membership events (so we can fetch their
660
+ # associated membership).
661
+ memberships_to_fetch: set[str] = set()
662
+
663
+ # TODO: Only pull out membership events?
664
+ state_changes = await self.store.get_current_state_deltas_for_rooms(
665
+ joined_room_ids, from_token=from_token.room_key, to_token=now_token.room_key
666
+ )
667
+ for delta in state_changes:
668
+ if delta.event_type != EventTypes.Member:
669
+ continue
670
+
671
+ room_to_deltas.setdefault(delta.room_id, []).append(delta)
672
+ if delta.event_id:
673
+ memberships_to_fetch.add(delta.event_id)
674
+ if delta.prev_event_id:
675
+ memberships_to_fetch.add(delta.prev_event_id)
676
+
677
+ # Fetch all the memberships for the membership events
678
+ event_id_to_memberships: Mapping[str, EventIdMembership | None] = {}
679
+ if memberships_to_fetch:
680
+ event_id_to_memberships = await self.store.get_membership_from_event_ids(
681
+ memberships_to_fetch
682
+ )
683
+
684
+ joined_invited_knocked = (
685
+ Membership.JOIN,
686
+ Membership.INVITE,
687
+ Membership.KNOCK,
688
+ )
689
+
690
+ # We now want to find any user that have newly joined/invited/knocked,
691
+ # or newly left, similarly to above.
692
+ newly_joined_or_invited_or_knocked_users: set[str] = set()
693
+ newly_left_users: set[str] = set()
694
+ for _, deltas in room_to_deltas.items():
695
+ for delta in deltas:
696
+ # Get the prev/new memberships for the delta
697
+ new_membership = None
698
+ prev_membership = None
699
+ if delta.event_id:
700
+ m = event_id_to_memberships.get(delta.event_id)
701
+ if m is not None:
702
+ new_membership = m.membership
703
+ if delta.prev_event_id:
704
+ m = event_id_to_memberships.get(delta.prev_event_id)
705
+ if m is not None:
706
+ prev_membership = m.membership
707
+
708
+ # Check if a user has newly joined/invited/knocked, or left.
709
+ if new_membership in joined_invited_knocked:
710
+ if prev_membership not in joined_invited_knocked:
711
+ newly_joined_or_invited_or_knocked_users.add(delta.state_key)
712
+ newly_left_users.discard(delta.state_key)
713
+ elif prev_membership in joined_invited_knocked:
714
+ newly_joined_or_invited_or_knocked_users.discard(delta.state_key)
715
+ newly_left_users.add(delta.state_key)
716
+
717
+ # Now we actually calculate the device list entry with the information
718
+ # calculated above.
719
+ device_list_updates = await self.generate_sync_entry_for_device_list(
720
+ user_id=user_id,
721
+ since_token=from_token,
722
+ now_token=now_token,
723
+ joined_room_ids=joined_room_ids,
724
+ newly_joined_rooms=newly_joined_rooms,
725
+ newly_joined_or_invited_or_knocked_users=newly_joined_or_invited_or_knocked_users,
726
+ newly_left_rooms=newly_left_rooms,
727
+ newly_left_users=newly_left_users,
728
+ )
729
+
730
+ log_kv(
731
+ {
732
+ "changed": device_list_updates.changed,
733
+ "left": device_list_updates.left,
734
+ }
735
+ )
736
+
737
+ return device_list_updates
738
+
739
+ async def generate_sync_entry_for_device_list(
740
+ self,
741
+ user_id: str,
742
+ since_token: StreamToken,
743
+ now_token: StreamToken,
744
+ joined_room_ids: AbstractSet[str],
745
+ newly_joined_rooms: AbstractSet[str],
746
+ newly_joined_or_invited_or_knocked_users: AbstractSet[str],
747
+ newly_left_rooms: AbstractSet[str],
748
+ newly_left_users: AbstractSet[str],
749
+ ) -> DeviceListUpdates:
750
+ """Generate the DeviceListUpdates section of sync
751
+
752
+ Args:
753
+ sync_result_builder
754
+ newly_joined_rooms: Set of rooms user has joined since previous sync
755
+ newly_joined_or_invited_or_knocked_users: Set of users that have joined,
756
+ been invited to a room or are knocking on a room since
757
+ previous sync.
758
+ newly_left_rooms: Set of rooms user has left since previous sync
759
+ newly_left_users: Set of users that have left a room we're in since
760
+ previous sync
761
+ """
762
+ # Take a copy since these fields will be mutated later.
763
+ newly_joined_or_invited_or_knocked_users = set(
764
+ newly_joined_or_invited_or_knocked_users
765
+ )
766
+ newly_left_users = set(newly_left_users)
767
+
768
+ # We want to figure out what user IDs the client should refetch
769
+ # device keys for, and which users we aren't going to track changes
770
+ # for anymore.
771
+ #
772
+ # For the first step we check:
773
+ # a. if any users we share a room with have updated their devices,
774
+ # and
775
+ # b. we also check if we've joined any new rooms, or if a user has
776
+ # joined a room we're in.
777
+ #
778
+ # For the second step we just find any users we no longer share a
779
+ # room with by looking at all users that have left a room plus users
780
+ # that were in a room we've left.
781
+
782
+ users_that_have_changed = set()
783
+
784
+ # Step 1a, check for changes in devices of users we share a room
785
+ # with
786
+ users_that_have_changed = await self.get_device_changes_in_shared_rooms(
787
+ user_id,
788
+ joined_room_ids,
789
+ from_token=since_token,
790
+ now_token=now_token,
791
+ )
792
+
793
+ # Step 1b, check for newly joined rooms
794
+ for room_id in newly_joined_rooms:
795
+ joined_users = await self.store.get_users_in_room(room_id)
796
+ newly_joined_or_invited_or_knocked_users.update(joined_users)
797
+
798
+ # TODO: Check that these users are actually new, i.e. either they
799
+ # weren't in the previous sync *or* they left and rejoined.
800
+ users_that_have_changed.update(newly_joined_or_invited_or_knocked_users)
801
+
802
+ user_signatures_changed = await self.store.get_users_whose_signatures_changed(
803
+ user_id, since_token.device_list_key
804
+ )
805
+ users_that_have_changed.update(user_signatures_changed)
806
+
807
+ # Now find users that we no longer track
808
+ for room_id in newly_left_rooms:
809
+ left_users = await self.store.get_users_in_room(room_id)
810
+ newly_left_users.update(left_users)
811
+
812
+ # Remove any users that we still share a room with.
813
+ left_users_rooms = await self.store.get_rooms_for_users(newly_left_users)
814
+ for user_id, entries in left_users_rooms.items():
815
+ if any(rid in joined_room_ids for rid in entries):
816
+ newly_left_users.discard(user_id)
817
+
818
+ return DeviceListUpdates(changed=users_that_have_changed, left=newly_left_users)
819
+
820
+ async def on_federation_query_user_devices(self, user_id: str) -> JsonDict:
821
+ if not self.hs.is_mine(UserID.from_string(user_id)):
822
+ raise SynapseError(400, "User is not hosted on this homeserver")
823
+
824
+ stream_id, devices = await self.store.get_e2e_device_keys_for_federation_query(
825
+ user_id
826
+ )
827
+ master_key = await self.store.get_e2e_cross_signing_key(user_id, "master")
828
+ self_signing_key = await self.store.get_e2e_cross_signing_key(
829
+ user_id, "self_signing"
830
+ )
831
+
832
+ # Check if the application services have any results.
833
+ if self._query_appservices_for_keys:
834
+ # Query the appservice for all devices for this user.
835
+ query: dict[str, list[str] | None] = {user_id: None}
836
+
837
+ # Query the appservices for any keys.
838
+ appservice_results = await self._appservice_handler.query_keys(query)
839
+
840
+ # Merge results, overriding anything from the database.
841
+ appservice_devices = appservice_results.get("device_keys", {}).get(
842
+ user_id, {}
843
+ )
844
+
845
+ # Filter the database results to only those devices that the appservice has
846
+ # *not* responded with.
847
+ devices = [d for d in devices if d["device_id"] not in appservice_devices]
848
+ # Append the appservice response by wrapping each result in another dictionary.
849
+ devices.extend(
850
+ {"device_id": device_id, "keys": device}
851
+ for device_id, device in appservice_devices.items()
852
+ )
853
+
854
+ # TODO Handle cross-signing keys.
855
+
856
+ return {
857
+ "user_id": user_id,
858
+ "stream_id": stream_id,
859
+ "devices": devices,
860
+ "master_key": master_key,
861
+ "self_signing_key": self_signing_key,
862
+ }
863
+
864
+ async def handle_room_un_partial_stated(self, room_id: str) -> None:
865
+ """Handles sending appropriate device list updates in a room that has
866
+ gone from partial to full state.
867
+ """
868
+
869
+ await self._handle_room_un_partial_stated_client(
870
+ instance_name=random.choice(self._device_list_writers),
871
+ room_id=room_id,
872
+ )
873
+
874
+ @trace
875
+ @measure_func("notify_device_update")
876
+ async def notify_device_update(
877
+ self, user_id: str, device_ids: StrCollection
878
+ ) -> None:
879
+ """Notify that a user's device(s) has changed. Pokes the notifier, and
880
+ remote servers if the user is local.
881
+
882
+ Args:
883
+ user_id: The Matrix ID of the user who's device list has been updated.
884
+ device_ids: The device IDs that have changed.
885
+ """
886
+ await self._notify_device_update_client(
887
+ instance_name=random.choice(self._device_list_writers),
888
+ user_id=user_id,
889
+ device_ids=list(device_ids),
890
+ )
891
+
892
+ async def notify_user_signature_update(
893
+ self,
894
+ from_user_id: str,
895
+ user_ids: list[str],
896
+ ) -> None:
897
+ """Notify a device writer that a user have made new signatures of other users.
898
+
899
+ Args:
900
+ from_user_id: The Matrix ID of the user who's signatures have been updated.
901
+ user_ids: The Matrix IDs of the users that have changed.
902
+ """
903
+ await self._notify_user_signature_update_client(
904
+ instance_name=random.choice(self._device_list_writers),
905
+ from_user_id=from_user_id,
906
+ user_ids=user_ids,
907
+ )
908
+
909
+ async def handle_new_device_update(self) -> None:
910
+ """Wake up a device writer to send local device list changes as federation outbound pokes."""
911
+ # This is only sent to the first device writer to avoid cross-worker
912
+ # locks in _handle_new_device_update_async, as it makes assumptions
913
+ # about being the only instance running.
914
+ await self._handle_new_device_update_client(
915
+ instance_name=self._device_list_writers[0],
916
+ )
917
+
918
+ DEVICE_MSGS_DELETE_BATCH_LIMIT = 1000
919
+ DEVICE_MSGS_DELETE_SLEEP = Duration(milliseconds=100)
920
+
921
+ async def _delete_device_messages(
922
+ self,
923
+ task: ScheduledTask,
924
+ ) -> tuple[TaskStatus, JsonMapping | None, str | None]:
925
+ """Scheduler task to delete device messages in batch of `DEVICE_MSGS_DELETE_BATCH_LIMIT`."""
926
+ assert task.params is not None
927
+ user_id = task.params["user_id"]
928
+ device_id = task.params["device_id"]
929
+ up_to_stream_id = task.params["up_to_stream_id"]
930
+
931
+ # Delete the messages in batches to avoid too much DB load.
932
+ from_stream_id = None
933
+ while True:
934
+ from_stream_id, _ = await self.store.delete_messages_for_device_between(
935
+ user_id=user_id,
936
+ device_id=device_id,
937
+ from_stream_id=from_stream_id,
938
+ to_stream_id=up_to_stream_id,
939
+ limit=DeviceWriterHandler.DEVICE_MSGS_DELETE_BATCH_LIMIT,
940
+ )
941
+
942
+ if from_stream_id is None:
943
+ return TaskStatus.COMPLETE, None, None
944
+
945
+ await self.clock.sleep(DeviceWriterHandler.DEVICE_MSGS_DELETE_SLEEP)
946
+
947
+
948
+ class DeviceWriterHandler(DeviceHandler):
949
+ """
950
+ Superclass of the DeviceHandler which gets instantiated on workers that can
951
+ write to the device list stream.
952
+ """
953
+
954
+ def __init__(self, hs: "HomeServer"):
955
+ super().__init__(hs)
956
+ self.server_name = hs.hostname # nb must be called this for @measure_func
957
+ self.hs = hs # nb must be called this for @wrap_as_background_process
958
+
959
+ # We only need to poke the federation sender explicitly if its on the
960
+ # same instance. Other federation sender instances will get notified by
961
+ # `synapse.app.generic_worker.FederationSenderHandler` when it sees it
962
+ # in the device lists stream.
963
+ self.federation_sender = None
964
+ if hs.should_send_federation():
965
+ self.federation_sender = hs.get_federation_sender()
966
+
967
+ self._storage_controllers = hs.get_storage_controllers()
968
+
969
+ # There are a few things that are only handled on the main device list
970
+ # writer to avoid cross-worker locks
971
+ #
972
+ # This mainly concerns the `DeviceListUpdater` class, which is only
973
+ # instantiated on the first device list writer.
974
+ self._is_main_device_list_writer = (
975
+ hs.get_instance_name() == self._main_device_list_writer
976
+ )
977
+
978
+ # Whether `_handle_new_device_update_async` is currently processing.
979
+ self._handle_new_device_update_is_processing = False
980
+
981
+ # If a new device update may have happened while the loop was
982
+ # processing.
983
+ self._handle_new_device_update_new_data = False
984
+
985
+ # Only the main device list writer handles device list EDUs and converts
986
+ # device list updates to outbound federation pokes. This allows us to
987
+ # use in-memory per-user locks instead of cross-worker locks, and
988
+ # simplifies the logic for converting outbound pokes. This makes the
989
+ # device_list writers a little bit unbalanced in terms of load, but
990
+ # still unlocks local device changes (and therefore login/logouts) when
991
+ # rolling-restarting Synapse.
992
+ if self._is_main_device_list_writer:
993
+ # On start up check if there are any updates pending.
994
+ hs.get_clock().call_when_running(self._handle_new_device_update_async)
995
+ self.device_list_updater = DeviceListUpdater(hs, self)
996
+ hs.get_federation_registry().register_edu_handler(
997
+ EduTypes.DEVICE_LIST_UPDATE,
998
+ self.device_list_updater.incoming_device_list_update,
999
+ )
1000
+
1001
+ @trace
1002
+ @measure_func("notify_device_update")
1003
+ async def notify_device_update(
1004
+ self, user_id: str, device_ids: StrCollection
1005
+ ) -> None:
1006
+ """Notify that a user's device(s) has changed. Pokes the notifier, and
1007
+ remote servers if the user is local.
1008
+
1009
+ Args:
1010
+ user_id: The Matrix ID of the user who's device list has been updated.
1011
+ device_ids: The device IDs that have changed.
1012
+ """
1013
+ if not device_ids:
1014
+ # No changes to notify about, so this is a no-op.
1015
+ return
1016
+
1017
+ room_ids = await self.store.get_rooms_for_user(user_id)
1018
+
1019
+ position = await self.store.add_device_change_to_streams(
1020
+ user_id,
1021
+ device_ids,
1022
+ room_ids=room_ids,
1023
+ )
1024
+
1025
+ if not position:
1026
+ # This should only happen if there are no updates, so we bail.
1027
+ return
1028
+
1029
+ if logger.isEnabledFor(logging.DEBUG):
1030
+ for device_id in device_ids:
1031
+ logger.debug(
1032
+ "Notifying about update %r/%r, ID: %r", user_id, device_id, position
1033
+ )
1034
+
1035
+ # specify the user ID too since the user should always get their own device list
1036
+ # updates, even if they aren't in any rooms.
1037
+ self.notifier.on_new_event(
1038
+ StreamKeyType.DEVICE_LIST, position, users={user_id}, rooms=room_ids
1039
+ )
1040
+
1041
+ # We may need to do some processing asynchronously for local user IDs.
1042
+ if self.hs.is_mine_id(user_id):
1043
+ await self.handle_new_device_update()
1044
+
1045
+ async def notify_user_signature_update(
1046
+ self, from_user_id: str, user_ids: list[str]
1047
+ ) -> None:
1048
+ """Notify a user that they have made new signatures of other users.
1049
+
1050
+ Args:
1051
+ from_user_id: the user who made the signature
1052
+ user_ids: the users IDs that have new signatures
1053
+ """
1054
+
1055
+ position = await self.store.add_user_signature_change_to_streams(
1056
+ from_user_id, user_ids
1057
+ )
1058
+
1059
+ self.notifier.on_new_event(
1060
+ StreamKeyType.DEVICE_LIST, position, users=[from_user_id]
1061
+ )
1062
+
1063
+ async def handle_new_device_update(self) -> None:
1064
+ # _handle_new_device_update_async is only called on the first device
1065
+ # writer, as it makes assumptions about only having one instance running
1066
+ # at a time. If this is not the first device writer, we defer to the
1067
+ # superclass, which will make the call go through replication.
1068
+ if not self._is_main_device_list_writer:
1069
+ return await super().handle_new_device_update()
1070
+
1071
+ self._handle_new_device_update_async()
1072
+ return
1073
+
1074
+ @wrap_as_background_process("_handle_new_device_update_async")
1075
+ async def _handle_new_device_update_async(self) -> None:
1076
+ """Called when we have a new local device list update that we need to
1077
+ send out over federation.
1078
+
1079
+ This happens in the background so as not to block the original request
1080
+ that generated the device update.
1081
+ """
1082
+ # This should only ever be called on the main device list writer, as it
1083
+ # expects to only have a single instance of this loop running at a time.
1084
+ # See `handle_new_device_update`.
1085
+ assert self._is_main_device_list_writer
1086
+
1087
+ if self._handle_new_device_update_is_processing:
1088
+ self._handle_new_device_update_new_data = True
1089
+ return
1090
+
1091
+ self._handle_new_device_update_is_processing = True
1092
+
1093
+ # Note that this logic only deals with the minimum stream ID, and not
1094
+ # the full stream token. This means that oubound pokes are only sent
1095
+ # once every writer on the device_lists stream has caught up. This is
1096
+ # fine, it may only introduces a bit of lag on the outbound pokes.
1097
+ # To fix this, 'device_lists_changes_converted_stream_position' would
1098
+ # need to include the full stream token instead of just a stream ID.
1099
+ # We could also consider have each writer converting their own device
1100
+ # list updates, but that can quickly become complex to handle changes in
1101
+ # the list of device writers.
1102
+
1103
+ # The stream ID we processed previous iteration (if any), and the set of
1104
+ # hosts we've already poked about for this update. This is so that we
1105
+ # don't poke the same remote server about the same update repeatedly.
1106
+ current_stream_id = None
1107
+ hosts_already_sent_to: set[str] = set()
1108
+
1109
+ try:
1110
+ stream_id, room_id = await self.store.get_device_change_last_converted_pos()
1111
+
1112
+ while True:
1113
+ self._handle_new_device_update_new_data = False
1114
+ max_stream_id = self.store.get_device_stream_token().stream
1115
+ rows = await self.store.get_uncoverted_outbound_room_pokes(
1116
+ stream_id, room_id
1117
+ )
1118
+ if not rows:
1119
+ # If the DB returned nothing then there is nothing left to
1120
+ # do, *unless* a new device list update happened during the
1121
+ # DB query.
1122
+
1123
+ # Advance `(stream_id, room_id)`.
1124
+ # `max_stream_id` comes from *before* the query for unconverted
1125
+ # rows, which means that any unconverted rows must have a larger
1126
+ # stream ID.
1127
+ if max_stream_id > stream_id:
1128
+ stream_id, room_id = max_stream_id, ""
1129
+ await self.store.set_device_change_last_converted_pos(
1130
+ stream_id, room_id
1131
+ )
1132
+ else:
1133
+ assert max_stream_id == stream_id
1134
+ # Avoid moving `room_id` backwards.
1135
+
1136
+ if self._handle_new_device_update_new_data:
1137
+ continue
1138
+ else:
1139
+ return
1140
+
1141
+ for user_id, device_id, room_id, stream_id, opentracing_context in rows:
1142
+ hosts = set()
1143
+
1144
+ # Ignore any users that aren't ours
1145
+ if self.hs.is_mine_id(user_id):
1146
+ hosts = set(
1147
+ await self._storage_controllers.state.get_current_hosts_in_room_or_partial_state_approximation(
1148
+ room_id
1149
+ )
1150
+ )
1151
+ hosts.discard(self.server_name)
1152
+ # For rooms with partial state, `hosts` is merely an
1153
+ # approximation. When we transition to a full state room, we
1154
+ # will have to send out device list updates to any servers we
1155
+ # missed.
1156
+
1157
+ # Check if we've already sent this update to some hosts
1158
+ if current_stream_id == stream_id:
1159
+ hosts -= hosts_already_sent_to
1160
+
1161
+ await self.store.add_device_list_outbound_pokes(
1162
+ user_id=user_id,
1163
+ device_id=device_id,
1164
+ room_id=room_id,
1165
+ hosts=hosts,
1166
+ context=opentracing_context,
1167
+ )
1168
+
1169
+ await self.store.mark_redundant_device_lists_pokes(
1170
+ user_id=user_id,
1171
+ device_id=device_id,
1172
+ room_id=room_id,
1173
+ converted_upto_stream_id=stream_id,
1174
+ )
1175
+
1176
+ # Notify replication that we've updated the device list stream.
1177
+ self.notifier.notify_replication()
1178
+
1179
+ if hosts and self.federation_sender:
1180
+ logger.info(
1181
+ "Sending device list update notif for %r to: %r",
1182
+ user_id,
1183
+ hosts,
1184
+ )
1185
+ await self.federation_sender.send_device_messages(
1186
+ hosts, immediate=False
1187
+ )
1188
+ # TODO: when called, this isn't in a logging context.
1189
+ # This leads to log spam, sentry event spam, and massive
1190
+ # memory usage.
1191
+ # See https://github.com/matrix-org/synapse/issues/12552.
1192
+ # log_kv(
1193
+ # {"message": "sent device update to host", "host": host}
1194
+ # )
1195
+
1196
+ if current_stream_id != stream_id:
1197
+ # Clear the set of hosts we've already sent to as we're
1198
+ # processing a new update.
1199
+ hosts_already_sent_to.clear()
1200
+
1201
+ hosts_already_sent_to.update(hosts)
1202
+ current_stream_id = stream_id
1203
+
1204
+ # Advance `(stream_id, room_id)`.
1205
+ _, _, room_id, stream_id, _ = rows[-1]
1206
+ await self.store.set_device_change_last_converted_pos(
1207
+ stream_id, room_id
1208
+ )
1209
+
1210
+ finally:
1211
+ self._handle_new_device_update_is_processing = False
1212
+
1213
+ async def handle_room_un_partial_stated(self, room_id: str) -> None:
1214
+ """Handles sending appropriate device list updates in a room that has
1215
+ gone from partial to full state.
1216
+ """
1217
+
1218
+ # We defer to the device list updater to handle pending remote device
1219
+ # list updates.
1220
+ await self.device_list_updater.handle_room_un_partial_stated(room_id)
1221
+
1222
+ # Replay local updates.
1223
+ (
1224
+ join_event_id,
1225
+ device_lists_stream_id,
1226
+ ) = await self.store.get_join_event_id_and_device_lists_stream_id_for_partial_state(
1227
+ room_id
1228
+ )
1229
+
1230
+ # Get the local device list changes that have happened in the room since
1231
+ # we started joining. If there are no updates there's nothing left to do.
1232
+ changes = await self.store.get_device_list_changes_in_room(
1233
+ room_id, device_lists_stream_id
1234
+ )
1235
+ local_changes = {(u, d) for u, d in changes if self.hs.is_mine_id(u)}
1236
+ if not local_changes:
1237
+ return
1238
+
1239
+ # Note: We have persisted the full state at this point, we just haven't
1240
+ # cleared the `partial_room` flag.
1241
+ join_state_ids = await self._state_storage.get_state_ids_for_event(
1242
+ join_event_id, await_full_state=False
1243
+ )
1244
+ current_state_ids = await self.store.get_partial_current_state_ids(room_id)
1245
+
1246
+ # Now we need to work out all servers that might have been in the room
1247
+ # at any point during our join.
1248
+
1249
+ # First we look for any membership states that have changed between the
1250
+ # initial join and now...
1251
+ all_keys = set(join_state_ids)
1252
+ all_keys.update(current_state_ids)
1253
+
1254
+ potentially_changed_hosts = set()
1255
+ for etype, state_key in all_keys:
1256
+ if etype != EventTypes.Member:
1257
+ continue
1258
+
1259
+ prev = join_state_ids.get((etype, state_key))
1260
+ current = current_state_ids.get((etype, state_key))
1261
+
1262
+ if prev != current:
1263
+ potentially_changed_hosts.add(get_domain_from_id(state_key))
1264
+
1265
+ # ... then we add all the hosts that are currently joined to the room...
1266
+ current_hosts_in_room = await self.store.get_current_hosts_in_room(room_id)
1267
+ potentially_changed_hosts.update(current_hosts_in_room)
1268
+
1269
+ # ... and finally we remove any hosts that we were told about, as we
1270
+ # will have sent device list updates to those hosts when they happened.
1271
+ known_hosts_at_join = await self.store.get_partial_state_servers_at_join(
1272
+ room_id
1273
+ )
1274
+ assert known_hosts_at_join is not None
1275
+ potentially_changed_hosts.difference_update(known_hosts_at_join)
1276
+
1277
+ potentially_changed_hosts.discard(self.server_name)
1278
+
1279
+ if not potentially_changed_hosts:
1280
+ # Nothing to do.
1281
+ return
1282
+
1283
+ logger.info(
1284
+ "Found %d changed hosts to send device list updates to",
1285
+ len(potentially_changed_hosts),
1286
+ )
1287
+
1288
+ for user_id, device_id in local_changes:
1289
+ await self.store.add_device_list_outbound_pokes(
1290
+ user_id=user_id,
1291
+ device_id=device_id,
1292
+ room_id=room_id,
1293
+ hosts=potentially_changed_hosts,
1294
+ context=None,
1295
+ )
1296
+
1297
+ # Notify things that device lists need to be sent out.
1298
+ self.notifier.notify_replication()
1299
+ if self.federation_sender:
1300
+ await self.federation_sender.send_device_messages(
1301
+ potentially_changed_hosts, immediate=False
1302
+ )
1303
+
1304
+
1305
+ def _update_device_from_client_ips(
1306
+ device: JsonDict, client_ips: Mapping[tuple[str, str], DeviceLastConnectionInfo]
1307
+ ) -> None:
1308
+ ip = client_ips.get((device["user_id"], device["device_id"]))
1309
+ device.update(
1310
+ {
1311
+ "last_seen_user_agent": ip.user_agent if ip else None,
1312
+ "last_seen_ts": ip.last_seen if ip else None,
1313
+ "last_seen_ip": ip.ip if ip else None,
1314
+ }
1315
+ )
1316
+
1317
+
1318
+ class DeviceListWorkerUpdater:
1319
+ "Handles incoming device list updates from federation and contacts the main device list writer over replication"
1320
+
1321
+ def __init__(self, hs: "HomeServer"):
1322
+ self.store = hs.get_datastores().main
1323
+ self._notifier = hs.get_notifier()
1324
+ # On which instance the DeviceListUpdater is running
1325
+ # Must be kept in sync with DeviceHandler
1326
+ self._main_device_list_writer = hs.config.worker.writers.device_lists[0]
1327
+ self._multi_user_device_resync_client = (
1328
+ ReplicationMultiUserDevicesResyncRestServlet.make_client(hs)
1329
+ )
1330
+
1331
+ async def multi_user_device_resync(
1332
+ self,
1333
+ user_ids: list[str],
1334
+ ) -> dict[str, JsonMapping | None]:
1335
+ """
1336
+ Like `user_device_resync` but operates on multiple users **from the same origin**
1337
+ at once.
1338
+
1339
+ Returns:
1340
+ Dict from User ID to the same Dict as `user_device_resync`.
1341
+ """
1342
+
1343
+ if not user_ids:
1344
+ # Shortcut empty requests
1345
+ return {}
1346
+
1347
+ # This uses a per-user-id lock; to avoid using cross-worker locks, we
1348
+ # forward the request to the main device list writer.
1349
+ # See DeviceListUpdater
1350
+ return await self._multi_user_device_resync_client(
1351
+ instance_name=self._main_device_list_writer,
1352
+ user_ids=user_ids,
1353
+ )
1354
+
1355
+ async def process_cross_signing_key_update(
1356
+ self,
1357
+ user_id: str,
1358
+ master_key: JsonDict | None,
1359
+ self_signing_key: JsonDict | None,
1360
+ ) -> list[str]:
1361
+ """Process the given new master and self-signing key for the given remote user.
1362
+
1363
+ Args:
1364
+ user_id: The ID of the user these keys are for.
1365
+ master_key: The dict of the cross-signing master key as returned by the
1366
+ remote server.
1367
+ self_signing_key: The dict of the cross-signing self-signing key as returned
1368
+ by the remote server.
1369
+
1370
+ Return:
1371
+ The device IDs for the given keys.
1372
+ """
1373
+ device_ids = []
1374
+
1375
+ current_keys_map = await self.store.get_e2e_cross_signing_keys_bulk([user_id])
1376
+ current_keys = current_keys_map.get(user_id) or {}
1377
+
1378
+ if master_key and master_key != current_keys.get("master"):
1379
+ await self.store.set_e2e_cross_signing_key(user_id, "master", master_key)
1380
+ _, verify_key = get_verify_key_from_cross_signing_key(master_key)
1381
+ # verify_key is a VerifyKey from signedjson, which uses
1382
+ # .version to denote the portion of the key ID after the
1383
+ # algorithm and colon, which is the device ID
1384
+ device_ids.append(verify_key.version)
1385
+ if self_signing_key and self_signing_key != current_keys.get("self_signing"):
1386
+ await self.store.set_e2e_cross_signing_key(
1387
+ user_id, "self_signing", self_signing_key
1388
+ )
1389
+ _, verify_key = get_verify_key_from_cross_signing_key(self_signing_key)
1390
+ device_ids.append(verify_key.version)
1391
+
1392
+ return device_ids
1393
+
1394
+ async def handle_room_un_partial_stated(self, room_id: str) -> None:
1395
+ """Handles sending appropriate device list updates in a room that has
1396
+ gone from partial to full state.
1397
+ """
1398
+
1399
+ pending_updates = (
1400
+ await self.store.get_pending_remote_device_list_updates_for_room(room_id)
1401
+ )
1402
+
1403
+ for user_id, device_id in pending_updates:
1404
+ logger.info(
1405
+ "Got pending device list update in room %s: %s / %s",
1406
+ room_id,
1407
+ user_id,
1408
+ device_id,
1409
+ )
1410
+ position = await self.store.add_device_change_to_streams(
1411
+ user_id,
1412
+ [device_id],
1413
+ room_ids=[room_id],
1414
+ )
1415
+
1416
+ if not position:
1417
+ # This should only happen if there are no updates, which
1418
+ # shouldn't happen when we've passed in a non-empty set of
1419
+ # device IDs.
1420
+ continue
1421
+
1422
+ self._notifier.on_new_event(
1423
+ StreamKeyType.DEVICE_LIST, position, rooms=[room_id]
1424
+ )
1425
+
1426
+
1427
+ class DeviceListUpdater(DeviceListWorkerUpdater):
1428
+ """Handles incoming device list updates from federation and updates the DB.
1429
+
1430
+ This is only instanciated on the first device list writer, as it uses
1431
+ in-process linearizers for some operations."""
1432
+
1433
+ def __init__(self, hs: "HomeServer", device_handler: DeviceWriterHandler):
1434
+ super().__init__(hs)
1435
+
1436
+ self.hs = hs
1437
+ self.federation = hs.get_federation_client()
1438
+ self.server_name = hs.hostname # nb must be called this for @measure_func
1439
+ self.clock = hs.get_clock() # nb must be called this for @measure_func
1440
+ self.device_handler = device_handler
1441
+
1442
+ self._remote_edu_linearizer = Linearizer(
1443
+ name="remote_device_list", clock=self.clock
1444
+ )
1445
+ self._resync_linearizer = Linearizer(
1446
+ name="remote_device_resync", clock=self.clock
1447
+ )
1448
+
1449
+ # user_id -> list of updates waiting to be handled.
1450
+ self._pending_updates: dict[
1451
+ str, list[tuple[str, str, Iterable[str], JsonDict]]
1452
+ ] = {}
1453
+
1454
+ # Recently seen stream ids. We don't bother keeping these in the DB,
1455
+ # but they're useful to have them about to reduce the number of spurious
1456
+ # resyncs.
1457
+ self._seen_updates: ExpiringCache[str, set[str]] = ExpiringCache(
1458
+ cache_name="device_update_edu",
1459
+ server_name=self.server_name,
1460
+ hs=self.hs,
1461
+ clock=self.clock,
1462
+ max_len=10000,
1463
+ expiry_ms=30 * 60 * 1000,
1464
+ iterable=True,
1465
+ )
1466
+
1467
+ # Attempt to resync out of sync device lists every 30s.
1468
+ self._resync_retry_lock = Lock()
1469
+ self.clock.looping_call(
1470
+ self.hs.run_as_background_process,
1471
+ Duration(seconds=30),
1472
+ func=self._maybe_retry_device_resync,
1473
+ desc="_maybe_retry_device_resync",
1474
+ )
1475
+
1476
+ @trace
1477
+ async def incoming_device_list_update(
1478
+ self, origin: str, edu_content: JsonDict
1479
+ ) -> None:
1480
+ """Called on incoming device list update from federation. Responsible
1481
+ for parsing the EDU and adding to pending updates list.
1482
+ """
1483
+
1484
+ set_tag("origin", origin)
1485
+ set_tag("edu_content", str(edu_content))
1486
+ user_id = edu_content.pop("user_id")
1487
+ device_id = edu_content.pop("device_id")
1488
+ stream_id = str(edu_content.pop("stream_id")) # They may come as ints
1489
+ prev_ids = edu_content.pop("prev_id", [])
1490
+ if not isinstance(prev_ids, list):
1491
+ raise SynapseError(
1492
+ 400, "Device list update had an invalid 'prev_ids' field"
1493
+ )
1494
+ prev_ids = [str(p) for p in prev_ids] # They may come as ints
1495
+
1496
+ if get_domain_from_id(user_id) != origin:
1497
+ # TODO: Raise?
1498
+ logger.warning(
1499
+ "Got device list update edu for %r/%r from %r",
1500
+ user_id,
1501
+ device_id,
1502
+ origin,
1503
+ )
1504
+
1505
+ set_tag("error", True)
1506
+ log_kv(
1507
+ {
1508
+ "message": "Got a device list update edu from a user and "
1509
+ "device which does not match the origin of the request.",
1510
+ "user_id": user_id,
1511
+ "device_id": device_id,
1512
+ }
1513
+ )
1514
+ return
1515
+
1516
+ # Check if we are partially joining any rooms. If so we need to store
1517
+ # all device list updates so that we can handle them correctly once we
1518
+ # know who is in the room.
1519
+ # TODO(faster_joins): this fetches and processes a bunch of data that we don't
1520
+ # use. Could be replaced by a tighter query e.g.
1521
+ # SELECT EXISTS(SELECT 1 FROM partial_state_rooms)
1522
+ partial_rooms = await self.store.get_partial_state_room_resync_info()
1523
+ if partial_rooms:
1524
+ await self.store.add_remote_device_list_to_pending(
1525
+ user_id,
1526
+ device_id,
1527
+ )
1528
+ self._notifier.notify_replication()
1529
+
1530
+ room_ids = await self.store.get_rooms_for_user(user_id)
1531
+ if not room_ids:
1532
+ # We don't share any rooms with this user. Ignore update, as we
1533
+ # probably won't get any further updates.
1534
+ set_tag("error", True)
1535
+ log_kv(
1536
+ {
1537
+ "message": "Got an update from a user for which "
1538
+ "we don't share any rooms",
1539
+ "other user_id": user_id,
1540
+ }
1541
+ )
1542
+ logger.warning(
1543
+ "Got device list update edu for %r/%r, but don't share a room",
1544
+ user_id,
1545
+ device_id,
1546
+ )
1547
+ return
1548
+
1549
+ logger.debug("Received device list update for %r/%r", user_id, device_id)
1550
+
1551
+ self._pending_updates.setdefault(user_id, []).append(
1552
+ (device_id, stream_id, prev_ids, edu_content)
1553
+ )
1554
+
1555
+ await self._handle_device_updates(user_id)
1556
+
1557
+ @measure_func("_incoming_device_list_update")
1558
+ async def _handle_device_updates(self, user_id: str) -> None:
1559
+ "Actually handle pending updates."
1560
+
1561
+ async with self._remote_edu_linearizer.queue(user_id):
1562
+ pending_updates = self._pending_updates.pop(user_id, [])
1563
+ if not pending_updates:
1564
+ # This can happen since we batch updates
1565
+ return
1566
+
1567
+ for device_id, stream_id, prev_ids, _ in pending_updates:
1568
+ logger.debug(
1569
+ "Handling update %r/%r, ID: %r, prev: %r ",
1570
+ user_id,
1571
+ device_id,
1572
+ stream_id,
1573
+ prev_ids,
1574
+ )
1575
+
1576
+ # Given a list of updates we check if we need to resync. This
1577
+ # happens if we've missed updates.
1578
+ resync = await self._need_to_do_resync(user_id, pending_updates)
1579
+
1580
+ if logger.isEnabledFor(logging.INFO):
1581
+ logger.info(
1582
+ "Received device list update for %s, requiring resync: %s. Devices: %s",
1583
+ user_id,
1584
+ resync,
1585
+ ", ".join(u[0] for u in pending_updates),
1586
+ )
1587
+
1588
+ if resync:
1589
+ # We mark as stale up front in case we get restarted.
1590
+ await self.store.mark_remote_users_device_caches_as_stale([user_id])
1591
+ self.hs.run_as_background_process(
1592
+ "_maybe_retry_device_resync",
1593
+ self.multi_user_device_resync,
1594
+ [user_id],
1595
+ False,
1596
+ )
1597
+ else:
1598
+ # Simply update the single device, since we know that is the only
1599
+ # change (because of the single prev_id matching the current cache)
1600
+ for device_id, stream_id, _, content in pending_updates:
1601
+ await self.store.update_remote_device_list_cache_entry(
1602
+ user_id, device_id, content, stream_id
1603
+ )
1604
+
1605
+ await self.device_handler.notify_device_update(
1606
+ user_id, [device_id for device_id, _, _, _ in pending_updates]
1607
+ )
1608
+
1609
+ self._seen_updates.setdefault(user_id, set()).update(
1610
+ stream_id for _, stream_id, _, _ in pending_updates
1611
+ )
1612
+
1613
+ async def _need_to_do_resync(
1614
+ self, user_id: str, updates: Iterable[tuple[str, str, Iterable[str], JsonDict]]
1615
+ ) -> bool:
1616
+ """Given a list of updates for a user figure out if we need to do a full
1617
+ resync, or whether we have enough data that we can just apply the delta.
1618
+ """
1619
+ seen_updates: set[str] = self._seen_updates.get(user_id, set())
1620
+
1621
+ extremity = await self.store.get_device_list_last_stream_id_for_remote(user_id)
1622
+
1623
+ logger.debug("Current extremity for %r: %r", user_id, extremity)
1624
+
1625
+ stream_id_in_updates = set() # stream_ids in updates list
1626
+ for _, stream_id, prev_ids, _ in updates:
1627
+ if not prev_ids:
1628
+ # We always do a resync if there are no previous IDs
1629
+ return True
1630
+
1631
+ for prev_id in prev_ids:
1632
+ if prev_id == extremity:
1633
+ continue
1634
+ elif prev_id in seen_updates:
1635
+ continue
1636
+ elif prev_id in stream_id_in_updates:
1637
+ continue
1638
+ else:
1639
+ return True
1640
+
1641
+ stream_id_in_updates.add(stream_id)
1642
+
1643
+ return False
1644
+
1645
+ @trace
1646
+ async def _maybe_retry_device_resync(self) -> None:
1647
+ """Retry to resync device lists that are out of sync, except if another retry is
1648
+ in progress.
1649
+ """
1650
+ # If the lock can not be acquired we want to always return immediately instead of blocking here
1651
+ if not self._resync_retry_lock.acquire(blocking=False):
1652
+ return
1653
+ try:
1654
+ # Get all of the users that need resyncing.
1655
+ need_resync = await self.store.get_user_ids_requiring_device_list_resync()
1656
+
1657
+ # Filter out users whose host is marked as "down" up front.
1658
+ hosts = await filter_destinations_by_retry_limiter(
1659
+ {get_domain_from_id(u) for u in need_resync}, self.clock, self.store
1660
+ )
1661
+ hosts = set(hosts)
1662
+
1663
+ # Iterate over the set of user IDs.
1664
+ for user_id in need_resync:
1665
+ if get_domain_from_id(user_id) not in hosts:
1666
+ continue
1667
+
1668
+ try:
1669
+ # Try to resync the current user's devices list.
1670
+ result = (await self.multi_user_device_resync([user_id], False))[
1671
+ user_id
1672
+ ]
1673
+
1674
+ # user_device_resync only returns a result if it managed to
1675
+ # successfully resync and update the database. Updating the table
1676
+ # of users requiring resync isn't necessary here as
1677
+ # user_device_resync already does it (through
1678
+ # self.store.update_remote_device_list_cache).
1679
+ if result:
1680
+ logger.debug(
1681
+ "Successfully resynced the device list for %s",
1682
+ user_id,
1683
+ )
1684
+ except Exception as e:
1685
+ # If there was an issue resyncing this user, e.g. if the remote
1686
+ # server sent a malformed result, just log the error instead of
1687
+ # aborting all the subsequent resyncs.
1688
+ logger.debug(
1689
+ "Could not resync the device list for %s: %s",
1690
+ user_id,
1691
+ e,
1692
+ )
1693
+ finally:
1694
+ self._resync_retry_lock.release()
1695
+
1696
+ async def multi_user_device_resync(
1697
+ self, user_ids: list[str], mark_failed_as_stale: bool = True
1698
+ ) -> dict[str, JsonMapping | None]:
1699
+ """
1700
+ Like `user_device_resync` but operates on multiple users **from the same origin**
1701
+ at once.
1702
+
1703
+ Returns:
1704
+ Dict from User ID to the same Dict as `user_device_resync`.
1705
+ """
1706
+ if not user_ids:
1707
+ return {}
1708
+
1709
+ origins = {UserID.from_string(user_id).domain for user_id in user_ids}
1710
+
1711
+ if len(origins) != 1:
1712
+ raise InvalidAPICallError(f"Only one origin permitted, got {origins!r}")
1713
+
1714
+ result = {}
1715
+ failed = set()
1716
+ # TODO(Perf): Actually batch these up
1717
+ for user_id in user_ids:
1718
+ async with self._resync_linearizer.queue(user_id):
1719
+ (
1720
+ user_result,
1721
+ user_failed,
1722
+ ) = await self._user_device_resync_returning_failed(user_id)
1723
+ result[user_id] = user_result
1724
+ if user_failed:
1725
+ failed.add(user_id)
1726
+
1727
+ if mark_failed_as_stale:
1728
+ await self.store.mark_remote_users_device_caches_as_stale(failed)
1729
+
1730
+ return result
1731
+
1732
+ async def _user_device_resync_returning_failed(
1733
+ self, user_id: str
1734
+ ) -> tuple[JsonMapping | None, bool]:
1735
+ """Fetches all devices for a user and updates the device cache with them.
1736
+
1737
+ Args:
1738
+ user_id: The user's id whose device_list will be updated.
1739
+ Returns:
1740
+ - A dict with device info as under the "devices" in the result of this
1741
+ request:
1742
+ https://matrix.org/docs/spec/server_server/r0.1.2#get-matrix-federation-v1-user-devices-userid
1743
+ None when we weren't able to fetch the device info for some reason,
1744
+ e.g. due to a connection problem.
1745
+ - True iff the resync failed and the device list should be marked as stale.
1746
+ """
1747
+ # Check that we haven't gone and fetched the devices since we last
1748
+ # checked if we needed to resync these device lists.
1749
+ if await self.store.get_users_whose_devices_are_cached([user_id]):
1750
+ cached = await self.store.get_cached_devices_for_user(user_id)
1751
+ return cached, False
1752
+
1753
+ logger.debug("Attempting to resync the device list for %s", user_id)
1754
+ log_kv({"message": "Doing resync to update device list."})
1755
+ # Fetch all devices for the user.
1756
+ origin = get_domain_from_id(user_id)
1757
+ try:
1758
+ result = await self.federation.query_user_devices(origin, user_id)
1759
+ except NotRetryingDestination:
1760
+ return None, True
1761
+ except (RequestSendFailed, HttpResponseException) as e:
1762
+ logger.warning(
1763
+ "Failed to handle device list update for %s: %s",
1764
+ user_id,
1765
+ e,
1766
+ )
1767
+
1768
+ # We abort on exceptions rather than accepting the update
1769
+ # as otherwise synapse will 'forget' that its device list
1770
+ # is out of date. If we bail then we will retry the resync
1771
+ # next time we get a device list update for this user_id.
1772
+ # This makes it more likely that the device lists will
1773
+ # eventually become consistent.
1774
+ return None, True
1775
+ except FederationDeniedError as e:
1776
+ set_tag("error", True)
1777
+ log_kv({"reason": "FederationDeniedError"})
1778
+ logger.info(e)
1779
+ return None, False
1780
+ except Exception as e:
1781
+ set_tag("error", True)
1782
+ log_kv(
1783
+ {"message": "Exception raised by federation request", "exception": e}
1784
+ )
1785
+ logger.exception("Failed to handle device list update for %s", user_id)
1786
+
1787
+ return None, True
1788
+ log_kv({"result": result})
1789
+ stream_id = result["stream_id"]
1790
+ devices = result["devices"]
1791
+
1792
+ # Get the master key and the self-signing key for this user if provided in the
1793
+ # response (None if not in the response).
1794
+ # The response will not contain the user signing key, as this key is only used by
1795
+ # its owner, thus it doesn't make sense to send it over federation.
1796
+ master_key = result.get("master_key")
1797
+ self_signing_key = result.get("self_signing_key")
1798
+
1799
+ ignore_devices = False
1800
+ # If the remote server has more than ~1000 devices for this user
1801
+ # we assume that something is going horribly wrong (e.g. a bot
1802
+ # that logs in and creates a new device every time it tries to
1803
+ # send a message). Maintaining lots of devices per user in the
1804
+ # cache can cause serious performance issues as if this request
1805
+ # takes more than 60s to complete, internal replication from the
1806
+ # inbound federation worker to the synapse master may time out
1807
+ # causing the inbound federation to fail and causing the remote
1808
+ # server to retry, causing a DoS. So in this scenario we give
1809
+ # up on storing the total list of devices and only handle the
1810
+ # delta instead.
1811
+ if len(devices) > 1000:
1812
+ logger.warning(
1813
+ "Ignoring device list snapshot for %s as it has >1K devs (%d)",
1814
+ user_id,
1815
+ len(devices),
1816
+ )
1817
+ devices = []
1818
+ ignore_devices = True
1819
+ else:
1820
+ prev_stream_id = await self.store.get_device_list_last_stream_id_for_remote(
1821
+ user_id
1822
+ )
1823
+ cached_devices = await self.store.get_cached_devices_for_user(user_id)
1824
+
1825
+ # To ensure that a user with no devices is cached, we skip the resync only
1826
+ # if we have a stream_id from previously writing a cache entry.
1827
+ if prev_stream_id is not None and cached_devices == {
1828
+ d["device_id"]: d for d in devices
1829
+ }:
1830
+ logger.info(
1831
+ "Skipping device list resync for %s, as our cache matches already",
1832
+ user_id,
1833
+ )
1834
+ devices = []
1835
+ ignore_devices = True
1836
+
1837
+ for device in devices:
1838
+ logger.debug(
1839
+ "Handling resync update %r/%r, ID: %r",
1840
+ user_id,
1841
+ device["device_id"],
1842
+ stream_id,
1843
+ )
1844
+
1845
+ if not ignore_devices:
1846
+ await self.store.update_remote_device_list_cache(
1847
+ user_id, devices, stream_id
1848
+ )
1849
+ # mark the cache as valid, whether or not we actually processed any device
1850
+ # list updates.
1851
+ await self.store.mark_remote_user_device_cache_as_valid(user_id)
1852
+ device_ids = [device["device_id"] for device in devices]
1853
+
1854
+ # Handle cross-signing keys.
1855
+ cross_signing_device_ids = await self.process_cross_signing_key_update(
1856
+ user_id,
1857
+ master_key,
1858
+ self_signing_key,
1859
+ )
1860
+ device_ids = device_ids + cross_signing_device_ids
1861
+
1862
+ if device_ids:
1863
+ await self.device_handler.notify_device_update(user_id, device_ids)
1864
+
1865
+ # We clobber the seen updates since we've re-synced from a given
1866
+ # point.
1867
+ self._seen_updates[user_id] = {stream_id}
1868
+
1869
+ return result, False