aioappsrv/main.py

273 lines
6.8 KiB
Python
Raw Normal View History

import discord
2020-11-11 16:46:15 +01:00
import json
import logging
2020-11-12 14:22:48 +01:00
import nio
import os
2020-11-11 16:46:15 +01:00
def config_gen(config_file):
config_dict = {
"homeserver": "https://matrix.org",
"room_id": "room:matrix.org",
"username": "@name:matrix.org",
"password": "my-secret-password",
"channel_id": "channel",
"token": "my-secret-token"
}
if not os.path.exists(config_file):
with open(config_file, "w") as f:
json.dump(config_dict, f, indent=4)
print(f"Example configuration dumped to {config_file}")
exit()
with open(config_file, "r") as f:
config = json.loads(f.read())
return config
config = config_gen("config.json")
intents = discord.Intents.default()
intents.members = True
discord_client = discord.Client(intents=intents)
2020-11-12 15:42:23 +01:00
logging.basicConfig(level=logging.INFO)
2020-11-11 16:46:15 +01:00
@discord_client.event
async def on_ready():
print(f"Logged in as {discord_client.user}")
2020-11-12 14:22:48 +01:00
# Start Matrix bot
await create_matrix_client()
2020-11-11 16:46:15 +01:00
@discord_client.event
async def on_message(message):
2020-11-12 14:22:48 +01:00
# Don't respond to bots/webhooks
2020-11-11 16:46:15 +01:00
if message.author.bot:
return
2020-11-13 17:31:04 +01:00
# Replace mention/emote IDs with names
content = await process(message.content, "emote_")
content = await process(content, "mention")
2020-11-13 17:31:04 +01:00
content = f"<{message.author.name}> {content}"
# Append attachments to message
for attachment in message.attachments:
content += f"\n{attachment.url}"
2020-11-11 16:46:15 +01:00
2020-11-12 14:22:48 +01:00
if str(message.channel.id) == config["channel_id"]:
2020-11-13 17:31:04 +01:00
await message_send(content)
2020-11-12 14:22:48 +01:00
2020-11-14 14:43:25 +01:00
@discord_client.event
async def on_typing(channel, user, when):
2020-11-14 16:14:16 +01:00
channel_ = await get_channel()
2020-11-14 14:43:25 +01:00
# Don't act on bots
if user.bot:
return
if channel == channel_:
# Send typing event
await matrix_client.room_typing(config["room_id"], timeout=0)
2020-11-14 16:14:16 +01:00
async def get_channel():
channel = int(config["channel_id"])
channel = discord_client.get_channel(channel)
return channel
async def process(message, category):
# Replace emote names with emote IDs (Matrix -> Discord)
if category == "emote":
start = end = ":"
start_ = 1
# Replace emote IDs with emote names (Discord -> Matrix)
elif category == "emote_":
start = "<:"
start_ = 2
end = ">"
# Replace mentioned user IDs with names (Discord -> Matrix)
elif category == "mention":
start = "<@!"
start_ = 3
end = ">"
2020-11-12 14:22:48 +01:00
for item in message.split():
if item.startswith(start) and item.endswith(end):
item_ = item[start_:-1]
2020-11-11 16:46:15 +01:00
if category == "emote":
emote = discord.utils.get(discord_client.emojis, name=item_)
if emote is not None:
message = message.replace(item, str(emote))
elif category == "emote_":
emote_name = item_.split(":")[0]
message = message.replace(item, f":{emote_name}:")
elif category == "mention":
user = discord_client.get_user(int(item_))
message = message.replace(item, f"@{user.name}")
2020-11-12 14:22:48 +01:00
return message
2020-11-13 11:08:28 +01:00
async def webhook_send(author, avatar, message):
2020-11-14 16:14:16 +01:00
channel = await get_channel()
2020-11-12 14:22:48 +01:00
# Create webhook if it doesn't exist
hook_name = "matrix_bridge"
2020-11-14 16:14:16 +01:00
hooks = await channel.webhooks()
2020-11-11 16:46:15 +01:00
hook = discord.utils.get(hooks, name=hook_name)
if hook is None:
2020-11-14 16:14:16 +01:00
hook = await channel.create_webhook(name=hook_name)
2020-11-11 16:46:15 +01:00
2020-11-12 14:22:48 +01:00
# Replace emote names
message = await process(message, "emote")
2020-11-12 14:22:48 +01:00
2020-11-13 11:08:28 +01:00
await hook.send(username=author, avatar_url=avatar, content=message)
2020-11-11 16:46:15 +01:00
async def partial_mention(user):
2020-11-14 16:14:16 +01:00
channel = await get_channel()
# Get guild to parse member list
2020-11-14 16:14:16 +01:00
guild = channel.guild
# Remove "@"
user = user[1:]
for member in await guild.query_members(query=user):
user_mention = f"<@!{member.id}>"
return user_mention
return None
2020-11-11 16:46:15 +01:00
async def create_matrix_client():
homeserver = config["homeserver"]
username = config["username"]
password = config["password"]
2020-11-12 15:42:23 +01:00
timeout = 30000
2020-11-12 14:22:48 +01:00
global matrix_client
matrix_client = nio.AsyncClient(homeserver, username)
2020-11-12 15:42:23 +01:00
print(await matrix_client.login(password))
# Sync once before adding callback to avoid acting on old messages
await matrix_client.sync(timeout)
2020-11-13 17:31:04 +01:00
matrix_client.add_event_callback(message_callback, (nio.RoomMessageText,
nio.RoomMessageMedia))
2020-11-12 14:22:48 +01:00
2020-11-14 14:43:25 +01:00
matrix_client.add_ephemeral_callback(typing_callback, nio.EphemeralEvent)
2020-11-12 15:42:23 +01:00
# Sync forever
await matrix_client.sync_forever(timeout=timeout)
2020-11-12 14:22:48 +01:00
await matrix_client.logout()
2020-11-12 14:22:48 +01:00
await matrix_client.close()
2020-11-11 16:46:15 +01:00
2020-11-12 14:22:48 +01:00
async def message_send(message):
await matrix_client.room_send(
room_id=config["room_id"],
message_type="m.room.message",
content={
"msgtype": "m.text",
"body": message
}
)
async def message_callback(room, event):
2020-11-14 14:43:25 +01:00
# Don't act on activities in other rooms
if room.room_id != config["room_id"]:
return
message = event.body
2020-11-12 14:22:48 +01:00
if not message:
return
2020-11-12 15:42:23 +01:00
# Don't reply to ourselves
if event.sender == matrix_client.user:
return
2020-11-11 16:46:15 +01:00
2020-11-13 11:08:28 +01:00
author = event.sender[1:]
avatar = None
2020-11-13 17:31:04 +01:00
homeserver = author.split(":")[-1]
url = "https://matrix.org/_matrix/media/r0/download"
2020-11-14 13:39:44 +01:00
# Don't mention @everyone or @here
message = message.replace("@everyone", "@\u200Beveryone")
message = message.replace("@here", "@\u200Bhere")
# Replace partial mention of Discord user with ID
if message.startswith("@"):
user = message.split()[0]
user_mention = await partial_mention(user)
if user_mention is not None:
message = message.replace(user, user_mention)
2020-11-13 17:31:04 +01:00
# Get attachments
try:
attachment = event.url.split("/")[-1]
2020-11-14 16:25:37 +01:00
# Highlight attachment name
message = f"`{message}`"
2020-11-13 17:31:04 +01:00
message += f"\n{url}/{homeserver}/{attachment}"
except AttributeError:
pass
2020-11-13 11:08:28 +01:00
# Get avatar
for user in room.users.values():
if user.user_id == event.sender:
if user.avatar_url:
avatar = user.avatar_url.split("/")[-1]
2020-11-13 17:31:04 +01:00
avatar = f"{url}/{homeserver}/{avatar}"
2020-11-13 11:08:28 +01:00
break
await webhook_send(author, avatar, message)
2020-11-11 16:46:15 +01:00
2020-11-14 14:43:25 +01:00
async def typing_callback(room, event):
2020-11-14 16:14:16 +01:00
channel = await get_channel()
2020-11-14 14:43:25 +01:00
# Don't act on activities in other rooms
if room.room_id != config["room_id"]:
return
if room.typing_users:
# Don't act on ourselves
if len(room.typing_users) == 1 \
and room.typing_users[0] == matrix_client.user:
return
# Send typing event
2020-11-14 16:14:16 +01:00
async with channel.typing():
2020-11-14 14:43:25 +01:00
pass
2020-11-11 16:46:15 +01:00
def main():
2020-11-12 14:22:48 +01:00
# Start Discord bot
2020-11-11 16:46:15 +01:00
discord_client.run(config["token"])
2020-11-12 15:42:23 +01:00
if __name__ == "__main__":
main()