matrix-synapse 1.143.0__cp310-abi3-manylinux_2_28_aarch64.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 (1058) hide show
  1. matrix_synapse-1.143.0.dist-info/AUTHORS.rst +51 -0
  2. matrix_synapse-1.143.0.dist-info/LICENSE-AGPL-3.0 +661 -0
  3. matrix_synapse-1.143.0.dist-info/LICENSE-COMMERCIAL +6 -0
  4. matrix_synapse-1.143.0.dist-info/METADATA +385 -0
  5. matrix_synapse-1.143.0.dist-info/RECORD +1058 -0
  6. matrix_synapse-1.143.0.dist-info/WHEEL +4 -0
  7. matrix_synapse-1.143.0.dist-info/entry_points.txt +14 -0
  8. synapse/__init__.py +97 -0
  9. synapse/_scripts/__init__.py +0 -0
  10. synapse/_scripts/export_signing_key.py +109 -0
  11. synapse/_scripts/generate_config.py +83 -0
  12. synapse/_scripts/generate_log_config.py +56 -0
  13. synapse/_scripts/generate_signing_key.py +55 -0
  14. synapse/_scripts/generate_workers_map.py +318 -0
  15. synapse/_scripts/hash_password.py +95 -0
  16. synapse/_scripts/move_remote_media_to_new_store.py +128 -0
  17. synapse/_scripts/register_new_matrix_user.py +402 -0
  18. synapse/_scripts/review_recent_signups.py +212 -0
  19. synapse/_scripts/synapse_port_db.py +1604 -0
  20. synapse/_scripts/synctl.py +365 -0
  21. synapse/_scripts/update_synapse_database.py +130 -0
  22. synapse/api/__init__.py +20 -0
  23. synapse/api/auth/__init__.py +207 -0
  24. synapse/api/auth/base.py +406 -0
  25. synapse/api/auth/internal.py +299 -0
  26. synapse/api/auth/mas.py +436 -0
  27. synapse/api/auth/msc3861_delegated.py +617 -0
  28. synapse/api/auth_blocking.py +144 -0
  29. synapse/api/constants.py +362 -0
  30. synapse/api/errors.py +907 -0
  31. synapse/api/filtering.py +537 -0
  32. synapse/api/presence.py +102 -0
  33. synapse/api/ratelimiting.py +480 -0
  34. synapse/api/room_versions.py +535 -0
  35. synapse/api/urls.py +118 -0
  36. synapse/app/__init__.py +60 -0
  37. synapse/app/_base.py +862 -0
  38. synapse/app/admin_cmd.py +388 -0
  39. synapse/app/appservice.py +30 -0
  40. synapse/app/client_reader.py +30 -0
  41. synapse/app/complement_fork_starter.py +206 -0
  42. synapse/app/event_creator.py +29 -0
  43. synapse/app/federation_reader.py +30 -0
  44. synapse/app/federation_sender.py +30 -0
  45. synapse/app/frontend_proxy.py +30 -0
  46. synapse/app/generic_worker.py +474 -0
  47. synapse/app/homeserver.py +505 -0
  48. synapse/app/media_repository.py +30 -0
  49. synapse/app/phone_stats_home.py +296 -0
  50. synapse/app/pusher.py +30 -0
  51. synapse/app/synchrotron.py +30 -0
  52. synapse/app/user_dir.py +31 -0
  53. synapse/appservice/__init__.py +458 -0
  54. synapse/appservice/api.py +567 -0
  55. synapse/appservice/scheduler.py +564 -0
  56. synapse/config/__init__.py +27 -0
  57. synapse/config/__main__.py +62 -0
  58. synapse/config/_base.py +1106 -0
  59. synapse/config/_base.pyi +215 -0
  60. synapse/config/_util.py +99 -0
  61. synapse/config/account_validity.py +116 -0
  62. synapse/config/api.py +141 -0
  63. synapse/config/appservice.py +210 -0
  64. synapse/config/auth.py +80 -0
  65. synapse/config/auto_accept_invites.py +43 -0
  66. synapse/config/background_updates.py +44 -0
  67. synapse/config/cache.py +231 -0
  68. synapse/config/captcha.py +90 -0
  69. synapse/config/cas.py +116 -0
  70. synapse/config/consent.py +73 -0
  71. synapse/config/database.py +184 -0
  72. synapse/config/emailconfig.py +367 -0
  73. synapse/config/experimental.py +595 -0
  74. synapse/config/federation.py +114 -0
  75. synapse/config/homeserver.py +141 -0
  76. synapse/config/jwt.py +55 -0
  77. synapse/config/key.py +447 -0
  78. synapse/config/logger.py +390 -0
  79. synapse/config/mas.py +192 -0
  80. synapse/config/matrixrtc.py +66 -0
  81. synapse/config/metrics.py +84 -0
  82. synapse/config/modules.py +40 -0
  83. synapse/config/oembed.py +185 -0
  84. synapse/config/oidc.py +509 -0
  85. synapse/config/password_auth_providers.py +82 -0
  86. synapse/config/push.py +64 -0
  87. synapse/config/ratelimiting.py +254 -0
  88. synapse/config/redis.py +74 -0
  89. synapse/config/registration.py +296 -0
  90. synapse/config/repository.py +311 -0
  91. synapse/config/retention.py +162 -0
  92. synapse/config/room.py +88 -0
  93. synapse/config/room_directory.py +165 -0
  94. synapse/config/saml2.py +251 -0
  95. synapse/config/server.py +1170 -0
  96. synapse/config/server_notices.py +84 -0
  97. synapse/config/spam_checker.py +66 -0
  98. synapse/config/sso.py +121 -0
  99. synapse/config/stats.py +54 -0
  100. synapse/config/third_party_event_rules.py +40 -0
  101. synapse/config/tls.py +192 -0
  102. synapse/config/tracer.py +71 -0
  103. synapse/config/user_directory.py +47 -0
  104. synapse/config/user_types.py +42 -0
  105. synapse/config/voip.py +59 -0
  106. synapse/config/workers.py +642 -0
  107. synapse/crypto/__init__.py +20 -0
  108. synapse/crypto/context_factory.py +278 -0
  109. synapse/crypto/event_signing.py +194 -0
  110. synapse/crypto/keyring.py +931 -0
  111. synapse/event_auth.py +1266 -0
  112. synapse/events/__init__.py +667 -0
  113. synapse/events/auto_accept_invites.py +216 -0
  114. synapse/events/builder.py +387 -0
  115. synapse/events/presence_router.py +243 -0
  116. synapse/events/snapshot.py +559 -0
  117. synapse/events/utils.py +924 -0
  118. synapse/events/validator.py +305 -0
  119. synapse/federation/__init__.py +22 -0
  120. synapse/federation/federation_base.py +382 -0
  121. synapse/federation/federation_client.py +2132 -0
  122. synapse/federation/federation_server.py +1540 -0
  123. synapse/federation/persistence.py +70 -0
  124. synapse/federation/send_queue.py +531 -0
  125. synapse/federation/sender/__init__.py +1164 -0
  126. synapse/federation/sender/per_destination_queue.py +886 -0
  127. synapse/federation/sender/transaction_manager.py +210 -0
  128. synapse/federation/transport/__init__.py +28 -0
  129. synapse/federation/transport/client.py +1199 -0
  130. synapse/federation/transport/server/__init__.py +334 -0
  131. synapse/federation/transport/server/_base.py +429 -0
  132. synapse/federation/transport/server/federation.py +910 -0
  133. synapse/federation/units.py +133 -0
  134. synapse/handlers/__init__.py +20 -0
  135. synapse/handlers/account.py +162 -0
  136. synapse/handlers/account_data.py +360 -0
  137. synapse/handlers/account_validity.py +361 -0
  138. synapse/handlers/admin.py +615 -0
  139. synapse/handlers/appservice.py +989 -0
  140. synapse/handlers/auth.py +2481 -0
  141. synapse/handlers/cas.py +413 -0
  142. synapse/handlers/deactivate_account.py +363 -0
  143. synapse/handlers/delayed_events.py +599 -0
  144. synapse/handlers/device.py +1870 -0
  145. synapse/handlers/devicemessage.py +399 -0
  146. synapse/handlers/directory.py +545 -0
  147. synapse/handlers/e2e_keys.py +1834 -0
  148. synapse/handlers/e2e_room_keys.py +455 -0
  149. synapse/handlers/event_auth.py +390 -0
  150. synapse/handlers/events.py +201 -0
  151. synapse/handlers/federation.py +2039 -0
  152. synapse/handlers/federation_event.py +2419 -0
  153. synapse/handlers/identity.py +812 -0
  154. synapse/handlers/initial_sync.py +528 -0
  155. synapse/handlers/jwt.py +120 -0
  156. synapse/handlers/message.py +2347 -0
  157. synapse/handlers/oidc.py +1801 -0
  158. synapse/handlers/pagination.py +768 -0
  159. synapse/handlers/password_policy.py +102 -0
  160. synapse/handlers/presence.py +2633 -0
  161. synapse/handlers/profile.py +655 -0
  162. synapse/handlers/push_rules.py +164 -0
  163. synapse/handlers/read_marker.py +79 -0
  164. synapse/handlers/receipts.py +351 -0
  165. synapse/handlers/register.py +1059 -0
  166. synapse/handlers/relations.py +623 -0
  167. synapse/handlers/reports.py +98 -0
  168. synapse/handlers/room.py +2448 -0
  169. synapse/handlers/room_list.py +632 -0
  170. synapse/handlers/room_member.py +2365 -0
  171. synapse/handlers/room_member_worker.py +146 -0
  172. synapse/handlers/room_policy.py +186 -0
  173. synapse/handlers/room_summary.py +1057 -0
  174. synapse/handlers/saml.py +524 -0
  175. synapse/handlers/search.py +723 -0
  176. synapse/handlers/send_email.py +209 -0
  177. synapse/handlers/set_password.py +71 -0
  178. synapse/handlers/sliding_sync/__init__.py +1701 -0
  179. synapse/handlers/sliding_sync/extensions.py +969 -0
  180. synapse/handlers/sliding_sync/room_lists.py +2262 -0
  181. synapse/handlers/sliding_sync/store.py +128 -0
  182. synapse/handlers/sso.py +1291 -0
  183. synapse/handlers/state_deltas.py +82 -0
  184. synapse/handlers/stats.py +321 -0
  185. synapse/handlers/sync.py +3106 -0
  186. synapse/handlers/thread_subscriptions.py +190 -0
  187. synapse/handlers/typing.py +606 -0
  188. synapse/handlers/ui_auth/__init__.py +48 -0
  189. synapse/handlers/ui_auth/checkers.py +332 -0
  190. synapse/handlers/user_directory.py +783 -0
  191. synapse/handlers/worker_lock.py +371 -0
  192. synapse/http/__init__.py +105 -0
  193. synapse/http/additional_resource.py +62 -0
  194. synapse/http/client.py +1373 -0
  195. synapse/http/connectproxyclient.py +316 -0
  196. synapse/http/federation/__init__.py +19 -0
  197. synapse/http/federation/matrix_federation_agent.py +490 -0
  198. synapse/http/federation/srv_resolver.py +196 -0
  199. synapse/http/federation/well_known_resolver.py +367 -0
  200. synapse/http/matrixfederationclient.py +1873 -0
  201. synapse/http/proxy.py +290 -0
  202. synapse/http/proxyagent.py +497 -0
  203. synapse/http/replicationagent.py +202 -0
  204. synapse/http/request_metrics.py +309 -0
  205. synapse/http/server.py +1110 -0
  206. synapse/http/servlet.py +1018 -0
  207. synapse/http/site.py +825 -0
  208. synapse/http/types.py +27 -0
  209. synapse/logging/__init__.py +31 -0
  210. synapse/logging/_remote.py +261 -0
  211. synapse/logging/_terse_json.py +95 -0
  212. synapse/logging/context.py +1209 -0
  213. synapse/logging/formatter.py +62 -0
  214. synapse/logging/handlers.py +99 -0
  215. synapse/logging/loggers.py +25 -0
  216. synapse/logging/opentracing.py +1132 -0
  217. synapse/logging/scopecontextmanager.py +160 -0
  218. synapse/media/_base.py +830 -0
  219. synapse/media/filepath.py +417 -0
  220. synapse/media/media_repository.py +1580 -0
  221. synapse/media/media_storage.py +702 -0
  222. synapse/media/oembed.py +277 -0
  223. synapse/media/preview_html.py +556 -0
  224. synapse/media/storage_provider.py +195 -0
  225. synapse/media/thumbnailer.py +833 -0
  226. synapse/media/url_previewer.py +875 -0
  227. synapse/metrics/__init__.py +748 -0
  228. synapse/metrics/_gc.py +219 -0
  229. synapse/metrics/_reactor_metrics.py +171 -0
  230. synapse/metrics/_types.py +38 -0
  231. synapse/metrics/background_process_metrics.py +555 -0
  232. synapse/metrics/common_usage_metrics.py +94 -0
  233. synapse/metrics/jemalloc.py +248 -0
  234. synapse/module_api/__init__.py +2131 -0
  235. synapse/module_api/callbacks/__init__.py +50 -0
  236. synapse/module_api/callbacks/account_validity_callbacks.py +106 -0
  237. synapse/module_api/callbacks/media_repository_callbacks.py +157 -0
  238. synapse/module_api/callbacks/ratelimit_callbacks.py +78 -0
  239. synapse/module_api/callbacks/spamchecker_callbacks.py +991 -0
  240. synapse/module_api/callbacks/third_party_event_rules_callbacks.py +592 -0
  241. synapse/module_api/errors.py +42 -0
  242. synapse/notifier.py +970 -0
  243. synapse/push/__init__.py +212 -0
  244. synapse/push/bulk_push_rule_evaluator.py +635 -0
  245. synapse/push/clientformat.py +126 -0
  246. synapse/push/emailpusher.py +333 -0
  247. synapse/push/httppusher.py +564 -0
  248. synapse/push/mailer.py +1010 -0
  249. synapse/push/presentable_names.py +216 -0
  250. synapse/push/push_tools.py +114 -0
  251. synapse/push/push_types.py +141 -0
  252. synapse/push/pusher.py +87 -0
  253. synapse/push/pusherpool.py +501 -0
  254. synapse/push/rulekinds.py +33 -0
  255. synapse/py.typed +0 -0
  256. synapse/replication/__init__.py +20 -0
  257. synapse/replication/http/__init__.py +68 -0
  258. synapse/replication/http/_base.py +468 -0
  259. synapse/replication/http/account_data.py +297 -0
  260. synapse/replication/http/deactivate_account.py +81 -0
  261. synapse/replication/http/delayed_events.py +62 -0
  262. synapse/replication/http/devices.py +254 -0
  263. synapse/replication/http/federation.py +334 -0
  264. synapse/replication/http/login.py +106 -0
  265. synapse/replication/http/membership.py +364 -0
  266. synapse/replication/http/presence.py +133 -0
  267. synapse/replication/http/push.py +156 -0
  268. synapse/replication/http/register.py +172 -0
  269. synapse/replication/http/send_events.py +182 -0
  270. synapse/replication/http/state.py +82 -0
  271. synapse/replication/http/streams.py +101 -0
  272. synapse/replication/tcp/__init__.py +56 -0
  273. synapse/replication/tcp/client.py +552 -0
  274. synapse/replication/tcp/commands.py +569 -0
  275. synapse/replication/tcp/context.py +41 -0
  276. synapse/replication/tcp/external_cache.py +156 -0
  277. synapse/replication/tcp/handler.py +922 -0
  278. synapse/replication/tcp/protocol.py +608 -0
  279. synapse/replication/tcp/redis.py +509 -0
  280. synapse/replication/tcp/resource.py +348 -0
  281. synapse/replication/tcp/streams/__init__.py +96 -0
  282. synapse/replication/tcp/streams/_base.py +765 -0
  283. synapse/replication/tcp/streams/events.py +287 -0
  284. synapse/replication/tcp/streams/federation.py +92 -0
  285. synapse/replication/tcp/streams/partial_state.py +80 -0
  286. synapse/res/providers.json +29 -0
  287. synapse/res/templates/_base.html +29 -0
  288. synapse/res/templates/account_previously_renewed.html +6 -0
  289. synapse/res/templates/account_renewed.html +6 -0
  290. synapse/res/templates/add_threepid.html +8 -0
  291. synapse/res/templates/add_threepid.txt +6 -0
  292. synapse/res/templates/add_threepid_failure.html +7 -0
  293. synapse/res/templates/add_threepid_success.html +6 -0
  294. synapse/res/templates/already_in_use.html +12 -0
  295. synapse/res/templates/already_in_use.txt +10 -0
  296. synapse/res/templates/auth_success.html +21 -0
  297. synapse/res/templates/invalid_token.html +6 -0
  298. synapse/res/templates/mail-Element.css +7 -0
  299. synapse/res/templates/mail-Vector.css +7 -0
  300. synapse/res/templates/mail-expiry.css +4 -0
  301. synapse/res/templates/mail.css +156 -0
  302. synapse/res/templates/notice_expiry.html +46 -0
  303. synapse/res/templates/notice_expiry.txt +7 -0
  304. synapse/res/templates/notif.html +51 -0
  305. synapse/res/templates/notif.txt +22 -0
  306. synapse/res/templates/notif_mail.html +59 -0
  307. synapse/res/templates/notif_mail.txt +10 -0
  308. synapse/res/templates/password_reset.html +10 -0
  309. synapse/res/templates/password_reset.txt +7 -0
  310. synapse/res/templates/password_reset_confirmation.html +15 -0
  311. synapse/res/templates/password_reset_failure.html +7 -0
  312. synapse/res/templates/password_reset_success.html +6 -0
  313. synapse/res/templates/recaptcha.html +42 -0
  314. synapse/res/templates/registration.html +12 -0
  315. synapse/res/templates/registration.txt +10 -0
  316. synapse/res/templates/registration_failure.html +6 -0
  317. synapse/res/templates/registration_success.html +6 -0
  318. synapse/res/templates/registration_token.html +18 -0
  319. synapse/res/templates/room.html +33 -0
  320. synapse/res/templates/room.txt +9 -0
  321. synapse/res/templates/sso.css +129 -0
  322. synapse/res/templates/sso_account_deactivated.html +25 -0
  323. synapse/res/templates/sso_auth_account_details.html +186 -0
  324. synapse/res/templates/sso_auth_account_details.js +116 -0
  325. synapse/res/templates/sso_auth_bad_user.html +26 -0
  326. synapse/res/templates/sso_auth_confirm.html +27 -0
  327. synapse/res/templates/sso_auth_success.html +26 -0
  328. synapse/res/templates/sso_error.html +71 -0
  329. synapse/res/templates/sso_footer.html +19 -0
  330. synapse/res/templates/sso_login_idp_picker.html +60 -0
  331. synapse/res/templates/sso_new_user_consent.html +30 -0
  332. synapse/res/templates/sso_partial_profile.html +19 -0
  333. synapse/res/templates/sso_redirect_confirm.html +39 -0
  334. synapse/res/templates/style.css +33 -0
  335. synapse/res/templates/terms.html +27 -0
  336. synapse/rest/__init__.py +197 -0
  337. synapse/rest/admin/__init__.py +390 -0
  338. synapse/rest/admin/_base.py +72 -0
  339. synapse/rest/admin/background_updates.py +171 -0
  340. synapse/rest/admin/devices.py +221 -0
  341. synapse/rest/admin/event_reports.py +173 -0
  342. synapse/rest/admin/events.py +69 -0
  343. synapse/rest/admin/experimental_features.py +137 -0
  344. synapse/rest/admin/federation.py +243 -0
  345. synapse/rest/admin/media.py +540 -0
  346. synapse/rest/admin/registration_tokens.py +358 -0
  347. synapse/rest/admin/rooms.py +1061 -0
  348. synapse/rest/admin/scheduled_tasks.py +70 -0
  349. synapse/rest/admin/server_notice_servlet.py +132 -0
  350. synapse/rest/admin/statistics.py +132 -0
  351. synapse/rest/admin/username_available.py +58 -0
  352. synapse/rest/admin/users.py +1606 -0
  353. synapse/rest/client/__init__.py +20 -0
  354. synapse/rest/client/_base.py +113 -0
  355. synapse/rest/client/account.py +930 -0
  356. synapse/rest/client/account_data.py +319 -0
  357. synapse/rest/client/account_validity.py +103 -0
  358. synapse/rest/client/appservice_ping.py +125 -0
  359. synapse/rest/client/auth.py +218 -0
  360. synapse/rest/client/auth_metadata.py +122 -0
  361. synapse/rest/client/capabilities.py +121 -0
  362. synapse/rest/client/delayed_events.py +165 -0
  363. synapse/rest/client/devices.py +587 -0
  364. synapse/rest/client/directory.py +211 -0
  365. synapse/rest/client/events.py +116 -0
  366. synapse/rest/client/filter.py +112 -0
  367. synapse/rest/client/initial_sync.py +65 -0
  368. synapse/rest/client/keys.py +678 -0
  369. synapse/rest/client/knock.py +104 -0
  370. synapse/rest/client/login.py +750 -0
  371. synapse/rest/client/login_token_request.py +127 -0
  372. synapse/rest/client/logout.py +93 -0
  373. synapse/rest/client/matrixrtc.py +52 -0
  374. synapse/rest/client/media.py +285 -0
  375. synapse/rest/client/mutual_rooms.py +93 -0
  376. synapse/rest/client/notifications.py +137 -0
  377. synapse/rest/client/openid.py +109 -0
  378. synapse/rest/client/password_policy.py +69 -0
  379. synapse/rest/client/presence.py +131 -0
  380. synapse/rest/client/profile.py +291 -0
  381. synapse/rest/client/push_rule.py +331 -0
  382. synapse/rest/client/pusher.py +181 -0
  383. synapse/rest/client/read_marker.py +104 -0
  384. synapse/rest/client/receipts.py +165 -0
  385. synapse/rest/client/register.py +1067 -0
  386. synapse/rest/client/relations.py +138 -0
  387. synapse/rest/client/rendezvous.py +76 -0
  388. synapse/rest/client/reporting.py +207 -0
  389. synapse/rest/client/room.py +1669 -0
  390. synapse/rest/client/room_keys.py +426 -0
  391. synapse/rest/client/room_upgrade_rest_servlet.py +112 -0
  392. synapse/rest/client/sendtodevice.py +85 -0
  393. synapse/rest/client/sync.py +1131 -0
  394. synapse/rest/client/tags.py +129 -0
  395. synapse/rest/client/thirdparty.py +130 -0
  396. synapse/rest/client/thread_subscriptions.py +247 -0
  397. synapse/rest/client/tokenrefresh.py +52 -0
  398. synapse/rest/client/transactions.py +149 -0
  399. synapse/rest/client/user_directory.py +90 -0
  400. synapse/rest/client/versions.py +191 -0
  401. synapse/rest/client/voip.py +88 -0
  402. synapse/rest/consent/__init__.py +0 -0
  403. synapse/rest/consent/consent_resource.py +210 -0
  404. synapse/rest/health.py +38 -0
  405. synapse/rest/key/__init__.py +20 -0
  406. synapse/rest/key/v2/__init__.py +40 -0
  407. synapse/rest/key/v2/local_key_resource.py +125 -0
  408. synapse/rest/key/v2/remote_key_resource.py +302 -0
  409. synapse/rest/media/__init__.py +0 -0
  410. synapse/rest/media/config_resource.py +53 -0
  411. synapse/rest/media/create_resource.py +90 -0
  412. synapse/rest/media/download_resource.py +110 -0
  413. synapse/rest/media/media_repository_resource.py +113 -0
  414. synapse/rest/media/preview_url_resource.py +77 -0
  415. synapse/rest/media/thumbnail_resource.py +142 -0
  416. synapse/rest/media/upload_resource.py +187 -0
  417. synapse/rest/media/v1/__init__.py +39 -0
  418. synapse/rest/media/v1/_base.py +23 -0
  419. synapse/rest/media/v1/media_storage.py +23 -0
  420. synapse/rest/media/v1/storage_provider.py +23 -0
  421. synapse/rest/synapse/__init__.py +20 -0
  422. synapse/rest/synapse/client/__init__.py +93 -0
  423. synapse/rest/synapse/client/federation_whitelist.py +66 -0
  424. synapse/rest/synapse/client/jwks.py +77 -0
  425. synapse/rest/synapse/client/new_user_consent.py +115 -0
  426. synapse/rest/synapse/client/oidc/__init__.py +45 -0
  427. synapse/rest/synapse/client/oidc/backchannel_logout_resource.py +42 -0
  428. synapse/rest/synapse/client/oidc/callback_resource.py +48 -0
  429. synapse/rest/synapse/client/password_reset.py +129 -0
  430. synapse/rest/synapse/client/pick_idp.py +107 -0
  431. synapse/rest/synapse/client/pick_username.py +153 -0
  432. synapse/rest/synapse/client/rendezvous.py +58 -0
  433. synapse/rest/synapse/client/saml2/__init__.py +42 -0
  434. synapse/rest/synapse/client/saml2/metadata_resource.py +46 -0
  435. synapse/rest/synapse/client/saml2/response_resource.py +52 -0
  436. synapse/rest/synapse/client/sso_register.py +56 -0
  437. synapse/rest/synapse/client/unsubscribe.py +88 -0
  438. synapse/rest/synapse/mas/__init__.py +71 -0
  439. synapse/rest/synapse/mas/_base.py +55 -0
  440. synapse/rest/synapse/mas/devices.py +239 -0
  441. synapse/rest/synapse/mas/users.py +469 -0
  442. synapse/rest/well_known.py +148 -0
  443. synapse/server.py +1257 -0
  444. synapse/server_notices/__init__.py +0 -0
  445. synapse/server_notices/consent_server_notices.py +136 -0
  446. synapse/server_notices/resource_limits_server_notices.py +215 -0
  447. synapse/server_notices/server_notices_manager.py +388 -0
  448. synapse/server_notices/server_notices_sender.py +67 -0
  449. synapse/server_notices/worker_server_notices_sender.py +46 -0
  450. synapse/spam_checker_api/__init__.py +31 -0
  451. synapse/state/__init__.py +1022 -0
  452. synapse/state/v1.py +369 -0
  453. synapse/state/v2.py +984 -0
  454. synapse/static/client/login/index.html +47 -0
  455. synapse/static/client/login/js/jquery-3.4.1.min.js +2 -0
  456. synapse/static/client/login/js/login.js +291 -0
  457. synapse/static/client/login/spinner.gif +0 -0
  458. synapse/static/client/login/style.css +79 -0
  459. synapse/static/index.html +63 -0
  460. synapse/storage/__init__.py +43 -0
  461. synapse/storage/_base.py +245 -0
  462. synapse/storage/admin_client_config.py +25 -0
  463. synapse/storage/background_updates.py +1188 -0
  464. synapse/storage/controllers/__init__.py +57 -0
  465. synapse/storage/controllers/persist_events.py +1237 -0
  466. synapse/storage/controllers/purge_events.py +455 -0
  467. synapse/storage/controllers/state.py +950 -0
  468. synapse/storage/controllers/stats.py +119 -0
  469. synapse/storage/database.py +2719 -0
  470. synapse/storage/databases/__init__.py +175 -0
  471. synapse/storage/databases/main/__init__.py +420 -0
  472. synapse/storage/databases/main/account_data.py +1059 -0
  473. synapse/storage/databases/main/appservice.py +473 -0
  474. synapse/storage/databases/main/cache.py +911 -0
  475. synapse/storage/databases/main/censor_events.py +225 -0
  476. synapse/storage/databases/main/client_ips.py +815 -0
  477. synapse/storage/databases/main/delayed_events.py +562 -0
  478. synapse/storage/databases/main/deviceinbox.py +1271 -0
  479. synapse/storage/databases/main/devices.py +2578 -0
  480. synapse/storage/databases/main/directory.py +212 -0
  481. synapse/storage/databases/main/e2e_room_keys.py +689 -0
  482. synapse/storage/databases/main/end_to_end_keys.py +1894 -0
  483. synapse/storage/databases/main/event_federation.py +2508 -0
  484. synapse/storage/databases/main/event_push_actions.py +1933 -0
  485. synapse/storage/databases/main/events.py +3765 -0
  486. synapse/storage/databases/main/events_bg_updates.py +2910 -0
  487. synapse/storage/databases/main/events_forward_extremities.py +126 -0
  488. synapse/storage/databases/main/events_worker.py +2786 -0
  489. synapse/storage/databases/main/experimental_features.py +130 -0
  490. synapse/storage/databases/main/filtering.py +231 -0
  491. synapse/storage/databases/main/keys.py +291 -0
  492. synapse/storage/databases/main/lock.py +553 -0
  493. synapse/storage/databases/main/media_repository.py +1068 -0
  494. synapse/storage/databases/main/metrics.py +460 -0
  495. synapse/storage/databases/main/monthly_active_users.py +443 -0
  496. synapse/storage/databases/main/openid.py +60 -0
  497. synapse/storage/databases/main/presence.py +509 -0
  498. synapse/storage/databases/main/profile.py +539 -0
  499. synapse/storage/databases/main/purge_events.py +521 -0
  500. synapse/storage/databases/main/push_rule.py +970 -0
  501. synapse/storage/databases/main/pusher.py +793 -0
  502. synapse/storage/databases/main/receipts.py +1341 -0
  503. synapse/storage/databases/main/registration.py +3072 -0
  504. synapse/storage/databases/main/rejections.py +37 -0
  505. synapse/storage/databases/main/relations.py +1116 -0
  506. synapse/storage/databases/main/room.py +2779 -0
  507. synapse/storage/databases/main/roommember.py +2110 -0
  508. synapse/storage/databases/main/search.py +939 -0
  509. synapse/storage/databases/main/session.py +151 -0
  510. synapse/storage/databases/main/signatures.py +94 -0
  511. synapse/storage/databases/main/sliding_sync.py +603 -0
  512. synapse/storage/databases/main/state.py +1002 -0
  513. synapse/storage/databases/main/state_deltas.py +329 -0
  514. synapse/storage/databases/main/stats.py +789 -0
  515. synapse/storage/databases/main/stream.py +2577 -0
  516. synapse/storage/databases/main/tags.py +360 -0
  517. synapse/storage/databases/main/task_scheduler.py +225 -0
  518. synapse/storage/databases/main/thread_subscriptions.py +589 -0
  519. synapse/storage/databases/main/transactions.py +675 -0
  520. synapse/storage/databases/main/ui_auth.py +420 -0
  521. synapse/storage/databases/main/user_directory.py +1330 -0
  522. synapse/storage/databases/main/user_erasure_store.py +117 -0
  523. synapse/storage/databases/state/__init__.py +22 -0
  524. synapse/storage/databases/state/bg_updates.py +497 -0
  525. synapse/storage/databases/state/deletion.py +557 -0
  526. synapse/storage/databases/state/store.py +948 -0
  527. synapse/storage/engines/__init__.py +70 -0
  528. synapse/storage/engines/_base.py +154 -0
  529. synapse/storage/engines/postgres.py +261 -0
  530. synapse/storage/engines/sqlite.py +199 -0
  531. synapse/storage/invite_rule.py +112 -0
  532. synapse/storage/keys.py +40 -0
  533. synapse/storage/prepare_database.py +730 -0
  534. synapse/storage/push_rule.py +28 -0
  535. synapse/storage/roommember.py +88 -0
  536. synapse/storage/schema/README.md +4 -0
  537. synapse/storage/schema/__init__.py +186 -0
  538. synapse/storage/schema/common/delta/25/00background_updates.sql +40 -0
  539. synapse/storage/schema/common/delta/35/00background_updates_add_col.sql +36 -0
  540. synapse/storage/schema/common/delta/58/00background_update_ordering.sql +38 -0
  541. synapse/storage/schema/common/full_schemas/72/full.sql.postgres +8 -0
  542. synapse/storage/schema/common/full_schemas/72/full.sql.sqlite +6 -0
  543. synapse/storage/schema/common/schema_version.sql +60 -0
  544. synapse/storage/schema/main/delta/12/v12.sql +82 -0
  545. synapse/storage/schema/main/delta/13/v13.sql +38 -0
  546. synapse/storage/schema/main/delta/14/v14.sql +42 -0
  547. synapse/storage/schema/main/delta/15/appservice_txns.sql +50 -0
  548. synapse/storage/schema/main/delta/15/presence_indices.sql +2 -0
  549. synapse/storage/schema/main/delta/15/v15.sql +24 -0
  550. synapse/storage/schema/main/delta/16/events_order_index.sql +4 -0
  551. synapse/storage/schema/main/delta/16/remote_media_cache_index.sql +2 -0
  552. synapse/storage/schema/main/delta/16/remove_duplicates.sql +9 -0
  553. synapse/storage/schema/main/delta/16/room_alias_index.sql +3 -0
  554. synapse/storage/schema/main/delta/16/unique_constraints.sql +72 -0
  555. synapse/storage/schema/main/delta/16/users.sql +56 -0
  556. synapse/storage/schema/main/delta/17/drop_indexes.sql +37 -0
  557. synapse/storage/schema/main/delta/17/server_keys.sql +43 -0
  558. synapse/storage/schema/main/delta/17/user_threepids.sql +9 -0
  559. synapse/storage/schema/main/delta/18/server_keys_bigger_ints.sql +51 -0
  560. synapse/storage/schema/main/delta/19/event_index.sql +38 -0
  561. synapse/storage/schema/main/delta/20/dummy.sql +1 -0
  562. synapse/storage/schema/main/delta/20/pushers.py +93 -0
  563. synapse/storage/schema/main/delta/21/end_to_end_keys.sql +53 -0
  564. synapse/storage/schema/main/delta/21/receipts.sql +57 -0
  565. synapse/storage/schema/main/delta/22/receipts_index.sql +41 -0
  566. synapse/storage/schema/main/delta/22/user_threepids_unique.sql +19 -0
  567. synapse/storage/schema/main/delta/24/stats_reporting.sql +37 -0
  568. synapse/storage/schema/main/delta/25/fts.py +81 -0
  569. synapse/storage/schema/main/delta/25/guest_access.sql +44 -0
  570. synapse/storage/schema/main/delta/25/history_visibility.sql +44 -0
  571. synapse/storage/schema/main/delta/25/tags.sql +57 -0
  572. synapse/storage/schema/main/delta/26/account_data.sql +36 -0
  573. synapse/storage/schema/main/delta/27/account_data.sql +55 -0
  574. synapse/storage/schema/main/delta/27/forgotten_memberships.sql +45 -0
  575. synapse/storage/schema/main/delta/27/ts.py +61 -0
  576. synapse/storage/schema/main/delta/28/event_push_actions.sql +46 -0
  577. synapse/storage/schema/main/delta/28/events_room_stream.sql +39 -0
  578. synapse/storage/schema/main/delta/28/public_roms_index.sql +39 -0
  579. synapse/storage/schema/main/delta/28/receipts_user_id_index.sql +41 -0
  580. synapse/storage/schema/main/delta/28/upgrade_times.sql +40 -0
  581. synapse/storage/schema/main/delta/28/users_is_guest.sql +41 -0
  582. synapse/storage/schema/main/delta/29/push_actions.sql +54 -0
  583. synapse/storage/schema/main/delta/30/alias_creator.sql +35 -0
  584. synapse/storage/schema/main/delta/30/as_users.py +82 -0
  585. synapse/storage/schema/main/delta/30/deleted_pushers.sql +44 -0
  586. synapse/storage/schema/main/delta/30/presence_stream.sql +49 -0
  587. synapse/storage/schema/main/delta/30/public_rooms.sql +42 -0
  588. synapse/storage/schema/main/delta/30/push_rule_stream.sql +57 -0
  589. synapse/storage/schema/main/delta/30/threepid_guest_access_tokens.sql +43 -0
  590. synapse/storage/schema/main/delta/31/invites.sql +61 -0
  591. synapse/storage/schema/main/delta/31/local_media_repository_url_cache.sql +46 -0
  592. synapse/storage/schema/main/delta/31/pushers_0.py +92 -0
  593. synapse/storage/schema/main/delta/31/pushers_index.sql +41 -0
  594. synapse/storage/schema/main/delta/31/search_update.py +65 -0
  595. synapse/storage/schema/main/delta/32/events.sql +35 -0
  596. synapse/storage/schema/main/delta/32/openid.sql +9 -0
  597. synapse/storage/schema/main/delta/32/pusher_throttle.sql +42 -0
  598. synapse/storage/schema/main/delta/32/remove_indices.sql +52 -0
  599. synapse/storage/schema/main/delta/32/reports.sql +44 -0
  600. synapse/storage/schema/main/delta/33/access_tokens_device_index.sql +36 -0
  601. synapse/storage/schema/main/delta/33/devices.sql +40 -0
  602. synapse/storage/schema/main/delta/33/devices_for_e2e_keys.sql +38 -0
  603. synapse/storage/schema/main/delta/33/devices_for_e2e_keys_clear_unknown_device.sql +39 -0
  604. synapse/storage/schema/main/delta/33/event_fields.py +61 -0
  605. synapse/storage/schema/main/delta/33/remote_media_ts.py +43 -0
  606. synapse/storage/schema/main/delta/33/user_ips_index.sql +36 -0
  607. synapse/storage/schema/main/delta/34/appservice_stream.sql +42 -0
  608. synapse/storage/schema/main/delta/34/cache_stream.py +50 -0
  609. synapse/storage/schema/main/delta/34/device_inbox.sql +43 -0
  610. synapse/storage/schema/main/delta/34/push_display_name_rename.sql +39 -0
  611. synapse/storage/schema/main/delta/34/received_txn_purge.py +36 -0
  612. synapse/storage/schema/main/delta/35/contains_url.sql +36 -0
  613. synapse/storage/schema/main/delta/35/device_outbox.sql +58 -0
  614. synapse/storage/schema/main/delta/35/device_stream_id.sql +40 -0
  615. synapse/storage/schema/main/delta/35/event_push_actions_index.sql +36 -0
  616. synapse/storage/schema/main/delta/35/public_room_list_change_stream.sql +52 -0
  617. synapse/storage/schema/main/delta/35/stream_order_to_extrem.sql +56 -0
  618. synapse/storage/schema/main/delta/36/readd_public_rooms.sql +45 -0
  619. synapse/storage/schema/main/delta/37/remove_auth_idx.py +89 -0
  620. synapse/storage/schema/main/delta/37/user_threepids.sql +71 -0
  621. synapse/storage/schema/main/delta/38/postgres_fts_gist.sql +38 -0
  622. synapse/storage/schema/main/delta/39/appservice_room_list.sql +48 -0
  623. synapse/storage/schema/main/delta/39/device_federation_stream_idx.sql +35 -0
  624. synapse/storage/schema/main/delta/39/event_push_index.sql +36 -0
  625. synapse/storage/schema/main/delta/39/federation_out_position.sql +41 -0
  626. synapse/storage/schema/main/delta/39/membership_profile.sql +39 -0
  627. synapse/storage/schema/main/delta/40/current_state_idx.sql +36 -0
  628. synapse/storage/schema/main/delta/40/device_inbox.sql +40 -0
  629. synapse/storage/schema/main/delta/40/device_list_streams.sql +79 -0
  630. synapse/storage/schema/main/delta/40/event_push_summary.sql +57 -0
  631. synapse/storage/schema/main/delta/40/pushers.sql +58 -0
  632. synapse/storage/schema/main/delta/41/device_list_stream_idx.sql +36 -0
  633. synapse/storage/schema/main/delta/41/device_outbound_index.sql +35 -0
  634. synapse/storage/schema/main/delta/41/event_search_event_id_idx.sql +36 -0
  635. synapse/storage/schema/main/delta/41/ratelimit.sql +41 -0
  636. synapse/storage/schema/main/delta/42/current_state_delta.sql +48 -0
  637. synapse/storage/schema/main/delta/42/device_list_last_id.sql +52 -0
  638. synapse/storage/schema/main/delta/42/event_auth_state_only.sql +36 -0
  639. synapse/storage/schema/main/delta/42/user_dir.py +88 -0
  640. synapse/storage/schema/main/delta/43/blocked_rooms.sql +40 -0
  641. synapse/storage/schema/main/delta/43/quarantine_media.sql +36 -0
  642. synapse/storage/schema/main/delta/43/url_cache.sql +35 -0
  643. synapse/storage/schema/main/delta/43/user_share.sql +52 -0
  644. synapse/storage/schema/main/delta/44/expire_url_cache.sql +60 -0
  645. synapse/storage/schema/main/delta/45/group_server.sql +186 -0
  646. synapse/storage/schema/main/delta/45/profile_cache.sql +47 -0
  647. synapse/storage/schema/main/delta/46/drop_refresh_tokens.sql +36 -0
  648. synapse/storage/schema/main/delta/46/drop_unique_deleted_pushers.sql +54 -0
  649. synapse/storage/schema/main/delta/46/group_server.sql +51 -0
  650. synapse/storage/schema/main/delta/46/local_media_repository_url_idx.sql +43 -0
  651. synapse/storage/schema/main/delta/46/user_dir_null_room_ids.sql +54 -0
  652. synapse/storage/schema/main/delta/46/user_dir_typos.sql +43 -0
  653. synapse/storage/schema/main/delta/47/last_access_media.sql +35 -0
  654. synapse/storage/schema/main/delta/47/postgres_fts_gin.sql +36 -0
  655. synapse/storage/schema/main/delta/47/push_actions_staging.sql +47 -0
  656. synapse/storage/schema/main/delta/48/add_user_consent.sql +37 -0
  657. synapse/storage/schema/main/delta/48/add_user_ips_last_seen_index.sql +36 -0
  658. synapse/storage/schema/main/delta/48/deactivated_users.sql +44 -0
  659. synapse/storage/schema/main/delta/48/group_unique_indexes.py +67 -0
  660. synapse/storage/schema/main/delta/48/groups_joinable.sql +41 -0
  661. synapse/storage/schema/main/delta/49/add_user_consent_server_notice_sent.sql +39 -0
  662. synapse/storage/schema/main/delta/49/add_user_daily_visits.sql +40 -0
  663. synapse/storage/schema/main/delta/49/add_user_ips_last_seen_only_index.sql +36 -0
  664. synapse/storage/schema/main/delta/50/add_creation_ts_users_index.sql +38 -0
  665. synapse/storage/schema/main/delta/50/erasure_store.sql +40 -0
  666. synapse/storage/schema/main/delta/50/make_event_content_nullable.py +102 -0
  667. synapse/storage/schema/main/delta/51/e2e_room_keys.sql +58 -0
  668. synapse/storage/schema/main/delta/51/monthly_active_users.sql +46 -0
  669. synapse/storage/schema/main/delta/52/add_event_to_state_group_index.sql +38 -0
  670. synapse/storage/schema/main/delta/52/device_list_streams_unique_idx.sql +55 -0
  671. synapse/storage/schema/main/delta/52/e2e_room_keys.sql +72 -0
  672. synapse/storage/schema/main/delta/53/add_user_type_to_users.sql +38 -0
  673. synapse/storage/schema/main/delta/53/drop_sent_transactions.sql +35 -0
  674. synapse/storage/schema/main/delta/53/event_format_version.sql +35 -0
  675. synapse/storage/schema/main/delta/53/user_dir_populate.sql +49 -0
  676. synapse/storage/schema/main/delta/53/user_ips_index.sql +49 -0
  677. synapse/storage/schema/main/delta/53/user_share.sql +63 -0
  678. synapse/storage/schema/main/delta/53/user_threepid_id.sql +48 -0
  679. synapse/storage/schema/main/delta/53/users_in_public_rooms.sql +47 -0
  680. synapse/storage/schema/main/delta/54/account_validity_with_renewal.sql +49 -0
  681. synapse/storage/schema/main/delta/54/add_validity_to_server_keys.sql +42 -0
  682. synapse/storage/schema/main/delta/54/delete_forward_extremities.sql +42 -0
  683. synapse/storage/schema/main/delta/54/drop_legacy_tables.sql +49 -0
  684. synapse/storage/schema/main/delta/54/drop_presence_list.sql +35 -0
  685. synapse/storage/schema/main/delta/54/relations.sql +46 -0
  686. synapse/storage/schema/main/delta/54/stats.sql +99 -0
  687. synapse/storage/schema/main/delta/54/stats2.sql +47 -0
  688. synapse/storage/schema/main/delta/55/access_token_expiry.sql +37 -0
  689. synapse/storage/schema/main/delta/55/track_threepid_validations.sql +50 -0
  690. synapse/storage/schema/main/delta/55/users_alter_deactivated.sql +38 -0
  691. synapse/storage/schema/main/delta/56/add_spans_to_device_lists.sql +39 -0
  692. synapse/storage/schema/main/delta/56/current_state_events_membership.sql +41 -0
  693. synapse/storage/schema/main/delta/56/current_state_events_membership_mk2.sql +43 -0
  694. synapse/storage/schema/main/delta/56/delete_keys_from_deleted_backups.sql +44 -0
  695. synapse/storage/schema/main/delta/56/destinations_failure_ts.sql +44 -0
  696. synapse/storage/schema/main/delta/56/destinations_retry_interval_type.sql.postgres +18 -0
  697. synapse/storage/schema/main/delta/56/device_stream_id_insert.sql +39 -0
  698. synapse/storage/schema/main/delta/56/devices_last_seen.sql +43 -0
  699. synapse/storage/schema/main/delta/56/drop_unused_event_tables.sql +39 -0
  700. synapse/storage/schema/main/delta/56/event_expiry.sql +40 -0
  701. synapse/storage/schema/main/delta/56/event_labels.sql +49 -0
  702. synapse/storage/schema/main/delta/56/event_labels_background_update.sql +36 -0
  703. synapse/storage/schema/main/delta/56/fix_room_keys_index.sql +37 -0
  704. synapse/storage/schema/main/delta/56/hidden_devices.sql +37 -0
  705. synapse/storage/schema/main/delta/56/hidden_devices_fix.sql.sqlite +42 -0
  706. synapse/storage/schema/main/delta/56/nuke_empty_communities_from_db.sql +48 -0
  707. synapse/storage/schema/main/delta/56/public_room_list_idx.sql +35 -0
  708. synapse/storage/schema/main/delta/56/redaction_censor.sql +35 -0
  709. synapse/storage/schema/main/delta/56/redaction_censor2.sql +41 -0
  710. synapse/storage/schema/main/delta/56/redaction_censor3_fix_update.sql.postgres +25 -0
  711. synapse/storage/schema/main/delta/56/redaction_censor4.sql +35 -0
  712. synapse/storage/schema/main/delta/56/remove_tombstoned_rooms_from_directory.sql +38 -0
  713. synapse/storage/schema/main/delta/56/room_key_etag.sql +36 -0
  714. synapse/storage/schema/main/delta/56/room_membership_idx.sql +37 -0
  715. synapse/storage/schema/main/delta/56/room_retention.sql +52 -0
  716. synapse/storage/schema/main/delta/56/signing_keys.sql +75 -0
  717. synapse/storage/schema/main/delta/56/signing_keys_nonunique_signatures.sql +41 -0
  718. synapse/storage/schema/main/delta/56/stats_separated.sql +175 -0
  719. synapse/storage/schema/main/delta/56/unique_user_filter_index.py +46 -0
  720. synapse/storage/schema/main/delta/56/user_external_ids.sql +43 -0
  721. synapse/storage/schema/main/delta/56/users_in_public_rooms_idx.sql +36 -0
  722. synapse/storage/schema/main/delta/57/delete_old_current_state_events.sql +41 -0
  723. synapse/storage/schema/main/delta/57/device_list_remote_cache_stale.sql +44 -0
  724. synapse/storage/schema/main/delta/57/local_current_membership.py +111 -0
  725. synapse/storage/schema/main/delta/57/remove_sent_outbound_pokes.sql +40 -0
  726. synapse/storage/schema/main/delta/57/rooms_version_column.sql +43 -0
  727. synapse/storage/schema/main/delta/57/rooms_version_column_2.sql.postgres +35 -0
  728. synapse/storage/schema/main/delta/57/rooms_version_column_2.sql.sqlite +22 -0
  729. synapse/storage/schema/main/delta/57/rooms_version_column_3.sql.postgres +39 -0
  730. synapse/storage/schema/main/delta/57/rooms_version_column_3.sql.sqlite +23 -0
  731. synapse/storage/schema/main/delta/58/02remove_dup_outbound_pokes.sql +41 -0
  732. synapse/storage/schema/main/delta/58/03persist_ui_auth.sql +55 -0
  733. synapse/storage/schema/main/delta/58/05cache_instance.sql.postgres +30 -0
  734. synapse/storage/schema/main/delta/58/06dlols_unique_idx.py +83 -0
  735. synapse/storage/schema/main/delta/58/07add_method_to_thumbnail_constraint.sql.postgres +33 -0
  736. synapse/storage/schema/main/delta/58/07add_method_to_thumbnail_constraint.sql.sqlite +44 -0
  737. synapse/storage/schema/main/delta/58/07persist_ui_auth_ips.sql +44 -0
  738. synapse/storage/schema/main/delta/58/08_media_safe_from_quarantine.sql.postgres +18 -0
  739. synapse/storage/schema/main/delta/58/08_media_safe_from_quarantine.sql.sqlite +18 -0
  740. synapse/storage/schema/main/delta/58/09shadow_ban.sql +37 -0
  741. synapse/storage/schema/main/delta/58/10_pushrules_enabled_delete_obsolete.sql +47 -0
  742. synapse/storage/schema/main/delta/58/10drop_local_rejections_stream.sql +41 -0
  743. synapse/storage/schema/main/delta/58/10federation_pos_instance_name.sql +41 -0
  744. synapse/storage/schema/main/delta/58/11dehydration.sql +39 -0
  745. synapse/storage/schema/main/delta/58/11fallback.sql +43 -0
  746. synapse/storage/schema/main/delta/58/11user_id_seq.py +38 -0
  747. synapse/storage/schema/main/delta/58/12room_stats.sql +51 -0
  748. synapse/storage/schema/main/delta/58/13remove_presence_allow_inbound.sql +36 -0
  749. synapse/storage/schema/main/delta/58/14events_instance_name.sql +35 -0
  750. synapse/storage/schema/main/delta/58/14events_instance_name.sql.postgres +28 -0
  751. synapse/storage/schema/main/delta/58/15_catchup_destination_rooms.sql +61 -0
  752. synapse/storage/schema/main/delta/58/15unread_count.sql +45 -0
  753. synapse/storage/schema/main/delta/58/16populate_stats_process_rooms_fix.sql +41 -0
  754. synapse/storage/schema/main/delta/58/17_catchup_last_successful.sql +40 -0
  755. synapse/storage/schema/main/delta/58/18stream_positions.sql +41 -0
  756. synapse/storage/schema/main/delta/58/19instance_map.sql.postgres +25 -0
  757. synapse/storage/schema/main/delta/58/19txn_id.sql +59 -0
  758. synapse/storage/schema/main/delta/58/20instance_name_event_tables.sql +36 -0
  759. synapse/storage/schema/main/delta/58/20user_daily_visits.sql +37 -0
  760. synapse/storage/schema/main/delta/58/21as_device_stream.sql +36 -0
  761. synapse/storage/schema/main/delta/58/21drop_device_max_stream_id.sql +1 -0
  762. synapse/storage/schema/main/delta/58/22puppet_token.sql +36 -0
  763. synapse/storage/schema/main/delta/58/22users_have_local_media.sql +2 -0
  764. synapse/storage/schema/main/delta/58/23e2e_cross_signing_keys_idx.sql +36 -0
  765. synapse/storage/schema/main/delta/58/24drop_event_json_index.sql +38 -0
  766. synapse/storage/schema/main/delta/58/25user_external_ids_user_id_idx.sql +36 -0
  767. synapse/storage/schema/main/delta/58/26access_token_last_validated.sql +37 -0
  768. synapse/storage/schema/main/delta/58/27local_invites.sql +37 -0
  769. synapse/storage/schema/main/delta/58/28drop_last_used_column.sql.postgres +16 -0
  770. synapse/storage/schema/main/delta/58/28drop_last_used_column.sql.sqlite +62 -0
  771. synapse/storage/schema/main/delta/59/01ignored_user.py +85 -0
  772. synapse/storage/schema/main/delta/59/02shard_send_to_device.sql +37 -0
  773. synapse/storage/schema/main/delta/59/03shard_send_to_device_sequence.sql.postgres +25 -0
  774. synapse/storage/schema/main/delta/59/04_event_auth_chains.sql +71 -0
  775. synapse/storage/schema/main/delta/59/04_event_auth_chains.sql.postgres +16 -0
  776. synapse/storage/schema/main/delta/59/04drop_account_data.sql +36 -0
  777. synapse/storage/schema/main/delta/59/05cache_invalidation.sql +36 -0
  778. synapse/storage/schema/main/delta/59/06chain_cover_index.sql +36 -0
  779. synapse/storage/schema/main/delta/59/06shard_account_data.sql +39 -0
  780. synapse/storage/schema/main/delta/59/06shard_account_data.sql.postgres +32 -0
  781. synapse/storage/schema/main/delta/59/07shard_account_data_fix.sql +37 -0
  782. synapse/storage/schema/main/delta/59/08delete_pushers_for_deactivated_accounts.sql +39 -0
  783. synapse/storage/schema/main/delta/59/08delete_stale_pushers.sql +39 -0
  784. synapse/storage/schema/main/delta/59/09rejected_events_metadata.sql +45 -0
  785. synapse/storage/schema/main/delta/59/10delete_purged_chain_cover.sql +36 -0
  786. synapse/storage/schema/main/delta/59/11add_knock_members_to_stats.sql +39 -0
  787. synapse/storage/schema/main/delta/59/11drop_thumbnail_constraint.sql.postgres +22 -0
  788. synapse/storage/schema/main/delta/59/12account_validity_token_used_ts_ms.sql +37 -0
  789. synapse/storage/schema/main/delta/59/12presence_stream_instance.sql +37 -0
  790. synapse/storage/schema/main/delta/59/12presence_stream_instance_seq.sql.postgres +20 -0
  791. synapse/storage/schema/main/delta/59/13users_to_send_full_presence_to.sql +53 -0
  792. synapse/storage/schema/main/delta/59/14refresh_tokens.sql +53 -0
  793. synapse/storage/schema/main/delta/59/15locks.sql +56 -0
  794. synapse/storage/schema/main/delta/59/16federation_inbound_staging.sql +51 -0
  795. synapse/storage/schema/main/delta/60/01recreate_stream_ordering.sql.postgres +45 -0
  796. synapse/storage/schema/main/delta/60/02change_stream_ordering_columns.sql.postgres +30 -0
  797. synapse/storage/schema/main/delta/61/01change_appservices_txns.sql.postgres +23 -0
  798. synapse/storage/schema/main/delta/61/01insertion_event_lookups.sql +68 -0
  799. synapse/storage/schema/main/delta/61/02drop_redundant_room_depth_index.sql +37 -0
  800. synapse/storage/schema/main/delta/61/03recreate_min_depth.py +74 -0
  801. synapse/storage/schema/main/delta/62/01insertion_event_extremities.sql +43 -0
  802. synapse/storage/schema/main/delta/63/01create_registration_tokens.sql +42 -0
  803. synapse/storage/schema/main/delta/63/02delete_unlinked_email_pushers.sql +39 -0
  804. synapse/storage/schema/main/delta/63/02populate-rooms-creator.sql +36 -0
  805. synapse/storage/schema/main/delta/63/03session_store.sql +42 -0
  806. synapse/storage/schema/main/delta/63/04add_presence_stream_not_offline_index.sql +37 -0
  807. synapse/storage/schema/main/delta/64/01msc2716_chunk_to_batch_rename.sql.postgres +23 -0
  808. synapse/storage/schema/main/delta/64/01msc2716_chunk_to_batch_rename.sql.sqlite +37 -0
  809. synapse/storage/schema/main/delta/65/01msc2716_insertion_event_edges.sql +38 -0
  810. synapse/storage/schema/main/delta/65/03remove_hidden_devices_from_device_inbox.sql +41 -0
  811. synapse/storage/schema/main/delta/65/04_local_group_updates.sql +37 -0
  812. synapse/storage/schema/main/delta/65/05_remove_room_stats_historical_and_user_stats_historical.sql +38 -0
  813. synapse/storage/schema/main/delta/65/06remove_deleted_devices_from_device_inbox.sql +53 -0
  814. synapse/storage/schema/main/delta/65/07_arbitrary_relations.sql +37 -0
  815. synapse/storage/schema/main/delta/65/08_device_inbox_background_updates.sql +37 -0
  816. synapse/storage/schema/main/delta/65/10_expirable_refresh_tokens.sql +47 -0
  817. synapse/storage/schema/main/delta/65/11_devices_auth_provider_session.sql +46 -0
  818. synapse/storage/schema/main/delta/67/01drop_public_room_list_stream.sql +37 -0
  819. synapse/storage/schema/main/delta/68/01event_columns.sql +45 -0
  820. synapse/storage/schema/main/delta/68/02_msc2409_add_device_id_appservice_stream_type.sql +40 -0
  821. synapse/storage/schema/main/delta/68/03_delete_account_data_for_deactivated_accounts.sql +39 -0
  822. synapse/storage/schema/main/delta/68/04_refresh_tokens_index_next_token_id.sql +47 -0
  823. synapse/storage/schema/main/delta/68/04partial_state_rooms.sql +60 -0
  824. synapse/storage/schema/main/delta/68/05_delete_non_strings_from_event_search.sql.sqlite +22 -0
  825. synapse/storage/schema/main/delta/68/05partial_state_rooms_triggers.py +80 -0
  826. synapse/storage/schema/main/delta/68/06_msc3202_add_device_list_appservice_stream_type.sql +42 -0
  827. synapse/storage/schema/main/delta/69/01as_txn_seq.py +54 -0
  828. synapse/storage/schema/main/delta/69/01device_list_oubound_by_room.sql +57 -0
  829. synapse/storage/schema/main/delta/69/02cache_invalidation_index.sql +37 -0
  830. synapse/storage/schema/main/delta/70/01clean_table_purged_rooms.sql +39 -0
  831. synapse/storage/schema/main/delta/71/01rebuild_event_edges.sql.postgres +43 -0
  832. synapse/storage/schema/main/delta/71/01rebuild_event_edges.sql.sqlite +47 -0
  833. synapse/storage/schema/main/delta/71/01remove_noop_background_updates.sql +80 -0
  834. synapse/storage/schema/main/delta/71/02event_push_summary_unique.sql +37 -0
  835. synapse/storage/schema/main/delta/72/01add_room_type_to_state_stats.sql +38 -0
  836. synapse/storage/schema/main/delta/72/01event_push_summary_receipt.sql +54 -0
  837. synapse/storage/schema/main/delta/72/02event_push_actions_index.sql +38 -0
  838. synapse/storage/schema/main/delta/72/03bg_populate_events_columns.py +57 -0
  839. synapse/storage/schema/main/delta/72/03drop_event_reference_hashes.sql +36 -0
  840. synapse/storage/schema/main/delta/72/03remove_groups.sql +50 -0
  841. synapse/storage/schema/main/delta/72/04drop_column_application_services_state_last_txn.sql.postgres +17 -0
  842. synapse/storage/schema/main/delta/72/04drop_column_application_services_state_last_txn.sql.sqlite +40 -0
  843. synapse/storage/schema/main/delta/72/05receipts_event_stream_ordering.sql +38 -0
  844. synapse/storage/schema/main/delta/72/05remove_unstable_private_read_receipts.sql +38 -0
  845. synapse/storage/schema/main/delta/72/06add_consent_ts_to_users.sql +35 -0
  846. synapse/storage/schema/main/delta/72/06thread_notifications.sql +49 -0
  847. synapse/storage/schema/main/delta/72/07force_update_current_state_events_membership.py +67 -0
  848. synapse/storage/schema/main/delta/72/07thread_receipts.sql.postgres +30 -0
  849. synapse/storage/schema/main/delta/72/07thread_receipts.sql.sqlite +70 -0
  850. synapse/storage/schema/main/delta/72/08begin_cache_invalidation_seq_at_2.sql.postgres +23 -0
  851. synapse/storage/schema/main/delta/72/08thread_receipts.sql +39 -0
  852. synapse/storage/schema/main/delta/72/09partial_indices.sql.sqlite +56 -0
  853. synapse/storage/schema/main/delta/73/01event_failed_pull_attempts.sql +48 -0
  854. synapse/storage/schema/main/delta/73/02add_pusher_enabled.sql +35 -0
  855. synapse/storage/schema/main/delta/73/02room_id_indexes_for_purging.sql +41 -0
  856. synapse/storage/schema/main/delta/73/03pusher_device_id.sql +39 -0
  857. synapse/storage/schema/main/delta/73/03users_approved_column.sql +39 -0
  858. synapse/storage/schema/main/delta/73/04partial_join_details.sql +42 -0
  859. synapse/storage/schema/main/delta/73/04pending_device_list_updates.sql +47 -0
  860. synapse/storage/schema/main/delta/73/05old_push_actions.sql.postgres +22 -0
  861. synapse/storage/schema/main/delta/73/05old_push_actions.sql.sqlite +24 -0
  862. synapse/storage/schema/main/delta/73/06thread_notifications_thread_id_idx.sql +42 -0
  863. synapse/storage/schema/main/delta/73/08thread_receipts_non_null.sql.postgres +23 -0
  864. synapse/storage/schema/main/delta/73/08thread_receipts_non_null.sql.sqlite +76 -0
  865. synapse/storage/schema/main/delta/73/09partial_joined_via_destination.sql +37 -0
  866. synapse/storage/schema/main/delta/73/09threads_table.sql +49 -0
  867. synapse/storage/schema/main/delta/73/10_update_sqlite_fts4_tokenizer.py +71 -0
  868. synapse/storage/schema/main/delta/73/10login_tokens.sql +54 -0
  869. synapse/storage/schema/main/delta/73/11event_search_room_id_n_distinct.sql.postgres +33 -0
  870. synapse/storage/schema/main/delta/73/12refactor_device_list_outbound_pokes.sql +72 -0
  871. synapse/storage/schema/main/delta/73/13add_device_lists_index.sql +39 -0
  872. synapse/storage/schema/main/delta/73/20_un_partial_stated_room_stream.sql +51 -0
  873. synapse/storage/schema/main/delta/73/21_un_partial_stated_room_stream_seq.sql.postgres +20 -0
  874. synapse/storage/schema/main/delta/73/22_rebuild_user_dir_stats.sql +48 -0
  875. synapse/storage/schema/main/delta/73/22_un_partial_stated_event_stream.sql +53 -0
  876. synapse/storage/schema/main/delta/73/23_fix_thread_index.sql +52 -0
  877. synapse/storage/schema/main/delta/73/23_un_partial_stated_room_stream_seq.sql.postgres +20 -0
  878. synapse/storage/schema/main/delta/73/24_events_jump_to_date_index.sql +36 -0
  879. synapse/storage/schema/main/delta/73/25drop_presence.sql +36 -0
  880. synapse/storage/schema/main/delta/74/01_user_directory_stale_remote_users.sql +58 -0
  881. synapse/storage/schema/main/delta/74/02_set_device_id_for_pushers_bg_update.sql +38 -0
  882. synapse/storage/schema/main/delta/74/03_membership_tables_event_stream_ordering.sql.postgres +29 -0
  883. synapse/storage/schema/main/delta/74/03_membership_tables_event_stream_ordering.sql.sqlite +23 -0
  884. synapse/storage/schema/main/delta/74/03_room_membership_index.sql +38 -0
  885. synapse/storage/schema/main/delta/74/04_delete_e2e_backup_keys_for_deactivated_users.sql +36 -0
  886. synapse/storage/schema/main/delta/74/04_membership_tables_event_stream_ordering_triggers.py +87 -0
  887. synapse/storage/schema/main/delta/74/05_events_txn_id_device_id.sql +72 -0
  888. synapse/storage/schema/main/delta/74/90COMMENTS_destinations.sql.postgres +52 -0
  889. synapse/storage/schema/main/delta/76/01_add_profiles_full_user_id_column.sql +39 -0
  890. synapse/storage/schema/main/delta/76/02_add_user_filters_full_user_id_column.sql +39 -0
  891. synapse/storage/schema/main/delta/76/03_per_user_experimental_features.sql +46 -0
  892. synapse/storage/schema/main/delta/76/04_add_room_forgetter.sql +43 -0
  893. synapse/storage/schema/main/delta/77/01_add_profiles_not_valid_check.sql.postgres +16 -0
  894. synapse/storage/schema/main/delta/77/02_add_user_filters_not_valid_check.sql.postgres +16 -0
  895. synapse/storage/schema/main/delta/77/03bg_populate_full_user_id_profiles.sql +35 -0
  896. synapse/storage/schema/main/delta/77/04bg_populate_full_user_id_user_filters.sql +35 -0
  897. synapse/storage/schema/main/delta/77/05thread_notifications_backfill.sql +67 -0
  898. synapse/storage/schema/main/delta/77/06thread_notifications_not_null.sql.sqlite +102 -0
  899. synapse/storage/schema/main/delta/77/06thread_notifications_not_null_event_push_actions.sql.postgres +27 -0
  900. synapse/storage/schema/main/delta/77/06thread_notifications_not_null_event_push_actions_staging.sql.postgres +27 -0
  901. synapse/storage/schema/main/delta/77/06thread_notifications_not_null_event_push_summary.sql.postgres +29 -0
  902. synapse/storage/schema/main/delta/77/14bg_indices_event_stream_ordering.sql +39 -0
  903. synapse/storage/schema/main/delta/78/01_validate_and_update_profiles.py +99 -0
  904. synapse/storage/schema/main/delta/78/02_validate_and_update_user_filters.py +100 -0
  905. synapse/storage/schema/main/delta/78/03_remove_unused_indexes_user_filters.py +72 -0
  906. synapse/storage/schema/main/delta/78/03event_extremities_constraints.py +65 -0
  907. synapse/storage/schema/main/delta/78/04_add_full_user_id_index_user_filters.py +32 -0
  908. synapse/storage/schema/main/delta/79/03_read_write_locks_triggers.sql.postgres +102 -0
  909. synapse/storage/schema/main/delta/79/03_read_write_locks_triggers.sql.sqlite +72 -0
  910. synapse/storage/schema/main/delta/79/04_mitigate_stream_ordering_update_race.py +70 -0
  911. synapse/storage/schema/main/delta/79/05_read_write_locks_triggers.sql.postgres +69 -0
  912. synapse/storage/schema/main/delta/79/05_read_write_locks_triggers.sql.sqlite +65 -0
  913. synapse/storage/schema/main/delta/80/01_users_alter_locked.sql +35 -0
  914. synapse/storage/schema/main/delta/80/02_read_write_locks_unlogged.sql.postgres +30 -0
  915. synapse/storage/schema/main/delta/80/02_scheduled_tasks.sql +47 -0
  916. synapse/storage/schema/main/delta/80/03_read_write_locks_triggers.sql.postgres +37 -0
  917. synapse/storage/schema/main/delta/80/04_read_write_locks_deadlock.sql.postgres +71 -0
  918. synapse/storage/schema/main/delta/82/02_scheduled_tasks_index.sql +35 -0
  919. synapse/storage/schema/main/delta/82/04_add_indices_for_purging_rooms.sql +39 -0
  920. synapse/storage/schema/main/delta/82/05gaps.sql +44 -0
  921. synapse/storage/schema/main/delta/83/01_drop_old_tables.sql +43 -0
  922. synapse/storage/schema/main/delta/83/03_instance_name_receipts.sql.sqlite +17 -0
  923. synapse/storage/schema/main/delta/83/05_cross_signing_key_update_grant.sql +34 -0
  924. synapse/storage/schema/main/delta/83/06_event_push_summary_room.sql +36 -0
  925. synapse/storage/schema/main/delta/84/01_auth_links_stats.sql.postgres +20 -0
  926. synapse/storage/schema/main/delta/84/02_auth_links_index.sql +16 -0
  927. synapse/storage/schema/main/delta/84/03_auth_links_analyze.sql.postgres +16 -0
  928. synapse/storage/schema/main/delta/84/04_access_token_index.sql +15 -0
  929. synapse/storage/schema/main/delta/85/01_add_suspended.sql +14 -0
  930. synapse/storage/schema/main/delta/85/02_add_instance_names.sql +27 -0
  931. synapse/storage/schema/main/delta/85/03_new_sequences.sql.postgres +54 -0
  932. synapse/storage/schema/main/delta/85/04_cleanup_device_federation_outbox.sql +15 -0
  933. synapse/storage/schema/main/delta/85/05_add_instance_names_converted_pos.sql +16 -0
  934. synapse/storage/schema/main/delta/85/06_add_room_reports.sql +20 -0
  935. synapse/storage/schema/main/delta/86/01_authenticate_media.sql +15 -0
  936. synapse/storage/schema/main/delta/86/02_receipts_event_id_index.sql +15 -0
  937. synapse/storage/schema/main/delta/87/01_sliding_sync_memberships.sql +169 -0
  938. synapse/storage/schema/main/delta/87/02_per_connection_state.sql +81 -0
  939. synapse/storage/schema/main/delta/87/03_current_state_index.sql +19 -0
  940. synapse/storage/schema/main/delta/88/01_add_delayed_events.sql +43 -0
  941. synapse/storage/schema/main/delta/88/01_custom_profile_fields.sql +15 -0
  942. synapse/storage/schema/main/delta/88/02_fix_sliding_sync_membership_snapshots_forgotten_column.sql +21 -0
  943. synapse/storage/schema/main/delta/88/03_add_otk_ts_added_index.sql +18 -0
  944. synapse/storage/schema/main/delta/88/04_current_state_delta_index.sql +18 -0
  945. synapse/storage/schema/main/delta/88/05_drop_old_otks.sql.postgres +19 -0
  946. synapse/storage/schema/main/delta/88/05_drop_old_otks.sql.sqlite +19 -0
  947. synapse/storage/schema/main/delta/88/05_sliding_sync_room_config_index.sql +20 -0
  948. synapse/storage/schema/main/delta/88/06_events_received_ts_index.sql +17 -0
  949. synapse/storage/schema/main/delta/89/01_sliding_sync_membership_snapshot_index.sql +15 -0
  950. synapse/storage/schema/main/delta/90/01_add_column_participant_room_memberships_table.sql +16 -0
  951. synapse/storage/schema/main/delta/91/01_media_hash.sql +28 -0
  952. synapse/storage/schema/main/delta/92/01_remove_trigger.sql.postgres +16 -0
  953. synapse/storage/schema/main/delta/92/01_remove_trigger.sql.sqlite +16 -0
  954. synapse/storage/schema/main/delta/92/02_remove_populate_participant_bg_update.sql +17 -0
  955. synapse/storage/schema/main/delta/92/04_ss_membership_snapshot_idx.sql +16 -0
  956. synapse/storage/schema/main/delta/92/04_thread_subscriptions.sql +59 -0
  957. synapse/storage/schema/main/delta/92/04_thread_subscriptions_seq.sql.postgres +19 -0
  958. synapse/storage/schema/main/delta/92/05_fixup_max_depth_cap.sql +17 -0
  959. synapse/storage/schema/main/delta/92/05_thread_subscriptions_comments.sql.postgres +18 -0
  960. synapse/storage/schema/main/delta/92/06_device_federation_inbox_index.sql +16 -0
  961. synapse/storage/schema/main/delta/92/06_threads_last_sent_stream_ordering_comments.sql.postgres +24 -0
  962. synapse/storage/schema/main/delta/92/07_add_user_reports.sql +22 -0
  963. synapse/storage/schema/main/delta/92/07_event_txn_id_device_id_txn_id2.sql +15 -0
  964. synapse/storage/schema/main/delta/92/08_room_ban_redactions.sql +21 -0
  965. synapse/storage/schema/main/delta/92/08_thread_subscriptions_seq_fixup.sql.postgres +19 -0
  966. synapse/storage/schema/main/delta/92/09_thread_subscriptions_update.sql +20 -0
  967. synapse/storage/schema/main/delta/92/09_thread_subscriptions_update.sql.postgres +18 -0
  968. synapse/storage/schema/main/delta/93/01_add_delayed_events.sql +15 -0
  969. synapse/storage/schema/main/full_schemas/72/full.sql.postgres +1344 -0
  970. synapse/storage/schema/main/full_schemas/72/full.sql.sqlite +646 -0
  971. synapse/storage/schema/state/delta/23/drop_state_index.sql +35 -0
  972. synapse/storage/schema/state/delta/32/remove_state_indices.sql +38 -0
  973. synapse/storage/schema/state/delta/35/add_state_index.sql +36 -0
  974. synapse/storage/schema/state/delta/35/state.sql +41 -0
  975. synapse/storage/schema/state/delta/35/state_dedupe.sql +36 -0
  976. synapse/storage/schema/state/delta/47/state_group_seq.py +38 -0
  977. synapse/storage/schema/state/delta/56/state_group_room_idx.sql +36 -0
  978. synapse/storage/schema/state/delta/61/02state_groups_state_n_distinct.sql.postgres +34 -0
  979. synapse/storage/schema/state/delta/70/08_state_group_edges_unique.sql +36 -0
  980. synapse/storage/schema/state/delta/89/01_state_groups_deletion.sql +39 -0
  981. synapse/storage/schema/state/delta/90/02_delete_unreferenced_state_groups.sql +16 -0
  982. synapse/storage/schema/state/delta/90/03_remove_old_deletion_bg_update.sql +15 -0
  983. synapse/storage/schema/state/full_schemas/72/full.sql.postgres +30 -0
  984. synapse/storage/schema/state/full_schemas/72/full.sql.sqlite +20 -0
  985. synapse/storage/types.py +183 -0
  986. synapse/storage/util/__init__.py +20 -0
  987. synapse/storage/util/id_generators.py +928 -0
  988. synapse/storage/util/partial_state_events_tracker.py +194 -0
  989. synapse/storage/util/sequence.py +315 -0
  990. synapse/streams/__init__.py +43 -0
  991. synapse/streams/config.py +91 -0
  992. synapse/streams/events.py +203 -0
  993. synapse/synapse_rust/__init__.pyi +3 -0
  994. synapse/synapse_rust/acl.pyi +20 -0
  995. synapse/synapse_rust/events.pyi +136 -0
  996. synapse/synapse_rust/http_client.pyi +32 -0
  997. synapse/synapse_rust/push.pyi +86 -0
  998. synapse/synapse_rust/rendezvous.pyi +30 -0
  999. synapse/synapse_rust/segmenter.pyi +1 -0
  1000. synapse/synapse_rust.abi3.so +0 -0
  1001. synapse/types/__init__.py +1600 -0
  1002. synapse/types/handlers/__init__.py +93 -0
  1003. synapse/types/handlers/policy_server.py +16 -0
  1004. synapse/types/handlers/sliding_sync.py +908 -0
  1005. synapse/types/rest/__init__.py +25 -0
  1006. synapse/types/rest/client/__init__.py +413 -0
  1007. synapse/types/state.py +634 -0
  1008. synapse/types/storage/__init__.py +66 -0
  1009. synapse/util/__init__.py +169 -0
  1010. synapse/util/async_helpers.py +1045 -0
  1011. synapse/util/background_queue.py +142 -0
  1012. synapse/util/batching_queue.py +202 -0
  1013. synapse/util/caches/__init__.py +300 -0
  1014. synapse/util/caches/cached_call.py +143 -0
  1015. synapse/util/caches/deferred_cache.py +530 -0
  1016. synapse/util/caches/descriptors.py +692 -0
  1017. synapse/util/caches/dictionary_cache.py +346 -0
  1018. synapse/util/caches/expiringcache.py +249 -0
  1019. synapse/util/caches/lrucache.py +975 -0
  1020. synapse/util/caches/response_cache.py +322 -0
  1021. synapse/util/caches/stream_change_cache.py +370 -0
  1022. synapse/util/caches/treecache.py +189 -0
  1023. synapse/util/caches/ttlcache.py +197 -0
  1024. synapse/util/cancellation.py +63 -0
  1025. synapse/util/check_dependencies.py +335 -0
  1026. synapse/util/clock.py +567 -0
  1027. synapse/util/constants.py +22 -0
  1028. synapse/util/daemonize.py +165 -0
  1029. synapse/util/distributor.py +157 -0
  1030. synapse/util/events.py +134 -0
  1031. synapse/util/file_consumer.py +164 -0
  1032. synapse/util/frozenutils.py +57 -0
  1033. synapse/util/gai_resolver.py +178 -0
  1034. synapse/util/hash.py +38 -0
  1035. synapse/util/httpresourcetree.py +108 -0
  1036. synapse/util/iterutils.py +189 -0
  1037. synapse/util/json.py +56 -0
  1038. synapse/util/linked_list.py +156 -0
  1039. synapse/util/logcontext.py +46 -0
  1040. synapse/util/logformatter.py +28 -0
  1041. synapse/util/macaroons.py +325 -0
  1042. synapse/util/manhole.py +191 -0
  1043. synapse/util/metrics.py +339 -0
  1044. synapse/util/module_loader.py +116 -0
  1045. synapse/util/msisdn.py +51 -0
  1046. synapse/util/patch_inline_callbacks.py +250 -0
  1047. synapse/util/pydantic_models.py +63 -0
  1048. synapse/util/ratelimitutils.py +419 -0
  1049. synapse/util/retryutils.py +339 -0
  1050. synapse/util/rlimit.py +42 -0
  1051. synapse/util/rust.py +133 -0
  1052. synapse/util/sentinel.py +21 -0
  1053. synapse/util/stringutils.py +293 -0
  1054. synapse/util/task_scheduler.py +493 -0
  1055. synapse/util/templates.py +126 -0
  1056. synapse/util/threepids.py +123 -0
  1057. synapse/util/wheel_timer.py +112 -0
  1058. synapse/visibility.py +835 -0
