config.py 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. # mautrix-instagram - A Matrix-Instagram puppeting bridge.
  2. # Copyright (C) 2023 Tulir Asokan
  3. #
  4. # This program is free software: you can redistribute it and/or modify
  5. # it under the terms of the GNU Affero General Public License as published by
  6. # the Free Software Foundation, either version 3 of the License, or
  7. # (at your option) any later version.
  8. #
  9. # This program is distributed in the hope that it will be useful,
  10. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. # GNU Affero General Public License for more details.
  13. #
  14. # You should have received a copy of the GNU Affero General Public License
  15. # along with this program. If not, see <https://www.gnu.org/licenses/>.
  16. from __future__ import annotations
  17. from typing import Any, NamedTuple
  18. import os
  19. from mautrix.bridge.config import BaseBridgeConfig
  20. from mautrix.client import Client
  21. from mautrix.types import UserID
  22. from mautrix.util.config import ConfigUpdateHelper, ForbiddenDefault, ForbiddenKey
  23. Permissions = NamedTuple("Permissions", relay=bool, user=bool, admin=bool, level=str)
  24. class Config(BaseBridgeConfig):
  25. @property
  26. def forbidden_defaults(self) -> list[ForbiddenDefault]:
  27. return [
  28. *super().forbidden_defaults,
  29. ForbiddenDefault("appservice.database", "postgres://username:password@hostname/db"),
  30. ForbiddenDefault("bridge.permissions", ForbiddenKey("example.com")),
  31. ]
  32. def do_update(self, helper: ConfigUpdateHelper) -> None:
  33. super().do_update(helper)
  34. copy, copy_dict, base = helper
  35. copy("metrics.enabled")
  36. copy("metrics.listen_port")
  37. copy("instagram.device_seed")
  38. if base["instagram.device_seed"] == "generate":
  39. base["instagram.device_seed"] = self._new_token()
  40. copy("instagram.mqtt_keepalive")
  41. copy("bridge.username_template")
  42. copy("bridge.displayname_template")
  43. copy("bridge.private_chat_name_template")
  44. copy("bridge.group_chat_name_template")
  45. copy("bridge.displayname_max_length")
  46. copy("bridge.max_startup_thread_sync_count")
  47. copy("bridge.sync_with_custom_puppets")
  48. copy("bridge.sync_direct_chat_list")
  49. copy("bridge.double_puppet_server_map")
  50. copy("bridge.double_puppet_allow_discovery")
  51. copy("bridge.login_shared_secret_map")
  52. copy("bridge.federate_rooms")
  53. copy("bridge.backfill.enable_initial")
  54. copy("bridge.backfill.enable")
  55. copy("bridge.backfill.msc2716")
  56. copy("bridge.backfill.double_puppet_backfill")
  57. if "bridge.initial_chat_sync" in self:
  58. initial_chat_sync = self["bridge.initial_chat_sync"]
  59. base["bridge.backfill.max_conversations"] = self.get(
  60. "bridge.backfill.max_conversations", initial_chat_sync
  61. )
  62. else:
  63. copy("bridge.backfill.max_conversations")
  64. copy("bridge.backfill.min_sync_thread_delay")
  65. copy("bridge.backfill.unread_hours_threshold")
  66. copy("bridge.backfill.backoff.thread_list")
  67. copy("bridge.backfill.backoff.message_history")
  68. copy("bridge.backfill.incremental.max_pages")
  69. copy("bridge.backfill.incremental.max_total_pages")
  70. copy("bridge.backfill.incremental.page_delay")
  71. copy("bridge.backfill.incremental.post_batch_delay")
  72. copy("bridge.periodic_reconnect.interval")
  73. copy("bridge.periodic_reconnect.resync")
  74. copy("bridge.periodic_reconnect.always")
  75. if isinstance(self.get("bridge.private_chat_portal_meta", "default"), bool):
  76. base["bridge.private_chat_portal_meta"] = (
  77. "always" if self["bridge.private_chat_portal_meta"] else "default"
  78. )
  79. else:
  80. copy("bridge.private_chat_portal_meta")
  81. if base["bridge.private_chat_portal_meta"] not in ("default", "always", "never"):
  82. base["bridge.private_chat_portal_meta"] = "default"
  83. copy("bridge.delivery_receipts")
  84. copy("bridge.delivery_error_reports")
  85. copy("bridge.message_status_events")
  86. copy("bridge.resend_bridge_info")
  87. copy("bridge.unimportant_bridge_notices")
  88. copy("bridge.disable_bridge_notices")
  89. copy("bridge.caption_in_message")
  90. copy("bridge.bridge_notices")
  91. copy("bridge.bridge_matrix_typing")
  92. copy("bridge.provisioning.enabled")
  93. copy("bridge.provisioning.prefix")
  94. copy("bridge.provisioning.shared_secret")
  95. if base["bridge.provisioning.shared_secret"] == "generate":
  96. base["bridge.provisioning.shared_secret"] = self._new_token()
  97. copy("bridge.provisioning.segment_key")
  98. copy("bridge.provisioning.segment_user_id")
  99. copy("bridge.command_prefix")
  100. copy("bridge.get_proxy_api_url")
  101. copy("bridge.use_proxy_for_media")
  102. copy_dict("bridge.permissions")
  103. def _get_permissions(self, key: str) -> Permissions:
  104. level = self["bridge.permissions"].get(key, "")
  105. admin = level == "admin"
  106. user = level == "user" or admin
  107. relay = level == "relay" or user
  108. return Permissions(relay, user, admin, level)
  109. def get_permissions(self, mxid: UserID) -> Permissions:
  110. permissions = self["bridge.permissions"]
  111. if mxid in permissions:
  112. return self._get_permissions(mxid)
  113. _, homeserver = Client.parse_user_id(mxid)
  114. if homeserver in permissions:
  115. return self._get_permissions(homeserver)
  116. return self._get_permissions("*")