156 lines
5.1 KiB
TypeScript
156 lines
5.1 KiB
TypeScript
import { APIEmbed, MessageMentionTypes, Snowflake } from "discord.js";
|
|
import { GuildPluginData } from "knub";
|
|
import { allowTimeout } from "../../../RegExpRunner";
|
|
import { LogType } from "../../../data/LogType";
|
|
import { TypedTemplateSafeValueContainer } from "../../../templateFormatter";
|
|
import { isDiscordAPIError, MINUTES, zSnowflake } from "../../../utils";
|
|
import { MessageBuffer } from "../../../utils/MessageBuffer";
|
|
import { InternalPosterPlugin } from "../../InternalPoster/InternalPosterPlugin";
|
|
import { ILogTypeData, LogsPluginType, TLogChannel, TLogChannelMap } from "../types";
|
|
import { getLogMessage } from "./getLogMessage";
|
|
|
|
interface ExclusionData {
|
|
userId?: Snowflake | null;
|
|
bot?: boolean | null;
|
|
roles?: Snowflake[] | null;
|
|
channel?: Snowflake | null;
|
|
category?: Snowflake | null;
|
|
thread?: Snowflake | null;
|
|
messageTextContent?: string | null;
|
|
}
|
|
|
|
const DEFAULT_BATCH_TIME = 1000;
|
|
const MIN_BATCH_TIME = 250;
|
|
const MAX_BATCH_TIME = 5000;
|
|
|
|
async function shouldExclude(
|
|
pluginData: GuildPluginData<LogsPluginType>,
|
|
opts: TLogChannel,
|
|
exclusionData: ExclusionData,
|
|
): Promise<boolean> {
|
|
if (
|
|
opts.excluded_users &&
|
|
exclusionData.userId &&
|
|
opts.excluded_users.includes(zSnowflake.parse(exclusionData.userId))
|
|
) {
|
|
return true;
|
|
}
|
|
|
|
if (opts.exclude_bots && exclusionData.bot) {
|
|
return true;
|
|
}
|
|
|
|
if (opts.excluded_roles && exclusionData.roles) {
|
|
for (const role of exclusionData.roles) {
|
|
if (opts.excluded_roles.includes(await zSnowflake.parseAsync(role))) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (
|
|
opts.excluded_channels &&
|
|
exclusionData.channel &&
|
|
opts.excluded_channels.includes(await zSnowflake.parseAsync(exclusionData.channel))
|
|
) {
|
|
return true;
|
|
}
|
|
|
|
if (
|
|
opts.excluded_categories &&
|
|
exclusionData.category &&
|
|
opts.excluded_categories.includes(await zSnowflake.parseAsync(exclusionData.category))
|
|
) {
|
|
return true;
|
|
}
|
|
|
|
if (
|
|
opts.excluded_threads &&
|
|
exclusionData.thread &&
|
|
opts.excluded_threads.includes(await zSnowflake.parseAsync(exclusionData.thread))
|
|
) {
|
|
return true;
|
|
}
|
|
|
|
if (opts.excluded_message_regexes && exclusionData.messageTextContent) {
|
|
for (const regex of opts.excluded_message_regexes) {
|
|
const matches = await pluginData.state.regexRunner
|
|
.exec(regex, exclusionData.messageTextContent)
|
|
.catch(allowTimeout);
|
|
if (matches) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
export async function log<TLogType extends keyof ILogTypeData>(
|
|
pluginData: GuildPluginData<LogsPluginType>,
|
|
type: TLogType,
|
|
data: TypedTemplateSafeValueContainer<ILogTypeData[TLogType]>,
|
|
exclusionData: ExclusionData = {},
|
|
) {
|
|
const logChannels: TLogChannelMap = pluginData.config.get().channels;
|
|
const typeStr = LogType[type];
|
|
|
|
for (const [channelId, opts] of Object.entries(logChannels)) {
|
|
const channel = pluginData.guild.channels.cache.get(channelId as Snowflake);
|
|
if (!channel?.isTextBased()) continue;
|
|
if (pluginData.state.channelCooldowns.isOnCooldown(channelId)) continue;
|
|
if (opts!.include?.length && !opts!.include.includes(typeStr)) continue;
|
|
if (opts!.exclude && opts!.exclude.includes(typeStr)) continue;
|
|
if (await shouldExclude(pluginData, opts!, exclusionData)) continue;
|
|
|
|
const message = await getLogMessage(pluginData, type, data, {
|
|
format: opts!.format,
|
|
include_embed_timestamp: opts!.include_embed_timestamp,
|
|
timestamp_format: opts!.timestamp_format,
|
|
});
|
|
if (!message) return;
|
|
|
|
// Initialize message buffer for this channel
|
|
if (!pluginData.state.buffers.has(channelId)) {
|
|
const batchTime = Math.min(Math.max(opts!.batch_time ?? DEFAULT_BATCH_TIME, MIN_BATCH_TIME), MAX_BATCH_TIME);
|
|
const internalPosterPlugin = pluginData.getPlugin(InternalPosterPlugin);
|
|
pluginData.state.buffers.set(
|
|
channelId,
|
|
new MessageBuffer({
|
|
timeout: batchTime,
|
|
textSeparator: "\n",
|
|
consume: (part) => {
|
|
const parse: MessageMentionTypes[] = pluginData.config.get().allow_user_mentions ? ["users"] : [];
|
|
internalPosterPlugin
|
|
.sendMessage(channel, {
|
|
...part,
|
|
allowedMentions: { parse },
|
|
})
|
|
.catch((err) => {
|
|
if (isDiscordAPIError(err)) {
|
|
// Missing Access / Missing Permissions
|
|
// TODO: Show/log this somewhere
|
|
if (err.code === 50001 || err.code === 50013) {
|
|
pluginData.state.channelCooldowns.setCooldown(channelId, 2 * MINUTES);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// tslint:disable-next-line:no-console
|
|
console.warn(
|
|
`Error while sending ${typeStr} log to ${pluginData.guild.id}/${channelId}: ${err.message}`,
|
|
);
|
|
});
|
|
},
|
|
}),
|
|
);
|
|
}
|
|
|
|
// Add log message to buffer
|
|
const buffer = pluginData.state.buffers.get(channelId)!;
|
|
buffer.push({
|
|
content: typeof message === "string" ? message : message.content || "",
|
|
embeds: typeof message === "string" ? [] : ((message.embeds || []) as APIEmbed[]),
|
|
});
|
|
}
|
|
}
|