Skip to content
This repository has been archived by the owner on Oct 2, 2023. It is now read-only.

Feature/betheprofessional #36

Open
wants to merge 81 commits into
base: develop
Choose a base branch
from
Open
Changes from 17 commits
Commits
Show all changes
81 commits
Select commit Hold shift + click to select a range
c5259bc
Added new DB Models
Tert0 Jun 19, 2021
f699b7f
Updates Parameters
Tert0 Jun 19, 2021
6541268
Started Refactoring BeTheProfessional
Tert0 Jun 19, 2021
3102fcc
Fixed Model
Tert0 Jun 19, 2021
0171654
Fixed Model Primary Key
Tert0 Jun 19, 2021
7d8d668
Refactored function and the cog is now runnable
Tert0 Jun 19, 2021
d16256a
Added Group to DB Model
Tert0 Jun 20, 2021
58de9d3
New Translation
Tert0 Jun 20, 2021
dd0a7d3
Fixed bugs, added parent parser, added parents to params
Tert0 Jun 20, 2021
6c71d14
Added help for Group/Parent/Topic formating
Tert0 Jun 20, 2021
7727363
Fixed trailing comma
Tert0 Jun 20, 2021
5068770
Fixed trailing commas
Tert0 Jun 20, 2021
a2d8c98
Merge branch 'develop' into feature/betheprofessional
Tert0 Jun 20, 2021
cb71c22
Resolved PEP Problem
Tert0 Jun 20, 2021
df6cda3
Reformated with black
Tert0 Jun 20, 2021
281eaaf
Added group check and fixed unassing_topic command
Tert0 Jun 20, 2021
bbebc70
Improved List Feature (not ready with many bugs)
Tert0 Jun 20, 2021
5f9e129
Fixed Topic Check command
Tert0 Jun 20, 2021
fe70c36
New DB Model
Tert0 Jun 25, 2021
49e9511
New List Topics View, removed Group from DB and fixed Register Topic …
Tert0 Jun 25, 2021
46a02af
Fixed PEP
Tert0 Jun 25, 2021
2abcc71
Reformated with black
Tert0 Jun 25, 2021
1f95996
Merge branch 'develop' into feature/betheprofessional
Tert0 Jun 25, 2021
9f1d72b
Fixed PEP8
Tert0 Jun 25, 2021
171f720
Added new topic ping command
Tert0 Jun 25, 2021
eb34e26
Added Role Update and fixed logger
Tert0 Jun 25, 2021
79ce872
Refactored with black
Tert0 Jun 25, 2021
d9d997a
Fixed Formating
Tert0 Jun 25, 2021
54abdcf
Fixed Formating
Tert0 Jun 25, 2021
ad768ef
Optimised Import
Tert0 Jun 25, 2021
f315a1a
Added Topic Role Update Command
Tert0 Jun 26, 2021
c74966b
Merge branch 'develop' into feature/betheprofessional
Tert0 Aug 9, 2021
3873480
Fixed BeTheProfessional Update Role and Refactored
Tert0 Aug 9, 2021
986ab4e
Merge branch 'develop' into feature/betheprofessional
Tert0 Oct 4, 2021
fcd188b
Merge branch 'develop' into feature/betheprofessional
Tert0 Oct 5, 2021
c852e94
Made Topic Names complete Unique, Fixed Role Assign, Fixed Role Delete
Tert0 Oct 5, 2021
ce73f33
Fix PEP8
Tert0 Oct 11, 2021
da6e7eb
Merge branch 'develop' into feature/betheprofessional
Tert0 Jan 5, 2022
433fc67
Fixed DB Models, Updated Command Descriptions, Improved Top Topic Rol…
Tert0 Jan 8, 2022
2b4f90d
Added BTP Leaderboard Command
Tert0 Jan 9, 2022
23732c7
Added Redis Leaderboard Cache, Added Bypass Cache Permission, Added B…
Tert0 Jan 9, 2022
4ca2a3c
Refactored Code
Tert0 Jan 9, 2022
7c5f7bd
Reformatted with black
Tert0 Jan 9, 2022
6edfe58
PEP8
Tert0 Jan 9, 2022
f877b28
PEP8
Tert0 Jan 9, 2022
16dd9df
Trailing Comma
Tert0 Jan 9, 2022
3e30daa
Merge branch 'develop' into feature/betheprofessional
Tert0 Jan 9, 2022
8c9c3c7
Added Docs
Tert0 Jan 9, 2022
8353218
Merge remote-tracking branch 'origin/feature/betheprofessional' into …
Tert0 Jan 9, 2022
9e01906
Fixed MD Style
Tert0 Jan 9, 2022
64bb3f6
Fixed Role Update Loop + Reload Command, Added BTP Main Command, Adde…
Tert0 Jan 9, 2022
1254e49
Added read Permission and Permission Check, Added new commands to the…
Tert0 Jan 9, 2022
45666b6
Fixed recursive Topic delition
Tert0 Jan 9, 2022
3b65996
Implemented Signle (Un)Assign Help
Tert0 Jan 9, 2022
7aac1c9
Added reset for redis counter
Tert0 Jan 9, 2022
9c18006
Added missing awaits, Added usertopic command
Tert0 Jan 9, 2022
0718aa4
Added usertopics command docs
Tert0 Jan 9, 2022
4562c42
Reformatted with black
Tert0 Jan 9, 2022
ffeb298
Added Pagigantion and improved LB
Tert0 Jan 9, 2022
0fe9d4d
Merge branch 'develop' into feature/betheprofessional
Tert0 Mar 1, 2022
ae944d9
fixed documentation code style
Tert0 Mar 1, 2022
c8ef56f
optimized role update loop
Tert0 Mar 1, 2022
a417253
added second sort to role update loop
Tert0 Mar 1, 2022
8a4ea1c
Merge develop into feature/betheprofessional
Defelo Mar 14, 2022
cf790df
added TODOs
TheCataliasTNT2k Apr 8, 2022
b5d33c3
Removed typing import
Tert0 Apr 13, 2022
c446584
Resolved some TODO's
Tert0 Apr 13, 2022
38ffb81
Resolved some TODO's
Tert0 Apr 13, 2022
8f9988a
Merge branch 'v3.3' into feature/betheprofessional
Tert0 May 16, 2022
aa6202f
Added DB Relationships
Tert0 May 16, 2022
6be0b2f
Merge branch 'develop' into feature/betheprofessional
Tert0 May 16, 2022
e42986d
fix after merge
Tert0 May 16, 2022
5523e69
added confirmation on topic unregister and fixed codestyle stuff
Tert0 May 16, 2022
a91bccc
sorted imports
Tert0 May 16, 2022
0cea283
black+isort auto format
Tert0 May 16, 2022
51d3f4a
Merge branch 'develop' into feature/betheprofessional
Tert0 Jul 18, 2022
055793a
added assignable parameter, rewrote update roles logic and improved c…
Tert0 Jul 18, 2022
bab1770
Changed BTP Settings
Tert0 Jul 18, 2022
a2a0a91
fixed linter
Tert0 Jul 18, 2022
3ff8eba
resolved todos
Tert0 Jul 18, 2022
1924e3f
added a few todos
TheCataliasTNT2k Jul 25, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
284 changes: 159 additions & 125 deletions general/betheprofessional/cog.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import string
from typing import List
from typing import List, Union, Optional, Dict

