signal.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  1. # mautrix-signal - A Matrix-Signal puppeting bridge
  2. # Copyright (C) 2022 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 TYPE_CHECKING
  18. import asyncio
  19. import logging
  20. from mausignald import SignaldClient
  21. from mausignald.types import (
  22. Address,
  23. IncomingMessage,
  24. MessageData,
  25. OfferMessageType,
  26. OwnReadReceipt,
  27. ReceiptMessage,
  28. ReceiptType,
  29. TypingAction,
  30. TypingMessage,
  31. WebsocketConnectionStateChangeEvent,
  32. )
  33. from mautrix.types import MessageType
  34. from mautrix.util.logging import TraceLogger
  35. from . import portal as po, puppet as pu, user as u
  36. from .db import Message as DBMessage
  37. if TYPE_CHECKING:
  38. from .__main__ import SignalBridge
  39. # Typing notifications seem to get resent every 10 seconds and the timeout is around 15 seconds
  40. SIGNAL_TYPING_TIMEOUT = 15000
  41. class SignalHandler(SignaldClient):
  42. log: TraceLogger = logging.getLogger("mau.signal")
  43. loop: asyncio.AbstractEventLoop
  44. data_dir: str
  45. delete_unknown_accounts: bool
  46. def __init__(self, bridge: "SignalBridge") -> None:
  47. super().__init__(bridge.config["signal.socket_path"], loop=bridge.loop)
  48. self.data_dir = bridge.config["signal.data_dir"]
  49. self.delete_unknown_accounts = bridge.config["signal.delete_unknown_accounts_on_start"]
  50. self.add_event_handler(IncomingMessage, self.on_message)
  51. self.add_event_handler(
  52. WebsocketConnectionStateChangeEvent, self.on_websocket_connection_state_change
  53. )
  54. async def on_message(self, evt: IncomingMessage) -> None:
  55. sender = await pu.Puppet.get_by_address(evt.source)
  56. user = await u.User.get_by_username(evt.account)
  57. # TODO add lots of logging
  58. if evt.data_message:
  59. await self.handle_message(user, sender, evt.data_message)
  60. if evt.typing_message:
  61. await self.handle_typing(user, sender, evt.typing_message)
  62. if evt.receipt_message:
  63. await self.handle_receipt(sender, evt.receipt_message)
  64. if evt.call_message:
  65. await self.handle_call_message(user, sender, evt)
  66. if evt.sync_message:
  67. if evt.sync_message.read_messages:
  68. await self.handle_own_receipts(sender, evt.sync_message.read_messages)
  69. if evt.sync_message.sent:
  70. await self.handle_message(
  71. user,
  72. sender,
  73. evt.sync_message.sent.message,
  74. addr_override=evt.sync_message.sent.destination,
  75. )
  76. if evt.sync_message.contacts or evt.sync_message.contacts_complete:
  77. self.log.debug("Sync message includes contacts meta, syncing contacts...")
  78. await user.sync_contacts()
  79. if evt.sync_message.groups:
  80. self.log.debug("Sync message includes groups meta, syncing groups...")
  81. await user.sync_groups()
  82. @staticmethod
  83. async def on_websocket_connection_state_change(
  84. evt: WebsocketConnectionStateChangeEvent,
  85. ) -> None:
  86. user = await u.User.get_by_username(evt.account)
  87. user.on_websocket_connection_state_change(evt)
  88. async def handle_message(
  89. self,
  90. user: u.User,
  91. sender: pu.Puppet,
  92. msg: MessageData,
  93. addr_override: Address | None = None,
  94. ) -> None:
  95. if msg.profile_key_update:
  96. self.log.debug("Ignoring profile key update")
  97. return
  98. if msg.group_v2:
  99. portal = await po.Portal.get_by_chat_id(msg.group_v2.id, create=True)
  100. elif msg.group:
  101. portal = await po.Portal.get_by_chat_id(msg.group.group_id, create=True)
  102. else:
  103. portal = await po.Portal.get_by_chat_id(
  104. addr_override or sender.address, receiver=user.username, create=True
  105. )
  106. if addr_override and not sender.is_real_user:
  107. portal.log.debug(
  108. f"Ignoring own message {msg.timestamp} as user doesn't have double puppeting "
  109. "enabled"
  110. )
  111. return
  112. assert portal
  113. if not portal.mxid:
  114. if not msg.body and not msg.attachments and not msg.sticker and not msg.group_v2:
  115. user.log.debug(
  116. f"Ignoring message {msg.timestamp},"
  117. " probably not bridgeable as there's no portal yet"
  118. )
  119. return
  120. await portal.create_matrix_room(
  121. user, msg.group_v2 or msg.group or addr_override or sender.address
  122. )
  123. if not portal.mxid:
  124. user.log.warning(
  125. f"Failed to create room for incoming message {msg.timestamp}, dropping message"
  126. )
  127. return
  128. elif msg.group_v2 and msg.group_v2.revision > portal.revision:
  129. self.log.debug(f"Got new revision of {msg.group_v2.id}, updating info")
  130. await portal.update_info(user, msg.group_v2, sender)
  131. if msg.reaction:
  132. await portal.handle_signal_reaction(sender, msg.reaction, msg.timestamp)
  133. if msg.body or msg.attachments or msg.sticker:
  134. await portal.handle_signal_message(user, sender, msg)
  135. if msg.expires_in_seconds is not None:
  136. await portal.update_expires_in_seconds(sender, msg.expires_in_seconds)
  137. if msg.group and msg.group.type == "UPDATE":
  138. await portal.update_info(user, msg.group)
  139. if msg.remote_delete:
  140. await portal.handle_signal_delete(sender, msg.remote_delete.target_sent_timestamp)
  141. @staticmethod
  142. async def handle_call_message(user: u.User, sender: pu.Puppet, msg: IncomingMessage) -> None:
  143. assert msg.call_message
  144. portal = await po.Portal.get_by_chat_id(
  145. sender.address, receiver=user.username, create=True
  146. )
  147. if not portal.mxid:
  148. # FIXME
  149. # await portal.create_matrix_room(
  150. # user, (msg.group_v2 or msg.group or addr_override or sender.address)
  151. # )
  152. # if not portal.mxid:
  153. # user.log.debug(
  154. # f"Failed to create room for incoming message {msg.timestamp},"
  155. # " dropping message"
  156. # )
  157. return
  158. msg_html = f'<a href="https://matrix.to/#/{sender.mxid}">{sender.name}</a>'
  159. if msg.call_message.offer_message:
  160. call_type = {
  161. OfferMessageType.AUDIO_CALL: "voice call",
  162. OfferMessageType.VIDEO_CALL: "video call",
  163. }.get(msg.call_message.offer_message.type, "call")
  164. msg_html += f" started a {call_type} on Signal. Use the native app to answer the call."
  165. msg_type = MessageType.TEXT
  166. elif msg.call_message.hangup_message:
  167. msg_html += " ended a call on Signal."
  168. msg_type = MessageType.NOTICE
  169. else:
  170. portal.log.debug(f"Unhandled call message. Likely an ICE message. {msg.call_message}")
  171. return
  172. await sender.intent_for(portal).send_text(portal.mxid, html=msg_html, msgtype=msg_type)
  173. @staticmethod
  174. async def handle_own_receipts(sender: pu.Puppet, receipts: list[OwnReadReceipt]) -> None:
  175. for receipt in receipts:
  176. puppet = await pu.Puppet.get_by_address(receipt.sender, create=False)
  177. if not puppet:
  178. continue
  179. message = await DBMessage.find_by_sender_timestamp(puppet.address, receipt.timestamp)
  180. if not message:
  181. continue
  182. portal = await po.Portal.get_by_mxid(message.mx_room)
  183. if not portal or (portal.is_direct and not sender.is_real_user):
  184. continue
  185. await sender.intent_for(portal).mark_read(portal.mxid, message.mxid)
  186. @staticmethod
  187. async def handle_typing(user: u.User, sender: pu.Puppet, typing: TypingMessage) -> None:
  188. if typing.group_id:
  189. portal = await po.Portal.get_by_chat_id(typing.group_id)
  190. else:
  191. portal = await po.Portal.get_by_chat_id(sender.address, receiver=user.username)
  192. if not portal or not portal.mxid:
  193. return
  194. is_typing = typing.action == TypingAction.STARTED
  195. await sender.intent_for(portal).set_typing(
  196. portal.mxid, is_typing, ignore_cache=True, timeout=SIGNAL_TYPING_TIMEOUT
  197. )
  198. @staticmethod
  199. async def handle_receipt(sender: pu.Puppet, receipt: ReceiptMessage) -> None:
  200. if receipt.type != ReceiptType.READ:
  201. return
  202. messages = await DBMessage.find_by_timestamps(receipt.timestamps)
  203. for message in messages:
  204. portal = await po.Portal.get_by_mxid(message.mx_room)
  205. await sender.intent_for(portal).mark_read(portal.mxid, message.mxid)
  206. async def start(self) -> None:
  207. await self.connect()
  208. known_usernames = set()
  209. async for user in u.User.all_logged_in():
  210. # TODO report errors to user?
  211. known_usernames.add(user.username)
  212. if await self.subscribe(user.username):
  213. asyncio.create_task(user.sync())
  214. if self.delete_unknown_accounts:
  215. self.log.debug("Checking for unknown accounts to delete")
  216. for account in await self.list_accounts():
  217. if account.account_id not in known_usernames:
  218. self.log.warning(f"Unknown account ID {account.account_id}, deleting...")
  219. await self.delete_account(account.account_id)
  220. async def stop(self) -> None:
  221. await self.disconnect()