@@ -0,0 +1,2577 @@
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
+ # If the stream change cache is valid for the stream token, we can just
745
+ # use the result of that.
746
+ if from_key.stream >= self._events_stream_cache.get_earliest_known_position():
747
+ return self._events_stream_cache.get_entities_changed(
748
+ room_ids, from_key.stream
749
+ )
750
+
751
+ def get_rooms_that_have_updates_since_sliding_sync_table_txn(
752
+ txn: LoggingTransaction,
753
+ ) -> StrCollection:
754
+ sql = """
755
+ SELECT room_id
756
+ FROM sliding_sync_joined_rooms
757
+ WHERE {clause}
758
+ AND event_stream_ordering > ?
759
+ """
760
+
761
+ results: set[str] = set()
762
+ for batch in batch_iter(room_ids, 1000):
763
+ clause, args = make_in_list_sql_clause(
764
+ self.database_engine, "room_id", batch
765
+ )
766
+
767
+ args.append(from_key.stream)
768
+ txn.execute(sql.format(clause=clause), args)
769
+
770
+ results.update(row[0] for row in txn)
771
+
772
+ return results
773
+
774
+ return await self.db_pool.runInteraction(
775
+ "get_rooms_that_have_updates_since_sliding_sync_table",
776
+ get_rooms_that_have_updates_since_sliding_sync_table_txn,
777
+ )
778
+
779
+ async def paginate_room_events_by_stream_ordering(
780
+ self,
781
+ *,
782
+ room_id: str,
783
+ from_key: RoomStreamToken,
784
+ to_key: RoomStreamToken | None = None,
785
+ direction: Direction = Direction.BACKWARDS,
786
+ limit: int = 0,
787
+ ) -> tuple[list[EventBase], RoomStreamToken, bool]:
788
+ """
789
+ Paginate events by `stream_ordering` in the room from the `from_key` in the
790
+ given `direction` to the `to_key` or `limit`.
791
+
792
+ Args:
793
+ room_id
794
+ from_key: The token to stream from (starting point and heading in the given
795
+ direction)
796
+ to_key: The token representing the end stream position (end point)
797
+ direction: Indicates whether we are paginating forwards or backwards
798
+ from `from_key`.
799
+ limit: Maximum number of events to return
800
+
801
+ Returns:
802
+ The results as a list of events, a token that points to the end of
803
+ the result set, and a boolean to indicate if there were more events
804
+ but we hit the limit. If no events are returned then the end of the
805
+ stream has been reached (i.e. there are no events between `from_key`
806
+ and `to_key`).
807
+
808
+ When Direction.FORWARDS: from_key < x <= to_key, (ascending order)
809
+ When Direction.BACKWARDS: from_key >= x > to_key, (descending order)
810
+ """
811
+
812
+ # FIXME: When going forwards, we should enforce that the `to_key` is not `None`
813
+ # because we always need an upper bound when querying the events stream (as
814
+ # otherwise we'll potentially pick up events that are not fully persisted).
815
+
816
+ # We should only be working with `stream_ordering` tokens here
817
+ assert from_key is None or from_key.topological is None
818
+ assert to_key is None or to_key.topological is None
819
+
820
+ # We can bail early if we're looking forwards, and our `to_key` is already
821
+ # before our `from_key`.
822
+ if (
823
+ direction == Direction.FORWARDS
824
+ and to_key is not None
825
+ and to_key.is_before_or_eq(from_key)
826
+ ):
827
+ # Token selection matches what we do below if there are no rows
828
+ return [], to_key if to_key else from_key, False
829
+ # Or vice-versa, if we're looking backwards and our `from_key` is already before
830
+ # our `to_key`.
831
+ elif (
832
+ direction == Direction.BACKWARDS
833
+ and to_key is not None
834
+ and from_key.is_before_or_eq(to_key)
835
+ ):
836
+ # Token selection matches what we do below if there are no rows
837
+ return [], to_key if to_key else from_key, False
838
+
839
+ # We can do a quick sanity check to see if any events have been sent in the room
840
+ # since the earlier token.
841
+ has_changed = True
842
+ if direction == Direction.FORWARDS:
843
+ has_changed = self._events_stream_cache.has_entity_changed(
844
+ room_id, from_key.stream
845
+ )
846
+ elif direction == Direction.BACKWARDS:
847
+ if to_key is not None:
848
+ has_changed = self._events_stream_cache.has_entity_changed(
849
+ room_id, to_key.stream
850
+ )
851
+ else:
852
+ assert_never(direction)
853
+
854
+ if not has_changed:
855
+ # Token selection matches what we do below if there are no rows
856
+ return [], to_key if to_key else from_key, False
857
+
858
+ order, from_bound, to_bound = generate_pagination_bounds(
859
+ direction, from_key, to_key
860
+ )
861
+
862
+ bounds = generate_pagination_where_clause(
863
+ direction=direction,
864
+ # The empty string will shortcut downstream code to only use the
865
+ # `stream_ordering` column
866
+ column_names=("", "stream_ordering"),
867
+ from_token=from_bound,
868
+ to_token=to_bound,
869
+ engine=self.database_engine,
870
+ )
871
+
872
+ def f(txn: LoggingTransaction) -> tuple[list[_EventDictReturn], bool]:
873
+ sql = f"""
874
+ SELECT event_id, instance_name, stream_ordering
875
+ FROM events
876
+ WHERE
877
+ room_id = ?
878
+ AND not outlier
879
+ AND {bounds}
880
+ ORDER BY stream_ordering {order} LIMIT ?
881
+ """
882
+ txn.execute(sql, (room_id, 2 * limit))
883
+
884
+ # Get all the rows and check if we hit the limit.
885
+ fetched_rows = txn.fetchall()
886
+ limited = len(fetched_rows) >= 2 * limit
887
+
888
+ rows = [
889
+ _EventDictReturn(event_id, None, stream_ordering)
890
+ for event_id, instance_name, stream_ordering in fetched_rows
891
+ if _filter_results_by_stream(
892
+ lower_token=(
893
+ to_key if direction == Direction.BACKWARDS else from_key
894
+ ),
895
+ upper_token=(
896
+ from_key if direction == Direction.BACKWARDS else to_key
897
+ ),
898
+ instance_name=instance_name,
899
+ stream_ordering=stream_ordering,
900
+ )
901
+ ]
902
+
903
+ if len(rows) > limit:
904
+ limited = True
905
+
906
+ rows = rows[:limit]
907
+ return rows, limited
908
+
909
+ rows, limited = await self.db_pool.runInteraction(
910
+ "get_room_events_stream_for_room", f
911
+ )
912
+
913
+ ret = await self.get_events_as_list(
914
+ [r.event_id for r in rows], get_prev_content=True
915
+ )
916
+
917
+ if rows:
918
+ next_key = generate_next_token(
919
+ direction=direction,
920
+ last_topo_ordering=None,
921
+ last_stream_ordering=rows[-1].stream_ordering,
922
+ )
923
+ else:
924
+ # TODO (erikj): We should work out what to do here instead. (same as
925
+ # `_paginate_room_events_by_topological_ordering_txn(...)`)
926
+ next_key = to_key if to_key else from_key
927
+
928
+ return ret, next_key, limited
929
+
930
+ @trace
931
+ async def get_current_state_delta_membership_changes_for_user(
932
+ self,
933
+ user_id: str,
934
+ from_key: RoomStreamToken,
935
+ to_key: RoomStreamToken,
936
+ excluded_room_ids: list[str] | None = None,
937
+ ) -> list[CurrentStateDeltaMembership]:
938
+ """
939
+ Fetch membership events (and the previous event that was replaced by that one)
940
+ for a given user.
941
+
942
+ Note: This function only works with "live" tokens with `stream_ordering` only.
943
+
944
+ We're looking for membership changes in the token range (> `from_key` and <=
945
+ `to_key`).
946
+
947
+ Please be mindful to only use this with `from_key` and `to_key` tokens that are
948
+ recent enough to be after when the first local user joined the room. Otherwise,
949
+ the results may be incomplete or too greedy. For example, if you use a token
950
+ range before the first local user joined the room, you will see 0 events since
951
+ `current_state_delta_stream` tracks what the server thinks is the current state
952
+ of the room as time goes. It does not track how state progresses from the
953
+ beginning of the room. So for example, when you remotely join a room, the first
954
+ rows will just be the state when you joined and progress from there.
955
+
956
+ You can probably reasonably use this with `/sync` because the `to_key` passed in
957
+ will be the "current" now token and the range will cover when the user joined
958
+ the room.
959
+
960
+ Args:
961
+ user_id: The user ID to fetch membership events for.
962
+ from_key: The point in the stream to sync from (fetching events > this point).
963
+ to_key: The token to fetch rooms up to (fetching events <= this point).
964
+ excluded_room_ids: Optional list of room IDs to exclude from the results.
965
+
966
+ Returns:
967
+ All membership changes to the current state in the token range. Events are
968
+ sorted by `stream_ordering` ascending.
969
+
970
+ `event_id`/`sender` can be `None` when the server leaves a room (meaning
971
+ everyone locally left) or a state reset which removed the person from the
972
+ room. We can't tell the difference between the two cases with what's
973
+ available in the `current_state_delta_stream` table. To actually check for a
974
+ state reset, you need to check if a membership still exists in the room.
975
+ """
976
+
977
+ assert from_key.topological is None
978
+ assert to_key.topological is None
979
+
980
+ # Start by ruling out cases where a DB query is not necessary.
981
+ if from_key == to_key:
982
+ return []
983
+
984
+ if from_key:
985
+ has_changed = self._membership_stream_cache.has_entity_changed(
986
+ user_id, int(from_key.stream)
987
+ )
988
+ if not has_changed:
989
+ return []
990
+
991
+ def f(txn: LoggingTransaction) -> list[CurrentStateDeltaMembership]:
992
+ # To handle tokens with a non-empty instance_map we fetch more
993
+ # results than necessary and then filter down
994
+ min_from_id = from_key.stream
995
+ max_to_id = to_key.get_max_stream_pos()
996
+
997
+ args: list[Any] = [min_from_id, max_to_id, EventTypes.Member, user_id]
998
+
999
+ # TODO: It would be good to assert that the `from_token`/`to_token` is >=
1000
+ # the first row in `current_state_delta_stream` for the rooms we're
1001
+ # interested in. Otherwise, we will end up with empty results and not know
1002
+ # it.
1003
+
1004
+ # We could `COALESCE(e.stream_ordering, s.stream_id)` to get more accurate
1005
+ # stream positioning when available but given our usages, we can avoid the
1006
+ # complexity. Between two (valid) stream tokens, we will still get all of
1007
+ # the state changes. Since those events are persisted in a batch, valid
1008
+ # tokens will either be before or after the batch of events.
1009
+ #
1010
+ # `stream_ordering` from the `events` table is more accurate when available
1011
+ # since the `current_state_delta_stream` table only tracks that the current
1012
+ # state is at this stream position (not what stream position the state event
1013
+ # was added) and uses the *minimum* stream position for batches of events.
1014
+ sql = """
1015
+ SELECT
1016
+ s.room_id,
1017
+ e.event_id,
1018
+ s.instance_name,
1019
+ s.stream_id,
1020
+ m.membership,
1021
+ e.sender,
1022
+ s.prev_event_id,
1023
+ e_prev.instance_name AS prev_instance_name,
1024
+ e_prev.stream_ordering AS prev_stream_ordering,
1025
+ m_prev.membership AS prev_membership,
1026
+ e_prev.sender AS prev_sender
1027
+ FROM current_state_delta_stream AS s
1028
+ LEFT JOIN events AS e ON e.event_id = s.event_id
1029
+ LEFT JOIN room_memberships AS m ON m.event_id = s.event_id
1030
+ LEFT JOIN events AS e_prev ON e_prev.event_id = s.prev_event_id
1031
+ LEFT JOIN room_memberships AS m_prev ON m_prev.event_id = s.prev_event_id
1032
+ WHERE s.stream_id > ? AND s.stream_id <= ?
1033
+ AND s.type = ?
1034
+ AND s.state_key = ?
1035
+ ORDER BY s.stream_id ASC
1036
+ """
1037
+
1038
+ txn.execute(sql, args)
1039
+
1040
+ membership_changes: list[CurrentStateDeltaMembership] = []
1041
+ for (
1042
+ room_id,
1043
+ event_id,
1044
+ instance_name,
1045
+ stream_ordering,
1046
+ membership,
1047
+ sender,
1048
+ prev_event_id,
1049
+ prev_instance_name,
1050
+ prev_stream_ordering,
1051
+ prev_membership,
1052
+ prev_sender,
1053
+ ) in txn:
1054
+ assert room_id is not None
1055
+ assert stream_ordering is not None
1056
+
1057
+ if _filter_results_by_stream(
1058
+ from_key,
1059
+ to_key,
1060
+ instance_name,
1061
+ stream_ordering,
1062
+ ):
1063
+ # When the server leaves a room, it will insert new rows into the
1064
+ # `current_state_delta_stream` table with `event_id = null` for all
1065
+ # current state. This means we might already have a row for the
1066
+ # leave event and then another for the same leave where the
1067
+ # `event_id=null` but the `prev_event_id` is pointing back at the
1068
+ # earlier leave event. We don't want to report the leave, if we
1069
+ # already have a leave event.
1070
+ if event_id is None and prev_membership == Membership.LEAVE:
1071
+ continue
1072
+
1073
+ membership_change = CurrentStateDeltaMembership(
1074
+ room_id=room_id,
1075
+ # Event
1076
+ event_id=event_id,
1077
+ event_pos=PersistedEventPosition(
1078
+ # If instance_name is null we default to "master"
1079
+ instance_name=instance_name or "master",
1080
+ stream=stream_ordering,
1081
+ ),
1082
+ # When `s.event_id = null`, we won't be able to get respective
1083
+ # `room_membership` but can assume the user has left the room
1084
+ # because this only happens when the server leaves a room
1085
+ # (meaning everyone locally left) or a state reset which removed
1086
+ # the person from the room.
1087
+ membership=(
1088
+ membership if membership is not None else Membership.LEAVE
1089
+ ),
1090
+ # This will also be null for the same reasons if `s.event_id = null`
1091
+ sender=sender,
1092
+ # Prev event
1093
+ prev_event_id=prev_event_id,
1094
+ prev_event_pos=(
1095
+ PersistedEventPosition(
1096
+ # If instance_name is null we default to "master"
1097
+ instance_name=prev_instance_name or "master",
1098
+ stream=prev_stream_ordering,
1099
+ )
1100
+ if (prev_stream_ordering is not None)
1101
+ else None
1102
+ ),
1103
+ prev_membership=prev_membership,
1104
+ prev_sender=prev_sender,
1105
+ )
1106
+
1107
+ membership_changes.append(membership_change)
1108
+
1109
+ return membership_changes
1110
+
1111
+ membership_changes = await self.db_pool.runInteraction(
1112
+ "get_current_state_delta_membership_changes_for_user", f
1113
+ )
1114
+
1115
+ room_ids_to_exclude: AbstractSet[str] = set()
1116
+ if excluded_room_ids is not None:
1117
+ room_ids_to_exclude = set(excluded_room_ids)
1118
+
1119
+ return [
1120
+ membership_change
1121
+ for membership_change in membership_changes
1122
+ if membership_change.room_id not in room_ids_to_exclude
1123
+ ]
1124
+
1125
+ @trace
1126
+ async def get_sliding_sync_membership_changes(
1127
+ self,
1128
+ user_id: str,
1129
+ from_key: RoomStreamToken,
1130
+ to_key: RoomStreamToken,
1131
+ excluded_room_ids: AbstractSet[str] | None = None,
1132
+ ) -> dict[str, RoomsForUserStateReset]:
1133
+ """
1134
+ Fetch membership events that result in a meaningful membership change for a
1135
+ given user.
1136
+
1137
+ A meaningful membership changes is one where the `membership` value actually
1138
+ changes. This means memberships changes from `join` to `join` (like a display
1139
+ name change) will be filtered out since they result in no meaningful change.
1140
+
1141
+ Note: This function only works with "live" tokens with `stream_ordering` only.
1142
+
1143
+ We're looking for membership changes in the token range (> `from_key` and <=
1144
+ `to_key`).
1145
+
1146
+ Args:
1147
+ user_id: The user ID to fetch membership events for.
1148
+ from_key: The point in the stream to sync from (fetching events > this point).
1149
+ to_key: The token to fetch rooms up to (fetching events <= this point).
1150
+ excluded_room_ids: Optional list of room IDs to exclude from the results.
1151
+
1152
+ Returns:
1153
+ All meaningful membership changes to the current state in the token range.
1154
+ Events are sorted by `stream_ordering` ascending.
1155
+
1156
+ `event_id`/`sender` can be `None` when the server leaves a room (meaning
1157
+ everyone locally left) or a state reset which removed the person from the
1158
+ room. We can't tell the difference between the two cases with what's
1159
+ available in the `current_state_delta_stream` table. To actually check for a
1160
+ state reset, you need to check if a membership still exists in the room.
1161
+ """
1162
+
1163
+ assert from_key.topological is None
1164
+ assert to_key.topological is None
1165
+
1166
+ # Start by ruling out cases where a DB query is not necessary.
1167
+ if from_key == to_key:
1168
+ return {}
1169
+
1170
+ if from_key:
1171
+ has_changed = self._membership_stream_cache.has_entity_changed(
1172
+ user_id, int(from_key.stream)
1173
+ )
1174
+ if not has_changed:
1175
+ return {}
1176
+
1177
+ room_ids_to_exclude: AbstractSet[str] = set()
1178
+ if excluded_room_ids is not None:
1179
+ room_ids_to_exclude = excluded_room_ids
1180
+
1181
+ def f(txn: LoggingTransaction) -> dict[str, RoomsForUserStateReset]:
1182
+ # To handle tokens with a non-empty instance_map we fetch more
1183
+ # results than necessary and then filter down
1184
+ min_from_id = from_key.stream
1185
+ max_to_id = to_key.get_max_stream_pos()
1186
+
1187
+ # This query looks at membership changes in
1188
+ # `sliding_sync_membership_snapshots` which will not include users
1189
+ # that were state reset out of rooms; so we need to look for that
1190
+ # case in `current_state_delta_stream`.
1191
+ sql = """
1192
+ SELECT
1193
+ room_id,
1194
+ membership_event_id,
1195
+ event_instance_name,
1196
+ event_stream_ordering,
1197
+ membership,
1198
+ sender,
1199
+ prev_membership,
1200
+ room_version
1201
+ FROM
1202
+ (
1203
+ SELECT
1204
+ s.room_id,
1205
+ s.membership_event_id,
1206
+ s.event_instance_name,
1207
+ s.event_stream_ordering,
1208
+ s.membership,
1209
+ s.sender,
1210
+ m_prev.membership AS prev_membership
1211
+ FROM sliding_sync_membership_snapshots as s
1212
+ LEFT JOIN event_edges AS e ON e.event_id = s.membership_event_id
1213
+ LEFT JOIN room_memberships AS m_prev ON m_prev.event_id = e.prev_event_id
1214
+ WHERE s.user_id = ?
1215
+
1216
+ UNION ALL
1217
+
1218
+ SELECT
1219
+ s.room_id,
1220
+ e.event_id,
1221
+ s.instance_name,
1222
+ s.stream_id,
1223
+ m.membership,
1224
+ e.sender,
1225
+ m_prev.membership AS prev_membership
1226
+ FROM current_state_delta_stream AS s
1227
+ LEFT JOIN events AS e ON e.event_id = s.event_id
1228
+ LEFT JOIN room_memberships AS m ON m.event_id = s.event_id
1229
+ LEFT JOIN room_memberships AS m_prev ON m_prev.event_id = s.prev_event_id
1230
+ WHERE
1231
+ s.type = ?
1232
+ AND s.state_key = ?
1233
+ ) AS c
1234
+ INNER JOIN rooms USING (room_id)
1235
+ WHERE event_stream_ordering > ? AND event_stream_ordering <= ?
1236
+ ORDER BY event_stream_ordering ASC
1237
+ """
1238
+
1239
+ txn.execute(
1240
+ sql,
1241
+ (user_id, EventTypes.Member, user_id, min_from_id, max_to_id),
1242
+ )
1243
+
1244
+ membership_changes: dict[str, RoomsForUserStateReset] = {}
1245
+ for (
1246
+ room_id,
1247
+ membership_event_id,
1248
+ event_instance_name,
1249
+ event_stream_ordering,
1250
+ membership,
1251
+ sender,
1252
+ prev_membership,
1253
+ room_version_id,
1254
+ ) in txn:
1255
+ assert room_id is not None
1256
+ assert event_stream_ordering is not None
1257
+
1258
+ if room_id in room_ids_to_exclude:
1259
+ continue
1260
+
1261
+ if _filter_results_by_stream(
1262
+ from_key,
1263
+ to_key,
1264
+ event_instance_name,
1265
+ event_stream_ordering,
1266
+ ):
1267
+ # When the server leaves a room, it will insert new rows into the
1268
+ # `current_state_delta_stream` table with `event_id = null` for all
1269
+ # current state. This means we might already have a row for the
1270
+ # leave event and then another for the same leave where the
1271
+ # `event_id=null` but the `prev_event_id` is pointing back at the
1272
+ # earlier leave event. We don't want to report the leave, if we
1273
+ # already have a leave event.
1274
+ if (
1275
+ membership_event_id is None
1276
+ and prev_membership == Membership.LEAVE
1277
+ ):
1278
+ continue
1279
+
1280
+ if membership_event_id is None and room_id in membership_changes:
1281
+ # SUSPICIOUS: if we join a room and get state reset out of it
1282
+ # in the same queried window,
1283
+ # won't this ignore the 'state reset out of it' part?
1284
+ continue
1285
+
1286
+ # When `s.event_id = null`, we won't be able to get respective
1287
+ # `room_membership` but can assume the user has left the room
1288
+ # because this only happens when the server leaves a room
1289
+ # (meaning everyone locally left) or a state reset which removed
1290
+ # the person from the room.
1291
+ membership = (
1292
+ membership if membership is not None else Membership.LEAVE
1293
+ )
1294
+
1295
+ if membership == prev_membership:
1296
+ # If `membership` and `prev_membership` are the same then this
1297
+ # is not a meaningful change so we can skip it.
1298
+ # An example of this happening is when the user changes their display name.
1299
+ continue
1300
+
1301
+ membership_change = RoomsForUserStateReset(
1302
+ room_id=room_id,
1303
+ sender=sender,
1304
+ membership=membership,
1305
+ event_id=membership_event_id,
1306
+ event_pos=PersistedEventPosition(
1307
+ event_instance_name, event_stream_ordering
1308
+ ),
1309
+ room_version_id=room_version_id,
1310
+ )
1311
+
1312
+ membership_changes[room_id] = membership_change
1313
+
1314
+ return membership_changes
1315
+
1316
+ membership_changes = await self.db_pool.runInteraction(
1317
+ "get_sliding_sync_membership_changes", f
1318
+ )
1319
+
1320
+ return membership_changes
1321
+
1322
+ @cancellable
1323
+ async def get_membership_changes_for_user(
1324
+ self,
1325
+ user_id: str,
1326
+ from_key: RoomStreamToken,
1327
+ to_key: RoomStreamToken,
1328
+ excluded_rooms: list[str] | None = None,
1329
+ ) -> list[EventBase]:
1330
+ """Fetch membership events for a given user.
1331
+
1332
+ All such events whose stream ordering `s` lies in the range
1333
+ `from_key < s <= to_key` are returned. Events are ordered by ascending stream
1334
+ order.
1335
+ """
1336
+ # Start by ruling out cases where a DB query is not necessary.
1337
+ if from_key == to_key:
1338
+ return []
1339
+
1340
+ if from_key:
1341
+ has_changed = self._membership_stream_cache.has_entity_changed(
1342
+ user_id, int(from_key.stream)
1343
+ )
1344
+ if not has_changed:
1345
+ return []
1346
+
1347
+ def f(txn: LoggingTransaction) -> list[_EventDictReturn]:
1348
+ # To handle tokens with a non-empty instance_map we fetch more
1349
+ # results than necessary and then filter down
1350
+ min_from_id = from_key.stream
1351
+ max_to_id = to_key.get_max_stream_pos()
1352
+
1353
+ args: list[Any] = [user_id, min_from_id, max_to_id]
1354
+
1355
+ ignore_room_clause = ""
1356
+ if excluded_rooms is not None and len(excluded_rooms) > 0:
1357
+ ignore_room_clause, ignore_room_args = make_in_list_sql_clause(
1358
+ txn.database_engine, "e.room_id", excluded_rooms, negative=True
1359
+ )
1360
+ ignore_room_clause = f"AND {ignore_room_clause}"
1361
+ args += ignore_room_args
1362
+
1363
+ sql = """
1364
+ SELECT m.event_id, instance_name, topological_ordering, stream_ordering
1365
+ FROM events AS e, room_memberships AS m
1366
+ WHERE e.event_id = m.event_id
1367
+ AND m.user_id = ?
1368
+ AND e.stream_ordering > ? AND e.stream_ordering <= ?
1369
+ %s
1370
+ ORDER BY e.stream_ordering ASC
1371
+ """ % (ignore_room_clause,)
1372
+
1373
+ txn.execute(sql, args)
1374
+
1375
+ rows = [
1376
+ _EventDictReturn(event_id, None, stream_ordering)
1377
+ for event_id, instance_name, topological_ordering, stream_ordering in txn
1378
+ if _filter_results(
1379
+ from_key,
1380
+ to_key,
1381
+ instance_name,
1382
+ topological_ordering,
1383
+ stream_ordering,
1384
+ )
1385
+ ]
1386
+
1387
+ return rows
1388
+
1389
+ rows = await self.db_pool.runInteraction("get_membership_changes_for_user", f)
1390
+
1391
+ ret = await self.get_events_as_list(
1392
+ [r.event_id for r in rows], get_prev_content=True
1393
+ )
1394
+
1395
+ return ret
1396
+
1397
+ async def get_recent_events_for_room(
1398
+ self, room_id: str, limit: int, end_token: RoomStreamToken
1399
+ ) -> tuple[list[EventBase], RoomStreamToken]:
1400
+ """Get the most recent events in the room in topological ordering.
1401
+
1402
+ Args:
1403
+ room_id
1404
+ limit
1405
+ end_token: The stream token representing now.
1406
+
1407
+ Returns:
1408
+ A list of events and a token pointing to the start of the returned
1409
+ events. The events returned are in ascending topological order.
1410
+ """
1411
+
1412
+ rows, token = await self.get_recent_event_ids_for_room(
1413
+ room_id, limit, end_token
1414
+ )
1415
+
1416
+ events = await self.get_events_as_list(
1417
+ [r.event_id for r in rows], get_prev_content=True
1418
+ )
1419
+
1420
+ return events, token
1421
+
1422
+ async def get_recent_event_ids_for_room(
1423
+ self, room_id: str, limit: int, end_token: RoomStreamToken
1424
+ ) -> tuple[list[_EventDictReturn], RoomStreamToken]:
1425
+ """Get the most recent events in the room in topological ordering.
1426
+
1427
+ Args:
1428
+ room_id
1429
+ limit
1430
+ end_token: The stream token representing now.
1431
+
1432
+ Returns:
1433
+ A list of _EventDictReturn and a token pointing to the start of the
1434
+ returned events. The events returned are in ascending order.
1435
+ """
1436
+ # Allow a zero limit here, and no-op.
1437
+ if limit == 0:
1438
+ return [], end_token
1439
+
1440
+ rows, token, _ = await self.db_pool.runInteraction(
1441
+ "get_recent_event_ids_for_room",
1442
+ self._paginate_room_events_by_topological_ordering_txn,
1443
+ room_id,
1444
+ from_token=end_token,
1445
+ limit=limit,
1446
+ )
1447
+
1448
+ # We want to return the results in ascending order.
1449
+ rows.reverse()
1450
+
1451
+ return rows, token
1452
+
1453
+ async def get_room_event_before_stream_ordering(
1454
+ self, room_id: str, stream_ordering: int
1455
+ ) -> tuple[int, int, str] | None:
1456
+ """Gets details of the first event in a room at or before a stream ordering
1457
+
1458
+ Args:
1459
+ room_id:
1460
+ stream_ordering:
1461
+
1462
+ Returns:
1463
+ A tuple of (stream ordering, topological ordering, event_id)
1464
+ """
1465
+
1466
+ def _f(txn: LoggingTransaction) -> tuple[int, int, str] | None:
1467
+ sql = """
1468
+ SELECT stream_ordering, topological_ordering, event_id
1469
+ FROM events
1470
+ LEFT JOIN rejections USING (event_id)
1471
+ WHERE room_id = ?
1472
+ AND stream_ordering <= ?
1473
+ AND NOT outlier
1474
+ AND rejections.event_id IS NULL
1475
+ ORDER BY stream_ordering DESC
1476
+ LIMIT 1
1477
+ """
1478
+ txn.execute(sql, (room_id, stream_ordering))
1479
+ return cast(tuple[int, int, str] | None, txn.fetchone())
1480
+
1481
+ return await self.db_pool.runInteraction(
1482
+ "get_room_event_before_stream_ordering", _f
1483
+ )
1484
+
1485
+ async def get_last_event_id_in_room_before_stream_ordering(
1486
+ self,
1487
+ room_id: str,
1488
+ end_token: RoomStreamToken,
1489
+ ) -> str | None:
1490
+ """Returns the ID of the last event in a room at or before a stream ordering
1491
+
1492
+ Args:
1493
+ room_id
1494
+ end_token: The token used to stream from
1495
+
1496
+ Returns:
1497
+ The ID of the most recent event, or None if there are no events in the room
1498
+ before this stream ordering.
1499
+ """
1500
+ last_event_result = (
1501
+ await self.get_last_event_pos_in_room_before_stream_ordering(
1502
+ room_id, end_token
1503
+ )
1504
+ )
1505
+
1506
+ if last_event_result:
1507
+ return last_event_result[0]
1508
+
1509
+ return None
1510
+
1511
+ async def get_last_event_pos_in_room(
1512
+ self,
1513
+ room_id: str,
1514
+ event_types: StrCollection | None = None,
1515
+ ) -> tuple[str, PersistedEventPosition] | None:
1516
+ """
1517
+ Returns the ID and event position of the last event in a room.
1518
+
1519
+ Based on `get_last_event_pos_in_room_before_stream_ordering(...)`
1520
+
1521
+ Args:
1522
+ room_id
1523
+ event_types: Optional allowlist of event types to filter by
1524
+
1525
+ Returns:
1526
+ The ID of the most recent event and it's position, or None if there are no
1527
+ events in the room that match the given event types.
1528
+ """
1529
+
1530
+ def _get_last_event_pos_in_room_txn(
1531
+ txn: LoggingTransaction,
1532
+ ) -> tuple[str, PersistedEventPosition] | None:
1533
+ event_type_clause = ""
1534
+ event_type_args: list[str] = []
1535
+ if event_types is not None and len(event_types) > 0:
1536
+ event_type_clause, event_type_args = make_in_list_sql_clause(
1537
+ txn.database_engine, "type", event_types
1538
+ )
1539
+ event_type_clause = f"AND {event_type_clause}"
1540
+
1541
+ sql = f"""
1542
+ SELECT event_id, stream_ordering, instance_name
1543
+ FROM events
1544
+ LEFT JOIN rejections USING (event_id)
1545
+ WHERE room_id = ?
1546
+ {event_type_clause}
1547
+ AND NOT outlier
1548
+ AND rejections.event_id IS NULL
1549
+ ORDER BY stream_ordering DESC
1550
+ LIMIT 1
1551
+ """
1552
+
1553
+ txn.execute(
1554
+ sql,
1555
+ [room_id] + event_type_args,
1556
+ )
1557
+
1558
+ row = cast(tuple[str, int, str] | None, txn.fetchone())
1559
+ if row is not None:
1560
+ event_id, stream_ordering, instance_name = row
1561
+
1562
+ return event_id, PersistedEventPosition(
1563
+ # If instance_name is null we default to "master"
1564
+ instance_name or "master",
1565
+ stream_ordering,
1566
+ )
1567
+
1568
+ return None
1569
+
1570
+ return await self.db_pool.runInteraction(
1571
+ "get_last_event_pos_in_room",
1572
+ _get_last_event_pos_in_room_txn,
1573
+ )
1574
+
1575
+ @trace
1576
+ async def get_last_event_pos_in_room_before_stream_ordering(
1577
+ self,
1578
+ room_id: str,
1579
+ end_token: RoomStreamToken,
1580
+ event_types: StrCollection | None = None,
1581
+ ) -> tuple[str, PersistedEventPosition] | None:
1582
+ """
1583
+ Returns the ID and event position of the last event in a room at or before a
1584
+ stream ordering.
1585
+
1586
+ Args:
1587
+ room_id
1588
+ end_token: The token used to stream from
1589
+ event_types: Optional allowlist of event types to filter by
1590
+
1591
+ Returns:
1592
+ The ID of the most recent event and it's position, or None if there are no
1593
+ events in the room before this stream ordering.
1594
+ """
1595
+
1596
+ def get_last_event_pos_in_room_before_stream_ordering_txn(
1597
+ txn: LoggingTransaction,
1598
+ ) -> tuple[str, PersistedEventPosition] | None:
1599
+ # We're looking for the closest event at or before the token. We need to
1600
+ # handle the fact that the stream token can be a vector clock (with an
1601
+ # `instance_map`) and events can be persisted on different instances
1602
+ # (sharded event persisters). The first subquery handles the events that
1603
+ # would be within the vector clock and gets all rows between the minimum and
1604
+ # maximum stream ordering in the token which need to be filtered against the
1605
+ # `instance_map`. The second subquery handles the "before" case and finds
1606
+ # the first row before the token. We then filter out any results past the
1607
+ # token's vector clock and return the first row that matches.
1608
+ min_stream = end_token.stream
1609
+ max_stream = end_token.get_max_stream_pos()
1610
+
1611
+ event_type_clause = ""
1612
+ event_type_args: list[str] = []
1613
+ if event_types is not None and len(event_types) > 0:
1614
+ event_type_clause, event_type_args = make_in_list_sql_clause(
1615
+ txn.database_engine, "type", event_types
1616
+ )
1617
+ event_type_clause = f"AND {event_type_clause}"
1618
+
1619
+ # We use `UNION ALL` because we don't need any of the deduplication logic
1620
+ # (`UNION` is really a `UNION` + `DISTINCT`). `UNION ALL` does preserve the
1621
+ # ordering of the operand queries but there is no actual guarantee that it
1622
+ # has this behavior in all scenarios so we need the extra `ORDER BY` at the
1623
+ # bottom.
1624
+ sql = """
1625
+ SELECT * FROM (
1626
+ SELECT instance_name, stream_ordering, topological_ordering, event_id
1627
+ FROM events
1628
+ LEFT JOIN rejections USING (event_id)
1629
+ WHERE room_id = ?
1630
+ %s
1631
+ AND ? < stream_ordering AND stream_ordering <= ?
1632
+ AND NOT outlier
1633
+ AND rejections.event_id IS NULL
1634
+ ORDER BY stream_ordering DESC
1635
+ ) AS a
1636
+ UNION ALL
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 <= ?
1644
+ AND NOT outlier
1645
+ AND rejections.event_id IS NULL
1646
+ ORDER BY stream_ordering DESC
1647
+ LIMIT 1
1648
+ ) AS b
1649
+ ORDER BY stream_ordering DESC
1650
+ """ % (
1651
+ event_type_clause,
1652
+ event_type_clause,
1653
+ )
1654
+ txn.execute(
1655
+ sql,
1656
+ [room_id]
1657
+ + event_type_args
1658
+ + [min_stream, max_stream, room_id]
1659
+ + event_type_args
1660
+ + [min_stream],
1661
+ )
1662
+
1663
+ for instance_name, stream_ordering, topological_ordering, event_id in txn:
1664
+ if _filter_results(
1665
+ lower_token=None,
1666
+ upper_token=end_token,
1667
+ instance_name=instance_name,
1668
+ topological_ordering=topological_ordering,
1669
+ stream_ordering=stream_ordering,
1670
+ ):
1671
+ return event_id, PersistedEventPosition(
1672
+ # If instance_name is null we default to "master"
1673
+ instance_name or "master",
1674
+ stream_ordering,
1675
+ )
1676
+
1677
+ return None
1678
+
1679
+ return await self.db_pool.runInteraction(
1680
+ "get_last_event_pos_in_room_before_stream_ordering",
1681
+ get_last_event_pos_in_room_before_stream_ordering_txn,
1682
+ )
1683
+
1684
+ async def bulk_get_last_event_pos_in_room_before_stream_ordering(
1685
+ self,
1686
+ room_ids: StrCollection,
1687
+ end_token: RoomStreamToken,
1688
+ ) -> dict[str, int]:
1689
+ """Bulk fetch the stream position of the latest events in the given
1690
+ rooms
1691
+ """
1692
+
1693
+ # First we just get the latest positions for the room, as the vast
1694
+ # majority of them will be before the given end token anyway. By doing
1695
+ # this we can cache most rooms.
1696
+ uncapped_results = await self._bulk_get_max_event_pos(room_ids)
1697
+
1698
+ # Check that the stream position for the rooms are from before the
1699
+ # minimum position of the token. If not then we need to fetch more
1700
+ # rows.
1701
+ results: dict[str, int] = {}
1702
+ recheck_rooms: set[str] = set()
1703
+ min_token = end_token.stream
1704
+ for room_id, stream in uncapped_results.items():
1705
+ if stream is None:
1706
+ # Despite the function not directly setting None, the cache can!
1707
+ # See: https://github.com/element-hq/synapse/issues/17726
1708
+ continue
1709
+ if stream <= min_token:
1710
+ results[room_id] = stream
1711
+ else:
1712
+ recheck_rooms.add(room_id)
1713
+
1714
+ if not recheck_rooms:
1715
+ return results
1716
+
1717
+ # There shouldn't be many rooms that we need to recheck, so we do them
1718
+ # one-by-one.
1719
+ for room_id in recheck_rooms:
1720
+ result = await self.get_last_event_pos_in_room_before_stream_ordering(
1721
+ room_id, end_token
1722
+ )
1723
+ if result is not None:
1724
+ results[room_id] = result[1].stream
1725
+
1726
+ return results
1727
+
1728
+ @cached()
1729
+ async def _get_max_event_pos(self, room_id: str) -> int:
1730
+ raise NotImplementedError()
1731
+
1732
+ @cachedList(cached_method_name="_get_max_event_pos", list_name="room_ids")
1733
+ async def _bulk_get_max_event_pos(
1734
+ self, room_ids: StrCollection
1735
+ ) -> Mapping[str, int | None]:
1736
+ """Fetch the max position of a persisted event in the room."""
1737
+
1738
+ # We need to be careful not to return positions ahead of the current
1739
+ # positions, so we get the current token now and cap our queries to it.
1740
+ now_token = self.get_room_max_token()
1741
+ max_pos = now_token.get_max_stream_pos()
1742
+
1743
+ results: dict[str, int] = {}
1744
+
1745
+ # First, we check for the rooms in the stream change cache to see if we
1746
+ # can just use the latest position from it.
1747
+ missing_room_ids: set[str] = set()
1748
+ for room_id in room_ids:
1749
+ stream_pos = self._events_stream_cache.get_max_pos_of_last_change(room_id)
1750
+ if stream_pos is not None:
1751
+ results[room_id] = stream_pos
1752
+ else:
1753
+ missing_room_ids.add(room_id)
1754
+
1755
+ if not missing_room_ids:
1756
+ return results
1757
+
1758
+ # Next, we query the stream position from the DB. At first we fetch all
1759
+ # positions less than the *max* stream pos in the token, then filter
1760
+ # them down. We do this as a) this is a cheaper query, and b) the vast
1761
+ # majority of rooms will have a latest token from before the min stream
1762
+ # pos.
1763
+
1764
+ def bulk_get_max_event_pos_fallback_txn(
1765
+ txn: LoggingTransaction, batched_room_ids: StrCollection
1766
+ ) -> dict[str, int]:
1767
+ clause, args = make_in_list_sql_clause(
1768
+ self.database_engine, "room_id", batched_room_ids
1769
+ )
1770
+ sql = f"""
1771
+ SELECT room_id, (
1772
+ SELECT stream_ordering FROM events AS e
1773
+ LEFT JOIN rejections USING (event_id)
1774
+ WHERE e.room_id = r.room_id
1775
+ AND e.stream_ordering <= ?
1776
+ AND NOT outlier
1777
+ AND rejection_reason IS NULL
1778
+ ORDER BY stream_ordering DESC
1779
+ LIMIT 1
1780
+ )
1781
+ FROM rooms AS r
1782
+ WHERE {clause}
1783
+ """
1784
+ txn.execute(sql, [max_pos] + args)
1785
+ return {row[0]: row[1] for row in txn}
1786
+
1787
+ # It's easier to look at the `sliding_sync_joined_rooms` table and avoid all of
1788
+ # the joins and sub-queries.
1789
+ def bulk_get_max_event_pos_from_sliding_sync_tables_txn(
1790
+ txn: LoggingTransaction, batched_room_ids: StrCollection
1791
+ ) -> dict[str, int]:
1792
+ clause, args = make_in_list_sql_clause(
1793
+ self.database_engine, "room_id", batched_room_ids
1794
+ )
1795
+ sql = f"""
1796
+ SELECT room_id, event_stream_ordering
1797
+ FROM sliding_sync_joined_rooms
1798
+ WHERE {clause}
1799
+ ORDER BY event_stream_ordering DESC
1800
+ """
1801
+ txn.execute(sql, args)
1802
+ return {row[0]: row[1] for row in txn}
1803
+
1804
+ recheck_rooms: set[str] = set()
1805
+ for batched in batch_iter(room_ids, 1000):
1806
+ if await self.have_finished_sliding_sync_background_jobs():
1807
+ batch_results = await self.db_pool.runInteraction(
1808
+ "bulk_get_max_event_pos_from_sliding_sync_tables_txn",
1809
+ bulk_get_max_event_pos_from_sliding_sync_tables_txn,
1810
+ batched,
1811
+ )
1812
+ else:
1813
+ batch_results = await self.db_pool.runInteraction(
1814
+ "bulk_get_max_event_pos_fallback_txn",
1815
+ bulk_get_max_event_pos_fallback_txn,
1816
+ batched,
1817
+ )
1818
+ for room_id, stream_ordering in batch_results.items():
1819
+ if stream_ordering <= now_token.stream:
1820
+ results[room_id] = stream_ordering
1821
+ else:
1822
+ recheck_rooms.add(room_id)
1823
+
1824
+ # We now need to handle rooms where the above query returned a stream
1825
+ # position that was potentially too new. This should happen very rarely
1826
+ # so we just query the rooms one-by-one
1827
+ for room_id in recheck_rooms:
1828
+ result = await self.get_last_event_pos_in_room_before_stream_ordering(
1829
+ room_id, now_token
1830
+ )
1831
+ if result is not None:
1832
+ results[room_id] = result[1].stream
1833
+
1834
+ return results
1835
+
1836
+ async def get_current_room_stream_token_for_room_id(
1837
+ self, room_id: str
1838
+ ) -> RoomStreamToken:
1839
+ """Returns the current position of the rooms stream (historic token)."""
1840
+ stream_ordering = self.get_room_max_stream_ordering()
1841
+ topo = await self.db_pool.runInteraction(
1842
+ "_get_max_topological_txn", self._get_max_topological_txn, room_id
1843
+ )
1844
+ return RoomStreamToken(topological=topo, stream=stream_ordering)
1845
+
1846
+ @overload
1847
+ def get_stream_id_for_event_txn(
1848
+ self,
1849
+ txn: LoggingTransaction,
1850
+ event_id: str,
1851
+ allow_none: Literal[False] = False,
1852
+ ) -> int: ...
1853
+
1854
+ @overload
1855
+ def get_stream_id_for_event_txn(
1856
+ self,
1857
+ txn: LoggingTransaction,
1858
+ event_id: str,
1859
+ allow_none: bool = False,
1860
+ ) -> int | None: ...
1861
+
1862
+ def get_stream_id_for_event_txn(
1863
+ self,
1864
+ txn: LoggingTransaction,
1865
+ event_id: str,
1866
+ allow_none: bool = False,
1867
+ ) -> int | None:
1868
+ # Type ignore: we pass keyvalues a Dict[str, str]; the function wants
1869
+ # Dict[str, Any]. I think mypy is unhappy because Dict is invariant?
1870
+ return self.db_pool.simple_select_one_onecol_txn( # type: ignore[call-overload]
1871
+ txn=txn,
1872
+ table="events",
1873
+ keyvalues={"event_id": event_id},
1874
+ retcol="stream_ordering",
1875
+ allow_none=allow_none,
1876
+ )
1877
+
1878
+ async def get_position_for_event(self, event_id: str) -> PersistedEventPosition:
1879
+ """Get the persisted position for an event"""
1880
+ row = await self.db_pool.simple_select_one(
1881
+ table="events",
1882
+ keyvalues={"event_id": event_id},
1883
+ retcols=("stream_ordering", "instance_name"),
1884
+ desc="get_position_for_event",
1885
+ )
1886
+
1887
+ return PersistedEventPosition(row[1] or "master", row[0])
1888
+
1889
+ async def get_topological_token_for_event(self, event_id: str) -> RoomStreamToken:
1890
+ """The stream token for an event
1891
+ Args:
1892
+ event_id: The id of the event to look up a stream token for.
1893
+ Raises:
1894
+ StoreError if the event wasn't in the database.
1895
+ Returns:
1896
+ A `RoomStreamToken` topological token.
1897
+ """
1898
+ row = await self.db_pool.simple_select_one(
1899
+ table="events",
1900
+ keyvalues={"event_id": event_id},
1901
+ retcols=("stream_ordering", "topological_ordering"),
1902
+ desc="get_topological_token_for_event",
1903
+ )
1904
+ return RoomStreamToken(topological=row[1], stream=row[0])
1905
+
1906
+ async def get_current_topological_token(self, room_id: str, stream_key: int) -> int:
1907
+ """Gets the topological token in a room after or at the given stream
1908
+ ordering.
1909
+
1910
+ Args:
1911
+ room_id
1912
+ stream_key
1913
+ """
1914
+ if isinstance(self.database_engine, PostgresEngine):
1915
+ min_function = "LEAST"
1916
+ elif isinstance(self.database_engine, Sqlite3Engine):
1917
+ min_function = "MIN"
1918
+ else:
1919
+ raise RuntimeError(f"Unknown database engine {self.database_engine}")
1920
+
1921
+ # This query used to be
1922
+ # SELECT COALESCE(MIN(topological_ordering), 0) FROM events
1923
+ # WHERE room_id = ? and events.stream_ordering >= {stream_key}
1924
+ # which returns 0 if the stream_key is newer than any event in
1925
+ # the room. That's not wrong, but it seems to interact oddly with backfill,
1926
+ # requiring a second call to /messages to actually backfill from a remote
1927
+ # homeserver.
1928
+ #
1929
+ # Instead, rollback the stream ordering to that after the most recent event in
1930
+ # this room.
1931
+ sql = f"""
1932
+ WITH fallback(max_stream_ordering) AS (
1933
+ SELECT MAX(stream_ordering)
1934
+ FROM events
1935
+ WHERE room_id = ?
1936
+ )
1937
+ SELECT COALESCE(MIN(topological_ordering), 0) FROM events
1938
+ WHERE
1939
+ room_id = ?
1940
+ AND events.stream_ordering >= {min_function}(
1941
+ ?,
1942
+ (SELECT max_stream_ordering FROM fallback)
1943
+ )
1944
+ """
1945
+
1946
+ row = await self.db_pool.execute(
1947
+ "get_current_topological_token", sql, room_id, room_id, stream_key
1948
+ )
1949
+ return row[0][0] if row else 0
1950
+
1951
+ def _get_max_topological_txn(self, txn: LoggingTransaction, room_id: str) -> int:
1952
+ txn.execute(
1953
+ "SELECT MAX(topological_ordering) FROM events WHERE room_id = ?",
1954
+ (room_id,),
1955
+ )
1956
+
1957
+ rows = txn.fetchall()
1958
+ # An aggregate function like MAX() will always return one row per group
1959
+ # so we can safely rely on the lookup here. For example, when a we
1960
+ # lookup a `room_id` which does not exist, `rows` will look like
1961
+ # `[(None,)]`
1962
+ return rows[0][0] if rows[0][0] is not None else 0
1963
+
1964
+ async def get_events_around(
1965
+ self,
1966
+ room_id: str,
1967
+ event_id: str,
1968
+ before_limit: int,
1969
+ after_limit: int,
1970
+ event_filter: Filter | None = None,
1971
+ ) -> _EventsAround:
1972
+ """Retrieve events and pagination tokens around a given event in a
1973
+ room.
1974
+ """
1975
+
1976
+ results = await self.db_pool.runInteraction(
1977
+ "get_events_around",
1978
+ self._get_events_around_txn,
1979
+ room_id,
1980
+ event_id,
1981
+ before_limit,
1982
+ after_limit,
1983
+ event_filter,
1984
+ )
1985
+
1986
+ events_before = await self.get_events_as_list(
1987
+ list(results["before"]["event_ids"]), get_prev_content=True
1988
+ )
1989
+
1990
+ events_after = await self.get_events_as_list(
1991
+ list(results["after"]["event_ids"]), get_prev_content=True
1992
+ )
1993
+
1994
+ return _EventsAround(
1995
+ events_before=events_before,
1996
+ events_after=events_after,
1997
+ start=results["before"]["token"],
1998
+ end=results["after"]["token"],
1999
+ )
2000
+
2001
+ def _get_events_around_txn(
2002
+ self,
2003
+ txn: LoggingTransaction,
2004
+ room_id: str,
2005
+ event_id: str,
2006
+ before_limit: int,
2007
+ after_limit: int,
2008
+ event_filter: Filter | None,
2009
+ ) -> dict:
2010
+ """Retrieves event_ids and pagination tokens around a given event in a
2011
+ room.
2012
+
2013
+ Args:
2014
+ room_id
2015
+ event_id
2016
+ before_limit
2017
+ after_limit
2018
+ event_filter
2019
+
2020
+ Returns:
2021
+ dict
2022
+ """
2023
+
2024
+ row = self.db_pool.simple_select_one_txn(
2025
+ txn,
2026
+ "events",
2027
+ keyvalues={"event_id": event_id, "room_id": room_id},
2028
+ retcols=("stream_ordering", "topological_ordering"),
2029
+ )
2030
+ stream_ordering = int(row[0])
2031
+ topological_ordering = int(row[1])
2032
+
2033
+ # Paginating backwards includes the event at the token, but paginating
2034
+ # forward doesn't.
2035
+ before_token = RoomStreamToken(
2036
+ topological=topological_ordering - 1, stream=stream_ordering
2037
+ )
2038
+
2039
+ after_token = RoomStreamToken(
2040
+ topological=topological_ordering, stream=stream_ordering
2041
+ )
2042
+
2043
+ rows, start_token, _ = self._paginate_room_events_by_topological_ordering_txn(
2044
+ txn,
2045
+ room_id,
2046
+ before_token,
2047
+ direction=Direction.BACKWARDS,
2048
+ limit=before_limit,
2049
+ event_filter=event_filter,
2050
+ )
2051
+ events_before = [r.event_id for r in rows]
2052
+
2053
+ rows, end_token, _ = self._paginate_room_events_by_topological_ordering_txn(
2054
+ txn,
2055
+ room_id,
2056
+ after_token,
2057
+ direction=Direction.FORWARDS,
2058
+ limit=after_limit,
2059
+ event_filter=event_filter,
2060
+ )
2061
+ events_after = [r.event_id for r in rows]
2062
+
2063
+ return {
2064
+ "before": {"event_ids": events_before, "token": start_token},
2065
+ "after": {"event_ids": events_after, "token": end_token},
2066
+ }
2067
+
2068
+ async def get_all_new_event_ids_stream(
2069
+ self,
2070
+ from_id: int,
2071
+ current_id: int,
2072
+ limit: int,
2073
+ ) -> tuple[int, dict[str, int | None]]:
2074
+ """Get all new events
2075
+
2076
+ Returns all event ids with from_id < stream_ordering <= current_id.
2077
+
2078
+ Args:
2079
+ from_id: the stream_ordering of the last event we processed
2080
+ current_id: the stream_ordering of the most recently processed event
2081
+ limit: the maximum number of events to return
2082
+
2083
+ Returns:
2084
+ A tuple of (next_id, event_to_received_ts), where `next_id`
2085
+ is the next value to pass as `from_id` (it will either be the
2086
+ stream_ordering of the last returned event, or, if fewer than `limit`
2087
+ events were found, the `current_id`). The `event_to_received_ts` is
2088
+ a dictionary mapping event ID to the event `received_ts`, sorted by ascending
2089
+ stream_ordering.
2090
+ """
2091
+
2092
+ def get_all_new_event_ids_stream_txn(
2093
+ txn: LoggingTransaction,
2094
+ ) -> tuple[int, dict[str, int | None]]:
2095
+ sql = (
2096
+ "SELECT e.stream_ordering, e.event_id, e.received_ts"
2097
+ " FROM events AS e"
2098
+ " WHERE"
2099
+ " ? < e.stream_ordering AND e.stream_ordering <= ?"
2100
+ " ORDER BY e.stream_ordering ASC"
2101
+ " LIMIT ?"
2102
+ )
2103
+
2104
+ txn.execute(sql, (from_id, current_id, limit))
2105
+ rows = txn.fetchall()
2106
+
2107
+ upper_bound = current_id
2108
+ if len(rows) == limit:
2109
+ upper_bound = rows[-1][0]
2110
+
2111
+ event_to_received_ts: dict[str, int | None] = {
2112
+ row[1]: row[2] for row in rows
2113
+ }
2114
+ return upper_bound, event_to_received_ts
2115
+
2116
+ upper_bound, event_to_received_ts = await self.db_pool.runInteraction(
2117
+ "get_all_new_event_ids_stream", get_all_new_event_ids_stream_txn
2118
+ )
2119
+
2120
+ return upper_bound, event_to_received_ts
2121
+
2122
+ async def get_federation_out_pos(self, typ: str) -> int:
2123
+ if self._need_to_reset_federation_stream_positions:
2124
+ await self.db_pool.runInteraction(
2125
+ "_reset_federation_positions_txn", self._reset_federation_positions_txn
2126
+ )
2127
+ self._need_to_reset_federation_stream_positions = False
2128
+
2129
+ return await self.db_pool.simple_select_one_onecol(
2130
+ table="federation_stream_position",
2131
+ retcol="stream_id",
2132
+ keyvalues={"type": typ, "instance_name": self._instance_name},
2133
+ desc="get_federation_out_pos",
2134
+ )
2135
+
2136
+ async def update_federation_out_pos(self, typ: str, stream_id: int) -> None:
2137
+ if self._need_to_reset_federation_stream_positions:
2138
+ await self.db_pool.runInteraction(
2139
+ "_reset_federation_positions_txn", self._reset_federation_positions_txn
2140
+ )
2141
+ self._need_to_reset_federation_stream_positions = False
2142
+
2143
+ await self.db_pool.simple_update_one(
2144
+ table="federation_stream_position",
2145
+ keyvalues={"type": typ, "instance_name": self._instance_name},
2146
+ updatevalues={"stream_id": stream_id},
2147
+ desc="update_federation_out_pos",
2148
+ )
2149
+
2150
+ def _reset_federation_positions_txn(self, txn: LoggingTransaction) -> None:
2151
+ """Fiddles with the `federation_stream_position` table to make it match
2152
+ the configured federation sender instances during start up.
2153
+ """
2154
+
2155
+ # The federation sender instances may have changed, so we need to
2156
+ # massage the `federation_stream_position` table to have a row per type
2157
+ # per instance sending federation. If there is a mismatch we update the
2158
+ # table with the correct rows using the *minimum* stream ID seen. This
2159
+ # may result in resending of events/EDUs to remote servers, but that is
2160
+ # preferable to dropping them.
2161
+
2162
+ if not self._send_federation:
2163
+ return
2164
+
2165
+ # Pull out the configured instances. If we don't have a shard config then
2166
+ # we assume that we're the only instance sending.
2167
+ configured_instances = self._federation_shard_config.instances
2168
+ if not configured_instances:
2169
+ configured_instances = [self._instance_name]
2170
+ elif self._instance_name not in configured_instances:
2171
+ return
2172
+
2173
+ instances_in_table = self.db_pool.simple_select_onecol_txn(
2174
+ txn,
2175
+ table="federation_stream_position",
2176
+ keyvalues={},
2177
+ retcol="instance_name",
2178
+ )
2179
+
2180
+ if set(instances_in_table) == set(configured_instances):
2181
+ # Nothing to do
2182
+ return
2183
+
2184
+ sql = """
2185
+ SELECT type, MIN(stream_id) FROM federation_stream_position
2186
+ GROUP BY type
2187
+ """
2188
+ txn.execute(sql)
2189
+ min_positions = dict(
2190
+ cast(Iterable[tuple[str, int]], txn)
2191
+ ) # Map from type -> min position
2192
+
2193
+ # Ensure we do actually have some values here
2194
+ assert set(min_positions) == {"federation", "events"}
2195
+
2196
+ sql = """
2197
+ DELETE FROM federation_stream_position
2198
+ WHERE NOT (%s)
2199
+ """
2200
+ clause, args = make_in_list_sql_clause(
2201
+ txn.database_engine, "instance_name", configured_instances
2202
+ )
2203
+ txn.execute(sql % (clause,), args)
2204
+
2205
+ for typ, stream_id in min_positions.items():
2206
+ self.db_pool.simple_upsert_txn(
2207
+ txn,
2208
+ table="federation_stream_position",
2209
+ keyvalues={"type": typ, "instance_name": self._instance_name},
2210
+ values={"stream_id": stream_id},
2211
+ )
2212
+
2213
+ def has_room_changed_since(self, room_id: str, stream_id: int) -> bool:
2214
+ return self._events_stream_cache.has_entity_changed(room_id, stream_id)
2215
+
2216
+ def _paginate_room_events_by_topological_ordering_txn(
2217
+ self,
2218
+ txn: LoggingTransaction,
2219
+ room_id: str,
2220
+ from_token: RoomStreamToken,
2221
+ to_token: RoomStreamToken | None = None,
2222
+ direction: Direction = Direction.BACKWARDS,
2223
+ limit: int = 0,
2224
+ event_filter: Filter | None = None,
2225
+ ) -> tuple[list[_EventDictReturn], RoomStreamToken, bool]:
2226
+ """Returns list of events before or after a given token.
2227
+
2228
+ Args:
2229
+ txn
2230
+ room_id
2231
+ from_token: The token used to stream from
2232
+ to_token: A token which if given limits the results to only those before
2233
+ direction: Indicates whether we are paginating forwards or backwards
2234
+ from `from_key`.
2235
+ limit: The maximum number of events to return.
2236
+ event_filter: If provided filters the events to
2237
+ those that match the filter.
2238
+
2239
+ Returns:
2240
+ A list of _EventDictReturn, a token that points to the end of the
2241
+ result set, and a boolean to indicate if there were more events but
2242
+ we hit the limit. If no events are returned then the end of the
2243
+ stream has been reached (i.e. there are no events between
2244
+ `from_token` and `to_token`), or `limit` is zero.
2245
+ """
2246
+ # We can bail early if we're looking forwards, and our `to_key` is already
2247
+ # before our `from_token`.
2248
+ if (
2249
+ direction == Direction.FORWARDS
2250
+ and to_token is not None
2251
+ and to_token.is_before_or_eq(from_token)
2252
+ ):
2253
+ # Token selection matches what we do below if there are no rows
2254
+ return [], to_token if to_token else from_token, False
2255
+ # Or vice-versa, if we're looking backwards and our `from_token` is already before
2256
+ # our `to_token`.
2257
+ elif (
2258
+ direction == Direction.BACKWARDS
2259
+ and to_token is not None
2260
+ and from_token.is_before_or_eq(to_token)
2261
+ ):
2262
+ # Token selection matches what we do below if there are no rows
2263
+ return [], to_token if to_token else from_token, False
2264
+
2265
+ args: list[Any] = [room_id]
2266
+
2267
+ order, from_bound, to_bound = generate_pagination_bounds(
2268
+ direction, from_token, to_token
2269
+ )
2270
+
2271
+ bounds = generate_pagination_where_clause(
2272
+ direction=direction,
2273
+ column_names=("event.topological_ordering", "event.stream_ordering"),
2274
+ from_token=from_bound,
2275
+ to_token=to_bound,
2276
+ engine=self.database_engine,
2277
+ )
2278
+
2279
+ filter_clause, filter_args = filter_to_clause(event_filter)
2280
+
2281
+ if filter_clause:
2282
+ bounds += " AND " + filter_clause
2283
+ args.extend(filter_args)
2284
+
2285
+ # We fetch more events as we'll filter the result set
2286
+ requested_limit = int(limit) * 2
2287
+ args.append(int(limit) * 2)
2288
+
2289
+ select_keywords = "SELECT"
2290
+ join_clause = ""
2291
+ # Using DISTINCT in this SELECT query is quite expensive, because it
2292
+ # requires the engine to sort on the entire (not limited) result set,
2293
+ # i.e. the entire events table. Only use it in scenarios that could result
2294
+ # in the same event ID occurring multiple times in the results.
2295
+ needs_distinct = False
2296
+ if event_filter and event_filter.labels:
2297
+ # If we're not filtering on a label, then joining on event_labels will
2298
+ # return as many row for a single event as the number of labels it has. To
2299
+ # avoid this, only join if we're filtering on at least one label.
2300
+ join_clause += """
2301
+ LEFT JOIN event_labels
2302
+ USING (event_id, room_id, topological_ordering)
2303
+ """
2304
+ if len(event_filter.labels) > 1:
2305
+ # Multiple labels could cause the same event to appear multiple times.
2306
+ needs_distinct = True
2307
+
2308
+ # If there is a relation_senders and relation_types filter join to the
2309
+ # relations table to get events related to the current event.
2310
+ if event_filter and (
2311
+ event_filter.related_by_senders or event_filter.related_by_rel_types
2312
+ ):
2313
+ # Filtering by relations could cause the same event to appear multiple
2314
+ # times (since there's no limit on the number of relations to an event).
2315
+ needs_distinct = True
2316
+ join_clause += """
2317
+ LEFT JOIN event_relations AS relation ON (event.event_id = relation.relates_to_id)
2318
+ """
2319
+ if event_filter.related_by_senders:
2320
+ join_clause += """
2321
+ LEFT JOIN events AS related_event ON (relation.event_id = related_event.event_id)
2322
+ """
2323
+
2324
+ # If there is a not_rel_types filter join to the relations table to get
2325
+ # the event's relation information.
2326
+ if event_filter and (event_filter.rel_types or event_filter.not_rel_types):
2327
+ join_clause += """
2328
+ LEFT JOIN event_relations AS event_relation USING (event_id)
2329
+ """
2330
+
2331
+ if needs_distinct:
2332
+ select_keywords += " DISTINCT"
2333
+
2334
+ sql = """
2335
+ %(select_keywords)s
2336
+ event.event_id, event.instance_name,
2337
+ event.topological_ordering, event.stream_ordering
2338
+ FROM events AS event
2339
+ %(join_clause)s
2340
+ WHERE event.outlier = FALSE AND event.room_id = ? AND %(bounds)s
2341
+ ORDER BY event.topological_ordering %(order)s,
2342
+ event.stream_ordering %(order)s LIMIT ?
2343
+ """ % {
2344
+ "select_keywords": select_keywords,
2345
+ "join_clause": join_clause,
2346
+ "bounds": bounds,
2347
+ "order": order,
2348
+ }
2349
+ txn.execute(sql, args)
2350
+
2351
+ # Get all the rows and check if we hit the limit.
2352
+ fetched_rows = txn.fetchall()
2353
+ limited = len(fetched_rows) >= requested_limit
2354
+
2355
+ # Filter the result set.
2356
+ rows = [
2357
+ _EventDictReturn(event_id, topological_ordering, stream_ordering)
2358
+ for event_id, instance_name, topological_ordering, stream_ordering in fetched_rows
2359
+ if _filter_results(
2360
+ lower_token=(
2361
+ to_token if direction == Direction.BACKWARDS else from_token
2362
+ ),
2363
+ upper_token=(
2364
+ from_token if direction == Direction.BACKWARDS else to_token
2365
+ ),
2366
+ instance_name=instance_name,
2367
+ topological_ordering=topological_ordering,
2368
+ stream_ordering=stream_ordering,
2369
+ )
2370
+ ]
2371
+
2372
+ if len(rows) > limit:
2373
+ limited = True
2374
+
2375
+ rows = rows[:limit]
2376
+
2377
+ if rows:
2378
+ assert rows[-1].topological_ordering is not None
2379
+ next_token = generate_next_token(
2380
+ direction, rows[-1].topological_ordering, rows[-1].stream_ordering
2381
+ )
2382
+ else:
2383
+ # TODO (erikj): We should work out what to do here instead.
2384
+ next_token = to_token if to_token else from_token
2385
+
2386
+ return rows, next_token, limited
2387
+
2388
+ @trace
2389
+ @tag_args
2390
+ async def paginate_room_events_by_topological_ordering(
2391
+ self,
2392
+ *,
2393
+ room_id: str,
2394
+ from_key: RoomStreamToken,
2395
+ to_key: RoomStreamToken | None = None,
2396
+ direction: Direction = Direction.BACKWARDS,
2397
+ limit: int = 0,
2398
+ event_filter: Filter | None = None,
2399
+ ) -> tuple[list[EventBase], RoomStreamToken, bool]:
2400
+ """
2401
+ Paginate events by `topological_ordering` (tie-break with `stream_ordering`) in
2402
+ the room from the `from_key` in the given `direction` to the `to_key` or
2403
+ `limit`.
2404
+
2405
+ Args:
2406
+ room_id
2407
+ from_key: The token to stream from (starting point and heading in the given
2408
+ direction)
2409
+ to_key: The token representing the end stream position (end point)
2410
+ direction: Indicates whether we are paginating forwards or backwards
2411
+ from `from_key`.
2412
+ limit: Maximum number of events to return
2413
+ event_filter: If provided filters the events to those that match the filter.
2414
+
2415
+ Returns:
2416
+ The results as a list of events, a token that points to the end of
2417
+ the result set, and a boolean to indicate if there were more events
2418
+ but we hit the limit. If no events are returned then the end of the
2419
+ stream has been reached (i.e. there are no events between `from_key`
2420
+ and `to_key`).
2421
+
2422
+ When Direction.FORWARDS: from_key < x <= to_key, (ascending order)
2423
+ When Direction.BACKWARDS: from_key >= x > to_key, (descending order)
2424
+ """
2425
+
2426
+ # FIXME: When going forwards, we should enforce that the `to_key` is not `None`
2427
+ # because we always need an upper bound when querying the events stream (as
2428
+ # otherwise we'll potentially pick up events that are not fully persisted).
2429
+
2430
+ # We have these checks outside of the transaction function (txn) to save getting
2431
+ # a DB connection and switching threads if we don't need to.
2432
+ #
2433
+ # We can bail early if we're looking forwards, and our `to_key` is already
2434
+ # before our `from_key`.
2435
+ if (
2436
+ direction == Direction.FORWARDS
2437
+ and to_key is not None
2438
+ and to_key.is_before_or_eq(from_key)
2439
+ ):
2440
+ # Token selection matches what we do in `_paginate_room_events_txn` if there
2441
+ # are no rows
2442
+ return [], to_key if to_key else from_key, False
2443
+ # Or vice-versa, if we're looking backwards and our `from_key` is already before
2444
+ # our `to_key`.
2445
+ elif (
2446
+ direction == Direction.BACKWARDS
2447
+ and to_key is not None
2448
+ and from_key.is_before_or_eq(to_key)
2449
+ ):
2450
+ # Token selection matches what we do in `_paginate_room_events_txn` if there
2451
+ # are no rows
2452
+ return [], to_key if to_key else from_key, False
2453
+
2454
+ rows, token, limited = await self.db_pool.runInteraction(
2455
+ "paginate_room_events_by_topological_ordering",
2456
+ self._paginate_room_events_by_topological_ordering_txn,
2457
+ room_id,
2458
+ from_key,
2459
+ to_key,
2460
+ direction,
2461
+ limit,
2462
+ event_filter,
2463
+ )
2464
+
2465
+ events = await self.get_events_as_list(
2466
+ [r.event_id for r in rows], get_prev_content=True
2467
+ )
2468
+
2469
+ return events, token, limited
2470
+
2471
+ @cached()
2472
+ async def get_id_for_instance(self, instance_name: str) -> int:
2473
+ """Get a unique, immutable ID that corresponds to the given Synapse worker instance."""
2474
+
2475
+ def _get_id_for_instance_txn(txn: LoggingTransaction) -> int:
2476
+ instance_id = self.db_pool.simple_select_one_onecol_txn(
2477
+ txn,
2478
+ table="instance_map",
2479
+ keyvalues={"instance_name": instance_name},
2480
+ retcol="instance_id",
2481
+ allow_none=True,
2482
+ )
2483
+ if instance_id is not None:
2484
+ return instance_id
2485
+
2486
+ # If we don't have an entry upsert one.
2487
+ #
2488
+ # We could do this before the first check, and rely on the cache for
2489
+ # efficiency, but each UPSERT causes the next ID to increment which
2490
+ # can quickly bloat the size of the generated IDs for new instances.
2491
+ self.db_pool.simple_upsert_txn(
2492
+ txn,
2493
+ table="instance_map",
2494
+ keyvalues={"instance_name": instance_name},
2495
+ values={},
2496
+ )
2497
+
2498
+ return self.db_pool.simple_select_one_onecol_txn(
2499
+ txn,
2500
+ table="instance_map",
2501
+ keyvalues={"instance_name": instance_name},
2502
+ retcol="instance_id",
2503
+ )
2504
+
2505
+ return await self.db_pool.runInteraction(
2506
+ "get_id_for_instance", _get_id_for_instance_txn
2507
+ )
2508
+
2509
+ @cached()
2510
+ async def get_name_from_instance_id(self, instance_id: int) -> str:
2511
+ """Get the instance name from an ID previously returned by
2512
+ `get_id_for_instance`.
2513
+ """
2514
+
2515
+ return await self.db_pool.simple_select_one_onecol(
2516
+ table="instance_map",
2517
+ keyvalues={"instance_id": instance_id},
2518
+ retcol="instance_name",
2519
+ desc="get_name_from_instance_id",
2520
+ )
2521
+
2522
+ async def get_timeline_gaps(
2523
+ self,
2524
+ room_id: str,
2525
+ from_token: RoomStreamToken | None,
2526
+ to_token: RoomStreamToken,
2527
+ ) -> RoomStreamToken | None:
2528
+ """Check if there is a gap, and return a token that marks the position
2529
+ of the gap in the stream.
2530
+ """
2531
+
2532
+ sql = """
2533
+ SELECT instance_name, stream_ordering
2534
+ FROM timeline_gaps
2535
+ WHERE room_id = ? AND ? < stream_ordering AND stream_ordering <= ?
2536
+ ORDER BY stream_ordering
2537
+ """
2538
+
2539
+ rows = await self.db_pool.execute(
2540
+ "get_timeline_gaps",
2541
+ sql,
2542
+ room_id,
2543
+ from_token.stream if from_token else 0,
2544
+ to_token.get_max_stream_pos(),
2545
+ )
2546
+
2547
+ if not rows:
2548
+ return None
2549
+
2550
+ positions = [
2551
+ PersistedEventPosition(instance_name, stream_ordering)
2552
+ for instance_name, stream_ordering in rows
2553
+ ]
2554
+ if from_token:
2555
+ positions = [p for p in positions if p.persisted_after(from_token)]
2556
+
2557
+ positions = [p for p in positions if not p.persisted_after(to_token)]
2558
+
2559
+ if positions:
2560
+ # We return a stream token that ensures the event *at* the position
2561
+ # of the gap is included (as the gap is *before* the persisted
2562
+ # event).
2563
+ last_position = positions[-1]
2564
+ return RoomStreamToken(stream=last_position.stream - 1)
2565
+
2566
+ return None
2567
+
2568
+ @trace
2569
+ def get_rooms_that_might_have_updates(
2570
+ self, room_ids: StrCollection, from_token: RoomStreamToken
2571
+ ) -> StrCollection:
2572
+ """Filters given room IDs down to those that might have updates, i.e.
2573
+ removes rooms that definitely do not have updates.
2574
+ """
2575
+ return self._events_stream_cache.get_entities_changed(
2576
+ room_ids, from_token.stream
2577
+ )