422 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			422 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
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
 | 
						|
 | 
						|
log = logging.getLogger("red.fox_v3.ccrole")
 | 
						|
 | 
						|
 | 
						|
async def _get_roles_from_content(ctx, content):
 | 
						|
    content_list = content.split(",")
 | 
						|
    try:
 | 
						|
        role_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
 | 
						|
 | 
						|
 | 
						|
class CCRole(commands.Cog):
 | 
						|
    """
 | 
						|
    Custom commands
 | 
						|
    Creates commands used to display text and adjust roles
 | 
						|
    """
 | 
						|
 | 
						|
    def __init__(self, bot: Red):
 | 
						|
        super().__init__()
 | 
						|
        self.bot = bot
 | 
						|
        self.config = Config.get_conf(self, identifier=9999114111108101)
 | 
						|
        default_guild = {"cmdlist": {}, "settings": {}}
 | 
						|
 | 
						|
        self.config.register_guild(**default_guild)
 | 
						|
 | 
						|
    async def red_delete_data_for_user(self, **kwargs):
 | 
						|
        """Nothing to delete"""
 | 
						|
        return
 | 
						|
 | 
						|
    @commands.guild_only()
 | 
						|
    @commands.group()
 | 
						|
    async def ccrole(self, ctx: commands.Context):
 | 
						|
        """Custom commands management with roles
 | 
						|
 | 
						|
        Highly customizable custom commands with role management."""
 | 
						|
        pass
 | 
						|
 | 
						|
    @ccrole.command(name="add")
 | 
						|
    @checks.mod_or_permissions(administrator=True)
 | 
						|
    async def ccrole_add(self, ctx, command: str):
 | 
						|
        """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
 | 
						|
 | 
						|
        command = command.lower()
 | 
						|
        if command in self.bot.all_commands:
 | 
						|
            await ctx.send("That command is already a standard command.")
 | 
						|
            return
 | 
						|
 | 
						|
        guild = ctx.guild
 | 
						|
        author = ctx.author
 | 
						|
        channel = ctx.channel
 | 
						|
 | 
						|
        cmd_list = self.config.guild(guild).cmdlist
 | 
						|
 | 
						|
        if await cmd_list.get_raw(command, default=None):
 | 
						|
            await ctx.send(
 | 
						|
                "This command already exists. Delete it with `{}ccrole delete` first.".format(
 | 
						|
                    ctx.prefix
 | 
						|
                )
 | 
						|
            )
 | 
						|
            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
 | 
						|
 | 
						|
        try:
 | 
						|
            answer = await self.bot.wait_for("message", timeout=120, check=check)
 | 
						|
        except asyncio.TimeoutError:
 | 
						|
            await ctx.send("Timed out, canceling")
 | 
						|
            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(
 | 
						|
            "What roles should it remove? (Must be comma separated)\n"
 | 
						|
            "Say `None` to skip removing roles"
 | 
						|
        )
 | 
						|
        try:
 | 
						|
            answer = await self.bot.wait_for("message", timeout=120, check=check)
 | 
						|
        except asyncio.TimeoutError:
 | 
						|
            await ctx.send("Timed out, canceling")
 | 
						|
            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(
 | 
						|
            "What roles are allowed to use this command? (Must be comma separated)\n"
 | 
						|
            "Say `None` to allow all roles"
 | 
						|
        )
 | 
						|
 | 
						|
        try:
 | 
						|
            answer = await self.bot.wait_for("message", timeout=120, check=check)
 | 
						|
        except asyncio.TimeoutError:
 | 
						|
            await ctx.send("Timed out, canceling")
 | 
						|
            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"
 | 
						|
        )
 | 
						|
 | 
						|
        try:
 | 
						|
            answer = await self.bot.wait_for("message", timeout=120, check=check)
 | 
						|
        except asyncio.TimeoutError:
 | 
						|
            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`**")
 | 
						|
 | 
						|
        # Message to send
 | 
						|
        await ctx.send(
 | 
						|
            "What message should the bot say when using this command?\n"
 | 
						|
            "Say `None` to send no message and just react with ✅\n"
 | 
						|
            "Eval Options: `{author}`, `{target}`, `{server}`, `{channel}`, `{message}`\n"
 | 
						|
            "For example: `Welcome {target.mention} to {server.name}!`"
 | 
						|
        )
 | 
						|
 | 
						|
        try:
 | 
						|
            answer = await self.bot.wait_for("message", timeout=120, check=check)
 | 
						|
        except asyncio.TimeoutError:
 | 
						|
            await ctx.send("Timed out, canceling")
 | 
						|
            return
 | 
						|
 | 
						|
        text = None
 | 
						|
        if answer.content.upper() != "NONE":
 | 
						|
            text = answer.content
 | 
						|
 | 
						|
        # Save the command
 | 
						|
 | 
						|
        out = {
 | 
						|
            "text": text,
 | 
						|
            "aroles": arole_list,
 | 
						|
            "rroles": rrole_list,
 | 
						|
            "proles": prole_list,
 | 
						|
            "targeted": targeted,
 | 
						|
        }
 | 
						|
 | 
						|
        await cmd_list.set_raw(command, value=out)
 | 
						|
 | 
						|
        await ctx.send("Custom Command **`{}`** successfully added".format(command))
 | 
						|
 | 
						|
    @ccrole.command(name="delete")
 | 
						|
    @checks.mod_or_permissions(administrator=True)
 | 
						|
    async def ccrole_delete(self, ctx, command: str):
 | 
						|
        """Deletes a custom command
 | 
						|
 | 
						|
        Example:
 | 
						|
        `[p]ccrole delete yourcommand`"""
 | 
						|
        guild = ctx.guild
 | 
						|
        command = command.lower()
 | 
						|
        if not await self.config.guild(guild).cmdlist.get_raw(command, default=None):
 | 
						|
            await ctx.send("That command doesn't exist")
 | 
						|
        else:
 | 
						|
            await self.config.guild(guild).cmdlist.set_raw(command, value=None)
 | 
						|
            await ctx.send("Custom command successfully deleted.")
 | 
						|
 | 
						|
    @ccrole.command(name="details", aliases=["detail"])
 | 
						|
    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
 | 
						|
 | 
						|
        embed = discord.Embed(
 | 
						|
            title=command,
 | 
						|
            description="{} custom command".format(
 | 
						|
                "Targeted" if cmd["targeted"] else "Non-Targeted"
 | 
						|
            ),
 | 
						|
        )
 | 
						|
 | 
						|
        def process_roles(role_list):
 | 
						|
            if not role_list:
 | 
						|
                return "None"
 | 
						|
            return ", ".join(
 | 
						|
                discord.utils.get(ctx.guild.roles, id=roleid).name for roleid in role_list
 | 
						|
            )
 | 
						|
 | 
						|
        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)
 | 
						|
 | 
						|
        await ctx.send(embed=embed)
 | 
						|
 | 
						|
    @ccrole.command(name="list")
 | 
						|
    async def ccrole_list(self, ctx):
 | 
						|
        """Shows custom commands list"""
 | 
						|
        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(
 | 
						|
                "There are no custom commands in this server. Use `{}ccrole add` to start adding some.".format(
 | 
						|
                    ctx.prefix
 | 
						|
                )
 | 
						|
            )
 | 
						|
            return
 | 
						|
 | 
						|
        cmd_list = ", ".join(ctx.prefix + c for c in sorted(cmd_list.keys()))
 | 
						|
        cmd_list = "Custom commands:\n\n" + cmd_list
 | 
						|
 | 
						|
        if (
 | 
						|
            len(cmd_list) < 1500
 | 
						|
        ):  # I'm allowed to have arbitrary numbers for when it's too much to dm dammit
 | 
						|
            await ctx.send(box(cmd_list))
 | 
						|
        else:
 | 
						|
            for page in pagify(cmd_list, delims=[" ", "\n"]):
 | 
						|
                await ctx.author.send(box(page))
 | 
						|
            await ctx.send("Command list DM'd")
 | 
						|
 | 
						|
    @commands.Cog.listener()
 | 
						|
    async def on_message_without_command(self, message: discord.Message):
 | 
						|
        """
 | 
						|
        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-insensitivity tradition of ccrole
 | 
						|
        guild = ctx.guild
 | 
						|
        # message = ctx.message  # Unneeded since switch to `on_message_without_command` from `on_command_error`
 | 
						|
 | 
						|
        cmd_list = self.config.guild(guild).cmdlist
 | 
						|
        # cmd = message.content[len(prefix) :].split()[0].lower()
 | 
						|
        cmd = await cmd_list.get_raw(cmd, default=None)
 | 
						|
 | 
						|
        if cmd is not None:
 | 
						|
            await self.eval_cc(cmd, message, ctx)
 | 
						|
        else:
 | 
						|
            log.debug(f"No custom command named {ctx.invoked_with} found")
 | 
						|
 | 
						|
    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)
 | 
						|
        prefixes = sorted(prefix_list, key=lambda pfx: len(pfx), reverse=True)
 | 
						|
        for p in prefixes:
 | 
						|
            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 {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))
 |