-
-
Notifications
You must be signed in to change notification settings - Fork 47
/
Copy pathcog.py
276 lines (242 loc) · 12.4 KB
/
cog.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
"""
Cog implementing karma system. Users can give each other positive/negative karma points with reactions.
"""
import math
import disnake
from disnake.ext import commands, tasks
import utils
from buttons.embed import PaginationView
from cogs.base import Base
from cogs.grillbotapi.cog import GrillbotApi
from database.karma import KarmaDB
from features.leaderboard import LeaderboardPageSource
from features.reaction_context import ReactionContext
from rubbergod import Rubbergod
from utils import cooldowns
from utils.checks import PermissionsCheck
from . import features
from .messages_cz import MessagesCZ
class Karma(Base, commands.Cog):
def __init__(self, bot: Rubbergod):
super().__init__()
self.bot = bot
self.karma_helper = features.Karma(bot)
self.grillbot_api = GrillbotApi(bot)
self._leaderboard_formatter = utils.general.make_pts_column_row_formatter(KarmaDB.karma.name)
self._positive_formatter = utils.general.make_pts_column_row_formatter(KarmaDB.positive.name)
self._negative_formatter = utils.general.make_pts_column_row_formatter(KarmaDB.negative.name)
self.tasks = [self.sync_with_grillbot_task.start()]
async def handle_reaction(self, ctx: ReactionContext):
# handle karma vote
if ctx.message.content.startswith(MessagesCZ.vote_message_hack):
if ctx.emoji not in ["✅", "❌", "0⃣"]:
await ctx.message.remove_reaction(ctx.emoji, ctx.member)
else:
users = []
for reaction in ctx.message.reactions:
users.append(await reaction.users().flatten())
# Flatten the final list
users = [x for y in users for x in y]
if users.count(ctx.member) > 1:
await ctx.message.remove_reaction(ctx.emoji, ctx.member)
# reeval karma message
elif (
ctx.message.embeds
and ctx.message.embeds[0].title is not None
and ctx.message.embeds[0].title == "Karma zprávy"
and ctx.emoji == "🔁"
):
async with ctx.channel.typing():
await ctx.message.remove_reaction(ctx.emoji, ctx.member)
msg_converter = commands.MessageConverter()
try:
message = await msg_converter.convert(self, ctx.message.embeds[0].fields[0].value)
except commands.errors.MessageNotFound:
ctx.message.delete()
return
embed = await self.karma_helper.message_karma(ctx.member, message)
await ctx.message.edit(embed=embed)
# handle karma
elif (
ctx.member.id != ctx.message.author.id
and ctx.guild.id == self.config.guild_id
and ctx.message.channel.id not in self.config.karma_banned_channels
and ctx.message.channel.id != self.config.bettermeme_room
and self.config.karma_ban_role_id not in map(lambda x: x.id, ctx.member.roles)
):
emoji = utils.general.str_emoji_id(ctx.emoji)
KarmaDB.karma_emoji(ctx.message.author.id, ctx.member.id, emoji)
@commands.Cog.listener()
async def on_raw_reaction_remove(self, payload: disnake.RawReactionActionEvent):
ctx = await ReactionContext.from_payload(self.bot, payload)
if ctx is None:
return
if (
ctx.member.id != ctx.message.author.id
and ctx.guild.id == self.config.guild_id
and ctx.message.channel.id not in self.config.karma_banned_channels
and ctx.message.channel.id != self.config.bettermeme_room
and self.config.karma_ban_role_id not in map(lambda x: x.id, ctx.member.roles)
):
emoji = utils.general.str_emoji_id(ctx.emoji)
KarmaDB.karma_emoji_remove(ctx.message.author.id, ctx.member.id, emoji)
@cooldowns.default_cooldown
@commands.slash_command(name="karma", guild_ids=[Base.config.guild_id])
async def _karma(self, inter: disnake.ApplicationCommandInteraction):
pass
@_karma.sub_command(description=MessagesCZ.karma_brief)
async def me(self, inter: disnake.ApplicationCommandInteraction):
await inter.response.send_message(
self.karma_helper.karma_get(inter.author), ephemeral=PermissionsCheck.is_botroom(inter)
)
@commands.user_command(name="Karma uživatele")
async def stalk_app(self, inter: disnake.UserCommandInteraction, user: disnake.Member):
await inter.response.send_message(
self.karma_helper.karma_get(inter.author, user), ephemeral=PermissionsCheck.is_botroom(inter)
)
@_karma.sub_command(description=MessagesCZ.stalk_brief)
async def stalk(self, inter: disnake.ApplicationCommandInteraction, user: disnake.User):
await inter.response.send_message(
self.karma_helper.karma_get(inter.author, user), ephemeral=PermissionsCheck.is_botroom(inter)
)
@commands.cooldown(rate=1, per=300.0, type=commands.BucketType.guild)
@_karma.sub_command(description=MessagesCZ.getall_brief)
async def getall(self, inter: disnake.ApplicationCommandInteraction):
await inter.send(MessagesCZ.getall_response, ephemeral=PermissionsCheck.is_botroom(inter))
await self.karma_helper.emoji_list_all_values(inter, PermissionsCheck.is_botroom(inter))
@_karma.sub_command(description=MessagesCZ.get_brief)
async def get(self, inter: disnake.ApplicationCommandInteraction, emoji):
await self.karma_helper.emoji_get_value(inter, emoji, ephemeral=PermissionsCheck.is_botroom(inter))
@cooldowns.long_cooldown
@commands.message_command(name="Karma zprávy", guild_ids=[Base.config.guild_id])
async def message_app(self, inter: disnake.MessageCommandInteraction, message: disnake.Message):
await self._message(inter, message, ephemeral=PermissionsCheck.is_botroom(inter))
@cooldowns.long_cooldown
@_karma.sub_command(description=MessagesCZ.message_brief)
async def message(self, inter: disnake.ApplicationCommandInteraction, message: disnake.Message):
await self._message(inter, message)
async def _message(
self, inter: disnake.ApplicationCommandInteraction, message: disnake.Message, ephemeral: bool = False
):
await inter.response.defer(with_message=True, ephemeral=ephemeral)
embed = await self.karma_helper.message_karma(inter.author, message)
await inter.edit_original_response(embed=embed)
msg = await inter.original_message()
if not ephemeral:
await msg.add_reaction("🔁")
@cooldowns.long_cooldown
@_karma.sub_command(name="leaderboard", description=MessagesCZ.leaderboard_brief)
async def leaderboard(
self,
inter: disnake.ApplicationCommandInteraction,
direction: str = commands.Param(default="descending", choices=["ascending", "descending"]),
start: int = commands.Param(default=1, gt=0, lt=100000000, description=MessagesCZ.board_start_param),
):
"""
Get karma leaderboard
"""
await inter.response.defer(ephemeral=PermissionsCheck.is_botroom(inter))
if direction == "descending":
query = KarmaDB.leaderboard_query(KarmaDB.karma.desc())
title = "KARMA LEADERBOARD"
emote = "trophy"
else:
query = KarmaDB.leaderboard_query(KarmaDB.karma.asc())
title = "KARMA LEADERBOARD REVERSED"
emote = "coolStoryBob"
embed = disnake.Embed()
value_num = math.ceil(start / self.config.karma_grillbot_leaderboard_size)
value = MessagesCZ.karma_web if value_num == 1 else f"{MessagesCZ.karma_web}{value_num}"
embed.add_field(name=MessagesCZ.web_title, value=value)
page_source = LeaderboardPageSource(
bot=self.bot,
author=inter.author,
query=query,
row_formatter=self._leaderboard_formatter,
base_embed=embed,
title=title,
emote_name=emote,
member_id_col_name="member_ID",
)
page_num = page_source.get_page_number(start)
page = page_source.get_page(page_num)
embed = page_source.format_page(page)
view = PaginationView(inter.author, embeds=[embed], page_source=page_source)
await inter.edit_original_response(embed=embed, view=view)
view.message = await inter.original_message()
@cooldowns.long_cooldown
@_karma.sub_command(name="givingboard", description=MessagesCZ.givingboard_brief)
async def givingboard(
self,
inter: disnake.ApplicationCommandInteraction,
karma: str = commands.Param(default="positive", choices=["positive", "negative"]),
direction: str = commands.Param(default="descending", choices=["ascending", "descending"]),
start: int = commands.Param(default=1, gt=0, lt=100000000, description=MessagesCZ.board_start_param),
):
"""
Get the biggest positive/negative karma givers
"""
await inter.response.defer(ephemeral=PermissionsCheck.is_botroom(inter))
karma_column = KarmaDB.positive if karma == "positive" else KarmaDB.negative
order_karma = karma_column.desc() if direction == "descending" else karma_column.asc()
query = KarmaDB.leaderboard_query(order_karma)
formatter = self._positive_formatter if karma == "positive" else self._negative_formatter
title = "KARMA GIVINGBOARD" if karma == "positive" else "KARMA NEGATIVE GIVINGBOARD"
emote = "peepolove" if karma == "positive" else "gasbutton"
page_source = LeaderboardPageSource(
bot=self.bot,
author=inter.author,
query=query,
row_formatter=formatter,
title=title,
emote_name=emote,
member_id_col_name="member_ID",
)
page_num = page_source.get_page_number(start)
page = page_source.get_page(page_num)
embed = page_source.format_page(page)
view = PaginationView(inter.author, embeds=[embed], page_source=page_source)
await inter.edit_original_response(embed=embed, view=view)
view.message = await inter.original_message()
@PermissionsCheck.is_bot_admin()
@commands.slash_command(name="karma_mod", description=MessagesCZ.karma_brief)
async def _karma_mod(self, inter: disnake.ApplicationCommandInteraction):
pass
@PermissionsCheck.is_in_voteroom()
@_karma_mod.sub_command(name="revote", description=MessagesCZ.revote_brief)
async def revote(self, inter: disnake.ApplicationCommandInteraction, emoji):
"""Start a revote of the karma value for emojis."""
await inter.response.defer(ephemeral=True)
await self.karma_helper.emoji_revote_value(inter, emoji)
@PermissionsCheck.is_in_voteroom()
@_karma_mod.sub_command(name="vote", description=MessagesCZ.vote_brief)
async def vote(self, inter: disnake.ApplicationCommandInteraction):
"""Start a vote using emojis without a karma value."""
await inter.response.defer(ephemeral=True)
await self.karma_helper.emoji_vote_value(inter)
@_karma_mod.sub_command(name="give", description=MessagesCZ.give_brief)
async def give(self, inter: disnake.ApplicationCommandInteraction, users: str, karma: int):
await inter.response.defer()
await self.karma_helper.karma_give(inter, users, karma)
@_karma_mod.sub_command(name="transfer", description=MessagesCZ.transfer_brief)
async def transfer(
self, inter: disnake.ApplicationCommandInteraction, from_user: disnake.Member, to_user: disnake.Member
):
await inter.response.defer()
if from_user == to_user:
await inter.send(MessagesCZ.transfer_same_user)
return
await self.karma_helper.karma_transfer(inter, from_user, to_user)
@cooldowns.default_cooldown
@commands.guild_only()
async def karma(self, ctx: commands.Context):
if not (self.config.guild_id == ctx.guild.id):
await ctx.reply(MessagesCZ.server_warning)
return
command_id = utils.general.get_command_id(self.bot, "karma")
await ctx.reply(MessagesCZ.moved_command(name="karma", id=command_id))
@tasks.loop(minutes=int(Base.config.grillbot_api_karma_sync_interval))
async def sync_with_grillbot_task(self):
items = list(KarmaDB.leaderboard_query(KarmaDB.karma.asc()))
for chunk in utils.general.split_to_parts(items, 500):
await self.grillbot_api.post_karma_store(chunk)