refraction/src/index.ts
2022-06-14 12:38:21 +02:00

170 lines
3.8 KiB
TypeScript

import {
Client,
Intents,
Message,
MessageEmbed,
type MessageEmbedOptions,
} from 'discord.js';
import * as BuildConfig from './constants';
import { commands } from './commands';
import { filterMessage } from './filters';
import { parseLog } from './logs';
import {
parse as discordParse,
type SuccessfulParsedMessage,
} from 'discord-command-parser';
import random from 'just-random';
import { readFile } from 'fs/promises';
import { join } from 'path';
import { green, bold, blue, underline, yellow } from 'kleur/colors';
import 'dotenv/config';
import { getLatestMinecraftVersion } from './utils/remoteVersions';
export interface Command {
name: string;
aliases?: string[];
desc?: string;
examples?: string[];
exec(
m: Message,
p: SuccessfulParsedMessage<Message<boolean>>
): Promise<void> | void;
}
interface Tag {
name: string;
aliases?: Array<string>;
text?: string;
embed?: MessageEmbedOptions;
}
export const getTags = async (): Promise<Tag[]> => {
return JSON.parse(
await readFile(join(__dirname, 'tags.json'), { encoding: 'utf8' })
);
};
const client = new Client({
intents: [
Intents.FLAGS.GUILDS,
Intents.FLAGS.GUILD_MESSAGES,
Intents.FLAGS.DIRECT_MESSAGES,
Intents.FLAGS.GUILD_MEMBERS,
Intents.FLAGS.GUILD_PRESENCES,
Intents.FLAGS.GUILD_MESSAGE_REACTIONS,
Intents.FLAGS.GUILD_BANS,
],
});
client.once('ready', async () => {
console.log(green('Discord bot ready!'));
if (process.env.NODE_ENV !== 'development')
console.warn(yellow(bold('Running in production mode!')));
console.log(
'Invite link:',
blue(
underline(
client.generateInvite({
scopes: ['bot'],
permissions: ['ADMINISTRATOR'],
})
)
)
);
client.user?.presence.set({
activities: [{ name: `Minecraft ${await getLatestMinecraftVersion()}` }],
status: 'online',
});
client.on('messageCreate', async (e) => {
if (!e.content) return;
if (!e.channel.isText()) return;
if (e.author === client.user) return;
if (
process.env.NODE_ENV === 'development' &&
e.channelId !== BuildConfig.DEBUG_CHANNEL_ID
) {
return;
} else if (
process.env.NODE_ENV !== 'development' &&
e.channelId === BuildConfig.DEBUG_CHANNEL_ID
) {
return;
}
const messageIsOK = await filterMessage(e);
if (!messageIsOK) {
return;
}
if (e.cleanContent.match(BuildConfig.ETA_REGEX)) {
await e.reply(
`${random(BuildConfig.ETA_MESSAGES)} <:pofat:964546613194420294>`
);
}
const commanded = await parseMsg(e);
if (commanded) return;
const log = await parseLog(e.content);
if (log != null) {
e.reply({ embeds: [log] });
return;
}
});
});
async function parseMsg(e: Message) {
const parsed = discordParse(e, '!', {
allowBots: true,
});
if (!parsed.success) return false;
const cmd = commands.find(
(c) => c.name == parsed.command || c.aliases?.includes(parsed.command)
);
if (!cmd) {
// TODO: Do not read tags.json everytime there is a new message
const tag = await getTags().then((r) =>
r.find(
(t) => t.name == parsed.command || t.aliases?.includes(parsed.command)
)
);
if (tag) {
if (tag.text) {
e.reply(tag.text);
} else if (tag.embed) {
const em = new MessageEmbed(tag.embed);
e.reply({ embeds: [em] });
}
return true;
}
return false;
}
try {
await cmd.exec(e, parsed);
} catch (err: unknown) {
const em = new MessageEmbed()
.setTitle('Error')
.setColor('RED')
// @ts-expect-error no why
.setDescription(err['message'] as string);
e.reply({ embeds: [em] });
}
return true;
}
client.login(process.env.DISCORD_TOKEN);