2020-03-04 17:13:36 +00:00
using System.Text.RegularExpressions ;
2020-02-01 12:03:02 +00:00
using System.Threading.Tasks ;
2020-06-13 14:03:57 +00:00
using Dapper ;
2020-02-27 23:23:54 +00:00
2020-06-13 14:03:57 +00:00
using DSharpPlus.Entities ;
2020-02-01 12:03:02 +00:00
using PluralKit.Core ;
2020-02-12 14:16:19 +00:00
namespace PluralKit.Bot
2020-02-01 12:03:02 +00:00
{
public class MemberEdit
{
2020-06-13 14:03:57 +00:00
private readonly IDataStore _data ;
2020-06-13 17:36:43 +00:00
private readonly IDatabase _db ;
2020-02-01 12:03:02 +00:00
2020-06-13 17:36:43 +00:00
public MemberEdit ( IDataStore data , IDatabase db )
2020-02-01 12:03:02 +00:00
{
_data = data ;
2020-06-13 14:03:57 +00:00
_db = db ;
2020-02-01 12:03:02 +00:00
}
public async Task Name ( Context ctx , PKMember target ) {
// TODO: this method is pretty much a 1:1 copy/paste of the above creation method, find a way to clean?
if ( ctx . System = = null ) throw Errors . NoSystemError ;
if ( target . System ! = ctx . System . Id ) throw Errors . NotOwnMemberError ;
var newName = ctx . RemainderOrNull ( ) ? ? throw new PKSyntaxError ( "You must pass a new name for the member." ) ;
// Hard name length cap
if ( newName . Length > Limits . MaxMemberNameLength ) throw Errors . MemberNameTooLongError ( newName . Length ) ;
// Warn if there's already a member by this name
var existingMember = await _data . GetMemberByName ( ctx . System , newName ) ;
if ( existingMember ! = null ) {
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?" ) ;
if ( ! await ctx . PromptYesNo ( msg ) ) throw new PKError ( "Member renaming cancelled." ) ;
}
// Rename the member
target . Name = newName ;
await _data . SaveMember ( target ) ;
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." ) ;
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." ) ;
if ( ctx . Guild ! = null )
{
2020-06-13 14:03:57 +00:00
var memberGuildConfig = await _db . Execute ( c = > c . QueryOrInsertMemberGuildConfig ( ctx . Guild . Id , target . Id ) ) ;
2020-02-01 12:03:02 +00:00
if ( memberGuildConfig . DisplayName ! = null )
await ctx . Reply ( $"{Emojis.Note} Note that this member has a server name set ({memberGuildConfig.DisplayName.SanitizeMentions()}) in this server ({ctx.Guild.Name.SanitizeMentions()}), and will be proxied using that name here." ) ;
}
}
2020-03-04 17:13:36 +00:00
private void CheckReadMemberPermission ( Context ctx , PKMember target )
{
if ( ! target . MemberPrivacy . CanAccess ( ctx . LookupContextFor ( target . System ) ) )
throw Errors . LookupNotAllowed ;
}
private void CheckEditMemberPermission ( Context ctx , PKMember target )
{
if ( target . System ! = ctx . System ? . Id ) throw Errors . NotOwnMemberError ;
}
2020-02-01 12:03:02 +00:00
2020-03-04 17:13:36 +00:00
private static bool MatchClear ( Context ctx ) = >
ctx . Match ( "clear" ) | | ctx . MatchFlag ( "c" , "clear" ) ;
2020-02-27 23:23:54 +00:00
2020-03-04 17:13:36 +00:00
public async Task Description ( Context ctx , PKMember target ) {
if ( MatchClear ( ctx ) )
2020-02-27 23:23:54 +00:00
{
2020-03-04 17:13:36 +00:00
CheckEditMemberPermission ( ctx , target ) ;
target . Description = null ;
await _data . SaveMember ( target ) ;
await ctx . Reply ( $"{Emojis.Success} Member description cleared." ) ;
}
else if ( ! ctx . HasNext ( ) )
{
CheckReadMemberPermission ( ctx , target ) ;
2020-02-27 23:23:54 +00:00
if ( target . Description = = null )
2020-03-04 17:13:36 +00:00
if ( ctx . System ? . Id = = target . System )
await ctx . Reply ( $"This member does not have a description set. To set one, type `pk;member {target.Hid} description <description>`." ) ;
2020-02-27 23:23:54 +00:00
else
await ctx . Reply ( "This member does not have a description set." ) ;
else if ( ctx . MatchFlag ( "r" , "raw" ) )
2020-03-08 09:34:47 +00:00
await ctx . Reply ( $"```\n{target.Description.SanitizeMentions()}\n```" ) ;
2020-02-27 23:23:54 +00:00
else
2020-04-24 19:50:28 +00:00
await ctx . Reply ( embed : new DiscordEmbedBuilder ( )
2020-02-27 23:23:54 +00:00
. WithTitle ( "Member description" )
. WithDescription ( target . Description )
2020-03-04 17:13:36 +00:00
. AddField ( "\u200B" , $"To print the description with formatting, type `pk;member {target.Hid} description -raw`."
+ ( ctx . System ? . Id = = target . System ? $" To clear it, type `pk;member {target.Hid} description -clear`." : "" ) )
2020-02-27 23:23:54 +00:00
. Build ( ) ) ;
}
2020-03-04 17:13:36 +00:00
else
{
CheckEditMemberPermission ( ctx , target ) ;
2020-02-27 23:23:54 +00:00
2020-03-04 17:13:36 +00:00
var description = ctx . RemainderOrNull ( ) . NormalizeLineEndSpacing ( ) ;
if ( description . IsLongerThan ( Limits . MaxDescriptionLength ) )
throw Errors . DescriptionTooLongError ( description . Length ) ;
target . Description = description ;
await _data . SaveMember ( target ) ;
await ctx . Reply ( $"{Emojis.Success} Member description changed." ) ;
}
2020-02-01 12:03:02 +00:00
}
public async Task Pronouns ( Context ctx , PKMember target ) {
2020-03-04 17:13:36 +00:00
if ( MatchClear ( ctx ) )
{
CheckEditMemberPermission ( ctx , target ) ;
target . Pronouns = null ;
await _data . SaveMember ( target ) ;
await ctx . Reply ( $"{Emojis.Success} Member pronouns cleared." ) ;
}
else if ( ! ctx . HasNext ( ) )
{
CheckReadMemberPermission ( ctx , target ) ;
if ( target . Pronouns = = null )
if ( ctx . System ? . Id = = target . System )
await ctx . Reply ( $"This member does not have pronouns set. To set some, type `pk;member {target.Hid} pronouns <pronouns>`." ) ;
else
await ctx . Reply ( "This member does not have pronouns set." ) ;
else
await ctx . Reply ( $"**{target.Name.SanitizeMentions()}**'s pronouns are **{target.Pronouns.SanitizeMentions()}**."
+ ( ctx . System ? . Id = = target . System ? $" To clear them, type `pk;member {target.Hid} pronouns -clear`." : "" ) ) ;
}
else
{
CheckEditMemberPermission ( ctx , target ) ;
2020-02-01 12:03:02 +00:00
2020-03-04 17:13:36 +00:00
var pronouns = ctx . RemainderOrNull ( ) . NormalizeLineEndSpacing ( ) ;
if ( pronouns . IsLongerThan ( Limits . MaxPronounsLength ) )
throw Errors . MemberPronounsTooLongError ( pronouns . Length ) ;
target . Pronouns = pronouns ;
await _data . SaveMember ( target ) ;
await ctx . Reply ( $"{Emojis.Success} Member pronouns changed." ) ;
}
2020-02-01 12:03:02 +00:00
}
public async Task Color ( Context ctx , PKMember target )
{
var color = ctx . RemainderOrNull ( ) ;
2020-03-04 17:13:36 +00:00
if ( MatchClear ( ctx ) )
2020-02-01 12:03:02 +00:00
{
2020-03-04 17:13:36 +00:00
CheckEditMemberPermission ( ctx , target ) ;
target . Color = null ;
await _data . SaveMember ( target ) ;
await ctx . Reply ( $"{Emojis.Success} Member color cleared." ) ;
2020-02-01 12:03:02 +00:00
}
2020-03-04 17:13:36 +00:00
else if ( ! ctx . HasNext ( ) )
{
CheckReadMemberPermission ( ctx , target ) ;
2020-02-01 12:03:02 +00:00
2020-03-04 17:13:36 +00:00
if ( target . Color = = null )
if ( ctx . System ? . Id = = target . System )
await ctx . Reply (
$"This member does not have a color set. To set one, type `pk;member {target.Hid} color <color>`." ) ;
else
await ctx . Reply ( "This member does not have a color set." ) ;
else
2020-04-24 19:50:28 +00:00
await ctx . Reply ( embed : new DiscordEmbedBuilder ( )
2020-03-04 17:13:36 +00:00
. WithTitle ( "Member color" )
. WithColor ( target . Color . ToDiscordColor ( ) . Value )
. WithThumbnailUrl ( $"https://fakeimg.pl/256x256/{target.Color}/?text=%20" )
. WithDescription ( $"This member's color is **#{target.Color}**."
+ ( ctx . System ? . Id = = target . System ? $" To clear it, type `pk;member {target.Hid} color -clear`." : "" ) )
. Build ( ) ) ;
}
else
{
CheckEditMemberPermission ( ctx , target ) ;
2020-02-01 12:03:02 +00:00
2020-03-04 17:13:36 +00:00
if ( color . StartsWith ( "#" ) ) color = color . Substring ( 1 ) ;
if ( ! Regex . IsMatch ( color , "^[0-9a-fA-F]{6}$" ) ) throw Errors . InvalidColorError ( color ) ;
target . Color = color . ToLower ( ) ;
await _data . SaveMember ( target ) ;
2020-04-24 19:50:28 +00:00
await ctx . Reply ( embed : new DiscordEmbedBuilder ( )
2020-03-04 17:13:36 +00:00
. WithTitle ( $"{Emojis.Success} Member color changed." )
. WithColor ( target . Color . ToDiscordColor ( ) . Value )
. WithThumbnailUrl ( $"https://fakeimg.pl/256x256/{target.Color}/?text=%20" )
. Build ( ) ) ;
}
2020-02-01 12:03:02 +00:00
}
public async Task Birthday ( Context ctx , PKMember target )
{
2020-03-04 17:13:36 +00:00
if ( MatchClear ( ctx ) )
{
CheckEditMemberPermission ( ctx , target ) ;
target . Birthday = null ;
await _data . SaveMember ( target ) ;
await ctx . Reply ( $"{Emojis.Success} Member birthdate cleared." ) ;
}
else if ( ! ctx . HasNext ( ) )
2020-02-01 12:03:02 +00:00
{
2020-03-04 17:13:36 +00:00
CheckReadMemberPermission ( ctx , target ) ;
if ( target . Birthday = = null )
await ctx . Reply ( "This member does not have a birthdate set."
+ ( ctx . System ? . Id = = target . System ? $" To set one, type `pk;member {target.Hid} birthdate <birthdate>`." : "" ) ) ;
else
await ctx . Reply ( $"This member's birthdate is **{target.BirthdayString}**."
+ ( ctx . System ? . Id = = target . System ? $" To clear it, type `pk;member {target.Hid} birthdate -clear`." : "" ) ) ;
}
else
{
CheckEditMemberPermission ( ctx , target ) ;
var birthdayStr = ctx . RemainderOrNull ( ) ;
var birthday = DateUtils . ParseDate ( birthdayStr , true ) ;
if ( birthday = = null ) throw Errors . BirthdayParseError ( birthdayStr ) ;
target . Birthday = birthday ;
await _data . SaveMember ( target ) ;
await ctx . Reply ( $"{Emojis.Success} Member birthdate changed." ) ;
2020-02-01 12:03:02 +00:00
}
}
2020-03-04 17:13:36 +00:00
2020-04-24 19:50:28 +00:00
private async Task < DiscordEmbedBuilder > CreateMemberNameInfoEmbed ( Context ctx , PKMember target )
2020-03-04 17:13:36 +00:00
{
MemberGuildSettings memberGuildConfig = null ;
if ( ctx . Guild ! = null )
2020-06-13 14:03:57 +00:00
memberGuildConfig = await _db . Execute ( c = > c . QueryOrInsertMemberGuildConfig ( ctx . Guild . Id , target . Id ) ) ;
2020-02-01 12:03:02 +00:00
2020-04-24 19:50:28 +00:00
var eb = new DiscordEmbedBuilder ( ) . WithTitle ( $"Member names" )
2020-03-04 17:13:36 +00:00
. WithFooter ( $"Member ID: {target.Hid} | Active name in bold. Server name overrides display name, which overrides base name." ) ;
2020-02-01 12:03:02 +00:00
2020-03-04 17:13:36 +00:00
if ( target . DisplayName = = null & & memberGuildConfig ? . DisplayName = = null )
eb . AddField ( $"Name" , $"**{target.Name}**" ) ;
2020-02-01 12:03:02 +00:00
else
2020-03-04 17:13:36 +00:00
eb . AddField ( "Name" , target . Name ) ;
if ( target . DisplayName ! = null & & memberGuildConfig ? . DisplayName = = null )
eb . AddField ( $"Display Name" , $"**{target.DisplayName}**" ) ;
else
eb . AddField ( "Display Name" , target . DisplayName ? ? "*(none)*" ) ;
2020-02-01 12:03:02 +00:00
if ( ctx . Guild ! = null )
{
2020-03-04 17:13:36 +00:00
if ( memberGuildConfig ? . DisplayName ! = null )
eb . AddField ( $"Server Name (in {ctx.Guild.Name.SanitizeMentions()})" , $"**{memberGuildConfig.DisplayName}**" ) ;
else
eb . AddField ( $"Server Name (in {ctx.Guild.Name.SanitizeMentions()})" , memberGuildConfig ? . DisplayName ? ? "*(none)*" ) ;
2020-02-01 12:03:02 +00:00
}
2020-03-04 17:13:36 +00:00
return eb ;
2020-02-01 12:03:02 +00:00
}
2020-03-04 17:13:36 +00:00
public async Task DisplayName ( Context ctx , PKMember target )
2020-02-01 12:03:02 +00:00
{
2020-03-04 17:13:36 +00:00
async Task PrintSuccess ( string text )
{
var successStr = text ;
if ( ctx . Guild ! = null )
{
2020-06-13 14:03:57 +00:00
var memberGuildConfig = await _db . Execute ( c = > c . QueryOrInsertMemberGuildConfig ( ctx . Guild . Id , target . Id ) ) ;
2020-03-04 17:13:36 +00:00
if ( memberGuildConfig . DisplayName ! = null )
successStr + = $" However, this member has a server name set in this server ({ctx.Guild.Name.SanitizeMentions()}), and will be proxied using that name, \" { memberGuildConfig . DisplayName . SanitizeMentions ( ) } \ ", here." ;
}
await ctx . Reply ( successStr ) ;
}
2020-02-01 12:03:02 +00:00
2020-03-04 17:13:36 +00:00
if ( MatchClear ( ctx ) )
{
CheckEditMemberPermission ( ctx , target ) ;
target . DisplayName = null ;
await _data . SaveMember ( target ) ;
await PrintSuccess ( $"{Emojis.Success} Member display name cleared. This member will now be proxied using their member name \" { target . Name . SanitizeMentions ( ) } \ "." ) ;
}
else if ( ! ctx . HasNext ( ) )
{
// No perms check, display name isn't covered by member privacy
var eb = await CreateMemberNameInfoEmbed ( ctx , target ) ;
if ( ctx . System ? . Id = = target . System )
eb . WithDescription ( $"To change display name, type `pk;member {target.Hid} displayname <display name>`.\nTo clear it, type `pk;member {target.Hid} displayname -clear`." ) ;
await ctx . Reply ( embed : eb . Build ( ) ) ;
}
else
{
CheckEditMemberPermission ( ctx , target ) ;
var newDisplayName = ctx . RemainderOrNull ( ) ;
target . DisplayName = newDisplayName ;
await _data . SaveMember ( target ) ;
await PrintSuccess ( $"{Emojis.Success} Member display name changed. This member will now be proxied using the name \" { newDisplayName . SanitizeMentions ( ) } \ "." ) ;
}
}
public async Task ServerName ( Context ctx , PKMember target )
{
2020-02-01 12:03:02 +00:00
ctx . CheckGuildContext ( ) ;
2020-03-04 17:13:36 +00:00
if ( MatchClear ( ctx ) )
{
CheckEditMemberPermission ( ctx , target ) ;
2020-06-13 14:03:57 +00:00
await _db . Execute ( c = >
c . ExecuteAsync ( "update member_guild set display_name = null where member = @member and guild = @guild" ,
new { member = target . Id , guild = ctx . Guild . Id } ) ) ;
2020-02-01 12:03:02 +00:00
2020-03-04 17:13:36 +00:00
if ( target . DisplayName ! = null )
await ctx . Reply ( $"{Emojis.Success} Member server name cleared. This member will now be proxied using their global display name \" { target . DisplayName . SanitizeMentions ( ) } \ " in this server ({ctx.Guild.Name.SanitizeMentions()})." ) ;
else
await ctx . Reply ( $"{Emojis.Success} Member server name cleared. This member will now be proxied using their member name \" { target . Name . SanitizeMentions ( ) } \ " in this server ({ctx.Guild.Name.SanitizeMentions()})." ) ;
}
else if ( ! ctx . HasNext ( ) )
{
// No perms check, server name isn't covered by member privacy
var eb = await CreateMemberNameInfoEmbed ( ctx , target ) ;
if ( ctx . System ? . Id = = target . System )
eb . WithDescription ( $"To change server name, type `pk;member {target.Hid} servername <server name>`.\nTo clear it, type `pk;member {target.Hid} servername -clear`." ) ;
await ctx . Reply ( embed : eb . Build ( ) ) ;
}
2020-02-01 12:03:02 +00:00
else
2020-03-04 17:13:36 +00:00
{
CheckEditMemberPermission ( ctx , target ) ;
var newServerName = ctx . RemainderOrNull ( ) ;
2020-06-13 14:03:57 +00:00
await _db . Execute ( c = >
c . ExecuteAsync ( "update member_guild set display_name = @newServerName where member = @member and guild = @guild" ,
new { member = target . Id , guild = ctx . Guild . Id , newServerName } ) ) ;
2020-02-01 12:03:02 +00:00
2020-03-04 17:13:36 +00:00
await ctx . Reply ( $"{Emojis.Success} Member server name changed. This member will now be proxied using the name \" { newServerName . SanitizeMentions ( ) } \ " in this server ({ctx.Guild.Name.SanitizeMentions()})." ) ;
}
2020-02-01 12:03:02 +00:00
}
public async Task KeepProxy ( Context ctx , PKMember target )
{
if ( ctx . System = = null ) throw Errors . NoSystemError ;
if ( target . System ! = ctx . System . Id ) throw Errors . NotOwnMemberError ;
bool newValue ;
if ( ctx . Match ( "on" , "enabled" , "true" , "yes" ) ) newValue = true ;
else if ( ctx . Match ( "off" , "disabled" , "false" , "no" ) ) newValue = false ;
else if ( ctx . HasNext ( ) ) throw new PKSyntaxError ( "You must pass either \"on\" or \"off\"." ) ;
2020-03-04 17:13:36 +00:00
else
{
if ( target . KeepProxy )
await ctx . Reply ( "This member has keepproxy **enabled**, which means proxy tags will be **included** in the resulting message when proxying." ) ;
else
await ctx . Reply ( "This member has keepproxy **disabled**, which means proxy tags will **not** be included in the resulting message when proxying." ) ;
return ;
} ;
2020-02-01 12:03:02 +00:00
target . KeepProxy = newValue ;
await _data . SaveMember ( target ) ;
if ( newValue )
await ctx . Reply ( $"{Emojis.Success} Member proxy tags will now be included in the resulting message when proxying." ) ;
else
await ctx . Reply ( $"{Emojis.Success} Member proxy tags will now not be included in the resulting message when proxying." ) ;
}
2020-02-07 21:20:40 +00:00
public async Task Privacy ( Context ctx , PKMember target , PrivacyLevel ? newValueFromCommand )
2020-02-01 12:03:02 +00:00
{
if ( ctx . System = = null ) throw Errors . NoSystemError ;
if ( target . System ! = ctx . System . Id ) throw Errors . NotOwnMemberError ;
2020-02-07 21:20:40 +00:00
PrivacyLevel newValue ;
if ( ctx . Match ( "private" , "hide" , "hidden" , "on" , "enable" , "yes" ) ) newValue = PrivacyLevel . Private ;
else if ( ctx . Match ( "public" , "show" , "shown" , "displayed" , "off" , "disable" , "no" ) ) newValue = PrivacyLevel . Public ;
2020-02-01 12:03:02 +00:00
else if ( ctx . HasNext ( ) ) throw new PKSyntaxError ( "You must pass either \"private\" or \"public\"." ) ;
2020-03-04 17:13:36 +00:00
// If we're getting a value from command (eg. "pk;m <name> private" == always private, "pk;m <name> public == always public"), use that instead of parsing
else if ( newValueFromCommand ! = null ) newValue = newValueFromCommand . Value ;
else
{
if ( target . MemberPrivacy = = PrivacyLevel . Public )
2020-03-09 10:00:28 +00:00
await ctx . Reply ( "This member's privacy is currently set to **public**. This member will show up in member lists and will return all information when queried by other accounts." ) ;
2020-03-04 17:13:36 +00:00
else
2020-03-09 10:00:28 +00:00
await ctx . Reply ( "This member's privacy is currently set to **private**. This member will not show up in member lists and will return limited information when queried by other accounts." ) ;
2020-03-04 17:13:36 +00:00
return ;
}
2020-02-01 12:03:02 +00:00
2020-02-07 21:20:40 +00:00
target . MemberPrivacy = newValue ;
2020-02-01 12:03:02 +00:00
await _data . SaveMember ( target ) ;
2020-02-07 21:20:40 +00:00
if ( newValue = = PrivacyLevel . Private )
2020-02-01 12:03:02 +00:00
await ctx . Reply ( $"{Emojis.Success} Member privacy set to **private**. This member will no longer show up in member lists and will return limited information when queried by other accounts." ) ;
else
await ctx . Reply ( $"{Emojis.Success} Member privacy set to **public**. This member will now show up in member lists and will return all information when queried by other accounts." ) ;
}
public async Task Delete ( Context ctx , PKMember target )
{
if ( ctx . System = = null ) throw Errors . NoSystemError ;
if ( target . System ! = ctx . System . Id ) throw Errors . NotOwnMemberError ;
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!***__" ) ;
if ( ! await ctx . ConfirmWithReply ( target . Hid ) ) throw Errors . MemberDeleteCancelled ;
await _data . DeleteMember ( target ) ;
await ctx . Reply ( $"{Emojis.Success} Member deleted." ) ;
}
}
}