matrix-synapse 1.142.0rc3__cp314-abi3-musllinux_1_2_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.
- matrix_synapse-1.142.0rc3.dist-info/AUTHORS.rst +51 -0
- matrix_synapse-1.142.0rc3.dist-info/LICENSE-AGPL-3.0 +661 -0
- matrix_synapse-1.142.0rc3.dist-info/LICENSE-COMMERCIAL +6 -0
- matrix_synapse-1.142.0rc3.dist-info/METADATA +375 -0
- matrix_synapse-1.142.0rc3.dist-info/RECORD +1057 -0
- matrix_synapse-1.142.0rc3.dist-info/WHEEL +4 -0
- matrix_synapse-1.142.0rc3.dist-info/entry_points.txt +14 -0
- matrix_synapse.libs/libgcc_s-2d945d6c.so.1 +0 -0
- synapse/__init__.py +97 -0
- synapse/_scripts/__init__.py +0 -0
- synapse/_scripts/export_signing_key.py +109 -0
- synapse/_scripts/generate_config.py +83 -0
- synapse/_scripts/generate_log_config.py +56 -0
- synapse/_scripts/generate_signing_key.py +55 -0
- synapse/_scripts/generate_workers_map.py +318 -0
- synapse/_scripts/hash_password.py +95 -0
- synapse/_scripts/move_remote_media_to_new_store.py +128 -0
- synapse/_scripts/register_new_matrix_user.py +374 -0
- synapse/_scripts/review_recent_signups.py +212 -0
- synapse/_scripts/synapse_port_db.py +1603 -0
- synapse/_scripts/synctl.py +365 -0
- synapse/_scripts/update_synapse_database.py +130 -0
- synapse/api/__init__.py +20 -0
- synapse/api/auth/__init__.py +207 -0
- synapse/api/auth/base.py +406 -0
- synapse/api/auth/internal.py +299 -0
- synapse/api/auth/mas.py +457 -0
- synapse/api/auth/msc3861_delegated.py +617 -0
- synapse/api/auth_blocking.py +144 -0
- synapse/api/constants.py +362 -0
- synapse/api/errors.py +907 -0
- synapse/api/filtering.py +539 -0
- synapse/api/presence.py +104 -0
- synapse/api/ratelimiting.py +482 -0
- synapse/api/room_versions.py +535 -0
- synapse/api/urls.py +119 -0
- synapse/app/__init__.py +60 -0
- synapse/app/_base.py +866 -0
- synapse/app/admin_cmd.py +388 -0
- synapse/app/appservice.py +30 -0
- synapse/app/client_reader.py +30 -0
- synapse/app/complement_fork_starter.py +206 -0
- synapse/app/event_creator.py +29 -0
- synapse/app/federation_reader.py +30 -0
- synapse/app/federation_sender.py +30 -0
- synapse/app/frontend_proxy.py +30 -0
- synapse/app/generic_worker.py +475 -0
- synapse/app/homeserver.py +504 -0
- synapse/app/media_repository.py +30 -0
- synapse/app/phone_stats_home.py +296 -0
- synapse/app/pusher.py +30 -0
- synapse/app/synchrotron.py +30 -0
- synapse/app/user_dir.py +31 -0
- synapse/appservice/__init__.py +461 -0
- synapse/appservice/api.py +569 -0
- synapse/appservice/scheduler.py +567 -0
- synapse/config/__init__.py +27 -0
- synapse/config/__main__.py +62 -0
- synapse/config/_base.py +1108 -0
- synapse/config/_base.pyi +217 -0
- synapse/config/_util.py +99 -0
- synapse/config/account_validity.py +116 -0
- synapse/config/api.py +141 -0
- synapse/config/appservice.py +210 -0
- synapse/config/auth.py +80 -0
- synapse/config/auto_accept_invites.py +43 -0
- synapse/config/background_updates.py +44 -0
- synapse/config/cache.py +231 -0
- synapse/config/captcha.py +90 -0
- synapse/config/cas.py +116 -0
- synapse/config/consent.py +73 -0
- synapse/config/database.py +184 -0
- synapse/config/emailconfig.py +367 -0
- synapse/config/experimental.py +595 -0
- synapse/config/federation.py +114 -0
- synapse/config/homeserver.py +141 -0
- synapse/config/jwt.py +55 -0
- synapse/config/key.py +447 -0
- synapse/config/logger.py +390 -0
- synapse/config/mas.py +191 -0
- synapse/config/matrixrtc.py +66 -0
- synapse/config/metrics.py +84 -0
- synapse/config/modules.py +40 -0
- synapse/config/oembed.py +185 -0
- synapse/config/oidc.py +509 -0
- synapse/config/password_auth_providers.py +82 -0
- synapse/config/push.py +64 -0
- synapse/config/ratelimiting.py +254 -0
- synapse/config/redis.py +74 -0
- synapse/config/registration.py +296 -0
- synapse/config/repository.py +311 -0
- synapse/config/retention.py +162 -0
- synapse/config/room.py +88 -0
- synapse/config/room_directory.py +165 -0
- synapse/config/saml2.py +251 -0
- synapse/config/server.py +1170 -0
- synapse/config/server_notices.py +84 -0
- synapse/config/spam_checker.py +66 -0
- synapse/config/sso.py +121 -0
- synapse/config/stats.py +54 -0
- synapse/config/third_party_event_rules.py +40 -0
- synapse/config/tls.py +192 -0
- synapse/config/tracer.py +71 -0
- synapse/config/user_directory.py +47 -0
- synapse/config/user_types.py +44 -0
- synapse/config/voip.py +59 -0
- synapse/config/workers.py +642 -0
- synapse/crypto/__init__.py +20 -0
- synapse/crypto/context_factory.py +278 -0
- synapse/crypto/event_signing.py +194 -0
- synapse/crypto/keyring.py +931 -0
- synapse/event_auth.py +1266 -0
- synapse/events/__init__.py +668 -0
- synapse/events/auto_accept_invites.py +216 -0
- synapse/events/builder.py +387 -0
- synapse/events/presence_router.py +245 -0
- synapse/events/snapshot.py +559 -0
- synapse/events/utils.py +928 -0
- synapse/events/validator.py +305 -0
- synapse/federation/__init__.py +22 -0
- synapse/federation/federation_base.py +383 -0
- synapse/federation/federation_client.py +2134 -0
- synapse/federation/federation_server.py +1544 -0
- synapse/federation/persistence.py +71 -0
- synapse/federation/send_queue.py +532 -0
- synapse/federation/sender/__init__.py +1165 -0
- synapse/federation/sender/per_destination_queue.py +884 -0
- synapse/federation/sender/transaction_manager.py +210 -0
- synapse/federation/transport/__init__.py +28 -0
- synapse/federation/transport/client.py +1201 -0
- synapse/federation/transport/server/__init__.py +334 -0
- synapse/federation/transport/server/_base.py +429 -0
- synapse/federation/transport/server/federation.py +912 -0
- synapse/federation/units.py +133 -0
- synapse/handlers/__init__.py +20 -0
- synapse/handlers/account.py +162 -0
- synapse/handlers/account_data.py +362 -0
- synapse/handlers/account_validity.py +361 -0
- synapse/handlers/admin.py +618 -0
- synapse/handlers/appservice.py +991 -0
- synapse/handlers/auth.py +2494 -0
- synapse/handlers/cas.py +413 -0
- synapse/handlers/deactivate_account.py +363 -0
- synapse/handlers/delayed_events.py +635 -0
- synapse/handlers/device.py +1873 -0
- synapse/handlers/devicemessage.py +399 -0
- synapse/handlers/directory.py +554 -0
- synapse/handlers/e2e_keys.py +1834 -0
- synapse/handlers/e2e_room_keys.py +455 -0
- synapse/handlers/event_auth.py +390 -0
- synapse/handlers/events.py +201 -0
- synapse/handlers/federation.py +2043 -0
- synapse/handlers/federation_event.py +2420 -0
- synapse/handlers/identity.py +812 -0
- synapse/handlers/initial_sync.py +528 -0
- synapse/handlers/jwt.py +120 -0
- synapse/handlers/message.py +2347 -0
- synapse/handlers/oidc.py +1803 -0
- synapse/handlers/pagination.py +768 -0
- synapse/handlers/password_policy.py +102 -0
- synapse/handlers/presence.py +2638 -0
- synapse/handlers/profile.py +655 -0
- synapse/handlers/push_rules.py +164 -0
- synapse/handlers/read_marker.py +79 -0
- synapse/handlers/receipts.py +351 -0
- synapse/handlers/register.py +1060 -0
- synapse/handlers/relations.py +624 -0
- synapse/handlers/reports.py +98 -0
- synapse/handlers/room.py +2447 -0
- synapse/handlers/room_list.py +632 -0
- synapse/handlers/room_member.py +2365 -0
- synapse/handlers/room_member_worker.py +146 -0
- synapse/handlers/room_policy.py +186 -0
- synapse/handlers/room_summary.py +1057 -0
- synapse/handlers/saml.py +524 -0
- synapse/handlers/search.py +723 -0
- synapse/handlers/send_email.py +209 -0
- synapse/handlers/set_password.py +71 -0
- synapse/handlers/sliding_sync/__init__.py +1701 -0
- synapse/handlers/sliding_sync/extensions.py +970 -0
- synapse/handlers/sliding_sync/room_lists.py +2266 -0
- synapse/handlers/sliding_sync/store.py +128 -0
- synapse/handlers/sso.py +1292 -0
- synapse/handlers/state_deltas.py +82 -0
- synapse/handlers/stats.py +322 -0
- synapse/handlers/sync.py +3109 -0
- synapse/handlers/thread_subscriptions.py +190 -0
- synapse/handlers/typing.py +606 -0
- synapse/handlers/ui_auth/__init__.py +48 -0
- synapse/handlers/ui_auth/checkers.py +332 -0
- synapse/handlers/user_directory.py +783 -0
- synapse/handlers/worker_lock.py +365 -0
- synapse/http/__init__.py +106 -0
- synapse/http/additional_resource.py +62 -0
- synapse/http/client.py +1360 -0
- synapse/http/connectproxyclient.py +309 -0
- synapse/http/federation/__init__.py +19 -0
- synapse/http/federation/matrix_federation_agent.py +490 -0
- synapse/http/federation/srv_resolver.py +196 -0
- synapse/http/federation/well_known_resolver.py +367 -0
- synapse/http/matrixfederationclient.py +1875 -0
- synapse/http/proxy.py +290 -0
- synapse/http/proxyagent.py +497 -0
- synapse/http/replicationagent.py +203 -0
- synapse/http/request_metrics.py +309 -0
- synapse/http/server.py +1114 -0
- synapse/http/servlet.py +1019 -0
- synapse/http/site.py +825 -0
- synapse/http/types.py +27 -0
- synapse/logging/__init__.py +31 -0
- synapse/logging/_remote.py +261 -0
- synapse/logging/_terse_json.py +95 -0
- synapse/logging/context.py +1211 -0
- synapse/logging/formatter.py +63 -0
- synapse/logging/handlers.py +99 -0
- synapse/logging/loggers.py +25 -0
- synapse/logging/opentracing.py +1132 -0
- synapse/logging/scopecontextmanager.py +161 -0
- synapse/media/_base.py +827 -0
- synapse/media/filepath.py +417 -0
- synapse/media/media_repository.py +1580 -0
- synapse/media/media_storage.py +704 -0
- synapse/media/oembed.py +277 -0
- synapse/media/preview_html.py +559 -0
- synapse/media/storage_provider.py +195 -0
- synapse/media/thumbnailer.py +833 -0
- synapse/media/url_previewer.py +875 -0
- synapse/metrics/__init__.py +754 -0
- synapse/metrics/_gc.py +219 -0
- synapse/metrics/_reactor_metrics.py +171 -0
- synapse/metrics/_types.py +38 -0
- synapse/metrics/background_process_metrics.py +556 -0
- synapse/metrics/common_usage_metrics.py +94 -0
- synapse/metrics/jemalloc.py +248 -0
- synapse/module_api/__init__.py +2154 -0
- synapse/module_api/callbacks/__init__.py +50 -0
- synapse/module_api/callbacks/account_validity_callbacks.py +106 -0
- synapse/module_api/callbacks/media_repository_callbacks.py +160 -0
- synapse/module_api/callbacks/ratelimit_callbacks.py +79 -0
- synapse/module_api/callbacks/spamchecker_callbacks.py +1113 -0
- synapse/module_api/callbacks/third_party_event_rules_callbacks.py +599 -0
- synapse/module_api/errors.py +42 -0
- synapse/notifier.py +972 -0
- synapse/push/__init__.py +212 -0
- synapse/push/bulk_push_rule_evaluator.py +637 -0
- synapse/push/clientformat.py +126 -0
- synapse/push/emailpusher.py +333 -0
- synapse/push/httppusher.py +564 -0
- synapse/push/mailer.py +1012 -0
- synapse/push/presentable_names.py +216 -0
- synapse/push/push_tools.py +114 -0
- synapse/push/push_types.py +141 -0
- synapse/push/pusher.py +87 -0
- synapse/push/pusherpool.py +501 -0
- synapse/push/rulekinds.py +33 -0
- synapse/py.typed +0 -0
- synapse/replication/__init__.py +20 -0
- synapse/replication/http/__init__.py +68 -0
- synapse/replication/http/_base.py +468 -0
- synapse/replication/http/account_data.py +297 -0
- synapse/replication/http/deactivate_account.py +81 -0
- synapse/replication/http/delayed_events.py +62 -0
- synapse/replication/http/devices.py +254 -0
- synapse/replication/http/federation.py +334 -0
- synapse/replication/http/login.py +106 -0
- synapse/replication/http/membership.py +364 -0
- synapse/replication/http/presence.py +133 -0
- synapse/replication/http/push.py +156 -0
- synapse/replication/http/register.py +172 -0
- synapse/replication/http/send_events.py +182 -0
- synapse/replication/http/state.py +82 -0
- synapse/replication/http/streams.py +101 -0
- synapse/replication/tcp/__init__.py +56 -0
- synapse/replication/tcp/client.py +552 -0
- synapse/replication/tcp/commands.py +569 -0
- synapse/replication/tcp/context.py +41 -0
- synapse/replication/tcp/external_cache.py +156 -0
- synapse/replication/tcp/handler.py +942 -0
- synapse/replication/tcp/protocol.py +608 -0
- synapse/replication/tcp/redis.py +509 -0
- synapse/replication/tcp/resource.py +348 -0
- synapse/replication/tcp/streams/__init__.py +96 -0
- synapse/replication/tcp/streams/_base.py +766 -0
- synapse/replication/tcp/streams/events.py +287 -0
- synapse/replication/tcp/streams/federation.py +92 -0
- synapse/replication/tcp/streams/partial_state.py +80 -0
- synapse/res/providers.json +29 -0
- synapse/res/templates/_base.html +29 -0
- synapse/res/templates/account_previously_renewed.html +6 -0
- synapse/res/templates/account_renewed.html +6 -0
- synapse/res/templates/add_threepid.html +8 -0
- synapse/res/templates/add_threepid.txt +6 -0
- synapse/res/templates/add_threepid_failure.html +7 -0
- synapse/res/templates/add_threepid_success.html +6 -0
- synapse/res/templates/already_in_use.html +12 -0
- synapse/res/templates/already_in_use.txt +10 -0
- synapse/res/templates/auth_success.html +21 -0
- synapse/res/templates/invalid_token.html +6 -0
- synapse/res/templates/mail-Element.css +7 -0
- synapse/res/templates/mail-Vector.css +7 -0
- synapse/res/templates/mail-expiry.css +4 -0
- synapse/res/templates/mail.css +156 -0
- synapse/res/templates/notice_expiry.html +46 -0
- synapse/res/templates/notice_expiry.txt +7 -0
- synapse/res/templates/notif.html +51 -0
- synapse/res/templates/notif.txt +22 -0
- synapse/res/templates/notif_mail.html +59 -0
- synapse/res/templates/notif_mail.txt +10 -0
- synapse/res/templates/password_reset.html +10 -0
- synapse/res/templates/password_reset.txt +7 -0
- synapse/res/templates/password_reset_confirmation.html +15 -0
- synapse/res/templates/password_reset_failure.html +7 -0
- synapse/res/templates/password_reset_success.html +6 -0
- synapse/res/templates/recaptcha.html +42 -0
- synapse/res/templates/registration.html +12 -0
- synapse/res/templates/registration.txt +10 -0
- synapse/res/templates/registration_failure.html +6 -0
- synapse/res/templates/registration_success.html +6 -0
- synapse/res/templates/registration_token.html +18 -0
- synapse/res/templates/room.html +33 -0
- synapse/res/templates/room.txt +9 -0
- synapse/res/templates/sso.css +129 -0
- synapse/res/templates/sso_account_deactivated.html +25 -0
- synapse/res/templates/sso_auth_account_details.html +186 -0
- synapse/res/templates/sso_auth_account_details.js +116 -0
- synapse/res/templates/sso_auth_bad_user.html +26 -0
- synapse/res/templates/sso_auth_confirm.html +27 -0
- synapse/res/templates/sso_auth_success.html +26 -0
- synapse/res/templates/sso_error.html +71 -0
- synapse/res/templates/sso_footer.html +19 -0
- synapse/res/templates/sso_login_idp_picker.html +60 -0
- synapse/res/templates/sso_new_user_consent.html +30 -0
- synapse/res/templates/sso_partial_profile.html +19 -0
- synapse/res/templates/sso_redirect_confirm.html +39 -0
- synapse/res/templates/style.css +33 -0
- synapse/res/templates/terms.html +27 -0
- synapse/rest/__init__.py +197 -0
- synapse/rest/admin/__init__.py +390 -0
- synapse/rest/admin/_base.py +72 -0
- synapse/rest/admin/background_updates.py +171 -0
- synapse/rest/admin/devices.py +221 -0
- synapse/rest/admin/event_reports.py +173 -0
- synapse/rest/admin/events.py +69 -0
- synapse/rest/admin/experimental_features.py +137 -0
- synapse/rest/admin/federation.py +243 -0
- synapse/rest/admin/media.py +540 -0
- synapse/rest/admin/registration_tokens.py +358 -0
- synapse/rest/admin/rooms.py +1061 -0
- synapse/rest/admin/scheduled_tasks.py +70 -0
- synapse/rest/admin/server_notice_servlet.py +132 -0
- synapse/rest/admin/statistics.py +132 -0
- synapse/rest/admin/username_available.py +58 -0
- synapse/rest/admin/users.py +1608 -0
- synapse/rest/client/__init__.py +20 -0
- synapse/rest/client/_base.py +113 -0
- synapse/rest/client/account.py +930 -0
- synapse/rest/client/account_data.py +319 -0
- synapse/rest/client/account_validity.py +103 -0
- synapse/rest/client/appservice_ping.py +125 -0
- synapse/rest/client/auth.py +218 -0
- synapse/rest/client/auth_metadata.py +122 -0
- synapse/rest/client/capabilities.py +121 -0
- synapse/rest/client/delayed_events.py +111 -0
- synapse/rest/client/devices.py +587 -0
- synapse/rest/client/directory.py +211 -0
- synapse/rest/client/events.py +116 -0
- synapse/rest/client/filter.py +112 -0
- synapse/rest/client/initial_sync.py +65 -0
- synapse/rest/client/keys.py +678 -0
- synapse/rest/client/knock.py +104 -0
- synapse/rest/client/login.py +754 -0
- synapse/rest/client/login_token_request.py +127 -0
- synapse/rest/client/logout.py +93 -0
- synapse/rest/client/matrixrtc.py +52 -0
- synapse/rest/client/media.py +286 -0
- synapse/rest/client/mutual_rooms.py +93 -0
- synapse/rest/client/notifications.py +137 -0
- synapse/rest/client/openid.py +109 -0
- synapse/rest/client/password_policy.py +69 -0
- synapse/rest/client/presence.py +131 -0
- synapse/rest/client/profile.py +291 -0
- synapse/rest/client/push_rule.py +331 -0
- synapse/rest/client/pusher.py +181 -0
- synapse/rest/client/read_marker.py +104 -0
- synapse/rest/client/receipts.py +165 -0
- synapse/rest/client/register.py +1067 -0
- synapse/rest/client/relations.py +138 -0
- synapse/rest/client/rendezvous.py +76 -0
- synapse/rest/client/reporting.py +207 -0
- synapse/rest/client/room.py +1669 -0
- synapse/rest/client/room_keys.py +426 -0
- synapse/rest/client/room_upgrade_rest_servlet.py +112 -0
- synapse/rest/client/sendtodevice.py +85 -0
- synapse/rest/client/sync.py +1131 -0
- synapse/rest/client/tags.py +129 -0
- synapse/rest/client/thirdparty.py +130 -0
- synapse/rest/client/thread_subscriptions.py +247 -0
- synapse/rest/client/tokenrefresh.py +52 -0
- synapse/rest/client/transactions.py +149 -0
- synapse/rest/client/user_directory.py +90 -0
- synapse/rest/client/versions.py +191 -0
- synapse/rest/client/voip.py +88 -0
- synapse/rest/consent/__init__.py +0 -0
- synapse/rest/consent/consent_resource.py +210 -0
- synapse/rest/health.py +38 -0
- synapse/rest/key/__init__.py +20 -0
- synapse/rest/key/v2/__init__.py +40 -0
- synapse/rest/key/v2/local_key_resource.py +125 -0
- synapse/rest/key/v2/remote_key_resource.py +302 -0
- synapse/rest/media/__init__.py +0 -0
- synapse/rest/media/config_resource.py +53 -0
- synapse/rest/media/create_resource.py +90 -0
- synapse/rest/media/download_resource.py +110 -0
- synapse/rest/media/media_repository_resource.py +113 -0
- synapse/rest/media/preview_url_resource.py +77 -0
- synapse/rest/media/thumbnail_resource.py +142 -0
- synapse/rest/media/upload_resource.py +187 -0
- synapse/rest/media/v1/__init__.py +39 -0
- synapse/rest/media/v1/_base.py +23 -0
- synapse/rest/media/v1/media_storage.py +23 -0
- synapse/rest/media/v1/storage_provider.py +23 -0
- synapse/rest/synapse/__init__.py +20 -0
- synapse/rest/synapse/client/__init__.py +93 -0
- synapse/rest/synapse/client/federation_whitelist.py +66 -0
- synapse/rest/synapse/client/jwks.py +77 -0
- synapse/rest/synapse/client/new_user_consent.py +115 -0
- synapse/rest/synapse/client/oidc/__init__.py +45 -0
- synapse/rest/synapse/client/oidc/backchannel_logout_resource.py +42 -0
- synapse/rest/synapse/client/oidc/callback_resource.py +48 -0
- synapse/rest/synapse/client/password_reset.py +129 -0
- synapse/rest/synapse/client/pick_idp.py +107 -0
- synapse/rest/synapse/client/pick_username.py +153 -0
- synapse/rest/synapse/client/rendezvous.py +58 -0
- synapse/rest/synapse/client/saml2/__init__.py +42 -0
- synapse/rest/synapse/client/saml2/metadata_resource.py +46 -0
- synapse/rest/synapse/client/saml2/response_resource.py +52 -0
- synapse/rest/synapse/client/sso_register.py +56 -0
- synapse/rest/synapse/client/unsubscribe.py +88 -0
- synapse/rest/synapse/mas/__init__.py +71 -0
- synapse/rest/synapse/mas/_base.py +55 -0
- synapse/rest/synapse/mas/devices.py +239 -0
- synapse/rest/synapse/mas/users.py +469 -0
- synapse/rest/well_known.py +148 -0
- synapse/server.py +1258 -0
- synapse/server_notices/__init__.py +0 -0
- synapse/server_notices/consent_server_notices.py +136 -0
- synapse/server_notices/resource_limits_server_notices.py +215 -0
- synapse/server_notices/server_notices_manager.py +388 -0
- synapse/server_notices/server_notices_sender.py +67 -0
- synapse/server_notices/worker_server_notices_sender.py +46 -0
- synapse/spam_checker_api/__init__.py +31 -0
- synapse/state/__init__.py +1022 -0
- synapse/state/v1.py +370 -0
- synapse/state/v2.py +985 -0
- synapse/static/client/login/index.html +47 -0
- synapse/static/client/login/js/jquery-3.4.1.min.js +2 -0
- synapse/static/client/login/js/login.js +291 -0
- synapse/static/client/login/spinner.gif +0 -0
- synapse/static/client/login/style.css +79 -0
- synapse/static/index.html +63 -0
- synapse/storage/__init__.py +43 -0
- synapse/storage/_base.py +245 -0
- synapse/storage/admin_client_config.py +26 -0
- synapse/storage/background_updates.py +1189 -0
- synapse/storage/controllers/__init__.py +57 -0
- synapse/storage/controllers/persist_events.py +1239 -0
- synapse/storage/controllers/purge_events.py +456 -0
- synapse/storage/controllers/state.py +954 -0
- synapse/storage/controllers/stats.py +119 -0
- synapse/storage/database.py +2720 -0
- synapse/storage/databases/__init__.py +175 -0
- synapse/storage/databases/main/__init__.py +424 -0
- synapse/storage/databases/main/account_data.py +1060 -0
- synapse/storage/databases/main/appservice.py +473 -0
- synapse/storage/databases/main/cache.py +911 -0
- synapse/storage/databases/main/censor_events.py +225 -0
- synapse/storage/databases/main/client_ips.py +817 -0
- synapse/storage/databases/main/delayed_events.py +560 -0
- synapse/storage/databases/main/deviceinbox.py +1272 -0
- synapse/storage/databases/main/devices.py +2581 -0
- synapse/storage/databases/main/directory.py +212 -0
- synapse/storage/databases/main/e2e_room_keys.py +690 -0
- synapse/storage/databases/main/end_to_end_keys.py +1896 -0
- synapse/storage/databases/main/event_federation.py +2509 -0
- synapse/storage/databases/main/event_push_actions.py +1937 -0
- synapse/storage/databases/main/events.py +3746 -0
- synapse/storage/databases/main/events_bg_updates.py +2910 -0
- synapse/storage/databases/main/events_forward_extremities.py +126 -0
- synapse/storage/databases/main/events_worker.py +2784 -0
- synapse/storage/databases/main/experimental_features.py +130 -0
- synapse/storage/databases/main/filtering.py +231 -0
- synapse/storage/databases/main/keys.py +291 -0
- synapse/storage/databases/main/lock.py +553 -0
- synapse/storage/databases/main/media_repository.py +1070 -0
- synapse/storage/databases/main/metrics.py +460 -0
- synapse/storage/databases/main/monthly_active_users.py +443 -0
- synapse/storage/databases/main/openid.py +61 -0
- synapse/storage/databases/main/presence.py +511 -0
- synapse/storage/databases/main/profile.py +541 -0
- synapse/storage/databases/main/purge_events.py +511 -0
- synapse/storage/databases/main/push_rule.py +972 -0
- synapse/storage/databases/main/pusher.py +794 -0
- synapse/storage/databases/main/receipts.py +1342 -0
- synapse/storage/databases/main/registration.py +3076 -0
- synapse/storage/databases/main/rejections.py +38 -0
- synapse/storage/databases/main/relations.py +1118 -0
- synapse/storage/databases/main/room.py +2781 -0
- synapse/storage/databases/main/roommember.py +2112 -0
- synapse/storage/databases/main/search.py +941 -0
- synapse/storage/databases/main/session.py +151 -0
- synapse/storage/databases/main/signatures.py +94 -0
- synapse/storage/databases/main/sliding_sync.py +603 -0
- synapse/storage/databases/main/state.py +1006 -0
- synapse/storage/databases/main/state_deltas.py +329 -0
- synapse/storage/databases/main/stats.py +791 -0
- synapse/storage/databases/main/stream.py +2580 -0
- synapse/storage/databases/main/tags.py +360 -0
- synapse/storage/databases/main/task_scheduler.py +225 -0
- synapse/storage/databases/main/thread_subscriptions.py +591 -0
- synapse/storage/databases/main/transactions.py +681 -0
- synapse/storage/databases/main/ui_auth.py +420 -0
- synapse/storage/databases/main/user_directory.py +1331 -0
- synapse/storage/databases/main/user_erasure_store.py +117 -0
- synapse/storage/databases/state/__init__.py +22 -0
- synapse/storage/databases/state/bg_updates.py +499 -0
- synapse/storage/databases/state/deletion.py +558 -0
- synapse/storage/databases/state/store.py +949 -0
- synapse/storage/engines/__init__.py +70 -0
- synapse/storage/engines/_base.py +154 -0
- synapse/storage/engines/postgres.py +261 -0
- synapse/storage/engines/sqlite.py +199 -0
- synapse/storage/invite_rule.py +112 -0
- synapse/storage/keys.py +40 -0
- synapse/storage/prepare_database.py +731 -0
- synapse/storage/push_rule.py +28 -0
- synapse/storage/roommember.py +89 -0
- synapse/storage/schema/README.md +4 -0
- synapse/storage/schema/__init__.py +182 -0
- synapse/storage/schema/common/delta/25/00background_updates.sql +40 -0
- synapse/storage/schema/common/delta/35/00background_updates_add_col.sql +36 -0
- synapse/storage/schema/common/delta/58/00background_update_ordering.sql +38 -0
- synapse/storage/schema/common/full_schemas/72/full.sql.postgres +8 -0
- synapse/storage/schema/common/full_schemas/72/full.sql.sqlite +6 -0
- synapse/storage/schema/common/schema_version.sql +60 -0
- synapse/storage/schema/main/delta/12/v12.sql +82 -0
- synapse/storage/schema/main/delta/13/v13.sql +38 -0
- synapse/storage/schema/main/delta/14/v14.sql +42 -0
- synapse/storage/schema/main/delta/15/appservice_txns.sql +50 -0
- synapse/storage/schema/main/delta/15/presence_indices.sql +2 -0
- synapse/storage/schema/main/delta/15/v15.sql +24 -0
- synapse/storage/schema/main/delta/16/events_order_index.sql +4 -0
- synapse/storage/schema/main/delta/16/remote_media_cache_index.sql +2 -0
- synapse/storage/schema/main/delta/16/remove_duplicates.sql +9 -0
- synapse/storage/schema/main/delta/16/room_alias_index.sql +3 -0
- synapse/storage/schema/main/delta/16/unique_constraints.sql +72 -0
- synapse/storage/schema/main/delta/16/users.sql +56 -0
- synapse/storage/schema/main/delta/17/drop_indexes.sql +37 -0
- synapse/storage/schema/main/delta/17/server_keys.sql +43 -0
- synapse/storage/schema/main/delta/17/user_threepids.sql +9 -0
- synapse/storage/schema/main/delta/18/server_keys_bigger_ints.sql +51 -0
- synapse/storage/schema/main/delta/19/event_index.sql +38 -0
- synapse/storage/schema/main/delta/20/dummy.sql +1 -0
- synapse/storage/schema/main/delta/20/pushers.py +93 -0
- synapse/storage/schema/main/delta/21/end_to_end_keys.sql +53 -0
- synapse/storage/schema/main/delta/21/receipts.sql +57 -0
- synapse/storage/schema/main/delta/22/receipts_index.sql +41 -0
- synapse/storage/schema/main/delta/22/user_threepids_unique.sql +19 -0
- synapse/storage/schema/main/delta/24/stats_reporting.sql +37 -0
- synapse/storage/schema/main/delta/25/fts.py +81 -0
- synapse/storage/schema/main/delta/25/guest_access.sql +44 -0
- synapse/storage/schema/main/delta/25/history_visibility.sql +44 -0
- synapse/storage/schema/main/delta/25/tags.sql +57 -0
- synapse/storage/schema/main/delta/26/account_data.sql +36 -0
- synapse/storage/schema/main/delta/27/account_data.sql +55 -0
- synapse/storage/schema/main/delta/27/forgotten_memberships.sql +45 -0
- synapse/storage/schema/main/delta/27/ts.py +61 -0
- synapse/storage/schema/main/delta/28/event_push_actions.sql +46 -0
- synapse/storage/schema/main/delta/28/events_room_stream.sql +39 -0
- synapse/storage/schema/main/delta/28/public_roms_index.sql +39 -0
- synapse/storage/schema/main/delta/28/receipts_user_id_index.sql +41 -0
- synapse/storage/schema/main/delta/28/upgrade_times.sql +40 -0
- synapse/storage/schema/main/delta/28/users_is_guest.sql +41 -0
- synapse/storage/schema/main/delta/29/push_actions.sql +54 -0
- synapse/storage/schema/main/delta/30/alias_creator.sql +35 -0
- synapse/storage/schema/main/delta/30/as_users.py +82 -0
- synapse/storage/schema/main/delta/30/deleted_pushers.sql +44 -0
- synapse/storage/schema/main/delta/30/presence_stream.sql +49 -0
- synapse/storage/schema/main/delta/30/public_rooms.sql +42 -0
- synapse/storage/schema/main/delta/30/push_rule_stream.sql +57 -0
- synapse/storage/schema/main/delta/30/threepid_guest_access_tokens.sql +43 -0
- synapse/storage/schema/main/delta/31/invites.sql +61 -0
- synapse/storage/schema/main/delta/31/local_media_repository_url_cache.sql +46 -0
- synapse/storage/schema/main/delta/31/pushers_0.py +92 -0
- synapse/storage/schema/main/delta/31/pushers_index.sql +41 -0
- synapse/storage/schema/main/delta/31/search_update.py +65 -0
- synapse/storage/schema/main/delta/32/events.sql +35 -0
- synapse/storage/schema/main/delta/32/openid.sql +9 -0
- synapse/storage/schema/main/delta/32/pusher_throttle.sql +42 -0
- synapse/storage/schema/main/delta/32/remove_indices.sql +52 -0
- synapse/storage/schema/main/delta/32/reports.sql +44 -0
- synapse/storage/schema/main/delta/33/access_tokens_device_index.sql +36 -0
- synapse/storage/schema/main/delta/33/devices.sql +40 -0
- synapse/storage/schema/main/delta/33/devices_for_e2e_keys.sql +38 -0
- synapse/storage/schema/main/delta/33/devices_for_e2e_keys_clear_unknown_device.sql +39 -0
- synapse/storage/schema/main/delta/33/event_fields.py +61 -0
- synapse/storage/schema/main/delta/33/remote_media_ts.py +43 -0
- synapse/storage/schema/main/delta/33/user_ips_index.sql +36 -0
- synapse/storage/schema/main/delta/34/appservice_stream.sql +42 -0
- synapse/storage/schema/main/delta/34/cache_stream.py +50 -0
- synapse/storage/schema/main/delta/34/device_inbox.sql +43 -0
- synapse/storage/schema/main/delta/34/push_display_name_rename.sql +39 -0
- synapse/storage/schema/main/delta/34/received_txn_purge.py +36 -0
- synapse/storage/schema/main/delta/35/contains_url.sql +36 -0
- synapse/storage/schema/main/delta/35/device_outbox.sql +58 -0
- synapse/storage/schema/main/delta/35/device_stream_id.sql +40 -0
- synapse/storage/schema/main/delta/35/event_push_actions_index.sql +36 -0
- synapse/storage/schema/main/delta/35/public_room_list_change_stream.sql +52 -0
- synapse/storage/schema/main/delta/35/stream_order_to_extrem.sql +56 -0
- synapse/storage/schema/main/delta/36/readd_public_rooms.sql +45 -0
- synapse/storage/schema/main/delta/37/remove_auth_idx.py +89 -0
- synapse/storage/schema/main/delta/37/user_threepids.sql +71 -0
- synapse/storage/schema/main/delta/38/postgres_fts_gist.sql +38 -0
- synapse/storage/schema/main/delta/39/appservice_room_list.sql +48 -0
- synapse/storage/schema/main/delta/39/device_federation_stream_idx.sql +35 -0
- synapse/storage/schema/main/delta/39/event_push_index.sql +36 -0
- synapse/storage/schema/main/delta/39/federation_out_position.sql +41 -0
- synapse/storage/schema/main/delta/39/membership_profile.sql +39 -0
- synapse/storage/schema/main/delta/40/current_state_idx.sql +36 -0
- synapse/storage/schema/main/delta/40/device_inbox.sql +40 -0
- synapse/storage/schema/main/delta/40/device_list_streams.sql +79 -0
- synapse/storage/schema/main/delta/40/event_push_summary.sql +57 -0
- synapse/storage/schema/main/delta/40/pushers.sql +58 -0
- synapse/storage/schema/main/delta/41/device_list_stream_idx.sql +36 -0
- synapse/storage/schema/main/delta/41/device_outbound_index.sql +35 -0
- synapse/storage/schema/main/delta/41/event_search_event_id_idx.sql +36 -0
- synapse/storage/schema/main/delta/41/ratelimit.sql +41 -0
- synapse/storage/schema/main/delta/42/current_state_delta.sql +48 -0
- synapse/storage/schema/main/delta/42/device_list_last_id.sql +52 -0
- synapse/storage/schema/main/delta/42/event_auth_state_only.sql +36 -0
- synapse/storage/schema/main/delta/42/user_dir.py +88 -0
- synapse/storage/schema/main/delta/43/blocked_rooms.sql +40 -0
- synapse/storage/schema/main/delta/43/quarantine_media.sql +36 -0
- synapse/storage/schema/main/delta/43/url_cache.sql +35 -0
- synapse/storage/schema/main/delta/43/user_share.sql +52 -0
- synapse/storage/schema/main/delta/44/expire_url_cache.sql +60 -0
- synapse/storage/schema/main/delta/45/group_server.sql +186 -0
- synapse/storage/schema/main/delta/45/profile_cache.sql +47 -0
- synapse/storage/schema/main/delta/46/drop_refresh_tokens.sql +36 -0
- synapse/storage/schema/main/delta/46/drop_unique_deleted_pushers.sql +54 -0
- synapse/storage/schema/main/delta/46/group_server.sql +51 -0
- synapse/storage/schema/main/delta/46/local_media_repository_url_idx.sql +43 -0
- synapse/storage/schema/main/delta/46/user_dir_null_room_ids.sql +54 -0
- synapse/storage/schema/main/delta/46/user_dir_typos.sql +43 -0
- synapse/storage/schema/main/delta/47/last_access_media.sql +35 -0
- synapse/storage/schema/main/delta/47/postgres_fts_gin.sql +36 -0
- synapse/storage/schema/main/delta/47/push_actions_staging.sql +47 -0
- synapse/storage/schema/main/delta/48/add_user_consent.sql +37 -0
- synapse/storage/schema/main/delta/48/add_user_ips_last_seen_index.sql +36 -0
- synapse/storage/schema/main/delta/48/deactivated_users.sql +44 -0
- synapse/storage/schema/main/delta/48/group_unique_indexes.py +67 -0
- synapse/storage/schema/main/delta/48/groups_joinable.sql +41 -0
- synapse/storage/schema/main/delta/49/add_user_consent_server_notice_sent.sql +39 -0
- synapse/storage/schema/main/delta/49/add_user_daily_visits.sql +40 -0
- synapse/storage/schema/main/delta/49/add_user_ips_last_seen_only_index.sql +36 -0
- synapse/storage/schema/main/delta/50/add_creation_ts_users_index.sql +38 -0
- synapse/storage/schema/main/delta/50/erasure_store.sql +40 -0
- synapse/storage/schema/main/delta/50/make_event_content_nullable.py +102 -0
- synapse/storage/schema/main/delta/51/e2e_room_keys.sql +58 -0
- synapse/storage/schema/main/delta/51/monthly_active_users.sql +46 -0
- synapse/storage/schema/main/delta/52/add_event_to_state_group_index.sql +38 -0
- synapse/storage/schema/main/delta/52/device_list_streams_unique_idx.sql +55 -0
- synapse/storage/schema/main/delta/52/e2e_room_keys.sql +72 -0
- synapse/storage/schema/main/delta/53/add_user_type_to_users.sql +38 -0
- synapse/storage/schema/main/delta/53/drop_sent_transactions.sql +35 -0
- synapse/storage/schema/main/delta/53/event_format_version.sql +35 -0
- synapse/storage/schema/main/delta/53/user_dir_populate.sql +49 -0
- synapse/storage/schema/main/delta/53/user_ips_index.sql +49 -0
- synapse/storage/schema/main/delta/53/user_share.sql +63 -0
- synapse/storage/schema/main/delta/53/user_threepid_id.sql +48 -0
- synapse/storage/schema/main/delta/53/users_in_public_rooms.sql +47 -0
- synapse/storage/schema/main/delta/54/account_validity_with_renewal.sql +49 -0
- synapse/storage/schema/main/delta/54/add_validity_to_server_keys.sql +42 -0
- synapse/storage/schema/main/delta/54/delete_forward_extremities.sql +42 -0
- synapse/storage/schema/main/delta/54/drop_legacy_tables.sql +49 -0
- synapse/storage/schema/main/delta/54/drop_presence_list.sql +35 -0
- synapse/storage/schema/main/delta/54/relations.sql +46 -0
- synapse/storage/schema/main/delta/54/stats.sql +99 -0
- synapse/storage/schema/main/delta/54/stats2.sql +47 -0
- synapse/storage/schema/main/delta/55/access_token_expiry.sql +37 -0
- synapse/storage/schema/main/delta/55/track_threepid_validations.sql +50 -0
- synapse/storage/schema/main/delta/55/users_alter_deactivated.sql +38 -0
- synapse/storage/schema/main/delta/56/add_spans_to_device_lists.sql +39 -0
- synapse/storage/schema/main/delta/56/current_state_events_membership.sql +41 -0
- synapse/storage/schema/main/delta/56/current_state_events_membership_mk2.sql +43 -0
- synapse/storage/schema/main/delta/56/delete_keys_from_deleted_backups.sql +44 -0
- synapse/storage/schema/main/delta/56/destinations_failure_ts.sql +44 -0
- synapse/storage/schema/main/delta/56/destinations_retry_interval_type.sql.postgres +18 -0
- synapse/storage/schema/main/delta/56/device_stream_id_insert.sql +39 -0
- synapse/storage/schema/main/delta/56/devices_last_seen.sql +43 -0
- synapse/storage/schema/main/delta/56/drop_unused_event_tables.sql +39 -0
- synapse/storage/schema/main/delta/56/event_expiry.sql +40 -0
- synapse/storage/schema/main/delta/56/event_labels.sql +49 -0
- synapse/storage/schema/main/delta/56/event_labels_background_update.sql +36 -0
- synapse/storage/schema/main/delta/56/fix_room_keys_index.sql +37 -0
- synapse/storage/schema/main/delta/56/hidden_devices.sql +37 -0
- synapse/storage/schema/main/delta/56/hidden_devices_fix.sql.sqlite +42 -0
- synapse/storage/schema/main/delta/56/nuke_empty_communities_from_db.sql +48 -0
- synapse/storage/schema/main/delta/56/public_room_list_idx.sql +35 -0
- synapse/storage/schema/main/delta/56/redaction_censor.sql +35 -0
- synapse/storage/schema/main/delta/56/redaction_censor2.sql +41 -0
- synapse/storage/schema/main/delta/56/redaction_censor3_fix_update.sql.postgres +25 -0
- synapse/storage/schema/main/delta/56/redaction_censor4.sql +35 -0
- synapse/storage/schema/main/delta/56/remove_tombstoned_rooms_from_directory.sql +38 -0
- synapse/storage/schema/main/delta/56/room_key_etag.sql +36 -0
- synapse/storage/schema/main/delta/56/room_membership_idx.sql +37 -0
- synapse/storage/schema/main/delta/56/room_retention.sql +52 -0
- synapse/storage/schema/main/delta/56/signing_keys.sql +75 -0
- synapse/storage/schema/main/delta/56/signing_keys_nonunique_signatures.sql +41 -0
- synapse/storage/schema/main/delta/56/stats_separated.sql +175 -0
- synapse/storage/schema/main/delta/56/unique_user_filter_index.py +46 -0
- synapse/storage/schema/main/delta/56/user_external_ids.sql +43 -0
- synapse/storage/schema/main/delta/56/users_in_public_rooms_idx.sql +36 -0
- synapse/storage/schema/main/delta/57/delete_old_current_state_events.sql +41 -0
- synapse/storage/schema/main/delta/57/device_list_remote_cache_stale.sql +44 -0
- synapse/storage/schema/main/delta/57/local_current_membership.py +111 -0
- synapse/storage/schema/main/delta/57/remove_sent_outbound_pokes.sql +40 -0
- synapse/storage/schema/main/delta/57/rooms_version_column.sql +43 -0
- synapse/storage/schema/main/delta/57/rooms_version_column_2.sql.postgres +35 -0
- synapse/storage/schema/main/delta/57/rooms_version_column_2.sql.sqlite +22 -0
- synapse/storage/schema/main/delta/57/rooms_version_column_3.sql.postgres +39 -0
- synapse/storage/schema/main/delta/57/rooms_version_column_3.sql.sqlite +23 -0
- synapse/storage/schema/main/delta/58/02remove_dup_outbound_pokes.sql +41 -0
- synapse/storage/schema/main/delta/58/03persist_ui_auth.sql +55 -0
- synapse/storage/schema/main/delta/58/05cache_instance.sql.postgres +30 -0
- synapse/storage/schema/main/delta/58/06dlols_unique_idx.py +83 -0
- synapse/storage/schema/main/delta/58/07add_method_to_thumbnail_constraint.sql.postgres +33 -0
- synapse/storage/schema/main/delta/58/07add_method_to_thumbnail_constraint.sql.sqlite +44 -0
- synapse/storage/schema/main/delta/58/07persist_ui_auth_ips.sql +44 -0
- synapse/storage/schema/main/delta/58/08_media_safe_from_quarantine.sql.postgres +18 -0
- synapse/storage/schema/main/delta/58/08_media_safe_from_quarantine.sql.sqlite +18 -0
- synapse/storage/schema/main/delta/58/09shadow_ban.sql +37 -0
- synapse/storage/schema/main/delta/58/10_pushrules_enabled_delete_obsolete.sql +47 -0
- synapse/storage/schema/main/delta/58/10drop_local_rejections_stream.sql +41 -0
- synapse/storage/schema/main/delta/58/10federation_pos_instance_name.sql +41 -0
- synapse/storage/schema/main/delta/58/11dehydration.sql +39 -0
- synapse/storage/schema/main/delta/58/11fallback.sql +43 -0
- synapse/storage/schema/main/delta/58/11user_id_seq.py +38 -0
- synapse/storage/schema/main/delta/58/12room_stats.sql +51 -0
- synapse/storage/schema/main/delta/58/13remove_presence_allow_inbound.sql +36 -0
- synapse/storage/schema/main/delta/58/14events_instance_name.sql +35 -0
- synapse/storage/schema/main/delta/58/14events_instance_name.sql.postgres +28 -0
- synapse/storage/schema/main/delta/58/15_catchup_destination_rooms.sql +61 -0
- synapse/storage/schema/main/delta/58/15unread_count.sql +45 -0
- synapse/storage/schema/main/delta/58/16populate_stats_process_rooms_fix.sql +41 -0
- synapse/storage/schema/main/delta/58/17_catchup_last_successful.sql +40 -0
- synapse/storage/schema/main/delta/58/18stream_positions.sql +41 -0
- synapse/storage/schema/main/delta/58/19instance_map.sql.postgres +25 -0
- synapse/storage/schema/main/delta/58/19txn_id.sql +59 -0
- synapse/storage/schema/main/delta/58/20instance_name_event_tables.sql +36 -0
- synapse/storage/schema/main/delta/58/20user_daily_visits.sql +37 -0
- synapse/storage/schema/main/delta/58/21as_device_stream.sql +36 -0
- synapse/storage/schema/main/delta/58/21drop_device_max_stream_id.sql +1 -0
- synapse/storage/schema/main/delta/58/22puppet_token.sql +36 -0
- synapse/storage/schema/main/delta/58/22users_have_local_media.sql +2 -0
- synapse/storage/schema/main/delta/58/23e2e_cross_signing_keys_idx.sql +36 -0
- synapse/storage/schema/main/delta/58/24drop_event_json_index.sql +38 -0
- synapse/storage/schema/main/delta/58/25user_external_ids_user_id_idx.sql +36 -0
- synapse/storage/schema/main/delta/58/26access_token_last_validated.sql +37 -0
- synapse/storage/schema/main/delta/58/27local_invites.sql +37 -0
- synapse/storage/schema/main/delta/58/28drop_last_used_column.sql.postgres +16 -0
- synapse/storage/schema/main/delta/58/28drop_last_used_column.sql.sqlite +62 -0
- synapse/storage/schema/main/delta/59/01ignored_user.py +85 -0
- synapse/storage/schema/main/delta/59/02shard_send_to_device.sql +37 -0
- synapse/storage/schema/main/delta/59/03shard_send_to_device_sequence.sql.postgres +25 -0
- synapse/storage/schema/main/delta/59/04_event_auth_chains.sql +71 -0
- synapse/storage/schema/main/delta/59/04_event_auth_chains.sql.postgres +16 -0
- synapse/storage/schema/main/delta/59/04drop_account_data.sql +36 -0
- synapse/storage/schema/main/delta/59/05cache_invalidation.sql +36 -0
- synapse/storage/schema/main/delta/59/06chain_cover_index.sql +36 -0
- synapse/storage/schema/main/delta/59/06shard_account_data.sql +39 -0
- synapse/storage/schema/main/delta/59/06shard_account_data.sql.postgres +32 -0
- synapse/storage/schema/main/delta/59/07shard_account_data_fix.sql +37 -0
- synapse/storage/schema/main/delta/59/08delete_pushers_for_deactivated_accounts.sql +39 -0
- synapse/storage/schema/main/delta/59/08delete_stale_pushers.sql +39 -0
- synapse/storage/schema/main/delta/59/09rejected_events_metadata.sql +45 -0
- synapse/storage/schema/main/delta/59/10delete_purged_chain_cover.sql +36 -0
- synapse/storage/schema/main/delta/59/11add_knock_members_to_stats.sql +39 -0
- synapse/storage/schema/main/delta/59/11drop_thumbnail_constraint.sql.postgres +22 -0
- synapse/storage/schema/main/delta/59/12account_validity_token_used_ts_ms.sql +37 -0
- synapse/storage/schema/main/delta/59/12presence_stream_instance.sql +37 -0
- synapse/storage/schema/main/delta/59/12presence_stream_instance_seq.sql.postgres +20 -0
- synapse/storage/schema/main/delta/59/13users_to_send_full_presence_to.sql +53 -0
- synapse/storage/schema/main/delta/59/14refresh_tokens.sql +53 -0
- synapse/storage/schema/main/delta/59/15locks.sql +56 -0
- synapse/storage/schema/main/delta/59/16federation_inbound_staging.sql +51 -0
- synapse/storage/schema/main/delta/60/01recreate_stream_ordering.sql.postgres +45 -0
- synapse/storage/schema/main/delta/60/02change_stream_ordering_columns.sql.postgres +30 -0
- synapse/storage/schema/main/delta/61/01change_appservices_txns.sql.postgres +23 -0
- synapse/storage/schema/main/delta/61/01insertion_event_lookups.sql +68 -0
- synapse/storage/schema/main/delta/61/02drop_redundant_room_depth_index.sql +37 -0
- synapse/storage/schema/main/delta/61/03recreate_min_depth.py +74 -0
- synapse/storage/schema/main/delta/62/01insertion_event_extremities.sql +43 -0
- synapse/storage/schema/main/delta/63/01create_registration_tokens.sql +42 -0
- synapse/storage/schema/main/delta/63/02delete_unlinked_email_pushers.sql +39 -0
- synapse/storage/schema/main/delta/63/02populate-rooms-creator.sql +36 -0
- synapse/storage/schema/main/delta/63/03session_store.sql +42 -0
- synapse/storage/schema/main/delta/63/04add_presence_stream_not_offline_index.sql +37 -0
- synapse/storage/schema/main/delta/64/01msc2716_chunk_to_batch_rename.sql.postgres +23 -0
- synapse/storage/schema/main/delta/64/01msc2716_chunk_to_batch_rename.sql.sqlite +37 -0
- synapse/storage/schema/main/delta/65/01msc2716_insertion_event_edges.sql +38 -0
- synapse/storage/schema/main/delta/65/03remove_hidden_devices_from_device_inbox.sql +41 -0
- synapse/storage/schema/main/delta/65/04_local_group_updates.sql +37 -0
- synapse/storage/schema/main/delta/65/05_remove_room_stats_historical_and_user_stats_historical.sql +38 -0
- synapse/storage/schema/main/delta/65/06remove_deleted_devices_from_device_inbox.sql +53 -0
- synapse/storage/schema/main/delta/65/07_arbitrary_relations.sql +37 -0
- synapse/storage/schema/main/delta/65/08_device_inbox_background_updates.sql +37 -0
- synapse/storage/schema/main/delta/65/10_expirable_refresh_tokens.sql +47 -0
- synapse/storage/schema/main/delta/65/11_devices_auth_provider_session.sql +46 -0
- synapse/storage/schema/main/delta/67/01drop_public_room_list_stream.sql +37 -0
- synapse/storage/schema/main/delta/68/01event_columns.sql +45 -0
- synapse/storage/schema/main/delta/68/02_msc2409_add_device_id_appservice_stream_type.sql +40 -0
- synapse/storage/schema/main/delta/68/03_delete_account_data_for_deactivated_accounts.sql +39 -0
- synapse/storage/schema/main/delta/68/04_refresh_tokens_index_next_token_id.sql +47 -0
- synapse/storage/schema/main/delta/68/04partial_state_rooms.sql +60 -0
- synapse/storage/schema/main/delta/68/05_delete_non_strings_from_event_search.sql.sqlite +22 -0
- synapse/storage/schema/main/delta/68/05partial_state_rooms_triggers.py +80 -0
- synapse/storage/schema/main/delta/68/06_msc3202_add_device_list_appservice_stream_type.sql +42 -0
- synapse/storage/schema/main/delta/69/01as_txn_seq.py +54 -0
- synapse/storage/schema/main/delta/69/01device_list_oubound_by_room.sql +57 -0
- synapse/storage/schema/main/delta/69/02cache_invalidation_index.sql +37 -0
- synapse/storage/schema/main/delta/70/01clean_table_purged_rooms.sql +39 -0
- synapse/storage/schema/main/delta/71/01rebuild_event_edges.sql.postgres +43 -0
- synapse/storage/schema/main/delta/71/01rebuild_event_edges.sql.sqlite +47 -0
- synapse/storage/schema/main/delta/71/01remove_noop_background_updates.sql +80 -0
- synapse/storage/schema/main/delta/71/02event_push_summary_unique.sql +37 -0
- synapse/storage/schema/main/delta/72/01add_room_type_to_state_stats.sql +38 -0
- synapse/storage/schema/main/delta/72/01event_push_summary_receipt.sql +54 -0
- synapse/storage/schema/main/delta/72/02event_push_actions_index.sql +38 -0
- synapse/storage/schema/main/delta/72/03bg_populate_events_columns.py +57 -0
- synapse/storage/schema/main/delta/72/03drop_event_reference_hashes.sql +36 -0
- synapse/storage/schema/main/delta/72/03remove_groups.sql +50 -0
- synapse/storage/schema/main/delta/72/04drop_column_application_services_state_last_txn.sql.postgres +17 -0
- synapse/storage/schema/main/delta/72/04drop_column_application_services_state_last_txn.sql.sqlite +40 -0
- synapse/storage/schema/main/delta/72/05receipts_event_stream_ordering.sql +38 -0
- synapse/storage/schema/main/delta/72/05remove_unstable_private_read_receipts.sql +38 -0
- synapse/storage/schema/main/delta/72/06add_consent_ts_to_users.sql +35 -0
- synapse/storage/schema/main/delta/72/06thread_notifications.sql +49 -0
- synapse/storage/schema/main/delta/72/07force_update_current_state_events_membership.py +67 -0
- synapse/storage/schema/main/delta/72/07thread_receipts.sql.postgres +30 -0
- synapse/storage/schema/main/delta/72/07thread_receipts.sql.sqlite +70 -0
- synapse/storage/schema/main/delta/72/08begin_cache_invalidation_seq_at_2.sql.postgres +23 -0
- synapse/storage/schema/main/delta/72/08thread_receipts.sql +39 -0
- synapse/storage/schema/main/delta/72/09partial_indices.sql.sqlite +56 -0
- synapse/storage/schema/main/delta/73/01event_failed_pull_attempts.sql +48 -0
- synapse/storage/schema/main/delta/73/02add_pusher_enabled.sql +35 -0
- synapse/storage/schema/main/delta/73/02room_id_indexes_for_purging.sql +41 -0
- synapse/storage/schema/main/delta/73/03pusher_device_id.sql +39 -0
- synapse/storage/schema/main/delta/73/03users_approved_column.sql +39 -0
- synapse/storage/schema/main/delta/73/04partial_join_details.sql +42 -0
- synapse/storage/schema/main/delta/73/04pending_device_list_updates.sql +47 -0
- synapse/storage/schema/main/delta/73/05old_push_actions.sql.postgres +22 -0
- synapse/storage/schema/main/delta/73/05old_push_actions.sql.sqlite +24 -0
- synapse/storage/schema/main/delta/73/06thread_notifications_thread_id_idx.sql +42 -0
- synapse/storage/schema/main/delta/73/08thread_receipts_non_null.sql.postgres +23 -0
- synapse/storage/schema/main/delta/73/08thread_receipts_non_null.sql.sqlite +76 -0
- synapse/storage/schema/main/delta/73/09partial_joined_via_destination.sql +37 -0
- synapse/storage/schema/main/delta/73/09threads_table.sql +49 -0
- synapse/storage/schema/main/delta/73/10_update_sqlite_fts4_tokenizer.py +71 -0
- synapse/storage/schema/main/delta/73/10login_tokens.sql +54 -0
- synapse/storage/schema/main/delta/73/11event_search_room_id_n_distinct.sql.postgres +33 -0
- synapse/storage/schema/main/delta/73/12refactor_device_list_outbound_pokes.sql +72 -0
- synapse/storage/schema/main/delta/73/13add_device_lists_index.sql +39 -0
- synapse/storage/schema/main/delta/73/20_un_partial_stated_room_stream.sql +51 -0
- synapse/storage/schema/main/delta/73/21_un_partial_stated_room_stream_seq.sql.postgres +20 -0
- synapse/storage/schema/main/delta/73/22_rebuild_user_dir_stats.sql +48 -0
- synapse/storage/schema/main/delta/73/22_un_partial_stated_event_stream.sql +53 -0
- synapse/storage/schema/main/delta/73/23_fix_thread_index.sql +52 -0
- synapse/storage/schema/main/delta/73/23_un_partial_stated_room_stream_seq.sql.postgres +20 -0
- synapse/storage/schema/main/delta/73/24_events_jump_to_date_index.sql +36 -0
- synapse/storage/schema/main/delta/73/25drop_presence.sql +36 -0
- synapse/storage/schema/main/delta/74/01_user_directory_stale_remote_users.sql +58 -0
- synapse/storage/schema/main/delta/74/02_set_device_id_for_pushers_bg_update.sql +38 -0
- synapse/storage/schema/main/delta/74/03_membership_tables_event_stream_ordering.sql.postgres +29 -0
- synapse/storage/schema/main/delta/74/03_membership_tables_event_stream_ordering.sql.sqlite +23 -0
- synapse/storage/schema/main/delta/74/03_room_membership_index.sql +38 -0
- synapse/storage/schema/main/delta/74/04_delete_e2e_backup_keys_for_deactivated_users.sql +36 -0
- synapse/storage/schema/main/delta/74/04_membership_tables_event_stream_ordering_triggers.py +87 -0
- synapse/storage/schema/main/delta/74/05_events_txn_id_device_id.sql +72 -0
- synapse/storage/schema/main/delta/74/90COMMENTS_destinations.sql.postgres +52 -0
- synapse/storage/schema/main/delta/76/01_add_profiles_full_user_id_column.sql +39 -0
- synapse/storage/schema/main/delta/76/02_add_user_filters_full_user_id_column.sql +39 -0
- synapse/storage/schema/main/delta/76/03_per_user_experimental_features.sql +46 -0
- synapse/storage/schema/main/delta/76/04_add_room_forgetter.sql +43 -0
- synapse/storage/schema/main/delta/77/01_add_profiles_not_valid_check.sql.postgres +16 -0
- synapse/storage/schema/main/delta/77/02_add_user_filters_not_valid_check.sql.postgres +16 -0
- synapse/storage/schema/main/delta/77/03bg_populate_full_user_id_profiles.sql +35 -0
- synapse/storage/schema/main/delta/77/04bg_populate_full_user_id_user_filters.sql +35 -0
- synapse/storage/schema/main/delta/77/05thread_notifications_backfill.sql +67 -0
- synapse/storage/schema/main/delta/77/06thread_notifications_not_null.sql.sqlite +102 -0
- synapse/storage/schema/main/delta/77/06thread_notifications_not_null_event_push_actions.sql.postgres +27 -0
- synapse/storage/schema/main/delta/77/06thread_notifications_not_null_event_push_actions_staging.sql.postgres +27 -0
- synapse/storage/schema/main/delta/77/06thread_notifications_not_null_event_push_summary.sql.postgres +29 -0
- synapse/storage/schema/main/delta/77/14bg_indices_event_stream_ordering.sql +39 -0
- synapse/storage/schema/main/delta/78/01_validate_and_update_profiles.py +99 -0
- synapse/storage/schema/main/delta/78/02_validate_and_update_user_filters.py +100 -0
- synapse/storage/schema/main/delta/78/03_remove_unused_indexes_user_filters.py +72 -0
- synapse/storage/schema/main/delta/78/03event_extremities_constraints.py +65 -0
- synapse/storage/schema/main/delta/78/04_add_full_user_id_index_user_filters.py +32 -0
- synapse/storage/schema/main/delta/79/03_read_write_locks_triggers.sql.postgres +102 -0
- synapse/storage/schema/main/delta/79/03_read_write_locks_triggers.sql.sqlite +72 -0
- synapse/storage/schema/main/delta/79/04_mitigate_stream_ordering_update_race.py +70 -0
- synapse/storage/schema/main/delta/79/05_read_write_locks_triggers.sql.postgres +69 -0
- synapse/storage/schema/main/delta/79/05_read_write_locks_triggers.sql.sqlite +65 -0
- synapse/storage/schema/main/delta/80/01_users_alter_locked.sql +35 -0
- synapse/storage/schema/main/delta/80/02_read_write_locks_unlogged.sql.postgres +30 -0
- synapse/storage/schema/main/delta/80/02_scheduled_tasks.sql +47 -0
- synapse/storage/schema/main/delta/80/03_read_write_locks_triggers.sql.postgres +37 -0
- synapse/storage/schema/main/delta/80/04_read_write_locks_deadlock.sql.postgres +71 -0
- synapse/storage/schema/main/delta/82/02_scheduled_tasks_index.sql +35 -0
- synapse/storage/schema/main/delta/82/04_add_indices_for_purging_rooms.sql +39 -0
- synapse/storage/schema/main/delta/82/05gaps.sql +44 -0
- synapse/storage/schema/main/delta/83/01_drop_old_tables.sql +43 -0
- synapse/storage/schema/main/delta/83/03_instance_name_receipts.sql.sqlite +17 -0
- synapse/storage/schema/main/delta/83/05_cross_signing_key_update_grant.sql +34 -0
- synapse/storage/schema/main/delta/83/06_event_push_summary_room.sql +36 -0
- synapse/storage/schema/main/delta/84/01_auth_links_stats.sql.postgres +20 -0
- synapse/storage/schema/main/delta/84/02_auth_links_index.sql +16 -0
- synapse/storage/schema/main/delta/84/03_auth_links_analyze.sql.postgres +16 -0
- synapse/storage/schema/main/delta/84/04_access_token_index.sql +15 -0
- synapse/storage/schema/main/delta/85/01_add_suspended.sql +14 -0
- synapse/storage/schema/main/delta/85/02_add_instance_names.sql +27 -0
- synapse/storage/schema/main/delta/85/03_new_sequences.sql.postgres +54 -0
- synapse/storage/schema/main/delta/85/04_cleanup_device_federation_outbox.sql +15 -0
- synapse/storage/schema/main/delta/85/05_add_instance_names_converted_pos.sql +16 -0
- synapse/storage/schema/main/delta/85/06_add_room_reports.sql +20 -0
- synapse/storage/schema/main/delta/86/01_authenticate_media.sql +15 -0
- synapse/storage/schema/main/delta/86/02_receipts_event_id_index.sql +15 -0
- synapse/storage/schema/main/delta/87/01_sliding_sync_memberships.sql +169 -0
- synapse/storage/schema/main/delta/87/02_per_connection_state.sql +81 -0
- synapse/storage/schema/main/delta/87/03_current_state_index.sql +19 -0
- synapse/storage/schema/main/delta/88/01_add_delayed_events.sql +43 -0
- synapse/storage/schema/main/delta/88/01_custom_profile_fields.sql +15 -0
- synapse/storage/schema/main/delta/88/02_fix_sliding_sync_membership_snapshots_forgotten_column.sql +21 -0
- synapse/storage/schema/main/delta/88/03_add_otk_ts_added_index.sql +18 -0
- synapse/storage/schema/main/delta/88/04_current_state_delta_index.sql +18 -0
- synapse/storage/schema/main/delta/88/05_drop_old_otks.sql.postgres +19 -0
- synapse/storage/schema/main/delta/88/05_drop_old_otks.sql.sqlite +19 -0
- synapse/storage/schema/main/delta/88/05_sliding_sync_room_config_index.sql +20 -0
- synapse/storage/schema/main/delta/88/06_events_received_ts_index.sql +17 -0
- synapse/storage/schema/main/delta/89/01_sliding_sync_membership_snapshot_index.sql +15 -0
- synapse/storage/schema/main/delta/90/01_add_column_participant_room_memberships_table.sql +16 -0
- synapse/storage/schema/main/delta/91/01_media_hash.sql +28 -0
- synapse/storage/schema/main/delta/92/01_remove_trigger.sql.postgres +16 -0
- synapse/storage/schema/main/delta/92/01_remove_trigger.sql.sqlite +16 -0
- synapse/storage/schema/main/delta/92/02_remove_populate_participant_bg_update.sql +17 -0
- synapse/storage/schema/main/delta/92/04_ss_membership_snapshot_idx.sql +16 -0
- synapse/storage/schema/main/delta/92/04_thread_subscriptions.sql +59 -0
- synapse/storage/schema/main/delta/92/04_thread_subscriptions_seq.sql.postgres +19 -0
- synapse/storage/schema/main/delta/92/05_fixup_max_depth_cap.sql +17 -0
- synapse/storage/schema/main/delta/92/05_thread_subscriptions_comments.sql.postgres +18 -0
- synapse/storage/schema/main/delta/92/06_device_federation_inbox_index.sql +16 -0
- synapse/storage/schema/main/delta/92/06_threads_last_sent_stream_ordering_comments.sql.postgres +24 -0
- synapse/storage/schema/main/delta/92/07_add_user_reports.sql +22 -0
- synapse/storage/schema/main/delta/92/07_event_txn_id_device_id_txn_id2.sql +15 -0
- synapse/storage/schema/main/delta/92/08_room_ban_redactions.sql +21 -0
- synapse/storage/schema/main/delta/92/08_thread_subscriptions_seq_fixup.sql.postgres +19 -0
- synapse/storage/schema/main/delta/92/09_thread_subscriptions_update.sql +20 -0
- synapse/storage/schema/main/delta/92/09_thread_subscriptions_update.sql.postgres +18 -0
- synapse/storage/schema/main/full_schemas/72/full.sql.postgres +1344 -0
- synapse/storage/schema/main/full_schemas/72/full.sql.sqlite +646 -0
- synapse/storage/schema/state/delta/23/drop_state_index.sql +35 -0
- synapse/storage/schema/state/delta/32/remove_state_indices.sql +38 -0
- synapse/storage/schema/state/delta/35/add_state_index.sql +36 -0
- synapse/storage/schema/state/delta/35/state.sql +41 -0
- synapse/storage/schema/state/delta/35/state_dedupe.sql +36 -0
- synapse/storage/schema/state/delta/47/state_group_seq.py +38 -0
- synapse/storage/schema/state/delta/56/state_group_room_idx.sql +36 -0
- synapse/storage/schema/state/delta/61/02state_groups_state_n_distinct.sql.postgres +34 -0
- synapse/storage/schema/state/delta/70/08_state_group_edges_unique.sql +36 -0
- synapse/storage/schema/state/delta/89/01_state_groups_deletion.sql +39 -0
- synapse/storage/schema/state/delta/90/02_delete_unreferenced_state_groups.sql +16 -0
- synapse/storage/schema/state/delta/90/03_remove_old_deletion_bg_update.sql +15 -0
- synapse/storage/schema/state/full_schemas/72/full.sql.postgres +30 -0
- synapse/storage/schema/state/full_schemas/72/full.sql.sqlite +20 -0
- synapse/storage/types.py +185 -0
- synapse/storage/util/__init__.py +20 -0
- synapse/storage/util/id_generators.py +909 -0
- synapse/storage/util/partial_state_events_tracker.py +194 -0
- synapse/storage/util/sequence.py +315 -0
- synapse/streams/__init__.py +43 -0
- synapse/streams/config.py +92 -0
- synapse/streams/events.py +203 -0
- synapse/synapse_rust/__init__.pyi +3 -0
- synapse/synapse_rust/acl.pyi +20 -0
- synapse/synapse_rust/events.pyi +136 -0
- synapse/synapse_rust/http_client.pyi +32 -0
- synapse/synapse_rust/push.pyi +86 -0
- synapse/synapse_rust/rendezvous.pyi +30 -0
- synapse/synapse_rust/segmenter.pyi +1 -0
- synapse/synapse_rust.abi3.so +0 -0
- synapse/types/__init__.py +1600 -0
- synapse/types/handlers/__init__.py +93 -0
- synapse/types/handlers/policy_server.py +16 -0
- synapse/types/handlers/sliding_sync.py +909 -0
- synapse/types/rest/__init__.py +25 -0
- synapse/types/rest/client/__init__.py +415 -0
- synapse/types/state.py +635 -0
- synapse/types/storage/__init__.py +66 -0
- synapse/util/__init__.py +170 -0
- synapse/util/async_helpers.py +1067 -0
- synapse/util/batching_queue.py +202 -0
- synapse/util/caches/__init__.py +300 -0
- synapse/util/caches/cached_call.py +143 -0
- synapse/util/caches/deferred_cache.py +530 -0
- synapse/util/caches/descriptors.py +694 -0
- synapse/util/caches/dictionary_cache.py +350 -0
- synapse/util/caches/expiringcache.py +251 -0
- synapse/util/caches/lrucache.py +977 -0
- synapse/util/caches/response_cache.py +323 -0
- synapse/util/caches/stream_change_cache.py +370 -0
- synapse/util/caches/treecache.py +189 -0
- synapse/util/caches/ttlcache.py +197 -0
- synapse/util/cancellation.py +63 -0
- synapse/util/check_dependencies.py +335 -0
- synapse/util/clock.py +500 -0
- synapse/util/constants.py +22 -0
- synapse/util/daemonize.py +165 -0
- synapse/util/distributor.py +159 -0
- synapse/util/events.py +134 -0
- synapse/util/file_consumer.py +164 -0
- synapse/util/frozenutils.py +57 -0
- synapse/util/gai_resolver.py +180 -0
- synapse/util/hash.py +38 -0
- synapse/util/httpresourcetree.py +108 -0
- synapse/util/iterutils.py +189 -0
- synapse/util/json.py +56 -0
- synapse/util/linked_list.py +156 -0
- synapse/util/logcontext.py +46 -0
- synapse/util/logformatter.py +28 -0
- synapse/util/macaroons.py +325 -0
- synapse/util/manhole.py +191 -0
- synapse/util/metrics.py +340 -0
- synapse/util/module_loader.py +116 -0
- synapse/util/msisdn.py +51 -0
- synapse/util/patch_inline_callbacks.py +250 -0
- synapse/util/pydantic_models.py +56 -0
- synapse/util/ratelimitutils.py +420 -0
- synapse/util/retryutils.py +339 -0
- synapse/util/rlimit.py +42 -0
- synapse/util/rust.py +134 -0
- synapse/util/sentinel.py +21 -0
- synapse/util/stringutils.py +293 -0
- synapse/util/task_scheduler.py +493 -0
- synapse/util/templates.py +126 -0
- synapse/util/threepids.py +123 -0
- synapse/util/wheel_timer.py +112 -0
- synapse/visibility.py +836 -0
|
@@ -0,0 +1,1600 @@
|
|
|
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 2014-2016 OpenMarket Ltd
|
|
6
|
+
# Copyright (C) 2023 New Vector, Ltd
|
|
7
|
+
#
|
|
8
|
+
# This program is free software: you can redistribute it and/or modify
|
|
9
|
+
# it under the terms of the GNU Affero General Public License as
|
|
10
|
+
# published by the Free Software Foundation, either version 3 of the
|
|
11
|
+
# License, or (at your option) any later version.
|
|
12
|
+
#
|
|
13
|
+
# See the GNU Affero General Public License for more details:
|
|
14
|
+
# <https://www.gnu.org/licenses/agpl-3.0.html>.
|
|
15
|
+
#
|
|
16
|
+
# Originally licensed under the Apache License, Version 2.0:
|
|
17
|
+
# <http://www.apache.org/licenses/LICENSE-2.0>.
|
|
18
|
+
#
|
|
19
|
+
# [This file includes modifications made by New Vector Limited]
|
|
20
|
+
#
|
|
21
|
+
#
|
|
22
|
+
import abc
|
|
23
|
+
import logging
|
|
24
|
+
import re
|
|
25
|
+
import string
|
|
26
|
+
from enum import Enum
|
|
27
|
+
from typing import (
|
|
28
|
+
TYPE_CHECKING,
|
|
29
|
+
AbstractSet,
|
|
30
|
+
Any,
|
|
31
|
+
ClassVar,
|
|
32
|
+
Literal,
|
|
33
|
+
Mapping,
|
|
34
|
+
Match,
|
|
35
|
+
MutableMapping,
|
|
36
|
+
NoReturn,
|
|
37
|
+
Optional,
|
|
38
|
+
TypedDict,
|
|
39
|
+
TypeVar,
|
|
40
|
+
Union,
|
|
41
|
+
overload,
|
|
42
|
+
)
|
|
43
|
+
|
|
44
|
+
import attr
|
|
45
|
+
from immutabledict import immutabledict
|
|
46
|
+
from signedjson.key import decode_verify_key_bytes
|
|
47
|
+
from signedjson.types import VerifyKey
|
|
48
|
+
from typing_extensions import Self
|
|
49
|
+
from unpaddedbase64 import decode_base64
|
|
50
|
+
from zope.interface import Interface
|
|
51
|
+
|
|
52
|
+
from twisted.internet.defer import CancelledError
|
|
53
|
+
from twisted.internet.interfaces import (
|
|
54
|
+
IReactorCore,
|
|
55
|
+
IReactorPluggableNameResolver,
|
|
56
|
+
IReactorSSL,
|
|
57
|
+
IReactorTCP,
|
|
58
|
+
IReactorThreads,
|
|
59
|
+
IReactorTime,
|
|
60
|
+
IReactorUNIX,
|
|
61
|
+
)
|
|
62
|
+
|
|
63
|
+
from synapse.api.errors import Codes, SynapseError
|
|
64
|
+
from synapse.util.cancellation import cancellable
|
|
65
|
+
from synapse.util.stringutils import parse_and_validate_server_name
|
|
66
|
+
|
|
67
|
+
if TYPE_CHECKING:
|
|
68
|
+
from typing_extensions import Self
|
|
69
|
+
|
|
70
|
+
from synapse.appservice.api import ApplicationService
|
|
71
|
+
from synapse.events import EventBase
|
|
72
|
+
from synapse.storage.databases.main import DataStore, PurgeEventsStore
|
|
73
|
+
from synapse.storage.databases.main.appservice import ApplicationServiceWorkerStore
|
|
74
|
+
from synapse.storage.util.id_generators import MultiWriterIdGenerator
|
|
75
|
+
|
|
76
|
+
|
|
77
|
+
logger = logging.getLogger(__name__)
|
|
78
|
+
|
|
79
|
+
# Define a state map type from type/state_key to T (usually an event ID or
|
|
80
|
+
# event)
|
|
81
|
+
T = TypeVar("T")
|
|
82
|
+
StateKey = tuple[str, str]
|
|
83
|
+
StateMap = Mapping[StateKey, T]
|
|
84
|
+
MutableStateMap = MutableMapping[StateKey, T]
|
|
85
|
+
|
|
86
|
+
# JSON types. These could be made stronger, but will do for now.
|
|
87
|
+
# A "simple" (canonical) JSON value.
|
|
88
|
+
SimpleJsonValue = Optional[Union[str, int, bool]]
|
|
89
|
+
JsonValue = Union[list[SimpleJsonValue], tuple[SimpleJsonValue, ...], SimpleJsonValue]
|
|
90
|
+
# A JSON-serialisable dict.
|
|
91
|
+
JsonDict = dict[str, Any]
|
|
92
|
+
# A JSON-serialisable mapping; roughly speaking an immutable JSONDict.
|
|
93
|
+
# Useful when you have a TypedDict which isn't going to be mutated and you don't want
|
|
94
|
+
# to cast to JsonDict everywhere.
|
|
95
|
+
JsonMapping = Mapping[str, Any]
|
|
96
|
+
# A JSON-serialisable object.
|
|
97
|
+
JsonSerializable = object
|
|
98
|
+
|
|
99
|
+
# Collection[str] that does not include str itself; str being a Sequence[str]
|
|
100
|
+
# is very misleading and results in bugs.
|
|
101
|
+
#
|
|
102
|
+
# StrCollection is an unordered collection of strings. If ordering is important,
|
|
103
|
+
# StrSequence can be used instead.
|
|
104
|
+
StrCollection = Union[tuple[str, ...], list[str], AbstractSet[str]]
|
|
105
|
+
# Sequence[str] that does not include str itself; str being a Sequence[str]
|
|
106
|
+
# is very misleading and results in bugs.
|
|
107
|
+
#
|
|
108
|
+
# Unlike StrCollection, StrSequence is an ordered collection of strings.
|
|
109
|
+
StrSequence = Union[tuple[str, ...], list[str]]
|
|
110
|
+
|
|
111
|
+
|
|
112
|
+
# Note that this seems to require inheriting *directly* from Interface in order
|
|
113
|
+
# for mypy-zope to realize it is an interface.
|
|
114
|
+
class ISynapseThreadlessReactor(
|
|
115
|
+
IReactorTCP,
|
|
116
|
+
IReactorSSL,
|
|
117
|
+
IReactorUNIX,
|
|
118
|
+
IReactorPluggableNameResolver,
|
|
119
|
+
IReactorTime,
|
|
120
|
+
IReactorCore,
|
|
121
|
+
Interface,
|
|
122
|
+
):
|
|
123
|
+
"""
|
|
124
|
+
The interfaces necessary for Synapse to function (without threads).
|
|
125
|
+
|
|
126
|
+
Helpful because we use `twisted.internet.testing.MemoryReactorClock` in tests which
|
|
127
|
+
doesn't implement `IReactorThreads`.
|
|
128
|
+
"""
|
|
129
|
+
|
|
130
|
+
|
|
131
|
+
# Note that this seems to require inheriting *directly* from Interface in order
|
|
132
|
+
# for mypy-zope to realize it is an interface.
|
|
133
|
+
class ISynapseReactor(
|
|
134
|
+
ISynapseThreadlessReactor,
|
|
135
|
+
IReactorThreads,
|
|
136
|
+
Interface,
|
|
137
|
+
):
|
|
138
|
+
"""The interfaces necessary for Synapse to function."""
|
|
139
|
+
|
|
140
|
+
|
|
141
|
+
@attr.s(frozen=True, slots=True, auto_attribs=True)
|
|
142
|
+
class Requester:
|
|
143
|
+
"""
|
|
144
|
+
Represents the user making a request
|
|
145
|
+
|
|
146
|
+
Attributes:
|
|
147
|
+
user: id of the user making the request
|
|
148
|
+
access_token_id: *ID* of the access token used for this request, or
|
|
149
|
+
None for appservices, guests, and tokens generated by the admin API
|
|
150
|
+
is_guest: True if the user making this request is a guest user
|
|
151
|
+
shadow_banned: True if the user making this request has been shadow-banned.
|
|
152
|
+
device_id: device_id which was set at authentication time, or
|
|
153
|
+
None for appservices, guests, and tokens generated by the admin API
|
|
154
|
+
app_service: the AS requesting on behalf of the user
|
|
155
|
+
authenticated_entity: The entity that authenticated when making the request.
|
|
156
|
+
This is different to the user_id when an admin user or the server is
|
|
157
|
+
"puppeting" the user.
|
|
158
|
+
"""
|
|
159
|
+
|
|
160
|
+
user: "UserID"
|
|
161
|
+
access_token_id: Optional[int]
|
|
162
|
+
is_guest: bool
|
|
163
|
+
scope: set[str]
|
|
164
|
+
shadow_banned: bool
|
|
165
|
+
device_id: Optional[str]
|
|
166
|
+
app_service: Optional["ApplicationService"]
|
|
167
|
+
authenticated_entity: str
|
|
168
|
+
|
|
169
|
+
def serialize(self) -> dict[str, Any]:
|
|
170
|
+
"""Converts self to a type that can be serialized as JSON, and then
|
|
171
|
+
deserialized by `deserialize`
|
|
172
|
+
|
|
173
|
+
Returns:
|
|
174
|
+
dict
|
|
175
|
+
"""
|
|
176
|
+
return {
|
|
177
|
+
"user_id": self.user.to_string(),
|
|
178
|
+
"access_token_id": self.access_token_id,
|
|
179
|
+
"is_guest": self.is_guest,
|
|
180
|
+
"scope": list(self.scope),
|
|
181
|
+
"shadow_banned": self.shadow_banned,
|
|
182
|
+
"device_id": self.device_id,
|
|
183
|
+
"app_server_id": self.app_service.id if self.app_service else None,
|
|
184
|
+
"authenticated_entity": self.authenticated_entity,
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
@staticmethod
|
|
188
|
+
def deserialize(
|
|
189
|
+
store: "ApplicationServiceWorkerStore", input: dict[str, Any]
|
|
190
|
+
) -> "Requester":
|
|
191
|
+
"""Converts a dict that was produced by `serialize` back into a
|
|
192
|
+
Requester.
|
|
193
|
+
|
|
194
|
+
Args:
|
|
195
|
+
store: Used to convert AS ID to AS object
|
|
196
|
+
input: A dict produced by `serialize`
|
|
197
|
+
|
|
198
|
+
Returns:
|
|
199
|
+
Requester
|
|
200
|
+
"""
|
|
201
|
+
appservice = None
|
|
202
|
+
if input["app_server_id"]:
|
|
203
|
+
appservice = store.get_app_service_by_id(input["app_server_id"])
|
|
204
|
+
|
|
205
|
+
return Requester(
|
|
206
|
+
user=UserID.from_string(input["user_id"]),
|
|
207
|
+
access_token_id=input["access_token_id"],
|
|
208
|
+
is_guest=input["is_guest"],
|
|
209
|
+
scope=set(input.get("scope", [])),
|
|
210
|
+
shadow_banned=input["shadow_banned"],
|
|
211
|
+
device_id=input["device_id"],
|
|
212
|
+
app_service=appservice,
|
|
213
|
+
authenticated_entity=input["authenticated_entity"],
|
|
214
|
+
)
|
|
215
|
+
|
|
216
|
+
|
|
217
|
+
def create_requester(
|
|
218
|
+
user_id: Union[str, "UserID"],
|
|
219
|
+
access_token_id: Optional[int] = None,
|
|
220
|
+
is_guest: bool = False,
|
|
221
|
+
scope: StrCollection = (),
|
|
222
|
+
shadow_banned: bool = False,
|
|
223
|
+
device_id: Optional[str] = None,
|
|
224
|
+
app_service: Optional["ApplicationService"] = None,
|
|
225
|
+
authenticated_entity: Optional[str] = None,
|
|
226
|
+
) -> Requester:
|
|
227
|
+
"""
|
|
228
|
+
Create a new ``Requester`` object
|
|
229
|
+
|
|
230
|
+
Args:
|
|
231
|
+
user_id: id of the user making the request
|
|
232
|
+
access_token_id: *ID* of the access token used for this
|
|
233
|
+
request, or None if it came via the appservice API or similar
|
|
234
|
+
is_guest: True if the user making this request is a guest user
|
|
235
|
+
scope: the scope of the access token used for this request, if any
|
|
236
|
+
shadow_banned: True if the user making this request is shadow-banned.
|
|
237
|
+
device_id: device_id which was set at authentication time
|
|
238
|
+
app_service: the AS requesting on behalf of the user
|
|
239
|
+
authenticated_entity: The entity that authenticated when making the request.
|
|
240
|
+
This is different to the user_id when an admin user or the server is
|
|
241
|
+
"puppeting" the user.
|
|
242
|
+
|
|
243
|
+
Returns:
|
|
244
|
+
Requester
|
|
245
|
+
"""
|
|
246
|
+
if not isinstance(user_id, UserID):
|
|
247
|
+
user_id = UserID.from_string(user_id)
|
|
248
|
+
|
|
249
|
+
if authenticated_entity is None:
|
|
250
|
+
authenticated_entity = user_id.to_string()
|
|
251
|
+
|
|
252
|
+
scope = set(scope)
|
|
253
|
+
|
|
254
|
+
return Requester(
|
|
255
|
+
user_id,
|
|
256
|
+
access_token_id,
|
|
257
|
+
is_guest,
|
|
258
|
+
scope,
|
|
259
|
+
shadow_banned,
|
|
260
|
+
device_id,
|
|
261
|
+
app_service,
|
|
262
|
+
authenticated_entity,
|
|
263
|
+
)
|
|
264
|
+
|
|
265
|
+
|
|
266
|
+
def get_domain_from_id(string: str) -> str:
|
|
267
|
+
idx = string.find(":")
|
|
268
|
+
if idx == -1:
|
|
269
|
+
raise SynapseError(400, "Invalid ID: %r" % (string,))
|
|
270
|
+
return string[idx + 1 :]
|
|
271
|
+
|
|
272
|
+
|
|
273
|
+
def get_localpart_from_id(string: str) -> str:
|
|
274
|
+
idx = string.find(":")
|
|
275
|
+
if idx == -1:
|
|
276
|
+
raise SynapseError(400, "Invalid ID: %r" % (string,))
|
|
277
|
+
return string[1:idx]
|
|
278
|
+
|
|
279
|
+
|
|
280
|
+
DS = TypeVar("DS", bound="DomainSpecificString")
|
|
281
|
+
|
|
282
|
+
|
|
283
|
+
@attr.s(slots=True, frozen=True, repr=False, auto_attribs=True)
|
|
284
|
+
class DomainSpecificString(metaclass=abc.ABCMeta):
|
|
285
|
+
"""Common base class among ID/name strings that have a local part and a
|
|
286
|
+
domain name, prefixed with a sigil.
|
|
287
|
+
|
|
288
|
+
Has the fields:
|
|
289
|
+
|
|
290
|
+
'localpart' : The local part of the name (without the leading sigil)
|
|
291
|
+
'domain' : The domain part of the name
|
|
292
|
+
"""
|
|
293
|
+
|
|
294
|
+
SIGIL: ClassVar[str] = abc.abstractproperty() # type: ignore
|
|
295
|
+
|
|
296
|
+
localpart: str
|
|
297
|
+
domain: str
|
|
298
|
+
|
|
299
|
+
# Because this is a frozen class, it is deeply immutable.
|
|
300
|
+
def __copy__(self: DS) -> DS:
|
|
301
|
+
return self
|
|
302
|
+
|
|
303
|
+
def __deepcopy__(self: DS, memo: dict[str, object]) -> DS:
|
|
304
|
+
return self
|
|
305
|
+
|
|
306
|
+
@classmethod
|
|
307
|
+
def from_string(cls: type[DS], s: str) -> DS:
|
|
308
|
+
"""Parse the string given by 's' into a structure object."""
|
|
309
|
+
if len(s) < 1 or s[0:1] != cls.SIGIL:
|
|
310
|
+
raise SynapseError(
|
|
311
|
+
400,
|
|
312
|
+
"Expected %s string to start with '%s'" % (cls.__name__, cls.SIGIL),
|
|
313
|
+
Codes.INVALID_PARAM,
|
|
314
|
+
)
|
|
315
|
+
|
|
316
|
+
parts = s[1:].split(":", 1)
|
|
317
|
+
if len(parts) != 2:
|
|
318
|
+
raise SynapseError(
|
|
319
|
+
400,
|
|
320
|
+
"Expected %s of the form '%slocalname:domain'"
|
|
321
|
+
% (cls.__name__, cls.SIGIL),
|
|
322
|
+
Codes.INVALID_PARAM,
|
|
323
|
+
)
|
|
324
|
+
|
|
325
|
+
domain = parts[1]
|
|
326
|
+
# This code will need changing if we want to support multiple domain
|
|
327
|
+
# names on one HS
|
|
328
|
+
return cls(localpart=parts[0], domain=domain)
|
|
329
|
+
|
|
330
|
+
def to_string(self) -> str:
|
|
331
|
+
"""Return a string encoding the fields of the structure object."""
|
|
332
|
+
return "%s%s:%s" % (self.SIGIL, self.localpart, self.domain)
|
|
333
|
+
|
|
334
|
+
@classmethod
|
|
335
|
+
def is_valid(cls: type[DS], s: str) -> bool:
|
|
336
|
+
"""Parses the input string and attempts to ensure it is valid."""
|
|
337
|
+
# TODO: this does not reject an empty localpart or an overly-long string.
|
|
338
|
+
# See https://spec.matrix.org/v1.2/appendices/#identifier-grammar
|
|
339
|
+
try:
|
|
340
|
+
obj = cls.from_string(s)
|
|
341
|
+
# Apply additional validation to the domain. This is only done
|
|
342
|
+
# during is_valid (and not part of from_string) since it is
|
|
343
|
+
# possible for invalid data to exist in room-state, etc.
|
|
344
|
+
parse_and_validate_server_name(obj.domain)
|
|
345
|
+
return True
|
|
346
|
+
except Exception:
|
|
347
|
+
return False
|
|
348
|
+
|
|
349
|
+
__repr__ = to_string
|
|
350
|
+
|
|
351
|
+
|
|
352
|
+
@attr.s(slots=True, frozen=True, repr=False)
|
|
353
|
+
class UserID(DomainSpecificString):
|
|
354
|
+
"""Structure representing a user ID."""
|
|
355
|
+
|
|
356
|
+
SIGIL = "@"
|
|
357
|
+
|
|
358
|
+
|
|
359
|
+
@attr.s(slots=True, frozen=True, repr=False)
|
|
360
|
+
class RoomAlias(DomainSpecificString):
|
|
361
|
+
"""Structure representing a room name."""
|
|
362
|
+
|
|
363
|
+
SIGIL = "#"
|
|
364
|
+
|
|
365
|
+
|
|
366
|
+
@attr.s(slots=True, frozen=True, repr=False)
|
|
367
|
+
class RoomIdWithDomain(DomainSpecificString):
|
|
368
|
+
"""Structure representing a room ID with a domain suffix."""
|
|
369
|
+
|
|
370
|
+
SIGIL = "!"
|
|
371
|
+
|
|
372
|
+
|
|
373
|
+
# the set of urlsafe base64 characters, no padding.
|
|
374
|
+
ROOM_ID_PATTERN_DOMAINLESS = re.compile(r"^[A-Za-z0-9\-_]{43}$")
|
|
375
|
+
|
|
376
|
+
|
|
377
|
+
@attr.s(slots=True, frozen=True, auto_attribs=True, repr=False)
|
|
378
|
+
class RoomID:
|
|
379
|
+
"""Structure representing a room id without a domain.
|
|
380
|
+
There are two forms of room IDs:
|
|
381
|
+
- "!localpart:domain" used in most room versions prior to MSC4291.
|
|
382
|
+
- "!event_id_base_64" used in room versions post MSC4291.
|
|
383
|
+
This class will accept any room ID which meets either of these two criteria.
|
|
384
|
+
"""
|
|
385
|
+
|
|
386
|
+
SIGIL = "!"
|
|
387
|
+
id: str
|
|
388
|
+
room_id_with_domain: Optional[RoomIdWithDomain]
|
|
389
|
+
|
|
390
|
+
@classmethod
|
|
391
|
+
def is_valid(cls: type["RoomID"], s: str) -> bool:
|
|
392
|
+
if ":" in s:
|
|
393
|
+
return RoomIdWithDomain.is_valid(s)
|
|
394
|
+
try:
|
|
395
|
+
cls.from_string(s)
|
|
396
|
+
return True
|
|
397
|
+
except Exception:
|
|
398
|
+
return False
|
|
399
|
+
|
|
400
|
+
def get_domain(self) -> Optional[str]:
|
|
401
|
+
if not self.room_id_with_domain:
|
|
402
|
+
return None
|
|
403
|
+
return self.room_id_with_domain.domain
|
|
404
|
+
|
|
405
|
+
def to_string(self) -> str:
|
|
406
|
+
if self.room_id_with_domain:
|
|
407
|
+
return self.room_id_with_domain.to_string()
|
|
408
|
+
return self.id
|
|
409
|
+
|
|
410
|
+
__repr__ = to_string
|
|
411
|
+
|
|
412
|
+
@classmethod
|
|
413
|
+
def from_string(cls: type["RoomID"], s: str) -> "RoomID":
|
|
414
|
+
# sigil check
|
|
415
|
+
if len(s) < 1 or s[0] != cls.SIGIL:
|
|
416
|
+
raise SynapseError(
|
|
417
|
+
400,
|
|
418
|
+
"Expected %s string to start with '%s'" % (cls.__name__, cls.SIGIL),
|
|
419
|
+
Codes.INVALID_PARAM,
|
|
420
|
+
)
|
|
421
|
+
|
|
422
|
+
room_id_with_domain: Optional[RoomIdWithDomain] = None
|
|
423
|
+
if ":" in s:
|
|
424
|
+
room_id_with_domain = RoomIdWithDomain.from_string(s)
|
|
425
|
+
else:
|
|
426
|
+
# MSC4291 room IDs must be valid urlsafe unpadded base64
|
|
427
|
+
val = s[1:]
|
|
428
|
+
if not ROOM_ID_PATTERN_DOMAINLESS.match(val):
|
|
429
|
+
raise SynapseError(
|
|
430
|
+
400,
|
|
431
|
+
"Expected %s string to be valid urlsafe unpadded base64 '%s'"
|
|
432
|
+
% (cls.__name__, val),
|
|
433
|
+
Codes.INVALID_PARAM,
|
|
434
|
+
)
|
|
435
|
+
|
|
436
|
+
return cls(id=s, room_id_with_domain=room_id_with_domain)
|
|
437
|
+
|
|
438
|
+
|
|
439
|
+
@attr.s(slots=True, frozen=True, repr=False)
|
|
440
|
+
class EventID(DomainSpecificString):
|
|
441
|
+
"""Structure representing an event ID which is namespaced to a homeserver.
|
|
442
|
+
Room versions 3 and above are not supported by this grammar."""
|
|
443
|
+
|
|
444
|
+
SIGIL = "$"
|
|
445
|
+
|
|
446
|
+
|
|
447
|
+
MXID_LOCALPART_ALLOWED_CHARACTERS = set(
|
|
448
|
+
"_-./=+" + string.ascii_lowercase + string.digits
|
|
449
|
+
)
|
|
450
|
+
|
|
451
|
+
# Guest user IDs are purely numeric.
|
|
452
|
+
GUEST_USER_ID_PATTERN = re.compile(r"^\d+$")
|
|
453
|
+
|
|
454
|
+
|
|
455
|
+
def contains_invalid_mxid_characters(localpart: str) -> bool:
|
|
456
|
+
"""Check for characters not allowed in an mxid or groupid localpart
|
|
457
|
+
|
|
458
|
+
Args:
|
|
459
|
+
localpart: the localpart to be checked
|
|
460
|
+
use_extended_character_set: True to use the extended allowed characters
|
|
461
|
+
from MSC4009.
|
|
462
|
+
|
|
463
|
+
Returns:
|
|
464
|
+
True if there are any naughty characters
|
|
465
|
+
"""
|
|
466
|
+
return any(c not in MXID_LOCALPART_ALLOWED_CHARACTERS for c in localpart)
|
|
467
|
+
|
|
468
|
+
|
|
469
|
+
UPPER_CASE_PATTERN = re.compile(b"[A-Z_]")
|
|
470
|
+
|
|
471
|
+
# the following is a pattern which matches '=', and bytes which are not allowed in a mxid
|
|
472
|
+
# localpart.
|
|
473
|
+
#
|
|
474
|
+
# It works by:
|
|
475
|
+
# * building a string containing the allowed characters (excluding '=')
|
|
476
|
+
# * escaping every special character with a backslash (to stop '-' being interpreted as a
|
|
477
|
+
# range operator)
|
|
478
|
+
# * wrapping it in a '[^...]' regex
|
|
479
|
+
# * converting the whole lot to a 'bytes' sequence, so that we can use it to match
|
|
480
|
+
# bytes rather than strings
|
|
481
|
+
#
|
|
482
|
+
NON_MXID_CHARACTER_PATTERN = re.compile(
|
|
483
|
+
("[^%s]" % (re.escape("".join(MXID_LOCALPART_ALLOWED_CHARACTERS - {"="})),)).encode(
|
|
484
|
+
"ascii"
|
|
485
|
+
)
|
|
486
|
+
)
|
|
487
|
+
|
|
488
|
+
|
|
489
|
+
def map_username_to_mxid_localpart(
|
|
490
|
+
username: Union[str, bytes], case_sensitive: bool = False
|
|
491
|
+
) -> str:
|
|
492
|
+
"""Map a username onto a string suitable for a MXID
|
|
493
|
+
|
|
494
|
+
This follows the algorithm laid out at
|
|
495
|
+
https://matrix.org/docs/spec/appendices.html#mapping-from-other-character-sets.
|
|
496
|
+
|
|
497
|
+
Args:
|
|
498
|
+
username: username to be mapped
|
|
499
|
+
case_sensitive: true if TEST and test should be mapped
|
|
500
|
+
onto different mxids
|
|
501
|
+
|
|
502
|
+
Returns:
|
|
503
|
+
string suitable for a mxid localpart
|
|
504
|
+
"""
|
|
505
|
+
if not isinstance(username, bytes):
|
|
506
|
+
username = username.encode("utf-8")
|
|
507
|
+
|
|
508
|
+
# first we sort out upper-case characters
|
|
509
|
+
if case_sensitive:
|
|
510
|
+
|
|
511
|
+
def f1(m: Match[bytes]) -> bytes:
|
|
512
|
+
return b"_" + m.group().lower()
|
|
513
|
+
|
|
514
|
+
username = UPPER_CASE_PATTERN.sub(f1, username)
|
|
515
|
+
else:
|
|
516
|
+
username = username.lower()
|
|
517
|
+
|
|
518
|
+
# then we sort out non-ascii characters by converting to the hex equivalent.
|
|
519
|
+
def f2(m: Match[bytes]) -> bytes:
|
|
520
|
+
return b"=%02x" % (m.group()[0],)
|
|
521
|
+
|
|
522
|
+
username = NON_MXID_CHARACTER_PATTERN.sub(f2, username)
|
|
523
|
+
|
|
524
|
+
# we also do the =-escaping to mxids starting with an underscore.
|
|
525
|
+
username = re.sub(b"^_", b"=5f", username)
|
|
526
|
+
|
|
527
|
+
# we should now only have ascii bytes left, so can decode back to a string.
|
|
528
|
+
return username.decode("ascii")
|
|
529
|
+
|
|
530
|
+
|
|
531
|
+
@attr.s(frozen=True, slots=True, order=False)
|
|
532
|
+
class AbstractMultiWriterStreamToken(metaclass=abc.ABCMeta):
|
|
533
|
+
"""An abstract stream token class for streams that supports multiple
|
|
534
|
+
writers.
|
|
535
|
+
|
|
536
|
+
This works by keeping track of the stream position of each writer,
|
|
537
|
+
represented by a default `stream` attribute and a map of instance name to
|
|
538
|
+
stream position of any writers that are ahead of the default stream
|
|
539
|
+
position.
|
|
540
|
+
|
|
541
|
+
The values in `instance_map` must be greater than the `stream` attribute.
|
|
542
|
+
"""
|
|
543
|
+
|
|
544
|
+
stream: int = attr.ib(validator=attr.validators.instance_of(int), kw_only=True)
|
|
545
|
+
|
|
546
|
+
instance_map: "immutabledict[str, int]" = attr.ib(
|
|
547
|
+
factory=immutabledict,
|
|
548
|
+
validator=attr.validators.deep_mapping(
|
|
549
|
+
key_validator=attr.validators.instance_of(str),
|
|
550
|
+
value_validator=attr.validators.instance_of(int),
|
|
551
|
+
mapping_validator=attr.validators.instance_of(immutabledict),
|
|
552
|
+
),
|
|
553
|
+
kw_only=True,
|
|
554
|
+
)
|
|
555
|
+
|
|
556
|
+
def __attrs_post_init__(self) -> None:
|
|
557
|
+
# Enforce that all instances have a value greater than the min stream
|
|
558
|
+
# position.
|
|
559
|
+
for i, v in self.instance_map.items():
|
|
560
|
+
if v <= self.stream:
|
|
561
|
+
raise ValueError(
|
|
562
|
+
f"'instance_map' includes a stream position before the main 'stream' attribute. Instance: {i}"
|
|
563
|
+
)
|
|
564
|
+
|
|
565
|
+
@classmethod
|
|
566
|
+
@abc.abstractmethod
|
|
567
|
+
async def parse(cls, store: "DataStore", string: str) -> "Self":
|
|
568
|
+
"""Parse the string representation of the token."""
|
|
569
|
+
...
|
|
570
|
+
|
|
571
|
+
@abc.abstractmethod
|
|
572
|
+
async def to_string(self, store: "DataStore") -> str:
|
|
573
|
+
"""Serialize the token into its string representation."""
|
|
574
|
+
...
|
|
575
|
+
|
|
576
|
+
def copy_and_advance(self, other: "Self") -> "Self":
|
|
577
|
+
"""Return a new token such that if an event is after both this token and
|
|
578
|
+
the other token, then its after the returned token too.
|
|
579
|
+
"""
|
|
580
|
+
|
|
581
|
+
max_stream = max(self.stream, other.stream)
|
|
582
|
+
|
|
583
|
+
instance_map = {
|
|
584
|
+
instance: max(
|
|
585
|
+
self.instance_map.get(instance, self.stream),
|
|
586
|
+
other.instance_map.get(instance, other.stream),
|
|
587
|
+
)
|
|
588
|
+
for instance in set(self.instance_map).union(other.instance_map)
|
|
589
|
+
}
|
|
590
|
+
|
|
591
|
+
# Filter out any redundant entries.
|
|
592
|
+
instance_map = {i: s for i, s in instance_map.items() if s > max_stream}
|
|
593
|
+
|
|
594
|
+
return attr.evolve(
|
|
595
|
+
self, stream=max_stream, instance_map=immutabledict(instance_map)
|
|
596
|
+
)
|
|
597
|
+
|
|
598
|
+
def get_max_stream_pos(self) -> int:
|
|
599
|
+
"""Get the maximum stream position referenced in this token.
|
|
600
|
+
|
|
601
|
+
The corresponding "min" position is, by definition just `self.stream`.
|
|
602
|
+
|
|
603
|
+
This is used to handle tokens that have non-empty `instance_map`, and so
|
|
604
|
+
reference stream positions after the `self.stream` position.
|
|
605
|
+
"""
|
|
606
|
+
return max(self.instance_map.values(), default=self.stream)
|
|
607
|
+
|
|
608
|
+
def get_stream_pos_for_instance(self, instance_name: str) -> int:
|
|
609
|
+
"""Get the stream position that the given writer was at at this token."""
|
|
610
|
+
|
|
611
|
+
# If we don't have an entry for the instance we can assume that it was
|
|
612
|
+
# at `self.stream`.
|
|
613
|
+
return self.instance_map.get(instance_name, self.stream)
|
|
614
|
+
|
|
615
|
+
def is_before_or_eq(self, other_token: Self) -> bool:
|
|
616
|
+
"""Wether this token is before the other token, i.e. every constituent
|
|
617
|
+
part is before the other.
|
|
618
|
+
|
|
619
|
+
Essentially it is `self <= other`.
|
|
620
|
+
|
|
621
|
+
Note: if `self.is_before_or_eq(other_token) is False` then that does not
|
|
622
|
+
imply that the reverse is True.
|
|
623
|
+
"""
|
|
624
|
+
if self.stream > other_token.stream:
|
|
625
|
+
return False
|
|
626
|
+
|
|
627
|
+
instances = self.instance_map.keys() | other_token.instance_map.keys()
|
|
628
|
+
for instance in instances:
|
|
629
|
+
if self.instance_map.get(
|
|
630
|
+
instance, self.stream
|
|
631
|
+
) > other_token.instance_map.get(instance, other_token.stream):
|
|
632
|
+
return False
|
|
633
|
+
|
|
634
|
+
return True
|
|
635
|
+
|
|
636
|
+
def bound_stream_token(self, max_stream: int) -> "Self":
|
|
637
|
+
"""Bound the stream positions to a maximum value"""
|
|
638
|
+
|
|
639
|
+
min_pos = min(self.stream, max_stream)
|
|
640
|
+
return type(self)(
|
|
641
|
+
stream=min_pos,
|
|
642
|
+
instance_map=immutabledict(
|
|
643
|
+
{
|
|
644
|
+
k: min(s, max_stream)
|
|
645
|
+
for k, s in self.instance_map.items()
|
|
646
|
+
if min(s, max_stream) > min_pos
|
|
647
|
+
}
|
|
648
|
+
),
|
|
649
|
+
)
|
|
650
|
+
|
|
651
|
+
@classmethod
|
|
652
|
+
def from_generator(cls, generator: "MultiWriterIdGenerator") -> Self:
|
|
653
|
+
"""Get the current token out of a MultiWriterIdGenerator"""
|
|
654
|
+
|
|
655
|
+
# The `min_pos` is the minimum position that we know all instances
|
|
656
|
+
# have finished persisting to, so we only care about instances whose
|
|
657
|
+
# positions are ahead of that. (Instance positions can be behind the
|
|
658
|
+
# min position as there are times we can work out that the minimum
|
|
659
|
+
# position is ahead of the naive minimum across all current
|
|
660
|
+
# positions. See MultiWriterIdGenerator for details)
|
|
661
|
+
min_pos = generator.get_current_token()
|
|
662
|
+
positions = {
|
|
663
|
+
instance: position
|
|
664
|
+
for instance, position in generator.get_positions().items()
|
|
665
|
+
if position > min_pos
|
|
666
|
+
}
|
|
667
|
+
|
|
668
|
+
return cls(stream=min_pos, instance_map=immutabledict(positions))
|
|
669
|
+
|
|
670
|
+
|
|
671
|
+
@attr.s(frozen=True, slots=True, order=False)
|
|
672
|
+
class RoomStreamToken(AbstractMultiWriterStreamToken):
|
|
673
|
+
"""Tokens are positions between events. The token "s1" comes after event 1.
|
|
674
|
+
|
|
675
|
+
s0 s1
|
|
676
|
+
| |
|
|
677
|
+
[0] ▼ [1] ▼ [2]
|
|
678
|
+
|
|
679
|
+
Tokens can either be a point in the live event stream or a cursor going
|
|
680
|
+
through historic events.
|
|
681
|
+
|
|
682
|
+
When traversing the live event stream, events are ordered by
|
|
683
|
+
`stream_ordering` (when they arrived at the homeserver).
|
|
684
|
+
|
|
685
|
+
When traversing historic events, events are first ordered by their `depth`
|
|
686
|
+
(`topological_ordering` in the event graph) and tie-broken by
|
|
687
|
+
`stream_ordering` (when the event arrived at the homeserver).
|
|
688
|
+
|
|
689
|
+
If you're looking for more info about what a token with all of the
|
|
690
|
+
underscores means, ex.
|
|
691
|
+
`s2633508_17_338_6732159_1082514_541479_274711_265584_1`, see the docstring
|
|
692
|
+
for `StreamToken` below.
|
|
693
|
+
|
|
694
|
+
---
|
|
695
|
+
|
|
696
|
+
Live tokens start with an "s" followed by the `stream_ordering` of the event
|
|
697
|
+
that comes before the position of the token. Said another way:
|
|
698
|
+
`stream_ordering` uniquely identifies a persisted event. The live token
|
|
699
|
+
means "the position just after the event identified by `stream_ordering`".
|
|
700
|
+
An example token is:
|
|
701
|
+
|
|
702
|
+
s2633508
|
|
703
|
+
|
|
704
|
+
---
|
|
705
|
+
|
|
706
|
+
Historic tokens start with a "t" followed by the `depth`
|
|
707
|
+
(`topological_ordering` in the event graph) of the event that comes before
|
|
708
|
+
the position of the token, followed by "-", followed by the
|
|
709
|
+
`stream_ordering` of the event that comes before the position of the token.
|
|
710
|
+
An example token is:
|
|
711
|
+
|
|
712
|
+
t426-2633508
|
|
713
|
+
|
|
714
|
+
---
|
|
715
|
+
|
|
716
|
+
There is also a third mode for live tokens where the token starts with "m",
|
|
717
|
+
which is sometimes used when using sharded event persisters. In this case
|
|
718
|
+
the events stream is considered to be a set of streams (one for each writer)
|
|
719
|
+
and the token encodes the vector clock of positions of each writer in their
|
|
720
|
+
respective streams.
|
|
721
|
+
|
|
722
|
+
The format of the token in such case is an initial integer min position,
|
|
723
|
+
followed by the mapping of instance ID to position separated by '.' and '~':
|
|
724
|
+
|
|
725
|
+
m{min_pos}~{writer1}.{pos1}~{writer2}.{pos2}. ...
|
|
726
|
+
|
|
727
|
+
The `min_pos` corresponds to the minimum position all writers have persisted
|
|
728
|
+
up to, and then only writers that are ahead of that position need to be
|
|
729
|
+
encoded. An example token is:
|
|
730
|
+
|
|
731
|
+
m56~2.58~3.59
|
|
732
|
+
|
|
733
|
+
Which corresponds to a set of three (or more writers) where instances 2 and
|
|
734
|
+
3 (these are instance IDs that can be looked up in the DB to fetch the more
|
|
735
|
+
commonly used instance names) are at positions 58 and 59 respectively, and
|
|
736
|
+
all other instances are at position 56.
|
|
737
|
+
|
|
738
|
+
Note: The `RoomStreamToken` cannot have both a topological part and an
|
|
739
|
+
instance map.
|
|
740
|
+
|
|
741
|
+
---
|
|
742
|
+
|
|
743
|
+
For caching purposes, `RoomStreamToken`s and by extension, all their
|
|
744
|
+
attributes, must be hashable.
|
|
745
|
+
"""
|
|
746
|
+
|
|
747
|
+
topological: Optional[int] = attr.ib(
|
|
748
|
+
validator=attr.validators.optional(attr.validators.instance_of(int)),
|
|
749
|
+
kw_only=True,
|
|
750
|
+
default=None,
|
|
751
|
+
)
|
|
752
|
+
|
|
753
|
+
def __attrs_post_init__(self) -> None:
|
|
754
|
+
"""Validates that both `topological` and `instance_map` aren't set."""
|
|
755
|
+
|
|
756
|
+
if self.instance_map and self.topological:
|
|
757
|
+
raise ValueError(
|
|
758
|
+
"Cannot set both 'topological' and 'instance_map' on 'RoomStreamToken'."
|
|
759
|
+
)
|
|
760
|
+
|
|
761
|
+
super().__attrs_post_init__()
|
|
762
|
+
|
|
763
|
+
@classmethod
|
|
764
|
+
async def parse(cls, store: "PurgeEventsStore", string: str) -> "RoomStreamToken":
|
|
765
|
+
# Check that it looks like a Synapse token first. We do this so that
|
|
766
|
+
# we don't log at the exception-level for obviously incorrect tokens.
|
|
767
|
+
if not string or string[0] not in ("s", "t", "m"):
|
|
768
|
+
raise SynapseError(400, f"Invalid room stream token {string:!r}")
|
|
769
|
+
|
|
770
|
+
try:
|
|
771
|
+
if string[0] == "s":
|
|
772
|
+
return cls(topological=None, stream=int(string[1:]))
|
|
773
|
+
if string[0] == "t":
|
|
774
|
+
parts = string[1:].split("-", 1)
|
|
775
|
+
return cls(topological=int(parts[0]), stream=int(parts[1]))
|
|
776
|
+
if string[0] == "m":
|
|
777
|
+
parts = string[1:].split("~")
|
|
778
|
+
stream = int(parts[0])
|
|
779
|
+
|
|
780
|
+
instance_map = {}
|
|
781
|
+
for part in parts[1:]:
|
|
782
|
+
if not part:
|
|
783
|
+
# Handle tokens of the form `m5~`, which were created by
|
|
784
|
+
# a bug
|
|
785
|
+
continue
|
|
786
|
+
|
|
787
|
+
key, value = part.split(".")
|
|
788
|
+
instance_id = int(key)
|
|
789
|
+
pos = int(value)
|
|
790
|
+
|
|
791
|
+
instance_name = await store.get_name_from_instance_id(instance_id) # type: ignore[attr-defined]
|
|
792
|
+
instance_map[instance_name] = pos
|
|
793
|
+
|
|
794
|
+
return cls(
|
|
795
|
+
topological=None,
|
|
796
|
+
stream=stream,
|
|
797
|
+
instance_map=immutabledict(instance_map),
|
|
798
|
+
)
|
|
799
|
+
except CancelledError:
|
|
800
|
+
raise
|
|
801
|
+
except Exception:
|
|
802
|
+
# We log an exception here as even though this *might* be a client
|
|
803
|
+
# handing a bad token, its more likely that Synapse returned a bad
|
|
804
|
+
# token (and we really want to catch those!).
|
|
805
|
+
logger.exception("Failed to parse stream token: %r", string)
|
|
806
|
+
raise SynapseError(400, "Invalid room stream token %r" % (string,))
|
|
807
|
+
|
|
808
|
+
@classmethod
|
|
809
|
+
def parse_stream_token(cls, string: str) -> "RoomStreamToken":
|
|
810
|
+
try:
|
|
811
|
+
if string[0] == "s":
|
|
812
|
+
return cls(topological=None, stream=int(string[1:]))
|
|
813
|
+
except Exception:
|
|
814
|
+
pass
|
|
815
|
+
raise SynapseError(400, "Invalid room stream token %r" % (string,))
|
|
816
|
+
|
|
817
|
+
def copy_and_advance(self, other: "RoomStreamToken") -> "RoomStreamToken":
|
|
818
|
+
"""Return a new token such that if an event is after both this token and
|
|
819
|
+
the other token, then its after the returned token too.
|
|
820
|
+
"""
|
|
821
|
+
|
|
822
|
+
if self.topological or other.topological:
|
|
823
|
+
raise Exception("Can't advance topological tokens")
|
|
824
|
+
|
|
825
|
+
return super().copy_and_advance(other)
|
|
826
|
+
|
|
827
|
+
def as_historical_tuple(self) -> tuple[int, int]:
|
|
828
|
+
"""Returns a tuple of `(topological, stream)` for historical tokens.
|
|
829
|
+
|
|
830
|
+
Raises if not an historical token (i.e. doesn't have a topological part).
|
|
831
|
+
"""
|
|
832
|
+
if self.topological is None:
|
|
833
|
+
raise Exception(
|
|
834
|
+
"Cannot call `RoomStreamToken.as_historical_tuple` on live token"
|
|
835
|
+
)
|
|
836
|
+
|
|
837
|
+
return self.topological, self.stream
|
|
838
|
+
|
|
839
|
+
def get_stream_pos_for_instance(self, instance_name: str) -> int:
|
|
840
|
+
"""Get the stream position that the given writer was at at this token.
|
|
841
|
+
|
|
842
|
+
This only makes sense for "live" tokens that may have a vector clock
|
|
843
|
+
component, and so asserts that this is a "live" token.
|
|
844
|
+
"""
|
|
845
|
+
assert self.topological is None
|
|
846
|
+
|
|
847
|
+
# If we don't have an entry for the instance we can assume that it was
|
|
848
|
+
# at `self.stream`.
|
|
849
|
+
return self.instance_map.get(instance_name, self.stream)
|
|
850
|
+
|
|
851
|
+
async def to_string(self, store: "DataStore") -> str:
|
|
852
|
+
"""See class level docstring for information about the format."""
|
|
853
|
+
|
|
854
|
+
if self.topological is not None:
|
|
855
|
+
return "t%d-%d" % (self.topological, self.stream)
|
|
856
|
+
elif self.instance_map:
|
|
857
|
+
entries = []
|
|
858
|
+
for name, pos in self.instance_map.items():
|
|
859
|
+
if pos <= self.stream:
|
|
860
|
+
# Ignore instances who are below the minimum stream position
|
|
861
|
+
# (we might know they've advanced without seeing a recent
|
|
862
|
+
# write from them).
|
|
863
|
+
continue
|
|
864
|
+
|
|
865
|
+
instance_id = await store.get_id_for_instance(name)
|
|
866
|
+
entries.append(f"{instance_id}.{pos}")
|
|
867
|
+
|
|
868
|
+
if entries:
|
|
869
|
+
encoded_map = "~".join(entries)
|
|
870
|
+
return f"m{self.stream}~{encoded_map}"
|
|
871
|
+
return f"s{self.stream}"
|
|
872
|
+
else:
|
|
873
|
+
return "s%d" % (self.stream,)
|
|
874
|
+
|
|
875
|
+
def bound_stream_token(self, max_stream: int) -> "RoomStreamToken":
|
|
876
|
+
"""See super class"""
|
|
877
|
+
|
|
878
|
+
# This only makes sense for stream tokens.
|
|
879
|
+
assert self.topological is None
|
|
880
|
+
|
|
881
|
+
return super().bound_stream_token(max_stream)
|
|
882
|
+
|
|
883
|
+
def __str__(self) -> str:
|
|
884
|
+
instances = ", ".join(f"{k}: {v}" for k, v in sorted(self.instance_map.items()))
|
|
885
|
+
return (
|
|
886
|
+
f"RoomStreamToken(stream: {self.stream}, topological: {self.topological}, "
|
|
887
|
+
f"instances: {{{instances}}})"
|
|
888
|
+
)
|
|
889
|
+
|
|
890
|
+
|
|
891
|
+
@attr.s(frozen=True, slots=True, order=False)
|
|
892
|
+
class MultiWriterStreamToken(AbstractMultiWriterStreamToken):
|
|
893
|
+
"""A basic stream token class for streams that supports multiple writers."""
|
|
894
|
+
|
|
895
|
+
@classmethod
|
|
896
|
+
async def parse(cls, store: "DataStore", string: str) -> "MultiWriterStreamToken":
|
|
897
|
+
try:
|
|
898
|
+
if string[0].isdigit():
|
|
899
|
+
return cls(stream=int(string))
|
|
900
|
+
if string[0] == "m":
|
|
901
|
+
parts = string[1:].split("~")
|
|
902
|
+
stream = int(parts[0])
|
|
903
|
+
|
|
904
|
+
instance_map = {}
|
|
905
|
+
for part in parts[1:]:
|
|
906
|
+
if not part:
|
|
907
|
+
# Handle tokens of the form `m5~`, which were created by
|
|
908
|
+
# a bug
|
|
909
|
+
continue
|
|
910
|
+
|
|
911
|
+
key, value = part.split(".")
|
|
912
|
+
instance_id = int(key)
|
|
913
|
+
pos = int(value)
|
|
914
|
+
|
|
915
|
+
instance_name = await store.get_name_from_instance_id(instance_id)
|
|
916
|
+
instance_map[instance_name] = pos
|
|
917
|
+
|
|
918
|
+
return cls(
|
|
919
|
+
stream=stream,
|
|
920
|
+
instance_map=immutabledict(instance_map),
|
|
921
|
+
)
|
|
922
|
+
except CancelledError:
|
|
923
|
+
raise
|
|
924
|
+
except Exception:
|
|
925
|
+
# We log an exception here as even though this *might* be a client
|
|
926
|
+
# handing a bad token, its more likely that Synapse returned a bad
|
|
927
|
+
# token (and we really want to catch those!).
|
|
928
|
+
logger.exception("Failed to parse stream token: %r", string)
|
|
929
|
+
raise SynapseError(400, "Invalid stream token %r" % (string,))
|
|
930
|
+
|
|
931
|
+
async def to_string(self, store: "DataStore") -> str:
|
|
932
|
+
"""See class level docstring for information about the format."""
|
|
933
|
+
|
|
934
|
+
if self.instance_map:
|
|
935
|
+
entries = []
|
|
936
|
+
for name, pos in self.instance_map.items():
|
|
937
|
+
if pos <= self.stream:
|
|
938
|
+
# Ignore instances who are below the minimum stream position
|
|
939
|
+
# (we might know they've advanced without seeing a recent
|
|
940
|
+
# write from them).
|
|
941
|
+
continue
|
|
942
|
+
|
|
943
|
+
instance_id = await store.get_id_for_instance(name)
|
|
944
|
+
entries.append(f"{instance_id}.{pos}")
|
|
945
|
+
|
|
946
|
+
if entries:
|
|
947
|
+
encoded_map = "~".join(entries)
|
|
948
|
+
return f"m{self.stream}~{encoded_map}"
|
|
949
|
+
return str(self.stream)
|
|
950
|
+
else:
|
|
951
|
+
return str(self.stream)
|
|
952
|
+
|
|
953
|
+
@staticmethod
|
|
954
|
+
def is_stream_position_in_range(
|
|
955
|
+
low: Optional["AbstractMultiWriterStreamToken"],
|
|
956
|
+
high: Optional["AbstractMultiWriterStreamToken"],
|
|
957
|
+
instance_name: Optional[str],
|
|
958
|
+
pos: int,
|
|
959
|
+
) -> bool:
|
|
960
|
+
"""Checks if a given persisted position is between the two given tokens.
|
|
961
|
+
|
|
962
|
+
If `instance_name` is None then the row was persisted before multi
|
|
963
|
+
writer support.
|
|
964
|
+
"""
|
|
965
|
+
|
|
966
|
+
if low:
|
|
967
|
+
if instance_name:
|
|
968
|
+
low_stream = low.instance_map.get(instance_name, low.stream)
|
|
969
|
+
else:
|
|
970
|
+
low_stream = low.stream
|
|
971
|
+
|
|
972
|
+
if pos <= low_stream:
|
|
973
|
+
return False
|
|
974
|
+
|
|
975
|
+
if high:
|
|
976
|
+
if instance_name:
|
|
977
|
+
high_stream = high.instance_map.get(instance_name, high.stream)
|
|
978
|
+
else:
|
|
979
|
+
high_stream = high.stream
|
|
980
|
+
|
|
981
|
+
if high_stream < pos:
|
|
982
|
+
return False
|
|
983
|
+
|
|
984
|
+
return True
|
|
985
|
+
|
|
986
|
+
def __str__(self) -> str:
|
|
987
|
+
instances = ", ".join(f"{k}: {v}" for k, v in sorted(self.instance_map.items()))
|
|
988
|
+
return (
|
|
989
|
+
f"MultiWriterStreamToken(stream: {self.stream}, instances: {{{instances}}})"
|
|
990
|
+
)
|
|
991
|
+
|
|
992
|
+
|
|
993
|
+
class StreamKeyType(Enum):
|
|
994
|
+
"""Known stream types.
|
|
995
|
+
|
|
996
|
+
A stream is a list of entities ordered by an incrementing "stream token".
|
|
997
|
+
"""
|
|
998
|
+
|
|
999
|
+
ROOM = "room_key"
|
|
1000
|
+
PRESENCE = "presence_key"
|
|
1001
|
+
TYPING = "typing_key"
|
|
1002
|
+
RECEIPT = "receipt_key"
|
|
1003
|
+
ACCOUNT_DATA = "account_data_key"
|
|
1004
|
+
PUSH_RULES = "push_rules_key"
|
|
1005
|
+
TO_DEVICE = "to_device_key"
|
|
1006
|
+
DEVICE_LIST = "device_list_key"
|
|
1007
|
+
UN_PARTIAL_STATED_ROOMS = "un_partial_stated_rooms_key"
|
|
1008
|
+
THREAD_SUBSCRIPTIONS = "thread_subscriptions_key"
|
|
1009
|
+
|
|
1010
|
+
|
|
1011
|
+
@attr.s(slots=True, frozen=True, auto_attribs=True)
|
|
1012
|
+
class StreamToken:
|
|
1013
|
+
"""A collection of keys joined together by underscores in the following
|
|
1014
|
+
order and which represent the position in their respective streams.
|
|
1015
|
+
|
|
1016
|
+
ex. `s2633508_17_338_6732159_1082514_541479_274711_265584_1_379_4242`
|
|
1017
|
+
1. `room_key`: `s2633508` which is a `RoomStreamToken`
|
|
1018
|
+
- `RoomStreamToken`'s can also look like `t426-2633508` or `m56~2.58~3.59`
|
|
1019
|
+
- See the docstring for `RoomStreamToken` for more details.
|
|
1020
|
+
2. `presence_key`: `17`
|
|
1021
|
+
3. `typing_key`: `338`
|
|
1022
|
+
4. `receipt_key`: `6732159`
|
|
1023
|
+
5. `account_data_key`: `1082514`
|
|
1024
|
+
6. `push_rules_key`: `541479`
|
|
1025
|
+
7. `to_device_key`: `274711`
|
|
1026
|
+
8. `device_list_key`: `265584`
|
|
1027
|
+
9. `groups_key`: `1` (note that this key is now unused)
|
|
1028
|
+
10. `un_partial_stated_rooms_key`: `379`
|
|
1029
|
+
11. `thread_subscriptions_key`: 4242
|
|
1030
|
+
|
|
1031
|
+
You can see how many of these keys correspond to the various
|
|
1032
|
+
fields in a "/sync" response:
|
|
1033
|
+
```json
|
|
1034
|
+
{
|
|
1035
|
+
"next_batch": "s12_4_0_1_1_1_1_4_1_1",
|
|
1036
|
+
"presence": {
|
|
1037
|
+
"events": []
|
|
1038
|
+
},
|
|
1039
|
+
"device_lists": {
|
|
1040
|
+
"changed": []
|
|
1041
|
+
},
|
|
1042
|
+
"rooms": {
|
|
1043
|
+
"join": {
|
|
1044
|
+
"!QrZlfIDQLNLdZHqTnt:hs1": {
|
|
1045
|
+
"timeline": {
|
|
1046
|
+
"events": [],
|
|
1047
|
+
"prev_batch": "s10_4_0_1_1_1_1_4_1_1",
|
|
1048
|
+
"limited": false
|
|
1049
|
+
},
|
|
1050
|
+
"state": {
|
|
1051
|
+
"events": []
|
|
1052
|
+
},
|
|
1053
|
+
"account_data": {
|
|
1054
|
+
"events": []
|
|
1055
|
+
},
|
|
1056
|
+
"ephemeral": {
|
|
1057
|
+
"events": []
|
|
1058
|
+
}
|
|
1059
|
+
}
|
|
1060
|
+
}
|
|
1061
|
+
}
|
|
1062
|
+
}
|
|
1063
|
+
```
|
|
1064
|
+
|
|
1065
|
+
---
|
|
1066
|
+
|
|
1067
|
+
For caching purposes, `StreamToken`s and by extension, all their attributes,
|
|
1068
|
+
must be hashable.
|
|
1069
|
+
"""
|
|
1070
|
+
|
|
1071
|
+
room_key: RoomStreamToken = attr.ib(
|
|
1072
|
+
validator=attr.validators.instance_of(RoomStreamToken)
|
|
1073
|
+
)
|
|
1074
|
+
presence_key: int
|
|
1075
|
+
typing_key: int
|
|
1076
|
+
receipt_key: MultiWriterStreamToken = attr.ib(
|
|
1077
|
+
validator=attr.validators.instance_of(MultiWriterStreamToken)
|
|
1078
|
+
)
|
|
1079
|
+
account_data_key: int
|
|
1080
|
+
push_rules_key: int
|
|
1081
|
+
to_device_key: int
|
|
1082
|
+
device_list_key: MultiWriterStreamToken = attr.ib(
|
|
1083
|
+
validator=attr.validators.instance_of(MultiWriterStreamToken)
|
|
1084
|
+
)
|
|
1085
|
+
# Note that the groups key is no longer used and may have bogus values.
|
|
1086
|
+
groups_key: int
|
|
1087
|
+
un_partial_stated_rooms_key: int
|
|
1088
|
+
thread_subscriptions_key: int
|
|
1089
|
+
|
|
1090
|
+
_SEPARATOR = "_"
|
|
1091
|
+
START: ClassVar["StreamToken"]
|
|
1092
|
+
|
|
1093
|
+
@classmethod
|
|
1094
|
+
@cancellable
|
|
1095
|
+
async def from_string(cls, store: "DataStore", string: str) -> "StreamToken":
|
|
1096
|
+
"""
|
|
1097
|
+
Creates a RoomStreamToken from its textual representation.
|
|
1098
|
+
"""
|
|
1099
|
+
try:
|
|
1100
|
+
keys = string.split(cls._SEPARATOR)
|
|
1101
|
+
while len(keys) < len(attr.fields(cls)):
|
|
1102
|
+
# i.e. old token from before receipt_key
|
|
1103
|
+
keys.append("0")
|
|
1104
|
+
|
|
1105
|
+
(
|
|
1106
|
+
room_key,
|
|
1107
|
+
presence_key,
|
|
1108
|
+
typing_key,
|
|
1109
|
+
receipt_key,
|
|
1110
|
+
account_data_key,
|
|
1111
|
+
push_rules_key,
|
|
1112
|
+
to_device_key,
|
|
1113
|
+
device_list_key,
|
|
1114
|
+
groups_key,
|
|
1115
|
+
un_partial_stated_rooms_key,
|
|
1116
|
+
thread_subscriptions_key,
|
|
1117
|
+
) = keys
|
|
1118
|
+
|
|
1119
|
+
return cls(
|
|
1120
|
+
room_key=await RoomStreamToken.parse(store, room_key),
|
|
1121
|
+
presence_key=int(presence_key),
|
|
1122
|
+
typing_key=int(typing_key),
|
|
1123
|
+
receipt_key=await MultiWriterStreamToken.parse(store, receipt_key),
|
|
1124
|
+
account_data_key=int(account_data_key),
|
|
1125
|
+
push_rules_key=int(push_rules_key),
|
|
1126
|
+
to_device_key=int(to_device_key),
|
|
1127
|
+
device_list_key=await MultiWriterStreamToken.parse(
|
|
1128
|
+
store, device_list_key
|
|
1129
|
+
),
|
|
1130
|
+
groups_key=int(groups_key),
|
|
1131
|
+
un_partial_stated_rooms_key=int(un_partial_stated_rooms_key),
|
|
1132
|
+
thread_subscriptions_key=int(thread_subscriptions_key),
|
|
1133
|
+
)
|
|
1134
|
+
except CancelledError:
|
|
1135
|
+
raise
|
|
1136
|
+
except Exception:
|
|
1137
|
+
raise SynapseError(400, "Invalid stream token")
|
|
1138
|
+
|
|
1139
|
+
async def to_string(self, store: "DataStore") -> str:
|
|
1140
|
+
return self._SEPARATOR.join(
|
|
1141
|
+
[
|
|
1142
|
+
await self.room_key.to_string(store),
|
|
1143
|
+
str(self.presence_key),
|
|
1144
|
+
str(self.typing_key),
|
|
1145
|
+
await self.receipt_key.to_string(store),
|
|
1146
|
+
str(self.account_data_key),
|
|
1147
|
+
str(self.push_rules_key),
|
|
1148
|
+
str(self.to_device_key),
|
|
1149
|
+
await self.device_list_key.to_string(store),
|
|
1150
|
+
# Note that the groups key is no longer used, but it is still
|
|
1151
|
+
# serialized so that there will not be confusion in the future
|
|
1152
|
+
# if additional tokens are added.
|
|
1153
|
+
str(self.groups_key),
|
|
1154
|
+
str(self.un_partial_stated_rooms_key),
|
|
1155
|
+
str(self.thread_subscriptions_key),
|
|
1156
|
+
]
|
|
1157
|
+
)
|
|
1158
|
+
|
|
1159
|
+
@property
|
|
1160
|
+
def room_stream_id(self) -> int:
|
|
1161
|
+
return self.room_key.stream
|
|
1162
|
+
|
|
1163
|
+
def copy_and_advance(self, key: StreamKeyType, new_value: Any) -> "StreamToken":
|
|
1164
|
+
"""Advance the given key in the token to a new value if and only if the
|
|
1165
|
+
new value is after the old value.
|
|
1166
|
+
|
|
1167
|
+
:raises TypeError: if `key` is not the one of the keys tracked by a StreamToken.
|
|
1168
|
+
"""
|
|
1169
|
+
if key == StreamKeyType.ROOM:
|
|
1170
|
+
new_token = self.copy_and_replace(
|
|
1171
|
+
StreamKeyType.ROOM, self.room_key.copy_and_advance(new_value)
|
|
1172
|
+
)
|
|
1173
|
+
return new_token
|
|
1174
|
+
elif key == StreamKeyType.RECEIPT:
|
|
1175
|
+
new_token = self.copy_and_replace(
|
|
1176
|
+
StreamKeyType.RECEIPT, self.receipt_key.copy_and_advance(new_value)
|
|
1177
|
+
)
|
|
1178
|
+
return new_token
|
|
1179
|
+
elif key == StreamKeyType.DEVICE_LIST:
|
|
1180
|
+
new_token = self.copy_and_replace(
|
|
1181
|
+
StreamKeyType.DEVICE_LIST,
|
|
1182
|
+
self.device_list_key.copy_and_advance(new_value),
|
|
1183
|
+
)
|
|
1184
|
+
return new_token
|
|
1185
|
+
|
|
1186
|
+
new_token = self.copy_and_replace(key, new_value)
|
|
1187
|
+
new_id = new_token.get_field(key)
|
|
1188
|
+
old_id = self.get_field(key)
|
|
1189
|
+
|
|
1190
|
+
if old_id < new_id:
|
|
1191
|
+
return new_token
|
|
1192
|
+
else:
|
|
1193
|
+
return self
|
|
1194
|
+
|
|
1195
|
+
def copy_and_replace(self, key: StreamKeyType, new_value: Any) -> "StreamToken":
|
|
1196
|
+
return attr.evolve(self, **{key.value: new_value})
|
|
1197
|
+
|
|
1198
|
+
@overload
|
|
1199
|
+
def get_field(self, key: Literal[StreamKeyType.ROOM]) -> RoomStreamToken: ...
|
|
1200
|
+
|
|
1201
|
+
@overload
|
|
1202
|
+
def get_field(
|
|
1203
|
+
self,
|
|
1204
|
+
key: Literal[
|
|
1205
|
+
StreamKeyType.RECEIPT,
|
|
1206
|
+
StreamKeyType.DEVICE_LIST,
|
|
1207
|
+
],
|
|
1208
|
+
) -> MultiWriterStreamToken: ...
|
|
1209
|
+
|
|
1210
|
+
@overload
|
|
1211
|
+
def get_field(
|
|
1212
|
+
self,
|
|
1213
|
+
key: Literal[
|
|
1214
|
+
StreamKeyType.ACCOUNT_DATA,
|
|
1215
|
+
StreamKeyType.PRESENCE,
|
|
1216
|
+
StreamKeyType.PUSH_RULES,
|
|
1217
|
+
StreamKeyType.TO_DEVICE,
|
|
1218
|
+
StreamKeyType.TYPING,
|
|
1219
|
+
StreamKeyType.UN_PARTIAL_STATED_ROOMS,
|
|
1220
|
+
StreamKeyType.THREAD_SUBSCRIPTIONS,
|
|
1221
|
+
],
|
|
1222
|
+
) -> int: ...
|
|
1223
|
+
|
|
1224
|
+
@overload
|
|
1225
|
+
def get_field(
|
|
1226
|
+
self, key: StreamKeyType
|
|
1227
|
+
) -> Union[int, RoomStreamToken, MultiWriterStreamToken]: ...
|
|
1228
|
+
|
|
1229
|
+
def get_field(
|
|
1230
|
+
self, key: StreamKeyType
|
|
1231
|
+
) -> Union[int, RoomStreamToken, MultiWriterStreamToken]:
|
|
1232
|
+
"""Returns the stream ID for the given key."""
|
|
1233
|
+
return getattr(self, key.value)
|
|
1234
|
+
|
|
1235
|
+
def is_before_or_eq(self, other_token: "StreamToken") -> bool:
|
|
1236
|
+
"""Wether this token is before the other token, i.e. every constituent
|
|
1237
|
+
part is before the other.
|
|
1238
|
+
|
|
1239
|
+
Essentially it is `self <= other`.
|
|
1240
|
+
|
|
1241
|
+
Note: if `self.is_before_or_eq(other_token) is False` then that does not
|
|
1242
|
+
imply that the reverse is True.
|
|
1243
|
+
"""
|
|
1244
|
+
|
|
1245
|
+
for _, key in StreamKeyType.__members__.items():
|
|
1246
|
+
if key == StreamKeyType.TYPING:
|
|
1247
|
+
# Typing stream is allowed to "reset", and so comparisons don't
|
|
1248
|
+
# really make sense as is.
|
|
1249
|
+
# TODO: Figure out a better way of tracking resets.
|
|
1250
|
+
continue
|
|
1251
|
+
|
|
1252
|
+
self_value = self.get_field(key)
|
|
1253
|
+
other_value = other_token.get_field(key)
|
|
1254
|
+
|
|
1255
|
+
if isinstance(self_value, RoomStreamToken):
|
|
1256
|
+
assert isinstance(other_value, RoomStreamToken)
|
|
1257
|
+
if not self_value.is_before_or_eq(other_value):
|
|
1258
|
+
return False
|
|
1259
|
+
elif isinstance(self_value, MultiWriterStreamToken):
|
|
1260
|
+
assert isinstance(other_value, MultiWriterStreamToken)
|
|
1261
|
+
if not self_value.is_before_or_eq(other_value):
|
|
1262
|
+
return False
|
|
1263
|
+
else:
|
|
1264
|
+
assert isinstance(other_value, int)
|
|
1265
|
+
if self_value > other_value:
|
|
1266
|
+
return False
|
|
1267
|
+
|
|
1268
|
+
return True
|
|
1269
|
+
|
|
1270
|
+
def __str__(self) -> str:
|
|
1271
|
+
return (
|
|
1272
|
+
f"StreamToken(room: {self.room_key}, presence: {self.presence_key}, "
|
|
1273
|
+
f"typing: {self.typing_key}, receipt: {self.receipt_key}, "
|
|
1274
|
+
f"account_data: {self.account_data_key}, push_rules: {self.push_rules_key}, "
|
|
1275
|
+
f"to_device: {self.to_device_key}, device_list: {self.device_list_key}, "
|
|
1276
|
+
f"groups: {self.groups_key}, un_partial_stated_rooms: {self.un_partial_stated_rooms_key},"
|
|
1277
|
+
f"thread_subscriptions: {self.thread_subscriptions_key})"
|
|
1278
|
+
)
|
|
1279
|
+
|
|
1280
|
+
|
|
1281
|
+
StreamToken.START = StreamToken(
|
|
1282
|
+
room_key=RoomStreamToken(stream=0),
|
|
1283
|
+
presence_key=0,
|
|
1284
|
+
typing_key=0,
|
|
1285
|
+
receipt_key=MultiWriterStreamToken(stream=0),
|
|
1286
|
+
account_data_key=0,
|
|
1287
|
+
push_rules_key=0,
|
|
1288
|
+
to_device_key=0,
|
|
1289
|
+
device_list_key=MultiWriterStreamToken(stream=0),
|
|
1290
|
+
groups_key=0,
|
|
1291
|
+
un_partial_stated_rooms_key=0,
|
|
1292
|
+
thread_subscriptions_key=0,
|
|
1293
|
+
)
|
|
1294
|
+
|
|
1295
|
+
|
|
1296
|
+
@attr.s(slots=True, frozen=True, auto_attribs=True)
|
|
1297
|
+
class SlidingSyncStreamToken:
|
|
1298
|
+
"""The same as a `StreamToken`, but includes an extra field at the start for
|
|
1299
|
+
the sliding sync connection token (separated by a '/'). This is used to
|
|
1300
|
+
store per-connection state.
|
|
1301
|
+
|
|
1302
|
+
This then looks something like:
|
|
1303
|
+
5/s2633508_17_338_6732159_1082514_541479_274711_265584_1_379
|
|
1304
|
+
|
|
1305
|
+
Attributes:
|
|
1306
|
+
stream_token: Token representing the position of all the standard
|
|
1307
|
+
streams.
|
|
1308
|
+
connection_position: Token used by sliding sync to track updates to any
|
|
1309
|
+
per-connection state stored by Synapse.
|
|
1310
|
+
"""
|
|
1311
|
+
|
|
1312
|
+
stream_token: StreamToken
|
|
1313
|
+
connection_position: int
|
|
1314
|
+
|
|
1315
|
+
@staticmethod
|
|
1316
|
+
@cancellable
|
|
1317
|
+
async def from_string(store: "DataStore", string: str) -> "SlidingSyncStreamToken":
|
|
1318
|
+
"""Creates a SlidingSyncStreamToken from its textual representation."""
|
|
1319
|
+
try:
|
|
1320
|
+
connection_position_str, stream_token_str = string.split("/", 1)
|
|
1321
|
+
connection_position = int(connection_position_str)
|
|
1322
|
+
stream_token = await StreamToken.from_string(store, stream_token_str)
|
|
1323
|
+
|
|
1324
|
+
return SlidingSyncStreamToken(
|
|
1325
|
+
stream_token=stream_token,
|
|
1326
|
+
connection_position=connection_position,
|
|
1327
|
+
)
|
|
1328
|
+
except CancelledError:
|
|
1329
|
+
raise
|
|
1330
|
+
except Exception:
|
|
1331
|
+
raise SynapseError(400, "Invalid stream token")
|
|
1332
|
+
|
|
1333
|
+
async def to_string(self, store: "DataStore") -> str:
|
|
1334
|
+
"""Serializes the token to a string"""
|
|
1335
|
+
stream_token_str = await self.stream_token.to_string(store)
|
|
1336
|
+
return f"{self.connection_position}/{stream_token_str}"
|
|
1337
|
+
|
|
1338
|
+
|
|
1339
|
+
@attr.s(slots=True, frozen=True, auto_attribs=True)
|
|
1340
|
+
class ThreadSubscriptionsToken:
|
|
1341
|
+
"""
|
|
1342
|
+
Token for a position in the thread subscriptions stream.
|
|
1343
|
+
|
|
1344
|
+
Format: `ts<stream_id>`
|
|
1345
|
+
"""
|
|
1346
|
+
|
|
1347
|
+
stream_id: int
|
|
1348
|
+
|
|
1349
|
+
@staticmethod
|
|
1350
|
+
def from_string(s: str) -> "ThreadSubscriptionsToken":
|
|
1351
|
+
if not s.startswith("ts"):
|
|
1352
|
+
raise ValueError("thread subscription token must start with `ts`")
|
|
1353
|
+
|
|
1354
|
+
return ThreadSubscriptionsToken(stream_id=int(s[2:]))
|
|
1355
|
+
|
|
1356
|
+
def to_string(self) -> str:
|
|
1357
|
+
return f"ts{self.stream_id}"
|
|
1358
|
+
|
|
1359
|
+
|
|
1360
|
+
@attr.s(slots=True, frozen=True, auto_attribs=True)
|
|
1361
|
+
class PersistedPosition:
|
|
1362
|
+
"""Position of a newly persisted row with instance that persisted it."""
|
|
1363
|
+
|
|
1364
|
+
instance_name: str
|
|
1365
|
+
stream: int
|
|
1366
|
+
|
|
1367
|
+
def persisted_after(self, token: AbstractMultiWriterStreamToken) -> bool:
|
|
1368
|
+
"""
|
|
1369
|
+
Checks whether this position happened after the token
|
|
1370
|
+
"""
|
|
1371
|
+
return token.get_stream_pos_for_instance(self.instance_name) < self.stream
|
|
1372
|
+
|
|
1373
|
+
|
|
1374
|
+
@attr.s(slots=True, frozen=True, auto_attribs=True)
|
|
1375
|
+
class PersistedEventPosition(PersistedPosition):
|
|
1376
|
+
"""Position of a newly persisted event with instance that persisted it.
|
|
1377
|
+
|
|
1378
|
+
This can be used to test whether the event is persisted before or after a
|
|
1379
|
+
RoomStreamToken.
|
|
1380
|
+
"""
|
|
1381
|
+
|
|
1382
|
+
def to_room_stream_token(self) -> RoomStreamToken:
|
|
1383
|
+
"""Converts the position to a room stream token such that events
|
|
1384
|
+
persisted in the same room after this position will be after the
|
|
1385
|
+
returned `RoomStreamToken`.
|
|
1386
|
+
|
|
1387
|
+
Note: no guarantees are made about ordering w.r.t. events in other
|
|
1388
|
+
rooms.
|
|
1389
|
+
"""
|
|
1390
|
+
# Doing the naive thing satisfies the desired properties described in
|
|
1391
|
+
# the docstring.
|
|
1392
|
+
return RoomStreamToken(stream=self.stream)
|
|
1393
|
+
|
|
1394
|
+
|
|
1395
|
+
@attr.s(slots=True, frozen=True, auto_attribs=True)
|
|
1396
|
+
class ThirdPartyInstanceID:
|
|
1397
|
+
appservice_id: Optional[str]
|
|
1398
|
+
network_id: Optional[str]
|
|
1399
|
+
|
|
1400
|
+
# Deny iteration because it will bite you if you try to create a singleton
|
|
1401
|
+
# set by:
|
|
1402
|
+
# users = set(user)
|
|
1403
|
+
def __iter__(self) -> NoReturn:
|
|
1404
|
+
raise ValueError("Attempted to iterate a %s" % (type(self).__name__,))
|
|
1405
|
+
|
|
1406
|
+
# Because this class is a frozen class, it is deeply immutable.
|
|
1407
|
+
def __copy__(self) -> "ThirdPartyInstanceID":
|
|
1408
|
+
return self
|
|
1409
|
+
|
|
1410
|
+
def __deepcopy__(self, memo: dict[str, object]) -> "ThirdPartyInstanceID":
|
|
1411
|
+
return self
|
|
1412
|
+
|
|
1413
|
+
@classmethod
|
|
1414
|
+
def from_string(cls, s: str) -> "ThirdPartyInstanceID":
|
|
1415
|
+
bits = s.split("|", 2)
|
|
1416
|
+
if len(bits) != 2:
|
|
1417
|
+
raise SynapseError(400, "Invalid ID %r" % (s,))
|
|
1418
|
+
|
|
1419
|
+
return cls(appservice_id=bits[0], network_id=bits[1])
|
|
1420
|
+
|
|
1421
|
+
def to_string(self) -> str:
|
|
1422
|
+
return "%s|%s" % (self.appservice_id, self.network_id)
|
|
1423
|
+
|
|
1424
|
+
__str__ = to_string
|
|
1425
|
+
|
|
1426
|
+
|
|
1427
|
+
@attr.s(slots=True, frozen=True, auto_attribs=True)
|
|
1428
|
+
class ReadReceipt:
|
|
1429
|
+
"""Information about a read-receipt"""
|
|
1430
|
+
|
|
1431
|
+
room_id: str
|
|
1432
|
+
receipt_type: str
|
|
1433
|
+
user_id: str
|
|
1434
|
+
event_ids: list[str]
|
|
1435
|
+
thread_id: Optional[str]
|
|
1436
|
+
data: JsonDict
|
|
1437
|
+
|
|
1438
|
+
|
|
1439
|
+
@attr.s(slots=True, frozen=True, auto_attribs=True)
|
|
1440
|
+
class DeviceListUpdates:
|
|
1441
|
+
"""
|
|
1442
|
+
An object containing a diff of information regarding other users' device lists,
|
|
1443
|
+
intended for a recipient to carry out device list tracking.
|
|
1444
|
+
|
|
1445
|
+
Attributes:
|
|
1446
|
+
changed: A set of users who have updated their device identity or
|
|
1447
|
+
cross-signing keys, or who now share an encrypted room with.
|
|
1448
|
+
left: A set of users who the recipient no longer needs to track the device lists of.
|
|
1449
|
+
Typically when those users no longer share any end-to-end encryption enabled rooms.
|
|
1450
|
+
"""
|
|
1451
|
+
|
|
1452
|
+
# We need to use a factory here, otherwise `set` is not evaluated at
|
|
1453
|
+
# object instantiation, but instead at class definition instantiation.
|
|
1454
|
+
# The latter happening only once, thus always giving you the same sets
|
|
1455
|
+
# across multiple DeviceListUpdates instances.
|
|
1456
|
+
# Also see: don't define mutable default arguments.
|
|
1457
|
+
changed: set[str] = attr.ib(factory=set)
|
|
1458
|
+
left: set[str] = attr.ib(factory=set)
|
|
1459
|
+
|
|
1460
|
+
def __bool__(self) -> bool:
|
|
1461
|
+
return bool(self.changed or self.left)
|
|
1462
|
+
|
|
1463
|
+
|
|
1464
|
+
def get_verify_key_from_cross_signing_key(
|
|
1465
|
+
key_info: Mapping[str, Any],
|
|
1466
|
+
) -> tuple[str, VerifyKey]:
|
|
1467
|
+
"""Get the key ID and signedjson verify key from a cross-signing key dict
|
|
1468
|
+
|
|
1469
|
+
Args:
|
|
1470
|
+
key_info: a cross-signing key dict, which must have a "keys"
|
|
1471
|
+
property that has exactly one item in it
|
|
1472
|
+
|
|
1473
|
+
Returns:
|
|
1474
|
+
the key ID and verify key for the cross-signing key
|
|
1475
|
+
"""
|
|
1476
|
+
# make sure that a `keys` field is provided
|
|
1477
|
+
if "keys" not in key_info:
|
|
1478
|
+
raise ValueError("Invalid key")
|
|
1479
|
+
keys = key_info["keys"]
|
|
1480
|
+
# and that it contains exactly one key
|
|
1481
|
+
if len(keys) == 1:
|
|
1482
|
+
key_id, key_data = next(iter(keys.items()))
|
|
1483
|
+
return key_id, decode_verify_key_bytes(key_id, decode_base64(key_data))
|
|
1484
|
+
else:
|
|
1485
|
+
raise ValueError("Invalid key")
|
|
1486
|
+
|
|
1487
|
+
|
|
1488
|
+
@attr.s(auto_attribs=True, frozen=True, slots=True)
|
|
1489
|
+
class UserInfo:
|
|
1490
|
+
"""Holds information about a user. Result of get_user_by_id.
|
|
1491
|
+
|
|
1492
|
+
Attributes:
|
|
1493
|
+
user_id: ID of the user.
|
|
1494
|
+
appservice_id: Application service ID that created this user.
|
|
1495
|
+
consent_server_notice_sent: Version of policy documents the user has been sent.
|
|
1496
|
+
consent_version: Version of policy documents the user has consented to.
|
|
1497
|
+
consent_ts: Time the user consented
|
|
1498
|
+
creation_ts: Creation timestamp of the user.
|
|
1499
|
+
is_admin: True if the user is an admin.
|
|
1500
|
+
is_deactivated: True if the user has been deactivated.
|
|
1501
|
+
is_guest: True if the user is a guest user.
|
|
1502
|
+
is_shadow_banned: True if the user has been shadow-banned.
|
|
1503
|
+
user_type: User type (None for normal user, 'support' and 'bot' other options).
|
|
1504
|
+
approved: If the user has been "approved" to register on the server.
|
|
1505
|
+
locked: Whether the user's account has been locked
|
|
1506
|
+
suspended: Whether the user's account is currently suspended
|
|
1507
|
+
"""
|
|
1508
|
+
|
|
1509
|
+
user_id: UserID
|
|
1510
|
+
appservice_id: Optional[int]
|
|
1511
|
+
consent_server_notice_sent: Optional[str]
|
|
1512
|
+
consent_version: Optional[str]
|
|
1513
|
+
consent_ts: Optional[int]
|
|
1514
|
+
user_type: Optional[str]
|
|
1515
|
+
creation_ts: int
|
|
1516
|
+
is_admin: bool
|
|
1517
|
+
is_deactivated: bool
|
|
1518
|
+
is_guest: bool
|
|
1519
|
+
is_shadow_banned: bool
|
|
1520
|
+
approved: bool
|
|
1521
|
+
locked: bool
|
|
1522
|
+
suspended: bool
|
|
1523
|
+
|
|
1524
|
+
|
|
1525
|
+
class UserProfile(TypedDict):
|
|
1526
|
+
user_id: str
|
|
1527
|
+
display_name: Optional[str]
|
|
1528
|
+
avatar_url: Optional[str]
|
|
1529
|
+
|
|
1530
|
+
|
|
1531
|
+
@attr.s(auto_attribs=True, frozen=True, slots=True)
|
|
1532
|
+
class RetentionPolicy:
|
|
1533
|
+
min_lifetime: Optional[int] = None
|
|
1534
|
+
max_lifetime: Optional[int] = None
|
|
1535
|
+
|
|
1536
|
+
|
|
1537
|
+
class TaskStatus(str, Enum):
|
|
1538
|
+
"""Status of a scheduled task"""
|
|
1539
|
+
|
|
1540
|
+
# Task is scheduled but not active
|
|
1541
|
+
SCHEDULED = "scheduled"
|
|
1542
|
+
# Task is active and probably running, and if not
|
|
1543
|
+
# will be run on next scheduler loop run
|
|
1544
|
+
ACTIVE = "active"
|
|
1545
|
+
# Task has completed successfully
|
|
1546
|
+
COMPLETE = "complete"
|
|
1547
|
+
# Task is over and either returned a failed status, or had an exception
|
|
1548
|
+
FAILED = "failed"
|
|
1549
|
+
|
|
1550
|
+
|
|
1551
|
+
@attr.s(auto_attribs=True, frozen=True, slots=True)
|
|
1552
|
+
class ScheduledTask:
|
|
1553
|
+
"""Description of a scheduled task"""
|
|
1554
|
+
|
|
1555
|
+
# Id used to identify the task
|
|
1556
|
+
id: str
|
|
1557
|
+
# Name of the action to be run by this task
|
|
1558
|
+
action: str
|
|
1559
|
+
# Current status of this task
|
|
1560
|
+
status: TaskStatus
|
|
1561
|
+
# If the status is SCHEDULED then this represents when it should be launched,
|
|
1562
|
+
# otherwise it represents the last time this task got a change of state.
|
|
1563
|
+
# In milliseconds since epoch in system time timezone, usually UTC.
|
|
1564
|
+
timestamp: int
|
|
1565
|
+
# Optionally bind a task to some resource id for easy retrieval
|
|
1566
|
+
resource_id: Optional[str]
|
|
1567
|
+
# Optional parameters that will be passed to the function ran by the task
|
|
1568
|
+
params: Optional[JsonMapping]
|
|
1569
|
+
# Optional result that can be updated by the running task
|
|
1570
|
+
result: Optional[JsonMapping]
|
|
1571
|
+
# Optional error that should be assigned a value when the status is FAILED
|
|
1572
|
+
error: Optional[str]
|
|
1573
|
+
|
|
1574
|
+
|
|
1575
|
+
@attr.s(auto_attribs=True, frozen=True, slots=True)
|
|
1576
|
+
class EventOrderings:
|
|
1577
|
+
stream: int
|
|
1578
|
+
"""
|
|
1579
|
+
The stream_ordering of the event.
|
|
1580
|
+
Negative numbers mean the event was backfilled.
|
|
1581
|
+
"""
|
|
1582
|
+
|
|
1583
|
+
topological: int
|
|
1584
|
+
"""
|
|
1585
|
+
The topological_ordering of the event.
|
|
1586
|
+
Currently this is equivalent to the `depth` attributes of
|
|
1587
|
+
the PDU.
|
|
1588
|
+
"""
|
|
1589
|
+
|
|
1590
|
+
@staticmethod
|
|
1591
|
+
def from_event(event: "EventBase") -> "EventOrderings":
|
|
1592
|
+
"""
|
|
1593
|
+
Get the orderings from an event.
|
|
1594
|
+
|
|
1595
|
+
Preconditions:
|
|
1596
|
+
- the event must have been persisted (otherwise it won't have a stream ordering)
|
|
1597
|
+
"""
|
|
1598
|
+
stream = event.internal_metadata.stream_ordering
|
|
1599
|
+
assert stream is not None
|
|
1600
|
+
return EventOrderings(stream, event.depth)
|