PluralKit/PluralKit.Bot/Commands/MemberCommands.cs

295 lines
15 KiB
C#
Raw Normal View History

using System.Linq;
2019-05-11 21:56:56 +00:00
using System.Text.RegularExpressions;
2019-04-27 14:30:34 +00:00
using System.Threading.Tasks;
using Discord;
2019-05-13 20:44:49 +00:00
using NodaTime;
2019-10-05 05:41:00 +00:00
using PluralKit.Bot.CommandSystem;
2019-07-09 22:19:18 +00:00
using PluralKit.Core;
2019-04-27 14:30:34 +00:00
namespace PluralKit.Bot.Commands
{
2019-10-05 05:41:00 +00:00
public class MemberCommands
2019-04-27 14:30:34 +00:00
{
2019-10-05 05:41:00 +00:00
private SystemStore _systems;
private MemberStore _members;
private EmbedService _embeds;
private ProxyCacheService _proxyCache;
2019-04-27 14:30:34 +00:00
2019-10-05 05:41:00 +00:00
public MemberCommands(SystemStore systems, MemberStore members, EmbedService embeds, ProxyCacheService proxyCache)
{
_systems = systems;
_members = members;
_embeds = embeds;
_proxyCache = proxyCache;
}
2019-04-27 14:30:34 +00:00
2019-10-05 05:41:00 +00:00
public async Task NewMember(Context ctx) {
if (ctx.System == null) throw Errors.NoSystemError;
var memberName = ctx.RemainderOrNull() ?? throw new PKSyntaxError("You must pass a member name.");
// Hard name length cap
if (memberName.Length > Limits.MaxMemberNameLength) throw Errors.MemberNameTooLongError(memberName.Length);
2019-04-27 14:30:34 +00:00
// Warn if member name will be unproxyable (with/without tag)
2019-10-05 05:41:00 +00:00
if (memberName.Length > ctx.System.MaxMemberNameLength) {
var msg = await ctx.Reply($"{Emojis.Warn} Member name too long ({memberName.Length} > {ctx.System.MaxMemberNameLength} characters), this member will be unproxyable. Do you want to create it anyway? (You can change the name later, or set a member display name)");
if (!await ctx.PromptYesNo(msg)) throw new PKError("Member creation cancelled.");
2019-04-27 14:30:34 +00:00
}
// Warn if there's already a member by this name
2019-10-05 05:41:00 +00:00
var existingMember = await _members.GetByName(ctx.System, memberName);
2019-04-27 14:30:34 +00:00
if (existingMember != null) {
2019-10-18 11:14:36 +00:00
var msg = await ctx.Reply($"{Emojis.Warn} You already have a member in your system with the name \"{existingMember.Name.SanitizeMentions()}\" (with ID `{existingMember.Hid}`). Do you want to create another member with the same name?");
2019-10-05 05:41:00 +00:00
if (!await ctx.PromptYesNo(msg)) throw new PKError("Member creation cancelled.");
2019-04-27 14:30:34 +00:00
}
// Enforce per-system member limit
var memberCount = await _members.MemberCount(ctx.System);
if (memberCount >= Limits.MaxMemberCount)
throw Errors.MemberLimitReachedError;
2019-04-27 14:30:34 +00:00
// Create the member
2019-10-05 05:41:00 +00:00
var member = await _members.Create(ctx.System, memberName);
memberCount++;
2019-04-27 14:30:34 +00:00
2019-04-29 17:43:09 +00:00
// Send confirmation and space hint
2019-10-18 11:14:36 +00:00
await ctx.Reply($"{Emojis.Success} Member \"{memberName.SanitizeMentions()}\" (`{member.Hid}`) registered! See the user guide for commands for editing this member: https://pluralkit.me/guide#member-management");
if (memberName.Contains(" "))
await ctx.Reply($"{Emojis.Note} Note that this member's name contains spaces. You will need to surround it with \"double quotes\" when using commands referring to it, or just use the member's 5-character ID (which is `{member.Hid}`).");
if (memberCount >= Limits.MaxMemberCount)
await ctx.Reply($"{Emojis.Warn} You have reached the per-system member limit ({Limits.MaxMemberCount}). You will be unable to create additional members until existing members are deleted.");
else if (memberCount >= Limits.MaxMembersWarnThreshold)
await ctx.Reply($"{Emojis.Warn} You are approaching the per-system member limit ({memberCount} / {Limits.MaxMemberCount} members). Please review your member list for unused or duplicate members.");
2019-10-05 05:41:00 +00:00
await _proxyCache.InvalidateResultsForSystem(ctx.System);
2019-04-27 14:30:34 +00:00
}
2019-10-05 05:41:00 +00:00
public async Task RenameMember(Context ctx, PKMember target) {
2019-04-29 17:43:09 +00:00
// TODO: this method is pretty much a 1:1 copy/paste of the above creation method, find a way to clean?
2019-10-05 05:41:00 +00:00
if (ctx.System == null) throw Errors.NoSystemError;
if (target.System != ctx.System.Id) throw Errors.NotOwnMemberError;
var newName = ctx.RemainderOrNull();
2019-04-29 17:43:09 +00:00
// Hard name length cap
if (newName.Length > Limits.MaxMemberNameLength) throw Errors.MemberNameTooLongError(newName.Length);
2019-08-09 08:12:38 +00:00
// Warn if member name will be unproxyable (with/without tag), only if member doesn't have a display name
2019-10-05 05:41:00 +00:00
if (target.DisplayName == null && newName.Length > ctx.System.MaxMemberNameLength) {
var msg = await ctx.Reply($"{Emojis.Warn} New member name too long ({newName.Length} > {ctx.System.MaxMemberNameLength} characters), this member will be unproxyable. Do you want to change it anyway? (You can set a member display name instead)");
if (!await ctx.PromptYesNo(msg)) throw new PKError("Member renaming cancelled.");
2019-04-29 17:43:09 +00:00
}
// Warn if there's already a member by this name
2019-10-05 05:41:00 +00:00
var existingMember = await _members.GetByName(ctx.System, newName);
2019-04-29 17:43:09 +00:00
if (existingMember != null) {
2019-10-18 11:14:36 +00:00
var msg = await ctx.Reply($"{Emojis.Warn} You already have a member in your system with the name \"{existingMember.Name.SanitizeMentions()}\" (`{existingMember.Hid}`). Do you want to rename this member to that name too?");
2019-10-05 05:41:00 +00:00
if (!await ctx.PromptYesNo(msg)) throw new PKError("Member renaming cancelled.");
2019-04-29 17:43:09 +00:00
}
2019-05-11 21:56:56 +00:00
// Rename the member
2019-10-05 05:41:00 +00:00
target.Name = newName;
await _members.Save(target);
2019-04-29 17:43:09 +00:00
2019-10-05 05:41:00 +00:00
await ctx.Reply($"{Emojis.Success} Member renamed.");
if (newName.Contains(" ")) await ctx.Reply($"{Emojis.Note} Note that this member's name now contains spaces. You will need to surround it with \"double quotes\" when using commands referring to it.");
2019-10-18 11:14:36 +00:00
if (target.DisplayName != null) await ctx.Reply($"{Emojis.Note} Note that this member has a display name set ({target.DisplayName.SanitizeMentions()}), and will be proxied using that name instead.");
2019-10-05 05:41:00 +00:00
await _proxyCache.InvalidateResultsForSystem(ctx.System);
2019-04-29 17:43:09 +00:00
}
2019-10-05 05:41:00 +00:00
public async Task MemberDescription(Context ctx, PKMember target) {
if (ctx.System == null) throw Errors.NoSystemError;
if (target.System != ctx.System.Id) throw Errors.NotOwnMemberError;
2019-04-29 17:43:09 +00:00
2019-10-05 05:41:00 +00:00
var description = ctx.RemainderOrNull();
2019-05-11 21:56:56 +00:00
if (description.IsLongerThan(Limits.MaxDescriptionLength)) throw Errors.DescriptionTooLongError(description.Length);
2019-04-29 18:33:21 +00:00
2019-10-05 05:41:00 +00:00
target.Description = description;
await _members.Save(target);
2019-04-29 18:33:21 +00:00
2019-10-05 05:41:00 +00:00
await ctx.Reply($"{Emojis.Success} Member description {(description == null ? "cleared" : "changed")}.");
2019-04-29 18:33:21 +00:00
}
2019-10-05 05:41:00 +00:00
public async Task MemberPronouns(Context ctx, PKMember target) {
if (ctx.System == null) throw Errors.NoSystemError;
if (target.System != ctx.System.Id) throw Errors.NotOwnMemberError;
2019-04-29 18:33:21 +00:00
2019-10-05 05:41:00 +00:00
var pronouns = ctx.RemainderOrNull();
2019-05-11 21:56:56 +00:00
if (pronouns.IsLongerThan(Limits.MaxPronounsLength)) throw Errors.MemberPronounsTooLongError(pronouns.Length);
2019-04-29 18:36:09 +00:00
2019-10-05 05:41:00 +00:00
target.Pronouns = pronouns;
await _members.Save(target);
2019-04-29 18:36:09 +00:00
2019-10-05 05:41:00 +00:00
await ctx.Reply($"{Emojis.Success} Member pronouns {(pronouns == null ? "cleared" : "changed")}.");
2019-04-29 18:36:09 +00:00
}
2019-10-05 05:41:00 +00:00
public async Task MemberColor(Context ctx, PKMember target)
2019-05-11 21:56:56 +00:00
{
2019-10-05 05:41:00 +00:00
if (ctx.System == null) throw Errors.NoSystemError;
if (target.System != ctx.System.Id) throw Errors.NotOwnMemberError;
var color = ctx.RemainderOrNull();
2019-05-11 21:56:56 +00:00
if (color != null)
{
if (color.StartsWith("#")) color = color.Substring(1);
if (!Regex.IsMatch(color, "^[0-9a-fA-F]{6}$")) throw Errors.InvalidColorError(color);
2019-05-11 21:56:56 +00:00
}
2019-10-05 05:41:00 +00:00
target.Color = color;
await _members.Save(target);
2019-05-11 21:56:56 +00:00
2019-10-05 05:41:00 +00:00
await ctx.Reply($"{Emojis.Success} Member color {(color == null ? "cleared" : "changed")}.");
2019-05-11 21:56:56 +00:00
}
2019-05-11 22:44:02 +00:00
2019-10-05 05:41:00 +00:00
public async Task MemberBirthday(Context ctx, PKMember target)
2019-05-13 20:44:49 +00:00
{
2019-10-05 05:41:00 +00:00
if (ctx.System == null) throw Errors.NoSystemError;
if (target.System != ctx.System.Id) throw Errors.NotOwnMemberError;
2019-05-13 20:44:49 +00:00
LocalDate? date = null;
2019-10-05 05:41:00 +00:00
var birthday = ctx.RemainderOrNull();
2019-05-13 20:44:49 +00:00
if (birthday != null)
{
date = PluralKit.Utils.ParseDate(birthday, true);
if (date == null) throw Errors.BirthdayParseError(birthday);
}
2019-10-05 05:41:00 +00:00
target.Birthday = date;
await _members.Save(target);
2019-05-13 20:44:49 +00:00
2019-10-05 05:41:00 +00:00
await ctx.Reply($"{Emojis.Success} Member birthdate {(date == null ? "cleared" : $"changed to {target.BirthdayString}")}.");
2019-05-13 20:44:49 +00:00
}
2019-10-05 05:41:00 +00:00
public async Task MemberProxy(Context ctx, PKMember target)
2019-05-13 20:56:22 +00:00
{
2019-10-05 05:41:00 +00:00
if (ctx.System == null) throw Errors.NoSystemError;
if (target.System != ctx.System.Id) throw Errors.NotOwnMemberError;
2019-05-13 20:56:22 +00:00
// Handling the clear case in an if here to keep the body dedented
2019-10-05 05:41:00 +00:00
var exampleProxy = ctx.RemainderOrNull();
2019-05-13 20:56:22 +00:00
if (exampleProxy == null)
{
// Just reset and send OK message
2019-10-05 05:41:00 +00:00
target.Prefix = null;
target.Suffix = null;
await _members.Save(target);
await ctx.Reply($"{Emojis.Success} Member proxy tags cleared.");
2019-05-13 20:56:22 +00:00
return;
}
// Make sure there's one and only one instance of "text" in the example proxy given
var prefixAndSuffix = exampleProxy.Split("text");
if (prefixAndSuffix.Length < 2) throw Errors.ProxyMustHaveText;
if (prefixAndSuffix.Length > 2) throw Errors.ProxyMultipleText;
// If the prefix/suffix is empty, use "null" instead (for DB)
2019-10-05 05:41:00 +00:00
target.Prefix = prefixAndSuffix[0].Length > 0 ? prefixAndSuffix[0] : null;
target.Suffix = prefixAndSuffix[1].Length > 0 ? prefixAndSuffix[1] : null;
await _members.Save(target);
2019-10-18 11:14:36 +00:00
await ctx.Reply($"{Emojis.Success} Member proxy tags changed to `{target.ProxyString.SanitizeMentions()}`. Try proxying now!");
2019-10-05 05:41:00 +00:00
await _proxyCache.InvalidateResultsForSystem(ctx.System);
2019-05-13 20:56:22 +00:00
}
2019-10-05 05:41:00 +00:00
public async Task MemberDelete(Context ctx, PKMember target)
2019-05-13 21:08:44 +00:00
{
2019-10-05 05:41:00 +00:00
if (ctx.System == null) throw Errors.NoSystemError;
if (target.System != ctx.System.Id) throw Errors.NotOwnMemberError;
2019-10-18 11:14:36 +00:00
await ctx.Reply($"{Emojis.Warn} Are you sure you want to delete \"{target.Name.SanitizeMentions()}\"? If so, reply to this message with the member's ID (`{target.Hid}`). __***This cannot be undone!***__");
2019-10-05 05:41:00 +00:00
if (!await ctx.ConfirmWithReply(target.Hid)) throw Errors.MemberDeleteCancelled;
await _members.Delete(target);
await ctx.Reply($"{Emojis.Success} Member deleted.");
await _proxyCache.InvalidateResultsForSystem(ctx.System);
2019-05-13 21:08:44 +00:00
}
2019-10-05 05:41:00 +00:00
public async Task MemberAvatar(Context ctx, PKMember target)
2019-07-16 18:17:04 +00:00
{
2019-10-05 05:41:00 +00:00
if (ctx.System == null) throw Errors.NoSystemError;
if (target.System != ctx.System.Id) throw Errors.NotOwnMemberError;
2019-07-16 18:17:04 +00:00
2019-10-05 05:41:00 +00:00
if (await ctx.MatchUser() is IUser user)
{
if (user.AvatarId == null) throw Errors.UserHasNoAvatar;
target.AvatarUrl = user.GetAvatarUrl(ImageFormat.Png, size: 256);
await _members.Save(target);
2019-10-05 05:41:00 +00:00
var embed = new EmbedBuilder().WithImageUrl(target.AvatarUrl).Build();
await ctx.Reply(
$"{Emojis.Success} Member avatar changed to {user.Username}'s avatar! {Emojis.Warn} Please note that if {user.Username} changes their avatar, the webhook's avatar will need to be re-set.", embed: embed);
2019-07-16 18:17:04 +00:00
2019-10-05 05:41:00 +00:00
}
else if (ctx.RemainderOrNull() is string url)
{
await Utils.VerifyAvatarOrThrow(url);
target.AvatarUrl = url;
await _members.Save(target);
2019-10-05 05:41:00 +00:00
var embed = new EmbedBuilder().WithImageUrl(url).Build();
await ctx.Reply($"{Emojis.Success} Member avatar changed.", embed: embed);
}
else if (ctx.Message.Attachments.FirstOrDefault() is Attachment attachment)
{
await Utils.VerifyAvatarOrThrow(attachment.Url);
target.AvatarUrl = attachment.Url;
await _members.Save(target);
2019-10-05 05:41:00 +00:00
await ctx.Reply($"{Emojis.Success} Member avatar changed to attached image. Please note that if you delete the message containing the attachment, the avatar will stop working.");
}
else
{
target.AvatarUrl = null;
await _members.Save(target);
await ctx.Reply($"{Emojis.Success} Member avatar cleared.");
}
2019-10-05 05:41:00 +00:00
await _proxyCache.InvalidateResultsForSystem(ctx.System);
}
2019-10-05 05:41:00 +00:00
public async Task MemberDisplayName(Context ctx, PKMember target)
2019-08-09 08:12:38 +00:00
{
2019-10-05 05:41:00 +00:00
if (ctx.System == null) throw Errors.NoSystemError;
if (target.System != ctx.System.Id) throw Errors.NotOwnMemberError;
var newDisplayName = ctx.RemainderOrNull();
2019-08-09 08:12:38 +00:00
// Refuse if proxy name will be unproxyable (with/without tag)
2019-10-05 05:41:00 +00:00
if (newDisplayName != null && newDisplayName.Length > ctx.System.MaxMemberNameLength)
throw Errors.DisplayNameTooLong(newDisplayName, ctx.System.MaxMemberNameLength);
2019-08-09 08:12:38 +00:00
2019-10-05 05:41:00 +00:00
target.DisplayName = newDisplayName;
await _members.Save(target);
2019-08-09 08:12:38 +00:00
var successStr = $"{Emojis.Success} ";
if (newDisplayName != null)
{
successStr +=
2019-10-18 11:14:36 +00:00
$"Member display name changed. This member will now be proxied using the name \"{newDisplayName.SanitizeMentions()}\".";
2019-08-09 08:12:38 +00:00
}
else
{
successStr += $"Member display name cleared. ";
// If we're removing display name and the *real* name will be unproxyable, warn.
2019-10-05 05:41:00 +00:00
if (target.Name.Length > ctx.System.MaxMemberNameLength)
2019-08-09 08:12:38 +00:00
successStr +=
2019-10-05 05:41:00 +00:00
$" {Emojis.Warn} This member's actual name is too long ({target.Name.Length} > {ctx.System.MaxMemberNameLength} characters), and thus cannot be proxied.";
2019-08-09 08:12:38 +00:00
else
2019-10-18 11:14:36 +00:00
successStr += $"This member will now be proxied using their member name \"{target.Name.SanitizeMentions()}\".";
2019-08-09 08:12:38 +00:00
}
2019-10-05 05:41:00 +00:00
await ctx.Reply(successStr);
2019-10-05 05:41:00 +00:00
await _proxyCache.InvalidateResultsForSystem(ctx.System);
2019-05-11 22:44:02 +00:00
}
2019-10-05 05:41:00 +00:00
public async Task ViewMember(Context ctx, PKMember target)
2019-04-27 14:30:34 +00:00
{
2019-10-05 05:41:00 +00:00
var system = await _systems.GetById(target.System);
await ctx.Reply(embed: await _embeds.CreateMemberEmbed(system, target));
2019-04-27 14:30:34 +00:00
}
}
}