You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
Fox-V3/ccrole/ccrole.py

453 lines
15 KiB

import asyncio
import logging
import re
import discord
from discord.ext.commands.view import StringView
from redbot.core import Config, checks, commands
from redbot.core.bot import Red
from redbot.core.utils.chat_formatting import box, pagify
from redbot.core.utils.mod import get_audit_reason
7 years ago
log = logging.getLogger("red.fox_v3.ccrole")
7 years ago
async def get_roles_from_content(ctx, content):
content_list = content.split(",")
try:
role_id_list = [
discord.utils.get(ctx.guild.roles, name=role.strip(" ")).id for role in content_list
]
except (discord.HTTPException, AttributeError): # None.id is attribute error
return None
else:
return role_id_list
def transform_parameter(result, message, target):
"""
For security reasons only specific objects are allowed
Internals are ignored
Copied from redbot.cogs.customcom.CustomCommands.transform_parameter and added `target`
"""
raw_result = "{" + result + "}"
objects = {
"message": message,
"author": message.author,
"channel": message.channel,
"server": message.guild,
"guild": message.guild,
"target": target,
}
if result in objects:
return str(objects[result])
try:
first, second = result.split(".")
except ValueError:
return raw_result
if first in objects and not second.startswith("_"):
first = objects[first]
else:
return raw_result
return str(getattr(first, second, raw_result))
def format_cc(cmd, message, target):
out = cmd["text"]
results = re.findall("{([^}]+)}", out)
for result in results:
param = transform_parameter(result, message, target)
out = out.replace("{" + result + "}", param)
return out
async def eval_cc(cmd, message: discord.Message, ctx: commands.Context):
"""Does all the work"""
if cmd["proles"] and not (set(role.id for role in message.author.roles) & set(cmd["proles"])):
log.debug(f"{message.author} missing required role to execute {ctx.invoked_with}")
return # Not authorized, do nothing
if cmd["targeted"]:
view: StringView = ctx.view
view.skip_ws()
# guild: discord.Guild = ctx.guild
# print(f"Guild: {guild}")
target = view.get_quoted_word()
# print(f"Target: {target}")
if target:
# target = discord.utils.get(guild.members, mention=target)
try:
target = await commands.MemberConverter().convert(ctx, target)
except commands.BadArgument:
target = None
else:
target = None
if not target:
out_message = (
f"This custom command is targeted! @mention a target\n`"
f"{ctx.invoked_with} <target>`"
)
await ctx.send(out_message)
return
else:
target = message.author
reason = get_audit_reason(message.author)
if cmd["aroles"]:
arole_list = [
discord.utils.get(message.guild.roles, id=roleid) for roleid in cmd["aroles"]
]
try:
await target.add_roles(*arole_list, reason=reason)
except discord.Forbidden:
log.exception(f"Permission error: Unable to add roles")
await ctx.send("Permission error: Unable to add roles")
if cmd["rroles"]:
rrole_list = [
discord.utils.get(message.guild.roles, id=roleid) for roleid in cmd["rroles"]
]
try:
await target.remove_roles(*rrole_list, reason=reason)
except discord.Forbidden:
log.exception(f"Permission error: Unable to remove roles")
await ctx.send("Permission error: Unable to remove roles")
if cmd["text"] is not None:
out_message = format_cc(cmd, message, target)
await ctx.send(out_message, allowed_mentions=discord.AllowedMentions())
else:
await ctx.tick()
class CCRole(commands.Cog):
"""
Custom commands
Creates commands used to display text and adjust roles
"""
7 years ago
def __init__(self, bot: Red):
super().__init__()
7 years ago
self.bot = bot
7 years ago
self.config = Config.get_conf(self, identifier=9999114111108101)
6 years ago
default_guild = {"cmdlist": {}, "settings": {}}
7 years ago
self.config.register_guild(**default_guild)
7 years ago
async def red_delete_data_for_user(self, **kwargs):
"""Nothing to delete"""
return
async def _query_for_roles(self, ctx, message, timeout, check):
m: discord.Message = await ctx.send(message)
try:
await m.add_reaction("\N{WHITE HEAVY CHECK MARK}")
except discord.HTTPException:
log.exception("Unable to add reaction to ccrole setup message")
pass
def reaction_check(payload):
return payload.message_id == m.id and payload.user_id == ctx.author.id
tasks = [
asyncio.ensure_future(self.bot.wait_for("message", timeout=timeout, check=check)),
asyncio.ensure_future(
self.bot.wait_for("raw_reaction_add", timeout=timeout, check=reaction_check)
),
]
done, pending = await asyncio.wait(
tasks, timeout=timeout, return_when=asyncio.FIRST_COMPLETED
)
for task in pending:
task.cancel()
if len(done) == 0:
return {'success': False, 'message': "Timed out, canceling"}
payload_or_message = done.pop().result()
# try:
# answer = await self.bot.wait_for("message", timeout=timeout, check=check)
# except asyncio.TimeoutError:
# return None, "Timed out, canceling"
role_list = []
if (
isinstance(payload_or_message, discord.Message)
and payload_or_message.content.upper() != "NONE"
):
role_list = await get_roles_from_content(ctx, payload_or_message.content)
if role_list is None:
return {'success': False, 'message': "Invalid answer, canceling"}
# Either it was a reaction or None
return {'success': True, 'roles': role_list, 'message': "Success"}
6 years ago
@commands.guild_only()
@commands.group()
async def ccrole(self, ctx: commands.Context):
7 years ago
"""Custom commands management with roles
Highly customizable custom commands with role management."""
7 years ago
if not ctx.invoked_subcommand:
pass
7 years ago
7 years ago
@ccrole.command(name="add")
7 years ago
@checks.mod_or_permissions(administrator=True)
async def ccrole_add(self, ctx, command: str):
7 years ago
"""Adds a custom command with roles
When adding text, put arguments in `{}` to eval them
Options: `{author}`, `{target}`, `{server}`, `{channel}`, `{message}`"""
# TODO: Clean this up so it's not so repetitive
# The call/answer format has better options as well
# Saying "none" over and over can trigger automod actions as well
# Also, allow `ctx.tick()` instead of sending a message
7 years ago
command = command.lower()
7 years ago
if command in self.bot.all_commands:
await ctx.send("That command is already a standard command.")
7 years ago
return
7 years ago
guild = ctx.guild
7 years ago
cmd_list = self.config.guild(guild).cmdlist
7 years ago
if await cmd_list.get_raw(command, default=None):
6 years ago
await ctx.send(
"This command already exists. Delete it with `{}ccrole delete` first.".format(
ctx.prefix
)
)
return
7 years ago
def check(m):
return m.author == ctx.author and m.channel == ctx.channel
# Roles to add
results = await self._query_for_roles(
ctx,
"What roles should it add? (Must be **comma separated**)\n"
"Say `None` or react with tick to skip adding roles",
120,
check,
6 years ago
)
if not results['success']:
await ctx.send(results['message'])
return
arole_list = results['roles']
# Roles to remove
results = await self._query_for_roles(
ctx,
"What roles should it remove? (Must be comma separated)\n"
"Say `None` or react with tick to skip removing roles",
120,
check,
6 years ago
)
if not results['success']:
await ctx.send(results['message'])
return
rrole_list = results['roles']
# Roles to allow use
results = await self._query_for_roles(
ctx,
"What roles are allowed to use this command? (Must be comma separated)\n"
"Say `None` or react with tick to allow all roles",
120,
check,
6 years ago
)
if not results['success']:
await ctx.send(results['message'])
return
prole_list = results['roles']
# Selfrole
await ctx.send(
"Is this a targeted command?(yes/no)\n" "No will make this a selfrole command"
)
7 years ago
try:
6 years ago
answer = await self.bot.wait_for("message", timeout=120, check=check)
7 years ago
except asyncio.TimeoutError:
await ctx.send("Timed out, canceling")
return
targeted = answer.content.upper() in ["Y", "YES"]
await ctx.send(
"This command will be **`{}`**".format("targeted" if targeted else "selfrole")
)
# Message to send
await ctx.send(
6 years ago
"What message should the bot say when using this command?\n"
"Say `None` to send no message and just react with ✅\n"
6 years ago
"Eval Options: `{author}`, `{target}`, `{server}`, `{channel}`, `{message}`\n"
"For example: `Welcome {target.mention} to {server.name}!`"
)
7 years ago
try:
6 years ago
answer = await self.bot.wait_for("message", timeout=120, check=check)
7 years ago
except asyncio.TimeoutError:
await ctx.send("Timed out, canceling")
return
text = None
if answer.content.upper() != "NONE":
text = answer.content
# Save the command
6 years ago
out = {
"text": text,
"aroles": arole_list,
"rroles": rrole_list,
"proles": prole_list,
"targeted": targeted,
}
7 years ago
await cmd_list.set_raw(command, value=out)
7 years ago
await ctx.send("Custom Command **`{}`** successfully added".format(command))
7 years ago
@ccrole.command(name="delete")
7 years ago
@checks.mod_or_permissions(administrator=True)
async def ccrole_delete(self, ctx, command: str):
7 years ago
"""Deletes a custom command
7 years ago
7 years ago
Example:
7 years ago
`[p]ccrole delete yourcommand`"""
7 years ago
guild = ctx.guild
7 years ago
command = command.lower()
if not await self.config.guild(guild).cmdlist.get_raw(command, default=None):
7 years ago
await ctx.send("That command doesn't exist")
7 years ago
else:
await self.config.guild(guild).cmdlist.set_raw(command, value=None)
7 years ago
await ctx.send("Custom command successfully deleted.")
7 years ago
@ccrole.command(name="details", aliases=["detail"])
7 years ago
async def ccrole_details(self, ctx, command: str):
"""Provide details about passed custom command"""
guild = ctx.guild
command = command.lower()
cmd = await self.config.guild(guild).cmdlist.get_raw(command, default=None)
if cmd is None:
await ctx.send("That command doesn't exist")
return
6 years ago
embed = discord.Embed(
title=command,
description="{} custom command".format(
"Targeted" if cmd["targeted"] else "Non-Targeted"
),
)
7 years ago
def process_roles(role_list):
if not role_list:
return "None"
6 years ago
return ", ".join(
[discord.utils.get(ctx.guild.roles, id=roleid).name for roleid in role_list]
)
7 years ago
embed.add_field(name="Text", value="```{}```".format(cmd["text"]), inline=False)
embed.add_field(name="Adds Roles", value=process_roles(cmd["aroles"]), inline=False)
embed.add_field(name="Removes Roles", value=process_roles(cmd["rroles"]), inline=False)
embed.add_field(name="Role Restrictions", value=process_roles(cmd["proles"]), inline=False)
7 years ago
await ctx.send(embed=embed)
7 years ago
@ccrole.command(name="list")
async def ccrole_list(self, ctx):
7 years ago
"""Shows custom commands list"""
7 years ago
guild = ctx.guild
cmd_list = await self.config.guild(guild).cmdlist()
cmd_list = {k: v for k, v in cmd_list.items() if v}
if not cmd_list:
await ctx.send(
f"There are no custom commands in this server.\n"
f"Use `{ctx.prefix}ccrole add` to start adding some."
6 years ago
)
7 years ago
return
cmd_list = ", ".join([ctx.prefix + c for c in sorted(cmd_list.keys())])
cmd_list = "Custom commands:\n\n" + cmd_list
7 years ago
6 years ago
if (
4 years ago
len(cmd_list) < 1500
6 years ago
): # I'm allowed to have arbitrary numbers for when it's too much to dm dammit
await ctx.send(box(cmd_list))
7 years ago
else:
for page in pagify(cmd_list, delims=[" ", "\n"]):
7 years ago
await ctx.author.send(box(page))
await ctx.send("Command list DM'd")
7 years ago
@commands.Cog.listener()
async def on_message_without_command(self, message: discord.Message):
5 years ago
"""
Credit to:
https://github.com/Cog-Creators/Red-DiscordBot/blob/V3/develop/redbot/cogs/customcom/customcom.py#L508
for the message filtering
"""
# This covers message.author.bot check
if not await self.bot.message_eligible_as_command(message):
return
###########
is_private = isinstance(message.channel, discord.abc.PrivateChannel)
if is_private or len(message.content) < 2:
return
if await self.bot.cog_disabled_in_guild(self, message.guild):
return
ctx = await self.bot.get_context(message)
if ctx.prefix is None:
return
###########
# Thank you Cog-Creators
cmd = ctx.invoked_with
cmd = cmd.lower() # Continues the proud case_insentivity tradition of ccrole
guild = ctx.guild
# Unneeded since switch to `on_message_without_command` from `on_command_error`
# message = ctx.message
7 years ago
7 years ago
cmdlist = self.config.guild(guild).cmdlist
# cmd = message.content[len(prefix) :].split()[0].lower()
cmd = await cmdlist.get_raw(cmd, default=None)
if cmd is not None:
await eval_cc(cmd, message, ctx)
else:
log.debug(f"No custom command named {ctx.invoked_with} found")
7 years ago
async def get_prefix(self, message: discord.Message) -> str:
"""
Borrowed from alias cog
Tries to determine what prefix is used in a message object.
Looks to identify from longest prefix to smallest.
Will raise ValueError if no prefix is found.
:param message: Message object
:return:
"""
content = message.content
prefix_list = await self.bot.command_prefix(self.bot, message)
6 years ago
prefixes = sorted(prefix_list, key=lambda pfx: len(pfx), reverse=True)
7 years ago
for p in prefixes:
if content.startswith(p):
7 years ago
return p
7 years ago
raise ValueError