from discord import Role, Guild, Member, Embed
from discord import Member, Embed, Role
from discord.ext import commands
from discord.ext.commands import guild_only, Context, CommandError, UserInputError

@@ -10,9 +10,9 @@
from PyDrocsid.database import db, select
from PyDrocsid.embeds import send_long_embed
from PyDrocsid.translations import t
from PyDrocsid.util import calculate_edit_distance, check_role_assignable
from PyDrocsid.util import calculate_edit_distance
from .colors import Colors
from .models import BTPRole
from .models import BTPUser, BTPTopic
from .permissions import BeTheProfessionalPermission
from ...contributor import Contributor
from ...pubsub import send_to_changelog
@@ -25,94 +25,111 @@ def split_topics(topics: str) -> List[str]:
return [topic for topic in map(str.strip, topics.replace(";", ",").split(",")) if topic]


async def parse_topics(guild: Guild, topics: str, author: Member) -> List[Role]:
roles: List[Role] = []
all_topics: List[Role] = await list_topics(guild)
for topic in split_topics(topics):
for role in guild.roles:
if role.name.lower() == topic.lower():
if role in all_topics:
break
if not role.managed and role >= guild.me.top_role:
raise CommandError(t.youre_not_the_first_one(topic, author.mention))
else:
if all_topics:

def dist(name: str) -> int:
return calculate_edit_distance(name.lower(), topic.lower())

best_match = min([r.name for r in all_topics], key=dist)
async def split_parents(topics: List[str]) -> List[tuple[str, str, Union[BTPTopic, None]]]:
result: List[tuple[str, str, Union[BTPTopic, None]]] = []
for topic in topics:
topic_tree = topic.split("/")
if len(topic_tree) > 3 or len(topic_tree) < 2:
raise CommandError(t.group_parent_format_help)
group = topic_tree[0]
query = select(BTPTopic).filter_by(name=topic_tree[1])
parent: Union[BTPTopic, None, CommandError] = (
(await db.first(query) if await db.exists(query) else CommandError(t.parent_not_exists(topic_tree[1])))
if len(topic_tree) > 2
else None
)
if isinstance(parent, CommandError):
raise parent
if parent is not None:
if group != parent.group:
raise CommandError(t.group_not_parent_group(group, parent.group))
topic = topic_tree[-1]
result.append((topic, group, parent))
return result


async def parse_topics(topics_str: str) -> List[BTPTopic]:
Tert0 marked this conversation as resolved.
Show resolved Hide resolved
topics: List[BTPTopic] = []
all_topics: List[BTPTopic] = await get_topics()
for topic in split_topics(topics_str):
query = select(BTPTopic).filter_by(name=topic)
topic_db = await db.first(query)
if not (await db.exists(query)) and len(all_topics) > 0:

def dist(name: str) -> int:
return calculate_edit_distance(name.lower(), topic.lower())

best_match = min([r.name for r in all_topics], key=dist)
if best_match:
raise CommandError(t.topic_not_found_did_you_mean(topic, best_match))
raise CommandError(t.topic_not_found(topic))
roles.append(role)
return roles


async def list_topics(guild: Guild) -> List[Role]:
roles: List[Role] = []
async for btp_role in await db.stream(select(BTPRole)):
if (role := guild.get_role(btp_role.role_id)) is None:
await db.delete(btp_role)
else:
roles.append(role)
return roles


async def unregister_roles(ctx: Context, topics: str, *, delete_roles: bool):
guild: Guild = ctx.guild
roles: List[Role] = []
btp_roles: List[BTPRole] = []
names = split_topics(topics)
if not names:
raise UserInputError

for topic in names:
for role in guild.roles:
if role.name.lower() == topic.lower():
break
else:
raise CommandError(t.topic_not_registered(topic))
if (btp_role := await db.first(select(BTPRole).filter_by(role_id=role.id))) is None:
raise CommandError(t.topic_not_registered(topic))

roles.append(role)
btp_roles.append(btp_role)
else:
raise CommandError(t.topic_not_found(topic))
elif not (await db.exists(query)):
raise CommandError(t.no_topics_registered)
topics.append(topic_db)
return topics

for role, btp_role in zip(roles, btp_roles):
if delete_roles:
check_role_assignable(role)
await role.delete()
await db.delete(btp_role)

embed = Embed(title=t.betheprofessional, colour=Colors.BeTheProfessional)
embed.description = t.topics_unregistered(cnt=len(roles))
await send_to_changelog(
ctx.guild,
t.log_topics_unregistered(cnt=len(roles), topics=", ".join(f"`{r}`" for r in roles)),
)
await send_long_embed(ctx, embed)
async def get_topics() -> List[BTPTopic]:
topics: List[BTPTopic] = []
async for topic in await db.stream(select(BTPTopic)):
topics.append(topic)
return topics


