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,2589 @@
1
+ #
2
+ # This file is licensed under the Affero General Public License (AGPL) version 3.
3
+ #
4
+ # Copyright 2019 The Matrix.org Foundation C.I.C.
5
+ # Copyright 2017 Vector Creations Ltd
6
+ # Copyright 2014-2016 OpenMarket Ltd
7
+ # Copyright (C) 2023 New Vector, Ltd
8
+ #
9
+ # This program is free software: you can redistribute it and/or modify
10
+ # it under the terms of the GNU Affero General Public License as
11
+ # published by the Free Software Foundation, either version 3 of the
12
+ # License, or (at your option) any later version.
13
+ #
14
+ # See the GNU Affero General Public License for more details:
15
+ # <https://www.gnu.org/licenses/agpl-3.0.html>.
16
+ #
17
+ # Originally licensed under the Apache License, Version 2.0:
18
+ # <http://www.apache.org/licenses/LICENSE-2.0>.
19
+ #
20
+ # [This file includes modifications made by New Vector Limited]
21
+ #
22
+ #
23
+
24
+ """This module is responsible for getting events from the DB for pagination
25
+ and event streaming.
26
+
27
+ The order it returns events in depend on whether we are streaming forwards or
28
+ are paginating backwards. We do this because we want to handle out of order
29
+ messages nicely, while still returning them in the correct order when we
30
+ paginate bacwards.
31
+
32
+ This is implemented by keeping two ordering columns: stream_ordering and
33
+ topological_ordering. Stream ordering is basically insertion/received order
34
+ (except for events from backfill requests). The topological_ordering is a
35
+ weak ordering of events based on the pdu graph.
36
+
37
+ This means that we have to have two different types of tokens, depending on
38
+ what sort order was used:
39
+ - stream tokens are of the form: "s%d", which maps directly to the column
40
+ - topological tokems: "t%d-%d", where the integers map to the topological
41
+ and stream ordering columns respectively.
42
+ """
43
+
44
+ import logging
45
+ from typing import (
46
+ TYPE_CHECKING,
47
+ AbstractSet,
48
+ Any,
49
+ Collection,
50
+ Iterable,
51
+ Literal,
52
+ Mapping,
53
+ Protocol,
54
+ cast,
55
+ overload,
56
+ )
57
+
58
+ import attr
59
+ from typing_extensions import assert_never
60
+
61
+ from twisted.internet import defer
62
+
63
+ from synapse.api.constants import Direction, EventTypes, Membership
64
+ from synapse.api.filtering import Filter
65
+ from synapse.events import EventBase
66
+ from synapse.logging.context import make_deferred_yieldable, run_in_background
67
+ from synapse.logging.opentracing import tag_args, trace
68
+ from synapse.storage._base import SQLBaseStore
69
+ from synapse.storage.database import (
70
+ DatabasePool,
71
+ LoggingDatabaseConnection,
72
+ LoggingTransaction,
73
+ make_in_list_sql_clause,
74
+ )
75
+ from synapse.storage.databases.main.events_worker import EventsWorkerStore
76
+ from synapse.storage.engines import BaseDatabaseEngine, PostgresEngine, Sqlite3Engine
77
+ from synapse.storage.roommember import RoomsForUserStateReset
78
+ from synapse.storage.util.id_generators import MultiWriterIdGenerator
79
+ from synapse.types import PersistedEventPosition, RoomStreamToken, StrCollection
80
+ from synapse.util.caches.descriptors import cached, cachedList
81
+ from synapse.util.caches.stream_change_cache import StreamChangeCache
82
+ from synapse.util.cancellation import cancellable
83
+ from synapse.util.iterutils import batch_iter
84
+
85
+ if TYPE_CHECKING:
86
+ from synapse.server import HomeServer
87
+
88
+ logger = logging.getLogger(__name__)
89
+
90
+
91
+ MAX_STREAM_SIZE = 1000
92
+
93
+
94
+ _STREAM_TOKEN = "stream"
95
+ _TOPOLOGICAL_TOKEN = "topological"
96
+
97
+
98
+ class PaginateFunction(Protocol):
99
+ async def __call__(
100
+ self,
101
+ *,
102
+ room_id: str,
103
+ from_key: RoomStreamToken,
104
+ to_key: RoomStreamToken | None = None,
105
+ direction: Direction = Direction.BACKWARDS,
106
+ limit: int = 0,
107
+ ) -> tuple[list[EventBase], RoomStreamToken, bool]: ...
108
+
109
+
110
+ # Used as return values for pagination APIs
111
+ @attr.s(slots=True, frozen=True, auto_attribs=True)
112
+ class _EventDictReturn:
113
+ event_id: str
114
+ topological_ordering: int | None
115
+ stream_ordering: int
116
+
117
+
118
+ @attr.s(slots=True, frozen=True, auto_attribs=True)
119
+ class _EventsAround:
120
+ events_before: list[EventBase]
121
+ events_after: list[EventBase]
122
+ start: RoomStreamToken
123
+ end: RoomStreamToken
124
+
125
+
126
+ @attr.s(slots=True, frozen=True, auto_attribs=True)
127
+ class CurrentStateDeltaMembership:
128
+ """
129
+ Attributes:
130
+ event_id: The "current" membership event ID in this room.
131
+ event_pos: The position of the "current" membership event in the event stream.
132
+ prev_event_id: The previous membership event in this room that was replaced by
133
+ the "current" one. May be `None` if there was no previous membership event.
134
+ room_id: The room ID of the membership event.
135
+ membership: The membership state of the user in the room
136
+ sender: The person who sent the membership event
137
+ """
138
+
139
+ room_id: str
140
+ # Event
141
+ event_id: str | None
142
+ event_pos: PersistedEventPosition
143
+ membership: str
144
+ sender: str | None
145
+ # Prev event
146
+ prev_event_id: str | None
147
+ prev_event_pos: PersistedEventPosition | None
148
+ prev_membership: str | None
149
+ prev_sender: str | None
150
+
151
+
152
+ def generate_pagination_where_clause(
153
+ direction: Direction,
154
+ column_names: tuple[str, str],
155
+ from_token: tuple[int | None, int] | None,
156
+ to_token: tuple[int | None, int] | None,
157
+ engine: BaseDatabaseEngine,
158
+ ) -> str:
159
+ """Creates an SQL expression to bound the columns by the pagination
160
+ tokens.
161
+
162
+ For example creates an SQL expression like:
163
+
164
+ (6, 7) >= (topological_ordering, stream_ordering)
165
+ AND (5, 3) < (topological_ordering, stream_ordering)
166
+
167
+ would be generated for dir=b, from_token=(6, 7) and to_token=(5, 3).
168
+
169
+ Note that tokens are considered to be after the row they are in, e.g. if
170
+ a row A has a token T, then we consider A to be before T. This convention
171
+ is important when figuring out inequalities for the generated SQL, and
172
+ produces the following result:
173
+ - If paginating forwards then we exclude any rows matching the from
174
+ token, but include those that match the to token.
175
+ - If paginating backwards then we include any rows matching the from
176
+ token, but include those that match the to token.
177
+
178
+ Args:
179
+ direction: Whether we're paginating backwards or forwards.
180
+ column_names: The column names to bound. Must *not* be user defined as
181
+ these get inserted directly into the SQL statement without escapes.
182
+ from_token: The start point for the pagination. This is an exclusive
183
+ minimum bound if direction is forwards, and an inclusive maximum bound if
184
+ direction is backwards.
185
+ to_token: The endpoint point for the pagination. This is an inclusive
186
+ maximum bound if direction is forwards, and an exclusive minimum bound if
187
+ direction is backwards.
188
+ engine: The database engine to generate the clauses for
189
+
190
+ Returns:
191
+ The sql expression
192
+ """
193
+
194
+ where_clause = []
195
+ if from_token:
196
+ where_clause.append(
197
+ _make_generic_sql_bound(
198
+ bound=">=" if direction == Direction.BACKWARDS else "<",
199
+ column_names=column_names,
200
+ values=from_token,
201
+ engine=engine,
202
+ )
203
+ )
204
+
205
+ if to_token:
206
+ where_clause.append(
207
+ _make_generic_sql_bound(
208
+ bound="<" if direction == Direction.BACKWARDS else ">=",
209
+ column_names=column_names,
210
+ values=to_token,
211
+ engine=engine,
212
+ )
213
+ )
214
+
215
+ return " AND ".join(where_clause)
216
+
217
+
218
+ def generate_pagination_bounds(
219
+ direction: Direction,
220
+ from_token: RoomStreamToken | None,
221
+ to_token: RoomStreamToken | None,
222
+ ) -> tuple[str, tuple[int | None, int] | None, tuple[int | None, int] | None]:
223
+ """
224
+ Generate a start and end point for this page of events.
225
+
226
+ Args:
227
+ direction: Whether pagination is going forwards or backwards.
228
+ from_token: The token to start pagination at, or None to start at the first value.
229
+ to_token: The token to end pagination at, or None to not limit the end point.
230
+
231
+ Returns:
232
+ A three tuple of:
233
+
234
+ ASC or DESC for sorting of the query.
235
+
236
+ The starting position as a tuple of ints representing
237
+ (topological position, stream position) or None if no from_token was
238
+ provided. The topological position may be None for live tokens.
239
+
240
+ The end position in the same format as the starting position, or None
241
+ if no to_token was provided.
242
+ """
243
+
244
+ # Tokens really represent positions between elements, but we use
245
+ # the convention of pointing to the event before the gap. Hence
246
+ # we have a bit of asymmetry when it comes to equalities.
247
+ if direction == Direction.BACKWARDS:
248
+ order = "DESC"
249
+ else:
250
+ order = "ASC"
251
+
252
+ # The bounds for the stream tokens are complicated by the fact
253
+ # that we need to handle the instance_map part of the tokens. We do this
254
+ # by fetching all events between the min stream token and the maximum
255
+ # stream token (as returned by `RoomStreamToken.get_max_stream_pos`) and
256
+ # then filtering the results.
257
+ from_bound: tuple[int | None, int] | None = None
258
+ if from_token:
259
+ if from_token.topological is not None:
260
+ from_bound = from_token.as_historical_tuple()
261
+ elif direction == Direction.BACKWARDS:
262
+ from_bound = (
263
+ None,
264
+ from_token.get_max_stream_pos(),
265
+ )
266
+ else:
267
+ from_bound = (
268
+ None,
269
+ from_token.stream,
270
+ )
271
+
272
+ to_bound: tuple[int | None, int] | None = None
273
+ if to_token:
274
+ if to_token.topological is not None:
275
+ to_bound = to_token.as_historical_tuple()
276
+ elif direction == Direction.BACKWARDS:
277
+ to_bound = (
278
+ None,
279
+ to_token.stream,
280
+ )
281
+ else:
282
+ to_bound = (
283
+ None,
284
+ to_token.get_max_stream_pos(),
285
+ )
286
+
287
+ return order, from_bound, to_bound
288
+
289
+
290
+ def generate_next_token(
291
+ direction: Direction, last_topo_ordering: int | None, last_stream_ordering: int
292
+ ) -> RoomStreamToken:
293
+ """
294
+ Generate the next room stream token based on the currently returned data.
295
+
296
+ Args:
297
+ direction: Whether pagination is going forwards or backwards.
298
+ last_topo_ordering: The last topological ordering being returned.
299
+ last_stream_ordering: The last stream ordering being returned.
300
+
301
+ Returns:
302
+ A new RoomStreamToken to return to the client.
303
+ """
304
+ if direction == Direction.BACKWARDS:
305
+ # Tokens are positions between events.
306
+ # This token points *after* the last event in the chunk.
307
+ # We need it to point to the event before it in the chunk
308
+ # when we are going backwards so we subtract one from the
309
+ # stream part.
310
+ last_stream_ordering -= 1
311
+ return RoomStreamToken(topological=last_topo_ordering, stream=last_stream_ordering)
312
+
313
+
314
+ def _make_generic_sql_bound(
315
+ bound: str,
316
+ column_names: tuple[str, str],
317
+ values: tuple[int | None, int],
318
+ engine: BaseDatabaseEngine,
319
+ ) -> str:
320
+ """Create an SQL expression that bounds the given column names by the
321
+ values, e.g. create the equivalent of `(1, 2) < (col1, col2)`.
322
+
323
+ Only works with two columns.
324
+
325
+ Older versions of SQLite don't support that syntax so we have to expand it
326
+ out manually.
327
+
328
+ Args:
329
+ bound: The comparison operator to use. One of ">", "<", ">=",
330
+ "<=", where the values are on the left and columns on the right.
331
+ names: The column names. Must *not* be user defined
332
+ as these get inserted directly into the SQL statement without
333
+ escapes.
334
+ values: The values to bound the columns by. If
335
+ the first value is None then only creates a bound on the second
336
+ column.
337
+ engine: The database engine to generate the SQL for
338
+
339
+ Returns:
340
+ The SQL statement
341
+ """
342
+
343
+ assert bound in (">", "<", ">=", "<=")
344
+
345
+ name1, name2 = column_names
346
+ val1, val2 = values
347
+
348
+ if val1 is None:
349
+ val2 = int(val2)
350
+ return "(%d %s %s)" % (val2, bound, name2)
351
+
352
+ val1 = int(val1)
353
+ val2 = int(val2)
354
+
355
+ if isinstance(engine, PostgresEngine):
356
+ # Postgres doesn't optimise ``(x < a) OR (x=a AND y<b)`` as well
357
+ # as it optimises ``(x,y) < (a,b)`` on multicolumn indexes. So we
358
+ # use the later form when running against postgres.
359
+ return "((%d,%d) %s (%s,%s))" % (val1, val2, bound, name1, name2)
360
+
361
+ # We want to generate queries of e.g. the form:
362
+ #
363
+ # (val1 < name1 OR (val1 = name1 AND val2 <= name2))
364
+ #
365
+ # which is equivalent to (val1, val2) < (name1, name2)
366
+
367
+ return """(
368
+ {val1:d} {strict_bound} {name1}
369
+ OR ({val1:d} = {name1} AND {val2:d} {bound} {name2})
370
+ )""".format(
371
+ name1=name1,
372
+ val1=val1,
373
+ name2=name2,
374
+ val2=val2,
375
+ strict_bound=bound[0], # The first bound must always be strict equality here
376
+ bound=bound,
377
+ )
378
+
379
+
380
+ def _filter_results(
381
+ lower_token: RoomStreamToken | None,
382
+ upper_token: RoomStreamToken | None,
383
+ instance_name: str | None,
384
+ topological_ordering: int,
385
+ stream_ordering: int,
386
+ ) -> bool:
387
+ """Returns True if the event persisted by the given instance at the given
388
+ topological/stream_ordering falls between the two tokens (taking a None
389
+ token to mean unbounded).
390
+
391
+ Used to filter results from fetching events in the DB against the given
392
+ tokens. This is necessary to handle the case where the tokens include
393
+ position maps, which we handle by fetching more than necessary from the DB
394
+ and then filtering (rather than attempting to construct a complicated SQL
395
+ query).
396
+
397
+ The `instance_name` arg is optional to handle historic rows, and is
398
+ interpreted as if it was "master".
399
+ """
400
+
401
+ if instance_name is None:
402
+ instance_name = "master"
403
+
404
+ event_historical_tuple = (
405
+ topological_ordering,
406
+ stream_ordering,
407
+ )
408
+
409
+ if lower_token:
410
+ if lower_token.topological is not None:
411
+ # If these are historical tokens we compare the `(topological, stream)`
412
+ # tuples.
413
+ if event_historical_tuple <= lower_token.as_historical_tuple():
414
+ return False
415
+
416
+ else:
417
+ # If these are live tokens we compare the stream ordering against the
418
+ # writers stream position.
419
+ if stream_ordering <= lower_token.get_stream_pos_for_instance(
420
+ instance_name
421
+ ):
422
+ return False
423
+
424
+ if upper_token:
425
+ if upper_token.topological is not None:
426
+ if upper_token.as_historical_tuple() < event_historical_tuple:
427
+ return False
428
+ else:
429
+ if upper_token.get_stream_pos_for_instance(instance_name) < stream_ordering:
430
+ return False
431
+
432
+ return True
433
+
434
+
435
+ def _filter_results_by_stream(
436
+ lower_token: RoomStreamToken | None,
437
+ upper_token: RoomStreamToken | None,
438
+ instance_name: str | None,
439
+ stream_ordering: int,
440
+ ) -> bool:
441
+ """
442
+ This function only works with "live" tokens with `stream_ordering` only. See
443
+ `_filter_results(...)` if you want to work with all tokens.
444
+
445
+ Returns True if the event persisted by the given instance at the given
446
+ stream_ordering falls between the two tokens (taking a None
447
+ token to mean unbounded).
448
+
449
+ The token range is defined by > `lower_token` and <= `upper_token`.
450
+
451
+ Used to filter results from fetching events in the DB against the given
452
+ tokens. This is necessary to handle the case where the tokens include
453
+ position maps, which we handle by fetching more than necessary from the DB
454
+ and then filtering (rather than attempting to construct a complicated SQL
455
+ query).
456
+
457
+ The `instance_name` arg is optional to handle historic rows, and is
458
+ interpreted as if it was "master".
459
+ """
460
+ if instance_name is None:
461
+ instance_name = "master"
462
+
463
+ if lower_token:
464
+ assert lower_token.topological is None
465
+
466
+ # If these are live tokens we compare the stream ordering against the
467
+ # writers stream position.
468
+ if stream_ordering <= lower_token.get_stream_pos_for_instance(instance_name):
469
+ return False
470
+
471
+ if upper_token:
472
+ assert upper_token.topological is None
473
+
474
+ if upper_token.get_stream_pos_for_instance(instance_name) < stream_ordering:
475
+ return False
476
+
477
+ return True
478
+
479
+
480
+ def filter_to_clause(event_filter: Filter | None) -> tuple[str, list[str]]:
481
+ # NB: This may create SQL clauses that don't optimise well (and we don't
482
+ # have indices on all possible clauses). E.g. it may create
483
+ # "room_id == X AND room_id != X", which postgres doesn't optimise.
484
+
485
+ if not event_filter:
486
+ return "", []
487
+
488
+ clauses = []
489
+ args = []
490
+
491
+ # Handle event types with potential wildcard characters
492
+ if event_filter.types:
493
+ type_clauses = []
494
+ for typ in event_filter.types:
495
+ if "*" in typ:
496
+ type_clauses.append("event.type LIKE ?")
497
+ typ = typ.replace("*", "%") # Replace * with % for SQL LIKE pattern
498
+ else:
499
+ type_clauses.append("event.type = ?")
500
+ args.append(typ)
501
+ clauses.append("(%s)" % " OR ".join(type_clauses))
502
+
503
+ # Handle event types to exclude with potential wildcard characters
504
+ for typ in event_filter.not_types:
505
+ if "*" in typ:
506
+ clauses.append("event.type NOT LIKE ?")
507
+ typ = typ.replace("*", "%")
508
+ else:
509
+ clauses.append("event.type != ?")
510
+ args.append(typ)
511
+
512
+ if event_filter.senders:
513
+ clauses.append(
514
+ "(%s)" % " OR ".join("event.sender = ?" for _ in event_filter.senders)
515
+ )
516
+ args.extend(event_filter.senders)
517
+
518
+ for sender in event_filter.not_senders:
519
+ clauses.append("event.sender != ?")
520
+ args.append(sender)
521
+
522
+ if event_filter.rooms:
523
+ clauses.append(
524
+ "(%s)" % " OR ".join("event.room_id = ?" for _ in event_filter.rooms)
525
+ )
526
+ args.extend(event_filter.rooms)
527
+
528
+ for room_id in event_filter.not_rooms:
529
+ clauses.append("event.room_id != ?")
530
+ args.append(room_id)
531
+
532
+ if event_filter.contains_url:
533
+ clauses.append("event.contains_url = ?")
534
+ args.append(event_filter.contains_url)
535
+
536
+ # We're only applying the "labels" filter on the database query, because applying the
537
+ # "not_labels" filter via a SQL query is non-trivial. Instead, we let
538
+ # event_filter.check_fields apply it, which is not as efficient but makes the
539
+ # implementation simpler.
540
+ if event_filter.labels:
541
+ clauses.append("(%s)" % " OR ".join("label = ?" for _ in event_filter.labels))
542
+ args.extend(event_filter.labels)
543
+
544
+ # Filter on relation_senders / relation types from the joined tables.
545
+ if event_filter.related_by_senders:
546
+ clauses.append(
547
+ "(%s)"
548
+ % " OR ".join(
549
+ "related_event.sender = ?" for _ in event_filter.related_by_senders
550
+ )
551
+ )
552
+ args.extend(event_filter.related_by_senders)
553
+
554
+ if event_filter.related_by_rel_types:
555
+ clauses.append(
556
+ "(%s)"
557
+ % " OR ".join(
558
+ "relation_type = ?" for _ in event_filter.related_by_rel_types
559
+ )
560
+ )
561
+ args.extend(event_filter.related_by_rel_types)
562
+
563
+ if event_filter.rel_types:
564
+ clauses.append(
565
+ "(%s)"
566
+ % " OR ".join(
567
+ "event_relation.relation_type = ?" for _ in event_filter.rel_types
568
+ )
569
+ )
570
+ args.extend(event_filter.rel_types)
571
+
572
+ if event_filter.not_rel_types:
573
+ clauses.append(
574
+ "((%s) OR event_relation.relation_type IS NULL)"
575
+ % " AND ".join(
576
+ "event_relation.relation_type != ?" for _ in event_filter.not_rel_types
577
+ )
578
+ )
579
+ args.extend(event_filter.not_rel_types)
580
+
581
+ return " AND ".join(clauses), args
582
+
583
+
584
+ class StreamWorkerStore(EventsWorkerStore, SQLBaseStore):
585
+ def __init__(
586
+ self,
587
+ database: DatabasePool,
588
+ db_conn: LoggingDatabaseConnection,
589
+ hs: "HomeServer",
590
+ ):
591
+ super().__init__(database, db_conn, hs)
592
+
593
+ self._instance_name = hs.get_instance_name()
594
+ self._send_federation = hs.should_send_federation()
595
+ self._federation_shard_config = hs.config.worker.federation_shard_config
596
+
597
+ # If we're a process that sends federation we may need to reset the
598
+ # `federation_stream_position` table to match the current sharding
599
+ # config. We don't do this now as otherwise two processes could conflict
600
+ # during startup which would cause one to die.
601
+ self._need_to_reset_federation_stream_positions = self._send_federation
602
+
603
+ events_max = self.get_room_max_stream_ordering()
604
+ event_cache_prefill, min_event_val = self.db_pool.get_cache_dict(
605
+ db_conn,
606
+ "events",
607
+ entity_column="room_id",
608
+ stream_column="stream_ordering",
609
+ max_value=events_max,
610
+ )
611
+ self._events_stream_cache = StreamChangeCache(
612
+ name="EventsRoomStreamChangeCache",
613
+ server_name=self.server_name,
614
+ current_stream_pos=min_event_val,
615
+ prefilled_cache=event_cache_prefill,
616
+ )
617
+ self._membership_stream_cache = StreamChangeCache(
618
+ name="MembershipStreamChangeCache",
619
+ server_name=self.server_name,
620
+ current_stream_pos=events_max,
621
+ )
622
+
623
+ self._stream_order_on_start = self.get_room_max_stream_ordering()
624
+ self._min_stream_order_on_start = self.get_room_min_stream_ordering()
625
+
626
+ def get_room_max_stream_ordering(self) -> int:
627
+ """Get the stream_ordering of regular events that we have committed up to
628
+
629
+ Returns the maximum stream id such that all stream ids less than or
630
+ equal to it have been successfully persisted.
631
+ """
632
+ return self._stream_id_gen.get_current_token()
633
+
634
+ def get_room_min_stream_ordering(self) -> int:
635
+ """Get the stream_ordering of backfilled events that we have committed up to
636
+
637
+ Backfilled events use *negative* stream orderings, so this returns the
638
+ minimum negative stream id such that all stream ids greater than or
639
+ equal to it have been successfully persisted.
640
+ """
641
+ return self._backfill_id_gen.get_current_token()
642
+
643
+ def get_room_max_token(self) -> RoomStreamToken:
644
+ """Get a `RoomStreamToken` that marks the current maximum persisted
645
+ position of the events stream. Useful to get a token that represents
646
+ "now".
647
+
648
+ The token returned is a "live" token that may have an instance_map
649
+ component.
650
+ """
651
+
652
+ return RoomStreamToken.from_generator(self._stream_id_gen)
653
+
654
+ def get_events_stream_id_generator(self) -> MultiWriterIdGenerator:
655
+ return self._stream_id_gen
656
+
657
+ async def get_room_events_stream_for_rooms(
658
+ self,
659
+ *,
660
+ room_ids: Collection[str],
661
+ from_key: RoomStreamToken,
662
+ to_key: RoomStreamToken | None = None,
663
+ direction: Direction = Direction.BACKWARDS,
664
+ limit: int = 0,
665
+ ) -> dict[str, tuple[list[EventBase], RoomStreamToken, bool]]:
666
+ """Get new room events in stream ordering since `from_key`.
667
+
668
+ Args:
669
+ room_ids
670
+ from_key: The token to stream from (starting point and heading in the given
671
+ direction)
672
+ to_key: The token representing the end stream position (end point)
673
+ limit: Maximum number of events to return
674
+ direction: Indicates whether we are paginating forwards or backwards
675
+ from `from_key`.
676
+
677
+ Returns:
678
+ A map from room id to a tuple containing:
679
+ - list of recent events in the room
680
+ - stream ordering key for the start of the chunk of events returned.
681
+ - a boolean to indicate if there were more events but we hit the limit
682
+
683
+ When Direction.FORWARDS: from_key < x <= to_key, (ascending order)
684
+ When Direction.BACKWARDS: from_key >= x > to_key, (descending order)
685
+ """
686
+ if direction == Direction.FORWARDS:
687
+ room_ids = self._events_stream_cache.get_entities_changed(
688
+ room_ids, from_key.stream
689
+ )
690
+ elif direction == Direction.BACKWARDS:
691
+ if to_key is not None:
692
+ room_ids = self._events_stream_cache.get_entities_changed(
693
+ room_ids, to_key.stream
694
+ )
695
+ else:
696
+ assert_never(direction)
697
+
698
+ if not room_ids:
699
+ return {}
700
+
701
+ results = {}
702
+ room_ids = list(room_ids)
703
+ for rm_ids in (room_ids[i : i + 20] for i in range(0, len(room_ids), 20)):
704
+ res = await make_deferred_yieldable(
705
+ defer.gatherResults(
706
+ [
707
+ run_in_background(
708
+ self.paginate_room_events_by_stream_ordering,
709
+ room_id=room_id,
710
+ from_key=from_key,
711
+ to_key=to_key,
712
+ direction=direction,
713
+ limit=limit,
714
+ )
715
+ for room_id in rm_ids
716
+ ],
717
+ consumeErrors=True,
718
+ )
719
+ )
720
+ results.update(dict(zip(rm_ids, res)))
721
+
722
+ return results
723
+
724
+ def get_rooms_that_changed(
725
+ self, room_ids: Collection[str], from_key: RoomStreamToken
726
+ ) -> set[str]:
727
+ """Given a list of rooms and a token, return rooms where there may have
728
+ been changes.
729
+ """
730
+ from_id = from_key.stream
731
+ return {
732
+ room_id
733
+ for room_id in room_ids
734
+ if self._events_stream_cache.has_entity_changed(room_id, from_id)
735
+ }
736
+
737
+ async def get_rooms_that_have_updates_since_sliding_sync_table(
738
+ self,
739
+ room_ids: StrCollection,
740
+ from_key: RoomStreamToken,
741
+ ) -> StrCollection:
742
+ """Return the rooms that probably have had updates since the given
743
+ token (changes that are > `from_key`).
744
+
745
+ May return false positives, but must not return false negatives.
746
+
747
+ If `have_finished_sliding_sync_background_jobs` is False, then we return
748
+ all the room IDs, as we can't be sure that the sliding sync table is
749
+ fully populated.
750
+ """
751
+ # If the stream change cache is valid for the stream token, we can just
752
+ # use the result of that.
753
+ if from_key.stream >= self._events_stream_cache.get_earliest_known_position():
754
+ return self._events_stream_cache.get_entities_changed(
755
+ room_ids, from_key.stream
756
+ )
757
+
758
+ if not self.have_finished_sliding_sync_background_jobs():
759
+ # If the table hasn't been populated yet, we have to assume all rooms
760
+ # have updates.
761
+ return room_ids
762
+
763
+ def get_rooms_that_have_updates_since_sliding_sync_table_txn(
764
+ txn: LoggingTransaction,
765
+ ) -> StrCollection:
766
+ sql = """
767
+ SELECT room_id
768
+ FROM sliding_sync_joined_rooms
769
+ WHERE {clause}
770
+ AND event_stream_ordering > ?
771
+ """
772
+
773
+ results: set[str] = set()
774
+ for batch in batch_iter(room_ids, 1000):
775
+ clause, args = make_in_list_sql_clause(
776
+ self.database_engine, "room_id", batch
777
+ )
778
+
779
+ args.append(from_key.stream)
780
+ txn.execute(sql.format(clause=clause), args)
781
+
782
+ results.update(row[0] for row in txn)
783
+
784
+ return results
785
+
786
+ return await self.db_pool.runInteraction(
787
+ "get_rooms_that_have_updates_since_sliding_sync_table",
788
+ get_rooms_that_have_updates_since_sliding_sync_table_txn,
789
+ )
790
+
791
+ async def paginate_room_events_by_stream_ordering(
792
+ self,
793
+ *,
794
+ room_id: str,
795
+ from_key: RoomStreamToken,
796
+ to_key: RoomStreamToken | None = None,
797
+ direction: Direction = Direction.BACKWARDS,
798
+ limit: int = 0,
799
+ ) -> tuple[list[EventBase], RoomStreamToken, bool]:
800
+ """
801
+ Paginate events by `stream_ordering` in the room from the `from_key` in the
802
+ given `direction` to the `to_key` or `limit`.
803
+
804
+ Args:
805
+ room_id
806
+ from_key: The token to stream from (starting point and heading in the given
807
+ direction)
808
+ to_key: The token representing the end stream position (end point)
809
+ direction: Indicates whether we are paginating forwards or backwards
810
+ from `from_key`.
811
+ limit: Maximum number of events to return
812
+
813
+ Returns:
814
+ The results as a list of events, a token that points to the end of
815
+ the result set, and a boolean to indicate if there were more events
816
+ but we hit the limit. If no events are returned then the end of the
817
+ stream has been reached (i.e. there are no events between `from_key`
818
+ and `to_key`).
819
+
820
+ When Direction.FORWARDS: from_key < x <= to_key, (ascending order)
821
+ When Direction.BACKWARDS: from_key >= x > to_key, (descending order)
822
+ """
823
+
824
+ # FIXME: When going forwards, we should enforce that the `to_key` is not `None`
825
+ # because we always need an upper bound when querying the events stream (as
826
+ # otherwise we'll potentially pick up events that are not fully persisted).
827
+
828
+ # We should only be working with `stream_ordering` tokens here
829
+ assert from_key is None or from_key.topological is None
830
+ assert to_key is None or to_key.topological is None
831
+
832
+ # We can bail early if we're looking forwards, and our `to_key` is already
833
+ # before our `from_key`.
834
+ if (
835
+ direction == Direction.FORWARDS
836
+ and to_key is not None
837
+ and to_key.is_before_or_eq(from_key)
838
+ ):
839
+ # Token selection matches what we do below if there are no rows
840
+ return [], to_key if to_key else from_key, False
841
+ # Or vice-versa, if we're looking backwards and our `from_key` is already before
842
+ # our `to_key`.
843
+ elif (
844
+ direction == Direction.BACKWARDS
845
+ and to_key is not None
846
+ and from_key.is_before_or_eq(to_key)
847
+ ):
848
+ # Token selection matches what we do below if there are no rows
849
+ return [], to_key if to_key else from_key, False
850
+
851
+ # We can do a quick sanity check to see if any events have been sent in the room
852
+ # since the earlier token.
853
+ has_changed = True
854
+ if direction == Direction.FORWARDS:
855
+ has_changed = self._events_stream_cache.has_entity_changed(
856
+ room_id, from_key.stream
857
+ )
858
+ elif direction == Direction.BACKWARDS:
859
+ if to_key is not None:
860
+ has_changed = self._events_stream_cache.has_entity_changed(
861
+ room_id, to_key.stream
862
+ )
863
+ else:
864
+ assert_never(direction)
865
+
866
+ if not has_changed:
867
+ # Token selection matches what we do below if there are no rows
868
+ return [], to_key if to_key else from_key, False
869
+
870
+ order, from_bound, to_bound = generate_pagination_bounds(
871
+ direction, from_key, to_key
872
+ )
873
+
874
+ bounds = generate_pagination_where_clause(
875
+ direction=direction,
876
+ # The empty string will shortcut downstream code to only use the
877
+ # `stream_ordering` column
878
+ column_names=("", "stream_ordering"),
879
+ from_token=from_bound,
880
+ to_token=to_bound,
881
+ engine=self.database_engine,
882
+ )
883
+
884
+ def f(txn: LoggingTransaction) -> tuple[list[_EventDictReturn], bool]:
885
+ sql = f"""
886
+ SELECT event_id, instance_name, stream_ordering
887
+ FROM events
888
+ WHERE
889
+ room_id = ?
890
+ AND not outlier
891
+ AND {bounds}
892
+ ORDER BY stream_ordering {order} LIMIT ?
893
+ """
894
+ txn.execute(sql, (room_id, 2 * limit))
895
+
896
+ # Get all the rows and check if we hit the limit.
897
+ fetched_rows = txn.fetchall()
898
+ limited = len(fetched_rows) >= 2 * limit
899
+
900
+ rows = [
901
+ _EventDictReturn(event_id, None, stream_ordering)
902
+ for event_id, instance_name, stream_ordering in fetched_rows
903
+ if _filter_results_by_stream(
904
+ lower_token=(
905
+ to_key if direction == Direction.BACKWARDS else from_key
906
+ ),
907
+ upper_token=(
908
+ from_key if direction == Direction.BACKWARDS else to_key
909
+ ),
910
+ instance_name=instance_name,
911
+ stream_ordering=stream_ordering,
912
+ )
913
+ ]
914
+
915
+ if len(rows) > limit:
916
+ limited = True
917
+
918
+ rows = rows[:limit]
919
+ return rows, limited
920
+
921
+ rows, limited = await self.db_pool.runInteraction(
922
+ "get_room_events_stream_for_room", f
923
+ )
924
+
925
+ ret = await self.get_events_as_list(
926
+ [r.event_id for r in rows], get_prev_content=True
927
+ )
928
+
929
+ if rows:
930
+ next_key = generate_next_token(
931
+ direction=direction,
932
+ last_topo_ordering=None,
933
+ last_stream_ordering=rows[-1].stream_ordering,
934
+ )
935
+ else:
936
+ # TODO (erikj): We should work out what to do here instead. (same as
937
+ # `_paginate_room_events_by_topological_ordering_txn(...)`)
938
+ next_key = to_key if to_key else from_key
939
+
940
+ return ret, next_key, limited
941
+
942
+ @trace
943
+ async def get_current_state_delta_membership_changes_for_user(
944
+ self,
945
+ user_id: str,
946
+ from_key: RoomStreamToken,
947
+ to_key: RoomStreamToken,
948
+ excluded_room_ids: list[str] | None = None,
949
+ ) -> list[CurrentStateDeltaMembership]:
950
+ """
951
+ Fetch membership events (and the previous event that was replaced by that one)
952
+ for a given user.
953
+
954
+ Note: This function only works with "live" tokens with `stream_ordering` only.
955
+
956
+ We're looking for membership changes in the token range (> `from_key` and <=
957
+ `to_key`).
958
+
959
+ Please be mindful to only use this with `from_key` and `to_key` tokens that are
960
+ recent enough to be after when the first local user joined the room. Otherwise,
961
+ the results may be incomplete or too greedy. For example, if you use a token
962
+ range before the first local user joined the room, you will see 0 events since
963
+ `current_state_delta_stream` tracks what the server thinks is the current state
964
+ of the room as time goes. It does not track how state progresses from the
965
+ beginning of the room. So for example, when you remotely join a room, the first
966
+ rows will just be the state when you joined and progress from there.
967
+
968
+ You can probably reasonably use this with `/sync` because the `to_key` passed in
969
+ will be the "current" now token and the range will cover when the user joined
970
+ the room.
971
+
972
+ Args:
973
+ user_id: The user ID to fetch membership events for.
974
+ from_key: The point in the stream to sync from (fetching events > this point).
975
+ to_key: The token to fetch rooms up to (fetching events <= this point).
976
+ excluded_room_ids: Optional list of room IDs to exclude from the results.
977
+
978
+ Returns:
979
+ All membership changes to the current state in the token range. Events are
980
+ sorted by `stream_ordering` ascending.
981
+
982
+ `event_id`/`sender` can be `None` when the server leaves a room (meaning
983
+ everyone locally left) or a state reset which removed the person from the
984
+ room. We can't tell the difference between the two cases with what's
985
+ available in the `current_state_delta_stream` table. To actually check for a
986
+ state reset, you need to check if a membership still exists in the room.
987
+ """
988
+
989
+ assert from_key.topological is None
990
+ assert to_key.topological is None
991
+
992
+ # Start by ruling out cases where a DB query is not necessary.
993
+ if from_key == to_key:
994
+ return []
995
+
996
+ if from_key:
997
+ has_changed = self._membership_stream_cache.has_entity_changed(
998
+ user_id, int(from_key.stream)
999
+ )
1000
+ if not has_changed:
1001
+ return []
1002
+
1003
+ def f(txn: LoggingTransaction) -> list[CurrentStateDeltaMembership]:
1004
+ # To handle tokens with a non-empty instance_map we fetch more
1005
+ # results than necessary and then filter down
1006
+ min_from_id = from_key.stream
1007
+ max_to_id = to_key.get_max_stream_pos()
1008
+
1009
+ args: list[Any] = [min_from_id, max_to_id, EventTypes.Member, user_id]
1010
+
1011
+ # TODO: It would be good to assert that the `from_token`/`to_token` is >=
1012
+ # the first row in `current_state_delta_stream` for the rooms we're
1013
+ # interested in. Otherwise, we will end up with empty results and not know
1014
+ # it.
1015
+
1016
+ # We could `COALESCE(e.stream_ordering, s.stream_id)` to get more accurate
1017
+ # stream positioning when available but given our usages, we can avoid the
1018
+ # complexity. Between two (valid) stream tokens, we will still get all of
1019
+ # the state changes. Since those events are persisted in a batch, valid
1020
+ # tokens will either be before or after the batch of events.
1021
+ #
1022
+ # `stream_ordering` from the `events` table is more accurate when available
1023
+ # since the `current_state_delta_stream` table only tracks that the current
1024
+ # state is at this stream position (not what stream position the state event
1025
+ # was added) and uses the *minimum* stream position for batches of events.
1026
+ sql = """
1027
+ SELECT
1028
+ s.room_id,
1029
+ e.event_id,
1030
+ s.instance_name,
1031
+ s.stream_id,
1032
+ m.membership,
1033
+ e.sender,
1034
+ s.prev_event_id,
1035
+ e_prev.instance_name AS prev_instance_name,
1036
+ e_prev.stream_ordering AS prev_stream_ordering,
1037
+ m_prev.membership AS prev_membership,
1038
+ e_prev.sender AS prev_sender
1039
+ FROM current_state_delta_stream AS s
1040
+ LEFT JOIN events AS e ON e.event_id = s.event_id
1041
+ LEFT JOIN room_memberships AS m ON m.event_id = s.event_id
1042
+ LEFT JOIN events AS e_prev ON e_prev.event_id = s.prev_event_id
1043
+ LEFT JOIN room_memberships AS m_prev ON m_prev.event_id = s.prev_event_id
1044
+ WHERE s.stream_id > ? AND s.stream_id <= ?
1045
+ AND s.type = ?
1046
+ AND s.state_key = ?
1047
+ ORDER BY s.stream_id ASC
1048
+ """
1049
+
1050
+ txn.execute(sql, args)
1051
+
1052
+ membership_changes: list[CurrentStateDeltaMembership] = []
1053
+ for (
1054
+ room_id,
1055
+ event_id,
1056
+ instance_name,
1057
+ stream_ordering,
1058
+ membership,
1059
+ sender,
1060
+ prev_event_id,
1061
+ prev_instance_name,
1062
+ prev_stream_ordering,
1063
+ prev_membership,
1064
+ prev_sender,
1065
+ ) in txn:
1066
+ assert room_id is not None
1067
+ assert stream_ordering is not None
1068
+
1069
+ if _filter_results_by_stream(
1070
+ from_key,
1071
+ to_key,
1072
+ instance_name,
1073
+ stream_ordering,
1074
+ ):
1075
+ # When the server leaves a room, it will insert new rows into the
1076
+ # `current_state_delta_stream` table with `event_id = null` for all
1077
+ # current state. This means we might already have a row for the
1078
+ # leave event and then another for the same leave where the
1079
+ # `event_id=null` but the `prev_event_id` is pointing back at the
1080
+ # earlier leave event. We don't want to report the leave, if we
1081
+ # already have a leave event.
1082
+ if event_id is None and prev_membership == Membership.LEAVE:
1083
+ continue
1084
+
1085
+ membership_change = CurrentStateDeltaMembership(
1086
+ room_id=room_id,
1087
+ # Event
1088
+ event_id=event_id,
1089
+ event_pos=PersistedEventPosition(
1090
+ # If instance_name is null we default to "master"
1091
+ instance_name=instance_name or "master",
1092
+ stream=stream_ordering,
1093
+ ),
1094
+ # When `s.event_id = null`, we won't be able to get respective
1095
+ # `room_membership` but can assume the user has left the room
1096
+ # because this only happens when the server leaves a room
1097
+ # (meaning everyone locally left) or a state reset which removed
1098
+ # the person from the room.
1099
+ membership=(
1100
+ membership if membership is not None else Membership.LEAVE
1101
+ ),
1102
+ # This will also be null for the same reasons if `s.event_id = null`
1103
+ sender=sender,
1104
+ # Prev event
1105
+ prev_event_id=prev_event_id,
1106
+ prev_event_pos=(
1107
+ PersistedEventPosition(
1108
+ # If instance_name is null we default to "master"
1109
+ instance_name=prev_instance_name or "master",
1110
+ stream=prev_stream_ordering,
1111
+ )
1112
+ if (prev_stream_ordering is not None)
1113
+ else None
1114
+ ),
1115
+ prev_membership=prev_membership,
1116
+ prev_sender=prev_sender,
1117
+ )
1118
+
1119
+ membership_changes.append(membership_change)
1120
+
1121
+ return membership_changes
1122
+
1123
+ membership_changes = await self.db_pool.runInteraction(
1124
+ "get_current_state_delta_membership_changes_for_user", f
1125
+ )
1126
+
1127
+ room_ids_to_exclude: AbstractSet[str] = set()
1128
+ if excluded_room_ids is not None:
1129
+ room_ids_to_exclude = set(excluded_room_ids)
1130
+
1131
+ return [
1132
+ membership_change
1133
+ for membership_change in membership_changes
1134
+ if membership_change.room_id not in room_ids_to_exclude
1135
+ ]
1136
+
1137
+ @trace
1138
+ async def get_sliding_sync_membership_changes(
1139
+ self,
1140
+ user_id: str,
1141
+ from_key: RoomStreamToken,
1142
+ to_key: RoomStreamToken,
1143
+ excluded_room_ids: AbstractSet[str] | None = None,
1144
+ ) -> dict[str, RoomsForUserStateReset]:
1145
+ """
1146
+ Fetch membership events that result in a meaningful membership change for a
1147
+ given user.
1148
+
1149
+ A meaningful membership changes is one where the `membership` value actually
1150
+ changes. This means memberships changes from `join` to `join` (like a display
1151
+ name change) will be filtered out since they result in no meaningful change.
1152
+
1153
+ Note: This function only works with "live" tokens with `stream_ordering` only.
1154
+
1155
+ We're looking for membership changes in the token range (> `from_key` and <=
1156
+ `to_key`).
1157
+
1158
+ Args:
1159
+ user_id: The user ID to fetch membership events for.
1160
+ from_key: The point in the stream to sync from (fetching events > this point).
1161
+ to_key: The token to fetch rooms up to (fetching events <= this point).
1162
+ excluded_room_ids: Optional list of room IDs to exclude from the results.
1163
+
1164
+ Returns:
1165
+ All meaningful membership changes to the current state in the token range.
1166
+ Events are sorted by `stream_ordering` ascending.
1167
+
1168
+ `event_id`/`sender` can be `None` when the server leaves a room (meaning
1169
+ everyone locally left) or a state reset which removed the person from the
1170
+ room. We can't tell the difference between the two cases with what's
1171
+ available in the `current_state_delta_stream` table. To actually check for a
1172
+ state reset, you need to check if a membership still exists in the room.
1173
+ """
1174
+
1175
+ assert from_key.topological is None
1176
+ assert to_key.topological is None
1177
+
1178
+ # Start by ruling out cases where a DB query is not necessary.
1179
+ if from_key == to_key:
1180
+ return {}
1181
+
1182
+ if from_key:
1183
+ has_changed = self._membership_stream_cache.has_entity_changed(
1184
+ user_id, int(from_key.stream)
1185
+ )
1186
+ if not has_changed:
1187
+ return {}
1188
+
1189
+ room_ids_to_exclude: AbstractSet[str] = set()
1190
+ if excluded_room_ids is not None:
1191
+ room_ids_to_exclude = excluded_room_ids
1192
+
1193
+ def f(txn: LoggingTransaction) -> dict[str, RoomsForUserStateReset]:
1194
+ # To handle tokens with a non-empty instance_map we fetch more
1195
+ # results than necessary and then filter down
1196
+ min_from_id = from_key.stream
1197
+ max_to_id = to_key.get_max_stream_pos()
1198
+
1199
+ # This query looks at membership changes in
1200
+ # `sliding_sync_membership_snapshots` which will not include users
1201
+ # that were state reset out of rooms; so we need to look for that
1202
+ # case in `current_state_delta_stream`.
1203
+ sql = """
1204
+ SELECT
1205
+ room_id,
1206
+ membership_event_id,
1207
+ event_instance_name,
1208
+ event_stream_ordering,
1209
+ membership,
1210
+ sender,
1211
+ prev_membership,
1212
+ room_version
1213
+ FROM
1214
+ (
1215
+ SELECT
1216
+ s.room_id,
1217
+ s.membership_event_id,
1218
+ s.event_instance_name,
1219
+ s.event_stream_ordering,
1220
+ s.membership,
1221
+ s.sender,
1222
+ m_prev.membership AS prev_membership
1223
+ FROM sliding_sync_membership_snapshots as s
1224
+ LEFT JOIN event_edges AS e ON e.event_id = s.membership_event_id
1225
+ LEFT JOIN room_memberships AS m_prev ON m_prev.event_id = e.prev_event_id
1226
+ WHERE s.user_id = ?
1227
+
1228
+ UNION ALL
1229
+
1230
+ SELECT
1231
+ s.room_id,
1232
+ e.event_id,
1233
+ s.instance_name,
1234
+ s.stream_id,
1235
+ m.membership,
1236
+ e.sender,
1237
+ m_prev.membership AS prev_membership
1238
+ FROM current_state_delta_stream AS s
1239
+ LEFT JOIN events AS e ON e.event_id = s.event_id
1240
+ LEFT JOIN room_memberships AS m ON m.event_id = s.event_id
1241
+ LEFT JOIN room_memberships AS m_prev ON m_prev.event_id = s.prev_event_id
1242
+ WHERE
1243
+ s.type = ?
1244
+ AND s.state_key = ?
1245
+ ) AS c
1246
+ INNER JOIN rooms USING (room_id)
1247
+ WHERE event_stream_ordering > ? AND event_stream_ordering <= ?
1248
+ ORDER BY event_stream_ordering ASC
1249
+ """
1250
+
1251
+ txn.execute(
1252
+ sql,
1253
+ (user_id, EventTypes.Member, user_id, min_from_id, max_to_id),
1254
+ )
1255
+
1256
+ membership_changes: dict[str, RoomsForUserStateReset] = {}
1257
+ for (
1258
+ room_id,
1259
+ membership_event_id,
1260
+ event_instance_name,
1261
+ event_stream_ordering,
1262
+ membership,
1263
+ sender,
1264
+ prev_membership,
1265
+ room_version_id,
1266
+ ) in txn:
1267
+ assert room_id is not None
1268
+ assert event_stream_ordering is not None
1269
+
1270
+ if room_id in room_ids_to_exclude:
1271
+ continue
1272
+
1273
+ if _filter_results_by_stream(
1274
+ from_key,
1275
+ to_key,
1276
+ event_instance_name,
1277
+ event_stream_ordering,
1278
+ ):
1279
+ # When the server leaves a room, it will insert new rows into the
1280
+ # `current_state_delta_stream` table with `event_id = null` for all
1281
+ # current state. This means we might already have a row for the
1282
+ # leave event and then another for the same leave where the
1283
+ # `event_id=null` but the `prev_event_id` is pointing back at the
1284
+ # earlier leave event. We don't want to report the leave, if we
1285
+ # already have a leave event.
1286
+ if (
1287
+ membership_event_id is None
1288
+ and prev_membership == Membership.LEAVE
1289
+ ):
1290
+ continue
1291
+
1292
+ if membership_event_id is None and room_id in membership_changes:
1293
+ # SUSPICIOUS: if we join a room and get state reset out of it
1294
+ # in the same queried window,
1295
+ # won't this ignore the 'state reset out of it' part?
1296
+ continue
1297
+
1298
+ # When `s.event_id = null`, we won't be able to get respective
1299
+ # `room_membership` but can assume the user has left the room
1300
+ # because this only happens when the server leaves a room
1301
+ # (meaning everyone locally left) or a state reset which removed
1302
+ # the person from the room.
1303
+ membership = (
1304
+ membership if membership is not None else Membership.LEAVE
1305
+ )
1306
+
1307
+ if membership == prev_membership:
1308
+ # If `membership` and `prev_membership` are the same then this
1309
+ # is not a meaningful change so we can skip it.
1310
+ # An example of this happening is when the user changes their display name.
1311
+ continue
1312
+
1313
+ membership_change = RoomsForUserStateReset(
1314
+ room_id=room_id,
1315
+ sender=sender,
1316
+ membership=membership,
1317
+ event_id=membership_event_id,
1318
+ event_pos=PersistedEventPosition(
1319
+ event_instance_name, event_stream_ordering
1320
+ ),
1321
+ room_version_id=room_version_id,
1322
+ )
1323
+
1324
+ membership_changes[room_id] = membership_change
1325
+
1326
+ return membership_changes
1327
+
1328
+ membership_changes = await self.db_pool.runInteraction(
1329
+ "get_sliding_sync_membership_changes", f
1330
+ )
1331
+
1332
+ return membership_changes
1333
+
1334
+ @cancellable
1335
+ async def get_membership_changes_for_user(
1336
+ self,
1337
+ user_id: str,
1338
+ from_key: RoomStreamToken,
1339
+ to_key: RoomStreamToken,
1340
+ excluded_rooms: list[str] | None = None,
1341
+ ) -> list[EventBase]:
1342
+ """Fetch membership events for a given user.
1343
+
1344
+ All such events whose stream ordering `s` lies in the range
1345
+ `from_key < s <= to_key` are returned. Events are ordered by ascending stream
1346
+ order.
1347
+ """
1348
+ # Start by ruling out cases where a DB query is not necessary.
1349
+ if from_key == to_key:
1350
+ return []
1351
+
1352
+ if from_key:
1353
+ has_changed = self._membership_stream_cache.has_entity_changed(
1354
+ user_id, int(from_key.stream)
1355
+ )
1356
+ if not has_changed:
1357
+ return []
1358
+
1359
+ def f(txn: LoggingTransaction) -> list[_EventDictReturn]:
1360
+ # To handle tokens with a non-empty instance_map we fetch more
1361
+ # results than necessary and then filter down
1362
+ min_from_id = from_key.stream
1363
+ max_to_id = to_key.get_max_stream_pos()
1364
+
1365
+ args: list[Any] = [user_id, min_from_id, max_to_id]
1366
+
1367
+ ignore_room_clause = ""
1368
+ if excluded_rooms is not None and len(excluded_rooms) > 0:
1369
+ ignore_room_clause, ignore_room_args = make_in_list_sql_clause(
1370
+ txn.database_engine, "e.room_id", excluded_rooms, negative=True
1371
+ )
1372
+ ignore_room_clause = f"AND {ignore_room_clause}"
1373
+ args += ignore_room_args
1374
+
1375
+ sql = """
1376
+ SELECT m.event_id, instance_name, topological_ordering, stream_ordering
1377
+ FROM events AS e, room_memberships AS m
1378
+ WHERE e.event_id = m.event_id
1379
+ AND m.user_id = ?
1380
+ AND e.stream_ordering > ? AND e.stream_ordering <= ?
1381
+ %s
1382
+ ORDER BY e.stream_ordering ASC
1383
+ """ % (ignore_room_clause,)
1384
+
1385
+ txn.execute(sql, args)
1386
+
1387
+ rows = [
1388
+ _EventDictReturn(event_id, None, stream_ordering)
1389
+ for event_id, instance_name, topological_ordering, stream_ordering in txn
1390
+ if _filter_results(
1391
+ from_key,
1392
+ to_key,
1393
+ instance_name,
1394
+ topological_ordering,
1395
+ stream_ordering,
1396
+ )
1397
+ ]
1398
+
1399
+ return rows
1400
+
1401
+ rows = await self.db_pool.runInteraction("get_membership_changes_for_user", f)
1402
+
1403
+ ret = await self.get_events_as_list(
1404
+ [r.event_id for r in rows], get_prev_content=True
1405
+ )
1406
+
1407
+ return ret
1408
+
1409
+ async def get_recent_events_for_room(
1410
+ self, room_id: str, limit: int, end_token: RoomStreamToken
1411
+ ) -> tuple[list[EventBase], RoomStreamToken]:
1412
+ """Get the most recent events in the room in topological ordering.
1413
+
1414
+ Args:
1415
+ room_id
1416
+ limit
1417
+ end_token: The stream token representing now.
1418
+
1419
+ Returns:
1420
+ A list of events and a token pointing to the start of the returned
1421
+ events. The events returned are in ascending topological order.
1422
+ """
1423
+
1424
+ rows, token = await self.get_recent_event_ids_for_room(
1425
+ room_id, limit, end_token
1426
+ )
1427
+
1428
+ events = await self.get_events_as_list(
1429
+ [r.event_id for r in rows], get_prev_content=True
1430
+ )
1431
+
1432
+ return events, token
1433
+
1434
+ async def get_recent_event_ids_for_room(
1435
+ self, room_id: str, limit: int, end_token: RoomStreamToken
1436
+ ) -> tuple[list[_EventDictReturn], RoomStreamToken]:
1437
+ """Get the most recent events in the room in topological ordering.
1438
+
1439
+ Args:
1440
+ room_id
1441
+ limit
1442
+ end_token: The stream token representing now.
1443
+
1444
+ Returns:
1445
+ A list of _EventDictReturn and a token pointing to the start of the
1446
+ returned events. The events returned are in ascending order.
1447
+ """
1448
+ # Allow a zero limit here, and no-op.
1449
+ if limit == 0:
1450
+ return [], end_token
1451
+
1452
+ rows, token, _ = await self.db_pool.runInteraction(
1453
+ "get_recent_event_ids_for_room",
1454
+ self._paginate_room_events_by_topological_ordering_txn,
1455
+ room_id,
1456
+ from_token=end_token,
1457
+ limit=limit,
1458
+ )
1459
+
1460
+ # We want to return the results in ascending order.
1461
+ rows.reverse()
1462
+
1463
+ return rows, token
1464
+
1465
+ async def get_room_event_before_stream_ordering(
1466
+ self, room_id: str, stream_ordering: int
1467
+ ) -> tuple[int, int, str] | None:
1468
+ """Gets details of the first event in a room at or before a stream ordering
1469
+
1470
+ Args:
1471
+ room_id:
1472
+ stream_ordering:
1473
+
1474
+ Returns:
1475
+ A tuple of (stream ordering, topological ordering, event_id)
1476
+ """
1477
+
1478
+ def _f(txn: LoggingTransaction) -> tuple[int, int, str] | None:
1479
+ sql = """
1480
+ SELECT stream_ordering, topological_ordering, event_id
1481
+ FROM events
1482
+ LEFT JOIN rejections USING (event_id)
1483
+ WHERE room_id = ?
1484
+ AND stream_ordering <= ?
1485
+ AND NOT outlier
1486
+ AND rejections.event_id IS NULL
1487
+ ORDER BY stream_ordering DESC
1488
+ LIMIT 1
1489
+ """
1490
+ txn.execute(sql, (room_id, stream_ordering))
1491
+ return cast(tuple[int, int, str] | None, txn.fetchone())
1492
+
1493
+ return await self.db_pool.runInteraction(
1494
+ "get_room_event_before_stream_ordering", _f
1495
+ )
1496
+
1497
+ async def get_last_event_id_in_room_before_stream_ordering(
1498
+ self,
1499
+ room_id: str,
1500
+ end_token: RoomStreamToken,
1501
+ ) -> str | None:
1502
+ """Returns the ID of the last event in a room at or before a stream ordering
1503
+
1504
+ Args:
1505
+ room_id
1506
+ end_token: The token used to stream from
1507
+
1508
+ Returns:
1509
+ The ID of the most recent event, or None if there are no events in the room
1510
+ before this stream ordering.
1511
+ """
1512
+ last_event_result = (
1513
+ await self.get_last_event_pos_in_room_before_stream_ordering(
1514
+ room_id, end_token
1515
+ )
1516
+ )
1517
+
1518
+ if last_event_result:
1519
+ return last_event_result[0]
1520
+
1521
+ return None
1522
+
1523
+ async def get_last_event_pos_in_room(
1524
+ self,
1525
+ room_id: str,
1526
+ event_types: StrCollection | None = None,
1527
+ ) -> tuple[str, PersistedEventPosition] | None:
1528
+ """
1529
+ Returns the ID and event position of the last event in a room.
1530
+
1531
+ Based on `get_last_event_pos_in_room_before_stream_ordering(...)`
1532
+
1533
+ Args:
1534
+ room_id
1535
+ event_types: Optional allowlist of event types to filter by
1536
+
1537
+ Returns:
1538
+ The ID of the most recent event and it's position, or None if there are no
1539
+ events in the room that match the given event types.
1540
+ """
1541
+
1542
+ def _get_last_event_pos_in_room_txn(
1543
+ txn: LoggingTransaction,
1544
+ ) -> tuple[str, PersistedEventPosition] | None:
1545
+ event_type_clause = ""
1546
+ event_type_args: list[str] = []
1547
+ if event_types is not None and len(event_types) > 0:
1548
+ event_type_clause, event_type_args = make_in_list_sql_clause(
1549
+ txn.database_engine, "type", event_types
1550
+ )
1551
+ event_type_clause = f"AND {event_type_clause}"
1552
+
1553
+ sql = f"""
1554
+ SELECT event_id, stream_ordering, instance_name
1555
+ FROM events
1556
+ LEFT JOIN rejections USING (event_id)
1557
+ WHERE room_id = ?
1558
+ {event_type_clause}
1559
+ AND NOT outlier
1560
+ AND rejections.event_id IS NULL
1561
+ ORDER BY stream_ordering DESC
1562
+ LIMIT 1
1563
+ """
1564
+
1565
+ txn.execute(
1566
+ sql,
1567
+ [room_id] + event_type_args,
1568
+ )
1569
+
1570
+ row = cast(tuple[str, int, str] | None, txn.fetchone())
1571
+ if row is not None:
1572
+ event_id, stream_ordering, instance_name = row
1573
+
1574
+ return event_id, PersistedEventPosition(
1575
+ # If instance_name is null we default to "master"
1576
+ instance_name or "master",
1577
+ stream_ordering,
1578
+ )
1579
+
1580
+ return None
1581
+
1582
+ return await self.db_pool.runInteraction(
1583
+ "get_last_event_pos_in_room",
1584
+ _get_last_event_pos_in_room_txn,
1585
+ )
1586
+
1587
+ @trace
1588
+ async def get_last_event_pos_in_room_before_stream_ordering(
1589
+ self,
1590
+ room_id: str,
1591
+ end_token: RoomStreamToken,
1592
+ event_types: StrCollection | None = None,
1593
+ ) -> tuple[str, PersistedEventPosition] | None:
1594
+ """
1595
+ Returns the ID and event position of the last event in a room at or before a
1596
+ stream ordering.
1597
+
1598
+ Args:
1599
+ room_id
1600
+ end_token: The token used to stream from
1601
+ event_types: Optional allowlist of event types to filter by
1602
+
1603
+ Returns:
1604
+ The ID of the most recent event and it's position, or None if there are no
1605
+ events in the room before this stream ordering.
1606
+ """
1607
+
1608
+ def get_last_event_pos_in_room_before_stream_ordering_txn(
1609
+ txn: LoggingTransaction,
1610
+ ) -> tuple[str, PersistedEventPosition] | None:
1611
+ # We're looking for the closest event at or before the token. We need to
1612
+ # handle the fact that the stream token can be a vector clock (with an
1613
+ # `instance_map`) and events can be persisted on different instances
1614
+ # (sharded event persisters). The first subquery handles the events that
1615
+ # would be within the vector clock and gets all rows between the minimum and
1616
+ # maximum stream ordering in the token which need to be filtered against the
1617
+ # `instance_map`. The second subquery handles the "before" case and finds
1618
+ # the first row before the token. We then filter out any results past the
1619
+ # token's vector clock and return the first row that matches.
1620
+ min_stream = end_token.stream
1621
+ max_stream = end_token.get_max_stream_pos()
1622
+
1623
+ event_type_clause = ""
1624
+ event_type_args: list[str] = []
1625
+ if event_types is not None and len(event_types) > 0:
1626
+ event_type_clause, event_type_args = make_in_list_sql_clause(
1627
+ txn.database_engine, "type", event_types
1628
+ )
1629
+ event_type_clause = f"AND {event_type_clause}"
1630
+
1631
+ # We use `UNION ALL` because we don't need any of the deduplication logic
1632
+ # (`UNION` is really a `UNION` + `DISTINCT`). `UNION ALL` does preserve the
1633
+ # ordering of the operand queries but there is no actual guarantee that it
1634
+ # has this behavior in all scenarios so we need the extra `ORDER BY` at the
1635
+ # bottom.
1636
+ sql = """
1637
+ SELECT * FROM (
1638
+ SELECT instance_name, stream_ordering, topological_ordering, event_id
1639
+ FROM events
1640
+ LEFT JOIN rejections USING (event_id)
1641
+ WHERE room_id = ?
1642
+ %s
1643
+ AND ? < stream_ordering AND stream_ordering <= ?
1644
+ AND NOT outlier
1645
+ AND rejections.event_id IS NULL
1646
+ ORDER BY stream_ordering DESC
1647
+ ) AS a
1648
+ UNION ALL
1649
+ SELECT * FROM (
1650
+ SELECT instance_name, stream_ordering, topological_ordering, event_id
1651
+ FROM events
1652
+ LEFT JOIN rejections USING (event_id)
1653
+ WHERE room_id = ?
1654
+ %s
1655
+ AND stream_ordering <= ?
1656
+ AND NOT outlier
1657
+ AND rejections.event_id IS NULL
1658
+ ORDER BY stream_ordering DESC
1659
+ LIMIT 1
1660
+ ) AS b
1661
+ ORDER BY stream_ordering DESC
1662
+ """ % (
1663
+ event_type_clause,
1664
+ event_type_clause,
1665
+ )
1666
+ txn.execute(
1667
+ sql,
1668
+ [room_id]
1669
+ + event_type_args
1670
+ + [min_stream, max_stream, room_id]
1671
+ + event_type_args
1672
+ + [min_stream],
1673
+ )
1674
+
1675
+ for instance_name, stream_ordering, topological_ordering, event_id in txn:
1676
+ if _filter_results(
1677
+ lower_token=None,
1678
+ upper_token=end_token,
1679
+ instance_name=instance_name,
1680
+ topological_ordering=topological_ordering,
1681
+ stream_ordering=stream_ordering,
1682
+ ):
1683
+ return event_id, PersistedEventPosition(
1684
+ # If instance_name is null we default to "master"
1685
+ instance_name or "master",
1686
+ stream_ordering,
1687
+ )
1688
+
1689
+ return None
1690
+
1691
+ return await self.db_pool.runInteraction(
1692
+ "get_last_event_pos_in_room_before_stream_ordering",
1693
+ get_last_event_pos_in_room_before_stream_ordering_txn,
1694
+ )
1695
+
1696
+ async def bulk_get_last_event_pos_in_room_before_stream_ordering(
1697
+ self,
1698
+ room_ids: StrCollection,
1699
+ end_token: RoomStreamToken,
1700
+ ) -> dict[str, int]:
1701
+ """Bulk fetch the stream position of the latest events in the given
1702
+ rooms
1703
+ """
1704
+
1705
+ # First we just get the latest positions for the room, as the vast
1706
+ # majority of them will be before the given end token anyway. By doing
1707
+ # this we can cache most rooms.
1708
+ uncapped_results = await self._bulk_get_max_event_pos(room_ids)
1709
+
1710
+ # Check that the stream position for the rooms are from before the
1711
+ # minimum position of the token. If not then we need to fetch more
1712
+ # rows.
1713
+ results: dict[str, int] = {}
1714
+ recheck_rooms: set[str] = set()
1715
+ min_token = end_token.stream
1716
+ for room_id, stream in uncapped_results.items():
1717
+ if stream is None:
1718
+ # Despite the function not directly setting None, the cache can!
1719
+ # See: https://github.com/element-hq/synapse/issues/17726
1720
+ continue
1721
+ if stream <= min_token:
1722
+ results[room_id] = stream
1723
+ else:
1724
+ recheck_rooms.add(room_id)
1725
+
1726
+ if not recheck_rooms:
1727
+ return results
1728
+
1729
+ # There shouldn't be many rooms that we need to recheck, so we do them
1730
+ # one-by-one.
1731
+ for room_id in recheck_rooms:
1732
+ result = await self.get_last_event_pos_in_room_before_stream_ordering(
1733
+ room_id, end_token
1734
+ )
1735
+ if result is not None:
1736
+ results[room_id] = result[1].stream
1737
+
1738
+ return results
1739
+
1740
+ @cached()
1741
+ async def _get_max_event_pos(self, room_id: str) -> int:
1742
+ raise NotImplementedError()
1743
+
1744
+ @cachedList(cached_method_name="_get_max_event_pos", list_name="room_ids")
1745
+ async def _bulk_get_max_event_pos(
1746
+ self, room_ids: StrCollection
1747
+ ) -> Mapping[str, int | None]:
1748
+ """Fetch the max position of a persisted event in the room."""
1749
+
1750
+ # We need to be careful not to return positions ahead of the current
1751
+ # positions, so we get the current token now and cap our queries to it.
1752
+ now_token = self.get_room_max_token()
1753
+ max_pos = now_token.get_max_stream_pos()
1754
+
1755
+ results: dict[str, int] = {}
1756
+
1757
+ # First, we check for the rooms in the stream change cache to see if we
1758
+ # can just use the latest position from it.
1759
+ missing_room_ids: set[str] = set()
1760
+ for room_id in room_ids:
1761
+ stream_pos = self._events_stream_cache.get_max_pos_of_last_change(room_id)
1762
+ if stream_pos is not None:
1763
+ results[room_id] = stream_pos
1764
+ else:
1765
+ missing_room_ids.add(room_id)
1766
+
1767
+ if not missing_room_ids:
1768
+ return results
1769
+
1770
+ # Next, we query the stream position from the DB. At first we fetch all
1771
+ # positions less than the *max* stream pos in the token, then filter
1772
+ # them down. We do this as a) this is a cheaper query, and b) the vast
1773
+ # majority of rooms will have a latest token from before the min stream
1774
+ # pos.
1775
+
1776
+ def bulk_get_max_event_pos_fallback_txn(
1777
+ txn: LoggingTransaction, batched_room_ids: StrCollection
1778
+ ) -> dict[str, int]:
1779
+ clause, args = make_in_list_sql_clause(
1780
+ self.database_engine, "room_id", batched_room_ids
1781
+ )
1782
+ sql = f"""
1783
+ SELECT room_id, (
1784
+ SELECT stream_ordering FROM events AS e
1785
+ LEFT JOIN rejections USING (event_id)
1786
+ WHERE e.room_id = r.room_id
1787
+ AND e.stream_ordering <= ?
1788
+ AND NOT outlier
1789
+ AND rejection_reason IS NULL
1790
+ ORDER BY stream_ordering DESC
1791
+ LIMIT 1
1792
+ )
1793
+ FROM rooms AS r
1794
+ WHERE {clause}
1795
+ """
1796
+ txn.execute(sql, [max_pos] + args)
1797
+ return {row[0]: row[1] for row in txn}
1798
+
1799
+ # It's easier to look at the `sliding_sync_joined_rooms` table and avoid all of
1800
+ # the joins and sub-queries.
1801
+ def bulk_get_max_event_pos_from_sliding_sync_tables_txn(
1802
+ txn: LoggingTransaction, batched_room_ids: StrCollection
1803
+ ) -> dict[str, int]:
1804
+ clause, args = make_in_list_sql_clause(
1805
+ self.database_engine, "room_id", batched_room_ids
1806
+ )
1807
+ sql = f"""
1808
+ SELECT room_id, event_stream_ordering
1809
+ FROM sliding_sync_joined_rooms
1810
+ WHERE {clause}
1811
+ ORDER BY event_stream_ordering DESC
1812
+ """
1813
+ txn.execute(sql, args)
1814
+ return {row[0]: row[1] for row in txn}
1815
+
1816
+ recheck_rooms: set[str] = set()
1817
+ for batched in batch_iter(room_ids, 1000):
1818
+ if await self.have_finished_sliding_sync_background_jobs():
1819
+ batch_results = await self.db_pool.runInteraction(
1820
+ "bulk_get_max_event_pos_from_sliding_sync_tables_txn",
1821
+ bulk_get_max_event_pos_from_sliding_sync_tables_txn,
1822
+ batched,
1823
+ )
1824
+ else:
1825
+ batch_results = await self.db_pool.runInteraction(
1826
+ "bulk_get_max_event_pos_fallback_txn",
1827
+ bulk_get_max_event_pos_fallback_txn,
1828
+ batched,
1829
+ )
1830
+ for room_id, stream_ordering in batch_results.items():
1831
+ if stream_ordering <= now_token.stream:
1832
+ results[room_id] = stream_ordering
1833
+ else:
1834
+ recheck_rooms.add(room_id)
1835
+
1836
+ # We now need to handle rooms where the above query returned a stream
1837
+ # position that was potentially too new. This should happen very rarely
1838
+ # so we just query the rooms one-by-one
1839
+ for room_id in recheck_rooms:
1840
+ result = await self.get_last_event_pos_in_room_before_stream_ordering(
1841
+ room_id, now_token
1842
+ )
1843
+ if result is not None:
1844
+ results[room_id] = result[1].stream
1845
+
1846
+ return results
1847
+
1848
+ async def get_current_room_stream_token_for_room_id(
1849
+ self, room_id: str
1850
+ ) -> RoomStreamToken:
1851
+ """Returns the current position of the rooms stream (historic token)."""
1852
+ stream_ordering = self.get_room_max_stream_ordering()
1853
+ topo = await self.db_pool.runInteraction(
1854
+ "_get_max_topological_txn", self._get_max_topological_txn, room_id
1855
+ )
1856
+ return RoomStreamToken(topological=topo, stream=stream_ordering)
1857
+
1858
+ @overload
1859
+ def get_stream_id_for_event_txn(
1860
+ self,
1861
+ txn: LoggingTransaction,
1862
+ event_id: str,
1863
+ allow_none: Literal[False] = False,
1864
+ ) -> int: ...
1865
+
1866
+ @overload
1867
+ def get_stream_id_for_event_txn(
1868
+ self,
1869
+ txn: LoggingTransaction,
1870
+ event_id: str,
1871
+ allow_none: bool = False,
1872
+ ) -> int | None: ...
1873
+
1874
+ def get_stream_id_for_event_txn(
1875
+ self,
1876
+ txn: LoggingTransaction,
1877
+ event_id: str,
1878
+ allow_none: bool = False,
1879
+ ) -> int | None:
1880
+ # Type ignore: we pass keyvalues a Dict[str, str]; the function wants
1881
+ # Dict[str, Any]. I think mypy is unhappy because Dict is invariant?
1882
+ return self.db_pool.simple_select_one_onecol_txn( # type: ignore[call-overload]
1883
+ txn=txn,
1884
+ table="events",
1885
+ keyvalues={"event_id": event_id},
1886
+ retcol="stream_ordering",
1887
+ allow_none=allow_none,
1888
+ )
1889
+
1890
+ async def get_position_for_event(self, event_id: str) -> PersistedEventPosition:
1891
+ """Get the persisted position for an event"""
1892
+ row = await self.db_pool.simple_select_one(
1893
+ table="events",
1894
+ keyvalues={"event_id": event_id},
1895
+ retcols=("stream_ordering", "instance_name"),
1896
+ desc="get_position_for_event",
1897
+ )
1898
+
1899
+ return PersistedEventPosition(row[1] or "master", row[0])
1900
+
1901
+ async def get_topological_token_for_event(self, event_id: str) -> RoomStreamToken:
1902
+ """The stream token for an event
1903
+ Args:
1904
+ event_id: The id of the event to look up a stream token for.
1905
+ Raises:
1906
+ StoreError if the event wasn't in the database.
1907
+ Returns:
1908
+ A `RoomStreamToken` topological token.
1909
+ """
1910
+ row = await self.db_pool.simple_select_one(
1911
+ table="events",
1912
+ keyvalues={"event_id": event_id},
1913
+ retcols=("stream_ordering", "topological_ordering"),
1914
+ desc="get_topological_token_for_event",
1915
+ )
1916
+ return RoomStreamToken(topological=row[1], stream=row[0])
1917
+
1918
+ async def get_current_topological_token(self, room_id: str, stream_key: int) -> int:
1919
+ """Gets the topological token in a room after or at the given stream
1920
+ ordering.
1921
+
1922
+ Args:
1923
+ room_id
1924
+ stream_key
1925
+ """
1926
+ if isinstance(self.database_engine, PostgresEngine):
1927
+ min_function = "LEAST"
1928
+ elif isinstance(self.database_engine, Sqlite3Engine):
1929
+ min_function = "MIN"
1930
+ else:
1931
+ raise RuntimeError(f"Unknown database engine {self.database_engine}")
1932
+
1933
+ # This query used to be
1934
+ # SELECT COALESCE(MIN(topological_ordering), 0) FROM events
1935
+ # WHERE room_id = ? and events.stream_ordering >= {stream_key}
1936
+ # which returns 0 if the stream_key is newer than any event in
1937
+ # the room. That's not wrong, but it seems to interact oddly with backfill,
1938
+ # requiring a second call to /messages to actually backfill from a remote
1939
+ # homeserver.
1940
+ #
1941
+ # Instead, rollback the stream ordering to that after the most recent event in
1942
+ # this room.
1943
+ sql = f"""
1944
+ WITH fallback(max_stream_ordering) AS (
1945
+ SELECT MAX(stream_ordering)
1946
+ FROM events
1947
+ WHERE room_id = ?
1948
+ )
1949
+ SELECT COALESCE(MIN(topological_ordering), 0) FROM events
1950
+ WHERE
1951
+ room_id = ?
1952
+ AND events.stream_ordering >= {min_function}(
1953
+ ?,
1954
+ (SELECT max_stream_ordering FROM fallback)
1955
+ )
1956
+ """
1957
+
1958
+ row = await self.db_pool.execute(
1959
+ "get_current_topological_token", sql, room_id, room_id, stream_key
1960
+ )
1961
+ return row[0][0] if row else 0
1962
+
1963
+ def _get_max_topological_txn(self, txn: LoggingTransaction, room_id: str) -> int:
1964
+ txn.execute(
1965
+ "SELECT MAX(topological_ordering) FROM events WHERE room_id = ?",
1966
+ (room_id,),
1967
+ )
1968
+
1969
+ rows = txn.fetchall()
1970
+ # An aggregate function like MAX() will always return one row per group
1971
+ # so we can safely rely on the lookup here. For example, when a we
1972
+ # lookup a `room_id` which does not exist, `rows` will look like
1973
+ # `[(None,)]`
1974
+ return rows[0][0] if rows[0][0] is not None else 0
1975
+
1976
+ async def get_events_around(
1977
+ self,
1978
+ room_id: str,
1979
+ event_id: str,
1980
+ before_limit: int,
1981
+ after_limit: int,
1982
+ event_filter: Filter | None = None,
1983
+ ) -> _EventsAround:
1984
+ """Retrieve events and pagination tokens around a given event in a
1985
+ room.
1986
+ """
1987
+
1988
+ results = await self.db_pool.runInteraction(
1989
+ "get_events_around",
1990
+ self._get_events_around_txn,
1991
+ room_id,
1992
+ event_id,
1993
+ before_limit,
1994
+ after_limit,
1995
+ event_filter,
1996
+ )
1997
+
1998
+ events_before = await self.get_events_as_list(
1999
+ list(results["before"]["event_ids"]), get_prev_content=True
2000
+ )
2001
+
2002
+ events_after = await self.get_events_as_list(
2003
+ list(results["after"]["event_ids"]), get_prev_content=True
2004
+ )
2005
+
2006
+ return _EventsAround(
2007
+ events_before=events_before,
2008
+ events_after=events_after,
2009
+ start=results["before"]["token"],
2010
+ end=results["after"]["token"],
2011
+ )
2012
+
2013
+ def _get_events_around_txn(
2014
+ self,
2015
+ txn: LoggingTransaction,
2016
+ room_id: str,
2017
+ event_id: str,
2018
+ before_limit: int,
2019
+ after_limit: int,
2020
+ event_filter: Filter | None,
2021
+ ) -> dict:
2022
+ """Retrieves event_ids and pagination tokens around a given event in a
2023
+ room.
2024
+
2025
+ Args:
2026
+ room_id
2027
+ event_id
2028
+ before_limit
2029
+ after_limit
2030
+ event_filter
2031
+
2032
+ Returns:
2033
+ dict
2034
+ """
2035
+
2036
+ row = self.db_pool.simple_select_one_txn(
2037
+ txn,
2038
+ "events",
2039
+ keyvalues={"event_id": event_id, "room_id": room_id},
2040
+ retcols=("stream_ordering", "topological_ordering"),
2041
+ )
2042
+ stream_ordering = int(row[0])
2043
+ topological_ordering = int(row[1])
2044
+
2045
+ # Paginating backwards includes the event at the token, but paginating
2046
+ # forward doesn't.
2047
+ before_token = RoomStreamToken(
2048
+ topological=topological_ordering - 1, stream=stream_ordering
2049
+ )
2050
+
2051
+ after_token = RoomStreamToken(
2052
+ topological=topological_ordering, stream=stream_ordering
2053
+ )
2054
+
2055
+ rows, start_token, _ = self._paginate_room_events_by_topological_ordering_txn(
2056
+ txn,
2057
+ room_id,
2058
+ before_token,
2059
+ direction=Direction.BACKWARDS,
2060
+ limit=before_limit,
2061
+ event_filter=event_filter,
2062
+ )
2063
+ events_before = [r.event_id for r in rows]
2064
+
2065
+ rows, end_token, _ = self._paginate_room_events_by_topological_ordering_txn(
2066
+ txn,
2067
+ room_id,
2068
+ after_token,
2069
+ direction=Direction.FORWARDS,
2070
+ limit=after_limit,
2071
+ event_filter=event_filter,
2072
+ )
2073
+ events_after = [r.event_id for r in rows]
2074
+
2075
+ return {
2076
+ "before": {"event_ids": events_before, "token": start_token},
2077
+ "after": {"event_ids": events_after, "token": end_token},
2078
+ }
2079
+
2080
+ async def get_all_new_event_ids_stream(
2081
+ self,
2082
+ from_id: int,
2083
+ current_id: int,
2084
+ limit: int,
2085
+ ) -> tuple[int, dict[str, int | None]]:
2086
+ """Get all new events
2087
+
2088
+ Returns all event ids with from_id < stream_ordering <= current_id.
2089
+
2090
+ Args:
2091
+ from_id: the stream_ordering of the last event we processed
2092
+ current_id: the stream_ordering of the most recently processed event
2093
+ limit: the maximum number of events to return
2094
+
2095
+ Returns:
2096
+ A tuple of (next_id, event_to_received_ts), where `next_id`
2097
+ is the next value to pass as `from_id` (it will either be the
2098
+ stream_ordering of the last returned event, or, if fewer than `limit`
2099
+ events were found, the `current_id`). The `event_to_received_ts` is
2100
+ a dictionary mapping event ID to the event `received_ts`, sorted by ascending
2101
+ stream_ordering.
2102
+ """
2103
+
2104
+ def get_all_new_event_ids_stream_txn(
2105
+ txn: LoggingTransaction,
2106
+ ) -> tuple[int, dict[str, int | None]]:
2107
+ sql = (
2108
+ "SELECT e.stream_ordering, e.event_id, e.received_ts"
2109
+ " FROM events AS e"
2110
+ " WHERE"
2111
+ " ? < e.stream_ordering AND e.stream_ordering <= ?"
2112
+ " ORDER BY e.stream_ordering ASC"
2113
+ " LIMIT ?"
2114
+ )
2115
+
2116
+ txn.execute(sql, (from_id, current_id, limit))
2117
+ rows = txn.fetchall()
2118
+
2119
+ upper_bound = current_id
2120
+ if len(rows) == limit:
2121
+ upper_bound = rows[-1][0]
2122
+
2123
+ event_to_received_ts: dict[str, int | None] = {
2124
+ row[1]: row[2] for row in rows
2125
+ }
2126
+ return upper_bound, event_to_received_ts
2127
+
2128
+ upper_bound, event_to_received_ts = await self.db_pool.runInteraction(
2129
+ "get_all_new_event_ids_stream", get_all_new_event_ids_stream_txn
2130
+ )
2131
+
2132
+ return upper_bound, event_to_received_ts
2133
+
2134
+ async def get_federation_out_pos(self, typ: str) -> int:
2135
+ if self._need_to_reset_federation_stream_positions:
2136
+ await self.db_pool.runInteraction(
2137
+ "_reset_federation_positions_txn", self._reset_federation_positions_txn
2138
+ )
2139
+ self._need_to_reset_federation_stream_positions = False
2140
+
2141
+ return await self.db_pool.simple_select_one_onecol(
2142
+ table="federation_stream_position",
2143
+ retcol="stream_id",
2144
+ keyvalues={"type": typ, "instance_name": self._instance_name},
2145
+ desc="get_federation_out_pos",
2146
+ )
2147
+
2148
+ async def update_federation_out_pos(self, typ: str, stream_id: int) -> None:
2149
+ if self._need_to_reset_federation_stream_positions:
2150
+ await self.db_pool.runInteraction(
2151
+ "_reset_federation_positions_txn", self._reset_federation_positions_txn
2152
+ )
2153
+ self._need_to_reset_federation_stream_positions = False
2154
+
2155
+ await self.db_pool.simple_update_one(
2156
+ table="federation_stream_position",
2157
+ keyvalues={"type": typ, "instance_name": self._instance_name},
2158
+ updatevalues={"stream_id": stream_id},
2159
+ desc="update_federation_out_pos",
2160
+ )
2161
+
2162
+ def _reset_federation_positions_txn(self, txn: LoggingTransaction) -> None:
2163
+ """Fiddles with the `federation_stream_position` table to make it match
2164
+ the configured federation sender instances during start up.
2165
+ """
2166
+
2167
+ # The federation sender instances may have changed, so we need to
2168
+ # massage the `federation_stream_position` table to have a row per type
2169
+ # per instance sending federation. If there is a mismatch we update the
2170
+ # table with the correct rows using the *minimum* stream ID seen. This
2171
+ # may result in resending of events/EDUs to remote servers, but that is
2172
+ # preferable to dropping them.
2173
+
2174
+ if not self._send_federation:
2175
+ return
2176
+
2177
+ # Pull out the configured instances. If we don't have a shard config then
2178
+ # we assume that we're the only instance sending.
2179
+ configured_instances = self._federation_shard_config.instances
2180
+ if not configured_instances:
2181
+ configured_instances = [self._instance_name]
2182
+ elif self._instance_name not in configured_instances:
2183
+ return
2184
+
2185
+ instances_in_table = self.db_pool.simple_select_onecol_txn(
2186
+ txn,
2187
+ table="federation_stream_position",
2188
+ keyvalues={},
2189
+ retcol="instance_name",
2190
+ )
2191
+
2192
+ if set(instances_in_table) == set(configured_instances):
2193
+ # Nothing to do
2194
+ return
2195
+
2196
+ sql = """
2197
+ SELECT type, MIN(stream_id) FROM federation_stream_position
2198
+ GROUP BY type
2199
+ """
2200
+ txn.execute(sql)
2201
+ min_positions = dict(
2202
+ cast(Iterable[tuple[str, int]], txn)
2203
+ ) # Map from type -> min position
2204
+
2205
+ # Ensure we do actually have some values here
2206
+ assert set(min_positions) == {"federation", "events"}
2207
+
2208
+ sql = """
2209
+ DELETE FROM federation_stream_position
2210
+ WHERE NOT (%s)
2211
+ """
2212
+ clause, args = make_in_list_sql_clause(
2213
+ txn.database_engine, "instance_name", configured_instances
2214
+ )
2215
+ txn.execute(sql % (clause,), args)
2216
+
2217
+ for typ, stream_id in min_positions.items():
2218
+ self.db_pool.simple_upsert_txn(
2219
+ txn,
2220
+ table="federation_stream_position",
2221
+ keyvalues={"type": typ, "instance_name": self._instance_name},
2222
+ values={"stream_id": stream_id},
2223
+ )
2224
+
2225
+ def has_room_changed_since(self, room_id: str, stream_id: int) -> bool:
2226
+ return self._events_stream_cache.has_entity_changed(room_id, stream_id)
2227
+
2228
+ def _paginate_room_events_by_topological_ordering_txn(
2229
+ self,
2230
+ txn: LoggingTransaction,
2231
+ room_id: str,
2232
+ from_token: RoomStreamToken,
2233
+ to_token: RoomStreamToken | None = None,
2234
+ direction: Direction = Direction.BACKWARDS,
2235
+ limit: int = 0,
2236
+ event_filter: Filter | None = None,
2237
+ ) -> tuple[list[_EventDictReturn], RoomStreamToken, bool]:
2238
+ """Returns list of events before or after a given token.
2239
+
2240
+ Args:
2241
+ txn
2242
+ room_id
2243
+ from_token: The token used to stream from
2244
+ to_token: A token which if given limits the results to only those before
2245
+ direction: Indicates whether we are paginating forwards or backwards
2246
+ from `from_key`.
2247
+ limit: The maximum number of events to return.
2248
+ event_filter: If provided filters the events to
2249
+ those that match the filter.
2250
+
2251
+ Returns:
2252
+ A list of _EventDictReturn, a token that points to the end of the
2253
+ result set, and a boolean to indicate if there were more events but
2254
+ we hit the limit. If no events are returned then the end of the
2255
+ stream has been reached (i.e. there are no events between
2256
+ `from_token` and `to_token`), or `limit` is zero.
2257
+ """
2258
+ # We can bail early if we're looking forwards, and our `to_key` is already
2259
+ # before our `from_token`.
2260
+ if (
2261
+ direction == Direction.FORWARDS
2262
+ and to_token is not None
2263
+ and to_token.is_before_or_eq(from_token)
2264
+ ):
2265
+ # Token selection matches what we do below if there are no rows
2266
+ return [], to_token if to_token else from_token, False
2267
+ # Or vice-versa, if we're looking backwards and our `from_token` is already before
2268
+ # our `to_token`.
2269
+ elif (
2270
+ direction == Direction.BACKWARDS
2271
+ and to_token is not None
2272
+ and from_token.is_before_or_eq(to_token)
2273
+ ):
2274
+ # Token selection matches what we do below if there are no rows
2275
+ return [], to_token if to_token else from_token, False
2276
+
2277
+ args: list[Any] = [room_id]
2278
+
2279
+ order, from_bound, to_bound = generate_pagination_bounds(
2280
+ direction, from_token, to_token
2281
+ )
2282
+
2283
+ bounds = generate_pagination_where_clause(
2284
+ direction=direction,
2285
+ column_names=("event.topological_ordering", "event.stream_ordering"),
2286
+ from_token=from_bound,
2287
+ to_token=to_bound,
2288
+ engine=self.database_engine,
2289
+ )
2290
+
2291
+ filter_clause, filter_args = filter_to_clause(event_filter)
2292
+
2293
+ if filter_clause:
2294
+ bounds += " AND " + filter_clause
2295
+ args.extend(filter_args)
2296
+
2297
+ # We fetch more events as we'll filter the result set
2298
+ requested_limit = int(limit) * 2
2299
+ args.append(int(limit) * 2)
2300
+
2301
+ select_keywords = "SELECT"
2302
+ join_clause = ""
2303
+ # Using DISTINCT in this SELECT query is quite expensive, because it
2304
+ # requires the engine to sort on the entire (not limited) result set,
2305
+ # i.e. the entire events table. Only use it in scenarios that could result
2306
+ # in the same event ID occurring multiple times in the results.
2307
+ needs_distinct = False
2308
+ if event_filter and event_filter.labels:
2309
+ # If we're not filtering on a label, then joining on event_labels will
2310
+ # return as many row for a single event as the number of labels it has. To
2311
+ # avoid this, only join if we're filtering on at least one label.
2312
+ join_clause += """
2313
+ LEFT JOIN event_labels
2314
+ USING (event_id, room_id, topological_ordering)
2315
+ """
2316
+ if len(event_filter.labels) > 1:
2317
+ # Multiple labels could cause the same event to appear multiple times.
2318
+ needs_distinct = True
2319
+
2320
+ # If there is a relation_senders and relation_types filter join to the
2321
+ # relations table to get events related to the current event.
2322
+ if event_filter and (
2323
+ event_filter.related_by_senders or event_filter.related_by_rel_types
2324
+ ):
2325
+ # Filtering by relations could cause the same event to appear multiple
2326
+ # times (since there's no limit on the number of relations to an event).
2327
+ needs_distinct = True
2328
+ join_clause += """
2329
+ LEFT JOIN event_relations AS relation ON (event.event_id = relation.relates_to_id)
2330
+ """
2331
+ if event_filter.related_by_senders:
2332
+ join_clause += """
2333
+ LEFT JOIN events AS related_event ON (relation.event_id = related_event.event_id)
2334
+ """
2335
+
2336
+ # If there is a not_rel_types filter join to the relations table to get
2337
+ # the event's relation information.
2338
+ if event_filter and (event_filter.rel_types or event_filter.not_rel_types):
2339
+ join_clause += """
2340
+ LEFT JOIN event_relations AS event_relation USING (event_id)
2341
+ """
2342
+
2343
+ if needs_distinct:
2344
+ select_keywords += " DISTINCT"
2345
+
2346
+ sql = """
2347
+ %(select_keywords)s
2348
+ event.event_id, event.instance_name,
2349
+ event.topological_ordering, event.stream_ordering
2350
+ FROM events AS event
2351
+ %(join_clause)s
2352
+ WHERE event.outlier = FALSE AND event.room_id = ? AND %(bounds)s
2353
+ ORDER BY event.topological_ordering %(order)s,
2354
+ event.stream_ordering %(order)s LIMIT ?
2355
+ """ % {
2356
+ "select_keywords": select_keywords,
2357
+ "join_clause": join_clause,
2358
+ "bounds": bounds,
2359
+ "order": order,
2360
+ }
2361
+ txn.execute(sql, args)
2362
+
2363
+ # Get all the rows and check if we hit the limit.
2364
+ fetched_rows = txn.fetchall()
2365
+ limited = len(fetched_rows) >= requested_limit
2366
+
2367
+ # Filter the result set.
2368
+ rows = [
2369
+ _EventDictReturn(event_id, topological_ordering, stream_ordering)
2370
+ for event_id, instance_name, topological_ordering, stream_ordering in fetched_rows
2371
+ if _filter_results(
2372
+ lower_token=(
2373
+ to_token if direction == Direction.BACKWARDS else from_token
2374
+ ),
2375
+ upper_token=(
2376
+ from_token if direction == Direction.BACKWARDS else to_token
2377
+ ),
2378
+ instance_name=instance_name,
2379
+ topological_ordering=topological_ordering,
2380
+ stream_ordering=stream_ordering,
2381
+ )
2382
+ ]
2383
+
2384
+ if len(rows) > limit:
2385
+ limited = True
2386
+
2387
+ rows = rows[:limit]
2388
+
2389
+ if rows:
2390
+ assert rows[-1].topological_ordering is not None
2391
+ next_token = generate_next_token(
2392
+ direction, rows[-1].topological_ordering, rows[-1].stream_ordering
2393
+ )
2394
+ else:
2395
+ # TODO (erikj): We should work out what to do here instead.
2396
+ next_token = to_token if to_token else from_token
2397
+
2398
+ return rows, next_token, limited
2399
+
2400
+ @trace
2401
+ @tag_args
2402
+ async def paginate_room_events_by_topological_ordering(
2403
+ self,
2404
+ *,
2405
+ room_id: str,
2406
+ from_key: RoomStreamToken,
2407
+ to_key: RoomStreamToken | None = None,
2408
+ direction: Direction = Direction.BACKWARDS,
2409
+ limit: int = 0,
2410
+ event_filter: Filter | None = None,
2411
+ ) -> tuple[list[EventBase], RoomStreamToken, bool]:
2412
+ """
2413
+ Paginate events by `topological_ordering` (tie-break with `stream_ordering`) in
2414
+ the room from the `from_key` in the given `direction` to the `to_key` or
2415
+ `limit`.
2416
+
2417
+ Args:
2418
+ room_id
2419
+ from_key: The token to stream from (starting point and heading in the given
2420
+ direction)
2421
+ to_key: The token representing the end stream position (end point)
2422
+ direction: Indicates whether we are paginating forwards or backwards
2423
+ from `from_key`.
2424
+ limit: Maximum number of events to return
2425
+ event_filter: If provided filters the events to those that match the filter.
2426
+
2427
+ Returns:
2428
+ The results as a list of events, a token that points to the end of
2429
+ the result set, and a boolean to indicate if there were more events
2430
+ but we hit the limit. If no events are returned then the end of the
2431
+ stream has been reached (i.e. there are no events between `from_key`
2432
+ and `to_key`).
2433
+
2434
+ When Direction.FORWARDS: from_key < x <= to_key, (ascending order)
2435
+ When Direction.BACKWARDS: from_key >= x > to_key, (descending order)
2436
+ """
2437
+
2438
+ # FIXME: When going forwards, we should enforce that the `to_key` is not `None`
2439
+ # because we always need an upper bound when querying the events stream (as
2440
+ # otherwise we'll potentially pick up events that are not fully persisted).
2441
+
2442
+ # We have these checks outside of the transaction function (txn) to save getting
2443
+ # a DB connection and switching threads if we don't need to.
2444
+ #
2445
+ # We can bail early if we're looking forwards, and our `to_key` is already
2446
+ # before our `from_key`.
2447
+ if (
2448
+ direction == Direction.FORWARDS
2449
+ and to_key is not None
2450
+ and to_key.is_before_or_eq(from_key)
2451
+ ):
2452
+ # Token selection matches what we do in `_paginate_room_events_txn` if there
2453
+ # are no rows
2454
+ return [], to_key if to_key else from_key, False
2455
+ # Or vice-versa, if we're looking backwards and our `from_key` is already before
2456
+ # our `to_key`.
2457
+ elif (
2458
+ direction == Direction.BACKWARDS
2459
+ and to_key is not None
2460
+ and from_key.is_before_or_eq(to_key)
2461
+ ):
2462
+ # Token selection matches what we do in `_paginate_room_events_txn` if there
2463
+ # are no rows
2464
+ return [], to_key if to_key else from_key, False
2465
+
2466
+ rows, token, limited = await self.db_pool.runInteraction(
2467
+ "paginate_room_events_by_topological_ordering",
2468
+ self._paginate_room_events_by_topological_ordering_txn,
2469
+ room_id,
2470
+ from_key,
2471
+ to_key,
2472
+ direction,
2473
+ limit,
2474
+ event_filter,
2475
+ )
2476
+
2477
+ events = await self.get_events_as_list(
2478
+ [r.event_id for r in rows], get_prev_content=True
2479
+ )
2480
+
2481
+ return events, token, limited
2482
+
2483
+ @cached()
2484
+ async def get_id_for_instance(self, instance_name: str) -> int:
2485
+ """Get a unique, immutable ID that corresponds to the given Synapse worker instance."""
2486
+
2487
+ def _get_id_for_instance_txn(txn: LoggingTransaction) -> int:
2488
+ instance_id = self.db_pool.simple_select_one_onecol_txn(
2489
+ txn,
2490
+ table="instance_map",
2491
+ keyvalues={"instance_name": instance_name},
2492
+ retcol="instance_id",
2493
+ allow_none=True,
2494
+ )
2495
+ if instance_id is not None:
2496
+ return instance_id
2497
+
2498
+ # If we don't have an entry upsert one.
2499
+ #
2500
+ # We could do this before the first check, and rely on the cache for
2501
+ # efficiency, but each UPSERT causes the next ID to increment which
2502
+ # can quickly bloat the size of the generated IDs for new instances.
2503
+ self.db_pool.simple_upsert_txn(
2504
+ txn,
2505
+ table="instance_map",
2506
+ keyvalues={"instance_name": instance_name},
2507
+ values={},
2508
+ )
2509
+
2510
+ return self.db_pool.simple_select_one_onecol_txn(
2511
+ txn,
2512
+ table="instance_map",
2513
+ keyvalues={"instance_name": instance_name},
2514
+ retcol="instance_id",
2515
+ )
2516
+
2517
+ return await self.db_pool.runInteraction(
2518
+ "get_id_for_instance", _get_id_for_instance_txn
2519
+ )
2520
+
2521
+ @cached()
2522
+ async def get_name_from_instance_id(self, instance_id: int) -> str:
2523
+ """Get the instance name from an ID previously returned by
2524
+ `get_id_for_instance`.
2525
+ """
2526
+
2527
+ return await self.db_pool.simple_select_one_onecol(
2528
+ table="instance_map",
2529
+ keyvalues={"instance_id": instance_id},
2530
+ retcol="instance_name",
2531
+ desc="get_name_from_instance_id",
2532
+ )
2533
+
2534
+ async def get_timeline_gaps(
2535
+ self,
2536
+ room_id: str,
2537
+ from_token: RoomStreamToken | None,
2538
+ to_token: RoomStreamToken,
2539
+ ) -> RoomStreamToken | None:
2540
+ """Check if there is a gap, and return a token that marks the position
2541
+ of the gap in the stream.
2542
+ """
2543
+
2544
+ sql = """
2545
+ SELECT instance_name, stream_ordering
2546
+ FROM timeline_gaps
2547
+ WHERE room_id = ? AND ? < stream_ordering AND stream_ordering <= ?
2548
+ ORDER BY stream_ordering
2549
+ """
2550
+
2551
+ rows = await self.db_pool.execute(
2552
+ "get_timeline_gaps",
2553
+ sql,
2554
+ room_id,
2555
+ from_token.stream if from_token else 0,
2556
+ to_token.get_max_stream_pos(),
2557
+ )
2558
+
2559
+ if not rows:
2560
+ return None
2561
+
2562
+ positions = [
2563
+ PersistedEventPosition(instance_name, stream_ordering)
2564
+ for instance_name, stream_ordering in rows
2565
+ ]
2566
+ if from_token:
2567
+ positions = [p for p in positions if p.persisted_after(from_token)]
2568
+
2569
+ positions = [p for p in positions if not p.persisted_after(to_token)]
2570
+
2571
+ if positions:
2572
+ # We return a stream token that ensures the event *at* the position
2573
+ # of the gap is included (as the gap is *before* the persisted
2574
+ # event).
2575
+ last_position = positions[-1]
2576
+ return RoomStreamToken(stream=last_position.stream - 1)
2577
+
2578
+ return None
2579
+
2580
+ @trace
2581
+ def get_rooms_that_might_have_updates(
2582
+ self, room_ids: StrCollection, from_token: RoomStreamToken
2583
+ ) -> StrCollection:
2584
+ """Filters given room IDs down to those that might have updates, i.e.
2585
+ removes rooms that definitely do not have updates.
2586
+ """
2587
+ return self._events_stream_cache.get_entities_changed(
2588
+ room_ids, from_token.stream
2589
+ )