-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathbot.py
143 lines (114 loc) · 5.15 KB
/
bot.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
import os
import sentry_sdk
import discord
import requests
import re
import emoji
from discord.ext.commands import BadArgument
from discord.ext import commands, tasks
from datetime import datetime
from dotenv import load_dotenv
from helpers.config import DEBUG, SENTRY_KEY, VATSCA_MEMBER_ROLE, VATSIM_MEMBER_ROLE, VATSIM_SUBDIVISION, GUILD_ID, BOT_TOKEN, REACTION_ROLES, REACTION_MESSAGE_IDS, REACTION_EMOJI, ROLE_REASONS
from helpers.members import get_division_members
from helpers.message import staff_roles
from helpers import config
load_dotenv('.env')
intents = discord.Intents.all()
intents.message_content = True
bot = commands.Bot(command_prefix='/', description=config.DESCRIPTION, intents=intents, help_command=None, case_insensitive=True)
if DEBUG == False:
sentry_sdk.init(
dsn=SENTRY_KEY,
# Set traces_sample_rate to 1.0 to capture 100%
# of transactions for performance monitoring.
# We recommend adjusting this value in production.
traces_sample_rate=1.0
)
"""
Bot event that sets bots rich presence in Discord profile
"""
@bot.event
async def on_ready() -> None:
print(f'Bot started. \nUsername: {bot.user.name}. \nID: {bot.user.id}', flush=True)
try:
await bot.change_presence(activity=config.activity(), status=config.status())
await bot.tree.sync() # Sync global commands (might take up to 1 hour to reflect globally)
except BadArgument as e:
print(f'Error changing presence. Exception - {e}', flush=True)
except discord.HTTPException as e:
print(f"Failed to sync commands due to rate limiting: {e}")
@bot.event
async def on_member_update(before_update, user: discord.User):
"""
Function checks member and assigns role according to the username
:param before_update:
:param user:
:return:
"""
# Check if the nick has been changed as it's usually the only thing we care about
if(before_update.nick == user.nick):
return
guild = bot.get_guild(GUILD_ID)
vatsca_member = discord.utils.get(guild.roles, id=VATSCA_MEMBER_ROLE)
vatsim_member = discord.utils.get(guild.roles, id=VATSIM_MEMBER_ROLE)
try:
cid = re.findall(r'\d+', str(user.nick))
if len(cid) < 1:
raise ValueError
api_data = await get_division_members()
should_have_vatsca = False
for entry in api_data:
if int(entry['id']) == int(cid[0]) and str(entry["subdivision"]) == str(VATSIM_SUBDIVISION):
should_have_vatsca = True
if vatsim_member in user.roles:
if vatsca_member not in user.roles and should_have_vatsca == True:
await user.add_roles(vatsca_member)
elif vatsca_member in user.roles and should_have_vatsca == False:
await user.remove_roles(vatsca_member)
elif vatsim_member not in user.roles and vatsca_member in user.roles:
await user.remove_roles(vatsca_member)
except ValueError as e:
# This happens when a CID is not found, ignore it
print("Tried to find an ID but it threw a ValueError, not found", flush=True)
except Exception as e:
print(e, flush=True)
@bot.event
async def on_raw_reaction_add(payload):
channel = bot.get_channel(payload.channel_id)
msg = await channel.fetch_message(payload.message_id)
guild = bot.get_guild(GUILD_ID)
emojies = emoji.demojize(payload.emoji.name)
for message in REACTION_MESSAGE_IDS:
if int(msg.id) == int(message) and emojies in REACTION_EMOJI:
role = discord.utils.get(guild.roles, id=int(REACTION_ROLES[emojies]))
user = guild.get_member(payload.user_id)
if role not in user.roles:
await user.add_roles(role, reason=ROLE_REASONS['reaction_add'])
await user.send(f'You have been given the `{role.name}` role because you reacted with {payload.emoji}')
@bot.event
async def on_raw_reaction_remove(payload):
channel = bot.get_channel(payload.channel_id)
msg = await channel.fetch_message(payload.message_id)
guild = bot.get_guild(GUILD_ID)
emojies = emoji.demojize(payload.emoji.name)
for message in REACTION_MESSAGE_IDS:
if int(msg.id) == int(message) and emojies in REACTION_EMOJI:
role = discord.utils.get(guild.roles, id=int(REACTION_ROLES[emojies]))
user = guild.get_member(payload.user_id)
if role in user.roles:
await user.remove_roles(role, reason=ROLE_REASONS['reaction_remove'])
await user.send(f'You no longer have the `{role.name}` role because you removed your reaction.')
@bot.tree.error
async def on_app_command_error(interaction: discord.Interaction, error: discord.app_commands.AppCommandError):
ctx: commands.Context = await bot.get_context(interaction)
interaction._baton = ctx
print(error, flush=True)
await ctx.send(f'{error}\n\nIf you think this was a mistake, please contact the Web team', ephemeral=True)
@bot.event
async def on_connect():
await config.load_cogs(bot)
if __name__ == "__main__":
try:
bot.run(BOT_TOKEN)
except Exception as e:
print(f'Error starting the bot. Exception - {e}', flush=True)