signal.py 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. # mautrix-signal - A Matrix-Signal puppeting bridge
  2. # Copyright (C) 2020 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 typing import Optional
  17. import base64
  18. import json
  19. from mautrix.bridge.commands import HelpSection, command_handler, SECTION_ADMIN
  20. from mausignald.types import Address
  21. from .. import puppet as pu, portal as po
  22. from .auth import make_qr, remove_extra_chars
  23. from .typehint import CommandEvent
  24. try:
  25. import qrcode
  26. import PIL as _
  27. except ImportError:
  28. qrcode = None
  29. SECTION_SIGNAL = HelpSection("Signal actions", 20, "")
  30. async def _get_puppet_from_cmd(evt: CommandEvent) -> Optional['pu.Puppet']:
  31. if len(evt.args) == 0 or not evt.args[0].startswith("+"):
  32. await evt.reply(f"**Usage:** `$cmdprefix+sp {evt.command} <phone>` "
  33. "(enter phone number in international format)")
  34. return None
  35. phone = "".join(evt.args).translate(remove_extra_chars)
  36. if not phone[1:].isdecimal():
  37. await evt.reply(f"**Usage:** `$cmdprefix+sp {evt.command} <phone>` "
  38. "(enter phone number in international format)")
  39. return None
  40. return await pu.Puppet.get_by_address(Address(number=phone))
  41. def _format_safety_number(number: str) -> str:
  42. line_size = 20
  43. chunk_size = 5
  44. return "\n".join(" ".join([number[chunk:chunk + chunk_size]
  45. for chunk in range(line, line + line_size, chunk_size)])
  46. for line in range(0, len(number), line_size))
  47. def _pill(puppet: 'pu.Puppet') -> str:
  48. return f"[{puppet.name}](https://matrix.to/#/{puppet.mxid})"
  49. @command_handler(needs_auth=True, management_only=False, help_section=SECTION_SIGNAL,
  50. help_text="Open a private chat portal with a specific phone number",
  51. help_args="<_phone_>")
  52. async def pm(evt: CommandEvent) -> None:
  53. puppet = await _get_puppet_from_cmd(evt)
  54. if not puppet:
  55. return
  56. portal = await po.Portal.get_by_chat_id(puppet.address, receiver=evt.sender.username,
  57. create=True)
  58. if portal.mxid:
  59. await evt.reply(f"You already have a private chat with {puppet.name}: "
  60. f"[{portal.mxid}](https://matrix.to/#/{portal.mxid})")
  61. await portal.main_intent.invite_user(portal.mxid, evt.sender.mxid)
  62. return
  63. await portal.create_matrix_room(evt.sender, puppet.address)
  64. await evt.reply(f"Created a portal room with {_pill(puppet)} and invited you to it")
  65. @command_handler(needs_auth=True, management_only=False, help_section=SECTION_SIGNAL,
  66. help_text="View the safety number of a specific user",
  67. help_args="[--qr] [_phone_]")
  68. async def safety_number(evt: CommandEvent) -> None:
  69. show_qr = evt.args and evt.args[0].lower() == "--qr"
  70. if show_qr:
  71. if not qrcode:
  72. await evt.reply("Can't generate QR code: qrcode and/or PIL not installed")
  73. return
  74. evt.args = evt.args[1:]
  75. if len(evt.args) == 0 and evt.portal and evt.portal.is_direct:
  76. puppet = await pu.Puppet.get_by_address(evt.portal.chat_id)
  77. else:
  78. puppet = await _get_puppet_from_cmd(evt)
  79. if not puppet:
  80. return
  81. resp = await evt.bridge.signal.get_identities(evt.sender.username, puppet.address)
  82. if not resp.identities:
  83. await evt.reply(f"No identities found for {_pill(puppet)}")
  84. return
  85. most_recent = resp.identities[0]
  86. for identity in resp.identities:
  87. if identity.added > most_recent.added:
  88. most_recent = identity
  89. uuid = most_recent.address.uuid or "unknown"
  90. await evt.reply(f"### {puppet.name}\n\n"
  91. f"**UUID:** {uuid} \n"
  92. f"**Trust level:** {most_recent.trust_level} \n"
  93. f"**Safety number:**\n"
  94. f"```\n{_format_safety_number(most_recent.safety_number)}\n```")
  95. if show_qr and most_recent.qr_code_data:
  96. data = base64.b64decode(most_recent.qr_code_data)
  97. content = await make_qr(evt.main_intent, data, "verification-qr.png")
  98. await evt.main_intent.send_message(evt.room_id, content)
  99. @command_handler(needs_auth=True, management_only=False, help_section=SECTION_SIGNAL,
  100. help_text="Set your Signal profile name", help_args="<_name_>")
  101. async def set_profile_name(evt: CommandEvent) -> None:
  102. await evt.bridge.signal.set_profile(evt.sender.username, name=" ".join(evt.args))
  103. await evt.reply("Successfully updated profile name")
  104. @command_handler(needs_auth=True, management_only=False, help_section=SECTION_SIGNAL,
  105. help_text="Mark another user's safety number as trusted",
  106. help_args="<_recipient phone_> <_safety number_>")
  107. async def mark_trusted(evt: CommandEvent) -> None:
  108. number = evt.args[0].translate(remove_extra_chars)
  109. safety_num = "".join(evt.args[1:]).replace("\n", "")
  110. if len(safety_num) != 60 or not safety_num.isdecimal():
  111. await evt.reply("That doesn't look like a valid safety number")
  112. return
  113. msg = await evt.bridge.signal.trust(evt.sender.username, Address(number=number),
  114. fingerprint=safety_num, trust_level="TRUSTED_VERIFIED")
  115. await evt.reply(msg)
  116. @command_handler(needs_admin=False, needs_auth=True, help_section=SECTION_SIGNAL,
  117. help_text="Sync data from Signal")
  118. async def sync(evt: CommandEvent) -> None:
  119. await evt.sender.sync()
  120. await evt.reply("Sync complete")
  121. @command_handler(needs_admin=True, needs_auth=False, help_section=SECTION_ADMIN,
  122. help_text="Send raw requests to signald",
  123. help_args="[--user] <type> <_json_>")
  124. async def raw(evt: CommandEvent) -> None:
  125. add_username = False
  126. while True:
  127. flag = evt.args[0].lower()
  128. if flag == "--user":
  129. add_username = True
  130. else:
  131. break
  132. evt.args = evt.args[1:]
  133. type = evt.args[0]
  134. version = "v0"
  135. if "." in type:
  136. version, type = type.split(".", 1)
  137. try:
  138. args = json.loads(" ".join(evt.args[1:]))
  139. except json.JSONDecodeError as e:
  140. await evt.reply(f"JSON decode error: {e}")
  141. return
  142. if add_username:
  143. if version == "v0" or (version == "v1" and type in ("send", "react")):
  144. args["username"] = evt.sender.username
  145. else:
  146. args["account"] = evt.sender.username
  147. if version:
  148. args["version"] = version
  149. try:
  150. resp_type, resp_data = await evt.bridge.signal._raw_request(type, **args)
  151. except Exception as e:
  152. await evt.reply(f"Error sending request: {e}")
  153. else:
  154. if resp_data is None:
  155. await evt.reply(f"Got reply `{resp_type}` with no content")
  156. else:
  157. await evt.reply(f"Got reply `{resp_type}`:\n\n"
  158. f"```json\n{json.dumps(resp_data, indent=2)}\n```")