Passed
Branch BonHowi (e58854)
by Bartosz
01:17
created

LeaderboardsCog.create_role()   A

Complexity

Conditions 2

Size

Total Lines 6
Code Lines 5

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
eloc 5
dl 0
loc 6
rs 10
c 0
b 0
f 0
cc 2
nop 3
1
from datetime import datetime
2
3
import discord
4
import pandas as pd
5
from discord.ext import commands, tasks
6
from discord.utils import get
7
from discord_slash import cog_ext
8
import cogs.cogbase as cogbase
9
from cogs.databasecog import DatabaseCog
10
11
legend_multiplier = 5
12
13
14
class LeaderboardsCog(cogbase.BaseCog):
15
    def __init__(self, base):
16
        super().__init__(base)
17
        self.update_leaderboards_loop.start()
18
19
    # Send leaderboards to specified channel
20
    async def update_leaderboards(self, channel: int, ch_type: str):
21
        top_ch = self.bot.get_channel(channel)
22
        spots_df = await DatabaseCog.db_get_spots_df()
23
        spots_df = pd.DataFrame(spots_df)
24
        spots_df["total"] = spots_df["legendary"] * legend_multiplier + spots_df["rare"]
25
        spots_df_top = spots_df.sort_values(ch_type, ascending=False).head(15)
26
        spots_df_top = spots_df_top.reset_index(drop=True)
27
        await top_ch.purge()
28
29
        top_print = []
30
        for index, row in spots_df_top.iterrows():
31
            member_stats = [f"**[{index + 1}]**  {row['display_name']} - {row[ch_type]}"]
32
            top_print.append(member_stats)
33
        top_print = ['\n'.join([elem for elem in sublist]) for sublist in top_print]
34
        top_print = "\n".join(top_print)
35
        embed_command = discord.Embed(title=f"TOP 15 {ch_type.upper()}", description=top_print,
36
                                      color=0xf1c232)
37
        member = self.bot.get_user(spots_df_top['member_id'].iloc[0])
38
        embed_command.set_thumbnail(url=f'{member.avatar_url}')
39
        await top_ch.send(embed=embed_command)
40
41
    # Update member spotting role(total/common)
42
    async def update_role(self, guild, guild_member, spot_roles, common: bool):
43
        roles_type = "common" if common else "total"
44
        try:
45
            spots_df = await DatabaseCog.db_get_member_stats(guild_member.id)
46
            spots_df["total"] = spots_df["legendary"] * legend_multiplier + spots_df["rare"]
47
            roles_list = [key for (key, value) in spot_roles.items() if spots_df.at[0, roles_type] >= value]
48
            if roles_list:
49
                await self.create_role(guild, roles_list[-1])
50
                role_new = get(guild.roles, name=roles_list[-1])
51
                await guild_member.add_roles(role_new)
52
                if len(roles_list) > 1:
53
                    await self.create_role(guild, roles_list[-2])
54
                    role_old = get(guild.roles, name=roles_list[-2])
55
                    await self.create_role(guild, roles_list[-2])
56
                    await guild_member.remove_roles(role_old)
57
        except KeyError as e:
58
            print(e)
59
60
    # Update members' spotting roles
61
    async def update_member_roles(self):
62
        guild = self.bot.get_guild(self.bot.guild[0])
63
        spot_roles_total = self.bot.config["total_milestones"][0]
64
        spot_roles_common = self.bot.config["common_milestones"][0]
65
        for guild_member in guild.members:
66
            await self.update_role(guild, guild_member, spot_roles_total, False)
67
            await self.update_role(guild, guild_member, spot_roles_common, True)
68
69
    @tasks.loop(minutes=15)
70
    async def update_leaderboards_loop(self):
71
        await self.update_leaderboards(self.bot.ch_leaderboards, "total")
72
        await self.update_leaderboards(self.bot.ch_leaderboards_common, "common")
73
        await self.update_member_roles()
74
        now = datetime.now()
75
        dt_string = now.strftime("%d/%m/%Y %H:%M:%S")
76
        print(f'({dt_string})\t[{self.__class__.__name__}]: Leaderboards updated')
77
78
    @update_leaderboards_loop.before_loop
79
    async def before_update_leaderboards_loop(self):
80
        now = datetime.now()
81
        dt_string = now.strftime("%d/%m/%Y %H:%M:%S")
82
        print(f'({dt_string})\t[{self.__class__.__name__}]: Waiting until Bot is ready')
83
        await self.bot.wait_until_ready()
84
85
    # Get own spotting stats
86
    @cog_ext.cog_slash(name="myStats", guild_ids=cogbase.GUILD_IDS,
87
                       description="Get your spot stats",
88
                       default_permission=True)
89
    async def get_stats(self, ctx):
90
        spot_roles = self.bot.config["total_milestones"][0]
91
        guild = self.bot.get_guild(self.bot.guild[0])
92
        spots_df = await DatabaseCog.db_get_member_stats(ctx.author.id)
93
        spots_df["total"] = spots_df["legendary"] * legend_multiplier + spots_df["rare"]
94
95
        role_new = ""
96
        spots_for_new = -1
97
        roles_list = [key for (key, value) in spot_roles.items() if spots_df.at[0, "total"] < value]
98
        values_list = [value for (key, value) in spot_roles.items() if spots_df.at[0, "total"] < value]
99
        if roles_list:
100
            role_new = get(guild.roles, name=roles_list[0])
101
            spots_for_new = values_list[0]
102
103
        message = f"**Legends**: {spots_df.at[0, 'legendary']}\n" \
104
                  f"**Rares**: {spots_df.at[0, 'rare']}\n" \
105
                  f"**Commons**: {spots_df.at[0, 'common']}\n\n" \
106
                  f"**Total points**: {spots_df.at[0, 'total']}\n" \
107
                  f"**Progress**: {spots_df.at[0, 'total']}/{spots_for_new}\n" \
108
                  f"**Next role**: _{role_new}_"
109
110
        await ctx.send(f"{ctx.author.mention} stats:\n{message}", hidden=True)
111
112
113
def setup(bot: commands.Bot):
114
    bot.add_cog(LeaderboardsCog(bot))
115