2020-07-18 11:53:02 +00:00
using System ;
using System.Collections.Generic ;
2020-07-06 17:50:39 +00:00
using System.Linq ;
using System.Text ;
using System.Threading.Tasks ;
2020-06-29 21:51:12 +00:00
2020-07-18 11:30:54 +00:00
using Dapper ;
2020-06-29 21:51:12 +00:00
using DSharpPlus.Entities ;
2020-08-16 10:10:54 +00:00
using Humanizer ;
2020-06-29 21:51:12 +00:00
using PluralKit.Core ;
namespace PluralKit.Bot
{
public class Groups
{
private readonly IDatabase _db ;
2020-08-29 11:46:27 +00:00
private readonly ModelRepository _repo ;
2020-06-29 21:51:12 +00:00
2020-08-29 11:46:27 +00:00
public Groups ( IDatabase db , ModelRepository repo )
2020-06-29 21:51:12 +00:00
{
_db = db ;
2020-08-29 11:46:27 +00:00
_repo = repo ;
2020-06-29 21:51:12 +00:00
}
public async Task CreateGroup ( Context ctx )
{
ctx . CheckSystem ( ) ;
2020-08-16 10:10:54 +00:00
// Check group name length
2020-06-29 21:51:12 +00:00
var groupName = ctx . RemainderOrNull ( ) ? ? throw new PKSyntaxError ( "You must pass a group name." ) ;
if ( groupName . Length > Limits . MaxGroupNameLength )
2020-08-08 12:56:34 +00:00
throw new PKError ( $"Group name too long ({groupName.Length}/{Limits.MaxGroupNameLength} characters)." ) ;
2020-06-29 21:51:12 +00:00
await using var conn = await _db . Obtain ( ) ;
2020-08-16 10:10:54 +00:00
// Check group cap
var existingGroupCount = await conn . QuerySingleAsync < int > ( "select count(*) from groups where system = @System" , new { System = ctx . System . Id } ) ;
2020-07-18 11:30:54 +00:00
if ( existingGroupCount > = Limits . MaxGroupCount )
throw new PKError ( $"System has reached the maximum number of groups ({Limits.MaxGroupCount}). Please delete unused groups first in order to create new ones." ) ;
2020-08-16 10:10:54 +00:00
// Warn if there's already a group by this name
2020-08-29 11:46:27 +00:00
var existingGroup = await _repo . GetGroupByName ( conn , ctx . System . Id , groupName ) ;
2020-08-16 10:10:54 +00:00
if ( existingGroup ! = null ) {
var msg = $"{Emojis.Warn} You already have a group in your system with the name \" { existingGroup . Name } \ " (with ID `{existingGroup.Hid}`). Do you want to create another group with the same name?" ;
if ( ! await ctx . PromptYesNo ( msg ) )
throw new PKError ( "Group creation cancelled." ) ;
}
2020-07-18 11:30:54 +00:00
2020-08-29 11:46:27 +00:00
var newGroup = await _repo . CreateGroup ( conn , ctx . System . Id , groupName ) ;
2020-07-18 14:49:00 +00:00
var eb = new DiscordEmbedBuilder ( )
. W ithDescription ( $"Your new group, **{groupName}**, has been created, with the group ID **`{newGroup.Hid}`**.\nBelow are a couple of useful commands:" )
2020-08-25 20:33:04 +00:00
. AddField ( "View the group card" , $"> pk;group **{newGroup.Reference()}**" )
. AddField ( "Add members to the group" , $"> pk;group **{newGroup.Reference()}** add **MemberName**\n> pk;group **{newGroup.Reference()}** add **Member1** **Member2** **Member3** (and so on...)" )
. AddField ( "Set the description" , $"> pk;group **{newGroup.Reference()}** description **This is my new group, and here is the description!**" )
. AddField ( "Set the group icon" , $"> pk;group **{newGroup.Reference()}** icon\n*(with an image attached)*" ) ;
2020-07-18 14:49:00 +00:00
await ctx . Reply ( $"{Emojis.Success} Group created!" , eb . Build ( ) ) ;
2020-06-29 21:51:12 +00:00
}
2020-07-06 17:50:39 +00:00
public async Task RenameGroup ( Context ctx , PKGroup target )
{
ctx . CheckOwnGroup ( target ) ;
2020-08-16 10:10:54 +00:00
// Check group name length
2020-07-06 17:50:39 +00:00
var newName = ctx . RemainderOrNull ( ) ? ? throw new PKSyntaxError ( "You must pass a new group name." ) ;
if ( newName . Length > Limits . MaxGroupNameLength )
throw new PKError ( $"New group name too long ({newName.Length}/{Limits.MaxMemberNameLength} characters)." ) ;
2020-08-16 10:10:54 +00:00
2020-07-06 17:50:39 +00:00
await using var conn = await _db . Obtain ( ) ;
2020-08-16 10:10:54 +00:00
// Warn if there's already a group by this name
2020-08-29 11:46:27 +00:00
var existingGroup = await _repo . GetGroupByName ( conn , ctx . System . Id , newName ) ;
2020-08-16 10:10:54 +00:00
if ( existingGroup ! = null & & existingGroup . Id ! = target . Id ) {
var msg = $"{Emojis.Warn} You already have a group in your system with the name \" { existingGroup . Name } \ " (with ID `{existingGroup.Hid}`). Do you want to rename this member to that name too?" ;
if ( ! await ctx . PromptYesNo ( msg ) )
throw new PKError ( "Group creation cancelled." ) ;
}
2020-08-29 11:46:27 +00:00
await _repo . UpdateGroup ( conn , target . Id , new GroupPatch { Name = newName } ) ;
2020-07-06 17:50:39 +00:00
2020-08-20 19:43:17 +00:00
await ctx . Reply ( $"{Emojis.Success} Group name changed from **{target.Name}** to **{newName}**." ) ;
}
public async Task GroupDisplayName ( Context ctx , PKGroup target )
{
if ( ctx . MatchClear ( ) )
{
ctx . CheckOwnGroup ( target ) ;
var patch = new GroupPatch { DisplayName = Partial < string > . Null ( ) } ;
2020-08-29 11:46:27 +00:00
await _db . Execute ( conn = > _repo . UpdateGroup ( conn , target . Id , patch ) ) ;
2020-08-20 19:43:17 +00:00
await ctx . Reply ( $"{Emojis.Success} Group display name cleared." ) ;
}
else if ( ! ctx . HasNext ( ) )
{
// No perms check, display name isn't covered by member privacy
var eb = new DiscordEmbedBuilder ( )
. AddField ( "Name" , target . Name )
. AddField ( "Display Name" , target . DisplayName ? ? "*(none)*" ) ;
if ( ctx . System ? . Id = = target . System )
2020-08-25 20:33:04 +00:00
eb . WithDescription ( $"To change display name, type `pk;group {target.Reference()} displayname <display name>`.\nTo clear it, type `pk;group {target.Reference()} displayname -clear`." ) ;
2020-08-20 19:43:17 +00:00
await ctx . Reply ( embed : eb . Build ( ) ) ;
}
else
{
ctx . CheckOwnGroup ( target ) ;
var newDisplayName = ctx . RemainderOrNull ( ) ;
var patch = new GroupPatch { DisplayName = Partial < string > . Present ( newDisplayName ) } ;
2020-08-29 11:46:27 +00:00
await _db . Execute ( conn = > _repo . UpdateGroup ( conn , target . Id , patch ) ) ;
2020-08-20 19:43:17 +00:00
await ctx . Reply ( $"{Emojis.Success} Group display name changed." ) ;
}
2020-07-06 17:50:39 +00:00
}
public async Task GroupDescription ( Context ctx , PKGroup target )
{
if ( ctx . MatchClear ( ) )
{
ctx . CheckOwnGroup ( target ) ;
var patch = new GroupPatch { Description = Partial < string > . Null ( ) } ;
2020-08-29 11:46:27 +00:00
await _db . Execute ( conn = > _repo . UpdateGroup ( conn , target . Id , patch ) ) ;
2020-07-06 17:50:39 +00:00
await ctx . Reply ( $"{Emojis.Success} Group description cleared." ) ;
}
else if ( ! ctx . HasNext ( ) )
{
if ( target . Description = = null )
if ( ctx . System ? . Id = = target . System )
2020-08-25 20:33:04 +00:00
await ctx . Reply ( $"This group does not have a description set. To set one, type `pk;group {target.Reference()} description <description>`." ) ;
2020-07-06 17:50:39 +00:00
else
await ctx . Reply ( "This group does not have a description set." ) ;
else if ( ctx . MatchFlag ( "r" , "raw" ) )
await ctx . Reply ( $"```\n{target.Description}\n```" ) ;
else
await ctx . Reply ( embed : new DiscordEmbedBuilder ( )
. WithTitle ( "Group description" )
. WithDescription ( target . Description )
2020-08-25 20:33:04 +00:00
. AddField ( "\u200B" , $"To print the description with formatting, type `pk;group {target.Reference()} description -raw`."
+ ( ctx . System ? . Id = = target . System ? $" To clear it, type `pk;group {target.Reference()} description -clear`." : "" ) )
2020-07-06 17:50:39 +00:00
. Build ( ) ) ;
}
else
{
ctx . CheckOwnGroup ( target ) ;
var description = ctx . RemainderOrNull ( ) . NormalizeLineEndSpacing ( ) ;
if ( description . IsLongerThan ( Limits . MaxDescriptionLength ) )
throw Errors . DescriptionTooLongError ( description . Length ) ;
var patch = new GroupPatch { Description = Partial < string > . Present ( description ) } ;
2020-08-29 11:46:27 +00:00
await _db . Execute ( conn = > _repo . UpdateGroup ( conn , target . Id , patch ) ) ;
2020-07-06 17:50:39 +00:00
await ctx . Reply ( $"{Emojis.Success} Group description changed." ) ;
}
}
2020-08-08 13:09:42 +00:00
public async Task GroupIcon ( Context ctx , PKGroup target )
{
async Task ClearIcon ( )
{
ctx . CheckOwnGroup ( target ) ;
2020-08-29 11:46:27 +00:00
await _db . Execute ( c = > _repo . UpdateGroup ( c , target . Id , new GroupPatch { Icon = null } ) ) ;
2020-08-08 13:09:42 +00:00
await ctx . Reply ( $"{Emojis.Success} Group icon cleared." ) ;
}
async Task SetIcon ( ParsedImage img )
{
ctx . CheckOwnGroup ( target ) ;
if ( img . Url . Length > Limits . MaxUriLength )
throw Errors . InvalidUrl ( img . Url ) ;
await AvatarUtils . VerifyAvatarOrThrow ( img . Url ) ;
2020-08-29 11:46:27 +00:00
await _db . Execute ( c = > _repo . UpdateGroup ( c , target . Id , new GroupPatch { Icon = img . Url } ) ) ;
2020-08-08 13:09:42 +00:00
var msg = img . Source switch
{
AvatarSource . User = > $"{Emojis.Success} Group icon changed to {img.SourceUser?.Username}'s avatar!\n{Emojis.Warn} If {img.SourceUser?.Username} changes their avatar, the group icon will need to be re-set." ,
AvatarSource . Url = > $"{Emojis.Success} Group icon changed to the image at the given URL." ,
AvatarSource . Attachment = > $"{Emojis.Success} Group icon changed to attached image.\n{Emojis.Warn} If you delete the message containing the attachment, the group icon will stop working." ,
_ = > throw new ArgumentOutOfRangeException ( )
} ;
// The attachment's already right there, no need to preview it.
var hasEmbed = img . Source ! = AvatarSource . Attachment ;
await ( hasEmbed
? ctx . Reply ( msg , embed : new DiscordEmbedBuilder ( ) . WithImageUrl ( img . Url ) . Build ( ) )
: ctx . Reply ( msg ) ) ;
}
async Task ShowIcon ( )
{
if ( ( target . Icon ? . Trim ( ) ? ? "" ) . Length > 0 )
{
var eb = new DiscordEmbedBuilder ( )
. WithTitle ( "Group icon" )
. WithImageUrl ( target . Icon ) ;
if ( target . System = = ctx . System ? . Id )
{
2020-08-25 20:33:04 +00:00
eb . WithDescription ( $"To clear, use `pk;group {target.Reference()} icon -clear`." ) ;
2020-08-08 13:09:42 +00:00
}
await ctx . Reply ( embed : eb . Build ( ) ) ;
}
else
throw new PKSyntaxError ( "This group does not have an icon set. Set one by attaching an image to this command, or by passing an image URL or @mention." ) ;
}
if ( ctx . MatchClear ( ) )
await ClearIcon ( ) ;
else if ( await ctx . MatchImage ( ) is { } img )
await SetIcon ( img ) ;
else
await ShowIcon ( ) ;
}
2020-07-06 17:50:39 +00:00
public async Task ListSystemGroups ( Context ctx , PKSystem system )
{
if ( system = = null )
{
ctx . CheckSystem ( ) ;
system = ctx . System ;
}
2020-08-20 19:43:17 +00:00
ctx . CheckSystemPrivacy ( system , system . GroupListPrivacy ) ;
2020-08-16 10:10:54 +00:00
2020-07-06 17:50:39 +00:00
// TODO: integrate with the normal "search" system
await using var conn = await _db . Obtain ( ) ;
2020-07-18 11:53:02 +00:00
var pctx = LookupContext . ByNonOwner ;
2020-08-16 10:10:54 +00:00
if ( ctx . MatchFlag ( "a" , "all" ) )
{
if ( system . Id = = ctx . System . Id )
pctx = LookupContext . ByOwner ;
else
throw new PKError ( "You do not have permission to access this information." ) ;
}
2020-08-20 19:43:17 +00:00
2020-08-21 15:08:49 +00:00
var groups = ( await conn . QueryGroupList ( system . Id ) )
2020-08-16 10:10:54 +00:00
. Where ( g = > g . Visibility . CanAccess ( pctx ) )
2020-08-25 16:43:52 +00:00
. OrderBy ( g = > g . Name , StringComparer . InvariantCultureIgnoreCase )
2020-08-16 10:10:54 +00:00
. ToList ( ) ;
2020-07-06 17:50:39 +00:00
if ( groups . Count = = 0 )
{
if ( system . Id = = ctx . System ? . Id )
2020-08-20 19:43:17 +00:00
await ctx . Reply ( "This system has no groups. To create one, use the command `pk;group new <name>`." ) ;
2020-07-06 17:50:39 +00:00
else
2020-08-20 19:43:17 +00:00
await ctx . Reply ( "This system has no groups." ) ;
2020-07-06 17:50:39 +00:00
return ;
}
var title = system . Name ! = null ? $"Groups of {system.Name} (`{system.Hid}`)" : $"Groups of `{system.Hid}`" ;
await ctx . Paginate ( groups . ToAsyncEnumerable ( ) , groups . Count , 25 , title , Renderer ) ;
2020-08-21 15:08:49 +00:00
Task Renderer ( DiscordEmbedBuilder eb , IEnumerable < ListedGroup > page )
2020-07-06 17:50:39 +00:00
{
2020-08-20 19:43:17 +00:00
eb . WithSimpleLineContent ( page . Select ( g = >
{
if ( g . DisplayName ! = null )
2020-08-21 15:08:49 +00:00
return $"[`{g.Hid}`] **{g.Name}** ({g.DisplayName}) ({" member ".ToQuantity(g.MemberCount)})" ;
2020-08-20 19:43:17 +00:00
else
2020-08-21 15:08:49 +00:00
return $"[`{g.Hid}`] **{g.Name}** ({" member ".ToQuantity(g.MemberCount)})" ;
2020-08-20 19:43:17 +00:00
} ) ) ;
2020-08-16 10:10:54 +00:00
eb . WithFooter ( $"{groups.Count} total." ) ;
2020-07-06 17:50:39 +00:00
return Task . CompletedTask ;
}
}
2020-06-29 21:51:12 +00:00
public async Task ShowGroupCard ( Context ctx , PKGroup target )
{
await using var conn = await _db . Obtain ( ) ;
var system = await GetGroupSystem ( ctx , target , conn ) ;
2020-07-18 11:53:02 +00:00
var pctx = ctx . LookupContextFor ( system ) ;
2020-09-12 22:10:37 +00:00
var memberCount = ctx . MatchPrivateFlag ( pctx ) ? await _repo . GetGroupMemberCount ( conn , target . Id , PrivacyLevel . Public ) : await _repo . GetGroupMemberCount ( conn , target . Id ) ;
2020-06-29 21:51:12 +00:00
var nameField = target . Name ;
if ( system . Name ! = null )
nameField = $"{nameField} ({system.Name})" ;
var eb = new DiscordEmbedBuilder ( )
2020-07-18 11:53:02 +00:00
. WithAuthor ( nameField , iconUrl : DiscordUtils . WorkaroundForUrlBug ( target . IconFor ( pctx ) ) )
2020-06-29 21:51:12 +00:00
. WithFooter ( $"System ID: {system.Hid} | Group ID: {target.Hid} | Created on {target.Created.FormatZoned(system)}" ) ;
2020-08-20 19:43:17 +00:00
if ( target . DisplayName ! = null )
eb . AddField ( "Display Name" , target . DisplayName ) ;
if ( target . ListPrivacy . CanAccess ( pctx ) )
{
if ( memberCount = = 0 & & pctx = = LookupContext . ByOwner )
// Only suggest the add command if this is actually the owner lol
2020-08-25 20:33:04 +00:00
eb . AddField ( "Members (0)" , $"Add one with `pk;group {target.Reference()} add <member>`!" , true ) ;
2020-08-20 19:43:17 +00:00
else
2020-08-25 20:33:04 +00:00
eb . AddField ( $"Members ({memberCount})" , $"(see `pk;group {target.Reference()} list`)" , true ) ;
2020-08-20 19:43:17 +00:00
}
2020-07-18 11:19:53 +00:00
2020-07-18 11:53:02 +00:00
if ( target . DescriptionFor ( pctx ) is { } desc )
eb . AddField ( "Description" , desc ) ;
if ( target . IconFor ( pctx ) is { } icon )
eb . WithThumbnail ( icon ) ;
2020-07-07 22:56:06 +00:00
2020-06-29 21:51:12 +00:00
await ctx . Reply ( embed : eb . Build ( ) ) ;
}
2020-07-07 13:28:53 +00:00
public async Task AddRemoveMembers ( Context ctx , PKGroup target , AddRemoveOperation op )
{
ctx . CheckOwnGroup ( target ) ;
2020-07-07 17:34:44 +00:00
var members = await ParseMemberList ( ctx ) ;
2020-07-07 13:28:53 +00:00
await using var conn = await _db . Obtain ( ) ;
2020-08-16 10:10:54 +00:00
var existingMembersInGroup = ( await conn . QueryMemberList ( target . System ,
new DatabaseViewsExt . MemberListQueryOptions { GroupFilter = target . Id } ) )
2020-08-20 19:43:17 +00:00
. Select ( m = > m . Id . Value )
2020-08-16 10:10:54 +00:00
. ToHashSet ( ) ;
2020-07-07 13:28:53 +00:00
if ( op = = AddRemoveOperation . Add )
{
2020-08-16 10:10:54 +00:00
var membersNotInGroup = members
2020-08-20 19:43:17 +00:00
. Where ( m = > ! existingMembersInGroup . Contains ( m . Id . Value ) )
2020-08-16 10:10:54 +00:00
. Select ( m = > m . Id )
2020-08-20 19:43:17 +00:00
. Distinct ( )
2020-08-16 10:10:54 +00:00
. ToList ( ) ;
2020-08-29 11:46:27 +00:00
await _repo . AddMembersToGroup ( conn , target . Id , membersNotInGroup ) ;
2020-08-16 10:10:54 +00:00
if ( membersNotInGroup . Count = = members . Count )
await ctx . Reply ( $"{Emojis.Success} {" members ".ToQuantity(membersNotInGroup.Count)} added to group." ) ;
else
await ctx . Reply ( $"{Emojis.Success} {" members ".ToQuantity(membersNotInGroup.Count)} added to group ({" members ".ToQuantity(members.Count - membersNotInGroup.Count)} already in group)." ) ;
2020-07-07 13:28:53 +00:00
}
else if ( op = = AddRemoveOperation . Remove )
{
2020-08-16 10:10:54 +00:00
var membersInGroup = members
2020-08-20 19:43:17 +00:00
. Where ( m = > existingMembersInGroup . Contains ( m . Id . Value ) )
2020-08-16 10:10:54 +00:00
. Select ( m = > m . Id )
2020-08-20 19:43:17 +00:00
. Distinct ( )
2020-08-16 10:10:54 +00:00
. ToList ( ) ;
2020-08-29 11:46:27 +00:00
await _repo . RemoveMembersFromGroup ( conn , target . Id , membersInGroup ) ;
2020-08-16 10:10:54 +00:00
if ( membersInGroup . Count = = members . Count )
await ctx . Reply ( $"{Emojis.Success} {" members ".ToQuantity(membersInGroup.Count)} removed from group." ) ;
else
await ctx . Reply ( $"{Emojis.Success} {" members ".ToQuantity(membersInGroup.Count)} removed from group ({" members ".ToQuantity(members.Count - membersInGroup.Count)} already not in group)." ) ;
2020-07-07 13:28:53 +00:00
}
}
2020-07-07 17:34:23 +00:00
public async Task ListGroupMembers ( Context ctx , PKGroup target )
{
await using var conn = await _db . Obtain ( ) ;
2020-08-16 10:10:54 +00:00
2020-07-07 17:34:23 +00:00
var targetSystem = await GetGroupSystem ( ctx , target , conn ) ;
2020-08-20 19:43:17 +00:00
ctx . CheckSystemPrivacy ( targetSystem , target . ListPrivacy ) ;
2020-07-07 17:34:23 +00:00
var opts = ctx . ParseMemberListOptions ( ctx . LookupContextFor ( target . System ) ) ;
opts . GroupFilter = target . Id ;
2020-08-20 19:43:17 +00:00
var title = new StringBuilder ( $"Members of {target.DisplayName ?? target.Name} (`{target.Hid}`) in " ) ;
2020-07-07 17:34:23 +00:00
if ( targetSystem . Name ! = null )
title . Append ( $"{targetSystem.Name} (`{targetSystem.Hid}`)" ) ;
else
title . Append ( $"`{targetSystem.Hid}`" ) ;
if ( opts . Search ! = null )
title . Append ( $" matching **{opts.Search}**" ) ;
await ctx . RenderMemberList ( ctx . LookupContextFor ( target . System ) , _db , target . System , title . ToString ( ) , opts ) ;
}
2020-07-07 13:28:53 +00:00
public enum AddRemoveOperation
{
Add ,
Remove
}
2020-07-07 17:34:44 +00:00
private static async Task < List < PKMember > > ParseMemberList ( Context ctx )
{
// TODO: move this to a context extension and share with the switch command somewhere, after branch merge?
var members = new List < PKMember > ( ) ;
while ( ctx . HasNext ( ) )
{
var member = await ctx . MatchMember ( ) ;
if ( member = = null )
throw new PKSyntaxError ( ctx . CreateMemberNotFoundError ( ctx . PopArgument ( ) ) ) ; ;
if ( member . System ! = ctx . System . Id )
throw new PKError ( $"Member **{member.Name}** (`{member.Hid}`) is not in your own system, so you can't add it to a group." ) ;
members . Add ( member ) ;
}
if ( members . Count = = 0 )
throw new PKSyntaxError ( "You must pass one or more members." ) ;
return members ;
}
2020-07-18 11:53:02 +00:00
public async Task GroupPrivacy ( Context ctx , PKGroup target , PrivacyLevel ? newValueFromCommand )
{
ctx . CheckSystem ( ) . CheckOwnGroup ( target ) ;
// Display privacy settings
if ( ! ctx . HasNext ( ) & & newValueFromCommand = = null )
{
await ctx . Reply ( embed : new DiscordEmbedBuilder ( )
. WithTitle ( $"Current privacy settings for {target.Name}" )
. AddField ( "Description" , target . DescriptionPrivacy . Explanation ( ) )
. AddField ( "Icon" , target . IconPrivacy . Explanation ( ) )
2020-08-20 19:43:17 +00:00
. AddField ( "Member list" , target . ListPrivacy . Explanation ( ) )
2020-07-18 11:53:02 +00:00
. AddField ( "Visibility" , target . Visibility . Explanation ( ) )
2020-08-25 20:33:04 +00:00
. WithDescription ( $"To edit privacy settings, use the command:\n> pk;group **{target.Reference()}** privacy **<subject>** **<level>**\n\n- `subject` is one of `description`, `icon`, `members`, `visibility`, or `all`\n- `level` is either `public` or `private`." )
2020-07-18 11:53:02 +00:00
. Build ( ) ) ;
return ;
}
async Task SetAll ( PrivacyLevel level )
{
2020-08-29 11:46:27 +00:00
await _db . Execute ( c = > _repo . UpdateGroup ( c , target . Id , new GroupPatch ( ) . WithAllPrivacy ( level ) ) ) ;
2020-07-18 11:53:02 +00:00
if ( level = = PrivacyLevel . Private )
2020-08-08 12:56:34 +00:00
await ctx . Reply ( $"{Emojis.Success} All {target.Name}'s privacy settings have been set to **{level.LevelName()}**. Other accounts will now see nothing on the group card." ) ;
2020-07-18 11:53:02 +00:00
else
2020-08-08 12:56:34 +00:00
await ctx . Reply ( $"{Emojis.Success} All {target.Name}'s privacy settings have been set to **{level.LevelName()}**. Other accounts will now see everything on the group card." ) ;
2020-07-18 11:53:02 +00:00
}
async Task SetLevel ( GroupPrivacySubject subject , PrivacyLevel level )
{
2020-08-29 11:46:27 +00:00
await _db . Execute ( c = > _repo . UpdateGroup ( c , target . Id , new GroupPatch ( ) . WithPrivacy ( subject , level ) ) ) ;
2020-07-18 11:53:02 +00:00
var subjectName = subject switch
{
GroupPrivacySubject . Description = > "description privacy" ,
GroupPrivacySubject . Icon = > "icon privacy" ,
2020-08-20 19:43:17 +00:00
GroupPrivacySubject . List = > "member list" ,
2020-07-18 11:53:02 +00:00
GroupPrivacySubject . Visibility = > "visibility" ,
_ = > throw new ArgumentOutOfRangeException ( $"Unknown privacy subject {subject}" )
} ;
var explanation = ( subject , level ) switch
{
( GroupPrivacySubject . Description , PrivacyLevel . Private ) = > "This group's description is now hidden from other systems." ,
( GroupPrivacySubject . Icon , PrivacyLevel . Private ) = > "This group's icon is now hidden from other systems." ,
( GroupPrivacySubject . Visibility , PrivacyLevel . Private ) = > "This group is now hidden from group lists and member cards." ,
2020-08-20 19:43:17 +00:00
( GroupPrivacySubject . List , PrivacyLevel . Private ) = > "This group's member list is now hidden from other systems." ,
2020-07-18 11:53:02 +00:00
( GroupPrivacySubject . Description , PrivacyLevel . Public ) = > "This group's description is no longer hidden from other systems." ,
( GroupPrivacySubject . Icon , PrivacyLevel . Public ) = > "This group's icon is no longer hidden from other systems." ,
( GroupPrivacySubject . Visibility , PrivacyLevel . Public ) = > "This group is no longer hidden from group lists and member cards." ,
2020-08-20 19:43:17 +00:00
( GroupPrivacySubject . List , PrivacyLevel . Public ) = > "This group's member list is no longer hidden from other systems." ,
2020-07-18 11:53:02 +00:00
_ = > throw new InvalidOperationException ( $"Invalid subject/level tuple ({subject}, {level})" )
} ;
await ctx . Reply ( $"{Emojis.Success} {target.Name}'s **{subjectName}** has been set to **{level.LevelName()}**. {explanation}" ) ;
}
if ( ctx . Match ( "all" ) | | newValueFromCommand ! = null )
await SetAll ( newValueFromCommand ? ? ctx . PopPrivacyLevel ( ) ) ;
else
await SetLevel ( ctx . PopGroupPrivacySubject ( ) , ctx . PopPrivacyLevel ( ) ) ;
}
2020-07-07 17:34:44 +00:00
2020-08-08 12:56:34 +00:00
public async Task DeleteGroup ( Context ctx , PKGroup target )
{
ctx . CheckOwnGroup ( target ) ;
await ctx . Reply ( $"{Emojis.Warn} Are you sure you want to delete this group? If so, reply to this message with the group's ID (`{target.Hid}`).\n**Note: this action is permanent.**" ) ;
if ( ! await ctx . ConfirmWithReply ( target . Hid ) )
throw new PKError ( $"Group deletion cancelled. Note that you must reply with your group ID (`{target.Hid}`) *verbatim*." ) ;
2020-08-29 11:46:27 +00:00
await _db . Execute ( conn = > _repo . DeleteGroup ( conn , target . Id ) ) ;
2020-08-08 12:56:34 +00:00
await ctx . Reply ( $"{Emojis.Success} Group deleted." ) ;
}
2020-08-29 11:46:27 +00:00
private async Task < PKSystem > GetGroupSystem ( Context ctx , PKGroup target , IPKConnection conn )
2020-06-29 21:51:12 +00:00
{
var system = ctx . System ;
if ( system ? . Id = = target . System )
return system ;
2020-08-29 11:46:27 +00:00
return await _repo . GetSystem ( conn , target . System ) ! ;
2020-06-29 21:51:12 +00:00
}
}
}