2022-07-25 02:32:07 +02:00
|
|
|
import logging
|
2022-07-21 21:27:47 +02:00
|
|
|
from os import getenv
|
2023-03-02 21:44:39 +01:00
|
|
|
from typing import Optional
|
2022-07-21 21:27:47 +02:00
|
|
|
|
2023-04-17 20:04:22 +02:00
|
|
|
import sentry_sdk as sentry
|
2023-05-31 20:07:39 +02:00
|
|
|
from interactions import (
|
2022-07-21 20:50:23 +02:00
|
|
|
Client,
|
2023-03-02 21:44:39 +01:00
|
|
|
Guild,
|
2022-07-21 20:50:23 +02:00
|
|
|
Intents,
|
|
|
|
listen,
|
|
|
|
slash_command,
|
2023-03-02 21:44:39 +01:00
|
|
|
slash_option,
|
2022-07-21 20:50:23 +02:00
|
|
|
InteractionContext,
|
2023-02-06 20:20:20 +01:00
|
|
|
AuditLogEventType,
|
|
|
|
AuditLogEntry,
|
|
|
|
RoleSelectMenu,
|
2023-03-02 21:44:39 +01:00
|
|
|
is_owner,
|
|
|
|
check,
|
2023-05-31 20:07:39 +02:00
|
|
|
OptionType,
|
2023-03-02 21:44:39 +01:00
|
|
|
GuildChannel,
|
|
|
|
DM,
|
2022-07-21 20:50:23 +02:00
|
|
|
)
|
2023-05-31 20:07:39 +02:00
|
|
|
from interactions.api import events
|
|
|
|
from interactions.models.discord.embed import (
|
2022-07-21 20:50:23 +02:00
|
|
|
Embed,
|
|
|
|
EmbedAuthor,
|
|
|
|
EmbedField,
|
|
|
|
EmbedFooter,
|
|
|
|
)
|
|
|
|
|
|
|
|
from dotenv import load_dotenv
|
2022-10-15 03:35:42 +02:00
|
|
|
from database import GuildSettings as GuildSettingsModel, JoinLeave
|
2022-07-21 20:50:23 +02:00
|
|
|
|
2023-03-02 21:44:39 +01:00
|
|
|
|
2023-02-06 20:20:20 +01:00
|
|
|
class HeimdallrClient(Client):
|
|
|
|
@listen()
|
|
|
|
async def on_ready(self):
|
|
|
|
print("------------------------------------")
|
|
|
|
print(f"Bot '{bot.user.username}' is ready!")
|
|
|
|
print(f"This bot is owned by {bot.owner}")
|
2023-03-02 21:44:39 +01:00
|
|
|
print()
|
2023-02-06 20:20:20 +01:00
|
|
|
print("------------------------------------")
|
2022-07-21 20:50:23 +02:00
|
|
|
|
2023-02-06 20:20:20 +01:00
|
|
|
for guild in bot.guilds:
|
2023-03-02 21:44:39 +01:00
|
|
|
guild_q = GuildSettingsModel.get_or_none(
|
|
|
|
GuildSettingsModel.guild_id == guild.id
|
|
|
|
)
|
2023-02-06 20:20:20 +01:00
|
|
|
if guild_q is None:
|
|
|
|
guild_q = GuildSettingsModel(guild_id=guild.id)
|
|
|
|
guild_q.save()
|
2022-07-21 20:50:23 +02:00
|
|
|
|
2023-02-06 20:20:20 +01:00
|
|
|
@listen(events.MemberAdd)
|
|
|
|
async def on_member_join(self, event: events.MemberAdd):
|
|
|
|
joinleave_q: JoinLeave
|
|
|
|
joinleave_q, _ = JoinLeave.get_or_create(guild_id=event.guild.id)
|
2022-10-15 03:35:42 +02:00
|
|
|
|
2023-02-06 20:20:20 +01:00
|
|
|
if joinleave_q.message_channel is None:
|
|
|
|
return
|
|
|
|
channel = await bot.fetch_channel(joinleave_q.message_channel)
|
2022-07-21 20:50:23 +02:00
|
|
|
|
2023-02-06 20:20:20 +01:00
|
|
|
if not joinleave_q.join_message_enabled:
|
|
|
|
return
|
2022-07-25 02:32:07 +02:00
|
|
|
|
2023-02-06 20:20:20 +01:00
|
|
|
if joinleave_q.join_message is None or joinleave_q.join_message == "":
|
2023-03-02 21:44:39 +01:00
|
|
|
await channel.send(f"{event.member.mention} has joined the server!")
|
2023-02-06 20:20:20 +01:00
|
|
|
return
|
2022-07-25 02:32:07 +02:00
|
|
|
|
2023-03-02 21:44:39 +01:00
|
|
|
await channel.send(
|
|
|
|
str(joinleave_q.join_message).format(member=event.member, guild=event.guild)
|
|
|
|
)
|
2022-07-25 02:32:07 +02:00
|
|
|
|
2023-02-06 20:20:20 +01:00
|
|
|
@listen(events.MemberRemove)
|
|
|
|
async def on_member_leave(self, event: events.MemberRemove):
|
|
|
|
joinleave_q: JoinLeave
|
|
|
|
joinleave_q, _ = JoinLeave.get_or_create(guild_id=event.guild.id)
|
2022-07-25 02:32:07 +02:00
|
|
|
|
2023-02-06 20:20:20 +01:00
|
|
|
guildsettings_q: GuildSettingsModel
|
|
|
|
guildsettings_q, _ = GuildSettingsModel.get_or_create(guild_id=event.guild.id)
|
2022-07-25 02:32:07 +02:00
|
|
|
|
2023-02-06 20:20:20 +01:00
|
|
|
entry: AuditLogEntry
|
|
|
|
async for entry in event.guild.audit_log_history(
|
2023-03-02 21:44:39 +01:00
|
|
|
action_type=AuditLogEventType.MEMBER_KICK, limit=10
|
2023-02-06 20:20:20 +01:00
|
|
|
):
|
2023-03-02 21:44:39 +01:00
|
|
|
if (
|
|
|
|
entry.target_id != event.member.id
|
|
|
|
or guildsettings_q.admin_channel is None
|
|
|
|
):
|
2023-02-06 20:20:20 +01:00
|
|
|
continue
|
2022-07-25 02:32:07 +02:00
|
|
|
|
2023-02-06 20:20:20 +01:00
|
|
|
channel = await bot.fetch_channel(guildsettings_q.admin_channel)
|
2023-03-02 21:44:39 +01:00
|
|
|
await channel.send(
|
|
|
|
f"{event.member.mention} was kicked with reason: {entry.reason}"
|
|
|
|
)
|
2023-02-06 20:20:20 +01:00
|
|
|
break
|
2022-07-25 02:32:07 +02:00
|
|
|
|
2023-02-06 20:20:20 +01:00
|
|
|
if joinleave_q.message_channel is None:
|
|
|
|
return
|
|
|
|
channel = await bot.fetch_channel(joinleave_q.message_channel)
|
2022-07-25 02:32:07 +02:00
|
|
|
|
2023-02-06 20:20:20 +01:00
|
|
|
if not joinleave_q.leave_message_enabled:
|
|
|
|
return
|
2022-07-25 02:32:07 +02:00
|
|
|
|
2023-02-06 20:20:20 +01:00
|
|
|
if joinleave_q.leave_message is None or joinleave_q.leave_message == "":
|
2023-03-02 21:44:39 +01:00
|
|
|
await channel.send(f"{event.member.mention} has left the server!")
|
2023-02-06 20:20:20 +01:00
|
|
|
return
|
2022-07-25 02:32:07 +02:00
|
|
|
|
2023-03-02 21:44:39 +01:00
|
|
|
await channel.send(
|
|
|
|
str(joinleave_q.leave_message).format(
|
|
|
|
member=event.member, guild=event.guild
|
|
|
|
)
|
|
|
|
)
|
2022-07-21 20:50:23 +02:00
|
|
|
|
2023-02-06 20:20:20 +01:00
|
|
|
@slash_command(name="ping", description="Ping the bot")
|
|
|
|
async def ping_command(self, ctx: InteractionContext):
|
2023-04-17 20:04:22 +02:00
|
|
|
with sentry.start_transaction(op="ping"):
|
|
|
|
ctx.ephemeral = True
|
|
|
|
await ctx.send("Pong!", components=RoleSelectMenu(placeholder="HONK"))
|
2022-07-21 21:27:47 +02:00
|
|
|
|
2023-02-06 20:20:20 +01:00
|
|
|
@slash_command(name="bot-info", description="Get info about the bot")
|
|
|
|
async def bot_info_command(self, ctx: InteractionContext):
|
|
|
|
await ctx.send(
|
|
|
|
ephemeral=True,
|
|
|
|
embed=Embed(
|
|
|
|
title=f"{bot.user.username}",
|
|
|
|
description=f"This bot is owned by {bot.owner}",
|
|
|
|
color=0x00FF00,
|
|
|
|
timestamp=bot.user.created_at,
|
|
|
|
url=bot.user.avatar.as_url(),
|
|
|
|
author=EmbedAuthor(
|
|
|
|
name=bot.user.username,
|
|
|
|
icon_url=bot.user.avatar.as_url(),
|
2022-07-25 02:32:07 +02:00
|
|
|
),
|
2023-02-06 20:20:20 +01:00
|
|
|
footer=EmbedFooter(
|
|
|
|
text=f"Created at {bot.user.created_at}",
|
|
|
|
icon_url=bot.user.avatar.as_url(),
|
2022-07-25 02:32:07 +02:00
|
|
|
),
|
2023-02-06 20:20:20 +01:00
|
|
|
fields=[
|
|
|
|
EmbedField(
|
|
|
|
name="**Extensions**",
|
|
|
|
value=f"{', '.join(bot.ext.keys())}",
|
|
|
|
),
|
|
|
|
EmbedField(
|
|
|
|
name="**Guilds**",
|
2023-03-02 21:44:39 +01:00
|
|
|
value="\n".join(
|
|
|
|
[g.name for g in sorted(bot.guilds, key=lambda g: g.name)]
|
|
|
|
),
|
2023-02-06 20:20:20 +01:00
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
2023-03-02 21:44:39 +01:00
|
|
|
@check(is_owner())
|
|
|
|
@slash_command(
|
|
|
|
name="owner",
|
|
|
|
description="Owner-related commands",
|
|
|
|
dm_permission=True,
|
|
|
|
# Only available in the dev server.
|
|
|
|
scopes=[
|
|
|
|
387153131378835456, # BotTestServer
|
|
|
|
],
|
|
|
|
)
|
|
|
|
async def owner_cmd(self, ctx: InteractionContext):
|
|
|
|
pass
|
|
|
|
|
|
|
|
@owner_cmd.subcommand(
|
|
|
|
sub_cmd_name="reload-module",
|
|
|
|
sub_cmd_description="Reload a bot module",
|
|
|
|
)
|
|
|
|
@slash_option(
|
|
|
|
name="module",
|
|
|
|
description="The module to reload",
|
2023-05-31 20:07:39 +02:00
|
|
|
opt_type=OptionType.STRING,
|
2023-03-02 21:44:39 +01:00
|
|
|
required=True,
|
|
|
|
)
|
|
|
|
async def owner_reload(self, ctx: InteractionContext, module: str):
|
|
|
|
await ctx.defer(ephemeral=True)
|
|
|
|
|
|
|
|
mods = [ext.extension_name for ext in bot.get_extensions(module)]
|
|
|
|
|
|
|
|
if len(mods) == 1:
|
|
|
|
module = mods[0]
|
|
|
|
elif len(mods) < 1:
|
|
|
|
await ctx.send("No modules with that name are loaded.")
|
|
|
|
return
|
|
|
|
else:
|
|
|
|
modnames = "\n".join(mods)
|
|
|
|
await ctx.send("**Found multiple matching modules:**\n" f"{modnames}")
|
|
|
|
|
|
|
|
try:
|
|
|
|
self.reload_extension(module)
|
2023-05-31 20:07:39 +02:00
|
|
|
except Exception:
|
2023-03-02 21:44:39 +01:00
|
|
|
logging.warn(f"Failed to reload '{module}'.")
|
|
|
|
await ctx.send(f"Failed to reload '{module}'.")
|
|
|
|
else:
|
|
|
|
logging.info(f"Extension '{module}' reloaded.")
|
|
|
|
await ctx.send(f"Extension '{module}' reloaded.")
|
|
|
|
|
|
|
|
async def send_message_to_admins(
|
|
|
|
self,
|
|
|
|
guild: Guild | int,
|
|
|
|
content: Optional[str] = None,
|
|
|
|
embed: Optional[Embed] = None,
|
|
|
|
embeds: list[Embed] | None = None
|
|
|
|
):
|
|
|
|
"""Send a message to the guild's admin channel or admin.
|
|
|
|
|
|
|
|
Convenience method to send a message to the guild's admin/mod channel,
|
|
|
|
or to the owner of the server if an admin channel is not set.
|
|
|
|
"""
|
|
|
|
if embeds is None and embed is not None:
|
|
|
|
embeds = [embed]
|
|
|
|
elif embeds is not None and embed is not None:
|
|
|
|
embeds.append(embed)
|
|
|
|
elif content is None and embed is None and embeds is None:
|
|
|
|
raise ValueError("At least one of `content`, `embed`, or `embeds` must be set.")
|
|
|
|
|
|
|
|
guild_id = None
|
|
|
|
match guild:
|
|
|
|
case Guild():
|
|
|
|
guild_id = guild.id
|
|
|
|
case int():
|
|
|
|
guild_id = guild
|
|
|
|
guild = await self.fetch_guild(guild_id)
|
|
|
|
case _:
|
|
|
|
raise TypeError("Argument `guild` must be of type Guild or int")
|
|
|
|
|
|
|
|
guildsettings_q: GuildSettingsModel
|
|
|
|
guildsettings_q, _ = GuildSettingsModel.get_or_create(guild_id=guild_id)
|
|
|
|
|
|
|
|
channel: Optional[GuildChannel | DM]
|
|
|
|
if guildsettings_q.admin_channel == None:
|
|
|
|
owner = await guild.fetch_owner()
|
|
|
|
channel = await owner.fetch_dm()
|
|
|
|
|
|
|
|
content = (
|
|
|
|
content
|
|
|
|
+ "\nSet an admin channel to stop receiving these messages as DMs"
|
|
|
|
if content is not None
|
|
|
|
else "Set an admin channel to stop receiving these messages as DMs"
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
channel = await guild.fetch_channel(int(guildsettings_q.admin_channel))
|
|
|
|
|
|
|
|
await channel.send(content=content, embeds=embeds)
|
|
|
|
|
|
|
|
|
|
|
|
bot = HeimdallrClient(
|
|
|
|
intents=Intents.ALL,
|
|
|
|
debug_scope=387153131378835456,
|
2023-02-06 20:20:20 +01:00
|
|
|
sync_interactions=True,
|
|
|
|
fetch_members=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2022-08-11 01:39:15 +02:00
|
|
|
def set_loglevel(level: str):
|
|
|
|
loglevel = logging.WARNING
|
2022-07-21 20:50:23 +02:00
|
|
|
|
2022-08-29 02:50:55 +02:00
|
|
|
match str(level).lower().strip():
|
|
|
|
case "d" | "debug":
|
2022-08-11 01:39:15 +02:00
|
|
|
loglevel = logging.DEBUG
|
2022-08-29 02:50:55 +02:00
|
|
|
case "i" | "info" | "information":
|
2022-08-11 01:39:15 +02:00
|
|
|
loglevel = logging.INFO
|
|
|
|
|
2022-08-29 02:50:55 +02:00
|
|
|
case "w" | "warn" | "warning":
|
2022-08-11 01:39:15 +02:00
|
|
|
loglevel = logging.WARNING
|
|
|
|
|
2022-08-29 02:50:55 +02:00
|
|
|
case "e" | "error":
|
2022-08-11 01:39:15 +02:00
|
|
|
loglevel = logging.ERROR
|
|
|
|
|
2022-08-29 02:50:55 +02:00
|
|
|
case "c" | "critical":
|
2022-08-11 01:39:15 +02:00
|
|
|
loglevel = logging.CRITICAL
|
|
|
|
|
|
|
|
case _:
|
|
|
|
loglevel = logging.WARNING
|
|
|
|
|
|
|
|
logging.basicConfig(level=loglevel)
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
2023-03-02 21:44:39 +01:00
|
|
|
load_dotenv() # Load environment variables from .env file
|
2022-08-11 01:39:15 +02:00
|
|
|
set_loglevel(getenv("HEIMDALLR_LOGLEVEL"))
|
|
|
|
|
|
|
|
# Create basic tables
|
2022-07-25 02:32:07 +02:00
|
|
|
GuildSettingsModel.create_table()
|
|
|
|
JoinLeave.create_table()
|
|
|
|
|
2022-08-11 01:39:15 +02:00
|
|
|
# Load extensions
|
2022-10-15 02:51:21 +02:00
|
|
|
bot.load_extension("heimdallr.commands.admin")
|
2022-10-15 03:35:42 +02:00
|
|
|
bot.load_extension("heimdallr.commands.gatekeep")
|
2022-10-15 02:51:21 +02:00
|
|
|
bot.load_extension("heimdallr.commands.quote")
|
|
|
|
bot.load_extension("heimdallr.commands.infractions")
|
|
|
|
bot.load_extension("heimdallr.commands.self_roles")
|
|
|
|
bot.load_extension("heimdallr.commands.polls")
|
|
|
|
bot.load_extension("heimdallr.commands.bot_messages")
|
2023-02-06 20:20:40 +01:00
|
|
|
bot.load_extension("heimdallr.commands.modmail")
|
2023-04-17 20:04:22 +02:00
|
|
|
bot.load_extension("heimdallr.commands.keyword_notify")
|
2022-07-21 20:50:23 +02:00
|
|
|
bot.start(getenv("DISCORD_TOKEN"))
|
2022-08-11 01:39:15 +02:00
|
|
|
|
2023-03-02 21:44:39 +01:00
|
|
|
|
2022-08-11 01:39:15 +02:00
|
|
|
if __name__ == "__main__":
|
2023-03-02 21:44:39 +01:00
|
|
|
main()
|