PluralKit/PluralKit.Bot/Handlers/ReactionAdded.cs

238 lines
9.2 KiB
C#
Raw Normal View History

using System.Threading.Tasks;
using Myriad.Builders;
using Myriad.Cache;
using Myriad.Extensions;
2020-12-22 12:15:26 +00:00
using Myriad.Gateway;
using Myriad.Rest;
using Myriad.Rest.Exceptions;
using Myriad.Rest.Types;
2020-12-25 12:58:45 +00:00
using Myriad.Rest.Types.Requests;
using Myriad.Types;
2020-12-22 12:15:26 +00:00
using PluralKit.Core;
using Serilog;
namespace PluralKit.Bot
{
2020-12-22 12:15:26 +00:00
public class ReactionAdded: IEventHandler<MessageReactionAddEvent>
{
2020-08-29 11:46:27 +00:00
private readonly IDatabase _db;
private readonly ModelRepository _repo;
2020-10-23 10:18:28 +00:00
private readonly CommandMessageService _commandMessageService;
2020-08-29 11:46:27 +00:00
private readonly ILogger _logger;
private readonly IDiscordCache _cache;
2020-12-25 12:58:45 +00:00
private readonly EmbedService _embeds;
private readonly Bot _bot;
private readonly DiscordApiClient _rest;
2020-12-25 12:58:45 +00:00
public ReactionAdded(ILogger logger, IDatabase db, ModelRepository repo, CommandMessageService commandMessageService, IDiscordCache cache, Bot bot, DiscordApiClient rest, EmbedService embeds)
{
2020-08-29 11:46:27 +00:00
_db = db;
_repo = repo;
2020-10-23 10:18:28 +00:00
_commandMessageService = commandMessageService;
_cache = cache;
_bot = bot;
_rest = rest;
2020-12-25 12:58:45 +00:00
_embeds = embeds;
_logger = logger.ForContext<ReactionAdded>();
}
2020-12-22 12:15:26 +00:00
public async Task Handle(Shard shard, MessageReactionAddEvent evt)
{
await TryHandleProxyMessageReactions(evt);
}
private async ValueTask TryHandleProxyMessageReactions(MessageReactionAddEvent evt)
{
// Sometimes we get events from users that aren't in the user cache
// We just ignore all of those for now, should be quite rare...
if (!_cache.TryGetUser(evt.UserId, out var user))
return;
var channel = _cache.GetChannel(evt.ChannelId);
// check if it's a command message first
// since this can happen in DMs as well
if (evt.Emoji.Name == "\u274c")
{
await using var conn = await _db.Obtain();
var commandMsg = await _commandMessageService.GetCommandMessage(conn, evt.MessageId);
if (commandMsg != null)
{
await HandleCommandDeleteReaction(evt, commandMsg);
return;
}
}
// Proxied messages only exist in guild text channels, so skip checking if we're elsewhere
2021-07-08 16:45:59 +00:00
if (!DiscordUtils.IsValidGuildChannel(channel)) return;
2020-06-18 15:49:43 +00:00
// Ignore reactions from bots (we can't DM them anyway)
if (user.Bot) return;
2020-10-23 10:18:28 +00:00
switch (evt.Emoji.Name)
{
// Message deletion
case "\u274C": // Red X
2020-10-23 10:18:28 +00:00
{
await using var conn = await _db.Obtain();
var msg = await _repo.GetMessage(conn, evt.MessageId);
2020-10-23 10:18:28 +00:00
if (msg != null)
await HandleProxyDeleteReaction(evt, msg);
2020-10-23 10:18:28 +00:00
break;
}
case "\u2753": // Red question mark
case "\u2754": // White question mark
2020-10-23 10:18:28 +00:00
{
await using var conn = await _db.Obtain();
var msg = await _repo.GetMessage(conn, evt.MessageId);
2020-10-23 10:18:28 +00:00
if (msg != null)
await HandleQueryReaction(evt, msg);
2020-10-23 10:18:28 +00:00
break;
2020-10-23 10:18:28 +00:00
}
case "\U0001F514": // Bell
case "\U0001F6CE": // Bellhop bell
case "\U0001F3D3": // Ping pong paddle (lol)
case "\u23F0": // Alarm clock
case "\u2757": // Exclamation mark
2020-10-23 10:18:28 +00:00
{
await using var conn = await _db.Obtain();
var msg = await _repo.GetMessage(conn, evt.MessageId);
2020-10-23 10:18:28 +00:00
if (msg != null)
await HandlePingReaction(evt, msg);
break;
2020-10-23 10:18:28 +00:00
}
}
}
private async ValueTask HandleProxyDeleteReaction(MessageReactionAddEvent evt, FullMessage msg)
{
if (!_bot.PermissionsIn(evt.ChannelId).HasFlag(PermissionSet.ManageMessages))
return;
using var conn = await _db.Obtain();
var system = await _repo.GetSystemByAccount(conn, evt.UserId);
// Can only delete your own message
if (msg.System.Id != system.Id) return;
try
{
await _rest.DeleteMessage(evt.ChannelId, evt.MessageId);
}
catch (NotFoundException)
{
// Message was deleted by something/someone else before we got to it
}
await _db.Execute(c => _repo.DeleteMessage(c, evt.MessageId));
}
private async ValueTask HandleCommandDeleteReaction(MessageReactionAddEvent evt, CommandMessage msg)
{
// Can only delete your own message
if (msg.AuthorId != evt.UserId)
2020-10-23 10:18:28 +00:00
return;
try
{
await _rest.DeleteMessage(evt.ChannelId, evt.MessageId);
}
catch (NotFoundException)
{
// Message was deleted by something/someone else before we got to it
}
// No need to delete database row here, it'll get deleted by the once-per-minute scheduled task.
}
private async ValueTask HandleQueryReaction(MessageReactionAddEvent evt, FullMessage msg)
{
2020-12-25 12:58:45 +00:00
var guild = _cache.GetGuild(evt.GuildId!.Value);
// Try to DM the user info about the message
try
{
2020-12-25 12:58:45 +00:00
var dm = await _cache.GetOrCreateDmChannel(_rest, evt.UserId);
await _rest.CreateMessage(dm.Id, new MessageRequest
{
Embed = await _embeds.CreateMemberEmbed(msg.System, msg.Member, guild, LookupContext.ByNonOwner)
});
await _rest.CreateMessage(dm.Id, new MessageRequest
{
Embed = await _embeds.CreateMessageInfoEmbed(msg)
});
}
2021-03-18 10:38:28 +00:00
catch (ForbiddenException) { } // No permissions to DM, can't check for this :(
await TryRemoveOriginalReaction(evt);
}
private async ValueTask HandlePingReaction(MessageReactionAddEvent evt, FullMessage msg)
{
if (!_bot.PermissionsIn(evt.ChannelId).HasFlag(PermissionSet.ManageMessages))
return;
// Check if the "pinger" has permission to send messages in this channel
// (if not, PK shouldn't send messages on their behalf)
var member = await _rest.GetGuildMember(evt.GuildId!.Value, evt.UserId);
var requiredPerms = PermissionSet.ViewChannel | PermissionSet.SendMessages;
if (member == null || !_cache.PermissionsFor(evt.ChannelId, member).HasFlag(requiredPerms)) return;
if (msg.System.PingsEnabled)
{
// If the system has pings enabled, go ahead
await _rest.CreateMessage(evt.ChannelId, new()
{
Content = $"Psst, **{msg.Member.DisplayName()}** (<@{msg.Message.Sender}>), you have been pinged by <@{evt.UserId}>.",
Components = new []
{
new MessageComponent
{
Type = ComponentType.ActionRow,
Components = new[]
{
new MessageComponent
{
Style = ButtonStyle.Link,
Type = ComponentType.Button,
Label = "Jump",
Url = evt.JumpLink()
}
}
}
},
AllowedMentions = new AllowedMentions {Users = new[] {msg.Message.Sender}}
});
}
else
{
// If not, tell them in DMs (if we can)
try
{
2020-12-25 12:58:45 +00:00
var dm = await _cache.GetOrCreateDmChannel(_rest, evt.UserId);
await _rest.CreateMessage(dm.Id, new MessageRequest
{
Content = $"{Emojis.Error} {msg.Member.DisplayName()}'s system has disabled reaction pings. If you want to mention them anyway, you can copy/paste the following message:"
});
await _rest.CreateMessage(dm.Id, new MessageRequest {Content = $"<@{msg.Message.Sender}>".AsCode()});
}
2021-03-18 10:38:28 +00:00
catch (ForbiddenException) { }
}
await TryRemoveOriginalReaction(evt);
}
private async Task TryRemoveOriginalReaction(MessageReactionAddEvent evt)
{
if (_bot.PermissionsIn(evt.ChannelId).HasFlag(PermissionSet.ManageMessages))
2021-01-31 16:56:44 +00:00
await _rest.DeleteUserReaction(evt.ChannelId, evt.MessageId, evt.Emoji, evt.UserId);
}
}
}