class BeTheProfessionalCog(Cog, name="Self Assignable Topic Roles"):
CONTRIBUTORS = [Contributor.Defelo, Contributor.wolflu, Contributor.MaxiHuHe04, Contributor.AdriBloober]

@commands.command(name="?")
@guild_only()
async def list_topics(self, ctx: Context):
async def list_topics(self, ctx: Context, parent_topic: Optional[str]):
"""
list all registered topics
list all registered topics TODO
"""

parent: Union[None, BTPTopic, CommandError] = (
None
if parent_topic is None
else await db.first(select(BTPTopic).filter_by(name=parent_topic))
or CommandError(t.topic_not_found(parent_topic)) # noqa: W503
)
if isinstance(parent, CommandError):
raise parent
embed = Embed(title=t.available_topics_header, colour=Colors.BeTheProfessional)
out = [role.name for role in await list_topics(ctx.guild)]
grouped_topics: Dict[str, List[str]] = {}
out: List[BTPTopic] = [
topic
for topic in await db.all(select(BTPTopic).filter_by(parent=parent if parent is None else parent.id))
if topic.group is not None
]
if not out:
embed.colour = Colors.error
embed.description = t.no_topics_registered
await reply(ctx, embed=embed)
return

out.sort(key=str.lower)
embed.description = ", ".join(f"`{topic}`" for topic in out)
out.sort(key=lambda topic: topic.name.lower())
for topic in out:
if topic.group.title() not in grouped_topics.keys():
grouped_topics[topic.group] = [f"{topic.name}"]
else:
grouped_topics[topic.group.title()].append(f"{topic.name}")

for group in grouped_topics.keys():
embed.add_field(
name=group.title(),
value=", ".join(
[
f"`{topic.name}"
+ (
Tert0 marked this conversation as resolved.
Show resolved Hide resolved
f" ({c})`"
if (c := await db.count(select(BTPTopic).filter_by(parent=topic.id, group=topic.group))) > 0
else "`"
)
for topic in out
]
Tert0 marked this conversation as resolved.
Show resolved Hide resolved
),
inline=False,
)
await send_long_embed(ctx, embed)

@commands.command(name="+")
@@ -123,16 +140,17 @@ async def assign_topics(self, ctx: Context, *, topics: str):
"""

member: Member = ctx.author
roles: List[Role] = [r for r in await parse_topics(ctx.guild, topics, ctx.author) if r not in member.roles]

for role in roles:
check_role_assignable(role)

await member.add_roles(*roles)

topics: List[BTPTopic] = [
topic
for topic in await parse_topics(topics)
if (await db.exists(select(BTPTopic).filter_by(id=topic.id)))
and not (await db.exists(select(BTPUser).filter_by(user_id=member.id, topic=topic.id))) # noqa: W503
]
for topic in topics:
await BTPUser.create(member.id, topic.id)
embed = Embed(title=t.betheprofessional, colour=Colors.BeTheProfessional)
embed.description = t.topics_added(cnt=len(roles))
if not roles:
embed.description = t.topics_added(cnt=len(topics))
if not topics:
embed.colour = Colors.error

await reply(ctx, embed=embed)
@@ -143,21 +161,21 @@ async def unassign_topics(self, ctx: Context, *, topics: str):
"""
remove one or more topics (use * to remove all topics)
"""

member: Member = ctx.author
if topics.strip() == "*":
roles: List[Role] = await list_topics(ctx.guild)
topics: List[BTPTopic] = await get_topics()
else:
roles: List[Role] = await parse_topics(ctx.guild, topics, ctx.author)
roles = [r for r in roles if r in member.roles]
topics: List[BTPTopic] = await parse_topics(topics)
affected_topics: List[BTPTopic] = []
for topic in topics:
if await db.exists(select(BTPUser).filter_by(user_id=member.id, topic=topic.id)):
affected_topics.append(topic)

for role in roles:
check_role_assignable(role)

await member.remove_roles(*roles)
for topic in affected_topics:
await db.delete(await db.first(select(BTPUser).filter_by(topic=topic.id)))

embed = Embed(title=t.betheprofessional, colour=Colors.BeTheProfessional)
embed.description = t.topics_removed(cnt=len(roles))
embed.description = t.topics_removed(cnt=len(affected_topics))
await reply(ctx, embed=embed)

@commands.command(name="*")
@@ -168,43 +186,40 @@ async def register_topics(self, ctx: Context, *, topics: str):
register one or more new topics
"""

