signal.py 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  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. import base64
  18. import json
  19. from mausignald.errors import UnknownIdentityKey
  20. from mausignald.types import Address, TrustLevel
  21. from mautrix.bridge.commands import SECTION_ADMIN, HelpSection, command_handler
  22. from mautrix.types import EventID
  23. from .. import portal as po, puppet as pu
  24. from .auth import make_qr, remove_extra_chars
  25. from .typehint import CommandEvent
  26. try:
  27. import PIL as _
  28. import qrcode
  29. except ImportError:
  30. qrcode = None
  31. SECTION_SIGNAL = HelpSection("Signal actions", 20, "")
  32. async def _get_puppet_from_cmd(evt: CommandEvent) -> pu.Puppet | None:
  33. if len(evt.args) == 0 or not evt.args[0].startswith("+"):
  34. await evt.reply(
  35. f"**Usage:** `$cmdprefix+sp {evt.command} <phone>` "
  36. "(enter phone number in international format)"
  37. )
  38. return None
  39. phone = "".join(evt.args).translate(remove_extra_chars)
  40. if not phone[1:].isdecimal():
  41. await evt.reply(
  42. f"**Usage:** `$cmdprefix+sp {evt.command} <phone>` "
  43. "(enter phone number in international format)"
  44. )
  45. return None
  46. puppet: pu.Puppet = await pu.Puppet.get_by_address(Address(number=phone))
  47. if not puppet.uuid and evt.sender.username:
  48. uuid = await evt.bridge.signal.find_uuid(evt.sender.username, puppet.number)
  49. if uuid:
  50. await puppet.handle_uuid_receive(uuid)
  51. return puppet
  52. def _format_safety_number(number: str) -> str:
  53. line_size = 20
  54. chunk_size = 5
  55. return "\n".join(
  56. " ".join(
  57. [
  58. number[chunk : chunk + chunk_size]
  59. for chunk in range(line, line + line_size, chunk_size)
  60. ]
  61. )
  62. for line in range(0, len(number), line_size)
  63. )
  64. def _pill(puppet: "pu.Puppet") -> str:
  65. return f"[{puppet.name}](https://matrix.to/#/{puppet.mxid})"
  66. @command_handler(
  67. needs_auth=True,
  68. management_only=False,
  69. help_section=SECTION_SIGNAL,
  70. help_text="Open a private chat portal with a specific phone number",
  71. help_args="<_phone_>",
  72. )
  73. async def pm(evt: CommandEvent) -> None:
  74. puppet = await _get_puppet_from_cmd(evt)
  75. if not puppet:
  76. return
  77. portal = await po.Portal.get_by_chat_id(
  78. puppet.address, receiver=evt.sender.username, create=True
  79. )
  80. if portal.mxid:
  81. await evt.reply(
  82. f"You already have a private chat with {puppet.name}: "
  83. f"[{portal.mxid}](https://matrix.to/#/{portal.mxid})"
  84. )
  85. await portal.main_intent.invite_user(portal.mxid, evt.sender.mxid)
  86. return
  87. await portal.create_matrix_room(evt.sender, puppet.address)
  88. await evt.reply(f"Created a portal room with {_pill(puppet)} and invited you to it")
  89. @command_handler(
  90. needs_auth=True,
  91. management_only=False,
  92. help_section=SECTION_SIGNAL,
  93. help_text="Join a Signal group with an invite link",
  94. help_args="<_link_>",
  95. )
  96. async def join(evt: CommandEvent) -> EventID:
  97. if len(evt.args) == 0:
  98. return await evt.reply("**Usage:** `$cmdprefix+sp join <invite link>`")
  99. try:
  100. resp = await evt.bridge.signal.join_group(evt.sender.username, evt.args[0])
  101. if resp.pending_admin_approval:
  102. return await evt.reply(
  103. f"Successfully requested to join {resp.title}, waiting for admin approval."
  104. )
  105. else:
  106. return await evt.reply(f"Successfully joined {resp.title}")
  107. except Exception:
  108. evt.log.exception("Error trying to join group")
  109. await evt.reply("Failed to join group (see logs for more details)")
  110. @command_handler(
  111. needs_auth=True,
  112. management_only=False,
  113. help_section=SECTION_SIGNAL,
  114. help_text="Get the invite link to the current group",
  115. )
  116. async def invite_link(evt: CommandEvent) -> EventID:
  117. if not evt.is_portal:
  118. return await evt.reply("This is not a portal room.")
  119. group = await evt.bridge.signal.get_group(
  120. evt.sender.username, evt.portal.chat_id, evt.portal.revision
  121. )
  122. if not group:
  123. await evt.reply("Failed to get group info")
  124. elif not group.invite_link:
  125. await evt.reply("Invite link not available")
  126. else:
  127. await evt.reply(group.invite_link)
  128. @command_handler(
  129. needs_auth=True,
  130. management_only=False,
  131. help_section=SECTION_SIGNAL,
  132. help_text="View the safety number of a specific user",
  133. help_args="[--qr] [_phone_]",
  134. )
  135. async def safety_number(evt: CommandEvent) -> None:
  136. show_qr = evt.args and evt.args[0].lower() == "--qr"
  137. if show_qr:
  138. if not qrcode:
  139. await evt.reply("Can't generate QR code: qrcode and/or PIL not installed")
  140. return
  141. evt.args = evt.args[1:]
  142. if len(evt.args) == 0 and evt.portal and evt.portal.is_direct:
  143. puppet = await pu.Puppet.get_by_address(evt.portal.chat_id)
  144. else:
  145. puppet = await _get_puppet_from_cmd(evt)
  146. if not puppet:
  147. return
  148. resp = await evt.bridge.signal.get_identities(evt.sender.username, puppet.address)
  149. if not resp.identities:
  150. await evt.reply(f"No identities found for {_pill(puppet)}")
  151. return
  152. most_recent = resp.identities[0]
  153. for identity in resp.identities:
  154. if identity.added > most_recent.added:
  155. most_recent = identity
  156. uuid = resp.address.uuid or "unknown"
  157. await evt.reply(
  158. f"### {puppet.name}\n\n"
  159. f"**UUID:** {uuid} \n"
  160. f"**Trust level:** {most_recent.trust_level} \n"
  161. f"**Safety number:**\n"
  162. f"```\n{_format_safety_number(most_recent.safety_number)}\n```"
  163. )
  164. if show_qr and most_recent.qr_code_data:
  165. data = base64.b64decode(most_recent.qr_code_data)
  166. content = await make_qr(evt.main_intent, data, "verification-qr.png")
  167. await evt.main_intent.send_message(evt.room_id, content)
  168. @command_handler(
  169. needs_auth=True,
  170. management_only=False,
  171. help_section=SECTION_SIGNAL,
  172. help_text="Set your Signal profile name",
  173. help_args="<_name_>",
  174. )
  175. async def set_profile_name(evt: CommandEvent) -> None:
  176. await evt.bridge.signal.set_profile(evt.sender.username, name=" ".join(evt.args))
  177. await evt.reply("Successfully updated profile name")
  178. _trust_levels = [x.value for x in TrustLevel]
  179. @command_handler(
  180. needs_auth=True,
  181. management_only=False,
  182. help_section=SECTION_SIGNAL,
  183. help_text="Mark another user's safety number as trusted",
  184. help_args="<_recipient phone_> [_level_] <_safety number_>",
  185. )
  186. async def mark_trusted(evt: CommandEvent) -> EventID:
  187. if len(evt.args) < 2:
  188. return await evt.reply(
  189. "**Usage:** `$cmdprefix+sp mark-trusted <recipient phone> [level] <safety number>`"
  190. )
  191. number = evt.args[0].translate(remove_extra_chars)
  192. remaining_args = evt.args[1:]
  193. trust_level = TrustLevel.TRUSTED_VERIFIED
  194. if len(evt.args) > 2 and evt.args[1].upper() in _trust_levels:
  195. trust_level = TrustLevel(evt.args[1])
  196. remaining_args = evt.args[2:]
  197. safety_num = "".join(remaining_args).replace("\n", "")
  198. if len(safety_num) != 60 or not safety_num.isdecimal():
  199. return await evt.reply("That doesn't look like a valid safety number")
  200. try:
  201. await evt.bridge.signal.trust(
  202. evt.sender.username,
  203. Address(number=number),
  204. safety_number=safety_num,
  205. trust_level=trust_level,
  206. )
  207. except UnknownIdentityKey as e:
  208. return await evt.reply(f"Failed to mark {number} as {trust_level.human_str}: {e}")
  209. return await evt.reply(f"Successfully marked {number} as {trust_level.human_str}")
  210. @command_handler(
  211. needs_admin=False,
  212. needs_auth=True,
  213. help_section=SECTION_SIGNAL,
  214. help_text="Sync data from Signal",
  215. )
  216. async def sync(evt: CommandEvent) -> None:
  217. await evt.sender.sync()
  218. await evt.reply("Sync complete")
  219. @command_handler(
  220. needs_admin=True,
  221. needs_auth=False,
  222. help_section=SECTION_ADMIN,
  223. help_text="Send raw requests to signald",
  224. help_args="[--user] <type> <_json_>",
  225. )
  226. async def raw(evt: CommandEvent) -> None:
  227. add_username = False
  228. while True:
  229. flag = evt.args[0].lower()
  230. if flag == "--user":
  231. add_username = True
  232. else:
  233. break
  234. evt.args = evt.args[1:]
  235. type = evt.args[0]
  236. version = "v0"
  237. if "." in type:
  238. version, type = type.split(".", 1)
  239. try:
  240. args = json.loads(" ".join(evt.args[1:]))
  241. except json.JSONDecodeError as e:
  242. await evt.reply(f"JSON decode error: {e}")
  243. return
  244. if add_username:
  245. if version == "v0" or (version == "v1" and type in ("send", "react")):
  246. args["username"] = evt.sender.username
  247. else:
  248. args["account"] = evt.sender.username
  249. if version:
  250. args["version"] = version
  251. try:
  252. resp_type, resp_data = await evt.bridge.signal._raw_request(type, **args)
  253. except Exception as e:
  254. await evt.reply(f"Error sending request: {e}")
  255. else:
  256. if resp_data is None:
  257. await evt.reply(f"Got reply `{resp_type}` with no content")
  258. else:
  259. await evt.reply(
  260. f"Got reply `{resp_type}`:\n\n```json\n{json.dumps(resp_data, indent=2)}\n```"
  261. )