diff --git a/src/commands/mod/warning/warnings.ts b/src/commands/mod/warning/warnings.ts
new file mode 100644
index 0000000..b3428a3
--- /dev/null
+++ b/src/commands/mod/warning/warnings.ts
@@ -0,0 +1,184 @@
+// SPDX-License-Identifier: GPL-3.0-or-later
+/*
+ Animal Rights Advocates Discord Bot
+ Copyright (C) 2024 Anthony Berg
+
+ This program is free software: you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation, either version 3 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program. If not, see .
+*/
+
+import { Args, Command, RegisterBehavior } from '@sapphire/framework';
+import { ChannelType, EmbedBuilder } from 'discord.js';
+import type { Message, Guild, User } from 'discord.js';
+import IDs from '#utils/ids';
+import { fetchWarnings } from '#utils/database/warnings';
+
+export class RestrictLogsCommand extends Command {
+ public constructor(context: Command.LoaderContext, options: Command.Options) {
+ super(context, {
+ ...options,
+ name: 'warnings',
+ aliases: ['warninglog', 'warnlog'],
+ description: 'Shows all the warnings for the user',
+ preconditions: ['ModOnly'],
+ });
+ }
+
+ // Registers that this is a slash command
+ public override registerApplicationCommands(registry: Command.Registry) {
+ registry.registerChatInputCommand(
+ (builder) =>
+ builder
+ .setName(this.name)
+ .setDescription(this.description)
+ .addUserOption((option) =>
+ option
+ .setName('user')
+ .setDescription('User to check the warnings for')
+ .setRequired(true),
+ ),
+ {
+ behaviorWhenNotIdentical: RegisterBehavior.Overwrite,
+ },
+ );
+ }
+
+ // Command run
+ public async chatInputRun(interaction: Command.ChatInputCommandInteraction) {
+ // Get the arguments
+ const user = interaction.options.getUser('user', true);
+ const { guild } = interaction;
+
+ // Checks if all the variables are of the right type
+ if (guild === null) {
+ await interaction.reply({
+ content: 'Error fetching guild!',
+ ephemeral: true,
+ fetchReply: true,
+ });
+ return;
+ }
+
+ await interaction.deferReply({ ephemeral: true });
+
+ const info = await this.warnings(user, guild);
+
+ await interaction.editReply({
+ content: info.message,
+ embeds: info.embeds,
+ });
+ }
+
+ // Non Application Command method of banning a user
+ public async messageRun(message: Message, args: Args) {
+ // Get arguments
+ let user: User | undefined;
+ try {
+ user = await args.pick('user');
+ } catch {
+ user = undefined;
+ }
+
+ const { guild } = message;
+
+ if (guild === null) {
+ await message.react('❌');
+ await message.reply('Guild not found! Try again or contact a developer!');
+ return;
+ }
+
+ if (user === undefined) {
+ const { channel } = message;
+
+ if (channel.type !== ChannelType.GuildText) {
+ await message.react('❌');
+ await message.reply('User was not provided!');
+ return;
+ }
+
+ let topic: string[];
+
+ if (channel.parentId === IDs.categories.modMail) {
+ // Checks if the channel topic has the user's snowflake
+ if (channel.topic !== null) {
+ topic = channel.topic.split(' ');
+ // eslint-disable-next-line prefer-destructuring
+ const userId = topic[2];
+
+ user = guild.client.users.cache.get(userId);
+
+ if (user === undefined) {
+ user = await guild.client.users.fetch(userId);
+ }
+ }
+ }
+ }
+
+ if (user === undefined) {
+ await message.react('❌');
+ await message.reply('User was not provided!');
+ return;
+ }
+
+ const info = await this.warnings(user, guild);
+
+ await message.reply({ content: info.message, embeds: info.embeds });
+ }
+
+ private async warnings(user: User, guild: Guild) {
+ const info = {
+ message: '',
+ embeds: [] as EmbedBuilder[],
+ };
+
+ const warnings = await fetchWarnings(user.id);
+
+ if (warnings.length === 0) {
+ info.message = `${user} user has no warnings.`;
+ return info;
+ }
+
+ // Creates the embed to display the restrictions
+ const embed = new EmbedBuilder()
+ .setColor('#FF6700')
+ .setTitle(`${warnings.length} restrictions for ${user.tag}`)
+ .setThumbnail(user.displayAvatarURL())
+ .setFooter({ text: `ID: ${user.id}` });
+
+ // Add up to 10 of the latest restrictions to the embed
+ for (
+ let i = warnings.length > 10 ? warnings.length - 10 : 0;
+ i < warnings.length;
+ i += 1
+ ) {
+ // Get mod names
+ let mod = warnings[i].modId;
+ const modMember = guild.members.cache.get(mod);
+ if (modMember !== undefined) {
+ mod = modMember.displayName;
+ }
+
+ let warnTitle = `ID: ${warnings[i].id} | Moderator: ${mod} | `;
+
+ warnTitle += `Date: `;
+
+ embed.addFields({
+ name: warnTitle,
+ value: warnings[i].note,
+ });
+ }
+
+ info.embeds.push(embed);
+ return info;
+ }
+}
diff --git a/src/utils/database/warnings.ts b/src/utils/database/warnings.ts
index ea5d121..e181959 100644
--- a/src/utils/database/warnings.ts
+++ b/src/utils/database/warnings.ts
@@ -27,3 +27,17 @@ export async function addWarn(
},
});
}
+
+export async function fetchWarnings(userId: Snowflake) {
+ const warnings = await container.database.warning.findMany({
+ where: {
+ userId,
+ active: true,
+ },
+ orderBy: {
+ id: 'asc',
+ },
+ });
+
+ return warnings;
+}