guild: Guild = ctx.guild
names = split_topics(topics)
if not names:
topics: List[tuple[str, str, Union[BTPTopic, None]]] = await split_parents(names)
if not names or not topics:
raise UserInputError

valid_chars = set(string.ascii_letters + string.digits + " !#$%&'()+-./:<=>?[\\]^_`{|}~")
to_be_created: List[str] = []
roles: List[Role] = []
for topic in names:
if any(c not in valid_chars for c in topic):
registered_topics: List[tuple[str, str, Union[BTPTopic, None]]] = []
for topic in topics:
if any(c not in valid_chars for c in topic[0]):
raise CommandError(t.topic_invalid_chars(topic))

for role in guild.roles:
if role.name.lower() == topic.lower():
break
if await db.exists(select(BTPTopic).filter_by(name=topic[0])):
raise CommandError(
t.topic_already_registered(f"{topic[1]}/{topic[2].name + '/' if topic[2] else ''}{topic[0]}"),
)
else:
to_be_created.append(topic)
continue

if await db.exists(select(BTPRole).filter_by(role_id=role.id)):
raise CommandError(t.topic_already_registered(topic))

check_role_assignable(role)

roles.append(role)
registered_topics.append(topic)

for name in to_be_created:
roles.append(await guild.create_role(name=name, mentionable=True))

for role in roles:
await BTPRole.create(role.id)
for registered_topic in registered_topics:
await BTPTopic.create(
registered_topic[0],
None,
registered_topic[1],
registered_topic[2].id if registered_topic[2] is not None else None,
)

embed = Embed(title=t.betheprofessional, colour=Colors.BeTheProfessional)
embed.description = t.topics_registered(cnt=len(roles))
embed.description = t.topics_registered(cnt=len(registered_topics))
await send_to_changelog(
ctx.guild,
t.log_topics_registered(cnt=len(roles), topics=", ".join(f"`{r}`" for r in roles)),
t.log_topics_registered(
cnt=len(registered_topics),
topics=", ".join(f"`{r[0]}`" for r in registered_topics),
),
)
await reply(ctx, embed=embed)

@@ -216,14 +231,33 @@ async def delete_topics(self, ctx: Context, *, topics: str):
delete one or more topics
"""

await unregister_roles(ctx, topics, delete_roles=True)
topics: List[str] = split_topics(topics)

@commands.command(name="%")
@BeTheProfessionalPermission.manage.check
@guild_only()
async def unregister_topics(self, ctx: Context, *, topics: str):
"""
unregister one or more topics without deleting the roles
"""
delete_topics: list[BTPTopic] = []

for topic in topics:
if not await db.exists(select(BTPTopic).filter_by(name=topic)):
raise CommandError(t.topic_not_registered(topic))
else:
btp_topic = await db.first(select(BTPTopic).filter_by(name=topic))
delete_topics.append(btp_topic)
for child_topic in await db.all(
select(BTPTopic).filter_by(parent=btp_topic.id),
): # TODO Recursive? Fix more level childs
delete_topics.insert(0, child_topic)
for topic in delete_topics:
if topic.role_id is not None:
role: Role = ctx.guild.get_role(topic.role_id)
await role.delete()
for user_topic in await db.all(select(BTPUser).filter_by(topic=topic.id)):
await db.delete(user_topic)
await db.commit()
await db.delete(topic)

await unregister_roles(ctx, topics, delete_roles=False)
embed = Embed(title=t.betheprofessional, colour=Colors.BeTheProfessional)
embed.description = t.topics_unregistered(cnt=len(delete_topics))
await send_to_changelog(
ctx.guild,
t.log_topics_unregistered(cnt=len(delete_topics), topics=", ".join(f"`{r}`" for r in delete_topics)),
)
await send_long_embed(ctx, embed)
Loading