From affe8955e1d00146fd3c9dd9c3e253c18b82651a Mon Sep 17 00:00:00 2001 From: EmotionChild Date: Sun, 21 Aug 2022 19:21:53 +1200 Subject: [PATCH 1/5] Updated transcript lib --- SupportChild/lib/DiscordChatExporter.Core.dll | Bin 661504 -> 720384 bytes 1 file changed, 0 insertions(+), 0 deletions(-) diff --git a/SupportChild/lib/DiscordChatExporter.Core.dll b/SupportChild/lib/DiscordChatExporter.Core.dll index e27417ef6936970022096342365badcc1fe8b666..938808d4758a8fa288f78f54a836b676b42077b8 100644 GIT binary patch literal 720384 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

)1Hf5pu_X~&^o2&$Cwy!>u>G1Dw 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≧D)!sZ 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 literal 661504 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�-$PI0U-q)R7> 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 V{__pD3#&}GH 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 Date: Sun, 21 Aug 2022 19:32:26 +1200 Subject: [PATCH 2/5] Updated Resources.Designer.cs --- SupportChild/Properties/Resources.Designer.cs | 18 ------------------ 1 file changed, 18 deletions(-) diff --git a/SupportChild/Properties/Resources.Designer.cs b/SupportChild/Properties/Resources.Designer.cs index 4f8fc83..8cdb657 100644 --- a/SupportChild/Properties/Resources.Designer.cs +++ b/SupportChild/Properties/Resources.Designer.cs @@ -1,7 +1,6 @@ //------------------------------------------------------------------------------ // // This code was generated by a tool. -// Runtime Version:4.0.30319.42000 // // Changes to this file may cause incorrect behavior and will be lost if // the code is regenerated. @@ -12,13 +11,6 @@ namespace SupportChild.Properties { using System; - ///

- /// A strongly-typed resource class, for looking up localized strings, etc. - /// - // This class was auto-generated by the StronglyTypedResourceBuilder - // class via a tool like ResGen or Visual Studio. - // To add or remove a member, edit your .ResX file then rerun ResGen - // with the /str option, or rebuild your VS project. [global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "17.0.0.0")] [global::System.Diagnostics.DebuggerNonUserCodeAttribute()] [global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()] @@ -32,9 +24,6 @@ namespace SupportChild.Properties { internal Resources() { } - /// - /// Returns the cached ResourceManager instance used by this class. - /// [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)] internal static global::System.Resources.ResourceManager ResourceManager { get { @@ -46,10 +35,6 @@ namespace SupportChild.Properties { } } - /// - /// Overrides the current thread's CurrentUICulture property for all - /// resource lookups using this strongly typed resource class. - /// [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)] internal static global::System.Globalization.CultureInfo Culture { get { @@ -60,9 +45,6 @@ namespace SupportChild.Properties { } } - /// - /// Looks up a localized resource of type System.Byte[]. - /// internal static byte[] default_config { get { object obj = ResourceManager.GetObject("default_config", resourceCulture); From 0e5a93a6a68205cb8fb1016481d4f78e74a428d8 Mon Sep 17 00:00:00 2001 From: EmotionChild Date: Sun, 21 Aug 2022 19:33:27 +1200 Subject: [PATCH 3/5] Updated general files --- SupportChild/Config.cs | 206 ++--- SupportChild/Database.cs | 1389 ++++++++++++++++-------------- SupportChild/EventHandler.cs | 436 +++++----- SupportChild/Logger.cs | 68 ++ SupportChild/SupportChild.cs | 279 +++--- SupportChild/SupportChild.csproj | 66 +- SupportChild/Transcriber.cs | 81 +- SupportChild/Utilities.cs | 112 ++- SupportChild/default_config.yml | 122 +-- 9 files changed, 1405 insertions(+), 1354 deletions(-) create mode 100644 SupportChild/Logger.cs diff --git a/SupportChild/Config.cs b/SupportChild/Config.cs index 858e17b..69ad566 100644 --- a/SupportChild/Config.cs +++ b/SupportChild/Config.cs @@ -1,143 +1,95 @@ using System; -using System.Collections.Generic; using System.IO; -using System.Linq; using System.Text; -using DSharpPlus.Entities; +using DSharpPlus; +using Microsoft.Extensions.Logging; using Newtonsoft.Json.Linq; using SupportChild.Properties; using YamlDotNet.Serialization; -namespace SupportChild +namespace SupportChild; + +internal static class Config { - internal static class Config + internal static string token = ""; + internal static ulong logChannel; + internal static string welcomeMessage = ""; + internal static LogLevel logLevel = LogLevel.Information; + internal static TimestampFormat timestampFormat = TimestampFormat.RelativeTime; + internal static bool randomAssignment = false; + internal static bool randomAssignRoleOverride = false; + internal static string presenceType = "Playing"; + internal static string presenceText = ""; + internal static bool newCommandUsesSelector = false; + internal static int ticketLimit = 5; + + internal static bool ticketUpdatedNotifications = false; + internal static double ticketUpdatedNotificationDelay = 0.0; + internal static bool assignmentNotifications = false; + internal static bool closingNotifications = false; + + internal static string hostName = "127.0.0.1"; + internal static int port = 3306; + internal static string database = "supportchild"; + internal static string username = "supportchild"; + internal static string password = ""; + + public static void LoadConfig() { - internal static string token = ""; - internal static string prefix = ""; - internal static ulong logChannel; - internal static ulong ticketCategory; - internal static ulong reactionMessage; - internal static string welcomeMessage = ""; - internal static string logLevel = "Information"; - internal static string timestampFormat = "yyyy-MMM-dd HH:mm"; - internal static bool randomAssignment = false; - internal static bool randomAssignRoleOverride = false; - internal static string presenceType = "Playing"; - internal static string presenceText = ""; - - internal static bool ticketUpdatedNotifications = false; - internal static double ticketUpdatedNotificationDelay = 0.0; - internal static bool assignmentNotifications = false; - internal static bool closingNotifications = false; - - internal static string hostName = "127.0.0.1"; - internal static int port = 3306; - internal static string database = "supportbot"; - internal static string username = "supportbot"; - internal static string password = ""; - - private static readonly Dictionary permissions = new Dictionary + // Writes default config to file if it does not already exist + if (!File.Exists("./config.yml")) { - // Public commands - { "close", new ulong[]{ } }, - { "list", new ulong[]{ } }, - { "new", new ulong[]{ } }, - { "say", new ulong[]{ } }, - { "status", new ulong[]{ } }, - { "summary", new ulong[]{ } }, - { "transcript", new ulong[]{ } }, - // Moderator commands - { "add", new ulong[]{ } }, - { "addmessage", new ulong[]{ } }, - { "assign", new ulong[]{ } }, - { "blacklist", new ulong[]{ } }, - { "listassigned", new ulong[]{ } }, - { "listoldest", new ulong[]{ } }, - { "listunassigned", new ulong[]{ } }, - { "move", new ulong[]{ } }, - { "rassign", new ulong[]{ } }, - { "removemessage", new ulong[]{ } }, - { "setsummary", new ulong[]{ } }, - { "toggleactive", new ulong[]{ } }, - { "unassign", new ulong[]{ } }, - { "unblacklist", new ulong[]{ } }, - // Admin commands - { "addstaff", new ulong[]{ } }, - { "reload", new ulong[]{ } }, - { "removestaff", new ulong[]{ } }, - { "setticket", new ulong[]{ } }, - { "unsetticket", new ulong[]{ } }, - }; - - public static void LoadConfig() - { - // Writes default config to file if it does not already exist - if (!File.Exists("./config.yml")) - { - File.WriteAllText("./config.yml", Encoding.UTF8.GetString(Resources.default_config)); - } - - // Reads config contents into FileStream - FileStream stream = File.OpenRead("./config.yml"); - - // Converts the FileStream into a YAML object - IDeserializer deserializer = new DeserializerBuilder().Build(); - object yamlObject = deserializer.Deserialize(new StreamReader(stream)); - - // Converts the YAML object into a JSON object as the YAML ones do not support traversal or selection of nodes by name - ISerializer serializer = new SerializerBuilder().JsonCompatible().Build(); - JObject json = JObject.Parse(serializer.Serialize(yamlObject)); - - // Sets up the bot - token = json.SelectToken("bot.token").Value() ?? ""; - prefix = json.SelectToken("bot.prefix").Value() ?? ""; - logChannel = json.SelectToken("bot.log-channel").Value(); - ticketCategory = json.SelectToken("bot.ticket-category")?.Value() ?? 0; - reactionMessage = json.SelectToken("bot.reaction-message")?.Value() ?? 0; - welcomeMessage = json.SelectToken("bot.welcome-message").Value() ?? ""; - logLevel = json.SelectToken("bot.console-log-level").Value() ?? ""; - timestampFormat = json.SelectToken("bot.timestamp-format").Value() ?? "yyyy-MM-dd HH:mm"; - randomAssignment = json.SelectToken("bot.random-assignment")?.Value() ?? false; - randomAssignRoleOverride = json.SelectToken("bot.random-assign-role-override")?.Value() ?? false; - presenceType = json.SelectToken("bot.presence-type")?.Value() ?? "Playing"; - presenceText = json.SelectToken("bot.presence-text")?.Value() ?? ""; - - ticketUpdatedNotifications = json.SelectToken("notifications.ticket-updated")?.Value() ?? false; - ticketUpdatedNotificationDelay = json.SelectToken("notifications.ticket-updated-delay")?.Value() ?? 0.0; - assignmentNotifications = json.SelectToken("notifications.assignment")?.Value() ?? false; - closingNotifications = json.SelectToken("notifications.closing")?.Value() ?? false; - - // Reads database info - hostName = json.SelectToken("database.address")?.Value() ?? ""; - port = json.SelectToken("database.port")?.Value() ?? 3306; - database = json.SelectToken("database.name")?.Value() ?? "supportchild"; - username = json.SelectToken("database.user")?.Value() ?? "supportchild"; - password = json.SelectToken("database.password")?.Value() ?? ""; - - timestampFormat = timestampFormat.Trim(); - - foreach (KeyValuePair node in permissions.ToList()) - { - try - { - permissions[node.Key] = json.SelectToken("permissions." + node.Key).Value().Values().ToArray(); - } - catch (ArgumentNullException) - { - Console.WriteLine("Permission node '" + node.Key + "' was not found in the config, using default value: []"); - } - } + File.WriteAllText("./config.yml", Encoding.UTF8.GetString(Resources.default_config)); } - /// - /// Checks whether a user has a specific permission. - /// - /// The Discord user to check. - /// The permission name to check. - /// - public static bool HasPermission(DiscordMember member, string permission) + // Reads config contents into FileStream + FileStream stream = File.OpenRead("./config.yml"); + + // Converts the FileStream into a YAML object + IDeserializer deserializer = new DeserializerBuilder().Build(); + object yamlObject = deserializer.Deserialize(new StreamReader(stream)) ?? ""; + + // Converts the YAML object into a JSON object as the YAML ones do not support traversal or selection of nodes by name + ISerializer serializer = new SerializerBuilder().JsonCompatible().Build(); + JObject json = JObject.Parse(serializer.Serialize(yamlObject)); + + // Sets up the bot + token = json.SelectToken("bot.token")?.Value() ?? ""; + logChannel = json.SelectToken("bot.log-channel")?.Value() ?? 0; + welcomeMessage = json.SelectToken("bot.welcome-message")?.Value() ?? ""; + string stringLogLevel = json.SelectToken("bot.console-log-level")?.Value() ?? ""; + + if (!Enum.TryParse(stringLogLevel, true, out logLevel)) { - return member.Roles.Any(role => permissions[permission].Contains(role.Id)) || permissions[permission].Contains(member.Guild.Id); + logLevel = LogLevel.Information; + Logger.Warn("Log level '" + stringLogLevel + "' invalid, using 'Information' instead."); } + + string stringTimestampFormat = json.SelectToken("bot.timestamp-format")?.Value() ?? "RelativeTime"; + + if (!Enum.TryParse(stringTimestampFormat, true, out timestampFormat)) + { + timestampFormat = TimestampFormat.RelativeTime; + Logger.Warn("Timestamp '" + stringTimestampFormat + "' invalid, using 'RelativeTime' instead."); + } + + randomAssignment = json.SelectToken("bot.random-assignment")?.Value() ?? false; + randomAssignRoleOverride = json.SelectToken("bot.random-assign-role-override")?.Value() ?? false; + presenceType = json.SelectToken("bot.presence-type")?.Value() ?? "Playing"; + presenceText = json.SelectToken("bot.presence-text")?.Value() ?? ""; + newCommandUsesSelector = json.SelectToken("bot.new-command-uses-selector")?.Value() ?? false; + ticketLimit = json.SelectToken("bot.ticket-limit")?.Value() ?? 5; + + ticketUpdatedNotifications = json.SelectToken("notifications.ticket-updated")?.Value() ?? false; + ticketUpdatedNotificationDelay = json.SelectToken("notifications.ticket-updated-delay")?.Value() ?? 0.0; + assignmentNotifications = json.SelectToken("notifications.assignment")?.Value() ?? false; + closingNotifications = json.SelectToken("notifications.closing")?.Value() ?? false; + + // Reads database info + hostName = json.SelectToken("database.address")?.Value() ?? ""; + port = json.SelectToken("database.port")?.Value() ?? 3306; + database = json.SelectToken("database.name")?.Value() ?? "supportchild"; + username = json.SelectToken("database.user")?.Value() ?? "supportchild"; + password = json.SelectToken("database.password")?.Value() ?? ""; } } \ No newline at end of file diff --git a/SupportChild/Database.cs b/SupportChild/Database.cs index 57ecbca..e78160c 100644 --- a/SupportChild/Database.cs +++ b/SupportChild/Database.cs @@ -1,691 +1,818 @@ using System; using System.Linq; using System.Collections.Generic; +using DSharpPlus; using MySql.Data.MySqlClient; -namespace SupportChild +namespace SupportChild; + +public static class Database { - public static class Database + private static string connectionString = ""; + + private static readonly Random random = new Random(); + + public static void SetConnectionString(string host, int port, string database, string username, string password) { - private static string connectionString = ""; + connectionString = "server=" + host + + ";database=" + database + + ";port=" + port + + ";userid=" + username + + ";password=" + password; + } - private static Random random = new Random(); + public static MySqlConnection GetConnection() + { + return new MySqlConnection(connectionString); + } - public static void SetConnectionString(string host, int port, string database, string username, string password) + public static long GetNumberOfTickets() + { + try { - connectionString = "server=" + host + - ";database=" + database + - ";port=" + port + - ";userid=" + username + - ";password=" + password; + using MySqlConnection c = GetConnection(); + using MySqlCommand countTickets = new MySqlCommand("SELECT COUNT(*) FROM tickets", c); + c.Open(); + return (long)countTickets.ExecuteScalar(); } - public static MySqlConnection GetConnection() + catch (Exception e) { - return new MySqlConnection(connectionString); - } - public static long GetNumberOfTickets() - { - try - { - using (MySqlConnection c = GetConnection()) - { - MySqlCommand countTickets = new MySqlCommand("SELECT COUNT(*) FROM tickets", c); - c.Open(); - return (long)countTickets.ExecuteScalar(); - } - } - catch (Exception e) - { - Console.WriteLine("Error occured when attempting to count number of open tickets: " + e); - } - - return -1; - } - public static long GetNumberOfClosedTickets() - { - try - { - using (MySqlConnection c = GetConnection()) - { - MySqlCommand countTickets = new MySqlCommand("SELECT COUNT(*) FROM ticket_history", c); - c.Open(); - return (long)countTickets.ExecuteScalar(); - } - } - catch (Exception e) - { - Console.WriteLine("Error occured when attempting to count number of open tickets: " + e); - } - - return -1; - } - public static void SetupTables() - { - using (MySqlConnection c = GetConnection()) - { - MySqlCommand createTickets = new MySqlCommand( - "CREATE TABLE IF NOT EXISTS tickets(" + - "id INT UNSIGNED NOT NULL PRIMARY KEY AUTO_INCREMENT," + - "created_time DATETIME NOT NULL," + - "creator_id BIGINT UNSIGNED NOT NULL," + - "assigned_staff_id BIGINT UNSIGNED NOT NULL DEFAULT 0," + - "summary VARCHAR(5000) NOT NULL," + - "channel_id BIGINT UNSIGNED NOT NULL UNIQUE," + - "INDEX(created_time, assigned_staff_id, channel_id))", - c); - MySqlCommand createTicketHistory = new MySqlCommand( - "CREATE TABLE IF NOT EXISTS ticket_history(" + - "id INT UNSIGNED NOT NULL PRIMARY KEY," + - "created_time DATETIME NOT NULL," + - "closed_time DATETIME NOT NULL," + - "creator_id BIGINT UNSIGNED NOT NULL," + - "assigned_staff_id BIGINT UNSIGNED NOT NULL DEFAULT 0," + - "summary VARCHAR(5000) NOT NULL," + - "channel_id BIGINT UNSIGNED NOT NULL UNIQUE," + - "INDEX(created_time, closed_time, channel_id))", - c); - MySqlCommand createBlacklisted = new MySqlCommand( - "CREATE TABLE IF NOT EXISTS blacklisted_users(" + - "user_id BIGINT UNSIGNED NOT NULL PRIMARY KEY," + - "time DATETIME NOT NULL," + - "moderator_id BIGINT UNSIGNED NOT NULL," + - "INDEX(user_id, time))", - c); - MySqlCommand createStaffList = new MySqlCommand( - "CREATE TABLE IF NOT EXISTS staff(" + - "user_id BIGINT UNSIGNED NOT NULL PRIMARY KEY," + - "name VARCHAR(256) NOT NULL," + - "active BOOLEAN NOT NULL DEFAULT true)", - c); - MySqlCommand createMessages = new MySqlCommand( - "CREATE TABLE IF NOT EXISTS messages(" + - "identifier VARCHAR(256) NOT NULL PRIMARY KEY," + - "user_id BIGINT UNSIGNED NOT NULL," + - "message VARCHAR(5000) NOT NULL)", - c); - c.Open(); - createTickets.ExecuteNonQuery(); - createBlacklisted.ExecuteNonQuery(); - createTicketHistory.ExecuteNonQuery(); - createStaffList.ExecuteNonQuery(); - createMessages.ExecuteNonQuery(); - } - } - public static bool IsOpenTicket(ulong channelID) - { - using (MySqlConnection c = GetConnection()) - { - c.Open(); - MySqlCommand selection = new MySqlCommand(@"SELECT * FROM tickets WHERE channel_id=@channel_id", c); - selection.Parameters.AddWithValue("@channel_id", channelID); - selection.Prepare(); - MySqlDataReader results = selection.ExecuteReader(); - - // Check if ticket exists in the database - if (!results.Read()) - { - return false; - } - results.Close(); - } - return true; - } - public static bool TryGetOpenTicket(ulong channelID, out Ticket ticket) - { - using (MySqlConnection c = GetConnection()) - { - c.Open(); - MySqlCommand selection = new MySqlCommand(@"SELECT * FROM tickets WHERE channel_id=@channel_id", c); - selection.Parameters.AddWithValue("@channel_id", channelID); - selection.Prepare(); - MySqlDataReader results = selection.ExecuteReader(); - - // Check if ticket exists in the database - if (!results.Read()) - { - ticket = null; - return false; - } - - ticket = new Ticket(results); - results.Close(); - return true; - } - } - public static bool TryGetOpenTicketByID(uint id, out Ticket ticket) - { - using (MySqlConnection c = GetConnection()) - { - c.Open(); - MySqlCommand selection = new MySqlCommand(@"SELECT * FROM tickets WHERE id=@id", c); - selection.Parameters.AddWithValue("@id", id); - selection.Prepare(); - MySqlDataReader results = selection.ExecuteReader(); - - // Check if open ticket exists in the database - if (results.Read()) - { - ticket = new Ticket(results); - return true; - } - - ticket = null; - return false; - } - } - public static bool TryGetClosedTicket(uint id, out Ticket ticket) - { - using (MySqlConnection c = GetConnection()) - { - c.Open(); - MySqlCommand selection = new MySqlCommand(@"SELECT * FROM ticket_history WHERE id=@id", c); - selection.Parameters.AddWithValue("@id", id); - selection.Prepare(); - MySqlDataReader results = selection.ExecuteReader(); - - // Check if closed ticket exists in the database - if (results.Read()) - { - ticket = new Ticket(results); - return true; - } - - ticket = null; - return false; - } - } - public static bool TryGetOpenTickets(ulong userID, out List tickets) - { - tickets = null; - using (MySqlConnection c = GetConnection()) - { - c.Open(); - MySqlCommand selection = new MySqlCommand(@"SELECT * FROM tickets WHERE creator_id=@creator_id", c); - selection.Parameters.AddWithValue("@creator_id", userID); - selection.Prepare(); - MySqlDataReader results = selection.ExecuteReader(); - - if (!results.Read()) - { - return false; - } - - tickets = new List { new Ticket(results) }; - while (results.Read()) - { - tickets.Add(new Ticket(results)); - } - results.Close(); - return true; - } - } - public static bool TryGetOldestTickets(ulong userID, out List tickets, int listLimit) - { - tickets = null; - using (MySqlConnection c = GetConnection()) - { - c.Open(); - MySqlCommand selection = new MySqlCommand(@"SELECT * FROM tickets ORDER BY created_time ASC LIMIT @limit", c); - selection.Parameters.AddWithValue("@creator_id", userID); - selection.Parameters.AddWithValue("@limit", listLimit); - selection.Prepare(); - MySqlDataReader results = selection.ExecuteReader(); - - if (!results.Read()) - { - return false; - } - - tickets = new List { new Ticket(results) }; - while (results.Read()) - { - tickets.Add(new Ticket(results)); - } - results.Close(); - return true; - } - } - public static bool TryGetClosedTickets(ulong userID, out List tickets) - { - tickets = null; - using (MySqlConnection c = GetConnection()) - { - c.Open(); - MySqlCommand selection = new MySqlCommand(@"SELECT * FROM ticket_history WHERE creator_id=@creator_id", c); - selection.Parameters.AddWithValue("@creator_id", userID); - selection.Prepare(); - MySqlDataReader results = selection.ExecuteReader(); - - if (!results.Read()) - { - return false; - } - - tickets = new List { new Ticket(results) }; - while (results.Read()) - { - tickets.Add(new Ticket(results)); - } - results.Close(); - return true; - } - } - public static bool TryGetAssignedTickets(ulong staffID, out List tickets) - { - tickets = null; - using (MySqlConnection c = GetConnection()) - { - c.Open(); - MySqlCommand selection = new MySqlCommand(@"SELECT * FROM tickets WHERE assigned_staff_id=@assigned_staff_id", c); - selection.Parameters.AddWithValue("@assigned_staff_id", staffID); - selection.Prepare(); - MySqlDataReader results = selection.ExecuteReader(); - - if (!results.Read()) - { - return false; - } - - tickets = new List { new Ticket(results) }; - while (results.Read()) - { - tickets.Add(new Ticket(results)); - } - results.Close(); - return true; - } - } - public static long NewTicket(ulong memberID, ulong staffID, ulong ticketID) - { - using (MySqlConnection c = GetConnection()) - { - c.Open(); - MySqlCommand cmd = new MySqlCommand( - @"INSERT INTO tickets (created_time, creator_id, assigned_staff_id, summary, channel_id) VALUES (UTC_TIMESTAMP(), @creator_id, @assigned_staff_id, @summary, @channel_id);", - c); - cmd.Parameters.AddWithValue("@creator_id", memberID); - cmd.Parameters.AddWithValue("@assigned_staff_id", staffID); - cmd.Parameters.AddWithValue("@summary", ""); - cmd.Parameters.AddWithValue("@channel_id", ticketID); - cmd.ExecuteNonQuery(); - return cmd.LastInsertedId; - } + Logger.Error("Error occured when attempting to count number of open tickets: " + e); } - public static void ArchiveTicket(Ticket ticket) + return -1; + } + + public static long GetNumberOfClosedTickets() + { + try { - // Check if ticket already exists in the archive - if (TryGetClosedTicket(ticket.id, out Ticket _)) - { - using (MySqlConnection c = GetConnection()) - { - MySqlCommand deleteTicket = new MySqlCommand(@"DELETE FROM ticket_history WHERE id=@id OR channel_id=@channel_id", c); - deleteTicket.Parameters.AddWithValue("@id", ticket.id); - deleteTicket.Parameters.AddWithValue("@channel_id", ticket.channelID); - - c.Open(); - deleteTicket.Prepare(); - deleteTicket.ExecuteNonQuery(); - } - } - - using (MySqlConnection c = GetConnection()) - { - // Create an entry in the ticket history database - MySqlCommand archiveTicket = new MySqlCommand(@"INSERT INTO ticket_history (id, created_time, closed_time, creator_id, assigned_staff_id, summary, channel_id) VALUES (@id, @created_time, UTC_TIMESTAMP(), @creator_id, @assigned_staff_id, @summary, @channel_id);", c); - archiveTicket.Parameters.AddWithValue("@id", ticket.id); - archiveTicket.Parameters.AddWithValue("@created_time", ticket.createdTime); - archiveTicket.Parameters.AddWithValue("@creator_id", ticket.creatorID); - archiveTicket.Parameters.AddWithValue("@assigned_staff_id", ticket.assignedStaffID); - archiveTicket.Parameters.AddWithValue("@summary", ticket.summary); - archiveTicket.Parameters.AddWithValue("@channel_id", ticket.channelID); - - c.Open(); - archiveTicket.Prepare(); - archiveTicket.ExecuteNonQuery(); - } + using MySqlConnection c = GetConnection(); + using MySqlCommand countTickets = new MySqlCommand("SELECT COUNT(*) FROM ticket_history", c); + c.Open(); + return (long)countTickets.ExecuteScalar(); + } + catch (Exception e) + { + Logger.Error("Error occured when attempting to count number of open tickets: " + e); } - public static void DeleteOpenTicket(uint ticketID) + return -1; + } + + public static void SetupTables() + { + using MySqlConnection c = GetConnection(); + using MySqlCommand createTickets = new MySqlCommand( + "CREATE TABLE IF NOT EXISTS tickets(" + + "id INT UNSIGNED NOT NULL PRIMARY KEY AUTO_INCREMENT," + + "created_time DATETIME NOT NULL," + + "creator_id BIGINT UNSIGNED NOT NULL," + + "assigned_staff_id BIGINT UNSIGNED NOT NULL DEFAULT 0," + + "summary VARCHAR(5000) NOT NULL," + + "channel_id BIGINT UNSIGNED NOT NULL UNIQUE," + + "INDEX(created_time, assigned_staff_id, channel_id))", + c); + using MySqlCommand createTicketHistory = new MySqlCommand( + "CREATE TABLE IF NOT EXISTS ticket_history(" + + "id INT UNSIGNED NOT NULL PRIMARY KEY," + + "created_time DATETIME NOT NULL," + + "closed_time DATETIME NOT NULL," + + "creator_id BIGINT UNSIGNED NOT NULL," + + "assigned_staff_id BIGINT UNSIGNED NOT NULL DEFAULT 0," + + "summary VARCHAR(5000) NOT NULL," + + "channel_id BIGINT UNSIGNED NOT NULL UNIQUE," + + "INDEX(created_time, closed_time, channel_id))", + c); + using MySqlCommand createBlacklisted = new MySqlCommand( + "CREATE TABLE IF NOT EXISTS blacklisted_users(" + + "user_id BIGINT UNSIGNED NOT NULL PRIMARY KEY," + + "time DATETIME NOT NULL," + + "moderator_id BIGINT UNSIGNED NOT NULL," + + "INDEX(user_id, time))", + c); + using MySqlCommand createStaffList = new MySqlCommand( + "CREATE TABLE IF NOT EXISTS staff(" + + "user_id BIGINT UNSIGNED NOT NULL PRIMARY KEY," + + "name VARCHAR(256) NOT NULL," + + "active BOOLEAN NOT NULL DEFAULT true)", + c); + using MySqlCommand createMessages = new MySqlCommand( + "CREATE TABLE IF NOT EXISTS messages(" + + "identifier VARCHAR(256) NOT NULL PRIMARY KEY," + + "user_id BIGINT UNSIGNED NOT NULL," + + "message VARCHAR(5000) NOT NULL)", + c); + using MySqlCommand createCategories = new MySqlCommand( + "CREATE TABLE IF NOT EXISTS categories(" + + "name VARCHAR(256) NOT NULL UNIQUE," + + "category_id BIGINT UNSIGNED NOT NULL PRIMARY KEY)", + c); + c.Open(); + createTickets.ExecuteNonQuery(); + createBlacklisted.ExecuteNonQuery(); + createTicketHistory.ExecuteNonQuery(); + createStaffList.ExecuteNonQuery(); + createMessages.ExecuteNonQuery(); + createCategories.ExecuteNonQuery(); + } + + public static bool IsOpenTicket(ulong channelID) + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand selection = new MySqlCommand(@"SELECT * FROM tickets WHERE channel_id=@channel_id", c); + selection.Parameters.AddWithValue("@channel_id", channelID); + selection.Prepare(); + MySqlDataReader results = selection.ExecuteReader(); + + // Check if ticket exists in the database + if (!results.Read()) { - using (MySqlConnection c = GetConnection()) - { - MySqlCommand deletion = new MySqlCommand(@"DELETE FROM tickets WHERE id=@id", c); - deletion.Parameters.AddWithValue("@id", ticketID); - - c.Open(); - deletion.Prepare(); - deletion.ExecuteNonQuery(); - } - } - - public static bool IsBlacklisted(ulong userID) - { - using (MySqlConnection c = GetConnection()) - { - c.Open(); - MySqlCommand selection = new MySqlCommand(@"SELECT * FROM blacklisted_users WHERE user_id=@user_id", c); - selection.Parameters.AddWithValue("@user_id", userID); - selection.Prepare(); - MySqlDataReader results = selection.ExecuteReader(); - - // Check if user is blacklisted - if (results.Read()) - { - return true; - } - results.Close(); - } - return false; } - public static bool Blacklist(ulong blacklistedID, ulong staffID) + results.Close(); + return true; + } + + public static bool TryGetOpenTicket(ulong channelID, out Ticket ticket) + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand selection = new MySqlCommand(@"SELECT * FROM tickets WHERE channel_id=@channel_id", c); + selection.Parameters.AddWithValue("@channel_id", channelID); + selection.Prepare(); + MySqlDataReader results = selection.ExecuteReader(); + + // Check if ticket exists in the database + if (!results.Read()) { - using (MySqlConnection c = GetConnection()) + ticket = null; + return false; + } + + ticket = new Ticket(results); + results.Close(); + return true; + } + + public static bool TryGetOpenTicketByID(uint id, out Ticket ticket) + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand selection = new MySqlCommand(@"SELECT * FROM tickets WHERE id=@id", c); + selection.Parameters.AddWithValue("@id", id); + selection.Prepare(); + MySqlDataReader results = selection.ExecuteReader(); + + // Check if open ticket exists in the database + if (results.Read()) + { + ticket = new Ticket(results); + results.Close(); + return true; + } + + results.Close(); + ticket = null; + return false; + } + + public static bool TryGetClosedTicket(uint id, out Ticket ticket) + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand selection = new MySqlCommand(@"SELECT * FROM ticket_history WHERE id=@id", c); + selection.Parameters.AddWithValue("@id", id); + selection.Prepare(); + MySqlDataReader results = selection.ExecuteReader(); + + // Check if closed ticket exists in the database + if (results.Read()) + { + ticket = new Ticket(results); + results.Close(); + return true; + } + + ticket = null; + results.Close(); + return false; + } + + public static bool TryGetOpenTickets(ulong userID, out List tickets) + { + tickets = null; + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand selection = new MySqlCommand(@"SELECT * FROM tickets WHERE creator_id=@creator_id", c); + selection.Parameters.AddWithValue("@creator_id", userID); + selection.Prepare(); + MySqlDataReader results = selection.ExecuteReader(); + + if (!results.Read()) + { + return false; + } + + tickets = new List { new Ticket(results) }; + while (results.Read()) + { + tickets.Add(new Ticket(results)); + } + results.Close(); + return true; + } + + public static bool TryGetOpenTickets(out List tickets) + { + tickets = null; + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand selection = new MySqlCommand(@"SELECT * FROM tickets ORDER BY channel_id ASC", c); + selection.Prepare(); + MySqlDataReader results = selection.ExecuteReader(); + + if (!results.Read()) + { + return false; + } + + tickets = new List { new Ticket(results) }; + while (results.Read()) + { + tickets.Add(new Ticket(results)); + } + results.Close(); + return true; + } + + public static bool TryGetClosedTickets(ulong userID, out List tickets) + { + tickets = null; + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand selection = new MySqlCommand(@"SELECT * FROM ticket_history WHERE creator_id=@creator_id", c); + selection.Parameters.AddWithValue("@creator_id", userID); + selection.Prepare(); + MySqlDataReader results = selection.ExecuteReader(); + + if (!results.Read()) + { + return false; + } + + tickets = new List { new Ticket(results) }; + while (results.Read()) + { + tickets.Add(new Ticket(results)); + } + results.Close(); + return true; + } + + public static bool TryGetAssignedTickets(ulong staffID, out List tickets) + { + tickets = null; + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand selection = new MySqlCommand(@"SELECT * FROM tickets WHERE assigned_staff_id=@assigned_staff_id", c); + selection.Parameters.AddWithValue("@assigned_staff_id", staffID); + selection.Prepare(); + MySqlDataReader results = selection.ExecuteReader(); + + if (!results.Read()) + { + return false; + } + + tickets = new List { new Ticket(results) }; + while (results.Read()) + { + tickets.Add(new Ticket(results)); + } + results.Close(); + return true; + } + + public static long NewTicket(ulong memberID, ulong staffID, ulong ticketID) + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand cmd = new MySqlCommand(@"INSERT INTO tickets (created_time, creator_id, assigned_staff_id, summary, channel_id) VALUES (UTC_TIMESTAMP(), @creator_id, @assigned_staff_id, @summary, @channel_id);", c); + cmd.Parameters.AddWithValue("@creator_id", memberID); + cmd.Parameters.AddWithValue("@assigned_staff_id", staffID); + cmd.Parameters.AddWithValue("@summary", ""); + cmd.Parameters.AddWithValue("@channel_id", ticketID); + cmd.ExecuteNonQuery(); + return cmd.LastInsertedId; + } + + public static void ArchiveTicket(Ticket ticket) + { + // Check if ticket already exists in the archive + if (TryGetClosedTicket(ticket.id, out Ticket _)) + { + using MySqlConnection c = GetConnection(); + using MySqlCommand deleteTicket = new MySqlCommand(@"DELETE FROM ticket_history WHERE id=@id OR channel_id=@channel_id", c); + deleteTicket.Parameters.AddWithValue("@id", ticket.id); + deleteTicket.Parameters.AddWithValue("@channel_id", ticket.channelID); + + c.Open(); + deleteTicket.Prepare(); + deleteTicket.ExecuteNonQuery(); + } + + // Create an entry in the ticket history database + using MySqlConnection conn = GetConnection(); + using MySqlCommand archiveTicket = new MySqlCommand(@"INSERT INTO ticket_history (id, created_time, closed_time, creator_id, assigned_staff_id, summary, channel_id) VALUES (@id, @created_time, UTC_TIMESTAMP(), @creator_id, @assigned_staff_id, @summary, @channel_id);", conn); + archiveTicket.Parameters.AddWithValue("@id", ticket.id); + archiveTicket.Parameters.AddWithValue("@created_time", ticket.channelID.GetSnowflakeTime()); + archiveTicket.Parameters.AddWithValue("@creator_id", ticket.creatorID); + archiveTicket.Parameters.AddWithValue("@assigned_staff_id", ticket.assignedStaffID); + archiveTicket.Parameters.AddWithValue("@summary", ticket.summary); + archiveTicket.Parameters.AddWithValue("@channel_id", ticket.channelID); + + conn.Open(); + archiveTicket.Prepare(); + archiveTicket.ExecuteNonQuery(); + } + + public static bool DeleteOpenTicket(uint ticketID) + { + try + { + using MySqlConnection c = GetConnection(); + using MySqlCommand deletion = new MySqlCommand(@"DELETE FROM tickets WHERE id=@id", c); + deletion.Parameters.AddWithValue("@id", ticketID); + + c.Open(); + deletion.Prepare(); + return deletion.ExecuteNonQuery() > 0; + } + catch (MySqlException) + { + return false; + } + } + + public static bool IsBlacklisted(ulong userID) + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand selection = new MySqlCommand(@"SELECT * FROM blacklisted_users WHERE user_id=@user_id", c); + selection.Parameters.AddWithValue("@user_id", userID); + selection.Prepare(); + MySqlDataReader results = selection.ExecuteReader(); + + // Check if user is blacklisted + if (results.Read()) + { + return true; + } + results.Close(); + + return false; + } + + public static bool Blacklist(ulong blacklistedID, ulong staffID) + { + try + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand cmd = new MySqlCommand(@"INSERT INTO blacklisted_users (user_id,time,moderator_id) VALUES (@user_id, UTC_TIMESTAMP(), @moderator_id);", c); + cmd.Parameters.AddWithValue("@user_id", blacklistedID); + cmd.Parameters.AddWithValue("@moderator_id", staffID); + cmd.Prepare(); + return cmd.ExecuteNonQuery() > 0; + } + catch (MySqlException) + { + return false; + } + } + + public static bool Unblacklist(ulong blacklistedID) + { + try + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand cmd = new MySqlCommand(@"DELETE FROM blacklisted_users WHERE user_id=@user_id", c); + cmd.Parameters.AddWithValue("@user_id", blacklistedID); + cmd.Prepare(); + return cmd.ExecuteNonQuery() > 0; + } + catch (MySqlException) + { + return false; + } + } + + public static bool AssignStaff(Ticket ticket, ulong staffID) + { + try + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand update = new MySqlCommand(@"UPDATE tickets SET assigned_staff_id = @assigned_staff_id WHERE id = @id", c); + update.Parameters.AddWithValue("@assigned_staff_id", staffID); + update.Parameters.AddWithValue("@id", ticket.id); + update.Prepare(); + return update.ExecuteNonQuery() > 0; + } + catch (MySqlException) + { + return false; + } + } + + public static bool UnassignStaff(Ticket ticket) + { + return AssignStaff(ticket, 0); + } + + public static bool SetStaffActive(ulong staffID, bool active) + { + try + { + using MySqlConnection c = GetConnection(); + c.Open(); + MySqlCommand update = new MySqlCommand(@"UPDATE staff SET active = @active WHERE user_id = @user_id", c); + update.Parameters.AddWithValue("@user_id", staffID); + update.Parameters.AddWithValue("@active", active); + update.Prepare(); + return update.ExecuteNonQuery() > 0; + } + catch (MySqlException) + { + return false; + } + } + + public static StaffMember GetRandomActiveStaff(params ulong[] ignoredUserIDs) + { + List staffMembers = GetActiveStaff(ignoredUserIDs); + return staffMembers.Any() ? staffMembers[random.Next(staffMembers.Count)] : null; + } + + public static List GetActiveStaff(params ulong[] ignoredUserIDs) + { + bool first = true; + string filterString = ""; + foreach (ulong userID in ignoredUserIDs) + { + if (first) { - try - { - c.Open(); - MySqlCommand cmd = new MySqlCommand(@"INSERT INTO blacklisted_users (user_id,time,moderator_id) VALUES (@user_id, UTC_TIMESTAMP(), @moderator_id);", c); - cmd.Parameters.AddWithValue("@user_id", blacklistedID); - cmd.Parameters.AddWithValue("@moderator_id", staffID); - cmd.Prepare(); - return cmd.ExecuteNonQuery() > 0; - } - catch (MySqlException) - { - return false; - } + first = false; + filterString += "AND user_id != " + userID; } - } - public static bool Unblacklist(ulong blacklistedID) - { - using (MySqlConnection c = GetConnection()) + else { - try - { - c.Open(); - MySqlCommand cmd = new MySqlCommand(@"DELETE FROM blacklisted_users WHERE user_id=@user_id", c); - cmd.Parameters.AddWithValue("@user_id", blacklistedID); - cmd.Prepare(); - return cmd.ExecuteNonQuery() > 0; - } - catch (MySqlException) - { - return false; - } - - } - } - public static bool AssignStaff(Ticket ticket, ulong staffID) - { - using (MySqlConnection c = GetConnection()) - { - try - { - c.Open(); - MySqlCommand update = new MySqlCommand(@"UPDATE tickets SET assigned_staff_id = @assigned_staff_id WHERE id = @id", c); - update.Parameters.AddWithValue("@assigned_staff_id", staffID); - update.Parameters.AddWithValue("@id", ticket.id); - update.Prepare(); - return update.ExecuteNonQuery() > 0; - } - catch (MySqlException) - { - return false; - } - - } - } - public static bool UnassignStaff(Ticket ticket) - { - return AssignStaff(ticket, 0); - } - public static StaffMember GetRandomActiveStaff(ulong currentStaffID) - { - List staffMembers = GetActiveStaff(currentStaffID); - if (!staffMembers.Any()) - { - return null; - } - - return staffMembers[random.Next(staffMembers.Count)]; - } - - public static List GetActiveStaff(ulong currentStaffID = 0) - { - using (MySqlConnection c = GetConnection()) - { - c.Open(); - MySqlCommand selection = new MySqlCommand(@"SELECT * FROM staff WHERE active = true AND user_id != @user_id", c); - selection.Parameters.AddWithValue("@user_id", currentStaffID); - selection.Prepare(); - MySqlDataReader results = selection.ExecuteReader(); - - // Check if staff exists in the database - if (!results.Read()) - { - return new List(); - } - - List staffMembers = new List { new StaffMember(results) }; - while (results.Read()) - { - staffMembers.Add(new StaffMember(results)); - } - results.Close(); - - return staffMembers; + filterString += "&& user_id != " + userID; } } - public static List GetAllStaff(ulong currentStaffID = 0) + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand selection = new MySqlCommand(@"SELECT * FROM staff WHERE active = true " + filterString, c); + selection.Prepare(); + MySqlDataReader results = selection.ExecuteReader(); + + // Check if staff exists in the database + if (!results.Read()) { - using (MySqlConnection c = GetConnection()) - { - c.Open(); - MySqlCommand selection = new MySqlCommand(@"SELECT * FROM staff WHERE user_id != @user_id", c); - selection.Parameters.AddWithValue("@user_id", currentStaffID); - selection.Prepare(); - MySqlDataReader results = selection.ExecuteReader(); - - // Check if staff exist in the database - if (!results.Read()) - { - return new List(); - } - - List staffMembers = new List { new StaffMember(results) }; - while (results.Read()) - { - staffMembers.Add(new StaffMember(results)); - } - results.Close(); - - return staffMembers; - } + return new List(); } - public static bool IsStaff(ulong staffID) + List staffMembers = new List { new StaffMember(results) }; + while (results.Read()) { - using (MySqlConnection c = GetConnection()) - { - c.Open(); - MySqlCommand selection = new MySqlCommand(@"SELECT * FROM staff WHERE user_id=@user_id", c); - selection.Parameters.AddWithValue("@user_id", staffID); - selection.Prepare(); - MySqlDataReader results = selection.ExecuteReader(); - - // Check if ticket exists in the database - if (!results.Read()) - { - return false; - } - results.Close(); - return true; - } + staffMembers.Add(new StaffMember(results)); } - public static bool TryGetStaff(ulong staffID, out StaffMember staffMember) - { - using (MySqlConnection c = GetConnection()) - { - c.Open(); - MySqlCommand selection = new MySqlCommand(@"SELECT * FROM staff WHERE user_id=@user_id", c); - selection.Parameters.AddWithValue("@user_id", staffID); - selection.Prepare(); - MySqlDataReader results = selection.ExecuteReader(); + results.Close(); - // Check if ticket exists in the database - if (!results.Read()) - { - staffMember = null; - return false; - } - staffMember = new StaffMember(results); - results.Close(); - return true; + return staffMembers; + } + + public static List GetAllStaff(params ulong[] ignoredUserIDs) + { + bool first = true; + string filterString = ""; + foreach (ulong userID in ignoredUserIDs) + { + if (first) + { + first = false; + filterString += "WHERE user_id != " + userID; } + else + { + filterString += "&& user_id != " + userID; + } + } - public static List GetAllMessages() + + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand selection = new MySqlCommand(@"SELECT * FROM staff " + filterString, c); + selection.Prepare(); + MySqlDataReader results = selection.ExecuteReader(); + + // Check if staff exist in the database + if (!results.Read()) { - using (MySqlConnection c = GetConnection()) - { - c.Open(); - MySqlCommand selection = new MySqlCommand(@"SELECT * FROM messages", c); - selection.Prepare(); - MySqlDataReader results = selection.ExecuteReader(); - - // Check if messages exist in the database - if (!results.Read()) - { - return new List(); - } - - List messages = new List { new Message(results) }; - while (results.Read()) - { - messages.Add(new Message(results)); - } - results.Close(); - - return messages; - } + return new List(); } - public static bool TryGetMessage(string identifier, out Message message) + List staffMembers = new List { new StaffMember(results) }; + while (results.Read()) { - using (MySqlConnection c = GetConnection()) - { - c.Open(); - MySqlCommand selection = new MySqlCommand(@"SELECT * FROM messages WHERE identifier=@identifier", c); - selection.Parameters.AddWithValue("@identifier", identifier); - selection.Prepare(); - MySqlDataReader results = selection.ExecuteReader(); + staffMembers.Add(new StaffMember(results)); + } + results.Close(); - // Check if ticket exists in the database - if (!results.Read()) - { - message = null; - return false; - } - message = new Message(results); - results.Close(); - return true; - } + return staffMembers; + } + + public static bool IsStaff(ulong staffID) + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand selection = new MySqlCommand(@"SELECT * FROM staff WHERE user_id=@user_id", c); + selection.Parameters.AddWithValue("@user_id", staffID); + selection.Prepare(); + MySqlDataReader results = selection.ExecuteReader(); + + // Check if ticket exists in the database + if (!results.Read()) + { + return false; + } + results.Close(); + return true; + } + + public static bool TryGetStaff(ulong staffID, out StaffMember staffMember) + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand selection = new MySqlCommand(@"SELECT * FROM staff WHERE user_id=@user_id", c); + selection.Parameters.AddWithValue("@user_id", staffID); + selection.Prepare(); + MySqlDataReader results = selection.ExecuteReader(); + + // Check if ticket exists in the database + if (!results.Read()) + { + staffMember = null; + return false; + } + staffMember = new StaffMember(results); + results.Close(); + return true; + } + + public static List GetAllMessages() + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand selection = new MySqlCommand(@"SELECT * FROM messages", c); + selection.Prepare(); + MySqlDataReader results = selection.ExecuteReader(); + + // Check if messages exist in the database + if (!results.Read()) + { + return new List(); } - public static bool AddMessage(string identifier, ulong userID, string message) + List messages = new List { new Message(results) }; + while (results.Read()) { - using (MySqlConnection c = GetConnection()) - { - try - { - c.Open(); - MySqlCommand cmd = new MySqlCommand(@"INSERT INTO messages (identifier,user_id,message) VALUES (@identifier, @user_id, @message);", c); - cmd.Parameters.AddWithValue("@identifier", identifier); - cmd.Parameters.AddWithValue("@user_id", userID); - cmd.Parameters.AddWithValue("@message", message); - cmd.Prepare(); - return cmd.ExecuteNonQuery() > 0; - } - catch (MySqlException) - { - return false; - } - } + messages.Add(new Message(results)); + } + results.Close(); + + return messages; + } + + public static bool TryGetMessage(string identifier, out Message message) + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand selection = new MySqlCommand(@"SELECT * FROM messages WHERE identifier=@identifier", c); + selection.Parameters.AddWithValue("@identifier", identifier); + selection.Prepare(); + MySqlDataReader results = selection.ExecuteReader(); + + // Check if ticket exists in the database + if (!results.Read()) + { + message = null; + return false; + } + message = new Message(results); + results.Close(); + return true; + } + + public static bool AddMessage(string identifier, ulong userID, string message) + { + try + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand cmd = new MySqlCommand(@"INSERT INTO messages (identifier,user_id,message) VALUES (@identifier, @user_id, @message);", c); + cmd.Parameters.AddWithValue("@identifier", identifier); + cmd.Parameters.AddWithValue("@user_id", userID); + cmd.Parameters.AddWithValue("@message", message); + cmd.Prepare(); + return cmd.ExecuteNonQuery() > 0; + } + catch (MySqlException) + { + return false; + } + } + + public static bool RemoveMessage(string identifier) + { + try + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand cmd = new MySqlCommand(@"DELETE FROM messages WHERE identifier=@identifier", c); + cmd.Parameters.AddWithValue("@identifier", identifier); + cmd.Prepare(); + return cmd.ExecuteNonQuery() > 0; + } + catch (MySqlException) + { + return false; + } + } + + public static List GetAllCategories() + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand selection = new MySqlCommand(@"SELECT * FROM categories", c); + selection.Prepare(); + MySqlDataReader results = selection.ExecuteReader(); + + // Check if messages exist in the database + if (!results.Read()) + { + return new List(); } - public static bool RemoveMessage(string identifier) + List categories = new List { new Category(results) }; + while (results.Read()) { - using (MySqlConnection c = GetConnection()) - { - try - { - c.Open(); - MySqlCommand cmd = new MySqlCommand(@"DELETE FROM messages WHERE identifier=@identifier", c); - cmd.Parameters.AddWithValue("@identifier", identifier); - cmd.Prepare(); - return cmd.ExecuteNonQuery() > 0; - } - catch (MySqlException) - { - return false; - } + categories.Add(new Category(results)); + } + results.Close(); - } + return categories; + } + + public static bool TryGetCategory(ulong categoryID, out Category message) + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand selection = new MySqlCommand(@"SELECT * FROM categories WHERE category_id=@category_id", c); + selection.Parameters.AddWithValue("@category_id", categoryID); + selection.Prepare(); + MySqlDataReader results = selection.ExecuteReader(); + + // Check if ticket exists in the database + if (!results.Read()) + { + message = null; + return false; + } + message = new Category(results); + results.Close(); + return true; + } + + public static bool TryGetCategory(string name, out Category message) + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand selection = new MySqlCommand(@"SELECT * FROM categories WHERE name=@name", c); + selection.Parameters.AddWithValue("@name", name); + selection.Prepare(); + MySqlDataReader results = selection.ExecuteReader(); + + // Check if ticket exists in the database + if (!results.Read()) + { + message = null; + return false; + } + message = new Category(results); + results.Close(); + return true; + } + + public static bool AddCategory(string name, ulong categoryID) + { + try + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand cmd = new MySqlCommand(@"INSERT INTO categories (name,category_id) VALUES (@name, @category_id);", c); + cmd.Parameters.AddWithValue("@name", name); + cmd.Parameters.AddWithValue("@category_id", categoryID); + cmd.Prepare(); + return cmd.ExecuteNonQuery() > 0; + } + catch (MySqlException) + { + return false; + } + } + + public static bool RemoveCategory(ulong categoryID) + { + try + { + using MySqlConnection c = GetConnection(); + c.Open(); + using MySqlCommand cmd = new MySqlCommand(@"DELETE FROM categories WHERE category_id=@category_id", c); + cmd.Parameters.AddWithValue("@category_id", categoryID); + cmd.Prepare(); + return cmd.ExecuteNonQuery() > 0; + } + catch (MySqlException) + { + return false; + } + } + + public class Ticket + { + public uint id; + public ulong creatorID; + public ulong assignedStaffID; + public string summary; + public ulong channelID; + + public Ticket(MySqlDataReader reader) + { + id = reader.GetUInt32("id"); + creatorID = reader.GetUInt64("creator_id"); + assignedStaffID = reader.GetUInt64("assigned_staff_id"); + summary = reader.GetString("summary"); + channelID = reader.GetUInt64("channel_id"); } - public class Ticket + public string DiscordRelativeTime() { - public uint id; - public DateTime createdTime; - public ulong creatorID; - public ulong assignedStaffID; - public string summary; - public ulong channelID; - - public Ticket(MySqlDataReader reader) - { - this.id = reader.GetUInt32("id"); - this.createdTime = reader.GetDateTime("created_time"); - this.creatorID = reader.GetUInt64("creator_id"); - this.assignedStaffID = reader.GetUInt64("assigned_staff_id"); - this.summary = reader.GetString("summary"); - this.channelID = reader.GetUInt64("channel_id"); - } - - public string FormattedCreatedTime() - { - return this.createdTime.ToString(Config.timestampFormat); - } + return Formatter.Timestamp(channelID.GetSnowflakeTime(), Config.timestampFormat); } - public class StaffMember - { - public ulong userID; - public string name; - public bool active; + } + public class StaffMember + { + public ulong userID; + public string name; + public bool active; - public StaffMember(MySqlDataReader reader) - { - this.userID = reader.GetUInt64("user_id"); - this.name = reader.GetString("name"); - this.active = reader.GetBoolean("active"); - } + public StaffMember(MySqlDataReader reader) + { + userID = reader.GetUInt64("user_id"); + name = reader.GetString("name"); + active = reader.GetBoolean("active"); } + } - public class Message + public class Message + { + public string identifier; + public ulong userID; + public string message; + + public Message(MySqlDataReader reader) { - public string identifier; - public ulong userID; - public string message; + identifier = reader.GetString("identifier"); + userID = reader.GetUInt64("user_id"); + message = reader.GetString("message"); + } + } - public Message(MySqlDataReader reader) - { - this.identifier = reader.GetString("identifier"); - this.userID = reader.GetUInt64("user_id"); - this.message = reader.GetString("message"); - } + public class Category + { + public string name; + public ulong id; + + public Category(MySqlDataReader reader) + { + name = reader.GetString("name"); + id = reader.GetUInt64("category_id"); } } } \ No newline at end of file diff --git a/SupportChild/EventHandler.cs b/SupportChild/EventHandler.cs index e5f61f8..27b35a6 100644 --- a/SupportChild/EventHandler.cs +++ b/SupportChild/EventHandler.cs @@ -2,223 +2,153 @@ using System.Collections.Generic; using System.Threading.Tasks; using DSharpPlus; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; -using DSharpPlus.CommandsNext.Exceptions; using DSharpPlus.Entities; using DSharpPlus.EventArgs; using DSharpPlus.Exceptions; -using Microsoft.Extensions.Logging; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; +using DSharpPlus.SlashCommands.EventArgs; +using SupportChild.Commands; -namespace SupportChild +namespace SupportChild; + +internal static class EventHandler { - internal class EventHandler + internal static Task OnReady(DiscordClient client, ReadyEventArgs e) { - private DiscordClient discordClient; + Logger.Log("Client is ready to process events."); - //DateTime for the end of the cooldown - private static Dictionary reactionTicketCooldowns = new Dictionary(); - - public EventHandler(DiscordClient client) + // Checking activity type + if (!Enum.TryParse(Config.presenceType, true, out ActivityType activityType)) { - this.discordClient = client; + Logger.Log("Presence type '" + Config.presenceType + "' invalid, using 'Playing' instead."); + activityType = ActivityType.Playing; } - internal Task OnReady(DiscordClient client, ReadyEventArgs e) + client.UpdateStatusAsync(new DiscordActivity(Config.presenceText, activityType), UserStatus.Online); + return Task.CompletedTask; + } + + internal static Task OnGuildAvailable(DiscordClient _, GuildCreateEventArgs e) + { + Logger.Log("Guild available: " + e.Guild.Name); + + IReadOnlyDictionary roles = e.Guild.Roles; + + foreach ((ulong roleID, DiscordRole role) in roles) { - discordClient.Logger.Log(LogLevel.Information, "Client is ready to process events."); + Logger.Log(role.Name.PadRight(40, '.') + roleID); + } + return Task.CompletedTask; + } - // Checking activity type - if (!Enum.TryParse(Config.presenceType, true, out ActivityType activityType)) - { - Console.WriteLine("Presence type '" + Config.presenceType + "' invalid, using 'Playing' instead."); - activityType = ActivityType.Playing; - } + internal static Task OnClientError(DiscordClient _, ClientErrorEventArgs e) + { + Logger.Error("Client exception occured:\n" + e.Exception); + switch (e.Exception) + { + case BadRequestException ex: + Logger.Error("JSON Message: " + ex.JsonMessage); + break; + } + return Task.CompletedTask; + } - this.discordClient.UpdateStatusAsync(new DiscordActivity(Config.presenceText, activityType), UserStatus.Online); - return Task.CompletedTask; + internal static async Task OnMessageCreated(DiscordClient client, MessageCreateEventArgs e) + { + if (e.Author.IsBot) + { + return; } - internal Task OnGuildAvailable(DiscordClient client, GuildCreateEventArgs e) + // Check if ticket exists in the database and ticket notifications are enabled + if (!Database.TryGetOpenTicket(e.Channel.Id, out Database.Ticket ticket) || !Config.ticketUpdatedNotifications) { - discordClient.Logger.Log(LogLevel.Information, $"Guild available: {e.Guild.Name}"); - - IReadOnlyDictionary roles = e.Guild.Roles; - - foreach ((ulong roleID, DiscordRole role) in roles) - { - discordClient.Logger.Log(LogLevel.Information, role.Name.PadRight(40, '.') + roleID); - } - return Task.CompletedTask; + return; } - internal Task OnClientError(DiscordClient client, ClientErrorEventArgs e) + // Sends a DM to the assigned staff member if at least a day has gone by since the last message and the user sending the message isn't staff + IReadOnlyList messages = await e.Channel.GetMessagesAsync(2); + if (messages.Count > 1 && messages[1].Timestamp < DateTimeOffset.UtcNow.AddDays(Config.ticketUpdatedNotificationDelay * -1) && !Database.IsStaff(e.Author.Id)) { - discordClient.Logger.Log(LogLevel.Error, $"Exception occured: {e.Exception.GetType()}: {e.Exception}"); - - return Task.CompletedTask; - } - - internal async Task OnMessageCreated(DiscordClient client, MessageCreateEventArgs e) - { - if (e.Author.IsBot) + try { - return; - } - - // Check if ticket exists in the database and ticket notifications are enabled - if (!Database.TryGetOpenTicket(e.Channel.Id, out Database.Ticket ticket) || !Config.ticketUpdatedNotifications) - { - return; - } - - // Sends a DM to the assigned staff member if at least a day has gone by since the last message and the user sending the message isn't staff - IReadOnlyList messages = await e.Channel.GetMessagesAsync(2); - if (messages.Count > 1 && messages[1].Timestamp < DateTimeOffset.UtcNow.AddDays(Config.ticketUpdatedNotificationDelay * -1) && !Database.IsStaff(e.Author.Id)) - { - try + DiscordMember staffMember = await e.Guild.GetMemberAsync(ticket.assignedStaffID); + await staffMember.SendMessageAsync(new DiscordEmbedBuilder { - DiscordEmbed message = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "A ticket you are assigned to has been updated: " + e.Channel.Mention - }; - - DiscordMember staffMember = await e.Guild.GetMemberAsync(ticket.assignedStaffID); - await staffMember.SendMessageAsync(message); - } - catch (NotFoundException) { } - catch (UnauthorizedException) { } + Color = DiscordColor.Green, + Description = "A ticket you are assigned to has been updated: " + e.Channel.Mention + }); } + catch (NotFoundException) { } + catch (UnauthorizedException) { } } + } - internal Task OnCommandError(CommandsNextExtension commandSystem, CommandErrorEventArgs e) + internal static async Task OnCommandError(SlashCommandsExtension commandSystem, SlashCommandErrorEventArgs e) + { + switch (e.Exception) { - switch (e.Exception) - { - case CommandNotFoundException _: - return Task.CompletedTask; - case ChecksFailedException _: + case SlashExecutionChecksFailedException checksFailedException: + { + foreach (SlashCheckBaseAttribute attr in checksFailedException.FailedChecks) { - foreach (CheckBaseAttribute attr in ((ChecksFailedException)e.Exception).FailedChecks) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = this.ParseFailedCheck(attr) - }; - e.Context?.Channel?.SendMessageAsync(error); - } - return Task.CompletedTask; - } - - default: - { - discordClient.Logger.Log(LogLevel.Error, $"Exception occured: {e.Exception.GetType()}: {e.Exception}"); - DiscordEmbed error = new DiscordEmbedBuilder + await e.Context.Channel.SendMessageAsync(new DiscordEmbedBuilder { Color = DiscordColor.Red, - Description = "Internal error occured, please report this to the developer." - }; - e.Context?.Channel?.SendMessageAsync(error); - return Task.CompletedTask; + Description = ParseFailedCheck(attr) + }); } - } + return; + } + + case BadRequestException ex: + Logger.Error("Command exception occured:\n" + e.Exception); + Logger.Error("JSON Message: " + ex.JsonMessage); + return; + + default: + { + Logger.Error("Exception occured: " + e.Exception.GetType() + ": " + e.Exception); + await e.Context.Channel.SendMessageAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Internal error occured, please report this to the developer." + }); + return; + } + } + } + + internal static async Task OnMemberAdded(DiscordClient client, GuildMemberAddEventArgs e) + { + if (!Database.TryGetOpenTickets(e.Member.Id, out List ownTickets)) + { + return; } - internal async Task OnReactionAdded(DiscordClient client, MessageReactionAddEventArgs e) + foreach (Database.Ticket ticket in ownTickets) { - if (e.Message.Id != Config.reactionMessage) return; - - DiscordGuild guild = e.Message.Channel.Guild; - DiscordMember member = await guild.GetMemberAsync(e.User.Id); - - if (!Config.HasPermission(member, "new") || Database.IsBlacklisted(member.Id)) return; - if (reactionTicketCooldowns.ContainsKey(member.Id)) + try { - if (reactionTicketCooldowns[member.Id] > DateTime.Now) return; // cooldown has not expired - else reactionTicketCooldowns.Remove(member.Id); // cooldown exists but has expired, delete it - } - - - DiscordChannel category = guild.GetChannel(Config.ticketCategory); - DiscordChannel ticketChannel = await guild.CreateChannelAsync("ticket", ChannelType.Text, category); - - if (ticketChannel == null) return; - - ulong staffID = 0; - if (Config.randomAssignment) - { - staffID = Database.GetRandomActiveStaff(0)?.userID ?? 0; - } - - long id = Database.NewTicket(member.Id, staffID, ticketChannel.Id); - reactionTicketCooldowns.Add(member.Id, DateTime.Now.AddSeconds(10)); // add a cooldown which expires in 10 seconds - string ticketID = id.ToString("00000"); - - await ticketChannel.ModifyAsync(model => model.Name = "ticket-" + ticketID); - await ticketChannel.AddOverwriteAsync(member, Permissions.AccessChannels, Permissions.None); - await ticketChannel.SendMessageAsync("Hello, " + member.Mention + "!\n" + Config.welcomeMessage); - - // Remove user's reaction - await e.Message.DeleteReactionAsync(e.Emoji, e.User); - - // Refreshes the channel as changes were made to it above - ticketChannel = await SupportChild.GetClient().GetChannelAsync(ticketChannel.Id); - - if (staffID != 0) - { - DiscordEmbed assignmentMessage = new DiscordEmbedBuilder + DiscordChannel channel = await client.GetChannelAsync(ticket.channelID); + if (channel?.GuildId == e.Guild.Id) { - Color = DiscordColor.Green, - Description = "Ticket was randomly assigned to <@" + staffID + ">." - }; - await ticketChannel.SendMessageAsync(assignmentMessage); - - if (Config.assignmentNotifications) - { - DiscordEmbed message = new DiscordEmbedBuilder + await channel.SendMessageAsync(new DiscordEmbedBuilder { Color = DiscordColor.Green, - Description = "You have been randomly assigned to a newly opened support ticket: " + - ticketChannel.Mention - }; - - try - { - DiscordMember staffMember = await guild.GetMemberAsync(staffID); - await staffMember.SendMessageAsync(message); - } - catch (NotFoundException) - { - } - catch (UnauthorizedException) - { - } + Description = "User '" + e.Member.Username + "#" + e.Member.Discriminator + "' has rejoined the server, and has been re-added to the ticket." + }); } } - - // Log it if the log channel exists - DiscordChannel logChannel = guild.GetChannel(Config.logChannel); - if (logChannel != null) - { - DiscordEmbed logMessage = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Ticket " + ticketChannel.Mention + " opened by " + member.Mention + ".\n", - Footer = new DiscordEmbedBuilder.EmbedFooter {Text = "Ticket " + ticketID} - }; - await logChannel.SendMessageAsync(logMessage); - } + catch (Exception) { /* ignored */ } } + } - internal async Task OnMemberAdded(DiscordClient client, GuildMemberAddEventArgs e) + internal static async Task OnMemberRemoved(DiscordClient client, GuildMemberRemoveEventArgs e) + { + if (Database.TryGetOpenTickets(e.Member.Id, out List ownTickets)) { - if (!Database.TryGetOpenTickets(e.Member.Id, out List ownTickets)) - { - return; - } - foreach (Database.Ticket ticket in ownTickets) { try @@ -226,82 +156,120 @@ namespace SupportChild DiscordChannel channel = await client.GetChannelAsync(ticket.channelID); if (channel?.GuildId == e.Guild.Id) { - await channel.AddOverwriteAsync(e.Member, Permissions.AccessChannels, Permissions.None); - DiscordEmbed message = new DiscordEmbedBuilder() - .WithColor(DiscordColor.Green) - .WithDescription("User '" + e.Member.Username + "#" + e.Member.Discriminator + "' has rejoined the server, and has been re-added to the ticket."); - await channel.SendMessageAsync(message); + await channel.SendMessageAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "User '" + e.Member.Username + "#" + e.Member.Discriminator + "' has left the server." + }); } } - catch (Exception) { } + catch (Exception) { /* ignored */ } } } - internal async Task OnMemberRemoved(DiscordClient client, Guild​Member​Remove​Event​Args e) + if (Database.TryGetAssignedTickets(e.Member.Id, out List assignedTickets) && Config.logChannel != 0) { - if (Database.TryGetOpenTickets(e.Member.Id, out List ownTickets)) + DiscordChannel logChannel = await client.GetChannelAsync(Config.logChannel); + if (logChannel != null) { - foreach(Database.Ticket ticket in ownTickets) + foreach (Database.Ticket ticket in assignedTickets) { try { DiscordChannel channel = await client.GetChannelAsync(ticket.channelID); if (channel?.GuildId == e.Guild.Id) { - DiscordEmbed message = new DiscordEmbedBuilder() - .WithColor(DiscordColor.Red) - .WithDescription("User '" + e.Member.Username + "#" + e.Member.Discriminator + "' has left the server."); - await channel.SendMessageAsync(message); - } - } - catch (Exception) { } - } - } - - if (Database.TryGetAssignedTickets(e.Member.Id, out List assignedTickets) && Config.logChannel != 0) - { - DiscordChannel logChannel = await client.GetChannelAsync(Config.logChannel); - if (logChannel != null) - { - - foreach (Database.Ticket ticket in assignedTickets) - { - try - { - DiscordChannel channel = await client.GetChannelAsync(ticket.channelID); - if (channel?.GuildId == e.Guild.Id) + await logChannel.SendMessageAsync(new DiscordEmbedBuilder { - DiscordEmbed message = new DiscordEmbedBuilder() - .WithColor(DiscordColor.Red) - .WithDescription("Assigned staff member '" + e.Member.Username + "#" + e.Member.Discriminator + "' has left the server: <#" + channel.Id + ">"); - await logChannel.SendMessageAsync(message); - } + Color = DiscordColor.Red, + Description = "Assigned staff member '" + e.Member.Username + "#" + e.Member.Discriminator + "' has left the server: <#" + channel.Id + ">" + }); } - catch (Exception) { } } + catch (Exception) { /* ignored */ } } } } - - private string ParseFailedCheck(CheckBaseAttribute attr) - { - switch (attr) - { - case CooldownAttribute _: - return "You cannot use do that so often!"; - case RequireOwnerAttribute _: - return "Only the server owner can use that command!"; - case RequirePermissionsAttribute _: - return "You don't have permission to do that!"; - case RequireRolesAttribute _: - return "You do not have a required role!"; - case RequireUserPermissionsAttribute _: - return "You don't have permission to do that!"; - case RequireNsfwAttribute _: - return "This command can only be used in an NSFW channel!"; - default: - return "Unknown Discord API error occured, please try again later."; - } - } + } + + internal static async Task OnComponentInteractionCreated(DiscordClient client, ComponentInteractionCreateEventArgs e) + { + try + { + switch (e.Interaction.Data.ComponentType) + { + case ComponentType.Button: + switch (e.Id) + { + case "supportchild_closeconfirm": + await CloseCommand.OnConfirmed(e.Interaction); + return; + case { } when e.Id.StartsWith("supportchild_newcommandbutton"): + await NewCommand.OnCategorySelection(e.Interaction); + return; + case { } when e.Id.StartsWith("supportchild_newticketbutton"): + await CreateButtonPanelCommand.OnButtonUsed(e.Interaction); + return; + case "right": + return; + case "left": + return; + case "rightskip": + return; + case "leftskip": + return; + case "stop": + return; + default: + Logger.Warn("Unknown button press received! '" + e.Id + "'"); + return; + } + case ComponentType.Select: + switch (e.Id) + { + case { } when e.Id.StartsWith("supportchild_newcommandselector"): + await NewCommand.OnCategorySelection(e.Interaction); + return; + case { } when e.Id.StartsWith("supportchild_newticketselector"): + await CreateSelectionBoxPanelCommand.OnSelectionMenuUsed(e.Interaction); + return; + default: + Logger.Warn("Unknown selection box option received! '" + e.Id + "'"); + return; + } + case ComponentType.ActionRow: + Logger.Warn("Unknown action row received! '" + e.Id + "'"); + return; + case ComponentType.FormInput: + Logger.Warn("Unknown form input received! '" + e.Id + "'"); + return; + default: + Logger.Warn("Unknown interaction type received! '" + e.Interaction.Data.ComponentType + "'"); + break; + } + } + catch (DiscordException ex) + { + Logger.Error("Interaction Exception occurred: " + ex); + Logger.Error("JsomMessage: " + ex.JsonMessage); + } + catch (Exception ex) + { + Logger.Error("Interaction Exception occured: " + ex.GetType() + ": " + ex); + } + } + + private static string ParseFailedCheck(SlashCheckBaseAttribute attr) + { + return attr switch + { + SlashRequireDirectMessageAttribute => "This command can only be used in direct messages!", + SlashRequireOwnerAttribute => "Only the server owner can use that command!", + SlashRequirePermissionsAttribute => "You don't have permission to do that!", + SlashRequireBotPermissionsAttribute => "The bot doesn't have the required permissions to do that!", + SlashRequireUserPermissionsAttribute => "You don't have permission to do that!", + SlashRequireGuildAttribute => "This command has to be used in a Discord server!", + _ => "Unknown Discord API error occured, please try again later." + }; } } \ No newline at end of file diff --git a/SupportChild/Logger.cs b/SupportChild/Logger.cs new file mode 100644 index 0000000..53d8f5e --- /dev/null +++ b/SupportChild/Logger.cs @@ -0,0 +1,68 @@ +using Microsoft.Extensions.Logging; +using System; +using System.Reflection; + +namespace SupportChild; + +public static class Logger +{ + public static void Debug(string message) + { + try + { + SupportChild.discordClient.Logger.Log(LogLevel.Debug, new EventId(420, Assembly.GetEntryAssembly()?.GetName().Name), message); + } + catch (NullReferenceException) + { + Console.WriteLine("[DEBUG] " + message); + } + } + + public static void Log(string message) + { + try + { + SupportChild.discordClient.Logger.Log(LogLevel.Information, new EventId(420, Assembly.GetEntryAssembly()?.GetName().Name), message); + } + catch (NullReferenceException) + { + Console.WriteLine("[INFO] " + message); + } + } + + public static void Warn(string message) + { + try + { + SupportChild.discordClient.Logger.Log(LogLevel.Warning, new EventId(420, Assembly.GetEntryAssembly()?.GetName().Name), message); + } + catch (NullReferenceException) + { + Console.WriteLine("[WARNING] " + message); + } + } + + public static void Error(string message) + { + try + { + SupportChild.discordClient.Logger.Log(LogLevel.Error, new EventId(420, Assembly.GetEntryAssembly()?.GetName().Name), message); + } + catch (NullReferenceException) + { + Console.WriteLine("[ERROR] " + message); + } + } + + public static void Fatal(string message) + { + try + { + SupportChild.discordClient.Logger.Log(LogLevel.Critical, new EventId(420, Assembly.GetEntryAssembly()?.GetName().Name), message); + } + catch (NullReferenceException) + { + Console.WriteLine("[CRITICAL] " + message); + } + } +} \ No newline at end of file diff --git a/SupportChild/SupportChild.cs b/SupportChild/SupportChild.cs index 3f3c1b2..4af79a3 100644 --- a/SupportChild/SupportChild.cs +++ b/SupportChild/SupportChild.cs @@ -3,161 +3,148 @@ using System.IO; using System.Reflection; using System.Threading.Tasks; using DSharpPlus; -using DSharpPlus.CommandsNext; +using DSharpPlus.Interactivity; +using DSharpPlus.Interactivity.Enums; +using DSharpPlus.Interactivity.Extensions; +using DSharpPlus.SlashCommands; using Microsoft.Extensions.Logging; using SupportChild.Commands; -namespace SupportChild +namespace SupportChild; + +internal static class SupportChild { - internal class SupportChild + // Sets up a dummy client to use for logging + public static DiscordClient discordClient = new DiscordClient(new DiscordConfiguration { Token = "DUMMY_TOKEN", TokenType = TokenType.Bot, MinimumLogLevel = LogLevel.Debug }); + private static SlashCommandsExtension commands = null; + + private static void Main() { - internal static SupportChild instance; + MainAsync().GetAwaiter().GetResult(); + } - private DiscordClient discordClient = null; - private CommandsNextExtension commands = null; - private EventHandler eventHandler; - - static void Main(string[] args) + private static async Task MainAsync() + { + Logger.Log("Starting " + Assembly.GetEntryAssembly()?.GetName().Name + " version " + GetVersion() + "..."); + try { - new SupportChild().MainAsync().GetAwaiter().GetResult(); + Reload(); + + // Block this task until the program is closed. + await Task.Delay(-1); } - - private async Task MainAsync() + catch (Exception e) { - instance = this; - - Console.WriteLine("Starting SupportChild version " + GetVersion() + "..."); - try - { - this.Reload(); - - // Block this task until the program is closed. - await Task.Delay(-1); - } - catch (Exception e) - { - Console.WriteLine("Fatal error:"); - Console.WriteLine(e); - Console.ReadLine(); - } - } - - public static DiscordClient GetClient() - { - return instance.discordClient; - } - - public static string GetVersion() - { - Version version = Assembly.GetEntryAssembly()?.GetName().Version; - return version?.Major + "." + version?.Minor + "." + version?.Build + (version?.Revision == 0 ? "" : "-" + (char)(64 + version?.Revision ?? 0)); - } - - public async void Reload() - { - if (this.discordClient != null) - { - await this.discordClient.DisconnectAsync(); - this.discordClient.Dispose(); - Console.WriteLine("Discord client disconnected."); - } - - Console.WriteLine("Loading config \"" + Directory.GetCurrentDirectory() + Path.DirectorySeparatorChar + "config.yml\""); - Config.LoadConfig(); - - // Check if token is unset - if (Config.token == "" || Config.token == "") - { - Console.WriteLine("You need to set your bot token in the config and start the bot again."); - throw new ArgumentException("Invalid Discord bot token"); - } - - // Database connection and setup - try - { - Console.WriteLine("Connecting to database... (" + Config.hostName + ":" + Config.port + ")"); - Database.SetConnectionString(Config.hostName, Config.port, Config.database, Config.username, Config.password); - Database.SetupTables(); - } - catch (Exception e) - { - Console.WriteLine("Could not set up database tables, please confirm connection settings, status of the server and permissions of MySQL user. Error: " + e); - throw; - } - - Console.WriteLine("Setting up Discord client..."); - - // Checking log level - if (!Enum.TryParse(Config.logLevel, true, out LogLevel logLevel)) - { - Console.WriteLine("Log level '" + Config.logLevel + "' invalid, using 'Information' instead."); - logLevel = LogLevel.Information; - } - - // Setting up client configuration - DiscordConfiguration cfg = new DiscordConfiguration - { - Token = Config.token, - TokenType = TokenType.Bot, - MinimumLogLevel = logLevel, - AutoReconnect = true, - Intents = DiscordIntents.All - }; - - this.discordClient = new DiscordClient(cfg); - - this.eventHandler = new EventHandler(this.discordClient); - - Console.WriteLine("Hooking events..."); - this.discordClient.Ready += this.eventHandler.OnReady; - this.discordClient.GuildAvailable += this.eventHandler.OnGuildAvailable; - this.discordClient.ClientErrored += this.eventHandler.OnClientError; - this.discordClient.MessageCreated += this.eventHandler.OnMessageCreated; - this.discordClient.GuildMemberAdded += this.eventHandler.OnMemberAdded; - this.discordClient.GuildMemberRemoved += this.eventHandler.OnMemberRemoved; - if (Config.reactionMessage != 0) - { - this.discordClient.MessageReactionAdded += this.eventHandler.OnReactionAdded; - } - - Console.WriteLine("Registering commands..."); - commands = discordClient.UseCommandsNext(new CommandsNextConfiguration - { - StringPrefixes = new []{ Config.prefix } - }); - - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - this.commands.RegisterCommands(); - - Console.WriteLine("Hooking command events..."); - this.commands.CommandErrored += this.eventHandler.OnCommandError; - - Console.WriteLine("Connecting to Discord..."); - await this.discordClient.ConnectAsync(); + Logger.Fatal("Fatal error:\n" + e); + Console.ReadLine(); } } + + public static string GetVersion() + { + Version version = Assembly.GetEntryAssembly()?.GetName().Version; + return version?.Major + "." + version?.Minor + "." + version?.Build + (version?.Revision == 0 ? "" : "-" + (char)(64 + version?.Revision ?? 0)); + } + + public static async void Reload() + { + if (discordClient != null) + { + await discordClient.DisconnectAsync(); + discordClient.Dispose(); + Logger.Log("Discord client disconnected."); + } + + Logger.Log("Loading config \"" + Directory.GetCurrentDirectory() + Path.DirectorySeparatorChar + "config.yml\""); + Config.LoadConfig(); + + // Check if token is unset + if (Config.token is "" or "") + { + Logger.Fatal("You need to set your bot token in the config and start the bot again."); + throw new ArgumentException("Invalid Discord bot token"); + } + + // Database connection and setup + try + { + Logger.Log("Connecting to database... (" + Config.hostName + ":" + Config.port + ")"); + Database.SetConnectionString(Config.hostName, Config.port, Config.database, Config.username, Config.password); + Database.SetupTables(); + } + catch (Exception e) + { + Logger.Fatal("Could not set up database tables, please confirm connection settings, status of the server and permissions of MySQL user. Error: " + e); + throw; + } + + Logger.Log("Setting up Discord client..."); + + // Setting up client configuration + DiscordConfiguration cfg = new DiscordConfiguration + { + Token = Config.token, + TokenType = TokenType.Bot, + MinimumLogLevel = Config.logLevel, + AutoReconnect = true, + Intents = DiscordIntents.All + }; + + discordClient = new DiscordClient(cfg); + + Logger.Log("Hooking events..."); + discordClient.Ready += EventHandler.OnReady; + discordClient.GuildAvailable += EventHandler.OnGuildAvailable; + discordClient.ClientErrored += EventHandler.OnClientError; + discordClient.MessageCreated += EventHandler.OnMessageCreated; + discordClient.GuildMemberAdded += EventHandler.OnMemberAdded; + discordClient.GuildMemberRemoved += EventHandler.OnMemberRemoved; + discordClient.ComponentInteractionCreated += EventHandler.OnComponentInteractionCreated; + + discordClient.UseInteractivity(new InteractivityConfiguration + { + AckPaginationButtons = true, + PaginationBehaviour = PaginationBehaviour.Ignore, + PaginationDeletion = PaginationDeletion.DeleteMessage, + Timeout = TimeSpan.FromMinutes(15) + }); + + Logger.Log("Registering commands..."); + commands = discordClient.UseSlashCommands(); + + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + commands.RegisterCommands(); + + Logger.Log("Hooking command events..."); + commands.SlashCommandErrored += EventHandler.OnCommandError; + + Logger.Log("Connecting to Discord..."); + await discordClient.ConnectAsync(); + } } \ No newline at end of file diff --git a/SupportChild/SupportChild.csproj b/SupportChild/SupportChild.csproj index 5b31187..ea42f5f 100644 --- a/SupportChild/SupportChild.csproj +++ b/SupportChild/SupportChild.csproj @@ -5,7 +5,7 @@ ellie_icon.ico net6.0 win-x64;linux-x64 - 1.2.0 + SupportChild.SupportChild EmotionChild @@ -13,58 +13,58 @@ https://github.com/EmotionChild/SupportChild Git LICENSE - https://cdn.emotionchild.com/Ellie.png - A Discord support bot build for the Ellie's Home Discord server - 2.6.1.1 - 2.6.1.1 + https://cdn.discordapp.com/attachments/765441543100170271/914327948667011132/Ellie_Concept_2_transparent_ver.png + A Discord support ticket bot built for the Ellie's home server en - 1.2.0 + 3.0.0.1 + 1.3.0 + 3.0.0.1 + 3.0.0.1 + + + + full - + + - + - + + - - True - True - Resources.resx - + + ResXFileCodeGenerator + Resources.Designer.cs + + + + + - - ResXFileCodeGenerator - Resources.Designer.cs - + + lib\DiscordChatExporter.Core.dll + - - True - - + + True + True + Resources.resx + - - - - - - - lib\DiscordChatExporter.Core.dll - - - - + \ No newline at end of file diff --git a/SupportChild/Transcriber.cs b/SupportChild/Transcriber.cs index 9b10e62..0fd0d5c 100644 --- a/SupportChild/Transcriber.cs +++ b/SupportChild/Transcriber.cs @@ -3,60 +3,51 @@ using System.Threading.Tasks; using DiscordChatExporter.Core.Discord; using DiscordChatExporter.Core.Discord.Data; -using DiscordChatExporter.Core.Exceptions; using DiscordChatExporter.Core.Exporting; using DiscordChatExporter.Core.Exporting.Filtering; using DiscordChatExporter.Core.Exporting.Partitioning; -using DiscordChatExporter.Core.Utils.Extensions; -namespace SupportChild +namespace SupportChild; + +internal static class Transcriber { - internal static class Transcriber - { - internal static async Task ExecuteAsync(ulong channelID, uint ticketID) - { - DiscordClient discordClient = new DiscordClient(new AuthToken(AuthTokenKind.Bot, Config.token)); - ChannelExporter Exporter = new ChannelExporter(discordClient); + internal static async Task ExecuteAsync(ulong channelID, uint ticketID) + { + DiscordClient discordClient = new DiscordClient(Config.token); + ChannelExporter exporter = new ChannelExporter(discordClient); - if (!Directory.Exists("./transcripts")) - { - Directory.CreateDirectory("./transcripts"); - } + if (!Directory.Exists("./transcripts")) + { + Directory.CreateDirectory("./transcripts"); + } - string dateFormat = "yyyy-MMM-dd HH:mm"; + Channel channel = await discordClient.GetChannelAsync(new Snowflake(channelID)); + Guild guild = await discordClient.GetGuildAsync(channel.GuildId); - // Configure settings - if (Config.timestampFormat != "") - dateFormat = Config.timestampFormat; + ExportRequest request = new ExportRequest( + Guild: guild, + Channel: channel, + OutputPath: GetPath(ticketID), + Format: ExportFormat.HtmlDark, + After: null, + Before: null, + PartitionLimit: PartitionLimit.Null, + MessageFilter: MessageFilter.Null, + ShouldDownloadMedia: false, + ShouldReuseMedia: false, + DateFormat: "yyyy-MMM-dd HH:mm" + ); - Channel channel = await discordClient.GetChannelAsync(new Snowflake(channelID)); - Guild guild = await discordClient.GetGuildAsync(channel.GuildId); + await exporter.ExportChannelAsync(request); + } - ExportRequest request = new ExportRequest( - guild: guild, - channel: channel, - outputPath: GetPath(ticketID), - format: ExportFormat.HtmlDark, - after: null, - before: null, - partitionLimit: PartitionLimit.Null, - messageFilter: MessageFilter.Null, - shouldDownloadMedia: false, - shouldReuseMedia: false, - dateFormat: dateFormat - ); + internal static string GetPath(uint ticketNumber) + { + return "./transcripts/" + GetFilename(ticketNumber); + } - await Exporter.ExportChannelAsync(request); - } - - internal static string GetPath(uint ticketNumber) - { - return "./transcripts/" + GetFilename(ticketNumber); - } - - internal static string GetFilename(uint ticketNumber) - { - return "ticket-" + ticketNumber.ToString("00000") + ".html"; - } - } + internal static string GetFilename(uint ticketNumber) + { + return "ticket-" + ticketNumber.ToString("00000") + ".html"; + } } \ No newline at end of file diff --git a/SupportChild/Utilities.cs b/SupportChild/Utilities.cs index 06099d1..b554cfc 100644 --- a/SupportChild/Utilities.cs +++ b/SupportChild/Utilities.cs @@ -1,71 +1,65 @@ using System; using System.Collections.Generic; -using System.Security.Cryptography; +using System.Threading.Tasks; using DSharpPlus.Entities; -namespace SupportChild +namespace SupportChild; + +public static class Utilities { - public static class Utilities - { - public static List RandomizeList(List list) - { - RNGCryptoServiceProvider provider = new RNGCryptoServiceProvider(); - int n = list.Count; - while (n > 1) - { - byte[] box = new byte[1]; - do provider.GetBytes(box); - while (!(box[0] < n * (Byte.MaxValue / n))); - int k = (box[0] % n); - n--; - T value = list[k]; - list[k] = list[n]; - list[n] = value; - } + private static readonly Random rng = new Random(); - return list; - } + public static void Shuffle(this IList list) + { + int n = list.Count; + while (n > 1) + { + n--; + int k = rng.Next(n + 1); + (list[k], list[n]) = (list[n], list[k]); + } + } - public static string[] ParseIDs(string args) - { - if (string.IsNullOrEmpty(args)) - { - return new string[0]; - } - return args.Trim().Replace("<@!", "").Replace("<@", "").Replace(">", "").Split(); - } + public static LinkedList ParseListIntoMessages(List listItems) + { + LinkedList messages = new LinkedList(); - public static LinkedList ParseListIntoMessages(List listItems) - { - LinkedList messages = new LinkedList(); + foreach (string listItem in listItems) + { + if (messages.Last?.Value?.Length + listItem?.Length < 2048) + { + messages.Last.Value += listItem; + } + else + { + messages.AddLast(listItem); + } + } - foreach (string listItem in listItems) - { - if (messages.Last?.Value?.Length + listItem?.Length < 2048) - { - messages.Last.Value += listItem; - } - else - { - messages.AddLast(listItem); - } - } + return messages; + } - return messages; - } + public static async Task> GetVerifiedChannels() + { + List verifiedCategories = new List(); + foreach (Database.Category category in Database.GetAllCategories()) + { + DiscordChannel channel = null; + try + { + channel = await SupportChild.discordClient.GetChannelAsync(category.id); + } + catch (Exception) { /*ignored*/ } - public static DiscordRole GetRoleByName(DiscordGuild guild, string Name) - { - Name = Name.Trim().ToLower(); - foreach (DiscordRole role in guild.Roles.Values) - { - if (role.Name.ToLower().StartsWith(Name)) - { - return role; - } - } - - return null; - } - } + if (channel != null) + { + verifiedCategories.Add(category); + } + else + { + Logger.Warn("Category '" + category.name + "' (" + category.id + ") no longer exists! Ignoring..."); + } + } + return verifiedCategories; + } } \ No newline at end of file diff --git a/SupportChild/default_config.yml b/SupportChild/default_config.yml index 8dcb6e8..b24019a 100644 --- a/SupportChild/default_config.yml +++ b/SupportChild/default_config.yml @@ -1,84 +1,48 @@ bot: - # Bot token. - token: "" - # Command prefix. - prefix: "-" - # Channel where ticket logs are posted (recommended) - log-channel: 000000000000000000 - # Category where the ticket will be created, it will have the same permissions of that ticket plus read permissions for the user opening the ticket (recommended) - ticket-category: 000000000000000000 - # A message which will open new tickets when users react to it (optional) - reaction-message: 000000000000000000 - # Message posted when a ticket is opened. - welcome-message: "Please describe your issue below, and include all information needed for us to help you." - # Decides what messages are shown in console - # Possible values are: Critical, Error, Warning, Information, Debug. - console-log-level: "Information" - # Format for timestamps in transcripts and google sheets if used - timestamp-format: "yyyy-MM-dd HH:mm" - # Whether or not staff members should be randomly assigned tickets when they are made. Individual staff members can opt out using the toggleactive command. - random-assignment: true - # If set to true the rasssign command will include staff members set as inactive if a specific role is specified in the command. - # This can be useful if you have admins set as inactive to not automatically recieve tickets and then have moderators elevate tickets when needed. - random-assign-role-override: true - # Sets the type of activity for the bot to display in its presence status - # Possible values are: Playing, Streaming, ListeningTo, Watching, Competing - presence-type: "ListeningTo" - # Sets the activity text shown in the bot's status - presence-text: "-new" + # Bot token. + token: "" + # Channel where ticket logs are posted (recommended) + log-channel: 000000000000000000 + # Message posted when a ticket is opened. + welcome-message: "Please describe your issue below, and include all information needed for us to take action." + # Decides what messages are shown in console + # Possible values are: Critical, Error, Warning, Information, Debug. + console-log-level: "Information" + # One of the following: LongDate, LongDateTime, LongTime, RelativeTime, ShortDate, ShortDateTime, ShortTime + # More info: https://dsharpplus.github.io/api/DSharpPlus.TimestampFormat.html + timestamp-format: "RelativeTime" + # Whether or not staff members should be randomly assigned tickets when they are made. Individual staff members can opt out using the toggleactive command. + random-assignment: true + # If set to true the rasssign command will include staff members set as inactive if a specific role is specified in the command. + # This can be useful if you have admins set as inactive to not automatically receive tickets and then have moderators elevate tickets when needed. + random-assign-role-override: true + # Sets the type of activity for the bot to display in its presence status + # Possible values are: Playing, Streaming, ListeningTo, Watching, Competing + presence-type: "ListeningTo" + # Sets the activity text shown in the bot's status + presence-text: "/new" + # Set to true if you want the /new command to show a selection box instead of a series of buttons + new-command-uses-selector: false + # Number of tickets a single user can have open at a time, staff members are excluded from this + ticket-limit: 5 notifications: - # Notifies the assigned staff member when a new message is posted in a ticket if the ticket has been silent for a configurable amount of time - # Other staff members and bots do not trigger this. - ticket-updated: true - # The above notification will only be sent if the ticket has been silent for more than this amount of days. Default is 0.5 days. - ticket-updated-delay: 0.5 - # Notifies staff when they are assigned to tickets - assignment: true - # Notifies the user opening the ticket that their ticket was closed and includes the transcript - closing: true + # Notifies the assigned staff member when a new message is posted in a ticket if the ticket has been silent for a configurable amount of time + # Other staff members and bots do not trigger this. + ticket-updated: true + # The above notification will only be sent if the ticket has been silent for more than this amount of days. Default is 0.5 days. + ticket-updated-delay: 0.5 + # Notifies staff when they are assigned to tickets + assignment: true + # Notifies the user opening the ticket that their ticket was closed and includes the transcript + closing: true database: - # Address and port of the mysql server - address: "127.0.0.1" - port: 3306 - # Name of the database to use - name: "supportchild" - # Username and password for authentication - user: "" - password: "" - -# Set up which roles are allowed to use different commands. -# Example: -# new: [ 000000000000000000, 111111111111111111 ] -# They are grouped into suggested command groups below for first time setup. -permissions: - # Public commands - close: [] - list: [] - new: [] - say: [] - status: [] - summary: [] - transcript: [] - # Moderator commands - add: [] - addmessage: [] - assign: [] - blacklist: [] - listassigned: [] - listoldest: [] - listunassigned: [] - move: [] - rassign: [] - removemessage: [] - setsummary: [] - toggleactive: [] - unassign: [] - unblacklist: [] - # Admin commands - addstaff: [] - reload: [] - removestaff: [] - setticket: [] - unsetticket: [] \ No newline at end of file + # Address and port of the mysql server + address: "127.0.0.1" + port: 3306 + # Name of the database to use + name: "supportchild" + # Username and password for authentication + user: "" + password: "" \ No newline at end of file From 94df384934a3b8f82d1b3f98e51b4a8ca1268b10 Mon Sep 17 00:00:00 2001 From: EmotionChild Date: Sun, 21 Aug 2022 19:34:11 +1200 Subject: [PATCH 4/5] Updated commands --- SupportChild/Commands/AddCategoryCommand.cs | 71 ++++ SupportChild/Commands/AddCommand.cs | 145 +++----- SupportChild/Commands/AddMessageCommand.cs | 117 +++--- SupportChild/Commands/AddStaffCommand.cs | 127 +++---- SupportChild/Commands/AdminCommands.cs | 200 ++++++++++ SupportChild/Commands/AssignCommand.cs | 189 ++++------ SupportChild/Commands/BlacklistCommand.cs | 115 ++---- SupportChild/Commands/CloseCommand.cs | 212 ++++++----- .../Commands/CreateButtonPanelCommand.cs | 81 +++++ .../CreateSelectionBoxPanelCommand.cs | 79 ++++ SupportChild/Commands/ListAssignedCommand.cs | 117 +++--- SupportChild/Commands/ListCommand.cs | 162 ++++----- SupportChild/Commands/ListOldestCommand.cs | 71 ---- SupportChild/Commands/ListOpen.cs | 60 +++ .../Commands/ListUnassignedCommand.cs | 98 ++--- SupportChild/Commands/MoveCommand.cs | 159 ++++---- SupportChild/Commands/NewCommand.cs | 342 ++++++++++++------ SupportChild/Commands/RandomAssignCommand.cs | 289 +++++++-------- SupportChild/Commands/ReloadCommand.cs | 38 -- .../Commands/RemoveCategoryCommand.cs | 41 +++ SupportChild/Commands/RemoveMessageCommand.cs | 97 ++--- SupportChild/Commands/RemoveStaffCommand.cs | 127 ++----- SupportChild/Commands/SayCommand.cs | 111 +++--- SupportChild/Commands/SetSummaryCommand.cs | 89 ++--- SupportChild/Commands/SetTicketCommand.cs | 98 ----- SupportChild/Commands/StatusCommand.cs | 55 +-- SupportChild/Commands/SummaryCommand.cs | 76 ++-- SupportChild/Commands/ToggleActiveCommand.cs | 102 ++---- SupportChild/Commands/TranscriptCommand.cs | 241 +++++------- SupportChild/Commands/UnassignComand.cs | 91 ++--- SupportChild/Commands/UnblacklistCommand.cs | 115 ++---- SupportChild/Commands/UnsetTicketCommand.cs | 70 ---- 32 files changed, 1926 insertions(+), 2059 deletions(-) create mode 100644 SupportChild/Commands/AddCategoryCommand.cs create mode 100644 SupportChild/Commands/AdminCommands.cs create mode 100644 SupportChild/Commands/CreateButtonPanelCommand.cs create mode 100644 SupportChild/Commands/CreateSelectionBoxPanelCommand.cs delete mode 100644 SupportChild/Commands/ListOldestCommand.cs create mode 100644 SupportChild/Commands/ListOpen.cs delete mode 100644 SupportChild/Commands/ReloadCommand.cs create mode 100644 SupportChild/Commands/RemoveCategoryCommand.cs delete mode 100644 SupportChild/Commands/SetTicketCommand.cs delete mode 100644 SupportChild/Commands/UnsetTicketCommand.cs diff --git a/SupportChild/Commands/AddCategoryCommand.cs b/SupportChild/Commands/AddCategoryCommand.cs new file mode 100644 index 0000000..3656fdb --- /dev/null +++ b/SupportChild/Commands/AddCategoryCommand.cs @@ -0,0 +1,71 @@ +using System.Threading.Tasks; +using DSharpPlus.Entities; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; + +namespace SupportChild.Commands; + +public class AddCategoryCommand : ApplicationCommandModule +{ + [SlashRequireGuild] + [SlashCommand("addcategory", "Adds a category to the ticket bot letting users open tickets in them.")] + public async Task OnExecute(InteractionContext command, [Option("Title", "The name to display on buttons and in selection boxes.")] string title, [Option("Category", "The category to add.")] DiscordChannel category) + { + if (!category.IsCategory) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "That channel is not a category." + }, true); + return; + } + + if (string.IsNullOrWhiteSpace(title)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Invalid category title specified." + }, true); + return; + } + + if (Database.TryGetCategory(category.Id, out Database.Category _)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "That category is already registered." + }, true); + return; + } + + if (Database.TryGetCategory(title, out Database.Category _)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "There is already a category with that title." + }, true); + return; + } + + if (Database.AddCategory(title, category.Id)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Category added." + }, true); + } + else + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Error: Failed adding the category to the database." + }, true); + } + } +} \ No newline at end of file diff --git a/SupportChild/Commands/AddCommand.cs b/SupportChild/Commands/AddCommand.cs index 64e5935..47b92f8 100644 --- a/SupportChild/Commands/AddCommand.cs +++ b/SupportChild/Commands/AddCommand.cs @@ -1,107 +1,82 @@ using System; using System.Threading.Tasks; using DSharpPlus; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; using DSharpPlus.Entities; -using Microsoft.Extensions.Logging; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class AddCommand : ApplicationCommandModule { - public class AddCommand : BaseCommandModule + [SlashRequireGuild] + [SlashCommand("add", "Adds a user to a ticket")] + public async Task OnExecute(InteractionContext command, [Option("User", "User to add to ticket.")] DiscordUser user) { - [Command("add")] - [Description("Adds a user to a ticket.")] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) + // Check if ticket exists in the database + if (!Database.IsOpenTicket(command.Channel.Id)) { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "add")) + await command.CreateResponseAsync(new DiscordEmbedBuilder { - DiscordEmbed error = new DiscordEmbedBuilder + Color = DiscordColor.Red, + Description = "This channel is not a ticket." + }, true); + return; + } + + DiscordMember member; + try + { + member = (user == null ? command.Member : await command.Guild.GetMemberAsync(user.Id)); + + if (member == null) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder { Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the add command but did not have permission."); + Description = "Could not find that user in this server." + }, true); return; } - - // Check if ticket exists in the database - if (!Database.IsOpenTicket(command.Channel.Id)) + } + catch (Exception) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "This channel is not a ticket." - }; - await command.RespondAsync(error); - return; - } + Color = DiscordColor.Red, + Description = "Could not find that user in this server." + }, true); + return; + } - string[] parsedArgs = Utilities.ParseIDs(command.RawArgumentString); - foreach (string parsedArg in parsedArgs) + try + { + await command.Channel.AddOverwriteAsync(member, Permissions.AccessChannels); + await command.CreateResponseAsync(new DiscordEmbedBuilder { - if (!ulong.TryParse(parsedArg, out ulong userID)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Invalid ID/Mention. (Could not convert to numerical)" - }; - await command.RespondAsync(error); - continue; - } + Color = DiscordColor.Green, + Description = "Added " + member.Mention + " to ticket." + }); - DiscordMember mentionedMember; - try + // Log it if the log channel exists + DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); + if (logChannel != null) + { + await logChannel.SendMessageAsync(new DiscordEmbedBuilder { - mentionedMember = await command.Guild.GetMemberAsync(userID); - } - catch (Exception) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Invalid ID/Mention. (Could not find user on this server)" - }; - await command.RespondAsync(error); - continue; - } - - try - { - await command.Channel.AddOverwriteAsync(mentionedMember, Permissions.AccessChannels, Permissions.None); - DiscordEmbed message = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Added " + mentionedMember.Mention + " to ticket." - }; - await command.RespondAsync(message); - - // Log it if the log channel exists - DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); - if (logChannel != null) - { - DiscordEmbed logMessage = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = mentionedMember.Mention + " was added to " + command.Channel.Mention + - " by " + command.Member.Mention + "." - }; - await logChannel.SendMessageAsync(logMessage); - } - } - catch (Exception) - { - DiscordEmbed message = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Could not add <@" + parsedArg + "> to ticket, unknown error occured." - }; - await command.RespondAsync(message); - } + Color = DiscordColor.Green, + Description = member.Mention + " was added to " + command.Channel.Mention + + " by " + command.Member.Mention + "." + }); } } + catch (Exception) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Could not add " + member.Mention + " to ticket, unknown error occured." + }, true); + } } } \ No newline at end of file diff --git a/SupportChild/Commands/AddMessageCommand.cs b/SupportChild/Commands/AddMessageCommand.cs index a2e6606..6ffa491 100644 --- a/SupportChild/Commands/AddMessageCommand.cs +++ b/SupportChild/Commands/AddMessageCommand.cs @@ -1,76 +1,53 @@ -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; -using DSharpPlus.Entities; -using Microsoft.Extensions.Logging; -using System; -using System.Collections.Generic; -using System.Linq; -using System.Text; +using DSharpPlus.Entities; using System.Threading.Tasks; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class AddMessageCommand : ApplicationCommandModule { - public class AddMessageCommand : BaseCommandModule - { - [Command("addmessage")] - [Description("Adds a new message for the 'say' command.")] - public async Task OnExecute(CommandContext command, string identifier, [RemainingText] string message) - { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "addmessage")) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the addmessage command but did not have permission."); - return; - } + [SlashRequireGuild] + [SlashCommand("addmessage", "Adds a new message for the 'say' command.")] + public async Task OnExecute(InteractionContext command, + [Option("Identifier", "The identifier word used in the /say command.")] string identifier, + [Option("Message", "The message the /say command will return.")] string message) + { + if (string.IsNullOrEmpty(message)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "No message specified." + }, true); + return; + } - if (string.IsNullOrEmpty(message)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "No message specified." - }; - await command.RespondAsync(error); - return; - } - - if (Database.TryGetMessage(identifier.ToLower(), out Database.Message _)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "There is already a message with that identifier." - }; - await command.RespondAsync(error); - return; - } + if (Database.TryGetMessage(identifier.ToLower(), out Database.Message _)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "There is already a message with that identifier." + }, true); + return; + } - if(Database.AddMessage(identifier, command.Member.Id, message)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Message added." - }; - await command.RespondAsync(error); - return; - } - else - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Error: Failed adding the message to the database." - }; - await command.RespondAsync(error); - return; - } - } - } + if (Database.AddMessage(identifier, command.Member.Id, message)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Message added." + }, true); + } + else + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Error: Failed adding the message to the database." + }, true); + } + } } \ No newline at end of file diff --git a/SupportChild/Commands/AddStaffCommand.cs b/SupportChild/Commands/AddStaffCommand.cs index 03f5948..f0e48ea 100644 --- a/SupportChild/Commands/AddStaffCommand.cs +++ b/SupportChild/Commands/AddStaffCommand.cs @@ -1,96 +1,67 @@ using System; -using System.Linq; using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; using DSharpPlus.Entities; -using Microsoft.Extensions.Logging; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; using MySql.Data.MySqlClient; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class AddStaffCommand : ApplicationCommandModule { - public class AddStaffCommand : BaseCommandModule + [SlashRequireGuild] + [SlashCommand("addstaff", "Adds a new staff member.")] + public async Task OnExecute(InteractionContext command, [Option("User", "User to add to staff.")] DiscordUser user) { - [Command("addstaff")] - [Cooldown(1, 5, CooldownBucketType.User)] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) + DiscordMember staffMember = null; + try { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "addstaff")) + staffMember = user == null ? command.Member : await command.Guild.GetMemberAsync(user.Id); + + if (staffMember == null) { - DiscordEmbed error = new DiscordEmbedBuilder + await command.CreateResponseAsync(new DiscordEmbedBuilder { Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the addstaff command but did not have permission."); + Description = "Could not find that user in this server." + }, true); return; } - - ulong userID; - string[] parsedArgs = Utilities.ParseIDs(commandArgs); - - if (!parsedArgs.Any()) + } + catch (Exception) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder { - userID = command.Member.Id; - } - else if (!ulong.TryParse(parsedArgs[0], out userID)) + Color = DiscordColor.Red, + Description = "Could not find that user in this server." + }, true); + return; + } + + await using MySqlConnection c = Database.GetConnection(); + MySqlCommand cmd = Database.IsStaff(staffMember.Id) ? new MySqlCommand(@"UPDATE staff SET name = @name WHERE user_id = @user_id", c) : new MySqlCommand(@"INSERT INTO staff (user_id, name) VALUES (@user_id, @name);", c); + + c.Open(); + cmd.Parameters.AddWithValue("@user_id", staffMember.Id); + cmd.Parameters.AddWithValue("@name", staffMember.DisplayName); + cmd.ExecuteNonQuery(); + cmd.Dispose(); + + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = staffMember.Mention + " was added to staff." + }); + + // Log it if the log channel exists + DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); + if (logChannel != null) + { + await logChannel.SendMessageAsync(new DiscordEmbedBuilder { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Invalid ID/Mention. (Could not convert to numerical)" - }; - await command.RespondAsync(error); - return; - } - - DiscordMember member; - try - { - member = await command.Guild.GetMemberAsync(userID); - } - catch (Exception) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Invalid ID/Mention. (Could not find user on this server)" - }; - await command.RespondAsync(error); - return; - } - - using (MySqlConnection c = Database.GetConnection()) - { - MySqlCommand cmd = Database.IsStaff(userID) ? new MySqlCommand(@"UPDATE staff SET name = @name WHERE user_id = @user_id", c) : new MySqlCommand(@"INSERT INTO staff (user_id, name) VALUES (@user_id, @name);", c); - - c.Open(); - cmd.Parameters.AddWithValue("@user_id", userID); - cmd.Parameters.AddWithValue("@name", member.DisplayName); - cmd.ExecuteNonQuery(); - cmd.Dispose(); - - DiscordEmbed message = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = member.Mention + " was added to staff." - }; - await command.RespondAsync(message); - - // Log it if the log channel exists - DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); - if (logChannel != null) - { - DiscordEmbed logMessage = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = member.Mention + " was added to staff.\n", - }; - await logChannel.SendMessageAsync(logMessage); - } - } + Color = DiscordColor.Green, + Description = staffMember.Mention + " was added to staff.\n" + }); } } } \ No newline at end of file diff --git a/SupportChild/Commands/AdminCommands.cs b/SupportChild/Commands/AdminCommands.cs new file mode 100644 index 0000000..bf90847 --- /dev/null +++ b/SupportChild/Commands/AdminCommands.cs @@ -0,0 +1,200 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Threading.Tasks; +using DSharpPlus.Entities; +using DSharpPlus.Interactivity; +using DSharpPlus.Interactivity.Extensions; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; + +namespace SupportChild.Commands; + +[SlashCommandGroup("admin", "Administrative commands.")] +public class AdminCommands : ApplicationCommandModule +{ + [SlashRequireGuild] + [SlashCommand("listinvalid", "List tickets which channels have been deleted. Use /admin unsetticket to remove them.")] + public async Task ListInvalid(InteractionContext command) + { + if (!Database.TryGetOpenTickets(out List openTickets)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + 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 allChannels = new List(); + foreach (KeyValuePair guild in SupportChild.discordClient.Guilds) + { + try + { + allChannels.AddRange(await guild.Value.GetChannelsAsync()); + } + catch (Exception) { /*ignored*/ } + } + + // Check which tickets channels no longer exist + List listItems = new List(); + 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) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "All tickets are valid!" + }, true); + return; + } + + List embeds = new List(); + 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 listPages = new List(); + foreach (DiscordEmbedBuilder embed in embeds) + { + listPages.Add(new Page("", embed)); + } + + await command.Interaction.SendPaginatedResponseAsync(true, command.User, listPages); + } + + [SlashRequireGuild] + [SlashCommand("setticket", "Turns a channel into a ticket WARNING: Anyone will be able to delete the channel using /close.")] + public async Task SetTicket(InteractionContext command, [Option("User", "(Optional) The owner of the ticket.")] DiscordUser user = null) + { + // Check if ticket exists in the database + if (Database.IsOpenTicket(command.Channel.Id)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + 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); + string ticketID = id.ToString("00000"); + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Channel has been designated ticket " + ticketID + "." + }); + + // Log it if the log channel exists + DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); + if (logChannel != null) + { + await logChannel.SendMessageAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = command.Channel.Mention + " has been designated ticket " + ticketID + " by " + command.Member.Mention + "." + }); + } + } + + [SlashRequireGuild] + [SlashCommand("unsetticket", "Deletes a ticket from the ticket system without deleting the channel.")] + public async Task UnsetTicket(InteractionContext command, [Option("TicketID", "(Optional) Ticket to unset. Uses the channel you are in by default.")] long ticketID = 0) + { + Database.Ticket ticket; + + if (ticketID == 0) + { + // Check if ticket exists in the database + if (!Database.TryGetOpenTicket(command.Channel.Id, out ticket)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + 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)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "There is no ticket with this ticket ID." + }, true); + return; + } + } + + + if (Database.DeleteOpenTicket(ticket.id)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Channel has been undesignated as a ticket." + }); + + // Log it if the log channel exists + DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); + if (logChannel != null) + { + await logChannel.SendMessageAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = command.Channel.Mention + " has been undesignated as a ticket by " + command.Member.Mention + "." + }); + } + } + else + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Error: Failed removing ticket from database." + }, true); + } + } + + [SlashCommand("reload", "Reloads the bot config.")] + public async Task Reload(InteractionContext command) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Reloading bot application..." + }); + Logger.Log("Reloading bot..."); + SupportChild.Reload(); + } +} \ No newline at end of file diff --git a/SupportChild/Commands/AssignCommand.cs b/SupportChild/Commands/AssignCommand.cs index a4660f6..171760a 100644 --- a/SupportChild/Commands/AssignCommand.cs +++ b/SupportChild/Commands/AssignCommand.cs @@ -1,135 +1,102 @@ -using System.Linq; +using System; using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; using DSharpPlus.Entities; using DSharpPlus.Exceptions; -using Microsoft.Extensions.Logging; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class AssignCommand : ApplicationCommandModule { - public class AssignCommand : BaseCommandModule + [SlashRequireGuild] + [SlashCommand("assign", "Assigns a staff member to this ticket.")] + public async Task OnExecute(InteractionContext command, [Option("User", "(Optional) User to assign to this ticket.")] DiscordUser user = null) { - [Command("assign")] - [Description("Assigns a staff member to a ticket.")] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) + DiscordMember member = null; + try { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "assign")) + member = user == null ? command.Member : await command.Guild.GetMemberAsync(user.Id); + + if (member == null) { - DiscordEmbed error = new DiscordEmbedBuilder + await command.CreateResponseAsync(new DiscordEmbedBuilder { Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the assign command but did not have permission."); + Description = "Could not find that user in this server." + }, true); return; } - - // Check if ticket exists in the database - if (!Database.TryGetOpenTicket(command.Channel.Id, out Database.Ticket ticket)) + } + catch (Exception) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "This channel is not a ticket." - }; - await command.RespondAsync(error); - return; - } + Color = DiscordColor.Red, + Description = "Could not find that user in this server." + }, true); + return; + } - ulong staffID; - string[] parsedMessage = Utilities.ParseIDs(command.RawArgumentString); - - if (!parsedMessage.Any()) + // Check if ticket exists in the database + if (!Database.TryGetOpenTicket(command.Channel.Id, out Database.Ticket ticket)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder { - staffID = command.Member.Id; - } - else if (!ulong.TryParse(parsedMessage[0], out staffID)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Invalid ID/Mention. (Could not convert to numerical)" - }; - await command.RespondAsync(error); - return; - } + Color = DiscordColor.Red, + Description = "This channel is not a ticket." + }, true); + return; + } - DiscordMember staffMember = null; + if (!Database.IsStaff(member.Id)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Error: User is not registered as staff." + }, true); + return; + } + + if (!Database.AssignStaff(ticket, member.Id)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Error: Failed to assign " + member.Mention + " to ticket." + }, true); + return; + } + + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Assigned " + member.Mention + " to ticket." + }); + + if (Config.assignmentNotifications) + { try { - staffMember = await command.Guild.GetMemberAsync(staffID); - } - catch (NotFoundException) { } - - if (staffMember == null) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Error: Could not find user." - }; - await command.RespondAsync(error); - return; - } - - if (!Database.IsStaff(staffMember.Id)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Error: User is not registered as staff." - }; - await command.RespondAsync(error); - return; - } - - if (!Database.AssignStaff(ticket, staffID)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Error: Failed to assign " + staffMember.Mention + " to ticket." - }; - await command.RespondAsync(error); - return; - } - - DiscordEmbed feedback = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Assigned " + staffMember.Mention + " to ticket." - }; - await command.RespondAsync(feedback); - - if (Config.assignmentNotifications) - { - try - { - DiscordEmbed message = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "You have been assigned to a support ticket: " + command.Channel.Mention - }; - await staffMember.SendMessageAsync(message); - } - catch (UnauthorizedException) {} - - } - - // Log it if the log channel exists - DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); - if (logChannel != null) - { - DiscordEmbed logMessage = new DiscordEmbedBuilder + await member.SendMessageAsync(new DiscordEmbedBuilder { Color = DiscordColor.Green, - Description = staffMember.Mention + " was assigned to " + command.Channel.Mention + " by " + command.Member.Mention + "." - }; - await logChannel.SendMessageAsync(logMessage); + Description = "You have been assigned to a support ticket: " + command.Channel.Mention + }); } + catch (UnauthorizedException) { } + } + + // Log it if the log channel exists + DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); + if (logChannel != null) + { + await logChannel.SendMessageAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = member.Mention + " was assigned to " + command.Channel.Mention + " by " + command.Member.Mention + "." + }); } } } \ No newline at end of file diff --git a/SupportChild/Commands/BlacklistCommand.cs b/SupportChild/Commands/BlacklistCommand.cs index dedf9ad..33a6262 100644 --- a/SupportChild/Commands/BlacklistCommand.cs +++ b/SupportChild/Commands/BlacklistCommand.cs @@ -1,99 +1,54 @@ using System; using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; using DSharpPlus.Entities; -using DSharpPlus.Exceptions; -using Microsoft.Extensions.Logging; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class BlacklistCommand : ApplicationCommandModule { - public class BlacklistCommand : BaseCommandModule + [SlashRequireGuild] + [SlashCommand("blacklist", "Blacklists a user from opening tickets.")] + public async Task OnExecute(InteractionContext command, [Option("User", "User to blacklist.")] DiscordUser user) { - [Command("blacklist")] - [Description("Blacklists a user from opening tickets.")] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) + try { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "blacklist")) + if (!Database.Blacklist(user.Id, command.User.Id)) { - DiscordEmbed error = new DiscordEmbedBuilder + await command.CreateResponseAsync(new DiscordEmbedBuilder { Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the blacklist command but did not have permission."); + Description = user.Mention + " is already blacklisted." + }, true); return; } - string[] parsedArgs = Utilities.ParseIDs(command.RawArgumentString); - foreach (string parsedArg in parsedArgs) + await command.CreateResponseAsync(new DiscordEmbedBuilder { - if (ulong.TryParse(parsedArg, out ulong userId)) + Color = DiscordColor.Green, + Description = "Blacklisted " + user.Mention + "." + }, true); + + // Log it if the log channel exists + DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); + if (logChannel != null) + { + await logChannel.SendMessageAsync(new DiscordEmbedBuilder { - DiscordUser blacklistedUser = null; - try - { - blacklistedUser = await command.Client.GetUserAsync(userId); - } - catch (NotFoundException) { } - - if (blacklistedUser == null) - { - DiscordEmbed message = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Error: Could not find user." - }; - await command.RespondAsync(message); - continue; - } - - try - { - if (!Database.Blacklist(blacklistedUser.Id, command.User.Id)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = blacklistedUser.Mention + " is already blacklisted." - }; - await command.RespondAsync(error); - continue; - } - - DiscordEmbed message = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Blacklisted " + blacklistedUser.Mention + "." - }; - await command.RespondAsync(message); - - // Log it if the log channel exists - DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); - if (logChannel != null) - { - DiscordEmbed logMessage = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = blacklistedUser.Mention + " was blacklisted from opening tickets by " + command.Member.Mention + "." - }; - await logChannel.SendMessageAsync(logMessage); - } - } - catch (Exception) - { - DiscordEmbed message = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Error occured while blacklisting " + blacklistedUser.Mention + "." - }; - await command.RespondAsync(message); - throw; - } - } + Color = DiscordColor.Green, + Description = user.Mention + " was blacklisted from opening tickets by " + command.Member.Mention + "." + }); } } + catch (Exception) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Error occured while blacklisting " + user.Mention + "." + }, true); + throw; + } } } \ No newline at end of file diff --git a/SupportChild/Commands/CloseCommand.cs b/SupportChild/Commands/CloseCommand.cs index 51e5324..059d34f 100644 --- a/SupportChild/Commands/CloseCommand.cs +++ b/SupportChild/Commands/CloseCommand.cs @@ -2,117 +2,133 @@ using System.Collections.Generic; using System.IO; using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; +using DSharpPlus; using DSharpPlus.Entities; using DSharpPlus.Exceptions; -using Microsoft.Extensions.Logging; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class CloseCommand : ApplicationCommandModule { - public class CloseCommand : BaseCommandModule + [SlashRequireGuild] + [SlashCommand("close", "Closes a ticket.")] + public async Task OnExecute(InteractionContext command) { - [Command("close")] - [Cooldown(1, 5, CooldownBucketType.User)] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) + // Check if ticket exists in the database + if (!Database.TryGetOpenTicket(command.Channel.Id, out Database.Ticket _)) { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "close")) + await command.CreateResponseAsync(new DiscordEmbedBuilder { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the close command but did not have permission."); - return; - } + Color = DiscordColor.Red, + Description = "This channel is not a ticket." + }); + return; + } - ulong channelID = command.Channel.Id; - string channelName = command.Channel.Name; - - // Check if ticket exists in the database - if (!Database.TryGetOpenTicket(channelID, out Database.Ticket ticket)) + DiscordInteractionResponseBuilder confirmation = new DiscordInteractionResponseBuilder() + .AddEmbed(new DiscordEmbedBuilder { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "This channel is not a ticket." - }; - await command.RespondAsync(error); - return; - } + Color = DiscordColor.Cyan, + Description = "Are you sure you wish to close this ticket? You cannot re-open it again later." + }) + .AddComponents(new DiscordButtonComponent(ButtonStyle.Danger, "supportboi_closeconfirm", "Confirm")); + + + await command.CreateResponseAsync(confirmation); + } + + public static async Task OnConfirmed(DiscordInteraction interaction) + { + await interaction.CreateResponseAsync(InteractionResponseType.DeferredMessageUpdate); + ulong channelID = interaction.Channel.Id; + string channelName = interaction.Channel.Name; + + // Check if ticket exists in the database + if (!Database.TryGetOpenTicket(channelID, out Database.Ticket ticket)) + { + await interaction.EditOriginalResponseAsync(new DiscordWebhookBuilder().AddEmbed(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "This channel is not a ticket." + })); + return; + } + + // Build transcript + try + { + await Transcriber.ExecuteAsync(interaction.Channel.Id, ticket.id); + } + catch (Exception e) + { + Logger.Error("Exception occured when trying to save transcript while closing ticket: " + e); + await interaction.EditOriginalResponseAsync(new DiscordWebhookBuilder().AddEmbed(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "ERROR: Could not save transcript file. Aborting..." + })); + return; + } + + // Log it if the log channel exists + DiscordChannel logChannel = interaction.Guild.GetChannel(Config.logChannel); + if (logChannel != null) + { + DiscordEmbed embed = new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Ticket " + ticket.id.ToString("00000") + " closed by " + interaction.User.Mention + ".\n", + Footer = new DiscordEmbedBuilder.EmbedFooter { Text = '#' + channelName } + }; + + await using FileStream file = new FileStream(Transcriber.GetPath(ticket.id), FileMode.Open, FileAccess.Read); + DiscordMessageBuilder message = new DiscordMessageBuilder(); + message.WithEmbed(embed); + message.WithFiles(new Dictionary { { Transcriber.GetFilename(ticket.id), file } }); + + await logChannel.SendMessageAsync(message); + } + + if (Config.closingNotifications) + { + DiscordEmbed embed = new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Ticket " + ticket.id.ToString("00000") + " which you opened has now been closed, check the transcript for more info.\n", + Footer = new DiscordEmbedBuilder.EmbedFooter { Text = '#' + channelName } + }; - // Build transcript try { - await Transcriber.ExecuteAsync(command.Channel.Id, ticket.id); + DiscordMember staffMember = await interaction.Guild.GetMemberAsync(ticket.creatorID); + await using FileStream file = new FileStream(Transcriber.GetPath(ticket.id), FileMode.Open, FileAccess.Read); + + DiscordMessageBuilder message = new DiscordMessageBuilder(); + message.WithEmbed(embed); + message.WithFiles(new Dictionary { { Transcriber.GetFilename(ticket.id), file } }); + + await staffMember.SendMessageAsync(message); } - catch (Exception) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "ERROR: Could not save transcript file. Aborting..." - }; - await command.RespondAsync(error); - throw; - } - - // Log it if the log channel exists - DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); - if (logChannel != null) - { - DiscordEmbed embed = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Ticket " + ticket.id.ToString("00000") + " closed by " + command.Member.Mention + ".\n", - Footer = new DiscordEmbedBuilder.EmbedFooter { Text = '#' + channelName } - }; - - using (FileStream file = new FileStream(Transcriber.GetPath(ticket.id), FileMode.Open, FileAccess.Read)) - { - DiscordMessageBuilder message = new DiscordMessageBuilder(); - message.WithEmbed(embed); - message.WithFiles(new Dictionary() { { Transcriber.GetFilename(ticket.id), file } }); - - await logChannel.SendMessageAsync(message); - } - } - - if (Config.closingNotifications) - { - DiscordEmbed embed = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Ticket " + ticket.id.ToString("00000") + " which you opened has now been closed, check the transcript for more info.\n", - Footer = new DiscordEmbedBuilder.EmbedFooter { Text = '#' + channelName } - }; - - try - { - DiscordMember staffMember = await command.Guild.GetMemberAsync(ticket.creatorID); - - using (FileStream file = new FileStream(Transcriber.GetPath(ticket.id), FileMode.Open, FileAccess.Read)) - { - DiscordMessageBuilder message = new DiscordMessageBuilder(); - message.WithEmbed(embed); - message.WithFiles(new Dictionary() { { Transcriber.GetFilename(ticket.id), file } }); - - await staffMember.SendMessageAsync(message); - } - } - catch (NotFoundException) { } - catch (UnauthorizedException) { } - } - - Database.ArchiveTicket(ticket); - - // Delete the channel and database entry - await command.Channel.DeleteAsync("Ticket closed."); - - Database.DeleteOpenTicket(ticket.id); + catch (NotFoundException) { } + catch (UnauthorizedException) { } } + + Database.ArchiveTicket(ticket); + + await interaction.EditOriginalResponseAsync(new DiscordWebhookBuilder().AddEmbed(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Channel will be deleted in 3 seconds..." + })); + + + await Task.Delay(3000); + + // Delete the channel and database entry + await interaction.Channel.DeleteAsync("Ticket closed."); + + Database.DeleteOpenTicket(ticket.id); } } \ No newline at end of file diff --git a/SupportChild/Commands/CreateButtonPanelCommand.cs b/SupportChild/Commands/CreateButtonPanelCommand.cs new file mode 100644 index 0000000..15ee739 --- /dev/null +++ b/SupportChild/Commands/CreateButtonPanelCommand.cs @@ -0,0 +1,81 @@ +using System.Collections.Generic; +using System.Linq; +using System.Threading.Tasks; +using DSharpPlus; +using DSharpPlus.Entities; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; + +namespace SupportChild.Commands; + +public class CreateButtonPanelCommand : ApplicationCommandModule +{ + [SlashRequireGuild] + [SlashCommand("createbuttonpanel", "Creates a series of buttons which users can use to open new tickets in specific categories.")] + public async Task OnExecute(InteractionContext command) + { + DiscordMessageBuilder builder = new DiscordMessageBuilder().WithContent(" "); + List verifiedCategories = await Utilities.GetVerifiedChannels(); + + if (verifiedCategories.Count == 0) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Error: No registered categories found." + }, true); + return; + } + + verifiedCategories = verifiedCategories.OrderBy(x => x.name).ToList(); + + int nrOfButtons = 0; + for (int nrOfButtonRows = 0; nrOfButtonRows < 5 && nrOfButtons < verifiedCategories.Count; nrOfButtonRows++) + { + List buttonRow = new List(); + + for (; nrOfButtons < 5 * (nrOfButtonRows + 1) && nrOfButtons < verifiedCategories.Count; nrOfButtons++) + { + buttonRow.Add(new DiscordButtonComponent(ButtonStyle.Primary, "supportboi_newticketbutton " + verifiedCategories[nrOfButtons].id, verifiedCategories[nrOfButtons].name)); + } + builder.AddComponents(buttonRow); + } + + await command.Channel.SendMessageAsync(builder); + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Successfully created message, make sure to run this command again if you add new categories to the bot." + }, true); + } + + public static async Task OnButtonUsed(DiscordInteraction interaction) + { + await interaction.CreateResponseAsync(InteractionResponseType.DeferredChannelMessageWithSource, new DiscordInteractionResponseBuilder().AsEphemeral()); + + if (!ulong.TryParse(interaction.Data.CustomId.Replace("supportboi_newticketbutton ", ""), out ulong categoryID) || categoryID == 0) + { + Logger.Warn("Invalid ID: " + interaction.Data.CustomId.Replace("supportboi_newticketbutton ", "")); + return; + } + + (bool success, string message) = await NewCommand.OpenNewTicket(interaction.User.Id, interaction.ChannelId, categoryID); + + if (success) + { + await interaction.CreateFollowupMessageAsync(new DiscordFollowupMessageBuilder().AddEmbed(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = message + })); + } + else + { + await interaction.CreateFollowupMessageAsync(new DiscordFollowupMessageBuilder().AddEmbed(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = message + })); + } + } +} \ No newline at end of file diff --git a/SupportChild/Commands/CreateSelectionBoxPanelCommand.cs b/SupportChild/Commands/CreateSelectionBoxPanelCommand.cs new file mode 100644 index 0000000..395d1e5 --- /dev/null +++ b/SupportChild/Commands/CreateSelectionBoxPanelCommand.cs @@ -0,0 +1,79 @@ +using System.Collections.Generic; +using System.Linq; +using System.Threading.Tasks; +using DSharpPlus; +using DSharpPlus.Entities; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; + +namespace SupportChild.Commands; + +public class CreateSelectionBoxPanelCommand : ApplicationCommandModule +{ + [SlashRequireGuild] + [SlashCommand("createselectionboxpanel", "Creates a selection box which users can use to open new tickets in specific categories.")] + public async Task OnExecute(InteractionContext command, [Option("Message", "(Optional) The message to show in the selection box.")] string message = null) + { + DiscordMessageBuilder builder = new DiscordMessageBuilder() + .WithContent(" ") + .AddComponents(await GetSelectComponents(command, message ?? "Open new ticket...")); + + await command.Channel.SendMessageAsync(builder); + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Successfully created message, make sure to run this command again if you add new categories to the bot." + }, true); + } + + public static async Task> GetSelectComponents(InteractionContext command, string placeholder) + { + List verifiedCategories = await Utilities.GetVerifiedChannels(); + + if (verifiedCategories.Count == 0) return new List(); + + verifiedCategories = verifiedCategories.OrderBy(x => x.name).ToList(); + List selectionComponents = new List(); + int selectionOptions = 0; + for (int selectionBoxes = 0; selectionBoxes < 5 && selectionOptions < verifiedCategories.Count; selectionBoxes++) + { + List categoryOptions = new List(); + + for (; selectionOptions < 25 * (selectionBoxes + 1) && selectionOptions < verifiedCategories.Count; selectionOptions++) + { + categoryOptions.Add(new DiscordSelectComponentOption(verifiedCategories[selectionOptions].name, verifiedCategories[selectionOptions].id.ToString())); + } + selectionComponents.Add(new DiscordSelectComponent("supportboi_newticketselector" + selectionBoxes, placeholder, categoryOptions, false, 0, 1)); + } + + return selectionComponents; + } + + public static async Task OnSelectionMenuUsed(DiscordInteraction interaction) + { + if (interaction.Data.Values == null || interaction.Data.Values.Length <= 0) return; + + if (!ulong.TryParse(interaction.Data.Values[0], out ulong categoryID) || categoryID == 0) return; + + await interaction.CreateResponseAsync(InteractionResponseType.DeferredChannelMessageWithSource, new DiscordInteractionResponseBuilder().AsEphemeral()); + + (bool success, string message) = await NewCommand.OpenNewTicket(interaction.User.Id, interaction.ChannelId, categoryID); + + if (success) + { + await interaction.CreateFollowupMessageAsync(new DiscordFollowupMessageBuilder().AddEmbed(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = message + }).AsEphemeral()); + } + else + { + await interaction.CreateFollowupMessageAsync(new DiscordFollowupMessageBuilder().AddEmbed(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = message + }).AsEphemeral()); + } + } +} \ No newline at end of file diff --git a/SupportChild/Commands/ListAssignedCommand.cs b/SupportChild/Commands/ListAssignedCommand.cs index 351df6c..b9ce36f 100644 --- a/SupportChild/Commands/ListAssignedCommand.cs +++ b/SupportChild/Commands/ListAssignedCommand.cs @@ -1,76 +1,63 @@ using System.Collections.Generic; -using System.Linq; using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; using DSharpPlus.Entities; -using Microsoft.Extensions.Logging; +using DSharpPlus.Interactivity; +using DSharpPlus.Interactivity.Extensions; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class ListAssignedCommand : ApplicationCommandModule { - public class ListAssignedCommand : BaseCommandModule + [SlashRequireGuild] + [SlashCommand("listassigned", "Lists tickets assigned to a user.")] + public async Task OnExecute(InteractionContext command, [Option("User", "(Optional) User to list tickets for.")] DiscordUser user = null) { - [Command("listassigned")] - [Aliases("la")] - [Cooldown(1, 5, CooldownBucketType.User)] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) + DiscordUser listUser = user == null ? command.User : user; + + if (!Database.TryGetAssignedTickets(listUser.Id, out List assignedTickets)) { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "listassigned")) + await command.CreateResponseAsync(new DiscordEmbedBuilder { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the listassigned command but did not have permission."); - return; - } - - ulong staffID; - string[] parsedIDs = Utilities.ParseIDs(command.RawArgumentString); - - if (!parsedIDs.Any()) - { - staffID = command.Member.Id; - } - else if (!ulong.TryParse(parsedIDs[0], out staffID)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Invalid ID/Mention. (Could not convert to numerical)" - }; - await command.RespondAsync(error); - return; - } - - if (!Database.TryGetAssignedTickets(staffID, out List assignedTickets)) - { - DiscordEmbed error = new DiscordEmbedBuilder() - .WithColor(DiscordColor.Red) - .WithDescription("User does not have any assigned tickets."); - await command.RespondAsync(error); - return; - } - - List listItems = new List(); - foreach (Database.Ticket ticket in assignedTickets) - { - listItems.Add("**" + ticket.FormattedCreatedTime() + ":** <#" + ticket.channelID + "> by <@" + ticket.creatorID + ">\n"); - } - - LinkedList messages = Utilities.ParseListIntoMessages(listItems); - foreach (string message in messages) - { - DiscordEmbed channelInfo = new DiscordEmbedBuilder() - .WithTitle("Assigned tickets: ") - .WithColor(DiscordColor.Green) - .WithDescription(message); - await command.RespondAsync(channelInfo); - } - + Color = DiscordColor.Red, + Description = "User does not have any assigned tickets." + }); + return; } + + List listItems = new List(); + foreach (Database.Ticket ticket in assignedTickets) + { + listItems.Add("**" + ticket.DiscordRelativeTime() + ":** <#" + ticket.channelID + "> by <@" + ticket.creatorID + ">\n"); + } + + List embeds = new List(); + foreach (string message in Utilities.ParseListIntoMessages(listItems)) + { + embeds.Add(new DiscordEmbedBuilder + { + Title = "Assigned tickets: ", + Color = DiscordColor.Green, + 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 listPages = new List(); + foreach (DiscordEmbedBuilder embed in embeds) + { + listPages.Add(new Page("", embed)); + } + + await command.Interaction.SendPaginatedResponseAsync(true, command.User, listPages); } } \ No newline at end of file diff --git a/SupportChild/Commands/ListCommand.cs b/SupportChild/Commands/ListCommand.cs index 3916ed4..e2811b5 100644 --- a/SupportChild/Commands/ListCommand.cs +++ b/SupportChild/Commands/ListCommand.cs @@ -1,101 +1,99 @@ using System.Collections.Generic; -using System.Linq; using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; using DSharpPlus.Entities; -using Microsoft.Extensions.Logging; +using DSharpPlus.Interactivity; +using DSharpPlus.Interactivity.Extensions; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class ListCommand : ApplicationCommandModule { - public class ListCommand : BaseCommandModule + [SlashRequireGuild] + [SlashCommand("list", "Lists tickets opened by a user.")] + public async Task OnExecute(InteractionContext command, [Option("User", "(Optional) The user to get tickets by.")] DiscordUser user = null) { - [Command("list")] - [Cooldown(1, 5, CooldownBucketType.User)] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) + DiscordUser listUser = user == null ? command.User : user; + + List openEmbeds = new List(); + if (Database.TryGetOpenTickets(listUser.Id, out List openTickets)) { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "list")) + List listItems = new List(); + foreach (Database.Ticket ticket in openTickets) { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the list command but did not have permission."); - return; + listItems.Add("**" + ticket.DiscordRelativeTime() + ":** <#" + ticket.channelID + ">\n"); } - ulong userID; - string[] parsedMessage = Utilities.ParseIDs(command.RawArgumentString); - - if (!parsedMessage.Any()) + foreach (string message in Utilities.ParseListIntoMessages(listItems)) { - userID = command.Member.Id; - } - else if (!ulong.TryParse(parsedMessage[0], out userID)) - { - DiscordEmbed error = new DiscordEmbedBuilder + openEmbeds.Add(new DiscordEmbedBuilder { - Color = DiscordColor.Red, - Description = "Invalid ID/Mention. (Could not convert to numerical)" - }; - await command.RespondAsync(error); - return; + Color = DiscordColor.Green, + Description = message + }); } - if (Database.TryGetOpenTickets(userID, out List openTickets)) + // Add the titles + for (int i = 0; i < openEmbeds.Count; i++) { - List listItems = new List(); - foreach (Database.Ticket ticket in openTickets) - { - listItems.Add("**" + ticket.FormattedCreatedTime() + ":** <#" + ticket.channelID + ">\n"); - } - - LinkedList messages = Utilities.ParseListIntoMessages(listItems); - foreach (string message in messages) - { - DiscordEmbed channelInfo = new DiscordEmbedBuilder() - .WithTitle("Open tickets: ") - .WithColor(DiscordColor.Green) - .WithDescription(message); - await command.RespondAsync(channelInfo); - } - } - else - { - DiscordEmbed channelInfo = new DiscordEmbedBuilder() - .WithColor(DiscordColor.Green) - .WithDescription("User does not have any open tickets."); - await command.RespondAsync(channelInfo); - } - - if (Database.TryGetClosedTickets(userID, out List closedTickets)) - { - List listItems = new List(); - foreach (Database.Ticket ticket in closedTickets) - { - listItems.Add("**" + ticket.FormattedCreatedTime() + ":** Ticket " + ticket.id.ToString("00000") + "\n"); - } - - LinkedList messages = Utilities.ParseListIntoMessages(listItems); - foreach (string message in messages) - { - DiscordEmbed channelInfo = new DiscordEmbedBuilder() - .WithTitle("Closed tickets: ") - .WithColor(DiscordColor.Red) - .WithDescription(message); - await command.RespondAsync(channelInfo); - } - } - else - { - DiscordEmbed channelInfo = new DiscordEmbedBuilder() - .WithColor(DiscordColor.Red) - .WithDescription("User does not have any closed tickets."); - await command.RespondAsync(channelInfo); + openEmbeds[i].Title = $"Open tickets ({i + 1}/{openEmbeds.Count})"; } } + + List closedEmbeds = new List(); + if (Database.TryGetClosedTickets(listUser.Id, out List closedTickets)) + { + List listItems = new List(); + foreach (Database.Ticket ticket in closedTickets) + { + listItems.Add("**" + ticket.DiscordRelativeTime() + ":** Ticket " + ticket.id.ToString("00000") + "\n"); + } + + foreach (string message in Utilities.ParseListIntoMessages(listItems)) + { + closedEmbeds.Add(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = message + }); + } + + // Add the titles + for (int i = 0; i < closedEmbeds.Count; i++) + { + closedEmbeds[i].Title = $"Closed tickets ({i + 1}/{closedEmbeds.Count})"; + } + } + + // Merge the embed lists and add the footers + List embeds = new List(); + embeds.AddRange(openEmbeds); + embeds.AddRange(closedEmbeds); + for (int i = 0; i < embeds.Count; i++) + { + embeds[i].Footer = new DiscordEmbedBuilder.EmbedFooter + { + Text = $"Page {i + 1} / {embeds.Count}" + }; + } + + if (embeds.Count == 0) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Cyan, + Description = "User does not have any open or closed tickets." + }); + return; + } + + List listPages = new List(); + foreach (DiscordEmbedBuilder embed in embeds) + { + listPages.Add(new Page("", embed)); + } + + await command.Interaction.SendPaginatedResponseAsync(true, command.User, listPages); } } \ No newline at end of file diff --git a/SupportChild/Commands/ListOldestCommand.cs b/SupportChild/Commands/ListOldestCommand.cs deleted file mode 100644 index 394b8d0..0000000 --- a/SupportChild/Commands/ListOldestCommand.cs +++ /dev/null @@ -1,71 +0,0 @@ -using System.Collections.Generic; -using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; -using DSharpPlus.Entities; -using Microsoft.Extensions.Logging; - -namespace SupportChild.Commands -{ - public class ListOldestCommand : BaseCommandModule - { - [Command("listoldest")] - [Aliases("lo")] - [Cooldown(1, 5, CooldownBucketType.User)] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) - { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "listoldest")) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the listoldest command but did not have permission."); - return; - } - - int listLimit = 20; - if (!string.IsNullOrEmpty(command.RawArgumentString?.Trim() ?? "")) - { - if (!int.TryParse(command.RawArgumentString?.Trim(), out listLimit) || listLimit < 5 || listLimit > 100) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Invalid list amount. (Must be integer between 5 and 100)" - }; - await command.RespondAsync(error); - return; - } - } - - if (!Database.TryGetOldestTickets(command.Member.Id, out List openTickets, listLimit)) - { - DiscordEmbed channelInfo = new DiscordEmbedBuilder() - .WithColor(DiscordColor.Red) - .WithDescription("Could not fetch any open tickets."); - await command.RespondAsync(channelInfo); - return; - } - - List listItems = new List(); - foreach (Database.Ticket ticket in openTickets) - { - listItems.Add("**" + ticket.FormattedCreatedTime() + ":** <#" + ticket.channelID + "> by <@" + ticket.creatorID + ">\n"); - } - - LinkedList messages = Utilities.ParseListIntoMessages(listItems); - foreach (string message in messages) - { - DiscordEmbed channelInfo = new DiscordEmbedBuilder() - .WithTitle("The " + openTickets.Count + " oldest open tickets: ") - .WithColor(DiscordColor.Green) - .WithDescription(message?.Trim()); - await command.RespondAsync(channelInfo); - } - } - } -} \ No newline at end of file diff --git a/SupportChild/Commands/ListOpen.cs b/SupportChild/Commands/ListOpen.cs new file mode 100644 index 0000000..7775e17 --- /dev/null +++ b/SupportChild/Commands/ListOpen.cs @@ -0,0 +1,60 @@ +using System.Collections.Generic; +using System.Threading.Tasks; +using DSharpPlus.Entities; +using DSharpPlus.Interactivity; +using DSharpPlus.Interactivity.Extensions; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; + +namespace SupportChild.Commands; + +public class ListOpen : ApplicationCommandModule +{ + [SlashRequireGuild] + [SlashCommand("listopen", "Lists all open tickets, oldest first.")] + public async Task OnExecute(InteractionContext command) + { + if (!Database.TryGetOpenTickets(out List openTickets)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Could not fetch any open tickets." + }); + return; + } + + List listItems = new List(); + foreach (Database.Ticket ticket in openTickets) + { + listItems.Add("**" + ticket.DiscordRelativeTime() + ":** <#" + ticket.channelID + "> by <@" + ticket.creatorID + ">\n"); + } + + List embeds = new List(); + foreach (string message in Utilities.ParseListIntoMessages(listItems)) + { + embeds.Add(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + 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 listPages = new List(); + foreach (DiscordEmbedBuilder embed in embeds) + { + listPages.Add(new Page("", embed)); + } + + await command.Interaction.SendPaginatedResponseAsync(true, command.User, listPages); + } +} \ No newline at end of file diff --git a/SupportChild/Commands/ListUnassignedCommand.cs b/SupportChild/Commands/ListUnassignedCommand.cs index 19236cf..bb0bba7 100644 --- a/SupportChild/Commands/ListUnassignedCommand.cs +++ b/SupportChild/Commands/ListUnassignedCommand.cs @@ -1,55 +1,61 @@ using System.Collections.Generic; using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; using DSharpPlus.Entities; -using Microsoft.Extensions.Logging; +using DSharpPlus.Interactivity; +using DSharpPlus.Interactivity.Extensions; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class ListUnassignedCommand : ApplicationCommandModule { - public class ListUnassignedCommand : BaseCommandModule - { - [Command("listunassigned")] - [Aliases("lu")] - [Cooldown(1, 5, CooldownBucketType.User)] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) - { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "listunassigned")) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the listunassigned command but did not have permission."); - return; - } + [SlashRequireGuild] + [SlashCommand("listunassigned", "Lists unassigned tickets.")] + public async Task OnExecute(InteractionContext command) + { + if (!Database.TryGetAssignedTickets(0, out List unassignedTickets)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "There are no unassigned tickets." + }); + return; + } - if (!Database.TryGetAssignedTickets(0, out List unassignedTickets)) - { - DiscordEmbed response = new DiscordEmbedBuilder() - .WithColor(DiscordColor.Green) - .WithDescription("There are no unassigned tickets."); - await command.RespondAsync(response); - } + List listItems = new List(); + foreach (Database.Ticket ticket in unassignedTickets) + { + listItems.Add("**" + ticket.DiscordRelativeTime() + ":** <#" + ticket.channelID + "> by <@" + ticket.creatorID + ">\n"); + } - List listItems = new List(); - foreach (Database.Ticket ticket in unassignedTickets) - { - listItems.Add("**" + ticket.FormattedCreatedTime() + ":** <#" + ticket.channelID + "> by <@" + ticket.creatorID + ">\n"); - } + List embeds = new List(); + foreach (string message in Utilities.ParseListIntoMessages(listItems)) + { + embeds.Add(new DiscordEmbedBuilder + { + Title = "Unassigned tickets: ", + Color = DiscordColor.Green, + Description = message + }); + } - LinkedList messages = Utilities.ParseListIntoMessages(listItems); - foreach (string message in messages) - { - DiscordEmbed channelInfo = new DiscordEmbedBuilder() - .WithTitle("Unassigned tickets: ") - .WithColor(DiscordColor.Green) - .WithDescription(message?.Trim()); - await command.RespondAsync(channelInfo); - } - } - } + // Add the footers + for (int i = 0; i < embeds.Count; i++) + { + embeds[i].Footer = new DiscordEmbedBuilder.EmbedFooter + { + Text = $"Page {i + 1} / {embeds.Count}" + }; + } + + List listPages = new List(); + foreach (DiscordEmbedBuilder embed in embeds) + { + listPages.Add(new Page("", embed)); + } + + await command.Interaction.SendPaginatedResponseAsync(true, command.User, listPages); + } } \ No newline at end of file diff --git a/SupportChild/Commands/MoveCommand.cs b/SupportChild/Commands/MoveCommand.cs index 4ef0737..e81cf70 100644 --- a/SupportChild/Commands/MoveCommand.cs +++ b/SupportChild/Commands/MoveCommand.cs @@ -2,103 +2,82 @@ using System.Collections.Generic; using System.Linq; using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; using DSharpPlus.Entities; using DSharpPlus.Exceptions; -using Microsoft.Extensions.Logging; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class MoveCommand : ApplicationCommandModule { - public class MoveCommand : BaseCommandModule + [SlashRequireGuild] + [SlashCommand("move", "Moves a ticket to another category.")] + public async Task OnExecute(InteractionContext command, [Option("Category", "The category to move the ticket to. Only has to be the beginning of the name.")] string category) { - [Command("move")] - [Description("Moves a ticket to another category.")] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) + // Check if ticket exists in the database + if (!Database.TryGetOpenTicket(command.Channel.Id, out Database.Ticket _)) { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "move")) + await command.CreateResponseAsync(new DiscordEmbedBuilder { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the move command but did not have permission."); - return; - } - - // Check if ticket exists in the database - if (!Database.TryGetOpenTicket(command.Channel.Id, out Database.Ticket ticket)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "This channel is not a ticket." - }; - await command.RespondAsync(error); - return; - } - - if (string.IsNullOrEmpty(command.RawArgumentString)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Error: No category provided." - }; - await command.RespondAsync(error); - return; - } - - IReadOnlyList channels = await command.Guild.GetChannelsAsync(); - IEnumerable categories = channels.Where(x => x.IsCategory); - DiscordChannel category = categories.FirstOrDefault(x => x.Name.StartsWith(command.RawArgumentString.Trim(), StringComparison.OrdinalIgnoreCase)); - - if (category == null) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Error: Could not find a category by that name." - }; - await command.RespondAsync(error); - return; - } - - if (command.Channel.Id == category.Id) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Error: The ticket is already in that category." - }; - await command.RespondAsync(error); - return; - } - - try - { - await command.Channel.ModifyAsync(modifiedAttributes => modifiedAttributes.Parent = category); - } - catch (UnauthorizedException) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Error: Not authorized to move this ticket to that category." - }; - await command.RespondAsync(error); - return; - } - - DiscordEmbed feedback = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Ticket was moved to " + category.Mention - }; - await command.RespondAsync(feedback); + Color = DiscordColor.Red, + Description = "This channel is not a ticket." + }, true); + return; } + + if (string.IsNullOrEmpty(category)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Error: No category provided." + }, true); + return; + } + + IReadOnlyList channels = await command.Guild.GetChannelsAsync(); + IEnumerable categories = channels.Where(x => x.IsCategory); + DiscordChannel categoryChannel = categories.FirstOrDefault(x => x.Name.StartsWith(category.Trim(), StringComparison.OrdinalIgnoreCase)); + + if (categoryChannel == null) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Error: Could not find a category by that name." + }, true); + return; + } + + if (command.Channel.Id == categoryChannel.Id) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Error: The ticket is already in that category." + }, true); + return; + } + + try + { + await command.Channel.ModifyAsync(modifiedAttributes => modifiedAttributes.Parent = categoryChannel); + } + catch (UnauthorizedException) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Error: Not authorized to move this ticket to that category." + }, true); + return; + } + + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Ticket was moved to " + categoryChannel.Mention + }); } } \ No newline at end of file diff --git a/SupportChild/Commands/NewCommand.cs b/SupportChild/Commands/NewCommand.cs index 1345a00..0b77184 100644 --- a/SupportChild/Commands/NewCommand.cs +++ b/SupportChild/Commands/NewCommand.cs @@ -1,150 +1,276 @@ using System; +using System.Collections.Generic; +using System.Linq; using System.Threading.Tasks; using DSharpPlus; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; using DSharpPlus.Entities; using DSharpPlus.Exceptions; -using Microsoft.Extensions.Logging; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class NewCommand : ApplicationCommandModule { - public class NewCommand : BaseCommandModule + [SlashRequireGuild] + [SlashCommand("new", "Opens a new ticket.")] + public async Task OnExecute(InteractionContext command) { - [Command("new")] - [Cooldown(1, 5, CooldownBucketType.User)] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) + List verifiedCategories = await Utilities.GetVerifiedChannels(); + switch (verifiedCategories.Count) { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "new")) - { - DiscordEmbed error = new DiscordEmbedBuilder + case 0: + await command.CreateResponseAsync(new DiscordEmbedBuilder { Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the new command but did not have permission."); + Description = "Error: No registered categories found." + }, true); return; - } + case 1: + await command.DeferAsync(true); + (bool success, string message) = await OpenNewTicket(command.User.Id, command.Channel.Id, verifiedCategories[0].id); - // Check if user is blacklisted - if (Database.IsBlacklisted(command.User.Id)) - { - DiscordEmbed error = new DiscordEmbedBuilder + if (success) { - Color = DiscordColor.Red, - Description = "You are banned from opening tickets." - }; - await command.RespondAsync(error); - return; - } - - if (Database.IsOpenTicket(command.Channel.Id)) - { - DiscordEmbed error = new DiscordEmbedBuilder + await command.FollowUpAsync(new DiscordFollowupMessageBuilder().AddEmbed(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = message + }).AsEphemeral()); + } + else { - Color = DiscordColor.Red, - Description = "You cannot use this command in a ticket channel." - }; - await command.RespondAsync(error); + await command.FollowUpAsync(new DiscordFollowupMessageBuilder().AddEmbed(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = message + }).AsEphemeral()); + } return; - } - - DiscordChannel category = command.Guild.GetChannel(Config.ticketCategory); - DiscordChannel ticketChannel; - - try - { - ticketChannel = await command.Guild.CreateChannelAsync("ticket", ChannelType.Text, category); - } - catch (Exception) - { - DiscordEmbed error = new DiscordEmbedBuilder + default: + if (Config.newCommandUsesSelector) { - Color = DiscordColor.Red, - Description = "Error occured while creating ticket, " + command.Member.Mention + - "!\nIs the channel limit reached in the server or ticket category?" - }; - await command.RespondAsync(error); - return; - } - - if (ticketChannel == null) - { - DiscordEmbed error = new DiscordEmbedBuilder + await CreateSelector(command, verifiedCategories); + } + else { - Color = DiscordColor.Red, - Description = "Error occured while creating ticket, " + command.Member.Mention + - "!\nIs the channel limit reached in the server or ticket category?" - }; - await command.RespondAsync(error); + await CreateButtons(command, verifiedCategories); + } return; - } + } + } - ulong staffID = 0; - if (Config.randomAssignment) + public static async Task CreateButtons(InteractionContext command, List verifiedCategories) + { + DiscordInteractionResponseBuilder builder = new DiscordInteractionResponseBuilder().WithContent(" "); + int nrOfButtons = 0; + for (int nrOfButtonRows = 0; nrOfButtonRows < 5 && nrOfButtons < verifiedCategories.Count; nrOfButtonRows++) + { + List buttonRow = new List(); + + for (; nrOfButtons < 5 * (nrOfButtonRows + 1) && nrOfButtons < verifiedCategories.Count; nrOfButtons++) { - staffID = Database.GetRandomActiveStaff(0)?.userID ?? 0; + buttonRow.Add(new DiscordButtonComponent(ButtonStyle.Primary, "supportboi_newcommandbutton " + verifiedCategories[nrOfButtons].id, verifiedCategories[nrOfButtons].name)); } + builder.AddComponents(buttonRow); + } - long id = Database.NewTicket(command.Member.Id, staffID, ticketChannel.Id); - string ticketID = id.ToString("00000"); + await command.CreateResponseAsync(builder.AsEphemeral()); + } + + public static async Task CreateSelector(InteractionContext command, List verifiedCategories) + { + verifiedCategories = verifiedCategories.OrderBy(x => x.name).ToList(); + List selectionComponents = new List(); + int selectionOptions = 0; + for (int selectionBoxes = 0; selectionBoxes < 5 && selectionOptions < verifiedCategories.Count; selectionBoxes++) + { + List categoryOptions = new List(); + + for (; selectionOptions < 25 * (selectionBoxes + 1) && selectionOptions < verifiedCategories.Count; selectionOptions++) + { + categoryOptions.Add(new DiscordSelectComponentOption(verifiedCategories[selectionOptions].name, verifiedCategories[selectionOptions].id.ToString())); + } + selectionComponents.Add(new DiscordSelectComponent("supportboi_newcommandselector" + selectionBoxes, "Open new ticket...", categoryOptions, false, 0, 1)); + } + + await command.CreateResponseAsync(new DiscordInteractionResponseBuilder().AddComponents(selectionComponents).AsEphemeral()); + } + + public static async Task OnCategorySelection(DiscordInteraction interaction) + { + string stringID; + switch (interaction.Data.ComponentType) + { + case ComponentType.Button: + stringID = interaction.Data.CustomId.Replace("supportboi_newcommandbutton ", ""); + break; + case ComponentType.Select: + if (interaction.Data.Values == null || interaction.Data.Values.Length <= 0) return; + stringID = interaction.Data.Values[0]; + break; + + case ComponentType.ActionRow: + case ComponentType.FormInput: + default: + return; + } + + if (!ulong.TryParse(stringID, out ulong categoryID) || categoryID == 0) return; + + await interaction.CreateResponseAsync(InteractionResponseType.DeferredMessageUpdate, new DiscordInteractionResponseBuilder().AsEphemeral()); + + (bool success, string message) = await OpenNewTicket(interaction.User.Id, interaction.ChannelId, categoryID); + + if (success) + { + await interaction.EditOriginalResponseAsync(new DiscordWebhookBuilder().AddEmbed(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = message + })); + } + else + { + await interaction.EditOriginalResponseAsync(new DiscordWebhookBuilder().AddEmbed(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = message + })); + } + } + + public static async Task<(bool, string)> OpenNewTicket(ulong userID, ulong commandChannelID, ulong categoryID) + { + // Check if user is blacklisted + if (Database.IsBlacklisted(userID)) + { + return (false, "You are banned from opening tickets."); + } + + if (Database.IsOpenTicket(commandChannelID)) + { + return (false, "You cannot use this command in a ticket channel."); + } + + if (!Database.IsStaff(userID) && Database.TryGetOpenTickets(userID, out List ownTickets) && ownTickets.Count >= Config.ticketLimit) + { + return (false, "You have reached the limit for maximum open tickets."); + } + + DiscordChannel category = null; + try + { + category = await SupportChild.discordClient.GetChannelAsync(categoryID); + } + catch (Exception) { /*ignored*/ } + + if (category == null) + { + return (false, "Error: Could not find the category to place the ticket in."); + } + + DiscordMember member = null; + try + { + member = await category.Guild.GetMemberAsync(userID); + } + catch (Exception) { /*ignored*/ } + + if (member == null) + { + return (false, "Error: Could not find you on the Discord server."); + } + + DiscordChannel ticketChannel = null; + + try + { + ticketChannel = await category.Guild.CreateChannelAsync("ticket", ChannelType.Text, category); + } + catch (Exception) { /* ignored */ } + + if (ticketChannel == null) + { + return (false, "Error occured while creating ticket, " + member.Mention + + "!\nIs the channel limit reached in the server or ticket category?"); + } + + ulong staffID = 0; + if (Config.randomAssignment) + { + staffID = Database.GetRandomActiveStaff(0)?.userID ?? 0; + } + + long id = Database.NewTicket(member.Id, staffID, ticketChannel.Id); + string ticketID = id.ToString("00000"); + + try + { await ticketChannel.ModifyAsync(modifiedAttributes => modifiedAttributes.Name = "ticket-" + ticketID); - await ticketChannel.AddOverwriteAsync(command.Member, Permissions.AccessChannels, Permissions.None); + } + catch (DiscordException e) + { + Logger.Error("Exception occurred trying to modify channel: " + e); + Logger.Error("JsomMessage: " + e.JsonMessage); + } - await ticketChannel.SendMessageAsync("Hello, " + command.Member.Mention + "!\n" + Config.welcomeMessage); + try + { + await ticketChannel.AddOverwriteAsync(member, Permissions.AccessChannels); + } + catch (DiscordException e) + { + Logger.Error("Exception occurred trying to add channel permissions: " + e); + Logger.Error("JsomMessage: " + e.JsonMessage); + } - // Refreshes the channel as changes were made to it above - ticketChannel = command.Guild.GetChannel(ticketChannel.Id); + await ticketChannel.SendMessageAsync("Hello, " + member.Mention + "!\n" + Config.welcomeMessage); - if (staffID != 0) + // Refreshes the channel as changes were made to it above + ticketChannel = await SupportChild.discordClient.GetChannelAsync(ticketChannel.Id); + + if (staffID != 0) + { + await ticketChannel.SendMessageAsync(new DiscordEmbedBuilder { - DiscordEmbed assignmentMessage = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Ticket was randomly assigned to <@" + staffID + ">." - }; - await ticketChannel.SendMessageAsync(assignmentMessage); + Color = DiscordColor.Green, + Description = "Ticket was randomly assigned to <@" + staffID + ">." + }); - if (Config.assignmentNotifications) + if (Config.assignmentNotifications) + { + try { - DiscordEmbed message = new DiscordEmbedBuilder + DiscordMember staffMember = await category.Guild.GetMemberAsync(staffID); + await staffMember.SendMessageAsync(new DiscordEmbedBuilder { Color = DiscordColor.Green, Description = "You have been randomly assigned to a newly opened support ticket: " + - ticketChannel.Mention - }; - - try - { - DiscordMember staffMember = await command.Guild.GetMemberAsync(staffID); - await staffMember.SendMessageAsync(message); - } - catch (NotFoundException) {} - catch (UnauthorizedException) {} + ticketChannel.Mention + }); + } + catch (DiscordException e) + { + Logger.Error("Exception occurred assign random staff member: " + e); + Logger.Error("JsomMessage: " + e.JsonMessage); } } + } - DiscordEmbed response = new DiscordEmbedBuilder + // Log it if the log channel exists + DiscordChannel logChannel = category.Guild.GetChannel(Config.logChannel); + if (logChannel != null) + { + DiscordEmbed logMessage = new DiscordEmbedBuilder { Color = DiscordColor.Green, - Description = "Ticket opened, " + command.Member.Mention + "!\n" + ticketChannel.Mention + Description = "Ticket " + ticketChannel.Mention + " opened by " + member.Mention + ".\n", + Footer = new DiscordEmbedBuilder.EmbedFooter { Text = "Ticket " + ticketID } }; - await command.RespondAsync(response); - - // Log it if the log channel exists - DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); - if (logChannel != null) - { - DiscordEmbed logMessage = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Ticket " + ticketChannel.Mention + " opened by " + command.Member.Mention + ".\n", - Footer = new DiscordEmbedBuilder.EmbedFooter {Text = "Ticket " + ticketID} - }; - await logChannel.SendMessageAsync(logMessage); - } + await logChannel.SendMessageAsync(logMessage); } + + return (true, "Ticket opened, " + member.Mention + "!\n" + ticketChannel.Mention); } } \ No newline at end of file diff --git a/SupportChild/Commands/RandomAssignCommand.cs b/SupportChild/Commands/RandomAssignCommand.cs index 7b2a3e3..36d633d 100644 --- a/SupportChild/Commands/RandomAssignCommand.cs +++ b/SupportChild/Commands/RandomAssignCommand.cs @@ -2,181 +2,138 @@ using System.Collections.Generic; using System.Linq; using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; using DSharpPlus.Entities; using DSharpPlus.Exceptions; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; using Microsoft.Extensions.Logging; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class RandomAssignCommand : ApplicationCommandModule { - public class RandomAssignCommand : BaseCommandModule + [SlashRequireGuild] + [SlashCommand("rassign", "Randomly assigns a staff member to a ticket.")] + public async Task OnExecute(InteractionContext command, [Option("Role", "(Optional) Limit the random assignment to a specific role.")] DiscordRole role = null) { - [Command("rassign")] - [Description("Randomly assigns a staff member to a ticket.")] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArguments) + // Check if ticket exists in the database + if (!Database.TryGetOpenTicket(command.Channel.Id, out Database.Ticket ticket)) { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "rassign")) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the rassign command but did not have permission."); - return; - } - - // Check if ticket exists in the database - if (!Database.TryGetOpenTicket(command.Channel.Id, out Database.Ticket ticket)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "This channel is not a ticket." - }; - await command.RespondAsync(error); - return; - } - - // Get a random staff member who is verified to have the correct role if applicable - DiscordMember staffMember = await GetRandomVerifiedStaffMember(command, ticket); - if (staffMember == null) - { - return; - } - - // Attempt to assign the staff member to the ticket - if (!Database.AssignStaff(ticket, staffMember.Id)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Error: Failed to assign " + staffMember.Mention + " to ticket." - }; - await command.RespondAsync(error); - return; - } - - // Respond that the command was successful - DiscordEmbed feedback = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Randomly assigned " + staffMember.Mention + " to ticket." - }; - await command.RespondAsync(feedback); - - // Send a notification to the staff member if applicable - if (Config.assignmentNotifications) - { - try - { - DiscordEmbed message = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "You have been randomly assigned to a support ticket: " + command.Channel.Mention - }; - await staffMember.SendMessageAsync(message); - } - catch (UnauthorizedException) {} - } - - // Log it if the log channel exists - DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); - if (logChannel != null) - { - DiscordEmbed logMessage = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = staffMember.Mention + " was assigned to " + command.Channel.Mention + " by " + command.Member.Mention + "." - }; - await logChannel.SendMessageAsync(logMessage); - } - } - - private async Task GetRandomVerifiedStaffMember(CommandContext command, Database.Ticket ticket) - { - if (command.RawArguments.Any()) // An argument was provided, check if this can be parsed into a role - { - ulong roleID = 0; - - // Try to parse either discord mention or ID - string[] parsedMessage = Utilities.ParseIDs(command.RawArgumentString); - if (!ulong.TryParse(parsedMessage[0], out roleID)) - { - // Try to find role by name - roleID = Utilities.GetRoleByName(command.Guild, command.RawArgumentString)?.Id ?? 0; - } - - // Check if a role was found - if (roleID == 0) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Could not find a role by that name/ID." - }; - await command.RespondAsync(error); - return null; - } - - // Check if role rassign should override staff's active status - List staffMembers = Config.randomAssignRoleOverride - ? Database.GetAllStaff(ticket.assignedStaffID) - : Database.GetActiveStaff(ticket.assignedStaffID); - - // Randomize the list before checking for roles in order to reduce number of API calls - staffMembers = Utilities.RandomizeList(staffMembers); - - // Get the first staff member that has the role - foreach (Database.StaffMember sm in staffMembers) - { - try - { - DiscordMember verifiedMember = await command.Guild.GetMemberAsync(sm.userID); - if (verifiedMember?.Roles?.Any(role => role.Id == roleID) ?? false) - { - return verifiedMember; - } - } - catch (Exception e) - { - command.Client.Logger.Log(LogLevel.Information, e, "Error occured trying to find a staff member in the rassign command."); - } - } - } - else // No role was specified, any active staff will be picked - { - Database.StaffMember staffEntry = Database.GetRandomActiveStaff(ticket.assignedStaffID); - if (staffEntry == null) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Error: There are no other staff members to choose from." - }; - await command.RespondAsync(error); - return null; - } - - // Get the staff member from discord - try - { - return await command.Guild.GetMemberAsync(staffEntry.userID); - } - catch (NotFoundException) { } - } - - // Send a more generic error if we get to this point and still haven't found the staff member - DiscordEmbed err = new DiscordEmbedBuilder + await command.CreateResponseAsync(new DiscordEmbedBuilder { Color = DiscordColor.Red, - Description = "Error: Could not find an applicable staff member." - }; - await command.RespondAsync(err); - return null; + Description = "Error: This channel is not a ticket." + }, true); + return; + } + + // Get a random staff member who is verified to have the correct role if applicable + DiscordMember staffMember = await GetRandomVerifiedStaffMember(command, role, ticket); + if (staffMember == null) + { + return; + } + + // Attempt to assign the staff member to the ticket + if (!Database.AssignStaff(ticket, staffMember.Id)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Error: Failed to assign " + staffMember.Mention + " to ticket." + }, true); + return; + } + + // Respond that the command was successful + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Randomly assigned " + staffMember.Mention + " to ticket." + }); + + // Send a notification to the staff member if applicable + if (Config.assignmentNotifications) + { + try + { + await staffMember.SendMessageAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "You have been randomly assigned to a support ticket: " + command.Channel.Mention + }); + } + catch (UnauthorizedException) { } + } + + // Log it if the log channel exists + DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); + if (logChannel != null) + { + await logChannel.SendMessageAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = staffMember.Mention + " was randomly assigned to " + command.Channel.Mention + " by " + command.Member.Mention + "." + }); } } + + private static async Task GetRandomVerifiedStaffMember(InteractionContext command, DiscordRole targetRole, Database.Ticket ticket) + { + if (targetRole != null) // A role was provided + { + // Check if role rassign should override staff's active status + List staffMembers = Config.randomAssignRoleOverride + ? Database.GetAllStaff(ticket.assignedStaffID, ticket.creatorID) + : Database.GetActiveStaff(ticket.assignedStaffID, ticket.creatorID); + + // Randomize the list before checking for roles in order to reduce number of API calls + staffMembers.Shuffle(); + + // Get the first staff member that has the role + foreach (Database.StaffMember sm in staffMembers) + { + try + { + DiscordMember verifiedMember = await command.Guild.GetMemberAsync(sm.userID); + if (verifiedMember?.Roles?.Any(role => role.Id == targetRole.Id) ?? false) + { + return verifiedMember; + } + } + catch (Exception e) + { + command.Client.Logger.Log(LogLevel.Information, e, "Error occured trying to find a staff member in the rassign command."); + } + } + } + else // No role was specified, any active staff will be picked + { + Database.StaffMember staffEntry = Database.GetRandomActiveStaff(ticket.assignedStaffID, ticket.creatorID); + if (staffEntry == null) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Error: There are no other staff members to choose from." + }, true); + return null; + } + + // Get the staff member from discord + try + { + return await command.Guild.GetMemberAsync(staffEntry.userID); + } + catch (NotFoundException) { } + } + + // Send a more generic error if we get to this point and still haven't found the staff member + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Error: Could not find an applicable staff member." + }, true); + return null; + } } \ No newline at end of file diff --git a/SupportChild/Commands/ReloadCommand.cs b/SupportChild/Commands/ReloadCommand.cs deleted file mode 100644 index de426c0..0000000 --- a/SupportChild/Commands/ReloadCommand.cs +++ /dev/null @@ -1,38 +0,0 @@ -using System; -using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; -using DSharpPlus.Entities; -using Microsoft.Extensions.Logging; - -namespace SupportChild.Commands -{ - public class ReloadCommand : BaseCommandModule - { - [Command("reload")] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) - { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "reload")) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the reload command but did not have permission."); - return; - } - - DiscordEmbed message = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Reloading bot application..." - }; - await command.RespondAsync(message); - Console.WriteLine("Reloading bot..."); - SupportChild.instance.Reload(); - } - } -} \ No newline at end of file diff --git a/SupportChild/Commands/RemoveCategoryCommand.cs b/SupportChild/Commands/RemoveCategoryCommand.cs new file mode 100644 index 0000000..0cef6a4 --- /dev/null +++ b/SupportChild/Commands/RemoveCategoryCommand.cs @@ -0,0 +1,41 @@ +using System.Threading.Tasks; +using DSharpPlus.Entities; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; + +namespace SupportChild.Commands; + +public class RemoveCategoryCommand : ApplicationCommandModule +{ + [SlashRequireGuild] + [SlashCommand("removecategory", "Removes the ability for users to open tickets in a specific category.")] + public async Task OnExecute(InteractionContext command, [Option("Category", "The category to remove.")] DiscordChannel channel) + { + if (!Database.TryGetCategory(channel.Id, out Database.Category _)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "That category is not registered." + }, true); + return; + } + + if (Database.RemoveCategory(channel.Id)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Category removed." + }, true); + } + else + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Error: Failed removing the category from the database." + }, true); + } + } +} \ No newline at end of file diff --git a/SupportChild/Commands/RemoveMessageCommand.cs b/SupportChild/Commands/RemoveMessageCommand.cs index 96794e4..fc14a0f 100644 --- a/SupportChild/Commands/RemoveMessageCommand.cs +++ b/SupportChild/Commands/RemoveMessageCommand.cs @@ -1,66 +1,41 @@ -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; -using DSharpPlus.Entities; -using Microsoft.Extensions.Logging; -using System; -using System.Collections.Generic; -using System.Linq; -using System.Text; +using DSharpPlus.Entities; using System.Threading.Tasks; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class RemoveMessageCommand : ApplicationCommandModule { - public class RemoveMessageCommand : BaseCommandModule - { - [Command("removemessage")] - [Description("Removes a message from the 'say' command.")] - public async Task OnExecute(CommandContext command, string identifier) - { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "removemessage")) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the removemessage command but did not have permission."); - return; - } + [SlashRequireGuild] + [SlashCommand("removemessage", "Removes a message from the 'say' command.")] + public async Task OnExecute(InteractionContext command, [Option("Identifier", "The identifier word used in the /say command.")] string identifier) + { + if (!Database.TryGetMessage(identifier.ToLower(), out Database.Message _)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "There is no message with that identifier." + }, true); + return; + } - if (!Database.TryGetMessage(identifier.ToLower(), out Database.Message _)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "There is no message with that identifier." - }; - await command.RespondAsync(error); - return; - } - - if(Database.RemoveMessage(identifier)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Message removed." - }; - await command.RespondAsync(error); - return; - } - else - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Error: Failed removing the message from the database." - }; - await command.RespondAsync(error); - return; - } - - } - } + if (Database.RemoveMessage(identifier)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Message removed." + }, true); + } + else + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Error: Failed removing the message from the database." + }, true); + } + } } \ No newline at end of file diff --git a/SupportChild/Commands/RemoveStaffCommand.cs b/SupportChild/Commands/RemoveStaffCommand.cs index a48f96f..70e2162 100644 --- a/SupportChild/Commands/RemoveStaffCommand.cs +++ b/SupportChild/Commands/RemoveStaffCommand.cs @@ -1,104 +1,49 @@ -using System; -using System.Linq; -using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; +using System.Threading.Tasks; using DSharpPlus.Entities; -using Microsoft.Extensions.Logging; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; using MySql.Data.MySqlClient; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class RemoveStaffCommand : ApplicationCommandModule { - public class RemoveStaffCommand : BaseCommandModule + [SlashRequireGuild] + [SlashCommand("removestaff", "Removes a staff member.")] + public async Task OnExecute(InteractionContext command, [Option("User", "User to remove from staff.")] DiscordUser user) { - [Command("removestaff")] - [Cooldown(1, 5, CooldownBucketType.User)] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) + if (!Database.IsStaff(user.Id)) { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "removestaff")) + await command.CreateResponseAsync(new DiscordEmbedBuilder { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the removestaff command but did not have permission."); - return; - } + Color = DiscordColor.Red, + Description = "User is already not registered as staff." + }, true); + return; + } - ulong userID; - string[] parsedMessage = Utilities.ParseIDs(command.RawArgumentString); + await using MySqlConnection c = Database.GetConnection(); + c.Open(); + MySqlCommand deletion = new MySqlCommand(@"DELETE FROM staff WHERE user_id=@user_id", c); + deletion.Parameters.AddWithValue("@user_id", user.Id); + deletion.Prepare(); + deletion.ExecuteNonQuery(); - if (!parsedMessage.Any()) + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "User was removed from staff." + }, true); + + // Log it if the log channel exists + DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); + if (logChannel != null) + { + await logChannel.SendMessageAsync(new DiscordEmbedBuilder { - userID = command.Member.Id; - } - else if (!ulong.TryParse(parsedMessage[0], out userID)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Invalid ID/Mention. (Could not convert to numerical)" - }; - await command.RespondAsync(error); - return; - } - - try - { - await command.Client.GetUserAsync(userID); - } - catch (Exception) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Invalid ID/Mention. (Could not find user on Discord)" - }; - await command.RespondAsync(error); - return; - } - - if (!Database.IsStaff(userID)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "User is already not registered as staff." - }; - await command.RespondAsync(error); - return; - } - - using (MySqlConnection c = Database.GetConnection()) - { - c.Open(); - MySqlCommand deletion = new MySqlCommand(@"DELETE FROM staff WHERE user_id=@user_id", c); - deletion.Parameters.AddWithValue("@user_id", userID); - deletion.Prepare(); - deletion.ExecuteNonQuery(); - - DiscordEmbed message = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "User was removed from staff." - }; - await command.RespondAsync(message); - - // Log it if the log channel exists - DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); - if (logChannel != null) - { - DiscordEmbed logMessage = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "User was removed from staff.\n", - }; - await logChannel.SendMessageAsync(logMessage); - } - } + Color = DiscordColor.Green, + Description = "User was removed from staff.\n" + }); } } } \ No newline at end of file diff --git a/SupportChild/Commands/SayCommand.cs b/SupportChild/Commands/SayCommand.cs index 5eaedd1..b198b9c 100644 --- a/SupportChild/Commands/SayCommand.cs +++ b/SupportChild/Commands/SayCommand.cs @@ -1,78 +1,29 @@ -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; -using DSharpPlus.Entities; -using Microsoft.Extensions.Logging; +using DSharpPlus.Entities; using System.Collections.Generic; using System.Linq; using System.Threading.Tasks; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class SayCommand : ApplicationCommandModule { - public class SayCommand : BaseCommandModule + [SlashRequireGuild] + [SlashCommand("say", "Prints a message with information from staff. Use without identifier to list all identifiers.")] + public async Task OnExecute(InteractionContext command, [Option("Identifier", "(Optional) The identifier word to summon a message.")] string identifier = null) { - [Command("say")] - [Cooldown(1, 2, CooldownBucketType.Channel)] - [Description("Prints a message with information from staff.")] - public async Task OnExecute(CommandContext command, string identifier) + // Print list of all messages if no identifier is provided + if (identifier == null) { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "say")) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the say command but did not have permission."); - return; - } - - if (!Database.TryGetMessage(identifier.ToLower(), out Database.Message message)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "There is no message with that identifier." - }; - await command.RespondAsync(error); - return; - } - - DiscordEmbed reply = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = message.message - }; - await command.RespondAsync(reply); - } - - [Command("say")] - [Cooldown(1, 2.0, CooldownBucketType.Channel)] - [Description("Prints a list of staff messages.")] - public async Task OnExecute(CommandContext command) - { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "say")) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the say command but did not have permission."); - return; - } - - List messages = Database.GetAllMessages(); if (!messages.Any()) { - DiscordEmbed error = new DiscordEmbedBuilder() - .WithColor(DiscordColor.Red) - .WithDescription("There are no messages registered."); - await command.RespondAsync(error); + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "There are no messages registered." + }, true); return; } @@ -85,12 +36,32 @@ namespace SupportChild.Commands LinkedList listMessages = Utilities.ParseListIntoMessages(listItems); foreach (string listMessage in listMessages) { - DiscordEmbed channelInfo = new DiscordEmbedBuilder() - .WithTitle("Available messages: ") - .WithColor(DiscordColor.Green) - .WithDescription(listMessage); - await command.RespondAsync(channelInfo); + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Title = "Available messages: ", + Color = DiscordColor.Green, + Description = listMessage + }, true); } } + // Print specific message + else + { + if (!Database.TryGetMessage(identifier.ToLower(), out Database.Message message)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "There is no message with that identifier." + }, true); + return; + } + + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Cyan, + Description = message.message.Replace("\\n", "\n") + }); + } } } \ No newline at end of file diff --git a/SupportChild/Commands/SetSummaryCommand.cs b/SupportChild/Commands/SetSummaryCommand.cs index 81b0ca5..6d9e87e 100644 --- a/SupportChild/Commands/SetSummaryCommand.cs +++ b/SupportChild/Commands/SetSummaryCommand.cs @@ -1,63 +1,42 @@ using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; using DSharpPlus.Entities; -using Microsoft.Extensions.Logging; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; using MySql.Data.MySqlClient; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class SetSummaryCommand : ApplicationCommandModule { - public class SetSummaryCommand : BaseCommandModule - { - [Command("setsummary")] - [Cooldown(1, 5, CooldownBucketType.User)] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) - { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "setsummary")) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the setsummary command but did not have permission."); - return; - } + [SlashRequireGuild] + [SlashCommand("setsummary", "Sets a ticket's summary for the summary command.")] + public async Task OnExecute(InteractionContext command, [Option("Summary", "The ticket summary text.")] string summary) + { + ulong channelID = command.Channel.Id; + // Check if ticket exists in the database + if (!Database.TryGetOpenTicket(command.Channel.Id, out Database.Ticket _)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "This channel is not a ticket." + }); + return; + } - ulong channelID = command.Channel.Id; - // Check if ticket exists in the database - if (!Database.TryGetOpenTicket(command.Channel.Id, out Database.Ticket ticket)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "This channel is not a ticket." - }; - await command.RespondAsync(error); - return; - } + await using MySqlConnection c = Database.GetConnection(); + c.Open(); + MySqlCommand update = new MySqlCommand(@"UPDATE tickets SET summary = @summary WHERE channel_id = @channel_id", c); + update.Parameters.AddWithValue("@summary", summary); + update.Parameters.AddWithValue("@channel_id", channelID); + update.Prepare(); + update.ExecuteNonQuery(); + update.Dispose(); - string summary = command.Message.Content.Replace(Config.prefix + "setsummary", "").Trim(); - - using (MySqlConnection c = Database.GetConnection()) - { - c.Open(); - MySqlCommand update = new MySqlCommand(@"UPDATE tickets SET summary = @summary WHERE channel_id = @channel_id", c); - update.Parameters.AddWithValue("@summary", summary); - update.Parameters.AddWithValue("@channel_id", channelID); - update.Prepare(); - update.ExecuteNonQuery(); - update.Dispose(); - - DiscordEmbed message = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Summary set." - }; - await command.RespondAsync(message); - } - } - } + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Summary set." + }, true); + } } \ No newline at end of file diff --git a/SupportChild/Commands/SetTicketCommand.cs b/SupportChild/Commands/SetTicketCommand.cs deleted file mode 100644 index 11797ec..0000000 --- a/SupportChild/Commands/SetTicketCommand.cs +++ /dev/null @@ -1,98 +0,0 @@ -using System.Linq; -using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; -using DSharpPlus.Entities; -using Microsoft.Extensions.Logging; -using MySql.Data.MySqlClient; - -namespace SupportChild.Commands -{ - public class SetTicketCommand :BaseCommandModule - { - [Command("setticket")] - [Description("Turns a channel into a ticket, warning: this will let anyone with write access delete the channel using the close command.")] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) - { - using (MySqlConnection c = Database.GetConnection()) - { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "setticket")) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the setticket command but did not have permission."); - return; - } - - // Check if ticket exists in the database - if (Database.IsOpenTicket(command.Channel.Id)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "This channel is already a ticket." - }; - await command.RespondAsync(error); - return; - } - - ulong userID; - string[] parsedMessage = Utilities.ParseIDs(command.RawArgumentString); - - if (!parsedMessage.Any()) - { - userID = command.Member.Id; - } - else if (!ulong.TryParse(parsedMessage[0], out userID)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Invalid ID/Mention. (Could not convert to numerical)" - }; - await command.RespondAsync(error); - return; - } - - DiscordUser user = await command.Client.GetUserAsync(userID); - - if (user == null) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Invalid ID/Mention." - }; - await command.RespondAsync(error); - return; - } - - long id = Database.NewTicket(userID, 0, command.Channel.Id); - string ticketID = id.ToString("00000"); - DiscordEmbed message = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Channel has been designated ticket " + ticketID + "." - }; - await command.RespondAsync(message); - - // Log it if the log channel exists - DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); - if (logChannel != null) - { - DiscordEmbed logMessage = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = command.Channel.Mention + " has been designated ticket " + ticketID + " by " + command.Member.Mention + "." - }; - await logChannel.SendMessageAsync(logMessage); - } - } - } - } -} \ No newline at end of file diff --git a/SupportChild/Commands/StatusCommand.cs b/SupportChild/Commands/StatusCommand.cs index 8521b80..dec688b 100644 --- a/SupportChild/Commands/StatusCommand.cs +++ b/SupportChild/Commands/StatusCommand.cs @@ -1,41 +1,26 @@ using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; using DSharpPlus.Entities; -using Microsoft.Extensions.Logging; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class StatusCommand : ApplicationCommandModule { - public class StatusCommand : BaseCommandModule - { - [Command("status")] - [Cooldown(1, 5, CooldownBucketType.User)] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) - { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "status")) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the status command but did not have permission."); - return; - } + [SlashRequireGuild] + [SlashCommand("status", "Shows bot status and information.")] + public async Task OnExecute(InteractionContext command) + { + long openTickets = Database.GetNumberOfTickets(); + long closedTickets = Database.GetNumberOfClosedTickets(); - long openTickets = Database.GetNumberOfTickets(); - long closedTickets = Database.GetNumberOfClosedTickets(); - - DiscordEmbed botInfo = new DiscordEmbedBuilder() - .WithAuthor("EmotionChild/SupportChild @ GitHub", "https://github.com/EmotionChild/SupportChild", "https://cdn.emotionchild.com/Ellie.png") - .WithTitle("Bot information") - .WithColor(DiscordColor.Cyan) - .AddField("Version:", SupportChild.GetVersion(), false) - .AddField("Open tickets:", openTickets + "", true) - .AddField("Closed tickets (1.1.0+ tickets only):", closedTickets + " ", true); - await command.RespondAsync(botInfo); - } - } + DiscordEmbed botInfo = new DiscordEmbedBuilder() + .WithAuthor("KarlofDuty/SupportBoi @ GitHub", "https://github.com/EmotionChild/SupportChild", "https://cdn.discordapp.com/attachments/765441543100170271/914327948667011132/Ellie_Concept_2_transparent_ver.png") + .WithTitle("Bot information") + .WithColor(DiscordColor.Cyan) + .AddField("Version:", SupportChild.GetVersion()) + .AddField("Open tickets:", openTickets + "", true) + .AddField("Closed tickets:", closedTickets + " ", true); + await command.CreateResponseAsync(botInfo); + } } \ No newline at end of file diff --git a/SupportChild/Commands/SummaryCommand.cs b/SupportChild/Commands/SummaryCommand.cs index 360db00..69a0a3b 100644 --- a/SupportChild/Commands/SummaryCommand.cs +++ b/SupportChild/Commands/SummaryCommand.cs @@ -1,51 +1,35 @@ using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; using DSharpPlus.Entities; -using Microsoft.Extensions.Logging; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class SummaryCommand : ApplicationCommandModule { - public class SummaryCommand : BaseCommandModule - { - [Command("summary")] - [Cooldown(1, 5, CooldownBucketType.User)] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) - { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "summary")) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the summary command but did not have permission."); - return; - } - - if (Database.TryGetOpenTicket(command.Channel.Id, out Database.Ticket ticket)) - { - DiscordEmbed channelInfo = new DiscordEmbedBuilder() - .WithTitle("Channel information") - .WithColor(DiscordColor.Cyan) - .AddField("Ticket number:", ticket.id.ToString(), true) - .AddField("Ticket creator:", $"<@{ticket.creatorID}>", true) - .AddField("Assigned staff:", ticket.assignedStaffID == 0 ? "Unassigned." : $"<@{ticket.assignedStaffID}>", true) - .AddField("Creation time:", ticket.createdTime.ToString(Config.timestampFormat), true) - .AddField("Summary:", string.IsNullOrEmpty(ticket.summary) ? "No summary." : ticket.summary, false); - await command.RespondAsync(channelInfo); - } - else - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "This channel is not a ticket." - }; - await command.RespondAsync(error); - } - } - } + [SlashRequireGuild] + [SlashCommand("summary", "Lists tickets assigned to a user.")] + public async Task OnExecute(InteractionContext command) + { + if (Database.TryGetOpenTicket(command.Channel.Id, out Database.Ticket ticket)) + { + DiscordEmbed channelInfo = new DiscordEmbedBuilder() + .WithTitle("Channel information") + .WithColor(DiscordColor.Cyan) + .AddField("Ticket number:", ticket.id.ToString("00000"), true) + .AddField("Ticket creator:", $"<@{ticket.creatorID}>", true) + .AddField("Assigned staff:", ticket.assignedStaffID == 0 ? "Unassigned." : $"<@{ticket.assignedStaffID}>", true) + .AddField("Creation time:", ticket.DiscordRelativeTime(), true) + .AddField("Summary:", string.IsNullOrEmpty(ticket.summary) ? "No summary." : ticket.summary.Replace("\\n", "\n")); + await command.CreateResponseAsync(channelInfo); + } + else + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "This channel is not a ticket." + }, true); + } + } } \ No newline at end of file diff --git a/SupportChild/Commands/ToggleActiveCommand.cs b/SupportChild/Commands/ToggleActiveCommand.cs index b953367..1e07cfa 100644 --- a/SupportChild/Commands/ToggleActiveCommand.cs +++ b/SupportChild/Commands/ToggleActiveCommand.cs @@ -1,78 +1,44 @@ -using System.Linq; -using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; +using System.Threading.Tasks; using DSharpPlus.Entities; -using Microsoft.Extensions.Logging; -using MySql.Data.MySqlClient; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class ToggleActiveCommand : ApplicationCommandModule { - public class ToggleActiveCommand : BaseCommandModule + [SlashRequireGuild] + [SlashCommand("toggleactive", "Toggles active status for a staff member.")] + public async Task OnExecute(InteractionContext command, [Option("User", "(Optional) Staff member to toggle activity for.")] DiscordUser user = null) { - [Command("toggleactive")] - [Aliases("ta")] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) + DiscordUser staffUser = user == null ? command.User : user; + + // Check if ticket exists in the database + if (!Database.TryGetStaff(staffUser.Id, out Database.StaffMember staffMember)) { - using (MySqlConnection c = Database.GetConnection()) + await command.CreateResponseAsync(new DiscordEmbedBuilder { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "toggleactive")) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the toggleactive command but did not have permission."); - return; - } + Color = DiscordColor.Red, + Description = user == null ? "You have not been registered as staff." : "The user is not registered as staff." + }, true); + return; + } - ulong staffID; - string[] parsedMessage = Utilities.ParseIDs(command.RawArgumentString); - - if (!parsedMessage.Any()) - { - staffID = command.Member.Id; - } - else if (!ulong.TryParse(parsedMessage[0], out staffID)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Invalid ID/Mention. (Could not convert to numerical)" - }; - await command.RespondAsync(error); - return; - } - - // Check if ticket exists in the database - if (!Database.TryGetStaff(staffID, out Database.StaffMember staffMember)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You have not been registered as staff." - }; - await command.RespondAsync(error); - return; - } - - c.Open(); - MySqlCommand update = new MySqlCommand(@"UPDATE staff SET active = @active WHERE user_id = @user_id", c); - update.Parameters.AddWithValue("@user_id", staffID); - update.Parameters.AddWithValue("@active", !staffMember.active); - update.Prepare(); - update.ExecuteNonQuery(); - - DiscordEmbed message = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = staffMember.active ? "Staff member is now set as inactive and will no longer be randomly assigned any support tickets." : "Staff member is now set as active and will be randomly assigned support tickets again." - }; - await command.RespondAsync(message); - } + if (Database.SetStaffActive(staffUser.Id, !staffMember.active)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = staffMember.active ? "Staff member is now set as inactive and will no longer be randomly assigned any support tickets." : "Staff member is now set as active and will be randomly assigned support tickets again." + }, true); + } + else + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Error: Unable to update active status in database." + }, true); } } } \ No newline at end of file diff --git a/SupportChild/Commands/TranscriptCommand.cs b/SupportChild/Commands/TranscriptCommand.cs index 8fb24cd..0eb4bfc 100644 --- a/SupportChild/Commands/TranscriptCommand.cs +++ b/SupportChild/Commands/TranscriptCommand.cs @@ -2,171 +2,128 @@ using System.Collections.Generic; using System.IO; using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; using DSharpPlus.Entities; using DSharpPlus.Exceptions; -using Microsoft.Extensions.Logging; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class TranscriptCommand : ApplicationCommandModule { - public class TranscriptCommand : BaseCommandModule + [SlashRequireGuild] + [SlashCommand("transcript", "Creates a transcript of a ticket.")] + public async Task OnExecute(InteractionContext command, [Option("Ticket", "(Optional) Ticket number to get transcript of.")] long ticketID = 0) { - [Command("transcript")] - [Cooldown(1, 5, CooldownBucketType.User)] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) + await command.DeferAsync(true); + Database.Ticket ticket; + if (ticketID == 0) // If there are no arguments use current channel { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "transcript")) + if (Database.TryGetOpenTicket(command.Channel.Id, out ticket)) { - DiscordEmbed error = new DiscordEmbedBuilder + try + { + await Transcriber.ExecuteAsync(command.Channel.Id, ticket.id); + } + catch (Exception) + { + await command.EditResponseAsync(new DiscordWebhookBuilder().AddEmbed(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "ERROR: Could not save transcript file. Aborting..." + })); + throw; + } + } + else + { + await command.EditResponseAsync(new DiscordWebhookBuilder().AddEmbed(new DiscordEmbedBuilder { Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the transcript command but did not have permission."); + Description = "This channel is not a ticket." + })); return; } - - Database.Ticket ticket; - string strippedMessage = command.Message.Content.Replace(Config.prefix, ""); - string[] parsedMessage = strippedMessage.Replace("<@!", "").Replace("<@", "").Replace(">", "").Split(); - - // If there are no arguments use current channel - if (parsedMessage.Length < 2) + } + else + { + // If the ticket is still open, generate a new fresh transcript + if (Database.TryGetOpenTicketByID((uint)ticketID, out ticket) && ticket?.creatorID == command.Member.Id) { - if (Database.TryGetOpenTicket(command.Channel.Id, out ticket)) + try { - try - { - await Transcriber.ExecuteAsync(command.Channel.Id, ticket.id); - } - catch (Exception) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "ERROR: Could not save transcript file. Aborting..." - }; - await command.RespondAsync(error); - throw; - } + await Transcriber.ExecuteAsync(command.Channel.Id, ticket.id); } - else + catch (Exception) { - DiscordEmbed error = new DiscordEmbedBuilder + await command.EditResponseAsync(new DiscordWebhookBuilder().AddEmbed(new DiscordEmbedBuilder { Color = DiscordColor.Red, - Description = "This channel is not a ticket." - }; - await command.RespondAsync(error); - return; + Description = "ERROR: Could not save transcript file. Aborting..." + })); + throw; } + } - else + // If there is no open or closed ticket, send an error. If there is a closed ticket we will simply use the old transcript from when the ticket was closed. + else if (!Database.TryGetClosedTicket((uint)ticketID, out ticket) || (ticket?.creatorID != command.Member.Id && !Database.IsStaff(command.Member.Id))) { - // Check if argument is numerical, if not abort - if (!uint.TryParse(parsedMessage[1], out uint ticketID)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Argument must be a number." - }; - await command.RespondAsync(error); - return; - } - - // If the ticket is still open, generate a new fresh transcript - if (Database.TryGetOpenTicketByID(ticketID, out ticket) && ticket?.creatorID == command.Member.Id) - { - try - { - await Transcriber.ExecuteAsync(command.Channel.Id, ticket.id); - } - catch (Exception) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "ERROR: Could not save transcript file. Aborting..." - }; - await command.RespondAsync(error); - throw; - } - - } - // If there is no open or closed ticket, send an error. If there is a closed ticket we will simply use the old transcript from when the ticket was closed. - else if (!Database.TryGetClosedTicket(ticketID, out ticket) || (ticket?.creatorID != command.Member.Id && !Database.IsStaff(command.Member.Id))) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Could not find a closed ticket with that number which you opened." + (Config.HasPermission(command.Member, "list") ? "\n(Use the " + Config.prefix + "list command to see all your tickets)" : "") - }; - await command.RespondAsync(error); - return; - } - } - - // Log it if the log channel exists - DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); - if (logChannel != null) - { - DiscordEmbed embed = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Ticket " + ticket.id.ToString("00000") + " transcript generated by " + command.Member.Mention + ".\n", - Footer = new DiscordEmbedBuilder.EmbedFooter { Text = '#' + command.Channel.Name } - }; - - using (FileStream file = new FileStream(Transcriber.GetPath(ticket.id), FileMode.Open, FileAccess.Read)) - { - DiscordMessageBuilder message = new DiscordMessageBuilder(); - message.WithEmbed(embed); - message.WithFiles(new Dictionary() { { Transcriber.GetFilename(ticket.id), file } }); - - await logChannel.SendMessageAsync(message); - } - } - - try - { - // Send transcript privately - DiscordEmbed directMessageEmbed = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Transcript generated, " + command.Member.Mention + "!\n" - }; - - using (FileStream file = new FileStream(Transcriber.GetPath(ticket.id), FileMode.Open, FileAccess.Read)) - { - DiscordMessageBuilder directMessage = new DiscordMessageBuilder(); - directMessage.WithEmbed(directMessageEmbed); - directMessage.WithFiles(new Dictionary() { { Transcriber.GetFilename(ticket.id), file } }); - - await command.Member.SendMessageAsync(directMessage); - } - - // Respond to message directly - DiscordEmbed response = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Transcript sent, " + command.Member.Mention + "!\n" - }; - await command.RespondAsync(response); - } - catch (UnauthorizedException) - { - // Send transcript privately - DiscordEmbed error = new DiscordEmbedBuilder + await command.EditResponseAsync(new DiscordWebhookBuilder().AddEmbed(new DiscordEmbedBuilder { Color = DiscordColor.Red, - Description = "Not allowed to send direct message to you, " + command.Member.Mention + ", please check your privacy settings.\n" - }; - await command.RespondAsync(error); + Description = "Could not find a closed ticket with that number which you opened.\n(Use the /list command to see all your tickets)" + })); + return; } } + + // Log it if the log channel exists + DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); + if (logChannel != null) + { + await using FileStream file = new FileStream(Transcriber.GetPath(ticket.id), FileMode.Open, FileAccess.Read); + + DiscordMessageBuilder message = new DiscordMessageBuilder(); + message.WithEmbed(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Ticket " + ticket.id.ToString("00000") + " transcript generated by " + command.Member.Mention + ".\n", + Footer = new DiscordEmbedBuilder.EmbedFooter { Text = '#' + command.Channel.Name } + }); + message.WithFiles(new Dictionary { { Transcriber.GetFilename(ticket.id), file } }); + + await logChannel.SendMessageAsync(message); + } + + try + { + // Send transcript in a direct message + await using FileStream file = new FileStream(Transcriber.GetPath(ticket.id), FileMode.Open, FileAccess.Read); + + DiscordMessageBuilder directMessage = new DiscordMessageBuilder(); + directMessage.WithEmbed(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Transcript generated!\n" + }); + directMessage.WithFiles(new Dictionary { { Transcriber.GetFilename(ticket.id), file } }); + + await command.Member.SendMessageAsync(directMessage); + } + catch (UnauthorizedException) + { + await command.EditResponseAsync(new DiscordWebhookBuilder().AddEmbed(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Not allowed to send direct message to you, please check your privacy settings.\n" + })); + return; + } + + await command.EditResponseAsync(new DiscordWebhookBuilder().AddEmbed(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Transcript sent!\n" + })); } } \ No newline at end of file diff --git a/SupportChild/Commands/UnassignComand.cs b/SupportChild/Commands/UnassignComand.cs index 5b892a8..42e272c 100644 --- a/SupportChild/Commands/UnassignComand.cs +++ b/SupportChild/Commands/UnassignComand.cs @@ -1,71 +1,52 @@ using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; using DSharpPlus.Entities; -using Microsoft.Extensions.Logging; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class UnassignCommand : ApplicationCommandModule { - public class UnassignCommand : BaseCommandModule + [SlashRequireGuild] + [SlashCommand("unassign", "Unassigns a staff member from a ticket.")] + public async Task OnExecute(InteractionContext command) { - [Command("unassign")] - [Description("Unassigns a staff member from a ticket.")] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) + // Check if ticket exists in the database + if (!Database.TryGetOpenTicket(command.Channel.Id, out Database.Ticket ticket)) { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "unassign")) + await command.CreateResponseAsync(new DiscordEmbedBuilder { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the unassign command but did not have permission."); - return; - } + Color = DiscordColor.Red, + Description = "This channel is not a ticket." + }, true); + return; + } - // Check if ticket exists in the database - if (!Database.TryGetOpenTicket(command.Channel.Id, out Database.Ticket ticket)) + if (!Database.UnassignStaff(ticket)) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "This channel is not a ticket." - }; - await command.RespondAsync(error); - return; - } + Color = DiscordColor.Red, + Description = "Error: Failed to unassign staff member from ticket." + }, true); + return; + } - if (!Database.UnassignStaff(ticket)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Error: Failed to unassign staff from ticket." - }; - await command.RespondAsync(error); - return; - } + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Green, + Description = "Unassigned staff member from ticket." + }); - DiscordEmbed message = new DiscordEmbedBuilder + // Log it if the log channel exists + DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); + if (logChannel != null) + { + await logChannel.SendMessageAsync(new DiscordEmbedBuilder { Color = DiscordColor.Green, - Description = "Unassigned staff from ticket." - }; - await command.RespondAsync(message); - - // Log it if the log channel exists - DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); - if (logChannel != null) - { - DiscordEmbed logMessage = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Staff was unassigned from " + command.Channel.Mention + " by " + command.Member.Mention + "." - }; - await logChannel.SendMessageAsync(logMessage); - } + Description = "Staff member was unassigned from " + command.Channel.Mention + " by " + command.Member.Mention + "." + }); } } } \ No newline at end of file diff --git a/SupportChild/Commands/UnblacklistCommand.cs b/SupportChild/Commands/UnblacklistCommand.cs index e73e5b3..30d591a 100644 --- a/SupportChild/Commands/UnblacklistCommand.cs +++ b/SupportChild/Commands/UnblacklistCommand.cs @@ -1,99 +1,54 @@ using System; using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; using DSharpPlus.Entities; -using DSharpPlus.Exceptions; -using Microsoft.Extensions.Logging; +using DSharpPlus.SlashCommands; +using DSharpPlus.SlashCommands.Attributes; -namespace SupportChild.Commands +namespace SupportChild.Commands; + +public class UnblacklistCommand : ApplicationCommandModule { - public class UnblacklistCommand : BaseCommandModule + [SlashRequireGuild] + [SlashCommand("unblacklist", "Unblacklists a user from opening tickets.")] + public async Task OnExecute(InteractionContext command, [Option("User", "User to remove from blacklist.")] DiscordUser user) { - [Command("unblacklist")] - [Description("Un-blacklists a user from opening tickets.")] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) + try { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "unblacklist")) + if (!Database.Unblacklist(user.Id)) { - DiscordEmbed error = new DiscordEmbedBuilder + await command.CreateResponseAsync(new DiscordEmbedBuilder { Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the unblacklist command but did not have permission."); + Description = user.Mention + " is not blacklisted." + }, true); return; } - string[] words = Utilities.ParseIDs(command.RawArgumentString); - foreach (string word in words) + await command.CreateResponseAsync(new DiscordEmbedBuilder { - if (ulong.TryParse(word, out ulong userId)) + Color = DiscordColor.Green, + Description = "Removed " + user.Mention + " from blacklist." + }, true); + + // Log it if the log channel exists + DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); + if (logChannel != null) + { + await logChannel.SendMessageAsync(new DiscordEmbedBuilder { - DiscordUser blacklistedUser = null; - try - { - blacklistedUser = await command.Client.GetUserAsync(userId); - } - catch (NotFoundException) { } - - if (blacklistedUser == null) - { - DiscordEmbed message = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Error: Could not find user." - }; - await command.RespondAsync(message); - continue; - } - - try - { - if (!Database.Unblacklist(blacklistedUser.Id)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = blacklistedUser.Mention + " is not blacklisted." - }; - await command.RespondAsync(error); - continue; - } - - DiscordEmbed message = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Removed " + blacklistedUser.Mention + " from blacklist." - }; - await command.RespondAsync(message); - - // Log it if the log channel exists - DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); - if (logChannel != null) - { - DiscordEmbed logMessage = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = blacklistedUser.Mention + " was unblacklisted from opening tickets by " + command.Member.Mention + "." - }; - await logChannel.SendMessageAsync(logMessage); - } - } - catch (Exception) - { - DiscordEmbed message = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "Error occured while removing " + blacklistedUser.Mention + " from blacklist." - }; - await command.RespondAsync(message); - throw; - } - } + Color = DiscordColor.Green, + Description = user.Mention + " was unblacklisted from opening tickets by " + command.Member.Mention + "." + }); } } + catch (Exception) + { + await command.CreateResponseAsync(new DiscordEmbedBuilder + { + Color = DiscordColor.Red, + Description = "Error occured while removing " + user.Mention + " from blacklist." + }, true); + throw; + } } } \ No newline at end of file diff --git a/SupportChild/Commands/UnsetTicketCommand.cs b/SupportChild/Commands/UnsetTicketCommand.cs deleted file mode 100644 index 1ead160..0000000 --- a/SupportChild/Commands/UnsetTicketCommand.cs +++ /dev/null @@ -1,70 +0,0 @@ -using System.Threading.Tasks; -using DSharpPlus.CommandsNext; -using DSharpPlus.CommandsNext.Attributes; -using DSharpPlus.Entities; -using Microsoft.Extensions.Logging; -using MySql.Data.MySqlClient; - -namespace SupportChild.Commands -{ - public class UnsetTicketCommand : BaseCommandModule - { - [Command("unsetticket")] - [Description( - "Deletes a channel from the ticket system without deleting the channel.")] - public async Task OnExecute(CommandContext command, [RemainingText] string commandArgs) - { - using (MySqlConnection c = Database.GetConnection()) - { - // Check if the user has permission to use this command. - if (!Config.HasPermission(command.Member, "unsetticket")) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "You do not have permission to use this command." - }; - await command.RespondAsync(error); - command.Client.Logger.Log(LogLevel.Information, "User tried to use the unsetticket command but did not have permission."); - return; - } - - // Check if ticket exists in the database - if (!Database.TryGetOpenTicket(command.Channel.Id, out Database.Ticket ticket)) - { - DiscordEmbed error = new DiscordEmbedBuilder - { - Color = DiscordColor.Red, - Description = "This channel is not a ticket." - }; - await command.RespondAsync(error); - return; - } - - c.Open(); - MySqlCommand deletion = new MySqlCommand(@"DELETE FROM tickets WHERE channel_id=@channel_id", c); - deletion.Parameters.AddWithValue("@channel_id", command.Channel.Id); - deletion.Prepare(); - deletion.ExecuteNonQuery(); - DiscordEmbed message = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = "Channel has been undesignated as a ticket." - }; - await command.RespondAsync(message); - - // Log it if the log channel exists - DiscordChannel logChannel = command.Guild.GetChannel(Config.logChannel); - if (logChannel != null) - { - DiscordEmbed logMessage = new DiscordEmbedBuilder - { - Color = DiscordColor.Green, - Description = command.Channel.Mention + " has been undesignated as a ticket by " + command.Member.Mention + "." - }; - await logChannel.SendMessageAsync(logMessage); - } - } - } - } -} \ No newline at end of file From df1e669aa466b270922fd2c40d6eed31976dba2b Mon Sep 17 00:00:00 2001 From: Emotion Date: Sun, 21 Aug 2022 19:36:29 +1200 Subject: [PATCH 5/5] Create auto-approve.yml --- .github/workflows/auto-approve.yml | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) create mode 100644 .github/workflows/auto-approve.yml diff --git a/.github/workflows/auto-approve.yml b/.github/workflows/auto-approve.yml new file mode 100644 index 0000000..a1bb62f --- /dev/null +++ b/.github/workflows/auto-approve.yml @@ -0,0 +1,26 @@ +name: Auto approve + +on: pull_request + +permissions: + contents: read + +jobs: + auto-approve: + + name: Auto approve Pull Request + runs-on: ubuntu-latest + + # for hmarr/auto-approve-action to approve PRs + permissions: + pull-requests: write + + # Only run this on the main repo + if: github.event.pull_request.head.repo.full_name == 'EmotionChild/SupportChild' + + steps: + - name: Approve via actions + uses: hmarr/auto-approve-action@v2.2.1 + if: github.actor == 'EmotionChild' || github.actor == 'dependabot[bot]' + with: + github-token: "${{ secrets.GITHUB_TOKEN }}"