added first slashcommand and refactored the preview cog as result of breaking changes in slashcommand
This commit is contained in:
4
.gitignore
vendored
4
.gitignore
vendored
@@ -1,9 +1,9 @@
|
|||||||
jeevesbot/env.py
|
jeevesbot/env.py
|
||||||
jeevesbot/secret.json
|
jeevesbot/databases/*
|
||||||
|
jeevesbot/__pycache__/*
|
||||||
logs/
|
logs/
|
||||||
__init__.py
|
__init__.py
|
||||||
*.pyc
|
*.pyc
|
||||||
__pycache__/*
|
__pycache__/*
|
||||||
cogs/__pycache__/*
|
cogs/__pycache__/*
|
||||||
jeevesbot/__pycache__/*
|
|
||||||
.vscode/
|
.vscode/
|
||||||
@@ -1,4 +1,3 @@
|
|||||||
import discord
|
|
||||||
from discord.ext import commands
|
from discord.ext import commands
|
||||||
from logging import getLogger
|
from logging import getLogger
|
||||||
import typing
|
import typing
|
||||||
|
|||||||
@@ -1,69 +1,74 @@
|
|||||||
import discord
|
import discord
|
||||||
from discord.ext import commands
|
from discord.ext import commands
|
||||||
from jeevesbot import functions
|
from jeevesbot import functions, env
|
||||||
from logging import getLogger
|
from logging import getLogger
|
||||||
import re
|
import re
|
||||||
import pytube
|
import pytube
|
||||||
from pytube.exceptions import RegexMatchError
|
from pytube.exceptions import RegexMatchError, VideoUnavailable, ExtractError
|
||||||
|
|
||||||
# setup logging
|
# setup logging
|
||||||
log = getLogger(__name__)
|
log = getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
e = discord.Embed()
|
|
||||||
|
|
||||||
|
|
||||||
class Preview(commands.Cog):
|
class Preview(commands.Cog):
|
||||||
""" Ensures that high-risk channels don't display embedded links, but only gifs and youtube previews."""
|
""" Ensures that high-risk channels don't display embedded links, but only gifs and youtube previews."""
|
||||||
|
|
||||||
|
|
||||||
def __init__(self, bot):
|
def __init__(self, bot):
|
||||||
self.bot = bot
|
self.bot = bot
|
||||||
self.video_id_regex = re.compile(r'(?:youtube\.com/(?:[^/]+/.+/|(?:v|e(?:mbed)?)/|.*[?&]v=)|youtu\.be/|m\.youtube\.com/(?:watch\?v=|embed/|v/))([^"&?/ ]{11})')
|
self.video_id_regex = re.compile(r'(?:youtube\.com/(?:[^/]+/.+/|(?:v|e(?:mbed)?)/|.*[?&]v=)|youtu\.be/|m\.youtube\.com/(?:watch\?v=|embed/|v/))([^"&?/ ]{11})')
|
||||||
|
self.e = discord.Embed()
|
||||||
|
|
||||||
@commands.Cog.listener()
|
@commands.Cog.listener()
|
||||||
async def on_message(self, message):
|
async def on_message(self, message):
|
||||||
|
if message.author.bot:
|
||||||
|
return # Ignore messages from bots
|
||||||
|
|
||||||
|
|
||||||
if message.content.startswith('https://tenor.com/'):
|
if message.content.startswith('https://tenor.com/'):
|
||||||
roles = functions.checkrole(message.author.roles)
|
is_admin = message.author.guild_permissions.administrator
|
||||||
channel = functions.checkchannel(message.channel.id)
|
is_high_risk_channel = self.check_channel(message.channel.id)
|
||||||
embed_url = message.content
|
embed_url = message.content
|
||||||
follow_url = embed_url + '.gif'
|
follow_url = embed_url + '.gif'
|
||||||
full_url = await functions.resolve(follow_url)
|
full_url = await functions.resolve(follow_url)
|
||||||
gif_url = full_url.split('?')[0]
|
gif_url = full_url.split('?')[0]
|
||||||
embed = e.set_image(url=gif_url)
|
embed = self.e.set_image(url=gif_url)
|
||||||
if channel is True:
|
if is_high_risk_channel and not is_admin:
|
||||||
if roles is not True:
|
|
||||||
await message.channel.send(embed=embed)
|
await message.channel.send(embed=embed)
|
||||||
logline = (str(message.author) + ' requested a gif: ' + str(gif_url))
|
logline = (str(message.author) + ' requested a gif: ' + str(gif_url))
|
||||||
log.info(logline)
|
log.info(logline)
|
||||||
if message.content.endswith('.gif'):
|
|
||||||
roles = functions.checkrole(message.author.roles)
|
|
||||||
channel = functions.checkchannel(message.channel.id)
|
|
||||||
embed_url = message.content
|
|
||||||
embed = e.set_image(url=embed_url)
|
|
||||||
if channel is True:
|
|
||||||
if roles is not True:
|
|
||||||
await message.channel.send(embed=embed)
|
|
||||||
logline = (str(message.author) + ' requested a gif: ' + str(embed_url))
|
|
||||||
log.info(logline)
|
|
||||||
if message.content.startswith('https://giphy.com/'):
|
if message.content.startswith('https://giphy.com/'):
|
||||||
roles = functions.checkrole(message.author.roles)
|
is_admin = message.author.guild_permissions.administrator
|
||||||
channel = functions.checkchannel(message.channel.id)
|
is_high_risk_channel = self.check_channel(message.channel.id)
|
||||||
embed_url = message.content
|
embed_url = message.content
|
||||||
image_code = embed_url.split('-')[-1]
|
image_code = embed_url.split('-')[-1]
|
||||||
gif_url = 'https://media.giphy.com/media/' + image_code + '/giphy.gif'
|
gif_url = 'https://media.giphy.com/media/' + image_code + '/giphy.gif'
|
||||||
embed = e.set_image(url=gif_url)
|
embed = self.e.set_image(url=gif_url)
|
||||||
if channel is True:
|
if is_high_risk_channel and not is_admin:
|
||||||
if roles is not True:
|
|
||||||
await message.channel.send(embed=embed)
|
await message.channel.send(embed=embed)
|
||||||
logline = (str(message.author) + ' requested a gif: ' + str(gif_url))
|
logline = (str(message.author) + ' requested a gif: ' + str(gif_url))
|
||||||
log.info(logline)
|
log.info(logline)
|
||||||
if 'https://youtu' or 'https://m.youtu' or 'https://www.youtu' in message.content():
|
|
||||||
roles = functions.checkrole(message.author.roles)
|
|
||||||
channel = functions.checkchannel(message.channel.id)
|
if message.content.endswith('.gif'):
|
||||||
if channel is True:
|
is_admin = message.author.guild_permissions.administrator
|
||||||
if roles is not True:
|
is_high_risk_channel = self.check_channel(message.channel.id)
|
||||||
|
embed_url = message.content
|
||||||
|
embed = self.e.set_image(url=embed_url)
|
||||||
|
if is_high_risk_channel and not is_admin:
|
||||||
|
await message.channel.send(embed=embed)
|
||||||
|
logline = (str(message.author) + ' requested a gif: ' + str(embed_url))
|
||||||
|
log.info(logline)
|
||||||
|
|
||||||
|
|
||||||
|
if 'https://youtu' in message.content or 'https://m.youtu' in message.content or 'https://www.youtu' in message.content:
|
||||||
|
is_admin = message.author.guild_permissions.administrator
|
||||||
|
is_high_risk_channel = self.check_channel(message.channel.id)
|
||||||
|
if is_high_risk_channel and not is_admin:
|
||||||
url = message.content
|
url = message.content
|
||||||
|
try:
|
||||||
youtube = pytube.YouTube(url)
|
youtube = pytube.YouTube(url)
|
||||||
video_title = youtube.title
|
video_title = youtube.title
|
||||||
video_author = youtube.author
|
video_author = youtube.author
|
||||||
@@ -71,12 +76,19 @@ class Preview(commands.Cog):
|
|||||||
if video_id:
|
if video_id:
|
||||||
embed = discord.Embed()
|
embed = discord.Embed()
|
||||||
embed.set_image(url=f"https://img.youtube.com/vi/{video_id}/hqdefault.jpg")
|
embed.set_image(url=f"https://img.youtube.com/vi/{video_id}/hqdefault.jpg")
|
||||||
embed.set_author(name=f"{video_author}")
|
embed.set_author(name=video_author)
|
||||||
embed.add_field(name=f"", value=f"[{video_title}]({url})")
|
embed.add_field(name="", value=f"[{video_title}]({url})")
|
||||||
await message.channel.send(embed=embed)
|
await message.channel.send(embed=embed)
|
||||||
log.info(f'User {message.author} requested preview for {url}')
|
log.info(f'User {message.author} requested preview for {url}')
|
||||||
|
except (VideoUnavailable, ExtractError, KeyError) as e:
|
||||||
|
log.error(f'Error extracting YouTube video details for {url}: {e}')
|
||||||
|
await message.channel.send('Sorry, there was an error retrieving the YouTube video details.')
|
||||||
|
|
||||||
|
|
||||||
|
def check_channel(self, channel_id):
|
||||||
|
high_risk_channels = env.PREVIEWCHANNELS
|
||||||
|
return channel_id in high_risk_channels
|
||||||
|
|
||||||
def extract_video_id(self, url):
|
def extract_video_id(self, url):
|
||||||
try:
|
try:
|
||||||
match = self.video_id_regex.search(url)
|
match = self.video_id_regex.search(url)
|
||||||
|
|||||||
41
cogs/reminders.py
Normal file
41
cogs/reminders.py
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
import discord
|
||||||
|
from discord.ext import commands
|
||||||
|
from jeevesbot import env
|
||||||
|
import datetime
|
||||||
|
from jeevesbot.database import add_reminder
|
||||||
|
from logging import getLogger
|
||||||
|
|
||||||
|
|
||||||
|
# setup logging
|
||||||
|
log = getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class Reminders(commands.Cog):
|
||||||
|
""" Reminder command"""
|
||||||
|
def __init__(self, bot):
|
||||||
|
self.bot = bot
|
||||||
|
|
||||||
|
|
||||||
|
@discord.app_commands.command(name='remindme', description='Set a reminder - Use YY-MM-DD HH:MM:SS notation')
|
||||||
|
@discord.app_commands.guilds(discord.Object(id=env.GUILD_ID))
|
||||||
|
async def remindme(self, interaction: discord.Interaction, time: str, message: str):
|
||||||
|
try:
|
||||||
|
reminder_time = datetime.datetime.strptime(time, '%Y-%m-%d %H:%M:%S')
|
||||||
|
add_reminder(interaction.user.id, message, reminder_time.isoformat())
|
||||||
|
await interaction.response.send_message(f'Reminder set for {reminder_time}')
|
||||||
|
log.info(f'Reminder set by {interaction.user} for {reminder_time}: {message}')
|
||||||
|
except ValueError:
|
||||||
|
await interaction.response.send_message('Invalid time format. Use YYYY-MM-DD HH:MM:SS', ephemeral=True)
|
||||||
|
log.warn(f'Reminder set by {interaction.user} went wrong.')
|
||||||
|
|
||||||
|
|
||||||
|
@commands.Cog.listener()
|
||||||
|
async def on_ready(self):
|
||||||
|
log.info(f'module active')
|
||||||
|
|
||||||
|
async def setup(bot):
|
||||||
|
await bot.add_cog(Reminders(bot))
|
||||||
|
log.info(f'Added Reminders.remindme as command')
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
88
jeeves.py
88
jeeves.py
@@ -1,66 +1,108 @@
|
|||||||
#!/usr/bin/env python3.8
|
#!/usr/bin/env python3.8
|
||||||
|
|
||||||
import discord
|
import discord
|
||||||
from discord.ext import commands
|
from discord.ext import commands, tasks
|
||||||
from jeevesbot import env
|
from jeevesbot import env
|
||||||
|
from jeevesbot.database import init_db, get_due_reminders
|
||||||
import os
|
import os
|
||||||
import log
|
import log
|
||||||
import logging.config
|
import logging.config
|
||||||
from logging import getLogger
|
from logging import getLogger
|
||||||
|
import datetime
|
||||||
import asyncio
|
import asyncio
|
||||||
|
|
||||||
|
|
||||||
# setup root logger handlers
|
# Initialize the database
|
||||||
logging.config.dictConfig(log.LOGGING)
|
init_db()
|
||||||
|
|
||||||
|
|
||||||
# setup logging
|
# setup logging
|
||||||
|
logging.config.dictConfig(log.LOGGING)
|
||||||
log = getLogger(__name__)
|
log = getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
# setup discord.py bot
|
# setup discord.py bot
|
||||||
intents = discord.Intents().all()
|
intents = discord.Intents().all()
|
||||||
bot = commands.Bot(command_prefix='!', intents=intents, help_command=None)
|
intents.message_content = True
|
||||||
e = discord.Embed()
|
e = discord.Embed()
|
||||||
|
|
||||||
|
class Jeeves(commands.Bot):
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__(command_prefix='!', intents=intents, help_command=None)
|
||||||
|
self.guild_ids = [env.GUILD_ID]
|
||||||
|
|
||||||
@bot.command(name='load', hidden=True)
|
|
||||||
|
@commands.command(name='load', hidden=True)
|
||||||
@commands.has_permissions(administrator=True)
|
@commands.has_permissions(administrator=True)
|
||||||
async def load(ctx, extension):
|
async def load(self, ctx, extension):
|
||||||
bot.load_extension(f'cogs.{extension}')
|
self.load_extension(f'cogs.{extension}')
|
||||||
log.info(f'{ctx.message.author} loaded the {extension} module')
|
log.info(f'{ctx.message.author} loaded the {extension} module')
|
||||||
|
|
||||||
|
|
||||||
@bot.command(name='unload', hidden=True)
|
@commands.command(name='unload', hidden=True)
|
||||||
@commands.has_permissions(administrator=True)
|
@commands.has_permissions(administrator=True)
|
||||||
async def unload(ctx, extension):
|
async def unload(self, ctx, extension):
|
||||||
bot.unload_extension(f'cogs.{extension}')
|
self.unload_extension(f'cogs.{extension}')
|
||||||
log.info(f'{ctx.message.author} unloaded the {extension} module')
|
log.info(f'{ctx.message.author} unloaded the {extension} module')
|
||||||
|
|
||||||
|
|
||||||
@bot.command(name='reload', hidden=True)
|
@commands.command(name='reload', hidden=True)
|
||||||
@commands.has_permissions(administrator=True)
|
@commands.has_permissions(administrator=True)
|
||||||
async def reload(ctx, extension):
|
async def reload(self, ctx, extension):
|
||||||
bot.unload_extension(f'cogs.{extension}')
|
self.unload_extension(f'cogs.{extension}')
|
||||||
bot.load_extension(f'cogs.{extension}')
|
self.load_extension(f'cogs.{extension}')
|
||||||
log.info(f'{ctx.message.author} reloaded the {extension} module')
|
log.info(f'{ctx.message.author} reloaded the {extension} module')
|
||||||
|
|
||||||
|
|
||||||
async def load_extensions():
|
async def load_extensions(self):
|
||||||
for filename in os.listdir('./cogs'):
|
for filename in os.listdir('./cogs'):
|
||||||
if filename.endswith('.py'):
|
if filename.endswith('.py'):
|
||||||
await bot.load_extension(f'cogs.{filename[:-3]}')
|
await self.load_extension(f'cogs.{filename[:-3]}')
|
||||||
|
|
||||||
|
|
||||||
@bot.event
|
@tasks.loop(seconds=60)
|
||||||
async def on_ready():
|
async def check_reminders(self):
|
||||||
log.info(f'Active with ID:{bot.user.id} as {bot.user.name}')
|
now = datetime.datetime.now().isoformat()
|
||||||
|
reminders = get_due_reminders(now)
|
||||||
|
for reminder in reminders:
|
||||||
|
user = self.get_user(reminder[1])
|
||||||
|
if user:
|
||||||
|
try:
|
||||||
|
await user.send(reminder[2])
|
||||||
|
except Exception as e:
|
||||||
|
log.error(f'Error sending reminder to user {reminder[1]}: {e}')
|
||||||
|
|
||||||
|
|
||||||
|
async def on_ready(self):
|
||||||
|
log.info(f'Active with ID:{self.user.id} as {self.user.name}')
|
||||||
activity = discord.Activity(name='!help', type=discord.ActivityType.listening)
|
activity = discord.Activity(name='!help', type=discord.ActivityType.listening)
|
||||||
await bot.change_presence(activity=activity)
|
await self.change_presence(activity=activity)
|
||||||
|
# Sync commands for all guilds
|
||||||
|
for guild_id in self.guild_ids:
|
||||||
|
guild = discord.Object(id=guild_id)
|
||||||
|
try:
|
||||||
|
await self.tree.sync(guild=guild)
|
||||||
|
log.info(f'Successfully synced commands for guild {guild_id}')
|
||||||
|
except discord.errors.Forbidden as e:
|
||||||
|
log.error(f'Failed to sync commands for guild {guild_id}: {e}')
|
||||||
|
# Start the reminder check loop
|
||||||
|
if not self.check_reminders.is_running():
|
||||||
|
self.check_reminders.start()
|
||||||
|
|
||||||
|
|
||||||
|
async def on_command_error(self, ctx, error):
|
||||||
|
if isinstance(error, commands.CommandNotFound):
|
||||||
|
await ctx.send('Command not found.')
|
||||||
|
log.warning(f'Command not found: {ctx.message.content}')
|
||||||
|
else:
|
||||||
|
await ctx.send('An error occurred.')
|
||||||
|
log.error(f'An error occurred: {error}')
|
||||||
|
|
||||||
|
|
||||||
async def main():
|
async def main():
|
||||||
async with bot:
|
bot = Jeeves()
|
||||||
await load_extensions()
|
await bot.load_extensions()
|
||||||
await bot.start(env.TOKEN)
|
await bot.start(env.TOKEN)
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
asyncio.run(main())
|
asyncio.run(main())
|
||||||
|
|||||||
34
jeevesbot/database.py
Normal file
34
jeevesbot/database.py
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import sqlite3
|
||||||
|
|
||||||
|
def init_db():
|
||||||
|
conn = sqlite3.connect(r"jeevesbot/databases/reminders.db")
|
||||||
|
c = conn.cursor()
|
||||||
|
c.execute('''
|
||||||
|
CREATE TABLE IF NOT EXISTS reminders (
|
||||||
|
id INTEGER PRIMARY KEY,
|
||||||
|
user_id INTEGER,
|
||||||
|
message TEXT,
|
||||||
|
reminder_time TIMESTAMP
|
||||||
|
)
|
||||||
|
''')
|
||||||
|
conn.commit()
|
||||||
|
conn.close()
|
||||||
|
|
||||||
|
def add_reminder(user_id, message, reminder_time):
|
||||||
|
conn = sqlite3.connect(r"jeevesbot/databases/reminders.db")
|
||||||
|
c = conn.cursor()
|
||||||
|
c.execute('INSERT INTO reminders (user_id, message, reminder_time) VALUES (?, ?, ?)', (user_id, message, reminder_time))
|
||||||
|
conn.commit()
|
||||||
|
conn.close()
|
||||||
|
|
||||||
|
def get_due_reminders(current_time):
|
||||||
|
conn = sqlite3.connect(r"jeevesbot/databases/reminders.db")
|
||||||
|
c = conn.cursor()
|
||||||
|
c.execute('SELECT id, user_id, message FROM reminders WHERE reminder_time <= ?', (current_time,))
|
||||||
|
reminders = c.fetchall()
|
||||||
|
c.execute('DELETE FROM reminders WHERE reminder_time <= ?', (current_time,))
|
||||||
|
conn.commit()
|
||||||
|
conn.close()
|
||||||
|
return reminders
|
||||||
@@ -2,4 +2,5 @@
|
|||||||
|
|
||||||
TOKEN = 'discord-bot-token-here'
|
TOKEN = 'discord-bot-token-here'
|
||||||
ADMIN_ROLE = 'role-to-exclude-from-gifbot'
|
ADMIN_ROLE = 'role-to-exclude-from-gifbot'
|
||||||
PREVIEWCHANNELS = [add-channel-ids-for-bot-to-work-in]
|
GUILD_ID = 'id-of-guild'
|
||||||
|
PREVIEWCHANNELS = ['add-channel-ids-for-bot-to-work-in']
|
||||||
@@ -17,16 +17,3 @@ def roll(notation):
|
|||||||
result = int(roll)
|
result = int(roll)
|
||||||
return roll,result
|
return roll,result
|
||||||
|
|
||||||
# check if user has admin role and output True if it's the case.
|
|
||||||
def checkrole(roles):
|
|
||||||
for role in roles:
|
|
||||||
if str(role) == env.ADMIN_ROLE:
|
|
||||||
return True
|
|
||||||
|
|
||||||
# check if the source channel is in the list of channels that are watched by the bot.
|
|
||||||
def checkchannel(channelid):
|
|
||||||
if channelid in env.PREVIEWCHANNELS:
|
|
||||||
return True
|
|
||||||
else:
|
|
||||||
return False
|
|
||||||
|
|
||||||
|
|||||||
@@ -2,13 +2,14 @@
|
|||||||
aiohttp
|
aiohttp
|
||||||
async-timeout
|
async-timeout
|
||||||
dice
|
dice
|
||||||
discord.py
|
discord.py[voice]
|
||||||
docopt
|
docopt
|
||||||
multidict
|
multidict
|
||||||
pyparsing
|
pyparsing
|
||||||
typing-extensions
|
typing-extensions
|
||||||
pylint
|
pylint
|
||||||
pytube
|
pytube
|
||||||
|
sqlite3
|
||||||
|
|
||||||
# needs this version, otherwise TypeErrors will break stuff
|
# needs this version, otherwise TypeErrors will break stuff
|
||||||
yarl==1.4.2
|
yarl==1.4.2
|
||||||
|
|||||||
Reference in New Issue
Block a user