2024-10-29 22:10:37 +13:00
using System ;
2022-08-21 19:34:11 +12:00
using System.Collections.Generic ;
2024-12-26 18:36:20 +13:00
using System.ComponentModel ;
2024-12-26 19:50:47 +13:00
using System.IO ;
2022-08-21 19:34:11 +12:00
using System.Linq ;
2024-12-26 19:50:47 +13:00
using System.Net.Http ;
using System.Text ;
2022-08-21 19:34:11 +12:00
using System.Threading.Tasks ;
2024-12-26 18:36:20 +13:00
using DSharpPlus.Commands ;
using DSharpPlus.Commands.ContextChecks ;
using DSharpPlus.Commands.Processors.SlashCommands ;
2022-08-21 19:34:11 +12:00
using DSharpPlus.Entities ;
2024-12-26 18:57:41 +13:00
using DSharpPlus.Exceptions ;
2022-08-21 19:34:11 +12:00
using DSharpPlus.Interactivity ;
using DSharpPlus.Interactivity.Extensions ;
2024-12-26 19:50:47 +13:00
using Newtonsoft.Json ;
2022-08-21 19:34:11 +12:00
namespace SupportChild.Commands ;
2024-12-26 18:36:20 +13:00
[Command("admin")]
[Description("Administrative commands.")]
public class AdminCommands
2022-08-21 19:34:11 +12:00
{
2024-12-26 18:36:20 +13:00
[RequireGuild]
[Command("listinvalid")]
[Description("List tickets which channels have been deleted. Use /admin unsetticket <id> to remove them.")]
public async Task ListInvalid ( SlashCommandContext command )
2024-10-29 22:10:37 +13:00
{
if ( ! Database . TryGetOpenTickets ( out List < Database . Ticket > openTickets ) )
{
2024-12-26 18:36:20 +13:00
await command . RespondAsync ( new DiscordEmbedBuilder
2024-10-29 22:10:37 +13:00
{
Color = DiscordColor . Red ,
Description = "Could not get any open tickets from database."
} , true ) ;
}
// Get all channels in all guilds the bot is part of
List < DiscordChannel > allChannels = new List < DiscordChannel > ( ) ;
2024-12-26 18:36:20 +13:00
foreach ( KeyValuePair < ulong , DiscordGuild > guild in SupportChild . client . Guilds )
2024-10-29 22:10:37 +13:00
{
try
{
allChannels . AddRange ( await guild . Value . GetChannelsAsync ( ) ) ;
}
catch ( Exception ) { /*ignored*/ }
}
// Check which tickets channels no longer exist
List < string > listItems = new List < string > ( ) ;
foreach ( Database . Ticket ticket in openTickets )
{
if ( allChannels . All ( channel = > channel . Id ! = ticket . channelID ) )
{
listItems . Add ( "ID: **" + ticket . id . ToString ( "00000" ) + ":** <#" + ticket . channelID + ">\n" ) ;
}
}
if ( listItems . Count = = 0 )
{
2024-12-26 18:36:20 +13:00
await command . RespondAsync ( new DiscordEmbedBuilder
2024-10-29 22:10:37 +13:00
{
Color = DiscordColor . Green ,
Description = "All tickets are valid!"
} , true ) ;
return ;
}
List < DiscordEmbedBuilder > embeds = new List < DiscordEmbedBuilder > ( ) ;
foreach ( string message in Utilities . ParseListIntoMessages ( listItems ) )
{
embeds . Add ( new DiscordEmbedBuilder
{
Title = "Invalid tickets:" ,
Color = DiscordColor . Red ,
Description = message
} ) ;
}
// Add the footers
for ( int i = 0 ; i < embeds . Count ; i + + )
{
embeds [ i ] . Footer = new DiscordEmbedBuilder . EmbedFooter
{
Text = $"Page {i + 1} / {embeds.Count}"
} ;
}
List < Page > listPages = new List < Page > ( ) ;
foreach ( DiscordEmbedBuilder embed in embeds )
{
listPages . Add ( new Page ( "" , embed ) ) ;
}
await command . Interaction . SendPaginatedResponseAsync ( true , command . User , listPages ) ;
}
2024-12-26 18:36:20 +13:00
[RequireGuild]
[Command("setticket")]
[Description("Turns a channel into a ticket. WARNING: Anyone will be able to delete the channel using /close.")]
public async Task SetTicket ( SlashCommandContext command ,
[Parameter("user")] [ Description ( "(Optional) The owner of the ticket." ) ] DiscordUser user = null )
2024-10-29 22:10:37 +13:00
{
// Check if ticket exists in the database
if ( Database . IsOpenTicket ( command . Channel . Id ) )
{
2024-12-26 18:36:20 +13:00
await command . RespondAsync ( new DiscordEmbedBuilder
2024-10-29 22:10:37 +13:00
{
Color = DiscordColor . Red ,
Description = "This channel is already a ticket."
} , true ) ;
return ;
}
DiscordUser ticketUser = ( user = = null ? command . User : user ) ;
long id = Database . NewTicket ( ticketUser . Id , 0 , command . Channel . Id ) ;
2024-12-26 18:36:20 +13:00
await command . RespondAsync ( new DiscordEmbedBuilder
2024-10-29 22:10:37 +13:00
{
Color = DiscordColor . Green ,
2024-12-27 01:21:37 +13:00
Description = "Channel has been designated ticket " + id . ToString ( "00000" ) + "."
2024-10-29 22:10:37 +13:00
} ) ;
2024-12-26 18:57:41 +13:00
try
2024-10-29 22:10:37 +13:00
{
2024-12-26 18:57:41 +13:00
// Log it if the log channel exists
DiscordChannel logChannel = await SupportChild . client . GetChannelAsync ( Config . logChannel ) ;
2024-10-29 22:10:37 +13:00
await logChannel . SendMessageAsync ( new DiscordEmbedBuilder
{
Color = DiscordColor . Green ,
2024-12-27 01:21:37 +13:00
Description = command . Channel . Mention + " has been designated ticket " + id . ToString ( "00000" ) + " by " + command . Member ? . Mention + "." ,
Footer = new DiscordEmbedBuilder . EmbedFooter
{
Text = "Ticket: " + id . ToString ( "00000" )
}
2024-10-29 22:10:37 +13:00
} ) ;
}
2024-12-26 18:57:41 +13:00
catch ( NotFoundException )
{
Logger . Error ( "Could not find the log channel." ) ;
}
2024-10-29 22:10:37 +13:00
}
2024-12-26 18:36:20 +13:00
[RequireGuild]
[Command("unsetticket")]
[Description("Deletes a ticket from the ticket system without deleting the channel.")]
public async Task UnsetTicket ( SlashCommandContext command ,
[Parameter("ticket-id")] [ Description ( "(Optional) Ticket to unset. Uses the channel you are in by default." ) ] long ticketID = 0 )
2024-10-29 22:10:37 +13:00
{
Database . Ticket ticket ;
if ( ticketID = = 0 )
{
// Check if ticket exists in the database
if ( ! Database . TryGetOpenTicket ( command . Channel . Id , out ticket ) )
{
2024-12-26 18:36:20 +13:00
await command . RespondAsync ( new DiscordEmbedBuilder
2024-10-29 22:10:37 +13:00
{
Color = DiscordColor . Red ,
Description = "This channel is not a ticket!"
} , true ) ;
return ;
}
}
else
{
// Check if ticket exists in the database
if ( ! Database . TryGetOpenTicketByID ( ( uint ) ticketID , out ticket ) )
{
2024-12-26 18:36:20 +13:00
await command . RespondAsync ( new DiscordEmbedBuilder
2024-10-29 22:10:37 +13:00
{
Color = DiscordColor . Red ,
Description = "There is no ticket with this ticket ID."
} , true ) ;
return ;
}
}
if ( Database . DeleteOpenTicket ( ticket . id ) )
{
2024-12-26 18:36:20 +13:00
await command . RespondAsync ( new DiscordEmbedBuilder
2024-10-29 22:10:37 +13:00
{
Color = DiscordColor . Green ,
Description = "Channel has been undesignated as a ticket."
} ) ;
2024-12-26 18:57:41 +13:00
try
2024-10-29 22:10:37 +13:00
{
2024-12-26 18:57:41 +13:00
// Log it if the log channel exists
DiscordChannel logChannel = await SupportChild . client . GetChannelAsync ( Config . logChannel ) ;
2024-10-29 22:10:37 +13:00
await logChannel . SendMessageAsync ( new DiscordEmbedBuilder
{
Color = DiscordColor . Green ,
2024-12-27 01:21:37 +13:00
Description = command . Channel . Mention + " has been undesignated as a ticket by " + command . User . Mention + "." ,
Footer = new DiscordEmbedBuilder . EmbedFooter
{
Text = "Ticket: " + ticket . id . ToString ( "00000" )
}
2024-10-29 22:10:37 +13:00
} ) ;
}
2024-12-26 18:57:41 +13:00
catch ( NotFoundException )
{
Logger . Error ( "Could not find the log channel." ) ;
}
2024-10-29 22:10:37 +13:00
}
else
{
2024-12-26 18:36:20 +13:00
await command . RespondAsync ( new DiscordEmbedBuilder
2024-10-29 22:10:37 +13:00
{
Color = DiscordColor . Red ,
Description = "Error: Failed removing ticket from database."
} , true ) ;
}
}
2024-12-26 20:47:07 +13:00
[RequireGuild]
2024-12-26 18:36:20 +13:00
[Command("reload")]
[Description("Reloads the bot config.")]
public async Task Reload ( SlashCommandContext command )
2024-10-29 22:10:37 +13:00
{
2024-12-26 18:36:20 +13:00
await command . RespondAsync ( new DiscordEmbedBuilder
2024-10-29 22:10:37 +13:00
{
Color = DiscordColor . Green ,
Description = "Reloading bot application..."
} ) ;
2024-12-27 01:21:37 +13:00
try
{
DiscordChannel logChannel = await SupportChild . client . GetChannelAsync ( Config . logChannel ) ;
await logChannel . SendMessageAsync ( new DiscordEmbedBuilder
{
Color = DiscordColor . Green ,
Description = command . Channel . Mention + " reloaded the bot." ,
} ) ;
}
catch ( NotFoundException )
{
Logger . Error ( "Could not find the log channel." ) ;
}
2024-10-29 22:10:37 +13:00
Logger . Log ( "Reloading bot..." ) ;
2024-12-26 20:12:50 +13:00
await SupportChild . Reload ( ) ;
2024-10-29 22:10:37 +13:00
}
2024-12-26 19:50:47 +13:00
2024-12-26 20:47:07 +13:00
[RequireGuild]
2024-12-26 19:50:47 +13:00
[Command("getinterviewtemplates")]
[Description("Provides a copy of the interview templates which you can edit and then reupload.")]
public async Task GetInterviewTemplates ( SlashCommandContext command )
{
2024-12-26 20:12:50 +13:00
MemoryStream stream = new ( Encoding . UTF8 . GetBytes ( Database . GetInterviewTemplatesJSON ( ) ) ) ;
2024-12-26 20:22:05 +13:00
await command . RespondAsync ( new DiscordInteractionResponseBuilder ( ) . AddFile ( "interview-templates.json" , stream ) . AsEphemeral ( ) ) ;
2024-12-26 19:50:47 +13:00
}
2024-12-26 20:47:07 +13:00
[RequireGuild]
2024-12-26 19:50:47 +13:00
[Command("setinterviewtemplates")]
[Description("Uploads an interview template file.")]
public async Task SetInterviewTemplates ( SlashCommandContext command , [ Parameter ( "file" ) ] DiscordAttachment file )
{
if ( ! file . MediaType ? . Contains ( "application/json" ) ? ? false )
{
await command . RespondAsync ( new DiscordEmbedBuilder
{
Color = DiscordColor . Red ,
Description = "The uploaded file is not a JSON file according to Discord."
2024-12-26 20:22:05 +13:00
} , true ) ;
2024-12-26 19:50:47 +13:00
return ;
}
Stream stream = await new HttpClient ( ) . GetStreamAsync ( file . Url ) ;
string json = await new StreamReader ( stream ) . ReadToEndAsync ( ) ;
try
{
List < string > errors = [ ] ;
// Convert it to an interview object to validate the template
Dictionary < ulong , Interviewer . ValidatedInterviewQuestion > interview = JsonConvert . DeserializeObject < Dictionary < ulong , Interviewer . ValidatedInterviewQuestion > > ( json , new JsonSerializerSettings ( )
{
//NullValueHandling = NullValueHandling.Include,
MissingMemberHandling = MissingMemberHandling . Error ,
2024-12-27 00:38:35 +13:00
Error = delegate ( object sender , Newtonsoft . Json . Serialization . ErrorEventArgs args )
2024-12-26 19:50:47 +13:00
{
// I noticed the main exception mainly has information for developers, not administrators,
// so I switched to using the inner message if available.
if ( string . IsNullOrEmpty ( args . ErrorContext . Error . InnerException ? . Message ) )
{
errors . Add ( args . ErrorContext . Error . Message ) ;
}
else
{
errors . Add ( args . ErrorContext . Error . InnerException . Message ) ;
}
Logger . Debug ( "Exception occured when trying to upload interview template:\n" + args . ErrorContext . Error ) ;
args . ErrorContext . Handled = true ;
}
} ) ;
2024-12-26 23:23:25 +13:00
if ( interview ! = null )
{
foreach ( KeyValuePair < ulong , Interviewer . ValidatedInterviewQuestion > interviewRoot in interview )
{
interviewRoot . Value . Validate ( ref errors , out int summaryCount , out int summaryMaxLength ) ;
2024-12-27 00:47:23 +13:00
2024-12-26 23:23:25 +13:00
if ( summaryCount > 25 )
{
errors . Add ( "A summary cannot contain more than 25 fields, but you have " + summaryCount + " fields in one of your interview branches." ) ;
}
2024-12-27 00:47:23 +13:00
2024-12-26 23:23:25 +13:00
if ( summaryMaxLength > = 6000 )
{
errors . Add ( "A summary cannot contain more than 6000 characters, but one of your branches has the possibility of its summary reaching " + summaryMaxLength + " characters." ) ;
}
}
}
2024-12-26 19:50:47 +13:00
if ( errors . Count ! = 0 )
{
string errorString = string . Join ( "\n\n" , errors ) ;
if ( errorString . Length > 1500 )
{
errorString = errorString . Substring ( 0 , 1500 ) ;
}
await command . RespondAsync ( new DiscordEmbedBuilder
{
Color = DiscordColor . Red ,
Description = "The uploaded JSON structure could not be converted to an interview template.\n\nErrors:\n```\n" + errorString + "\n```" ,
Footer = new DiscordEmbedBuilder . EmbedFooter ( )
{
Text = "More detailed information may be available as debug messages in the bot logs."
}
2024-12-26 20:22:05 +13:00
} , true ) ;
2024-12-26 19:50:47 +13:00
return ;
}
Database . SetInterviewTemplates ( JsonConvert . SerializeObject ( interview , Formatting . Indented ) ) ;
}
catch ( Exception e )
{
await command . RespondAsync ( new DiscordEmbedBuilder
{
Color = DiscordColor . Red ,
Description = "The uploaded JSON structure could not be converted to an interview template.\n\nError message:\n```\n" + e . Message + "\n```"
2024-12-26 20:22:05 +13:00
} , true ) ;
2024-12-26 19:50:47 +13:00
return ;
}
await command . RespondAsync ( new DiscordEmbedBuilder
{
Color = DiscordColor . Green ,
Description = "Uploaded interview template."
2024-12-26 20:22:05 +13:00
} , true ) ;
2024-12-27 01:21:37 +13:00
try
{
DiscordChannel logChannel = await SupportChild . client . GetChannelAsync ( Config . logChannel ) ;
await logChannel . SendMessageAsync ( new DiscordEmbedBuilder
{
Color = DiscordColor . Green ,
Description = command . Channel . Mention + " uploaded new interview templates." ,
} ) ;
}
catch ( NotFoundException )
{
Logger . Error ( "Could not find the log channel." ) ;
}
2024-12-26 19:50:47 +13:00
}
2022-08-21 19:34:11 +12:00
}