Нет описания

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  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. import requests
  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. @discord_client.event
  34. async def on_ready():
  35. global guild
  36. global emojis
  37. global webhook
  38. global channel
  39. channel = discord_client.get_channel(discord_channel)
  40. guild = channel.guild
  41. for emoji in guild.emojis:
  42. if not emoji.animated:
  43. emojis[":{}:".format(emoji.name)] = "<:{}:{}>".format(emoji.name, emoji.id)
  44. else:
  45. emojis[":{}:".format(emoji.name)] = "<a:{}:{}>".format(emoji.name, emoji.id)
  46. @discord_client.event
  47. async def on_message(message):
  48. for message_id in message_delete_queue:
  49. message_delete_queue.remove(message_id)
  50. message = await channel.get_message(message_id)
  51. await message.delete()
  52. if message.author.name in unmatched_messages_cache.keys():
  53. message_id_cache[unmatched_messages_cache[message.author.name]] = message.id
  54. del unmatched_messages_cache[message.author.name]
  55. if message.author.discriminator == "0000" or message.channel.id != discord_channel: return
  56. username = message.author.name[:1] + "\u200B" + message.author.name[1:] + "#" + message.author.discriminator
  57. content = prepare_matrix_content(message)
  58. matrix_message_id = matrix_room.send_text("<{}> {}".format(username, content))['event_id']
  59. message_id_cache[message.id] = matrix_message_id
  60. @discord_client.event
  61. async def on_message_edit(before, after):
  62. if after.author.discriminator == "0000" or after.channel.id != discord_channel: return
  63. if after.content == before.content: return
  64. after.attachments = []
  65. username = after.author.name[:1] + "\u200B" + after.author.name[1:] + "#" + after.author.discriminator
  66. content = prepare_matrix_content(after)
  67. matrix_room.redact_message(message_id_cache[before.id], reason="Message Edited")
  68. message_id_cache[after.id] = matrix_room.send_text("<{}> {} (edited)".format(username, content))['event_id']
  69. @discord_client.event
  70. async def on_raw_message_delete(paylod):
  71. if payload.message_id in message_id_cache.keys():
  72. matrix_room.redact_message(message_id_cache[paylod.message_id], reason="Message Deleted")
  73. def send_webhook(username, avatar_url, content, matrix_id):
  74. unmatched_messages_cache[username] = matrix_id;
  75. payload = {}
  76. payload['content'] = content
  77. payload['avatar_url'] = avatar_url
  78. payload['username'] = username
  79. for tries in range(5):
  80. r = requests.post(webhook_url, data = payload)
  81. if r.ok:
  82. return
  83. else:
  84. print(r.status_code)
  85. def prepare_discord_content(content):
  86. content = content.replace("@everyone", "@\u200Beveryone")
  87. content = content.replace("@here", "@\u200Bhere")
  88. content = re.sub("</?del>", "~~", content)
  89. mentions = re.findall("(^|\s)(@(\w*))", content)
  90. for mention in mentions:
  91. member = guild.get_member_named(mention[2])
  92. if not member: continue
  93. content = content.replace(mention[1], member.mention)
  94. for emoji_name, emoji_id in emojis.items():
  95. if emoji_name in content:
  96. content = content.replace(emoji_name, emoji_id)
  97. return content
  98. def on_matrix_message(room, event):
  99. user = matrix_client.get_user(event['sender'])
  100. if event['type'] == "m.room.message" and not user.user_id == matrix_user_id:
  101. if event['content']['msgtype'] == "m.text":
  102. username = "{}{}".format(discord_prefix, user.get_display_name())
  103. avatar = user.get_avatar_url()
  104. content = prepare_discord_content(event['content']['body'])
  105. send_webhook(username, avatar, content, event['event_id'])
  106. if event['content']['msgtype'] in matrix_file_types:
  107. username = "{}{}".format(discord_prefix, user.get_display_name())
  108. avatar = user.get_avatar_url()
  109. content = matrix_homeserver + "/_matrix/media/v1/download/" + event['content']['url'][6:]
  110. send_webhook(username, avatar, content, event['event_id'])
  111. if event['type'] == "m.room.redaction" and not user.user_id == matrix_user_id:
  112. try:
  113. message_delete_queue.append(message_id_cache[event['redacts']])
  114. except KeyError:
  115. pass
  116. matrix_room.add_listener(on_matrix_message)
  117. matrix_client.start_listener_thread()
  118. discord_client.run(discord_token)