General fixes. Update Knub to 9.6.1. Add info and server commands.
This commit is contained in:
parent
7ded84b924
commit
5359d0d5fe
7 changed files with 245 additions and 24 deletions
6
package-lock.json
generated
6
package-lock.json
generated
|
@ -2181,9 +2181,9 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"knub": {
|
"knub": {
|
||||||
"version": "9.6.0",
|
"version": "9.6.1",
|
||||||
"resolved": "https://registry.npmjs.org/knub/-/knub-9.6.0.tgz",
|
"resolved": "https://registry.npmjs.org/knub/-/knub-9.6.1.tgz",
|
||||||
"integrity": "sha512-+a/woh8WnSxBkflNjCjvfGASadz80o/0Mot81K9sr8BvcITzeDtoOBaxzeiwCb5NWNtYz/Qp9M7ZZ6Jr5U45bg==",
|
"integrity": "sha512-Usydud/TYz8Il/8DLpMSTnAup3nespr3DLlPRnFzmO9mL0hQnCKtqAdp3WoDCmPb6uhqJ3zCe6AH66zRS7VC/w==",
|
||||||
"requires": {
|
"requires": {
|
||||||
"escape-string-regexp": "^1.0.5",
|
"escape-string-regexp": "^1.0.5",
|
||||||
"js-yaml": "^3.9.1",
|
"js-yaml": "^3.9.1",
|
||||||
|
|
|
@ -33,7 +33,7 @@
|
||||||
"escape-string-regexp": "^1.0.5",
|
"escape-string-regexp": "^1.0.5",
|
||||||
"humanize-duration": "^3.15.0",
|
"humanize-duration": "^3.15.0",
|
||||||
"knex": "^0.14.6",
|
"knex": "^0.14.6",
|
||||||
"knub": "^9.6.0",
|
"knub": "^9.6.1",
|
||||||
"lodash.at": "^4.6.0",
|
"lodash.at": "^4.6.0",
|
||||||
"lodash.difference": "^4.5.0",
|
"lodash.difference": "^4.5.0",
|
||||||
"lodash.intersection": "^4.4.0",
|
"lodash.intersection": "^4.4.0",
|
||||||
|
|
|
@ -6,8 +6,8 @@
|
||||||
"MEMBER_MUTE_EXPIRED": "🔉 **{member.user.username}#{member.user.discriminator}**'s mute expired",
|
"MEMBER_MUTE_EXPIRED": "🔉 **{member.user.username}#{member.user.discriminator}**'s mute expired",
|
||||||
"MEMBER_KICK": "👢 **{user.username}#{user.discriminator}** (`{user.id}`) was kicked by {mod.username}#{mod.discriminator}",
|
"MEMBER_KICK": "👢 **{user.username}#{user.discriminator}** (`{user.id}`) was kicked by {mod.username}#{mod.discriminator}",
|
||||||
"MEMBER_BAN": "🔨 **{user.username}#{user.discriminator}** (`{user.id}`) was banned by {mod.username}#{mod.discriminator}",
|
"MEMBER_BAN": "🔨 **{user.username}#{user.discriminator}** (`{user.id}`) was banned by {mod.username}#{mod.discriminator}",
|
||||||
"MEMBER_UNBAN": "🔓 **{user.username}#{user.discriminator}** (`{user.id}`) was unbanned by {mod.username}#{mod.discriminator}",
|
"MEMBER_UNBAN": "🔓 User (`{userId}`) was unbanned by {mod.username}#{mod.discriminator}",
|
||||||
"MEMBER_FORCEBAN": "🔨 User `{userId}` was forcebanned by {mod.username}#{mod.discriminator}",
|
"MEMBER_FORCEBAN": "🔨 User (`{userId}`) was forcebanned by {mod.username}#{mod.discriminator}",
|
||||||
"MEMBER_JOIN": "📥 **{member.user.username}#{member.user.discriminator}** (`{member.id}`) joined{new} (created {account_age} ago)",
|
"MEMBER_JOIN": "📥 **{member.user.username}#{member.user.discriminator}** (`{member.id}`) joined{new} (created {account_age} ago)",
|
||||||
"MEMBER_LEAVE": "📤 **{member.user.username}#{member.user.discriminator}** (`{member.id}`) left the server",
|
"MEMBER_LEAVE": "📤 **{member.user.username}#{member.user.discriminator}** (`{member.id}`) left the server",
|
||||||
"MEMBER_ROLE_ADD": "🔑 **{member.user.username}#{member.user.discriminator}** (`{member.id}`) role added **{role.name}** by {mod.username}#{mod.discriminator}",
|
"MEMBER_ROLE_ADD": "🔑 **{member.user.username}#{member.user.discriminator}** (`{member.id}`) role added **{role.name}** by {mod.username}#{mod.discriminator}",
|
||||||
|
|
|
@ -113,10 +113,14 @@ export class LogsPlugin extends Plugin {
|
||||||
);
|
);
|
||||||
const mod = relevantAuditLogEntry ? relevantAuditLogEntry.user : unknownUser;
|
const mod = relevantAuditLogEntry ? relevantAuditLogEntry.user : unknownUser;
|
||||||
|
|
||||||
this.serverLogs.log(LogType.MEMBER_BAN, {
|
this.serverLogs.log(
|
||||||
user: stripObjectToScalars(user),
|
LogType.MEMBER_BAN,
|
||||||
mod: stripObjectToScalars(mod)
|
{
|
||||||
});
|
user: stripObjectToScalars(user),
|
||||||
|
mod: stripObjectToScalars(mod)
|
||||||
|
},
|
||||||
|
user.id
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@d.event("guildBanRemove")
|
@d.event("guildBanRemove")
|
||||||
|
@ -128,10 +132,14 @@ export class LogsPlugin extends Plugin {
|
||||||
);
|
);
|
||||||
const mod = relevantAuditLogEntry ? relevantAuditLogEntry.user : unknownUser;
|
const mod = relevantAuditLogEntry ? relevantAuditLogEntry.user : unknownUser;
|
||||||
|
|
||||||
this.serverLogs.log(LogType.MEMBER_UNBAN, {
|
this.serverLogs.log(
|
||||||
user: stripObjectToScalars(user),
|
LogType.MEMBER_UNBAN,
|
||||||
mod: stripObjectToScalars(mod)
|
{
|
||||||
});
|
mod: stripObjectToScalars(mod),
|
||||||
|
userId: user.id
|
||||||
|
},
|
||||||
|
user.id
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@d.event("guildMemberUpdate")
|
@d.event("guildMemberUpdate")
|
||||||
|
|
|
@ -19,6 +19,17 @@ import { GuildLogs } from "../data/GuildLogs";
|
||||||
import { LogType } from "../data/LogType";
|
import { LogType } from "../data/LogType";
|
||||||
import Timer = NodeJS.Timer;
|
import Timer = NodeJS.Timer;
|
||||||
|
|
||||||
|
enum IgnoredEventType {
|
||||||
|
Ban = 1,
|
||||||
|
Unban,
|
||||||
|
Kick
|
||||||
|
}
|
||||||
|
|
||||||
|
interface IIgnoredEvent {
|
||||||
|
type: IgnoredEventType;
|
||||||
|
userId: string;
|
||||||
|
}
|
||||||
|
|
||||||
export class ModActionsPlugin extends Plugin {
|
export class ModActionsPlugin extends Plugin {
|
||||||
protected cases: GuildCases;
|
protected cases: GuildCases;
|
||||||
protected mutes: GuildMutes;
|
protected mutes: GuildMutes;
|
||||||
|
@ -26,11 +37,15 @@ export class ModActionsPlugin extends Plugin {
|
||||||
|
|
||||||
protected muteClearIntervalId: Timer;
|
protected muteClearIntervalId: Timer;
|
||||||
|
|
||||||
|
protected ignoredEvents: IIgnoredEvent[];
|
||||||
|
|
||||||
async onLoad() {
|
async onLoad() {
|
||||||
this.cases = new GuildCases(this.guildId);
|
this.cases = new GuildCases(this.guildId);
|
||||||
this.mutes = new GuildMutes(this.guildId);
|
this.mutes = new GuildMutes(this.guildId);
|
||||||
this.serverLogs = new GuildLogs(this.guildId);
|
this.serverLogs = new GuildLogs(this.guildId);
|
||||||
|
|
||||||
|
this.ignoredEvents = [];
|
||||||
|
|
||||||
// Check for expired mutes every 5s
|
// Check for expired mutes every 5s
|
||||||
this.clearExpiredMutes();
|
this.clearExpiredMutes();
|
||||||
this.muteClearIntervalId = setInterval(() => this.clearExpiredMutes(), 5000);
|
this.muteClearIntervalId = setInterval(() => this.clearExpiredMutes(), 5000);
|
||||||
|
@ -89,13 +104,37 @@ export class ModActionsPlugin extends Plugin {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ignoreEvent(type: IgnoredEventType, userId: any) {
|
||||||
|
this.ignoredEvents.push({ type, userId });
|
||||||
|
|
||||||
|
// Clear after expiry (15sec by default)
|
||||||
|
setTimeout(() => {
|
||||||
|
this.clearIgnoredEvent(type, userId);
|
||||||
|
}, 1000 * 15);
|
||||||
|
}
|
||||||
|
|
||||||
|
isEventIgnored(type: IgnoredEventType, userId: any) {
|
||||||
|
return this.ignoredEvents.some(info => type === info.type && userId === info.userId);
|
||||||
|
}
|
||||||
|
|
||||||
|
clearIgnoredEvent(type: IgnoredEventType, userId: any) {
|
||||||
|
this.ignoredEvents.splice(
|
||||||
|
this.ignoredEvents.findIndex(info => type === info.type && userId === info.userId),
|
||||||
|
1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Add a BAN action automatically when a user is banned.
|
* Add a BAN action automatically when a user is banned.
|
||||||
* Attempts to find the ban's details in the audit log.
|
* Attempts to find the ban's details in the audit log.
|
||||||
*/
|
*/
|
||||||
@d.event("guildBanAdd")
|
@d.event("guildBanAdd")
|
||||||
async onGuildBanAdd(guild: Guild, user: User) {
|
async onGuildBanAdd(guild: Guild, user: User) {
|
||||||
await sleep(1000); // Wait a moment for the audit log to update
|
if (this.isEventIgnored(IgnoredEventType.Ban, user.id)) {
|
||||||
|
this.clearIgnoredEvent(IgnoredEventType.Ban, user.id);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const relevantAuditLogEntry = await findRelevantAuditLogEntry(
|
const relevantAuditLogEntry = await findRelevantAuditLogEntry(
|
||||||
this.guild,
|
this.guild,
|
||||||
ErisConstants.AuditLogActions.MEMBER_BAN_ADD,
|
ErisConstants.AuditLogActions.MEMBER_BAN_ADD,
|
||||||
|
@ -125,6 +164,11 @@ export class ModActionsPlugin extends Plugin {
|
||||||
*/
|
*/
|
||||||
@d.event("guildBanRemove")
|
@d.event("guildBanRemove")
|
||||||
async onGuildBanRemove(guild: Guild, user: User) {
|
async onGuildBanRemove(guild: Guild, user: User) {
|
||||||
|
if (this.isEventIgnored(IgnoredEventType.Unban, user.id)) {
|
||||||
|
this.clearIgnoredEvent(IgnoredEventType.Unban, user.id);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const relevantAuditLogEntry = await findRelevantAuditLogEntry(
|
const relevantAuditLogEntry = await findRelevantAuditLogEntry(
|
||||||
this.guild,
|
this.guild,
|
||||||
ErisConstants.AuditLogActions.MEMBER_BAN_REMOVE,
|
ErisConstants.AuditLogActions.MEMBER_BAN_REMOVE,
|
||||||
|
@ -165,6 +209,11 @@ export class ModActionsPlugin extends Plugin {
|
||||||
|
|
||||||
@d.event("guildMemberRemove")
|
@d.event("guildMemberRemove")
|
||||||
async onGuildMemberRemove(_, member: Member) {
|
async onGuildMemberRemove(_, member: Member) {
|
||||||
|
if (this.isEventIgnored(IgnoredEventType.Kick, member.id)) {
|
||||||
|
this.clearIgnoredEvent(IgnoredEventType.Kick, member.id);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const kickAuditLogEntry = await findRelevantAuditLogEntry(
|
const kickAuditLogEntry = await findRelevantAuditLogEntry(
|
||||||
this.guild,
|
this.guild,
|
||||||
ErisConstants.AuditLogActions.MEMBER_KICK,
|
ErisConstants.AuditLogActions.MEMBER_KICK,
|
||||||
|
@ -398,6 +447,7 @@ export class ModActionsPlugin extends Plugin {
|
||||||
|
|
||||||
// Kick the user
|
// Kick the user
|
||||||
this.serverLogs.ignoreLog(LogType.MEMBER_KICK, args.member.id);
|
this.serverLogs.ignoreLog(LogType.MEMBER_KICK, args.member.id);
|
||||||
|
this.ignoreEvent(IgnoredEventType.Kick, args.member.id);
|
||||||
args.member.kick(args.reason);
|
args.member.kick(args.reason);
|
||||||
|
|
||||||
// Create a case for this action
|
// Create a case for this action
|
||||||
|
@ -442,6 +492,7 @@ export class ModActionsPlugin extends Plugin {
|
||||||
|
|
||||||
// Ban the user
|
// Ban the user
|
||||||
this.serverLogs.ignoreLog(LogType.MEMBER_BAN, args.member.id);
|
this.serverLogs.ignoreLog(LogType.MEMBER_BAN, args.member.id);
|
||||||
|
this.ignoreEvent(IgnoredEventType.Ban, args.member.id);
|
||||||
args.member.ban(1, args.reason);
|
args.member.ban(1, args.reason);
|
||||||
|
|
||||||
// Create a case for this action
|
// Create a case for this action
|
||||||
|
@ -462,9 +513,10 @@ export class ModActionsPlugin extends Plugin {
|
||||||
@d.command("unban", "<userId:string> [reason:string$]")
|
@d.command("unban", "<userId:string> [reason:string$]")
|
||||||
@d.permission("ban")
|
@d.permission("ban")
|
||||||
async unbanCmd(msg: Message, args: any) {
|
async unbanCmd(msg: Message, args: any) {
|
||||||
this.serverLogs.ignoreLog(LogType.MEMBER_UNBAN, args.member.id);
|
this.serverLogs.ignoreLog(LogType.MEMBER_UNBAN, args.userId);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
this.ignoreEvent(IgnoredEventType.Unban, args.userId);
|
||||||
await this.guild.unbanMember(args.userId, args.reason);
|
await this.guild.unbanMember(args.userId, args.reason);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
msg.channel.createMessage(errorMessage("Failed to unban member"));
|
msg.channel.createMessage(errorMessage("Failed to unban member"));
|
||||||
|
@ -496,7 +548,8 @@ export class ModActionsPlugin extends Plugin {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.serverLogs.ignoreLog(LogType.MEMBER_FORCEBAN, args.member.id);
|
this.ignoreEvent(IgnoredEventType.Ban, args.userId);
|
||||||
|
this.serverLogs.ignoreLog(LogType.MEMBER_BAN, args.userId);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await this.guild.banMember(args.userId, 1, args.reason);
|
await this.guild.banMember(args.userId, 1, args.reason);
|
||||||
|
|
|
@ -1,14 +1,26 @@
|
||||||
import { Plugin, decorators as d, reply } from "knub";
|
import { Plugin, decorators as d, reply } from "knub";
|
||||||
import { Channel, Message, TextChannel, User } from "eris";
|
import { Channel, Embed, EmbedOptions, Message, TextChannel, User, VoiceChannel } from "eris";
|
||||||
import { errorMessage, getMessages, stripObjectToScalars, successMessage } from "../utils";
|
import {
|
||||||
|
embedPadding,
|
||||||
|
errorMessage,
|
||||||
|
getMessages,
|
||||||
|
stripObjectToScalars,
|
||||||
|
successMessage,
|
||||||
|
trimLines
|
||||||
|
} from "../utils";
|
||||||
import { GuildLogs } from "../data/GuildLogs";
|
import { GuildLogs } from "../data/GuildLogs";
|
||||||
import { LogType } from "../data/LogType";
|
import { LogType } from "../data/LogType";
|
||||||
|
import moment from "moment-timezone";
|
||||||
|
import humanizeDuration from "humanize-duration";
|
||||||
|
import { GuildCases } from "../data/GuildCases";
|
||||||
|
import { CaseType } from "../data/CaseType";
|
||||||
|
|
||||||
const MAX_SEARCH_RESULTS = 15;
|
const MAX_SEARCH_RESULTS = 15;
|
||||||
const MAX_CLEAN_COUNT = 50;
|
const MAX_CLEAN_COUNT = 50;
|
||||||
|
|
||||||
export class UtilityPlugin extends Plugin {
|
export class UtilityPlugin extends Plugin {
|
||||||
protected logs: GuildLogs;
|
protected logs: GuildLogs;
|
||||||
|
protected cases: GuildCases;
|
||||||
|
|
||||||
getDefaultOptions() {
|
getDefaultOptions() {
|
||||||
return {
|
return {
|
||||||
|
@ -17,7 +29,8 @@ export class UtilityPlugin extends Plugin {
|
||||||
level: false,
|
level: false,
|
||||||
search: false,
|
search: false,
|
||||||
clean: false,
|
clean: false,
|
||||||
info: false
|
info: false,
|
||||||
|
server: true
|
||||||
},
|
},
|
||||||
overrides: [
|
overrides: [
|
||||||
{
|
{
|
||||||
|
@ -27,7 +40,8 @@ export class UtilityPlugin extends Plugin {
|
||||||
level: true,
|
level: true,
|
||||||
search: true,
|
search: true,
|
||||||
clean: true,
|
clean: true,
|
||||||
info: true
|
info: true,
|
||||||
|
server: true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
@ -36,6 +50,7 @@ export class UtilityPlugin extends Plugin {
|
||||||
|
|
||||||
onLoad() {
|
onLoad() {
|
||||||
this.logs = new GuildLogs(this.guildId);
|
this.logs = new GuildLogs(this.guildId);
|
||||||
|
this.cases = new GuildCases(this.guildId);
|
||||||
}
|
}
|
||||||
|
|
||||||
@d.command("roles")
|
@d.command("roles")
|
||||||
|
@ -122,8 +137,9 @@ export class UtilityPlugin extends Plugin {
|
||||||
m => m.id !== msg.id,
|
m => m.id !== msg.id,
|
||||||
args.count
|
args.count
|
||||||
);
|
);
|
||||||
if (messagesToClean.length > 0)
|
if (messagesToClean.length > 0) {
|
||||||
await this.cleanMessages(msg.channel, messagesToClean.map(m => m.id), msg.author);
|
await this.cleanMessages(msg.channel, messagesToClean.map(m => m.id), msg.author);
|
||||||
|
}
|
||||||
|
|
||||||
msg.channel.createMessage(
|
msg.channel.createMessage(
|
||||||
successMessage(
|
successMessage(
|
||||||
|
@ -147,8 +163,9 @@ export class UtilityPlugin extends Plugin {
|
||||||
m => m.id !== msg.id && m.author.id === args.userId,
|
m => m.id !== msg.id && m.author.id === args.userId,
|
||||||
args.count
|
args.count
|
||||||
);
|
);
|
||||||
if (messagesToClean.length > 0)
|
if (messagesToClean.length > 0) {
|
||||||
await this.cleanMessages(msg.channel, messagesToClean.map(m => m.id), msg.author);
|
await this.cleanMessages(msg.channel, messagesToClean.map(m => m.id), msg.author);
|
||||||
|
}
|
||||||
|
|
||||||
msg.channel.createMessage(
|
msg.channel.createMessage(
|
||||||
successMessage(
|
successMessage(
|
||||||
|
@ -172,8 +189,9 @@ export class UtilityPlugin extends Plugin {
|
||||||
m => m.id !== msg.id && m.author.bot,
|
m => m.id !== msg.id && m.author.bot,
|
||||||
args.count
|
args.count
|
||||||
);
|
);
|
||||||
if (messagesToClean.length > 0)
|
if (messagesToClean.length > 0) {
|
||||||
await this.cleanMessages(msg.channel, messagesToClean.map(m => m.id), msg.author);
|
await this.cleanMessages(msg.channel, messagesToClean.map(m => m.id), msg.author);
|
||||||
|
}
|
||||||
|
|
||||||
msg.channel.createMessage(
|
msg.channel.createMessage(
|
||||||
successMessage(
|
successMessage(
|
||||||
|
@ -181,4 +199,134 @@ export class UtilityPlugin extends Plugin {
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@d.command("info", "<userId:userId>")
|
||||||
|
@d.permission("info")
|
||||||
|
async infoCmd(msg: Message, args: { userId: string }) {
|
||||||
|
const embed: EmbedOptions = {
|
||||||
|
fields: []
|
||||||
|
};
|
||||||
|
|
||||||
|
const user = this.bot.users.get(args.userId);
|
||||||
|
if (user) {
|
||||||
|
const createdAt = moment(user.createdAt);
|
||||||
|
const accountAge = humanizeDuration(moment().valueOf() - user.createdAt, {
|
||||||
|
largest: 2,
|
||||||
|
round: true
|
||||||
|
});
|
||||||
|
|
||||||
|
embed.title = `${user.username}#${user.discriminator}`;
|
||||||
|
embed.thumbnail = { url: user.avatarURL };
|
||||||
|
|
||||||
|
embed.fields.push({
|
||||||
|
name: "User information",
|
||||||
|
value:
|
||||||
|
trimLines(`
|
||||||
|
ID: ${user.id}
|
||||||
|
Profile: <@!${user.id}>
|
||||||
|
Created: ${accountAge} ago (${createdAt.format("YYYY-MM-DD[T]HH:mm:ss")})
|
||||||
|
`) + embedPadding
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
embed.title = `Unknown user`;
|
||||||
|
}
|
||||||
|
|
||||||
|
const member = this.guild.members.get(args.userId);
|
||||||
|
if (member) {
|
||||||
|
const joinedAt = moment(member.joinedAt);
|
||||||
|
const joinAge = humanizeDuration(moment().valueOf() - member.joinedAt, {
|
||||||
|
largest: 2,
|
||||||
|
round: true
|
||||||
|
});
|
||||||
|
const roles = member.roles.map(id => this.guild.roles.get(id));
|
||||||
|
|
||||||
|
embed.fields.push({
|
||||||
|
name: "Member information",
|
||||||
|
value:
|
||||||
|
trimLines(`
|
||||||
|
Joined: ${joinAge} ago (${joinedAt.format("YYYY-MM-DD[T]HH:mm:ss")})
|
||||||
|
${roles.length > 0 ? "Roles: " + roles.map(r => r.name).join(", ") : ""}
|
||||||
|
`) + embedPadding
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const cases = await this.cases.getByUserId(args.userId);
|
||||||
|
if (cases.length > 0) {
|
||||||
|
cases.sort((a, b) => {
|
||||||
|
return a.created_at < b.created_at ? -1 : 1;
|
||||||
|
});
|
||||||
|
|
||||||
|
const caseSummaries = cases.map(c => {
|
||||||
|
return `${CaseType[c.type]} (#${c.case_number})`;
|
||||||
|
});
|
||||||
|
|
||||||
|
embed.fields.push({
|
||||||
|
name: "Cases",
|
||||||
|
value: trimLines(`
|
||||||
|
Total cases: ${cases.length}
|
||||||
|
Summary: ${caseSummaries.join(", ")}
|
||||||
|
`)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
msg.channel.createMessage({ embed });
|
||||||
|
}
|
||||||
|
|
||||||
|
@d.command("server")
|
||||||
|
@d.permission("server")
|
||||||
|
async serverCmd(msg: Message) {
|
||||||
|
await this.guild.fetchAllMembers();
|
||||||
|
|
||||||
|
const embed: EmbedOptions = {
|
||||||
|
fields: []
|
||||||
|
};
|
||||||
|
|
||||||
|
embed.thumbnail = { url: this.guild.iconURL };
|
||||||
|
|
||||||
|
const createdAt = moment(this.guild.createdAt);
|
||||||
|
const serverAge = humanizeDuration(moment().valueOf() - this.guild.createdAt, {
|
||||||
|
largest: 2,
|
||||||
|
round: true
|
||||||
|
});
|
||||||
|
|
||||||
|
embed.fields.push({
|
||||||
|
name: "Server information",
|
||||||
|
value:
|
||||||
|
trimLines(`
|
||||||
|
Created: ${serverAge} ago (${createdAt.format("YYYY-MM-DD[T]HH:mm:ss")})
|
||||||
|
Members: ${this.guild.memberCount}
|
||||||
|
${this.guild.features.length > 0 ? "Features: " + this.guild.features.join(", ") : ""}
|
||||||
|
`) + embedPadding
|
||||||
|
});
|
||||||
|
|
||||||
|
const textChannels = this.guild.channels.filter(channel => channel instanceof TextChannel);
|
||||||
|
const voiceChannels = this.guild.channels.filter(channel => channel instanceof VoiceChannel);
|
||||||
|
|
||||||
|
embed.fields.push({
|
||||||
|
name: "Counts",
|
||||||
|
value:
|
||||||
|
trimLines(`
|
||||||
|
Roles: ${this.guild.roles.size}
|
||||||
|
Text channels: ${textChannels.length}
|
||||||
|
Voice channels: ${voiceChannels.length}
|
||||||
|
`) + embedPadding
|
||||||
|
});
|
||||||
|
|
||||||
|
const onlineMembers = this.guild.members.filter(m => m.status === "online");
|
||||||
|
const dndMembers = this.guild.members.filter(m => m.status === "dnd");
|
||||||
|
const idleMembers = this.guild.members.filter(m => m.status === "idle");
|
||||||
|
const offlineMembers = this.guild.members.filter(m => m.status === "offline");
|
||||||
|
|
||||||
|
embed.fields.push({
|
||||||
|
name: "Members",
|
||||||
|
value: trimLines(`
|
||||||
|
Online: **${onlineMembers.length}**
|
||||||
|
Idle: **${idleMembers.length}**
|
||||||
|
DND: **${dndMembers.length}**
|
||||||
|
Offline: **${offlineMembers.length}**
|
||||||
|
`)
|
||||||
|
});
|
||||||
|
|
||||||
|
msg.channel.createMessage({ embed });
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
12
src/utils.ts
12
src/utils.ts
|
@ -226,3 +226,15 @@ export async function cleanMessagesInChannel(
|
||||||
await bot.deleteMessages(channel.id, ids, reason);
|
await bot.deleteMessages(channel.id, ids, reason);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function trimLines(str: string) {
|
||||||
|
return str
|
||||||
|
.trim()
|
||||||
|
.split("\n")
|
||||||
|
.map(l => l.trim())
|
||||||
|
.join("\n")
|
||||||
|
.trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
export const emptyEmbedValue = "\u200b";
|
||||||
|
export const embedPadding = "\n" + emptyEmbedValue;
|
||||||
|
|
Loading…
Add table
Reference in a new issue