Sin descripción

app.py 4.9KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. """
  2. Copyright 2018 Alex Taber ("astronautlevel")
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. """
  13. from matrix_client.client import MatrixClient
  14. import discord
  15. from discord import Webhook, RequestsWebhookAdapter
  16. import re
  17. from config import *
  18. matrix_client = MatrixClient(matrix_homeserver)
  19. token = matrix_client.login(matrix_username, matrix_password)
  20. discord_client = discord.Client()
  21. matrix_room = matrix_client.join_room(matrix_room_id)
  22. matrix_file_types = ('m.file', 'm.image', 'm.video', 'm.audio')
  23. message_id_cache = {}
  24. unmatched_messages_cache = {}
  25. message_delete_queue = []
  26. def prepare_matrix_content(message):
  27. attachments = "\n".join([x.url for x in message.attachments])
  28. content = message.clean_content + ("\n" + attachments if attachments != "" else "")
  29. return content
  30. guild = None
  31. channel = None
  32. emojis = {}
  33. webhook = None
  34. @discord_client.event
  35. async def on_ready():
  36. global guild
  37. global emojis
  38. global webhook
  39. global channel
  40. channel = discord_client.get_channel(discord_channel)
  41. guild = channel.guild
  42. emojis = {":{}:".format(emoji.name): "<:{}:{}>".format(emoji.name, emoji.id) for emoji in guild.emojis}
  43. webhook = Webhook.from_url(webhook_url, adapter=RequestsWebhookAdapter())
  44. @discord_client.event
  45. async def on_message(message):
  46. for message_id in message_delete_queue:
  47. message_delete_queue.remove(message_id)
  48. message = await channel.get_message(message_id)
  49. await message.delete()
  50. if message.author.name in unmatched_messages_cache.keys():
  51. message_id_cache[unmatched_messages_cache[message.author.name]] = message.id
  52. del unmatched_messages_cache[message.author.name]
  53. if message.author.discriminator == "0000" or message.channel.id != discord_channel: return
  54. username = message.author.name[:1] + "\u200B" + message.author.name[1:] + "#" + message.author.discriminator
  55. content = prepare_matrix_content(message)
  56. matrix_message_id = matrix_room.send_text("<{}> {}".format(username, content))['event_id']
  57. message_id_cache[message.id] = matrix_message_id
  58. @discord_client.event
  59. async def on_message_edit(before, after):
  60. if after.author.discriminator == "0000" or after.channel.id != discord_channel: return
  61. if after.content == before.content: return
  62. after.attachments = []
  63. username = after.author.name[:1] + "\u200B" + after.author.name[1:] + "#" + after.author.discriminator
  64. content = prepare_matrix_content(after)
  65. matrix_room.redact_message(message_id_cache[before.id], reason="Message Edited")
  66. message_id_cache[after.id] = matrix_room.send_text("<{}> {} (edited)".format(username, content))['event_id']
  67. @discord_client.event
  68. async def on_raw_message_delete(paylod):
  69. if payload.message_id in message_id_cache.keys():
  70. matrix_room.redact_message(message_id_cache[paylod.message_id], reason="Message Deleted")
  71. def send_webhook(username, avatar_url, content, matrix_id):
  72. unmatched_messages_cache[username] = matrix_id;
  73. webhook.send(content=content, username=username, avatar_url=avatar_url)
  74. def prepare_discord_content(content):
  75. content = content.replace("@everyone", "@\u200Beveryone")
  76. content = content.replace("@here", "@\u200Bhere")
  77. content = re.sub("</?del>", "~~", content)
  78. mentions = re.findall("(^|\s)(@(\w*))", content)
  79. for mention in mentions:
  80. member = guild.get_member_named(mention[2])
  81. if not member: continue
  82. content = content.replace(mention[1], member.mention)
  83. for emoji_name, emoji_id in emojis.items():
  84. if emoji_name in content:
  85. content = content.replace(emoji_name, emoji_id)
  86. return content
  87. def on_matrix_message(room, event):
  88. user = matrix_client.get_user(event['sender'])
  89. if event['type'] == "m.room.message" and not user.user_id == matrix_user_id:
  90. if event['content']['msgtype'] == "m.text":
  91. username = "{}{}".format(discord_prefix, user.get_display_name())
  92. avatar = user.get_avatar_url()
  93. content = prepare_discord_content(event['content']['body'])
  94. send_webhook(username, avatar, content, event['event_id'])
  95. if event['content']['msgtype'] in matrix_file_types:
  96. username = "{}{}".format(discord_prefix, user.get_display_name())
  97. avatar = user.get_avatar_url()
  98. content = matrix_homeserver + "/_matrix/media/v1/download/" + event['content']['url'][6:]
  99. send_webhook(username, avatar, content, event['event_id'])
  100. if event['type'] == "m.room.redaction" and not user.user_id == matrix_user_id:
  101. try:
  102. message_delete_queue.append(message_id_cache[event['redacts']])
  103. except KeyError:
  104. pass
  105. matrix_room.add_listener(on_matrix_message)
  106. matrix_client.start_listener_thread()
  107. discord_client.run(discord_token)