DiscordTickets/src/client.js

154 lines
4.6 KiB
JavaScript
Raw Normal View History

2022-08-02 19:25:53 +03:00
const { FrameworkClient } = require('@eartharoid/dbf');
2022-08-06 00:21:55 +03:00
const {
GatewayIntentBits,
Partials,
2022-08-06 00:21:55 +03:00
} = require('discord.js');
2024-12-20 04:24:32 +02:00
const logger = require('./lib/logger');
2022-05-06 02:27:27 +03:00
const { PrismaClient } = require('@prisma/client');
2022-07-16 01:19:42 +03:00
const Keyv = require('keyv');
2022-07-17 00:18:50 +03:00
const I18n = require('@eartharoid/i18n');
const fs = require('fs');
const { join } = require('path');
const YAML = require('yaml');
2022-08-06 00:21:55 +03:00
const TicketManager = require('./lib/tickets/manager');
const sqliteMiddleware = require('./lib/middleware/prisma-sqlite');
2023-06-20 22:59:08 +03:00
const ms = require('ms');
2022-05-05 23:29:28 +03:00
module.exports = class Client extends FrameworkClient {
2024-12-20 04:24:32 +02:00
constructor() {
2024-01-11 02:08:57 +02:00
super(
{
intents: [
...[
GatewayIntentBits.DirectMessages,
GatewayIntentBits.DirectMessageReactions,
GatewayIntentBits.DirectMessageTyping,
GatewayIntentBits.MessageContent,
GatewayIntentBits.Guilds,
GatewayIntentBits.GuildMembers,
GatewayIntentBits.GuildMessages,
],
...(process.env.PUBLIC_BOT !== 'true' ? [GatewayIntentBits.GuildPresences] : []),
],
2024-01-11 02:08:57 +02:00
partials: [
Partials.Channel,
Partials.Message,
Partials.Reaction,
],
2024-05-01 04:45:12 +03:00
shards: 'auto',
2024-01-11 02:08:57 +02:00
},
{ baseDir: __dirname },
);
2022-07-17 00:18:50 +03:00
2024-12-20 04:24:32 +02:00
this.config = {};
this.log = {};
this.init();
}
async init(reload = false) {
2022-07-17 00:18:50 +03:00
const locales = {};
fs.readdirSync(join(__dirname, 'i18n'))
.filter(file => file.endsWith('.yml'))
.forEach(file => {
const data = fs.readFileSync(join(__dirname, 'i18n/' + file), { encoding: 'utf8' });
const name = file.slice(0, file.length - 4);
locales[name] = YAML.parse(data);
});
/** @type {I18n} */
this.i18n = new I18n('en-GB', locales);
2024-12-20 04:24:32 +02:00
// to maintain references, these shouldn't be reassigned
Object.assign(this.config, YAML.parse(fs.readFileSync('./user/config.yml', 'utf8')));
Object.assign(this.log, logger(this.config));
this.banned_guilds = new Set(
(() => {
let array = fs.readFileSync('./user/banned-guilds.txt', 'utf8').trim().split(/\r?\n/);
if (array[0] === '') array = [];
return array;
})(),
);
this.log.info(`${this.banned_guilds.size} guilds are banned`);
if (reload) {
await this.initAfterLogin();
} else {
this.keyv = new Keyv();
this.tickets = new TicketManager(this);
this.supers = (process.env.SUPER ?? '').split(',');
/** @param {import('discord.js/typings').Interaction} interaction */
this.commands.interceptor = async interaction => {
if (!interaction.inGuild()) return;
const id = interaction.guildId;
const cacheKey = `cache/known/guild:${id}`;
if (await this.keyv.has(cacheKey)) return;
await this.prisma.guild.upsert({
create: {
id,
locale: this.i18n.locales.find(locale => locale === interaction.guild.preferredLocale), // undefined if not supported
},
update: {},
where: { id },
});
await this.keyv.set(cacheKey, true);
};
}
}
async initAfterLogin() {
for (const id of this.banned_guilds) {
if (this.guilds.cache.has(id)) {
this.log.info(`Leaving banned guild ${id}`);
await this.guilds.cache.get(id).leave();
}
}
2022-05-05 23:29:28 +03:00
}
2022-05-06 02:27:27 +03:00
2022-05-05 23:29:28 +03:00
async login(token) {
2023-06-25 15:43:10 +03:00
const levels = ['error', 'info', 'warn'];
if (this.config.logs.level === 'debug') levels.push('query');
2024-01-11 02:08:57 +02:00
const prisma_options = {
2023-06-25 15:43:10 +03:00
log: levels.map(level => ({
emit: 'event',
level,
})),
2024-01-11 02:08:57 +02:00
};
if (process.env.DB_PROVIDER === 'sqlite' && !process.env.DB_CONNECTION_URL) {
2024-12-20 04:24:32 +02:00
prisma_options.datasources = { db: { url: 'file:' + join(process.cwd(), './user/database.db') } };
2024-01-11 02:08:57 +02:00
}
/** @type {PrismaClient} */
this.prisma = new PrismaClient(prisma_options);
2023-06-25 15:43:10 +03:00
this.prisma.$on('error', e => this.log.error.prisma(`${e.target} ${e.message}`));
this.prisma.$on('info', e => this.log.info.prisma(`${e.target} ${e.message}`));
this.prisma.$on('warn', e => this.log.warn.prisma(`${e.target} ${e.message}`));
this.prisma.$on('query', e => this.log.debug.prisma(e));
2023-06-20 22:59:08 +03:00
2023-01-30 17:56:41 +02:00
if (process.env.DB_PROVIDER === 'sqlite') {
2023-06-20 22:59:08 +03:00
// rewrite queries that use unsupported features
2023-01-30 17:56:41 +02:00
this.prisma.$use(sqliteMiddleware);
2023-06-20 22:59:08 +03:00
// make sqlite faster (missing parentheses are not a mistake, `$queryRaw` is a tagged template literal)
2023-06-17 19:01:06 +03:00
this.log.debug(await this.prisma.$queryRaw`PRAGMA journal_mode=WAL;`); // https://www.sqlite.org/wal.html
this.log.debug(await this.prisma.$queryRaw`PRAGMA synchronous=normal;`); // https://www.sqlite.org/pragma.html#pragma_synchronous
2023-06-20 22:59:08 +03:00
setInterval(async () => {
this.log.debug(await this.prisma.$queryRaw`PRAGMA optimize;`); // https://www.sqlite.org/pragma.html#pragma_optimize
}, ms('6h'));
2023-01-30 17:56:41 +02:00
}
2023-06-20 22:59:08 +03:00
2022-05-05 23:29:28 +03:00
return super.login(token);
}
async destroy() {
await this.prisma.$disconnect();
return super.destroy();
}
};