Accept tick to skip, refactoring
This commit is contained in:
parent
18e5cc12ff
commit
835ea35617
354
ccrole/ccrole.py
354
ccrole/ccrole.py
@ -12,16 +12,117 @@ from redbot.core.utils.mod import get_audit_reason
|
||||
log = logging.getLogger("red.fox_v3.ccrole")
|
||||
|
||||
|
||||
async def _get_roles_from_content(ctx, content):
|
||||
async def get_roles_from_content(ctx, content):
|
||||
content_list = content.split(",")
|
||||
try:
|
||||
role_list = [
|
||||
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_list
|
||||
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):
|
||||
@ -42,6 +143,52 @@ class CCRole(commands.Cog):
|
||||
"""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 None, "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 None, "Invalid answer, canceling"
|
||||
|
||||
# Either it was a reaction or None
|
||||
return role_list, None
|
||||
|
||||
@commands.guild_only()
|
||||
@commands.group()
|
||||
async def ccrole(self, ctx: commands.Context):
|
||||
@ -70,8 +217,6 @@ class CCRole(commands.Cog):
|
||||
return
|
||||
|
||||
guild = ctx.guild
|
||||
author = ctx.author
|
||||
channel = ctx.channel
|
||||
|
||||
cmd_list = self.config.guild(guild).cmdlist
|
||||
|
||||
@ -83,68 +228,49 @@ class CCRole(commands.Cog):
|
||||
)
|
||||
return
|
||||
|
||||
# Roles to add
|
||||
await ctx.send(
|
||||
"What roles should it add? (Must be **comma separated**)\n"
|
||||
"Say `None` to skip adding roles"
|
||||
)
|
||||
|
||||
def check(m):
|
||||
return m.author == author and m.channel == channel
|
||||
return m.author == ctx.author and m.channel == ctx.channel
|
||||
|
||||
try:
|
||||
answer = await self.bot.wait_for("message", timeout=120, check=check)
|
||||
except asyncio.TimeoutError:
|
||||
await ctx.send("Timed out, canceling")
|
||||
# Roles to add
|
||||
arole_list, error_message = 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,
|
||||
)
|
||||
if arole_list is None:
|
||||
await ctx.send(error_message)
|
||||
return
|
||||
|
||||
arole_list = []
|
||||
if answer.content.upper() != "NONE":
|
||||
arole_list = await _get_roles_from_content(ctx, answer.content)
|
||||
if arole_list is None:
|
||||
await ctx.send("Invalid answer, canceling")
|
||||
return
|
||||
|
||||
# Roles to remove
|
||||
await ctx.send(
|
||||
rrole_list, error_message = await self._query_for_roles(
|
||||
ctx,
|
||||
"What roles should it remove? (Must be comma separated)\n"
|
||||
"Say `None` to skip removing roles"
|
||||
"Say `None` or react with tick to skip removing roles",
|
||||
120,
|
||||
check,
|
||||
)
|
||||
try:
|
||||
answer = await self.bot.wait_for("message", timeout=120, check=check)
|
||||
except asyncio.TimeoutError:
|
||||
await ctx.send("Timed out, canceling")
|
||||
if rrole_list is None:
|
||||
await ctx.send(error_message)
|
||||
return
|
||||
|
||||
rrole_list = []
|
||||
if answer.content.upper() != "NONE":
|
||||
rrole_list = await _get_roles_from_content(ctx, answer.content)
|
||||
if rrole_list is None:
|
||||
await ctx.send("Invalid answer, canceling")
|
||||
return
|
||||
|
||||
# Roles to use
|
||||
await ctx.send(
|
||||
# Roles to allow use
|
||||
prole_list, error_message = await self._query_for_roles(
|
||||
ctx,
|
||||
"What roles are allowed to use this command? (Must be comma separated)\n"
|
||||
"Say `None` to allow all roles"
|
||||
"Say `None` or react with tick to allow all roles",
|
||||
120,
|
||||
check,
|
||||
)
|
||||
|
||||
try:
|
||||
answer = await self.bot.wait_for("message", timeout=120, check=check)
|
||||
except asyncio.TimeoutError:
|
||||
await ctx.send("Timed out, canceling")
|
||||
if prole_list is None:
|
||||
await ctx.send(error_message)
|
||||
return
|
||||
|
||||
prole_list = []
|
||||
if answer.content.upper() != "NONE":
|
||||
prole_list = await _get_roles_from_content(ctx, answer.content)
|
||||
if prole_list is None:
|
||||
await ctx.send("Invalid answer, canceling")
|
||||
return
|
||||
|
||||
# Selfrole
|
||||
await ctx.send("Is this a targeted command?(yes/no)\n"
|
||||
"No will make this a selfrole command")
|
||||
await ctx.send(
|
||||
"Is this a targeted command?(yes/no)\n" "No will make this a selfrole command"
|
||||
)
|
||||
|
||||
try:
|
||||
answer = await self.bot.wait_for("message", timeout=120, check=check)
|
||||
@ -152,12 +278,11 @@ class CCRole(commands.Cog):
|
||||
await ctx.send("Timed out, canceling")
|
||||
return
|
||||
|
||||
if answer.content.upper() in ["Y", "YES"]:
|
||||
targeted = True
|
||||
await ctx.send("This command will be **`targeted`**")
|
||||
else:
|
||||
targeted = False
|
||||
await ctx.send("This command will be **`selfrole`**")
|
||||
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(
|
||||
@ -245,9 +370,8 @@ class CCRole(commands.Cog):
|
||||
cmd_list = {k: v for k, v in cmd_list.items() if v}
|
||||
if not cmd_list:
|
||||
await ctx.send(
|
||||
"There are no custom commands in this server. Use `{}ccrole add` to start adding some.".format(
|
||||
ctx.prefix
|
||||
)
|
||||
f"There are no custom commands in this server.\n"
|
||||
f"Use `{ctx.prefix}ccrole add` to start adding some."
|
||||
)
|
||||
return
|
||||
|
||||
@ -293,14 +417,16 @@ class CCRole(commands.Cog):
|
||||
cmd = ctx.invoked_with
|
||||
cmd = cmd.lower() # Continues the proud case_insentivity tradition of ccrole
|
||||
guild = ctx.guild
|
||||
# message = ctx.message # Unneeded since switch to `on_message_without_command` from `on_command_error`
|
||||
|
||||
# Unneeded since switch to `on_message_without_command` from `on_command_error`
|
||||
# message = ctx.message
|
||||
|
||||
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 self.eval_cc(cmd, message, ctx)
|
||||
await eval_cc(cmd, message, ctx)
|
||||
else:
|
||||
log.debug(f"No custom command named {ctx.invoked_with} found")
|
||||
|
||||
@ -321,103 +447,3 @@ class CCRole(commands.Cog):
|
||||
if content.startswith(p):
|
||||
return p
|
||||
raise ValueError
|
||||
|
||||
async def eval_cc(self, 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 = self.format_cc(cmd, message, target)
|
||||
await ctx.send(out_message, allowed_mentions=discord.AllowedMentions())
|
||||
else:
|
||||
await ctx.tick()
|
||||
|
||||
def format_cc(self, cmd, message, target):
|
||||
out = cmd["text"]
|
||||
results = re.findall("{([^}]+)\}", out)
|
||||
for result in results:
|
||||
param = self.transform_parameter(result, message, target)
|
||||
out = out.replace("{" + result + "}", param)
|
||||
return out
|
||||
|
||||
def transform_parameter(self, result, message, target):
|
||||
"""
|
||||
For security reasons only specific objects are allowed
|
||||
Internals are ignored
|
||||
Copied from 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))
|
||||
|
Loading…
x
Reference in New Issue
Block a user