mirror of
https://github.com/Cog-Creators/Red-DiscordBot.git
synced 2025-11-06 03:08:55 -05:00
[Admin] Code prettification and bugfixing (#3250)
* Facelift for Admin * Remove unnecessary converter, reorder existing steps * Delete admin.py * Delete __init__.py * Delete test_admin.py * Remove one extra unneeded check * Create 3250.bugfix.1.rst * Create 3250.bugfix.2.rst * Create 3250.bugfix.3.rst * Create 3250.bugfix.4.rst * Create 3250.misc.1.rst * Create 3250.misc.2.rst * Create 3250.misc.3.rst * Create 3250.breaking.1.rst * Create 3250.breaking.2.rst * ... * I hate black...
This commit is contained in:
parent
36e2cde04d
commit
e776b5ca1a
1
changelog.d/admin/3250.breaking.1.rst
Normal file
1
changelog.d/admin/3250.breaking.1.rst
Normal file
@ -0,0 +1 @@
|
|||||||
|
Changed ``[p]announce ignore`` and ``[p]announce channel`` to ``[p]announceset ignore`` and ``[p]announceset channel``.
|
||||||
1
changelog.d/admin/3250.breaking.2.rst
Normal file
1
changelog.d/admin/3250.breaking.2.rst
Normal file
@ -0,0 +1 @@
|
|||||||
|
Changed ``[p]selfrole <role>`` to ``[p]selfrole add <role>``, changed ``[p]selfrole add`` to ``[p]selfroleset add`` , and changed ``[p]selfrole delete`` to ``[p]selfroleset remove``.
|
||||||
2
changelog.d/admin/3250.bugfix.1.rst
Normal file
2
changelog.d/admin/3250.bugfix.1.rst
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
Improved the clairty of user facing messages in the admin cog when the user is not allowed
|
||||||
|
to do something due to Discord hierarchy rules.
|
||||||
1
changelog.d/admin/3250.bugfix.2.rst
Normal file
1
changelog.d/admin/3250.bugfix.2.rst
Normal file
@ -0,0 +1 @@
|
|||||||
|
Fixed some role managing commands not properly checking if the bot had manage_roles perms before attempting to manage roles.
|
||||||
1
changelog.d/admin/3250.bugfix.3.rst
Normal file
1
changelog.d/admin/3250.bugfix.3.rst
Normal file
@ -0,0 +1 @@
|
|||||||
|
Fixed ``[p]editrole`` commands not checking if roles to be edited are higher than the bot's highest role before trying to edit them.
|
||||||
1
changelog.d/admin/3250.bugfix.4.rst
Normal file
1
changelog.d/admin/3250.bugfix.4.rst
Normal file
@ -0,0 +1 @@
|
|||||||
|
Fixed ``[p]announce ignore`` and ``[p]announce channel`` not being able to be used by guild owners and administrators.
|
||||||
1
changelog.d/admin/3250.misc.1.rst
Normal file
1
changelog.d/admin/3250.misc.1.rst
Normal file
@ -0,0 +1 @@
|
|||||||
|
Removed Admin tests.
|
||||||
1
changelog.d/admin/3250.misc.2.rst
Normal file
1
changelog.d/admin/3250.misc.2.rst
Normal file
@ -0,0 +1 @@
|
|||||||
|
Removed the ``MemberDefaultAuthor`` converter because its only purpose seemed to be catching ``""`` "members".
|
||||||
1
changelog.d/admin/3250.misc.3.rst
Normal file
1
changelog.d/admin/3250.misc.3.rst
Normal file
@ -0,0 +1 @@
|
|||||||
|
The ``SelfRole`` converter now only makes a config get call when the role actually exists.
|
||||||
@ -7,7 +7,7 @@ from redbot.core import Config, checks, commands
|
|||||||
from redbot.core.i18n import Translator, cog_i18n
|
from redbot.core.i18n import Translator, cog_i18n
|
||||||
from redbot.core.utils.chat_formatting import box
|
from redbot.core.utils.chat_formatting import box
|
||||||
from .announcer import Announcer
|
from .announcer import Announcer
|
||||||
from .converters import MemberDefaultAuthor, SelfRole
|
from .converters import SelfRole
|
||||||
|
|
||||||
log = logging.getLogger("red.admin")
|
log = logging.getLogger("red.admin")
|
||||||
|
|
||||||
@ -20,40 +20,43 @@ GENERIC_FORBIDDEN = _(
|
|||||||
)
|
)
|
||||||
|
|
||||||
HIERARCHY_ISSUE_ADD = _(
|
HIERARCHY_ISSUE_ADD = _(
|
||||||
"I tried to add {role.name} to {member.display_name} but that role"
|
"I can not give {role.name} to {member.display_name}"
|
||||||
" is higher than my highest role in the Discord hierarchy so I was"
|
" because that role is higher than or equal to my highest role"
|
||||||
" unable to successfully add it. Please give me a higher role and "
|
" in the Discord hierarchy."
|
||||||
"try again."
|
|
||||||
)
|
)
|
||||||
|
|
||||||
HIERARCHY_ISSUE_REMOVE = _(
|
HIERARCHY_ISSUE_REMOVE = _(
|
||||||
"I tried to remove {role.name} from {member.display_name} but that role"
|
"I can not remove {role.name} from {member.display_name}"
|
||||||
" is higher than my highest role in the Discord hierarchy so I was"
|
" because that role is higher than or equal to my highest role"
|
||||||
" unable to successfully remove it. Please give me a higher role and "
|
" in the Discord hierarchy."
|
||||||
"try again."
|
)
|
||||||
|
|
||||||
|
ROLE_HIERARCHY_ISSUE = _(
|
||||||
|
"I can not edit {role.name}"
|
||||||
|
" because that role is higher than my or equal to highest role"
|
||||||
|
" in the Discord hierarchy."
|
||||||
)
|
)
|
||||||
|
|
||||||
USER_HIERARCHY_ISSUE_ADD = _(
|
USER_HIERARCHY_ISSUE_ADD = _(
|
||||||
"I tried to add {role.name} to {member.display_name} but that role"
|
"I can not let you give {role.name} to {member.display_name}"
|
||||||
" is higher than your highest role in the Discord hierarchy so I was"
|
" because that role is higher than or equal to your highest role"
|
||||||
" unable to successfully add it. Please get a higher role and "
|
" in the Discord hierarchy."
|
||||||
"try again."
|
|
||||||
)
|
)
|
||||||
|
|
||||||
USER_HIERARCHY_ISSUE_REMOVE = _(
|
USER_HIERARCHY_ISSUE_REMOVE = _(
|
||||||
"I tried to remove {role.name} from {member.display_name} but that role"
|
"I can not let you remove {role.name} from {member.display_name}"
|
||||||
" is higher than your highest role in the Discord hierarchy so I was"
|
" because that role is higher than or equal to your highest role"
|
||||||
" unable to successfully remove it. Please get a higher role and "
|
" in the Discord hierarchy."
|
||||||
"try again."
|
|
||||||
)
|
)
|
||||||
|
|
||||||
ROLE_USER_HIERARCHY_ISSUE = _(
|
ROLE_USER_HIERARCHY_ISSUE = _(
|
||||||
"I tried to edit {role.name} but that role"
|
"I can not let you edit {role.name}"
|
||||||
" is higher than your highest role in the Discord hierarchy so I was"
|
" because that role is higher than or equal to your highest role"
|
||||||
" unable to successfully add it. Please get a higher role and "
|
" in the Discord hierarchy."
|
||||||
"try again."
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
NEED_MANAGE_ROLES = _("I need manage roles permission to do that.")
|
||||||
|
|
||||||
RUNNING_ANNOUNCEMENT = _(
|
RUNNING_ANNOUNCEMENT = _(
|
||||||
"I am already announcing something. If you would like to make a"
|
"I am already announcing something. If you would like to make a"
|
||||||
" different announcement please use `{prefix}announce cancel`"
|
" different announcement please use `{prefix}announce cancel`"
|
||||||
@ -66,9 +69,8 @@ _ = T_
|
|||||||
class Admin(commands.Cog):
|
class Admin(commands.Cog):
|
||||||
"""A collection of server administration utilities."""
|
"""A collection of server administration utilities."""
|
||||||
|
|
||||||
def __init__(self, config=Config):
|
def __init__(self):
|
||||||
super().__init__()
|
self.conf = Config.get_conf(self, 8237492837454039, force_registration=True)
|
||||||
self.conf = config.get_conf(self, 8237492837454039, force_registration=True)
|
|
||||||
|
|
||||||
self.conf.register_global(serverlocked=False)
|
self.conf.register_global(serverlocked=False)
|
||||||
|
|
||||||
@ -86,10 +88,6 @@ class Admin(commands.Cog):
|
|||||||
except AttributeError:
|
except AttributeError:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
async def complain(ctx: commands.Context, message: str, **kwargs):
|
|
||||||
await ctx.send(message.format(**kwargs))
|
|
||||||
|
|
||||||
def is_announcing(self) -> bool:
|
def is_announcing(self) -> bool:
|
||||||
"""
|
"""
|
||||||
Is the bot currently announcing something?
|
Is the bot currently announcing something?
|
||||||
@ -121,13 +119,21 @@ class Admin(commands.Cog):
|
|||||||
return ctx.author.top_role > role
|
return ctx.author.top_role > role
|
||||||
|
|
||||||
async def _addrole(self, ctx: commands.Context, member: discord.Member, role: discord.Role):
|
async def _addrole(self, ctx: commands.Context, member: discord.Member, role: discord.Role):
|
||||||
|
if member is None:
|
||||||
|
member = ctx.author
|
||||||
|
if not self.pass_user_hierarchy_check(ctx, role):
|
||||||
|
await ctx.send(_(USER_HIERARCHY_ISSUE_ADD).format(role=role, member=member))
|
||||||
|
return
|
||||||
|
if not self.pass_hierarchy_check(ctx, role):
|
||||||
|
await ctx.send(_(HIERARCHY_ISSUE_ADD).format(role=role, member=member))
|
||||||
|
return
|
||||||
|
if not ctx.guild.me.guild_permissions.manage_roles:
|
||||||
|
await ctx.send(_(NEED_MANAGE_ROLES))
|
||||||
|
return
|
||||||
try:
|
try:
|
||||||
await member.add_roles(role)
|
await member.add_roles(role)
|
||||||
except discord.Forbidden:
|
except discord.Forbidden:
|
||||||
if not self.pass_hierarchy_check(ctx, role):
|
await ctx.send(_(GENERIC_FORBIDDEN))
|
||||||
await self.complain(ctx, T_(HIERARCHY_ISSUE_ADD), role=role, member=member)
|
|
||||||
else:
|
|
||||||
await self.complain(ctx, T_(GENERIC_FORBIDDEN))
|
|
||||||
else:
|
else:
|
||||||
await ctx.send(
|
await ctx.send(
|
||||||
_("I successfully added {role.name} to {member.display_name}").format(
|
_("I successfully added {role.name} to {member.display_name}").format(
|
||||||
@ -136,13 +142,21 @@ class Admin(commands.Cog):
|
|||||||
)
|
)
|
||||||
|
|
||||||
async def _removerole(self, ctx: commands.Context, member: discord.Member, role: discord.Role):
|
async def _removerole(self, ctx: commands.Context, member: discord.Member, role: discord.Role):
|
||||||
|
if member is None:
|
||||||
|
member = ctx.author
|
||||||
|
if not self.pass_user_hierarchy_check(ctx, role):
|
||||||
|
await ctx.send(_(USER_HIERARCHY_ISSUE_REMOVE).foramt(role=role, member=member))
|
||||||
|
return
|
||||||
|
if not self.pass_hierarchy_check(ctx, role):
|
||||||
|
await ctx.send(_(HIERARCHY_ISSUE_REMOVE).format(role=role, member=member))
|
||||||
|
return
|
||||||
|
if not ctx.guild.me.guild_permissions.manage_roles:
|
||||||
|
await ctx.send(_(NEED_MANAGE_ROLES))
|
||||||
|
return
|
||||||
try:
|
try:
|
||||||
await member.remove_roles(role)
|
await member.remove_roles(role)
|
||||||
except discord.Forbidden:
|
except discord.Forbidden:
|
||||||
if not self.pass_hierarchy_check(ctx, role):
|
await ctx.send(_(GENERIC_FORBIDDEN))
|
||||||
await self.complain(ctx, T_(HIERARCHY_ISSUE_REMOVE), role=role, member=member)
|
|
||||||
else:
|
|
||||||
await self.complain(ctx, T_(GENERIC_FORBIDDEN))
|
|
||||||
else:
|
else:
|
||||||
await ctx.send(
|
await ctx.send(
|
||||||
_("I successfully removed {role.name} from {member.display_name}").format(
|
_("I successfully removed {role.name} from {member.display_name}").format(
|
||||||
@ -154,37 +168,33 @@ class Admin(commands.Cog):
|
|||||||
@commands.guild_only()
|
@commands.guild_only()
|
||||||
@checks.admin_or_permissions(manage_roles=True)
|
@checks.admin_or_permissions(manage_roles=True)
|
||||||
async def addrole(
|
async def addrole(
|
||||||
self, ctx: commands.Context, rolename: discord.Role, *, user: MemberDefaultAuthor = None
|
self, ctx: commands.Context, rolename: discord.Role, *, user: discord.Member = None
|
||||||
):
|
):
|
||||||
"""Add a role to a user.
|
"""
|
||||||
|
Add a role to a user.
|
||||||
|
|
||||||
|
Use double quotes if the role contains spaces.
|
||||||
If user is left blank it defaults to the author of the command.
|
If user is left blank it defaults to the author of the command.
|
||||||
"""
|
"""
|
||||||
if user is None:
|
if user is None:
|
||||||
user = ctx.author
|
user = ctx.author
|
||||||
if self.pass_user_hierarchy_check(ctx, rolename):
|
await self._addrole(ctx, user, rolename)
|
||||||
# noinspection PyTypeChecker
|
|
||||||
await self._addrole(ctx, user, rolename)
|
|
||||||
else:
|
|
||||||
await self.complain(ctx, T_(USER_HIERARCHY_ISSUE_ADD), member=user, role=rolename)
|
|
||||||
|
|
||||||
@commands.command()
|
@commands.command()
|
||||||
@commands.guild_only()
|
@commands.guild_only()
|
||||||
@checks.admin_or_permissions(manage_roles=True)
|
@checks.admin_or_permissions(manage_roles=True)
|
||||||
async def removerole(
|
async def removerole(
|
||||||
self, ctx: commands.Context, rolename: discord.Role, *, user: MemberDefaultAuthor = None
|
self, ctx: commands.Context, rolename: discord.Role, *, user: discord.Member = None
|
||||||
):
|
):
|
||||||
"""Remove a role from a user.
|
"""
|
||||||
|
Remove a role from a user.
|
||||||
|
|
||||||
|
Use double quotes if the role contains spaces.
|
||||||
If user is left blank it defaults to the author of the command.
|
If user is left blank it defaults to the author of the command.
|
||||||
"""
|
"""
|
||||||
if user is None:
|
if user is None:
|
||||||
user = ctx.author
|
user = ctx.author
|
||||||
if self.pass_user_hierarchy_check(ctx, rolename):
|
await self._removerole(ctx, user, rolename)
|
||||||
# noinspection PyTypeChecker
|
|
||||||
await self._removerole(ctx, user, rolename)
|
|
||||||
else:
|
|
||||||
await self.complain(ctx, T_(USER_HIERARCHY_ISSUE_REMOVE), member=user, role=rolename)
|
|
||||||
|
|
||||||
@commands.group()
|
@commands.group()
|
||||||
@commands.guild_only()
|
@commands.guild_only()
|
||||||
@ -197,7 +207,8 @@ class Admin(commands.Cog):
|
|||||||
async def editrole_colour(
|
async def editrole_colour(
|
||||||
self, ctx: commands.Context, role: discord.Role, value: discord.Colour
|
self, ctx: commands.Context, role: discord.Role, value: discord.Colour
|
||||||
):
|
):
|
||||||
"""Edit a role's colour.
|
"""
|
||||||
|
Edit a role's colour.
|
||||||
|
|
||||||
Use double quotes if the role contains spaces.
|
Use double quotes if the role contains spaces.
|
||||||
Colour must be in hexadecimal format.
|
Colour must be in hexadecimal format.
|
||||||
@ -211,25 +222,30 @@ class Admin(commands.Cog):
|
|||||||
reason = "{}({}) changed the colour of role '{}'".format(author.name, author.id, role.name)
|
reason = "{}({}) changed the colour of role '{}'".format(author.name, author.id, role.name)
|
||||||
|
|
||||||
if not self.pass_user_hierarchy_check(ctx, role):
|
if not self.pass_user_hierarchy_check(ctx, role):
|
||||||
await self.complain(ctx, T_(ROLE_USER_HIERARCHY_ISSUE), role=role)
|
await ctx.send(_(ROLE_USER_HIERARCHY_ISSUE).format(role=role))
|
||||||
|
return
|
||||||
|
if not self.pass_hierarchy_check(ctx, role):
|
||||||
|
await ctx.send(_(ROLE_HIERARCHY_ISSUE).format(role=role))
|
||||||
|
return
|
||||||
|
if not ctx.guild.me.guild_permissions.manage_roles:
|
||||||
|
await ctx.send(_(NEED_MANAGE_ROLES))
|
||||||
return
|
return
|
||||||
|
|
||||||
try:
|
try:
|
||||||
await role.edit(reason=reason, color=value)
|
await role.edit(reason=reason, color=value)
|
||||||
except discord.Forbidden:
|
except discord.Forbidden:
|
||||||
await self.complain(ctx, T_(GENERIC_FORBIDDEN))
|
await ctx.send(_(GENERIC_FORBIDDEN))
|
||||||
else:
|
else:
|
||||||
log.info(reason)
|
log.info(reason)
|
||||||
await ctx.send(_("Done."))
|
await ctx.send(_("Done."))
|
||||||
|
|
||||||
@editrole.command(name="name")
|
@editrole.command(name="name")
|
||||||
@checks.admin_or_permissions(administrator=True)
|
async def edit_role_name(self, ctx: commands.Context, role: discord.Role, name: str):
|
||||||
async def edit_role_name(self, ctx: commands.Context, role: discord.Role, *, name: str):
|
"""
|
||||||
"""Edit a role's name.
|
Edit a role's name.
|
||||||
|
|
||||||
Use double quotes if the role or the name contain spaces.
|
Use double quotes if the role or the name contain spaces.
|
||||||
|
|
||||||
Examples:
|
Example:
|
||||||
`[p]editrole name \"The Transistor\" Test`
|
`[p]editrole name \"The Transistor\" Test`
|
||||||
"""
|
"""
|
||||||
author = ctx.message.author
|
author = ctx.message.author
|
||||||
@ -239,13 +255,18 @@ class Admin(commands.Cog):
|
|||||||
)
|
)
|
||||||
|
|
||||||
if not self.pass_user_hierarchy_check(ctx, role):
|
if not self.pass_user_hierarchy_check(ctx, role):
|
||||||
await self.complain(ctx, T_(ROLE_USER_HIERARCHY_ISSUE), role=role)
|
await ctx.send(_(ROLE_USER_HIERARCHY_ISSUE).format(role=role))
|
||||||
|
return
|
||||||
|
if not self.pass_hierarchy_check(ctx, role):
|
||||||
|
await ctx.send(_(ROLE_HIERARCHY_ISSUE).format(role=role))
|
||||||
|
return
|
||||||
|
if not ctx.guild.me.guild_permissions.manage_roles:
|
||||||
|
await ctx.send(_(NEED_MANAGE_ROLES))
|
||||||
return
|
return
|
||||||
|
|
||||||
try:
|
try:
|
||||||
await role.edit(reason=reason, name=name)
|
await role.edit(reason=reason, name=name)
|
||||||
except discord.Forbidden:
|
except discord.Forbidden:
|
||||||
await self.complain(ctx, T_(GENERIC_FORBIDDEN))
|
await ctx.send(_(GENERIC_FORBIDDEN))
|
||||||
else:
|
else:
|
||||||
log.info(reason)
|
log.info(reason)
|
||||||
await ctx.send(_("Done."))
|
await ctx.send(_("Done."))
|
||||||
@ -263,41 +284,44 @@ class Admin(commands.Cog):
|
|||||||
await ctx.send(_("The announcement has begun."))
|
await ctx.send(_("The announcement has begun."))
|
||||||
else:
|
else:
|
||||||
prefix = ctx.prefix
|
prefix = ctx.prefix
|
||||||
await self.complain(ctx, T_(RUNNING_ANNOUNCEMENT), prefix=prefix)
|
await ctx.send(_(RUNNING_ANNOUNCEMENT).format(prefix=prefix))
|
||||||
|
|
||||||
@announce.command(name="cancel")
|
@announce.command(name="cancel")
|
||||||
@checks.is_owner()
|
|
||||||
async def announce_cancel(self, ctx):
|
async def announce_cancel(self, ctx):
|
||||||
"""Cancel a running announce."""
|
"""Cancel a running announce."""
|
||||||
try:
|
if not self.is_announcing():
|
||||||
self.__current_announcer.cancel()
|
await ctx.send(_("There is no currently running announcement."))
|
||||||
except AttributeError:
|
return
|
||||||
pass
|
self.__current_announcer.cancel()
|
||||||
|
|
||||||
await ctx.send(_("The current announcement has been cancelled."))
|
await ctx.send(_("The current announcement has been cancelled."))
|
||||||
|
|
||||||
@announce.command(name="channel")
|
@commands.group()
|
||||||
@commands.guild_only()
|
@commands.guild_only()
|
||||||
@checks.guildowner_or_permissions(administrator=True)
|
@checks.guildowner_or_permissions(administrator=True)
|
||||||
async def announce_channel(self, ctx, *, channel: discord.TextChannel = None):
|
async def announceset(self, ctx):
|
||||||
"""Change the channel to which the bot makes announcements."""
|
"""Change how announcements are sent in this guild."""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@announceset.command(name="channel")
|
||||||
|
async def announceset_channel(self, ctx, *, channel: discord.TextChannel = None):
|
||||||
|
"""
|
||||||
|
Change the channel where the bot will send announcements.
|
||||||
|
|
||||||
|
If channel is left blank it defaults to the current channel.
|
||||||
|
"""
|
||||||
if channel is None:
|
if channel is None:
|
||||||
channel = ctx.channel
|
channel = ctx.channel
|
||||||
await self.conf.guild(ctx.guild).announce_channel.set(channel.id)
|
await self.conf.guild(ctx.guild).announce_channel.set(channel.id)
|
||||||
|
|
||||||
await ctx.send(
|
await ctx.send(
|
||||||
_("The announcement channel has been set to {channel.mention}").format(channel=channel)
|
_("The announcement channel has been set to {channel.mention}").format(channel=channel)
|
||||||
)
|
)
|
||||||
|
|
||||||
@announce.command(name="ignore")
|
@announceset.command(name="ignore")
|
||||||
@commands.guild_only()
|
async def announceset_ignore(self, ctx):
|
||||||
@checks.guildowner_or_permissions(administrator=True)
|
|
||||||
async def announce_ignore(self, ctx):
|
|
||||||
"""Toggle announcements being enabled this server."""
|
"""Toggle announcements being enabled this server."""
|
||||||
ignored = await self.conf.guild(ctx.guild).announce_ignore()
|
ignored = await self.conf.guild(ctx.guild).announce_ignore()
|
||||||
await self.conf.guild(ctx.guild).announce_ignore.set(not ignored)
|
await self.conf.guild(ctx.guild).announce_ignore.set(not ignored)
|
||||||
|
if ignored:
|
||||||
if ignored: # Keeping original logic....
|
|
||||||
await ctx.send(
|
await ctx.send(
|
||||||
_("The server {guild.name} will receive announcements.").format(guild=ctx.guild)
|
_("The server {guild.name} will receive announcements.").format(guild=ctx.guild)
|
||||||
)
|
)
|
||||||
@ -310,7 +334,7 @@ class Admin(commands.Cog):
|
|||||||
|
|
||||||
async def _valid_selfroles(self, guild: discord.Guild) -> Tuple[discord.Role]:
|
async def _valid_selfroles(self, guild: discord.Guild) -> Tuple[discord.Role]:
|
||||||
"""
|
"""
|
||||||
Returns a list of valid selfroles
|
Returns a tuple of valid selfroles
|
||||||
:param guild:
|
:param guild:
|
||||||
:return:
|
:return:
|
||||||
"""
|
"""
|
||||||
@ -327,12 +351,17 @@ class Admin(commands.Cog):
|
|||||||
return valid_roles
|
return valid_roles
|
||||||
|
|
||||||
@commands.guild_only()
|
@commands.guild_only()
|
||||||
@commands.group(invoke_without_command=True)
|
@commands.group()
|
||||||
async def selfrole(self, ctx: commands.Context, *, selfrole: SelfRole):
|
async def selfrole(self, ctx: commands.Context):
|
||||||
"""Add a role to yourself.
|
"""Apply selfroles."""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@selfrole.command(name="add")
|
||||||
|
async def selfrole_add(self, ctx: commands.Context, *, selfrole: SelfRole):
|
||||||
|
"""
|
||||||
|
Add a selfrole to yourself.
|
||||||
|
|
||||||
Server admins must have configured the role as user settable.
|
Server admins must have configured the role as user settable.
|
||||||
|
|
||||||
NOTE: The role is case sensitive!
|
NOTE: The role is case sensitive!
|
||||||
"""
|
"""
|
||||||
# noinspection PyTypeChecker
|
# noinspection PyTypeChecker
|
||||||
@ -340,38 +369,15 @@ class Admin(commands.Cog):
|
|||||||
|
|
||||||
@selfrole.command(name="remove")
|
@selfrole.command(name="remove")
|
||||||
async def selfrole_remove(self, ctx: commands.Context, *, selfrole: SelfRole):
|
async def selfrole_remove(self, ctx: commands.Context, *, selfrole: SelfRole):
|
||||||
"""Remove a selfrole from yourself.
|
"""
|
||||||
|
Remove a selfrole from yourself.
|
||||||
|
|
||||||
|
Server admins must have configured the role as user settable.
|
||||||
NOTE: The role is case sensitive!
|
NOTE: The role is case sensitive!
|
||||||
"""
|
"""
|
||||||
# noinspection PyTypeChecker
|
# noinspection PyTypeChecker
|
||||||
await self._removerole(ctx, ctx.author, selfrole)
|
await self._removerole(ctx, ctx.author, selfrole)
|
||||||
|
|
||||||
@selfrole.command(name="add")
|
|
||||||
@checks.admin_or_permissions(manage_roles=True)
|
|
||||||
async def selfrole_add(self, ctx: commands.Context, *, role: discord.Role):
|
|
||||||
"""Add a role to the list of available selfroles.
|
|
||||||
|
|
||||||
NOTE: The role is case sensitive!
|
|
||||||
"""
|
|
||||||
async with self.conf.guild(ctx.guild).selfroles() as curr_selfroles:
|
|
||||||
if role.id not in curr_selfroles:
|
|
||||||
curr_selfroles.append(role.id)
|
|
||||||
|
|
||||||
await ctx.send(_("The selfroles list has been successfully modified."))
|
|
||||||
|
|
||||||
@selfrole.command(name="delete")
|
|
||||||
@checks.admin_or_permissions(manage_roles=True)
|
|
||||||
async def selfrole_delete(self, ctx: commands.Context, *, role: SelfRole):
|
|
||||||
"""Remove a role from the list of available selfroles.
|
|
||||||
|
|
||||||
NOTE: The role is case sensitive!
|
|
||||||
"""
|
|
||||||
async with self.conf.guild(ctx.guild).selfroles() as curr_selfroles:
|
|
||||||
curr_selfroles.remove(role.id)
|
|
||||||
|
|
||||||
await ctx.send(_("The selfroles list has been successfully modified."))
|
|
||||||
|
|
||||||
@selfrole.command(name="list")
|
@selfrole.command(name="list")
|
||||||
async def selfrole_list(self, ctx: commands.Context):
|
async def selfrole_list(self, ctx: commands.Context):
|
||||||
"""
|
"""
|
||||||
@ -380,19 +386,45 @@ class Admin(commands.Cog):
|
|||||||
selfroles = await self._valid_selfroles(ctx.guild)
|
selfroles = await self._valid_selfroles(ctx.guild)
|
||||||
fmt_selfroles = "\n".join(["+ " + r.name for r in selfroles])
|
fmt_selfroles = "\n".join(["+ " + r.name for r in selfroles])
|
||||||
|
|
||||||
|
if not fmt_selfroles:
|
||||||
|
await ctx.send("There are currently no selfroles.")
|
||||||
|
return
|
||||||
|
|
||||||
msg = _("Available Selfroles:\n{selfroles}").format(selfroles=fmt_selfroles)
|
msg = _("Available Selfroles:\n{selfroles}").format(selfroles=fmt_selfroles)
|
||||||
await ctx.send(box(msg, "diff"))
|
await ctx.send(box(msg, "diff"))
|
||||||
|
|
||||||
async def _serverlock_check(self, guild: discord.Guild) -> bool:
|
@commands.group()
|
||||||
|
@checks.admin_or_permissions(manage_roles=True)
|
||||||
|
async def selfroleset(self, ctx: commands.Context):
|
||||||
|
"""Manage selfroles."""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@selfroleset.command(name="add")
|
||||||
|
async def selfroleset_add(self, ctx: commands.Context, *, role: discord.Role):
|
||||||
"""
|
"""
|
||||||
Checks if serverlocked is enabled.
|
Add a role to the list of available selfroles.
|
||||||
:param guild:
|
|
||||||
:return: True if locked and left server
|
NOTE: The role is case sensitive!
|
||||||
"""
|
"""
|
||||||
if await self.conf.serverlocked():
|
async with self.conf.guild(ctx.guild).selfroles() as curr_selfroles:
|
||||||
await guild.leave()
|
if role.id not in curr_selfroles:
|
||||||
return True
|
curr_selfroles.append(role.id)
|
||||||
return False
|
await ctx.send(_("Added."))
|
||||||
|
return
|
||||||
|
|
||||||
|
await ctx.send(_("That role is already a selfrole."))
|
||||||
|
|
||||||
|
@selfroleset.command(name="remove")
|
||||||
|
async def selfroleset_remove(self, ctx: commands.Context, *, role: SelfRole):
|
||||||
|
"""
|
||||||
|
Remove a role from the list of available selfroles.
|
||||||
|
|
||||||
|
NOTE: The role is case sensitive!
|
||||||
|
"""
|
||||||
|
async with self.conf.guild(ctx.guild).selfroles() as curr_selfroles:
|
||||||
|
curr_selfroles.remove(role.id)
|
||||||
|
|
||||||
|
await ctx.send(_("Removed."))
|
||||||
|
|
||||||
@commands.command()
|
@commands.command()
|
||||||
@checks.is_owner()
|
@checks.is_owner()
|
||||||
@ -408,8 +440,8 @@ class Admin(commands.Cog):
|
|||||||
|
|
||||||
# region Event Handlers
|
# region Event Handlers
|
||||||
async def on_guild_join(self, guild: discord.Guild):
|
async def on_guild_join(self, guild: discord.Guild):
|
||||||
if await self._serverlock_check(guild):
|
if await self.conf.serverlocked():
|
||||||
return
|
await guild.leave()
|
||||||
|
|
||||||
|
|
||||||
# endregion
|
# endregion
|
||||||
|
|||||||
@ -5,31 +5,18 @@ from redbot.core.i18n import Translator
|
|||||||
_ = Translator("AdminConverters", __file__)
|
_ = Translator("AdminConverters", __file__)
|
||||||
|
|
||||||
|
|
||||||
class MemberDefaultAuthor(commands.Converter):
|
|
||||||
async def convert(self, ctx: commands.Context, arg: str) -> discord.Member:
|
|
||||||
member_converter = commands.MemberConverter()
|
|
||||||
try:
|
|
||||||
member = await member_converter.convert(ctx, arg)
|
|
||||||
except commands.BadArgument:
|
|
||||||
if arg.strip() != "":
|
|
||||||
raise
|
|
||||||
else:
|
|
||||||
member = ctx.author
|
|
||||||
return member
|
|
||||||
|
|
||||||
|
|
||||||
class SelfRole(commands.Converter):
|
class SelfRole(commands.Converter):
|
||||||
async def convert(self, ctx: commands.Context, arg: str) -> discord.Role:
|
async def convert(self, ctx: commands.Context, arg: str) -> discord.Role:
|
||||||
admin = ctx.command.cog
|
admin = ctx.command.cog
|
||||||
if admin is None:
|
if admin is None:
|
||||||
raise commands.BadArgument(_("The Admin cog is not loaded."))
|
raise commands.BadArgument(_("The Admin cog is not loaded."))
|
||||||
|
|
||||||
conf = admin.conf
|
|
||||||
selfroles = await conf.guild(ctx.guild).selfroles()
|
|
||||||
|
|
||||||
role_converter = commands.RoleConverter()
|
role_converter = commands.RoleConverter()
|
||||||
role = await role_converter.convert(ctx, arg)
|
role = await role_converter.convert(ctx, arg)
|
||||||
|
|
||||||
|
conf = admin.conf
|
||||||
|
selfroles = await conf.guild(ctx.guild).selfroles()
|
||||||
|
|
||||||
if role.id not in selfroles:
|
if role.id not in selfroles:
|
||||||
raise commands.BadArgument(_("The provided role is not a valid selfrole."))
|
raise commands.BadArgument(_("The provided role is not a valid selfrole."))
|
||||||
return role
|
return role
|
||||||
|
|||||||
@ -1,20 +0,0 @@
|
|||||||
from unittest.mock import MagicMock
|
|
||||||
|
|
||||||
import pytest
|
|
||||||
|
|
||||||
from redbot.cogs.admin import Admin
|
|
||||||
from redbot.cogs.admin.announcer import Announcer
|
|
||||||
|
|
||||||
__all__ = ["admin", "announcer"]
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture()
|
|
||||||
def admin(config):
|
|
||||||
return Admin(config)
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture()
|
|
||||||
def announcer(admin):
|
|
||||||
a = Announcer(MagicMock(), "Some message", admin.conf)
|
|
||||||
yield a
|
|
||||||
a.cancel()
|
|
||||||
@ -1,44 +0,0 @@
|
|||||||
from unittest.mock import MagicMock
|
|
||||||
|
|
||||||
import pytest
|
|
||||||
|
|
||||||
from redbot.pytest.admin import *
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.asyncio
|
|
||||||
async def test_serverlock_check(admin, coroutine):
|
|
||||||
await admin.conf.serverlocked.set(True)
|
|
||||||
guild = MagicMock()
|
|
||||||
guild.leave = coroutine
|
|
||||||
|
|
||||||
# noinspection PyProtectedMember
|
|
||||||
ret = await admin._serverlock_check(guild)
|
|
||||||
|
|
||||||
assert ret is True
|
|
||||||
|
|
||||||
|
|
||||||
def test_announcer_initial_state(announcer):
|
|
||||||
assert announcer.active is None
|
|
||||||
|
|
||||||
|
|
||||||
def test_announcer_start(announcer):
|
|
||||||
announcer.announcer = object
|
|
||||||
announcer.start()
|
|
||||||
|
|
||||||
assert announcer.ctx.bot.loop.create_task.called
|
|
||||||
assert announcer.active is True
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.asyncio
|
|
||||||
async def test_announcer_ignore(announcer, empty_guild, empty_channel):
|
|
||||||
await announcer.config.guild(empty_guild).announce_channel.set(empty_channel.id)
|
|
||||||
|
|
||||||
guild = MagicMock()
|
|
||||||
guild.id = empty_guild.id
|
|
||||||
|
|
||||||
guild.get_channel.return_value = empty_channel
|
|
||||||
|
|
||||||
ret = await announcer._get_announce_channel(guild)
|
|
||||||
|
|
||||||
assert guild.get_channel.called
|
|
||||||
assert ret == empty_channel
|
|
||||||
Loading…
x
Reference in New Issue
Block a user