Add auto kick functionality #502
16 changed files with 467 additions and 4 deletions
|
@ -7,7 +7,7 @@
|
||||||
# any secret values.
|
# any secret values.
|
||||||
|
|
||||||
BOT_TOKEN=
|
BOT_TOKEN=
|
||||||
BOT_VER=3.2.3
|
BOT_VER=3.2.4
|
||||||
BOT_AUTHOR=Vylpes
|
BOT_AUTHOR=Vylpes
|
||||||
BOT_OWNERID=147392775707426816
|
BOT_OWNERID=147392775707426816
|
||||||
BOT_CLIENTID=682942374040961060
|
BOT_CLIENTID=682942374040961060
|
||||||
|
@ -23,4 +23,6 @@ DB_NAME=vylbot
|
||||||
DB_AUTH_USER=dev
|
DB_AUTH_USER=dev
|
||||||
DB_AUTH_PASS=dev
|
DB_AUTH_PASS=dev
|
||||||
DB_SYNC=true
|
DB_SYNC=true
|
||||||
DB_LOGGING=true
|
DB_LOGGING=true
|
||||||
|
DB_DATA_LOCATION=./.temp/database
|
||||||
|
DB_ROOT_HOST=0.0.0.0
|
|
@ -0,0 +1 @@
|
||||||
|
DROP TABLE auto_kick_config;
|
|
@ -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
|
||||||
|
);
|
|
@ -0,0 +1,2 @@
|
||||||
|
ALTER TABLE auto_kick_config
|
||||||
|
ADD PRIMARY KEY (Id);
|
|
@ -30,6 +30,7 @@
|
||||||
"@discordjs/rest": "^2.0.0",
|
"@discordjs/rest": "^2.0.0",
|
||||||
"@types/jest": "^29.0.0",
|
"@types/jest": "^29.0.0",
|
||||||
"@types/uuid": "^9.0.0",
|
"@types/uuid": "^9.0.0",
|
||||||
|
"cron": "^3.3.1",
|
||||||
"discord.js": "^14.3.0",
|
"discord.js": "^14.3.0",
|
||||||
"dotenv": "^16.0.0",
|
"dotenv": "^16.0.0",
|
||||||
"emoji-regex": "^10.0.0",
|
"emoji-regex": "^10.0.0",
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
import { Client, Partials } from "discord.js";
|
import { Client, Partials } from "discord.js";
|
||||||
import * as dotenv from "dotenv";
|
import * as dotenv from "dotenv";
|
||||||
import { createConnection } from "typeorm";
|
|
||||||
import { EventType } from "../constants/EventType";
|
import { EventType } from "../constants/EventType";
|
||||||
import ICommandItem from "../contracts/ICommandItem";
|
import ICommandItem from "../contracts/ICommandItem";
|
||||||
import IEventItem from "../contracts/IEventItem";
|
import IEventItem from "../contracts/IEventItem";
|
||||||
|
@ -12,14 +11,18 @@ import AppDataSource from "../database/dataSources/appDataSource";
|
||||||
import ButtonEventItem from "../contracts/ButtonEventItem";
|
import ButtonEventItem from "../contracts/ButtonEventItem";
|
||||||
import { ButtonEvent } from "../type/buttonEvent";
|
import { ButtonEvent } from "../type/buttonEvent";
|
||||||
import CacheHelper from "../helpers/CacheHelper";
|
import CacheHelper from "../helpers/CacheHelper";
|
||||||
|
import TimerHelper from "../helpers/TimerHelper";
|
||||||
|
import AutoKick from "../timers/AutoKick";
|
||||||
|
|
||||||
export class CoreClient extends Client {
|
export class CoreClient extends Client {
|
||||||
private static _commandItems: ICommandItem[];
|
private static _commandItems: ICommandItem[];
|
||||||
private static _eventItems: IEventItem[];
|
private static _eventItems: IEventItem[];
|
||||||
private static _buttonEvents: ButtonEventItem[];
|
private static _buttonEvents: ButtonEventItem[];
|
||||||
|
private static _baseClient: Client;
|
||||||
|
|
||||||
private _events: Events;
|
private _events: Events;
|
||||||
private _util: Util;
|
private _util: Util;
|
||||||
|
private _timerHelper: TimerHelper;
|
||||||
|
|
||||||
public static get commandItems(): ICommandItem[] {
|
public static get commandItems(): ICommandItem[] {
|
||||||
return this._commandItems;
|
return this._commandItems;
|
||||||
|
@ -33,6 +36,10 @@ export class CoreClient extends Client {
|
||||||
return this._buttonEvents;
|
return this._buttonEvents;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static get baseClient(): Client {
|
||||||
|
return this._baseClient;
|
||||||
|
}
|
||||||
|
|
||||||
constructor(intents: number[], partials: Partials[]) {
|
constructor(intents: number[], partials: Partials[]) {
|
||||||
super({ intents: intents, partials: partials });
|
super({ intents: intents, partials: partials });
|
||||||
dotenv.config();
|
dotenv.config();
|
||||||
|
@ -43,6 +50,7 @@ export class CoreClient extends Client {
|
||||||
|
|
||||||
this._events = new Events();
|
this._events = new Events();
|
||||||
this._util = new Util();
|
this._util = new Util();
|
||||||
|
this._timerHelper = new TimerHelper();
|
||||||
}
|
}
|
||||||
|
|
||||||
public async start() {
|
public async start() {
|
||||||
|
@ -51,8 +59,16 @@ export class CoreClient extends Client {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
CoreClient._baseClient = this;
|
||||||
|
|
||||||
await AppDataSource.initialize()
|
await AppDataSource.initialize()
|
||||||
.then(() => console.log("Data Source Initialized"))
|
.then(() => {
|
||||||
|
console.log("Data Source Initialized");
|
||||||
|
|
||||||
|
this._timerHelper.AddTimer("0 * * * *", "Europe/London", AutoKick, false);
|
||||||
|
|
||||||
|
this._timerHelper.StartAllTimers();
|
||||||
|
})
|
||||||
.catch((err) => console.error("Error Initialising Data Source", err));
|
.catch((err) => console.error("Error Initialising Data Source", err));
|
||||||
|
|
||||||
super.on("interactionCreate", this._events.onInteractionCreate);
|
super.on("interactionCreate", this._events.onInteractionCreate);
|
||||||
|
|
113
src/commands/autokick.ts
Normal file
113
src/commands/autokick.ts
Normal file
|
@ -0,0 +1,113 @@
|
||||||
|
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() {
|
||||||
|
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: 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: ChatInputCommandInteraction) {
|
||||||
|
if (!interaction.guildId) return;
|
||||||
|
|
||||||
|
await AutoKickHelper.UnsetSetting(interaction.guildId);
|
||||||
|
|
||||||
|
await interaction.reply("Unset the auto kick configuration for this server");
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,3 +1,5 @@
|
||||||
export default class EmbedColours {
|
export default class EmbedColours {
|
||||||
public static readonly Ok = 0x3050ba;
|
public static readonly Ok = 0x3050ba;
|
||||||
|
public static readonly Warning = 0xffbf00;
|
||||||
|
public static readonly Danger = 0xd2042d;
|
||||||
}
|
}
|
61
src/database/entities/AutoKickConfig.ts
Normal file
61
src/database/entities/AutoKickConfig.ts
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
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<AutoKickConfig | null> {
|
||||||
|
const repository = AppDataSource.getRepository(AutoKickConfig);
|
||||||
|
|
||||||
|
const query = repository
|
||||||
|
.createQueryBuilder("config")
|
||||||
|
.where("config.serverId = :serverId", { serverId })
|
||||||
|
.andWhere("config.roleId = :roleId", { roleId })
|
||||||
|
.getOne();
|
||||||
|
|
||||||
|
return query;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static async FetchAllByServerId(serverId: string): Promise<AutoKickConfig[]> {
|
||||||
|
const repository = AppDataSource.getRepository(AutoKickConfig);
|
||||||
|
|
||||||
|
const query = repository
|
||||||
|
.createQueryBuilder("config")
|
||||||
|
.where("config.serverId = :serverId", { serverId })
|
||||||
|
.getMany();
|
||||||
|
|
||||||
|
return query;
|
||||||
|
}
|
||||||
|
}
|
|
@ -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<void> {
|
||||||
|
MigrationHelper.Up("1732973911304-createAutoKickConfig", "3.2.4", [
|
||||||
|
"01-AutoKickConfig-Table",
|
||||||
|
"02-AutoKickConfig-Key",
|
||||||
|
], queryRunner)
|
||||||
|
}
|
||||||
|
|
||||||
|
public async down(queryRunner: QueryRunner): Promise<void> {
|
||||||
|
MigrationHelper.Down("1732973911304-createAutoKickConfig", "3.2.4", [
|
||||||
|
"01-AutoKickConfig",
|
||||||
|
], queryRunner)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
37
src/helpers/AutoKickHelper.ts
Normal file
37
src/helpers/AutoKickHelper.ts
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
import AutoKickConfig from "../database/entities/AutoKickConfig";
|
||||||
|
|
||||||
|
export default class AutoKickHelper {
|
||||||
|
public static async GetSetting(serverId: string): Promise<AutoKickConfig | null> {
|
||||||
|
const configs = await AutoKickConfig.FetchAllByServerId(serverId);
|
||||||
|
|
||||||
|
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 UnsetSetting(serverId: string) {
|
||||||
|
const configs = await AutoKickConfig.FetchAllByServerId(serverId);
|
||||||
|
|
||||||
|
for (let config of configs) {
|
||||||
|
await AutoKickConfig.Remove(AutoKickConfig, config);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
81
src/helpers/TimerHelper.ts
Normal file
81
src/helpers/TimerHelper.ts
Normal file
|
@ -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<string, primitive>;
|
||||||
|
onTick: ((context: Map<string, primitive>) => void) | ((context: Map<string, primitive>) => Promise<void>);
|
||||||
|
runOnStart: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default class TimerHelper {
|
||||||
|
private _timers: Timer[];
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
this._timers = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
public AddTimer(
|
||||||
|
cronTime: string,
|
||||||
|
timeZone: string,
|
||||||
|
onTick: ((context: Map<string, primitive>) => void) | ((context: Map<string, primitive>) => Promise<void>),
|
||||||
|
runOnStart: boolean = false): string {
|
||||||
|
const context = new Map<string, primitive>();
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -4,6 +4,7 @@ import { EventType } from "./constants/EventType";
|
||||||
// Command Imports
|
// Command Imports
|
||||||
import About from "./commands/about";
|
import About from "./commands/about";
|
||||||
import Audits from "./commands/audits";
|
import Audits from "./commands/audits";
|
||||||
|
import Autokick from "./commands/autokick";
|
||||||
import Ban from "./commands/ban";
|
import Ban from "./commands/ban";
|
||||||
import Bunny from "./commands/bunny";
|
import Bunny from "./commands/bunny";
|
||||||
import Clear from "./commands/clear";
|
import Clear from "./commands/clear";
|
||||||
|
@ -45,6 +46,7 @@ export default class Registry {
|
||||||
public static RegisterCommands() {
|
public static RegisterCommands() {
|
||||||
CoreClient.RegisterCommand("about", new About());
|
CoreClient.RegisterCommand("about", new About());
|
||||||
CoreClient.RegisterCommand("audits", new Audits());
|
CoreClient.RegisterCommand("audits", new Audits());
|
||||||
|
CoreClient.RegisterCommand("autokick", new Autokick());
|
||||||
CoreClient.RegisterCommand("ban", new Ban());
|
CoreClient.RegisterCommand("ban", new Ban());
|
||||||
CoreClient.RegisterCommand("bunny", new Bunny());
|
CoreClient.RegisterCommand("bunny", new Bunny());
|
||||||
CoreClient.RegisterCommand("clear", new Clear());
|
CoreClient.RegisterCommand("clear", new Clear());
|
||||||
|
|
97
src/timers/AutoKick.ts
Normal file
97
src/timers/AutoKick.ts
Normal file
|
@ -0,0 +1,97 @@
|
||||||
|
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;
|
||||||
|
const autoKickConfigs = await AutoKickConfig.FetchAll(AutoKickConfig);
|
||||||
|
|
||||||
|
for (let config of autoKickConfigs) {
|
||||||
|
const guild = client.guilds.cache.find(x => x.id == config.ServerId) || await client.guilds.fetch(config.ServerId);
|
||||||
|
|
||||||
|
if (!guild) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
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");
|
||||||
|
|
||||||
|
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.user.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.getTime() - 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.user.avatarURL())
|
||||||
|
.addFields([
|
||||||
|
{
|
||||||
|
name: "User",
|
||||||
|
value: `<@${member.user.id}> \`${member.user.username}\``,
|
||||||
|
inline: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "When To Kick",
|
||||||
|
value: `<t:${Math.round(whenToKick.getTime() / 1000)}:R>`,
|
||||||
|
inline: true,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
await channel.send({
|
||||||
|
embeds: [ embed ],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
1
src/type/primitive.ts
Normal file
1
src/type/primitive.ts
Normal file
|
@ -0,0 +1 @@
|
||||||
|
export type primitive = string | number | boolean;
|
18
yarn.lock
18
yarn.lock
|
@ -788,6 +788,11 @@
|
||||||
expect "^29.0.0"
|
expect "^29.0.0"
|
||||||
pretty-format "^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@*":
|
"@types/node@*":
|
||||||
version "22.7.5"
|
version "22.7.5"
|
||||||
resolved "https://registry.npmjs.org/@types/node/-/node-22.7.5.tgz"
|
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"
|
jest-util "^29.7.0"
|
||||||
prompts "^2.0.1"
|
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:
|
cross-spawn@^7.0.0, cross-spawn@^7.0.3:
|
||||||
version "7.0.3"
|
version "7.0.3"
|
||||||
resolved "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz"
|
resolved "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz"
|
||||||
|
@ -3076,6 +3089,11 @@ lru-cache@^5.1.1:
|
||||||
dependencies:
|
dependencies:
|
||||||
yallist "^3.0.2"
|
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:
|
magic-bytes.js@^1.10.0:
|
||||||
version "1.10.0"
|
version "1.10.0"
|
||||||
resolved "https://registry.npmjs.org/magic-bytes.js/-/magic-bytes.js-1.10.0.tgz"
|
resolved "https://registry.npmjs.org/magic-bytes.js/-/magic-bytes.js-1.10.0.tgz"
|
||||||
|
|
Loading…
Reference in a new issue