From 8e16a4fda6ca55e45f9f2c63e1e2636ed2df38ab Mon Sep 17 00:00:00 2001 From: Ethan Lane Date: Sat, 30 Nov 2024 13:53:13 +0000 Subject: [PATCH 01/10] Add AutoKickConfig to database --- .../Down/01-AutoKickConfig.sql | 1 + .../Up/01-AutoKickConfig-Table.sql | 10 ++++ .../Up/02-AutoKickConfig-Key.sql | 2 + src/database/entities/AutoKickConfig.ts | 50 +++++++++++++++++++ .../1732973911304-createAutoKickConfig.ts | 19 +++++++ 5 files changed, 82 insertions(+) create mode 100644 database/3.2.4/1732973911304-createAutoKickConfig/Down/01-AutoKickConfig.sql create mode 100644 database/3.2.4/1732973911304-createAutoKickConfig/Up/01-AutoKickConfig-Table.sql create mode 100644 database/3.2.4/1732973911304-createAutoKickConfig/Up/02-AutoKickConfig-Key.sql create mode 100644 src/database/entities/AutoKickConfig.ts create mode 100644 src/database/migrations/3.2.4/1732973911304-createAutoKickConfig.ts diff --git a/database/3.2.4/1732973911304-createAutoKickConfig/Down/01-AutoKickConfig.sql b/database/3.2.4/1732973911304-createAutoKickConfig/Down/01-AutoKickConfig.sql new file mode 100644 index 0000000..8412a02 --- /dev/null +++ b/database/3.2.4/1732973911304-createAutoKickConfig/Down/01-AutoKickConfig.sql @@ -0,0 +1 @@ +DROP TABLE auto_kick_config; diff --git a/database/3.2.4/1732973911304-createAutoKickConfig/Up/01-AutoKickConfig-Table.sql b/database/3.2.4/1732973911304-createAutoKickConfig/Up/01-AutoKickConfig-Table.sql new file mode 100644 index 0000000..e7d377a --- /dev/null +++ b/database/3.2.4/1732973911304-createAutoKickConfig/Up/01-AutoKickConfig-Table.sql @@ -0,0 +1,10 @@ +CREATE TABLE auto_kick_config ( + Id varchar(255) NOT NULL, + WhenCreated datetime NOT NULL, + WhenUpdated datetime NOT NULL, + ServerId varchar(255) NOT NULL, + RoleId varchar(255) NOT NULL, + KickTime int NOT NULL, + NoticeTime int NULL, + NoticeChannelId varchar(255) NULL +); diff --git a/database/3.2.4/1732973911304-createAutoKickConfig/Up/02-AutoKickConfig-Key.sql b/database/3.2.4/1732973911304-createAutoKickConfig/Up/02-AutoKickConfig-Key.sql new file mode 100644 index 0000000..e3fec43 --- /dev/null +++ b/database/3.2.4/1732973911304-createAutoKickConfig/Up/02-AutoKickConfig-Key.sql @@ -0,0 +1,2 @@ +ALTER TABLE auto_kick_config + ADD PRIMARY KEY (Id); diff --git a/src/database/entities/AutoKickConfig.ts b/src/database/entities/AutoKickConfig.ts new file mode 100644 index 0000000..1ee3625 --- /dev/null +++ b/src/database/entities/AutoKickConfig.ts @@ -0,0 +1,50 @@ +import {Column, Entity} from "typeorm"; +import AppDataSource from "../dataSources/appDataSource"; +import BaseEntity from "../../contracts/BaseEntity"; + +@Entity() +export default class AutoKickConfig extends BaseEntity { + constructor(serverId: string, roleId: string, kickTime: number, noticeTime?: number, noticeChannelId?: string) { + super(); + + this.ServerId = serverId; + this.RoleId = roleId; + this.KickTime = kickTime; + this.NoticeTime = noticeTime; + this.NoticeChannelId = noticeChannelId; + } + + @Column() + ServerId: string; + + @Column() + RoleId: string; + + @Column({ type: "int" }) + KickTime: number; + + @Column({ type: "int", nullable: true }) + NoticeTime?: number; + + @Column({ nullable: true }) + NoticeChannelId?: string; + + public UpdateBasicDetails(roleId: string, kickTime: number, noticeTime?: number, noticeChannelId?: string) { + this.RoleId = roleId; + this.KickTime = kickTime; + this.NoticeTime = noticeTime; + this.NoticeChannelId = noticeChannelId; + } + + public static async FetchOneByServerIdAndRoleId(serverId: string, roleId: string): Promise { + const repository = AppDataSource.getRepository(AutoKickConfig); + + const query = repository + .createQueryBuilder("config") + .where("config.serverId = :serverId", { serverId }) + .andWhere("config.roleId = :roleId", { roleId }) + .getOne(); + + return query; + } +} diff --git a/src/database/migrations/3.2.4/1732973911304-createAutoKickConfig.ts b/src/database/migrations/3.2.4/1732973911304-createAutoKickConfig.ts new file mode 100644 index 0000000..c4e9a74 --- /dev/null +++ b/src/database/migrations/3.2.4/1732973911304-createAutoKickConfig.ts @@ -0,0 +1,19 @@ +import { MigrationInterface, QueryRunner } from "typeorm"; +import MigrationHelper from "../../../helpers/MigrationHelper"; + +export class CreateAutoKickConfig1732973911304 implements MigrationInterface { + + public async up(queryRunner: QueryRunner): Promise { + MigrationHelper.Up("1732973911304-createAutoKickConfig", "3.2.4", [ + "01-AutoKickConfig-Table", + "02-AutoKickConfig-Key", + ], queryRunner) + } + + public async down(queryRunner: QueryRunner): Promise { + MigrationHelper.Down("1732973911304-createAutoKickConfig", "3.2.4", [ + "01-AutoKickConfig", + ], queryRunner) + } + +} -- 2.45.2 From bc3c16bdcbb943d1e8b262f3adbd1c6a999fcf2b Mon Sep 17 00:00:00 2001 From: Ethan Lane Date: Sat, 30 Nov 2024 13:53:24 +0000 Subject: [PATCH 02/10] WIP: Start creating autokick command --- src/commands/autokick.ts | 54 ++++++++++++++++++++++++++++++++++++++++ 1 file changed, 54 insertions(+) create mode 100644 src/commands/autokick.ts diff --git a/src/commands/autokick.ts b/src/commands/autokick.ts new file mode 100644 index 0000000..09318b3 --- /dev/null +++ b/src/commands/autokick.ts @@ -0,0 +1,54 @@ +import {CommandInteraction, PermissionFlagsBits, SlashCommandBuilder} from "discord.js"; +import {Command} from "../type/command"; + +export default class Autokick extends Command { + constructor() { + super(); + + this.CommandBuilder = new SlashCommandBuilder() + .setName("autokick") + .setDescription("Configure the auto kick functionality") + .setDefaultMemberPermissions(PermissionFlagsBits.KickMembers) + .addSubcommand(x => x + .setName("set") + .setDescription("Set the configuration") + .addRoleOption(y => y + .setName("role") + .setDescription("The role the user needs to be auto kicked") + .setRequired(true)) + .addStringOption(y => y + .setName("kicktime") + .setDescription("The time with the role before being kicked (Ex: 2h 30m)") + .setRequired(true)) + .addStringOption(y => y + .setName("noticetime") + .setDescription("The time before being kicked when a notification is sent (Ex: 2h 30m)")) + .addChannelOption(y => y + .setName("noticechannel") + .setDescription("The channel to send the notification to"))) + .addSubcommand(x => x + .setName("unset") + .setDescription("Unset the current configuration")); + } + + public override async execute(interaction: CommandInteraction) { + if (!interaction.isChatInputCommand()) return; + + const subcommand = interaction.options.getSubcommand(); + + switch (subcommand) { + case "set": + await this.set(interaction); + break; + case "unset": + await this.unset(interaction); + break; + } + } + + private async set(interaction: CommandInteraction) { + } + + private async unset(interaction: CommandInteraction) { + } +} -- 2.45.2 From 2b4ae91584e60df9295cd667399fe77c278a7560 Mon Sep 17 00:00:00 2001 From: Ethan Lane Date: Fri, 6 Dec 2024 20:27:17 +0000 Subject: [PATCH 03/10] Create helper class --- src/helpers/AutoKickHelper.ts | 25 +++++++++++++++++++++++++ 1 file changed, 25 insertions(+) create mode 100644 src/helpers/AutoKickHelper.ts diff --git a/src/helpers/AutoKickHelper.ts b/src/helpers/AutoKickHelper.ts new file mode 100644 index 0000000..7c3e2aa --- /dev/null +++ b/src/helpers/AutoKickHelper.ts @@ -0,0 +1,25 @@ +import AutoKickConfig from "../database/entities/AutoKickConfig"; + +export default class AutoKickHelper { + public static async SetSetting(serverId: string, roleId: string, kickTime: number, noticeTime?: number, noticeChannelId?: string) { + const config = await AutoKickConfig.FetchOneByServerIdAndRoleId(serverId, roleId); + + if (!config) { + const newConfig = new AutoKickConfig(serverId, roleId, kickTime, noticeTime, noticeChannelId); + await newConfig.Save(AutoKickConfig, newConfig); + + return; + } + + config.UpdateBasicDetails(roleId, kickTime, noticeTime, noticeChannelId); + await config.Save(AutoKickConfig, config); + } + + public static async ResetSetting(serverId: string, roleId: string) { + const config = await AutoKickConfig.FetchOneByServerIdAndRoleId(serverId, roleId); + + if (!config) return; + + await AutoKickConfig.Remove(AutoKickConfig, config); + } +} -- 2.45.2 From 8d1befcf25b4645472f70027414d8043c9b9c613 Mon Sep 17 00:00:00 2001 From: Ethan Lane Date: Fri, 6 Dec 2024 20:54:47 +0000 Subject: [PATCH 04/10] Create autokick command --- src/commands/autokick.ts | 65 +++++++++++++++++++++++-- src/database/entities/AutoKickConfig.ts | 11 +++++ src/helpers/AutoKickHelper.ts | 32 ++++++++---- src/registry.ts | 2 + 4 files changed, 97 insertions(+), 13 deletions(-) diff --git a/src/commands/autokick.ts b/src/commands/autokick.ts index 09318b3..760441e 100644 --- a/src/commands/autokick.ts +++ b/src/commands/autokick.ts @@ -1,5 +1,7 @@ -import {CommandInteraction, PermissionFlagsBits, SlashCommandBuilder} from "discord.js"; +import {ChatInputCommandInteraction, CommandInteraction, EmbedBuilder, PermissionFlagsBits, SlashCommandBuilder} from "discord.js"; import {Command} from "../type/command"; +import TimeLengthInput from "../helpers/TimeLengthInput"; +import AutoKickHelper from "../helpers/AutoKickHelper"; export default class Autokick extends Command { constructor() { @@ -46,9 +48,66 @@ export default class Autokick extends Command { } } - private async set(interaction: CommandInteraction) { + private async set(interaction: ChatInputCommandInteraction) { + if (!interaction.guildId) return; + + const roleOption = interaction.options.getRole("role", true); + const kickTimeOption = interaction.options.getString("kicktime", true); + const noticeTimeOption = interaction.options.getString("noticetime"); + const noticeChannelOption = interaction.options.getChannel("noticechannel"); + + const roleId = roleOption.id; + const kickTimeInput = new TimeLengthInput(kickTimeOption); + const noticeTimeInput = noticeTimeOption ? new TimeLengthInput(noticeTimeOption) : undefined; + const noticeChannelId = noticeChannelOption?.id; + + if ((noticeTimeInput && !noticeTimeOption) || (!noticeTimeInput && noticeChannelOption)) { + await interaction.reply("Both `noticetime` and `noticechannel` must be set if you want a notification embed"); + return; + } + + await AutoKickHelper.SetSetting(interaction.guildId, roleId, kickTimeInput.GetMilliseconds(), noticeTimeInput?.GetMilliseconds(), noticeChannelId); + + const embed = new EmbedBuilder() + .setTitle("Auto Kick") + .setDescription("Configured auto kick for this server") + .addFields([ + { + name: "Role", + value: roleOption.name, + inline: true, + }, + { + name: "Kick Time", + value: kickTimeInput.GetLengthShort(), + inline: true, + }, + ]); + + if (noticeTimeInput) { + embed.addFields([ + { + name: "Notice Time", + value: noticeTimeInput.GetLengthShort(), + }, + { + name: "Notice Channel", + value: noticeChannelOption!.name!, + inline: true, + }, + ]); + } + + await interaction.reply({ + embeds: [ embed ], + }); } - private async unset(interaction: CommandInteraction) { + private async unset(interaction: ChatInputCommandInteraction) { + if (!interaction.guildId) return; + + await AutoKickHelper.UnsetSetting(interaction.guildId); + + await interaction.reply("Unset the auto kick configuration for this server"); } } diff --git a/src/database/entities/AutoKickConfig.ts b/src/database/entities/AutoKickConfig.ts index 1ee3625..6a82ba5 100644 --- a/src/database/entities/AutoKickConfig.ts +++ b/src/database/entities/AutoKickConfig.ts @@ -47,4 +47,15 @@ export default class AutoKickConfig extends BaseEntity { return query; } + + public static async FetchAllByServerId(serverId: string): Promise { + const repository = AppDataSource.getRepository(AutoKickConfig); + + const query = repository + .createQueryBuilder("config") + .where("config.serverId = :serverId", { serverId }) + .getMany(); + + return query; + } } diff --git a/src/helpers/AutoKickHelper.ts b/src/helpers/AutoKickHelper.ts index 7c3e2aa..a02a934 100644 --- a/src/helpers/AutoKickHelper.ts +++ b/src/helpers/AutoKickHelper.ts @@ -1,25 +1,37 @@ import AutoKickConfig from "../database/entities/AutoKickConfig"; export default class AutoKickHelper { - public static async SetSetting(serverId: string, roleId: string, kickTime: number, noticeTime?: number, noticeChannelId?: string) { - const config = await AutoKickConfig.FetchOneByServerIdAndRoleId(serverId, roleId); + public static async GetSetting(serverId: string): Promise { + const configs = await AutoKickConfig.FetchAllByServerId(serverId); - if (!config) { - const newConfig = new AutoKickConfig(serverId, roleId, kickTime, noticeTime, noticeChannelId); - await newConfig.Save(AutoKickConfig, newConfig); + if (configs.length != 1) { + return null; + } + + return configs[0]; + } + + public static async SetSetting(serverId: string, roleId: string, kickTime: number, noticeTime?: number, noticeChannelId?: string) { + const configs = await AutoKickConfig.FetchAllByServerId(serverId); + + if (configs.length == 0) { + const config = new AutoKickConfig(serverId, roleId, kickTime, noticeTime, noticeChannelId); + await config.Save(AutoKickConfig, config); return; } + const config = configs[0]; + config.UpdateBasicDetails(roleId, kickTime, noticeTime, noticeChannelId); await config.Save(AutoKickConfig, config); } - public static async ResetSetting(serverId: string, roleId: string) { - const config = await AutoKickConfig.FetchOneByServerIdAndRoleId(serverId, roleId); + public static async UnsetSetting(serverId: string) { + const configs = await AutoKickConfig.FetchAllByServerId(serverId); - if (!config) return; - - await AutoKickConfig.Remove(AutoKickConfig, config); + for (let config of configs) { + await AutoKickConfig.Remove(AutoKickConfig, config); + } } } diff --git a/src/registry.ts b/src/registry.ts index a2d7391..36dd021 100644 --- a/src/registry.ts +++ b/src/registry.ts @@ -4,6 +4,7 @@ import { EventType } from "./constants/EventType"; // Command Imports import About from "./commands/about"; import Audits from "./commands/audits"; +import Autokick from "./commands/autokick"; import Ban from "./commands/ban"; import Bunny from "./commands/bunny"; import Clear from "./commands/clear"; @@ -45,6 +46,7 @@ export default class Registry { public static RegisterCommands() { CoreClient.RegisterCommand("about", new About()); CoreClient.RegisterCommand("audits", new Audits()); + CoreClient.RegisterCommand("autokick", new Autokick()); CoreClient.RegisterCommand("ban", new Ban()); CoreClient.RegisterCommand("bunny", new Bunny()); CoreClient.RegisterCommand("clear", new Clear()); -- 2.45.2 From 9e05d884cfcda4967d9c495fc33855d5a7bcf703 Mon Sep 17 00:00:00 2001 From: Ethan Lane Date: Mon, 16 Dec 2024 18:45:35 +0000 Subject: [PATCH 05/10] Add ability to run background tasks on a timer --- package.json | 1 + src/client/client.ts | 10 ++++- src/helpers/TimerHelper.ts | 81 ++++++++++++++++++++++++++++++++++++++ src/type/primitive.ts | 1 + yarn.lock | 18 +++++++++ 5 files changed, 109 insertions(+), 2 deletions(-) create mode 100644 src/helpers/TimerHelper.ts create mode 100644 src/type/primitive.ts diff --git a/package.json b/package.json index c5af404..27b06a3 100644 --- a/package.json +++ b/package.json @@ -30,6 +30,7 @@ "@discordjs/rest": "^2.0.0", "@types/jest": "^29.0.0", "@types/uuid": "^9.0.0", + "cron": "^3.3.1", "discord.js": "^14.3.0", "dotenv": "^16.0.0", "emoji-regex": "^10.0.0", diff --git a/src/client/client.ts b/src/client/client.ts index c1ce5da..b51ccda 100644 --- a/src/client/client.ts +++ b/src/client/client.ts @@ -1,6 +1,5 @@ import { Client, Partials } from "discord.js"; import * as dotenv from "dotenv"; -import { createConnection } from "typeorm"; import { EventType } from "../constants/EventType"; import ICommandItem from "../contracts/ICommandItem"; import IEventItem from "../contracts/IEventItem"; @@ -12,6 +11,7 @@ import AppDataSource from "../database/dataSources/appDataSource"; import ButtonEventItem from "../contracts/ButtonEventItem"; import { ButtonEvent } from "../type/buttonEvent"; import CacheHelper from "../helpers/CacheHelper"; +import TimerHelper from "../helpers/TimerHelper"; export class CoreClient extends Client { private static _commandItems: ICommandItem[]; @@ -20,6 +20,7 @@ export class CoreClient extends Client { private _events: Events; private _util: Util; + private _timerHelper: TimerHelper; public static get commandItems(): ICommandItem[] { return this._commandItems; @@ -43,6 +44,7 @@ export class CoreClient extends Client { this._events = new Events(); this._util = new Util(); + this._timerHelper = new TimerHelper(); } public async start() { @@ -52,7 +54,11 @@ export class CoreClient extends Client { } await AppDataSource.initialize() - .then(() => console.log("Data Source Initialized")) + .then(() => { + console.log("Data Source Initialized"); + + // this.timerHelper.AddTimer + }) .catch((err) => console.error("Error Initialising Data Source", err)); super.on("interactionCreate", this._events.onInteractionCreate); diff --git a/src/helpers/TimerHelper.ts b/src/helpers/TimerHelper.ts new file mode 100644 index 0000000..3cc4246 --- /dev/null +++ b/src/helpers/TimerHelper.ts @@ -0,0 +1,81 @@ +import {CronJob} from "cron"; +import {primitive} from "../type/primitive"; +import {v4} from "uuid"; + +interface Timer { + id: string; + job: CronJob; + context: Map; + onTick: ((context: Map) => void) | ((context: Map) => Promise); + runOnStart: boolean; +} + +export default class TimerHelper { + private _timers: Timer[]; + + constructor() { + this._timers = []; + } + + public AddTimer( + cronTime: string, + timeZone: string, + onTick: ((context: Map) => void) | ((context: Map) => Promise), + runOnStart: boolean = false): string { + const context = new Map(); + + const job = new CronJob( + cronTime, + () => { + onTick(context); + }, + null, + false, + timeZone, + ); + + const id = v4(); + + this._timers.push({ + id, + job, + context, + onTick, + runOnStart, + }); + + return id; + } + + public StartAllTimers() { + this._timers.forEach(timer => this.StartJob(timer)); + } + + public StopAllTimers() { + this._timers.forEach(timer => timer.job.stop()); + } + + public StartTimer(id: string) { + const timer = this._timers.find(x => x.id == id); + + if (!timer) return; + + this.StartJob(timer); + } + + public StopTimer(id: string) { + const timer = this._timers.find(x => x.id == id); + + if (!timer) return; + + timer.job.stop(); + } + + private StartJob(timer: Timer) { + timer.job.start(); + + if (timer.runOnStart) { + timer.onTick(timer.context); + } + } +} diff --git a/src/type/primitive.ts b/src/type/primitive.ts new file mode 100644 index 0000000..9093737 --- /dev/null +++ b/src/type/primitive.ts @@ -0,0 +1 @@ +export type primitive = string | number | boolean; diff --git a/yarn.lock b/yarn.lock index ced241e..24feeb5 100644 --- a/yarn.lock +++ b/yarn.lock @@ -788,6 +788,11 @@ expect "^29.0.0" pretty-format "^29.0.0" +"@types/luxon@~3.4.0": + version "3.4.2" + resolved "https://registry.yarnpkg.com/@types/luxon/-/luxon-3.4.2.tgz#e4fc7214a420173cea47739c33cdf10874694db7" + integrity sha512-TifLZlFudklWlMBfhubvgqTXRzLDI5pCbGa4P8a3wPyUQSW+1xQ5eDsreP9DWHX3tjq1ke96uYG/nwundroWcA== + "@types/node@*": version "22.7.5" resolved "https://registry.npmjs.org/@types/node/-/node-22.7.5.tgz" @@ -1458,6 +1463,14 @@ create-jest@^29.7.0: jest-util "^29.7.0" prompts "^2.0.1" +cron@^3.3.1: + version "3.3.1" + resolved "https://registry.yarnpkg.com/cron/-/cron-3.3.1.tgz#03c56b4a3ad52606160adfba1fab932c53838807" + integrity sha512-KpvuzJEbeTMTfLsXhUuDfsFYr8s5roUlLKb4fa68GszWrA4783C7q6m9yj4vyc6neyD/V9e0YiADSX2c+yRDXg== + dependencies: + "@types/luxon" "~3.4.0" + luxon "~3.5.0" + cross-spawn@^7.0.0, cross-spawn@^7.0.3: version "7.0.3" resolved "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz" @@ -3076,6 +3089,11 @@ lru-cache@^5.1.1: dependencies: yallist "^3.0.2" +luxon@~3.5.0: + version "3.5.0" + resolved "https://registry.yarnpkg.com/luxon/-/luxon-3.5.0.tgz#6b6f65c5cd1d61d1fd19dbf07ee87a50bf4b8e20" + integrity sha512-rh+Zjr6DNfUYR3bPwJEnuwDdqMbxZW7LOQfUN4B54+Cl+0o5zaU9RJ6bcidfDtC1cWCZXQ+nvX8bf6bAji37QQ== + magic-bytes.js@^1.10.0: version "1.10.0" resolved "https://registry.npmjs.org/magic-bytes.js/-/magic-bytes.js-1.10.0.tgz" -- 2.45.2 From 94a2965e8e60110d7811472352512d8a8474183f Mon Sep 17 00:00:00 2001 From: Ethan Lane Date: Mon, 16 Dec 2024 18:59:52 +0000 Subject: [PATCH 06/10] WIP: Start of creating autokick timer --- src/client/client.ts | 11 ++++++++++- src/timers/AutoKick.ts | 17 +++++++++++++++++ 2 files changed, 27 insertions(+), 1 deletion(-) create mode 100644 src/timers/AutoKick.ts diff --git a/src/client/client.ts b/src/client/client.ts index b51ccda..563a007 100644 --- a/src/client/client.ts +++ b/src/client/client.ts @@ -12,11 +12,14 @@ import ButtonEventItem from "../contracts/ButtonEventItem"; import { ButtonEvent } from "../type/buttonEvent"; import CacheHelper from "../helpers/CacheHelper"; import TimerHelper from "../helpers/TimerHelper"; +import AutoKick from "../timers/AutoKick"; export class CoreClient extends Client { private static _commandItems: ICommandItem[]; private static _eventItems: IEventItem[]; private static _buttonEvents: ButtonEventItem[]; + private static _baseClient: Client; + private _events: Events; private _util: Util; @@ -34,6 +37,10 @@ export class CoreClient extends Client { return this._buttonEvents; } + public static get baseClient(): Client { + return this._baseClient; + } + constructor(intents: number[], partials: Partials[]) { super({ intents: intents, partials: partials }); dotenv.config(); @@ -57,7 +64,7 @@ export class CoreClient extends Client { .then(() => { console.log("Data Source Initialized"); - // this.timerHelper.AddTimer + this._timerHelper.AddTimer("*/5 * * * *", "Europe/London", AutoKick, true); }) .catch((err) => console.error("Error Initialising Data Source", err)); @@ -72,6 +79,8 @@ export class CoreClient extends Client { this._util.loadEvents(this, CoreClient._eventItems); this._util.loadSlashCommands(this); + + CoreClient._baseClient = this; } public static RegisterCommand(name: string, command: Command, serverId?: string) { diff --git a/src/timers/AutoKick.ts b/src/timers/AutoKick.ts new file mode 100644 index 0000000..42a9e93 --- /dev/null +++ b/src/timers/AutoKick.ts @@ -0,0 +1,17 @@ +import {CoreClient} from "../client/client"; +import AutoKickConfig from "../database/entities/AutoKickConfig"; + +export default async function AutoKick() { + const client = CoreClient.baseClient; + const autoKickConfigs = await AutoKickConfig.FetchAll(AutoKickConfig); + + for (let config of autoKickConfigs) { + const guild = client.guilds.cache.find(x => x.id == config.ServerId) || client.guilds.fetch(config.ServerId); + + if (!guild) { + continue; + } + + console.log(typeof guild); + } +} -- 2.45.2 From afafe246a858249bf3dc9b4da2a8f0215140a50d Mon Sep 17 00:00:00 2001 From: Ethan Lane Date: Sat, 21 Dec 2024 14:24:38 +0000 Subject: [PATCH 07/10] WIP: Add to the timer the kick functionality --- .env.example | 6 ++++-- src/client/client.ts | 7 ++++--- src/timers/AutoKick.ts | 27 ++++++++++++++++++++++++--- 3 files changed, 32 insertions(+), 8 deletions(-) diff --git a/.env.example b/.env.example index 44b071b..e2591d0 100644 --- a/.env.example +++ b/.env.example @@ -7,7 +7,7 @@ # any secret values. BOT_TOKEN= -BOT_VER=3.2.3 +BOT_VER=3.2.4 BOT_AUTHOR=Vylpes BOT_OWNERID=147392775707426816 BOT_CLIENTID=682942374040961060 @@ -23,4 +23,6 @@ DB_NAME=vylbot DB_AUTH_USER=dev DB_AUTH_PASS=dev DB_SYNC=true -DB_LOGGING=true \ No newline at end of file +DB_LOGGING=true +DB_DATA_LOCATION=./.temp/database +DB_ROOT_HOST=0.0.0.0 \ No newline at end of file diff --git a/src/client/client.ts b/src/client/client.ts index 563a007..f6568e4 100644 --- a/src/client/client.ts +++ b/src/client/client.ts @@ -20,7 +20,6 @@ export class CoreClient extends Client { private static _buttonEvents: ButtonEventItem[]; private static _baseClient: Client; - private _events: Events; private _util: Util; private _timerHelper: TimerHelper; @@ -60,11 +59,15 @@ export class CoreClient extends Client { return; } + CoreClient._baseClient = this; + await AppDataSource.initialize() .then(() => { console.log("Data Source Initialized"); this._timerHelper.AddTimer("*/5 * * * *", "Europe/London", AutoKick, true); + + this._timerHelper.StartAllTimers(); }) .catch((err) => console.error("Error Initialising Data Source", err)); @@ -79,8 +82,6 @@ export class CoreClient extends Client { this._util.loadEvents(this, CoreClient._eventItems); this._util.loadSlashCommands(this); - - CoreClient._baseClient = this; } public static RegisterCommand(name: string, command: Command, serverId?: string) { diff --git a/src/timers/AutoKick.ts b/src/timers/AutoKick.ts index 42a9e93..8158519 100644 --- a/src/timers/AutoKick.ts +++ b/src/timers/AutoKick.ts @@ -6,12 +6,33 @@ export default async function AutoKick() { const autoKickConfigs = await AutoKickConfig.FetchAll(AutoKickConfig); for (let config of autoKickConfigs) { - const guild = client.guilds.cache.find(x => x.id == config.ServerId) || client.guilds.fetch(config.ServerId); - + const guild = client.guilds.cache.find(x => x.id == config.ServerId) || await client.guilds.fetch(config.ServerId); + if (!guild) { continue; } - console.log(typeof guild); + await guild.members.fetch(); + + const role = guild.roles.cache.find(x => x.id == config.RoleId); + + if (!role) { + continue; + } + + for (let memberEntity of role.members) { + const member = memberEntity[1]; + + if (!member.kickable) { + continue; + } + + const whenToKick = new Date(member.joinedTimestamp! + config.KickTime); + const now = new Date(); + + if (whenToKick < now) { + await member.kick("Auto Kicked"); + } + } } } -- 2.45.2 From 40a976419232759b73309d5a87affa4195d48bab Mon Sep 17 00:00:00 2001 From: Ethan Lane Date: Sun, 22 Dec 2024 18:16:30 +0000 Subject: [PATCH 08/10] Add notice channel sending functionality --- src/client/client.ts | 2 +- src/constants/EmbedColours.ts | 2 ++ src/timers/AutoKick.ts | 62 ++++++++++++++++++++++++++++++++++- 3 files changed, 64 insertions(+), 2 deletions(-) diff --git a/src/client/client.ts b/src/client/client.ts index f6568e4..59fb946 100644 --- a/src/client/client.ts +++ b/src/client/client.ts @@ -65,7 +65,7 @@ export class CoreClient extends Client { .then(() => { console.log("Data Source Initialized"); - this._timerHelper.AddTimer("*/5 * * * *", "Europe/London", AutoKick, true); + this._timerHelper.AddTimer("0 * * * *", "Europe/London", AutoKick, true); this._timerHelper.StartAllTimers(); }) diff --git a/src/constants/EmbedColours.ts b/src/constants/EmbedColours.ts index 023c77a..dd84b94 100644 --- a/src/constants/EmbedColours.ts +++ b/src/constants/EmbedColours.ts @@ -1,3 +1,5 @@ export default class EmbedColours { public static readonly Ok = 0x3050ba; + public static readonly Warning = 0xffbf00; + public static readonly Danger = 0xd2042d; } \ No newline at end of file diff --git a/src/timers/AutoKick.ts b/src/timers/AutoKick.ts index 8158519..a98aee8 100644 --- a/src/timers/AutoKick.ts +++ b/src/timers/AutoKick.ts @@ -1,5 +1,7 @@ +import { EmbedBuilder } from "discord.js"; import {CoreClient} from "../client/client"; import AutoKickConfig from "../database/entities/AutoKickConfig"; +import EmbedColours from "../constants/EmbedColours"; export default async function AutoKick() { const client = CoreClient.baseClient; @@ -31,7 +33,65 @@ export default async function AutoKick() { const now = new Date(); if (whenToKick < now) { - await member.kick("Auto Kicked"); + // await member.kick("Auto Kicked"); + console.log("Kicked"); + + if (config.NoticeChannelId) { + const channel = guild.channels.cache.find(x => x.id == config.NoticeChannelId) || await guild.channels.fetch(config.NoticeChannelId); + + if (!channel?.isSendable()) { + continue; + } + + const embed = new EmbedBuilder() + .setTitle("Auto Kicked User") + .setColor(EmbedColours.Danger) + .setThumbnail(member.avatarURL()) + .addFields([ + { + name: "User", + value: `<@${member.user.id}> \`${member.user.username}\``, + inline: true, + }, + ]); + + await channel.send({ + embeds: [ embed ], + }); + } + } else if (config.NoticeChannelId && config.NoticeTime) { + const whenToNotice = new Date(whenToKick.getMilliseconds() - config.NoticeTime); + + const channel = guild.channels.cache.find(x => x.id == config.NoticeChannelId) || await guild.channels.fetch(config.NoticeChannelId); + + if (!channel?.isSendable()) { + continue; + } + + if (now.getMonth() == whenToNotice.getMonth() + && now.getDate() == whenToNotice.getDate() + && now.getHours() == whenToNotice.getHours()) { + const embed = new EmbedBuilder() + .setTitle("Auto Kick Notice") + .setColor(EmbedColours.Warning) + .setThumbnail(member.avatarURL()) + .addFields([ + { + name: "User", + value: `<@${member.user.id}> \`${member.user.username}\``, + inline: true, + }, + { + name: "When To Kick", + value: ``, + inline: true, + }, + ]); + + await channel.send({ + embeds: [ embed ], + }); + } } } } -- 2.45.2 From 6c48da114af26436af81ba6c85e2a8169f1b4e6e Mon Sep 17 00:00:00 2001 From: Ethan Lane Date: Sat, 28 Dec 2024 16:38:07 +0000 Subject: [PATCH 09/10] Fix time checking using the wrong function --- src/timers/AutoKick.ts | 11 +++++------ 1 file changed, 5 insertions(+), 6 deletions(-) diff --git a/src/timers/AutoKick.ts b/src/timers/AutoKick.ts index a98aee8..61f743a 100644 --- a/src/timers/AutoKick.ts +++ b/src/timers/AutoKick.ts @@ -33,8 +33,7 @@ export default async function AutoKick() { const now = new Date(); if (whenToKick < now) { - // await member.kick("Auto Kicked"); - console.log("Kicked"); + await member.kick("Auto Kicked"); if (config.NoticeChannelId) { const channel = guild.channels.cache.find(x => x.id == config.NoticeChannelId) || await guild.channels.fetch(config.NoticeChannelId); @@ -46,7 +45,7 @@ export default async function AutoKick() { const embed = new EmbedBuilder() .setTitle("Auto Kicked User") .setColor(EmbedColours.Danger) - .setThumbnail(member.avatarURL()) + .setThumbnail(member.user.avatarURL()) .addFields([ { name: "User", @@ -60,7 +59,7 @@ export default async function AutoKick() { }); } } else if (config.NoticeChannelId && config.NoticeTime) { - const whenToNotice = new Date(whenToKick.getMilliseconds() - config.NoticeTime); + const whenToNotice = new Date(whenToKick.getTime() - config.NoticeTime); const channel = guild.channels.cache.find(x => x.id == config.NoticeChannelId) || await guild.channels.fetch(config.NoticeChannelId); @@ -74,7 +73,7 @@ export default async function AutoKick() { const embed = new EmbedBuilder() .setTitle("Auto Kick Notice") .setColor(EmbedColours.Warning) - .setThumbnail(member.avatarURL()) + .setThumbnail(member.user.avatarURL()) .addFields([ { name: "User", @@ -83,7 +82,7 @@ export default async function AutoKick() { }, { name: "When To Kick", - value: ``, + value: ``, inline: true, }, ]); -- 2.45.2 From 95eab832505eb5ab7c0add300cdbab036a72c5c4 Mon Sep 17 00:00:00 2001 From: Ethan Lane Date: Sat, 28 Dec 2024 16:42:54 +0000 Subject: [PATCH 10/10] Stop timer from running on start --- src/client/client.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/client/client.ts b/src/client/client.ts index 59fb946..86b34bd 100644 --- a/src/client/client.ts +++ b/src/client/client.ts @@ -65,7 +65,7 @@ export class CoreClient extends Client { .then(() => { console.log("Data Source Initialized"); - this._timerHelper.AddTimer("0 * * * *", "Europe/London", AutoKick, true); + this._timerHelper.AddTimer("0 * * * *", "Europe/London", AutoKick, false); this._timerHelper.StartAllTimers(); }) -- 2.45.2