From c834acdfecd1167ffbda2c0ae7ac4c6062959531 Mon Sep 17 00:00:00 2001 From: Drew Malzahn Date: Fri, 4 Jul 2025 10:39:32 -0400 Subject: [PATCH] WIP: Format code with Google Typescript formatting guidelines --- Justfile | 6 + biome.jsonc | 12 + commands/calendar/nag/checkin.ts | 20 +- commands/calendar/nag/common.ts | 20 +- commands/calendar/nag/nag.ts | 40 +-- commands/calendar/nag/unnag.ts | 17 +- commands/calendar/ping.ts | 12 +- commands/calendar/remind.ts | 52 +-- commands/quotes/quote.ts | 10 +- commands/quotes/quotes.json | 580 ++++++++++++++++++++++++------- database.ts | 24 +- index.ts | 38 +- 12 files changed, 596 insertions(+), 235 deletions(-) create mode 100644 Justfile create mode 100644 biome.jsonc diff --git a/Justfile b/Justfile new file mode 100644 index 0000000..d5a53f9 --- /dev/null +++ b/Justfile @@ -0,0 +1,6 @@ + +fmt: + just format + +format: + bunx --bun @biomejs/biome format --write *.ts commands/ diff --git a/biome.jsonc b/biome.jsonc new file mode 100644 index 0000000..cd42b79 --- /dev/null +++ b/biome.jsonc @@ -0,0 +1,12 @@ +{ + "javascript": { + // This is taken from the Google TS style guide: + // https://github.com/google/gts/blob/main/.prettierrc.json + "formatter": { + "bracketSpacing": false, + "quoteStyle": "single", + "trailingCommas": "all", + "arrowParentheses": "asNeeded" + } + } +} diff --git a/commands/calendar/nag/checkin.ts b/commands/calendar/nag/checkin.ts index 7a35429..1a751be 100644 --- a/commands/calendar/nag/checkin.ts +++ b/commands/calendar/nag/checkin.ts @@ -1,23 +1,21 @@ -import { ChatInputCommandInteraction, SlashCommandBuilder } from "discord.js"; +import {ChatInputCommandInteraction, SlashCommandBuilder} from 'discord.js'; -import { Settings } from "./common"; +import {Settings} from './common'; -import { Nag, CheckIn } from './common'; +import {Nag, CheckIn} from './common'; const data = new SlashCommandBuilder() - .setName("checkin") - .setDescription("Check-in for your daily nag") - .addStringOption((option) => + .setName('checkin') + .setDescription('Check-in for your daily nag') + .addStringOption(option => option - .setName("text") - .setDescription("Optional description of what you have achieved"), + .setName('text') + .setDescription('Optional description of what you have achieved'), ); async function initialize(settings: Settings) {} -function execute(interaction: ChatInputCommandInteraction) { - -} +function execute(interaction: ChatInputCommandInteraction) {} export default function () { return { diff --git a/commands/calendar/nag/common.ts b/commands/calendar/nag/common.ts index 2e5d3ab..dbb48e8 100644 --- a/commands/calendar/nag/common.ts +++ b/commands/calendar/nag/common.ts @@ -3,7 +3,7 @@ import { Client, SlashCommandBuilder, TextChannel, -} from "discord.js"; +} from 'discord.js'; import { Sequelize, Model, @@ -12,7 +12,7 @@ import { BOOLEAN, DATE, literal, -} from "sequelize"; +} from 'sequelize'; export interface Settings { client: Client; // Main Discord client object @@ -63,7 +63,7 @@ export async function initAndSyncTables(sequelize: Sequelize) { type: BOOLEAN, }, }, - { sequelize }, + {sequelize}, ); CheckIn.init( { @@ -76,14 +76,14 @@ export async function initAndSyncTables(sequelize: Sequelize) { allowNull: false, }, }, - { sequelize }, + {sequelize}, ); - Nag.hasOne(CheckIn, { foreignKey: "nagId" }); + Nag.hasOne(CheckIn, {foreignKey: 'nagId'}); await Nag.sync(); await CheckIn.sync(); } -import { Guild, Channel } from "discord.js"; +import {Guild, Channel} from 'discord.js'; export class Plugin { settings: Settings; @@ -106,7 +106,7 @@ export class Plugin { async triggerNag(nag: Nag) { const client = this.settings.client; - const chan = client.channels.cache.get("1234"); // TODO + const chan = client.channels.cache.get('1234'); // TODO if (!(chan instanceof TextChannel)) { return; // TODO } @@ -114,16 +114,16 @@ export class Plugin { const failText = nag.failText ?? `<@${nag.userId}> didn't complete "${nag.text}". Shame shame!`; - const mentionHere = nag.mentionHere ? "<@here> " : ""; + const mentionHere = nag.mentionHere ? '<@here> ' : ''; const msg = `${mentionHere}${failText}`; await chan.send(msg); } catch (error) { - console.log("Error while creating Nag:", error); // TODO + console.log('Error while creating Nag:', error); // TODO } } async loop() { - console.debug("nag.js main loop"); + console.debug('nag.js main loop'); // Find all nags where the last check-in was before (next check in) - (24 hours) } diff --git a/commands/calendar/nag/nag.ts b/commands/calendar/nag/nag.ts index 768dd84..62ae675 100644 --- a/commands/calendar/nag/nag.ts +++ b/commands/calendar/nag/nag.ts @@ -2,30 +2,30 @@ import { ChatInputCommandInteraction, Client, SlashCommandBuilder, -} from "discord.js"; -import { Sequelize, literal } from "sequelize"; +} from 'discord.js'; +import {Sequelize, literal} from 'sequelize'; -import { Nag, CheckIn, Settings } from "./common"; -import { Chrono } from "chrono-node"; +import {Nag, CheckIn, Settings} from './common'; +import {Chrono} from 'chrono-node'; const data = new SlashCommandBuilder() - .setName("nag") - .setDescription("Let Blitzcrank nag you every day about something") - .addStringOption((option) => + .setName('nag') + .setDescription('Let Blitzcrank nag you every day about something') + .addStringOption(option => option .setRequired(true) - .setName("text") - .setDescription("What you have to do every day"), + .setName('text') + .setDescription('What you have to do every day'), ) - .addStringOption((option) => + .addStringOption(option => option - .setName("failText") - .setDescription("Custom message to be broadcast on failure"), + .setName('failText') + .setDescription('Custom message to be broadcast on failure'), ) - .addBooleanOption((option) => + .addBooleanOption(option => option - .setName("mentionHere") - .setDescription("Whether to DM you or @ a channel") + .setName('mentionHere') + .setDescription('Whether to DM you or @ a channel') .setRequired(false), ); @@ -41,7 +41,7 @@ function lateCheckedInUsers() { async function initialize(settings: Settings) {} async function execute(interaction: ChatInputCommandInteraction) { - const text = interaction.options.getString("text"); + const text = interaction.options.getString('text'); if (text === null || text === undefined) { await interaction.reply("Nag can't have a blank `text`, try again."); return; @@ -49,15 +49,15 @@ async function execute(interaction: ChatInputCommandInteraction) { const nag = await Nag.create({ userId: interaction.user.id, text: text, - failText: interaction.options.getString("failText"), - mentionHere: interaction.options.getBoolean("mentionHere") ?? false, + failText: interaction.options.getString('failText'), + mentionHere: interaction.options.getBoolean('mentionHere') ?? false, }); await nag.save(); const chrono = new Chrono(); - const checkIn = chrono.parseDate("today at 9AM"); + const checkIn = chrono.parseDate('today at 9AM'); if (!checkIn) { await interaction.reply( - "Internal error while saving your nag. Tell Drew the bot is broken!!!", + 'Internal error while saving your nag. Tell Drew the bot is broken!!!', ); return; } diff --git a/commands/calendar/nag/unnag.ts b/commands/calendar/nag/unnag.ts index ceaa84e..b65548d 100644 --- a/commands/calendar/nag/unnag.ts +++ b/commands/calendar/nag/unnag.ts @@ -2,19 +2,16 @@ import { ChatInputCommandInteraction, Client, SlashCommandBuilder, -} from "discord.js"; -import { Sequelize } from "sequelize"; - -import { Settings } from './common' +} from 'discord.js'; +import {Sequelize} from 'sequelize'; +import {Settings} from './common'; const data = new SlashCommandBuilder() - .setName("unnag") - .setDescription("Remove a nag"); + .setName('unnag') + .setDescription('Remove a nag'); - -async function initialize(settings: Settings) { -} +async function initialize(settings: Settings) {} async function execute(interaction: ChatInputCommandInteraction) { return; @@ -24,6 +21,6 @@ export default function (settings: Settings) { return { data, execute, - initialize: async() => await initialize(settings), + initialize: async () => await initialize(settings), }; } diff --git a/commands/calendar/ping.ts b/commands/calendar/ping.ts index 772f6d8..9e62a1d 100644 --- a/commands/calendar/ping.ts +++ b/commands/calendar/ping.ts @@ -1,15 +1,15 @@ -import { type Client, SlashCommandBuilder } from "discord.js"; -import type { Sequelize } from "sequelize"; +import {type Client, SlashCommandBuilder} from 'discord.js'; +import type {Sequelize} from 'sequelize'; -export default function (settings: { client: Client; db: Sequelize }) { +export default function (settings: {client: Client; db: Sequelize}) { return { data: new SlashCommandBuilder() - .setName("ping") - .setDescription("Send a ping to the bot"), + .setName('ping') + .setDescription('Send a ping to the bot'), initialize: async () => {}, - execute: async (interaction) => { + execute: async interaction => { await interaction.reply( `Pong! This command was run by ${interaction.user.username}, who joined on ${interaction.member.joinedAt}.`, ); diff --git a/commands/calendar/remind.ts b/commands/calendar/remind.ts index c752bff..601a70c 100644 --- a/commands/calendar/remind.ts +++ b/commands/calendar/remind.ts @@ -5,10 +5,10 @@ import { TextChannel, type ChatInputCommandInteraction, type Client, -} from "discord.js"; -import { type Sequelize, Model, INTEGER, TEXT, DATE, BOOLEAN } from "sequelize"; -import * as sequelize from "sequelize"; -import * as chrono from "chrono-node"; +} from 'discord.js'; +import {type Sequelize, Model, INTEGER, TEXT, DATE, BOOLEAN} from 'sequelize'; +import * as sequelize from 'sequelize'; +import * as chrono from 'chrono-node'; // const REMINDERS_CHANNEL = "1062196593379520593"; // #bot-test-channel // const REMINDERS_CHANNEL = ""; // #general @@ -22,18 +22,18 @@ interface Settings { } const data = new SlashCommandBuilder() - .setName("remind") - .setDescription("Remind me to do something") - .addStringOption((option) => + .setName('remind') + .setDescription('Remind me to do something') + .addStringOption(option => option - .setName("when") - .setDescription("Short description of when you want the reminder") + .setName('when') + .setDescription('Short description of when you want the reminder') .setRequired(true), ) - .addStringOption((option) => + .addStringOption(option => option - .setName("what") - .setDescription("Short description of what you want to be reminded of") + .setName('what') + .setDescription('Short description of what you want to be reminded of') .setRequired(true), ); @@ -62,17 +62,17 @@ class Plugin { // if (this.settings.responseMode === "private") { // channel = getSendableTextChannel(client, reminder.requestChannel); // } - if (this.settings.responseMode === "public" || !channel) { + if (this.settings.responseMode === 'public' || !channel) { channel = getSendableTextChannel(client, publicChannel); } if (!channel) { - throw Error("Cannot find valid channel to send reminder on"); + throw Error('Cannot find valid channel to send reminder on'); } return channel; } async loop() { - console.debug("remind.js main loop"); + console.debug('remind.js main loop'); const results = await Reminder.findAll({ // NOTE: 'trigger' is the time when the reminder should go off. If trigger - // now is positive, trigger is in the future. If trigger - now <= @@ -87,7 +87,7 @@ class Plugin { const now = new Date(Date.now()); const delay = reminder.trigger.getTime() - now.getTime(); console.log( - `Callback for Reminder ${reminder.get("id")} triggering in ${delay}ms`, + `Callback for Reminder ${reminder.get('id')} triggering in ${delay}ms`, ); const channel = this.getChannel(reminder); setTimeout(async () => await triggerReminder(channel, reminder), delay); @@ -113,7 +113,7 @@ class Plugin { } async function initialize(settings: Settings) { - console.log("Initializing remind.js"); + console.log('Initializing remind.js'); // Populate Reminder table inside SQLite DB Reminder.init( @@ -136,7 +136,7 @@ async function initialize(settings: Settings) { requestChannel: TEXT, requestMessage: TEXT, }, - { sequelize: settings.db }, + {sequelize: settings.db}, ); await Reminder.sync(); @@ -144,13 +144,13 @@ async function initialize(settings: Settings) { console.debug(`Accessing channel ${settings.publicChannel}`); if (!getSendableTextChannel(settings.client, settings.publicChannel)) { throw Error( - "Invalid value for publicChannel; specify a channel the bot can access.", + 'Invalid value for publicChannel; specify a channel the bot can access.', ); } // Initialize the 'plugin' object which will store all state required to control the mainloop. if (settings.loopIntervalSec == null || settings.loopIntervalSec <= 0) { - console.log("Setting plugin loop interval to default of 60 seconds"); + console.log('Setting plugin loop interval to default of 60 seconds'); settings.loopIntervalSec = 60; } const plugin = new Plugin(settings); @@ -164,13 +164,13 @@ function getSendableTextChannel(client: Client, chanId: string) { throw Error("Can't check non-existent client"); } if (!channel) { - throw Error("No such channel is visible to Blitz"); + throw Error('No such channel is visible to Blitz'); } if (!channel?.isSendable()) { - throw Error("Channel is not a text channel, or otherwise not Sendable"); + throw Error('Channel is not a text channel, or otherwise not Sendable'); } if (!(channel instanceof TextChannel)) { - throw Error("Channel is not a guild channel"); + throw Error('Channel is not a guild channel'); } const permissions = channel?.permissionsFor(client.user); const requiredPerms = new PermissionsBitField([ @@ -178,7 +178,7 @@ function getSendableTextChannel(client: Client, chanId: string) { PermissionsBitField.Flags.ViewChannel, ]); if (!permissions?.has(requiredPerms)) { - throw Error("Missing required permissions: SendMessages, ViewChannel"); + throw Error('Missing required permissions: SendMessages, ViewChannel'); } return channel; } @@ -197,7 +197,7 @@ async function triggerReminder(channel: TextChannel, reminder: Reminder) { } async function execute(interaction: ChatInputCommandInteraction) { - const whenString = interaction.options.getString("when") ?? "now"; + const whenString = interaction.options.getString('when') ?? 'now'; const when = chrono.parseDate(whenString); if (!when) { await interaction.reply(`Sorry, I don't understand '${when}' as a date`); @@ -205,7 +205,7 @@ async function execute(interaction: ChatInputCommandInteraction) { } const reminder = await Reminder.create({ userId: interaction.user.id, - text: interaction.options.getString("what"), + text: interaction.options.getString('what'), trigger: when, // TODO requestMessage: interaction.id, requestChannel: interaction.channelId, diff --git a/commands/quotes/quote.ts b/commands/quotes/quote.ts index ff51c7f..7eb12cc 100644 --- a/commands/quotes/quote.ts +++ b/commands/quotes/quote.ts @@ -4,10 +4,10 @@ import { type ChatInputCommandInteraction, type Client, type InviteStageInstance, -} from "discord.js"; -import type { Sequelize } from "sequelize"; +} from 'discord.js'; +import type {Sequelize} from 'sequelize'; -import { quotes } from "./quotes.json"; +import {quotes} from './quotes.json'; async function execute(interaction: ChatInputCommandInteraction) { try { @@ -26,8 +26,8 @@ async function initialize() {} export default function (settings: {}) { return { data: new SlashCommandBuilder() - .setName("quote") - .setDescription("Print a quote from League of Legends."), + .setName('quote') + .setDescription('Print a quote from League of Legends.'), initialize: initialize, execute: execute, }; diff --git a/commands/quotes/quotes.json b/commands/quotes/quotes.json index 3211178..56817cc 100644 --- a/commands/quotes/quotes.json +++ b/commands/quotes/quotes.json @@ -3,169 +3,517 @@ { "author": "Blitzcrank", "quote": "A rolling golem gathers no rust." }, { "author": "Blitzcrank", "quote": "Fired up and ready to serve." }, { "author": "Blitzcrank", "quote": "Metal is harder than flesh." }, - { "author": "Camille", "quote": "[Speaking to Warwick] We are all monsters. Now, you are just one on the outside." }, + { + "author": "Camille", + "quote": "[Speaking to Warwick] We are all monsters. Now, you are just one on the outside." + }, { "author": "Camille", "quote": "Efficiency is paramount to success." }, { "author": "Camille", "quote": "Elegance never goes out of fashion." }, - { "author": "Camille", "quote": "Extremes are easy, it's the balance that is difficult." }, - { "author": "Camille", "quote": "I don't play the game, I make the rules." }, - { "author": "Camille", "quote": "I'm what you would call a 'deniable asset'." }, - { "author": "Camille", "quote": "It is not the weapon that defines you, but how you wield it." }, - { "author": "Camille", "quote": "It's not lies that cut, but the sharpness of the truth." }, + { + "author": "Camille", + "quote": "Extremes are easy, it's the balance that is difficult." + }, + { + "author": "Camille", + "quote": "I don't play the game, I make the rules." + }, + { + "author": "Camille", + "quote": "I'm what you would call a 'deniable asset'." + }, + { + "author": "Camille", + "quote": "It is not the weapon that defines you, but how you wield it." + }, + { + "author": "Camille", + "quote": "It's not lies that cut, but the sharpness of the truth." + }, { "author": "Camille", "quote": "Mediocrity is the root of all evil." }, - { "author": "Camille", "quote": "Morality is a beautiful servant and a dangerous master." }, - { "author": "Camille", "quote": "Precision is the difference between a butcher and a surgeon." }, - { "author": "Camille", "quote": "Privilege must be preserved at all costs." }, + { + "author": "Camille", + "quote": "Morality is a beautiful servant and a dangerous master." + }, + { + "author": "Camille", + "quote": "Precision is the difference between a butcher and a surgeon." + }, + { + "author": "Camille", + "quote": "Privilege must be preserved at all costs." + }, { "author": "Camille", "quote": "Progress is honed on necessary death." }, - { "author": "Camille", "quote": "Progress is served by technology, not controlled by it." }, - { "author": "Camille", "quote": "Regret is what tempers the steel of our soul." }, + { + "author": "Camille", + "quote": "Progress is served by technology, not controlled by it." + }, + { + "author": "Camille", + "quote": "Regret is what tempers the steel of our soul." + }, { "author": "Camille", "quote": "Results are all that matters." }, - { "author": "Camille", "quote": "Self-made women need to be more prevalent." }, - { "author": "Camille", "quote": "Sometimes scars are the most refined attire one can wear." }, - { "author": "Camille", "quote": "The right word cuts more deeply than a knife." }, - { "author": "Camille", "quote": "The task at hand is the only one that matters." }, - { "author": "Camille", "quote": "The world is not black or white, but a delicious shade of grey." }, + { + "author": "Camille", + "quote": "Self-made women need to be more prevalent." + }, + { + "author": "Camille", + "quote": "Sometimes scars are the most refined attire one can wear." + }, + { + "author": "Camille", + "quote": "The right word cuts more deeply than a knife." + }, + { + "author": "Camille", + "quote": "The task at hand is the only one that matters." + }, + { + "author": "Camille", + "quote": "The world is not black or white, but a delicious shade of grey." + }, { "author": "Camille", "quote": "Violence is a means to an end." }, - { "author": "Ekko", "quote": "A second chance? I thought I was on my fifth!" }, + { + "author": "Ekko", + "quote": "A second chance? I thought I was on my fifth!" + }, { "author": "Ekko", "quote": "Good a time as any to act reckless." }, - { "author": "Ekko", "quote": "It's not how much time you have, it's how you use it." }, + { + "author": "Ekko", + "quote": "It's not how much time you have, it's how you use it." + }, { "author": "Ekko", "quote": "Never had luck. Never needed it." }, { "author": "Ekko", "quote": "Time doesn't heal all wounds." }, - { "author": "Heimerdinger", "quote": "42... there's just something about that number." }, - { "author": "Heimerdinger", "quote": "I prefer a battle of wits, but you're unarmed!" }, - { "author": "Heimerdinger", "quote": "Why do chemists call helium, curium, and barium 'the medical elements'? Because, if you can't 'helium' or 'curium', you 'barium'! Hm hm!" }, + { + "author": "Heimerdinger", + "quote": "42... there's just something about that number." + }, + { + "author": "Heimerdinger", + "quote": "I prefer a battle of wits, but you're unarmed!" + }, + { + "author": "Heimerdinger", + "quote": "Why do chemists call helium, curium, and barium 'the medical elements'? Because, if you can't 'helium' or 'curium', you 'barium'! Hm hm!" + }, { "author": "Heimerdinger", "quote": "ヽ༼ຈل͜ຈ༽ノ raise your dongers" }, { "author": "Jhin", "quote": "Art must exist beyond reason." }, { "author": "Jhin", "quote": "Four!" }, { "author": "Jhin", "quote": "I cannot be good. I must be perfection." }, - { "author": "Jhin", "quote": "I swear each performance is the last, but I lie every time." }, - { "author": "Jhin", "quote": "In carnage, I bloom, like a flower in the dawn." }, - { "author": "Jhin", "quote": "It is by my will alone I set my mind in motion." }, - { "author": "Jhin", "quote": "It's fun to kill a man, to take all that he had, and could ever have." }, + { + "author": "Jhin", + "quote": "I swear each performance is the last, but I lie every time." + }, + { + "author": "Jhin", + "quote": "In carnage, I bloom, like a flower in the dawn." + }, + { + "author": "Jhin", + "quote": "It is by my will alone I set my mind in motion." + }, + { + "author": "Jhin", + "quote": "It's fun to kill a man, to take all that he had, and could ever have." + }, { "author": "Jhin", "quote": "You will learn what beauty truly is." }, - { "author": "Jinx", "quote": "Fishbones, you know what we oughta do? 'Do the laundry, wash dishes and pay some bills.' Stupid dumb rocket launcher..." }, + { + "author": "Jinx", + "quote": "Fishbones, you know what we oughta do? 'Do the laundry, wash dishes and pay some bills.' Stupid dumb rocket launcher..." + }, { "author": "Jinx", "quote": "I'm crazy! Got a doctor's note." }, { "author": "Jinx", "quote": "I'm trying to care! But I just... can't!" }, - { "author": "Jinx", "quote": "Rules are made to be broken... like buildings! Or people!" }, - { "author": "Joseph Miklos", "quote": "'It's only a short way'? Is that a short joke?!" }, + { + "author": "Jinx", + "quote": "Rules are made to be broken... like buildings! Or people!" + }, + { + "author": "Joseph Miklos", + "quote": "'It's only a short way'? Is that a short joke?!" + }, { "author": "Joseph Miklos", "quote": "I am evil! Stop laughing!" }, - { "author": "Joseph Miklos", "quote": "Know that if the tables were turned, I would show you no mercy!" }, + { + "author": "Joseph Miklos", + "quote": "Know that if the tables were turned, I would show you no mercy!" + }, { "author": "Joseph Miklos", "quote": "You will die by my hand!" }, { "author": "Mordekaiser", "quote": "Ah, life is a bitter shame." }, - { "author": "Mordekaiser", "quote": "All mortals reek with the stench of decaying flesh." }, - { "author": "Mordekaiser", "quote": "Fools fear death, the strong wield it." }, - { "author": "Mordekaiser", "quote": "I alone am the bastion between eternal existence and oblivion." }, - { "author": "Mordekaiser", "quote": "I carve my kingdom beyond, from the ashes of nothing, no mortals, not even gods, will stop me from claiming what is mine." }, - { "author": "Mordekaiser", "quote": "I have bent the realm of the dead to my will, this world shall be next." }, - { "author": "Mordekaiser", "quote": "I raise my iron fist to subjugate the living." }, - { "author": "Mordekaiser", "quote": "I will grind their petty souls into mortar." }, - { "author": "Mordekaiser", "quote": "I will silence the incessant thrum of mortal hearts." }, - { "author": "Mordekaiser", "quote": "In the world beyond, blackened ichor filled a crumbling sky, as souls withered to nothing. But I refused to fade." }, - { "author": "Mordekaiser", "quote": "Mortals plan in fear for tomorrow, I build for eternity." }, - { "author": "Mordekaiser", "quote": "Naive men pray to the gods; they will learn to pray to me." }, - { "author": "Mordekaiser", "quote": "Only the worthy receive the gift of Nightfall's kiss." }, - { "author": "Mordekaiser", "quote": "Shed the frailty of flesh, embrace the cold edge of iron." }, - { "author": "Mordekaiser", "quote": "The dead belong to me, the living shall be next." }, - { "author": "Mordekaiser", "quote": "The world has tried to forget my existence, time to remind them why they fear." }, + { + "author": "Mordekaiser", + "quote": "All mortals reek with the stench of decaying flesh." + }, + { + "author": "Mordekaiser", + "quote": "Fools fear death, the strong wield it." + }, + { + "author": "Mordekaiser", + "quote": "I alone am the bastion between eternal existence and oblivion." + }, + { + "author": "Mordekaiser", + "quote": "I carve my kingdom beyond, from the ashes of nothing, no mortals, not even gods, will stop me from claiming what is mine." + }, + { + "author": "Mordekaiser", + "quote": "I have bent the realm of the dead to my will, this world shall be next." + }, + { + "author": "Mordekaiser", + "quote": "I raise my iron fist to subjugate the living." + }, + { + "author": "Mordekaiser", + "quote": "I will grind their petty souls into mortar." + }, + { + "author": "Mordekaiser", + "quote": "I will silence the incessant thrum of mortal hearts." + }, + { + "author": "Mordekaiser", + "quote": "In the world beyond, blackened ichor filled a crumbling sky, as souls withered to nothing. But I refused to fade." + }, + { + "author": "Mordekaiser", + "quote": "Mortals plan in fear for tomorrow, I build for eternity." + }, + { + "author": "Mordekaiser", + "quote": "Naive men pray to the gods; they will learn to pray to me." + }, + { + "author": "Mordekaiser", + "quote": "Only the worthy receive the gift of Nightfall's kiss." + }, + { + "author": "Mordekaiser", + "quote": "Shed the frailty of flesh, embrace the cold edge of iron." + }, + { + "author": "Mordekaiser", + "quote": "The dead belong to me, the living shall be next." + }, + { + "author": "Mordekaiser", + "quote": "The world has tried to forget my existence, time to remind them why they fear." + }, { "author": "Mordekaiser", "quote": "Twice slain, thrice born." }, - { "author": "Mordekaiser", "quote": "Weaklings cower in the light, I bring eternal darkness." }, - { "author": "Poppy", "quote": "I'm in, one-hundred-percent! That's everything, right?" }, + { + "author": "Mordekaiser", + "quote": "Weaklings cower in the light, I bring eternal darkness." + }, + { + "author": "Poppy", + "quote": "I'm in, one-hundred-percent! That's everything, right?" + }, { "author": "Poppy", "quote": "I'm no hero—just a Yordle with a hammer." }, { "author": "Poppy", "quote": "Just had a thought—three pigtails!" }, - { "author": "Poppy", "quote": "The hammer does most of the work, I just swing it." }, + { + "author": "Poppy", + "quote": "The hammer does most of the work, I just swing it." + }, { "author": "Rammus", "quote": "OK." }, { "author": "Rammus", "quote": "🆗" }, { "author": "Senna", "quote": "I forgive. No one else has to." }, - { "author": "Senna", "quote": "I remember my nightmares. Wish I could remember to dream." }, + { + "author": "Senna", + "quote": "I remember my nightmares. Wish I could remember to dream." + }, { "author": "Sion", "quote": "A black eye for the earth!" }, { "author": "Sion", "quote": "Death had its chance." }, { "author": "Sion", "quote": "Noxus suffers no cowards." }, { "author": "Sion", "quote": "The quiet... eats at me." }, { "author": "Swain", "quote": "A calculated risk is no risk at all." }, - { "author": "Swain", "quote": "A new vantage, is all the advantage I need." }, + { + "author": "Swain", + "quote": "A new vantage, is all the advantage I need." + }, { "author": "Swain", "quote": "And to think, they called me a 'cripple'." }, { "author": "Swain", "quote": "Destiny marches—like any man." }, { "author": "Swain", "quote": "Diplomacy is a subtle art." }, - { "author": "Swain", "quote": "Hmph... I do so enjoy explaining things to idiots." }, - { "author": "Swain", "quote": "I cannot lead if I allow fools to stumble about before me." }, - { "author": "Swain", "quote": "I could kill them all. But it would be far crueler to show them that I am right." }, - { "author": "Swain", "quote": "I have killed more men with words than by my own hand. Not for lack of trying." }, - { "author": "Swain", "quote": "I suppose I should be grateful they have the decency to fear me." }, - { "author": "Swain", "quote": "I've heard what they call me. What a waste of their final words." }, - { "author": "Swain", "quote": "If they already call me a villain, what will they call me when I succeed?" }, + { + "author": "Swain", + "quote": "Hmph... I do so enjoy explaining things to idiots." + }, + { + "author": "Swain", + "quote": "I cannot lead if I allow fools to stumble about before me." + }, + { + "author": "Swain", + "quote": "I could kill them all. But it would be far crueler to show them that I am right." + }, + { + "author": "Swain", + "quote": "I have killed more men with words than by my own hand. Not for lack of trying." + }, + { + "author": "Swain", + "quote": "I suppose I should be grateful they have the decency to fear me." + }, + { + "author": "Swain", + "quote": "I've heard what they call me. What a waste of their final words." + }, + { + "author": "Swain", + "quote": "If they already call me a villain, what will they call me when I succeed?" + }, { "author": "Swain", "quote": "Is it not enough for Noxus to be strong?" }, - { "author": "Swain", "quote": "It is not the visions that haunt me—but what I do not see." }, - { "author": "Swain", "quote": "Never make a bargain with a demon... that you intend to keep." }, - { "author": "Swain", "quote": "One can read the future in battle lines, assuming one can read." }, - { "author": "Swain", "quote": "People often ask for a hero, when a villain is what they truly need." }, - { "author": "Swain", "quote": "Pity stays the hand of the merciful, but not mine." }, - { "author": "Swain", "quote": "Tell me again all the crimes I've committed, and I'll tell you the price of victory." }, - { "author": "Swain", "quote": "The more they try to kill me, the more they reveal I am on the right path." }, - { "author": "Swain", "quote": "The outcome was decided when they brought an army; and I brought a demon." }, + { + "author": "Swain", + "quote": "It is not the visions that haunt me—but what I do not see." + }, + { + "author": "Swain", + "quote": "Never make a bargain with a demon... that you intend to keep." + }, + { + "author": "Swain", + "quote": "One can read the future in battle lines, assuming one can read." + }, + { + "author": "Swain", + "quote": "People often ask for a hero, when a villain is what they truly need." + }, + { + "author": "Swain", + "quote": "Pity stays the hand of the merciful, but not mine." + }, + { + "author": "Swain", + "quote": "Tell me again all the crimes I've committed, and I'll tell you the price of victory." + }, + { + "author": "Swain", + "quote": "The more they try to kill me, the more they reveal I am on the right path." + }, + { + "author": "Swain", + "quote": "The outcome was decided when they brought an army; and I brought a demon." + }, { "author": "Swain", "quote": "The right to rule, held, in my hand." }, - { "author": "Swain", "quote": "There is always a choice. The truth is no exception." }, - { "author": "Swain", "quote": "They are blind to the cold logic of this world." }, - { "author": "Swain", "quote": "They are five steps from realizing: I am ten steps ahead." }, - { "author": "Swain", "quote": "They expect me to play fairly... We aren't even playing the same game." }, - { "author": "Swain", "quote": "What is one more demon, when I already have so many?" }, - { "author": "Swain", "quote": "Would they even struggle to survive, if they knew what was to come?" }, - { "author": "Swain", "quote": "You can sit on a throne, that doesn't make you a ruler. It only means you have an arse." }, - { "author": "Tahm Kench", "quote": "All creation is born famished and starving." }, - { "author": "Tahm Kench", "quote": "Child, you're a couple cows short of a steak!" }, - { "author": "Tahm Kench", "quote": "The only real sin is to deny a craving." }, - { "author": "Tahm Kench", "quote": "We all gourmandize from time to time." }, + { + "author": "Swain", + "quote": "There is always a choice. The truth is no exception." + }, + { + "author": "Swain", + "quote": "They are blind to the cold logic of this world." + }, + { + "author": "Swain", + "quote": "They are five steps from realizing: I am ten steps ahead." + }, + { + "author": "Swain", + "quote": "They expect me to play fairly... We aren't even playing the same game." + }, + { + "author": "Swain", + "quote": "What is one more demon, when I already have so many?" + }, + { + "author": "Swain", + "quote": "Would they even struggle to survive, if they knew what was to come?" + }, + { + "author": "Swain", + "quote": "You can sit on a throne, that doesn't make you a ruler. It only means you have an arse." + }, + { + "author": "Tahm Kench", + "quote": "All creation is born famished and starving." + }, + { + "author": "Tahm Kench", + "quote": "Child, you're a couple cows short of a steak!" + }, + { + "author": "Tahm Kench", + "quote": "The only real sin is to deny a craving." + }, + { + "author": "Tahm Kench", + "quote": "We all gourmandize from time to time." + }, { "author": "Teemo", "quote": "Size doesn't mean everything." }, { "author": "Thresh", "quote": "I am the thing under the bed." }, { "author": "Thresh", "quote": "Me, mad? Haha... quite likely." }, - { "author": "Urgot", "quote": "Cast into a pit of despair, I climbed out on the corpses." }, - { "author": "Urgot", "quote": "I am stronger than man, stronger than machine, I am an idea." }, - { "author": "Urgot", "quote": "I am the very definition of a self-made man." }, - { "author": "Urgot", "quote": "If they do not stop me, they will die. It is just that simple." }, + { + "author": "Urgot", + "quote": "Cast into a pit of despair, I climbed out on the corpses." + }, + { + "author": "Urgot", + "quote": "I am stronger than man, stronger than machine, I am an idea." + }, + { + "author": "Urgot", + "quote": "I am the very definition of a self-made man." + }, + { + "author": "Urgot", + "quote": "If they do not stop me, they will die. It is just that simple." + }, { "author": "Urgot", "quote": "Pain is the act of becoming." }, - { "author": "Urgot", "quote": "We will rise from the rubble, stronger than before." }, - { "author": "Urgot", "quote": "You cannot know strength... Until you are broken." }, - { "author": "Veigar", "quote": "'It's only a short way'? Is that a short joke?!" }, + { + "author": "Urgot", + "quote": "We will rise from the rubble, stronger than before." + }, + { + "author": "Urgot", + "quote": "You cannot know strength... Until you are broken." + }, + { + "author": "Veigar", + "quote": "'It's only a short way'? Is that a short joke?!" + }, { "author": "Veigar", "quote": "I am evil! Stop laughing!" }, - { "author": "Veigar", "quote": "Know that if the tables were turned, I would show you no mercy!" }, - { "author": "Vex", "quote": "'Death is the true meaning of life.' Whoa! That's deep." }, - { "author": "Vex", "quote": "[Speaking to Lux] Oh, no. Happiness and rainbows? I'm gonna barf twice." }, - { "author": "Vex", "quote": "And then I told her, 'Get outta my room!' And she said, 'This is my house, young lady and'... Oh, hang on, Shadow. I'll finish this later." }, + { + "author": "Veigar", + "quote": "Know that if the tables were turned, I would show you no mercy!" + }, + { + "author": "Vex", + "quote": "'Death is the true meaning of life.' Whoa! That's deep." + }, + { + "author": "Vex", + "quote": "[Speaking to Lux] Oh, no. Happiness and rainbows? I'm gonna barf twice." + }, + { + "author": "Vex", + "quote": "And then I told her, 'Get outta my room!' And she said, 'This is my house, young lady and'... Oh, hang on, Shadow. I'll finish this later." + }, { "author": "Vex", "quote": "Calm down, Shadow. I'm trying to sulk." }, - { "author": "Vex", "quote": "I am not cute. I am dark and forlorn and hopelessly morbid!" }, - { "author": "Vex", "quote": "I could start a club for people who hate people! Ehh, but no one would show up." }, - { "author": "Vex", "quote": "This is going to be... awful, in a very good way. A good, awful way. You know what I mean!" }, - { "author": "Vex", "quote": "Welcome to Sad Town. Population: Me. Everyone else get out." }, - { "author": "Viktor", "quote": "All that is logical is true, absolute, irrefutable." }, - { "author": "Viktor", "quote": "Choice is false. It is how we clothe and forgive the baser instincts that spur us to division." }, - { "author": "Viktor", "quote": "Emotion and logic cannot coexist. One must be shed to gain the other." }, + { + "author": "Vex", + "quote": "I am not cute. I am dark and forlorn and hopelessly morbid!" + }, + { + "author": "Vex", + "quote": "I could start a club for people who hate people! Ehh, but no one would show up." + }, + { + "author": "Vex", + "quote": "This is going to be... awful, in a very good way. A good, awful way. You know what I mean!" + }, + { + "author": "Vex", + "quote": "Welcome to Sad Town. Population: Me. Everyone else get out." + }, + { + "author": "Viktor", + "quote": "All that is logical is true, absolute, irrefutable." + }, + { + "author": "Viktor", + "quote": "Choice is false. It is how we clothe and forgive the baser instincts that spur us to division." + }, + { + "author": "Viktor", + "quote": "Emotion and logic cannot coexist. One must be shed to gain the other." + }, { "author": "Viktor", "quote": "Emotion... clashes with reason." }, - { "author": "Viktor", "quote": "Governed by instinct, humans are no more than flawed and flailing animals." }, - { "author": "Viktor", "quote": "Hexcorization requires no justification. What purpose is there in explaining a horseshoe to the horse?" }, + { + "author": "Viktor", + "quote": "Governed by instinct, humans are no more than flawed and flailing animals." + }, + { + "author": "Viktor", + "quote": "Hexcorization requires no justification. What purpose is there in explaining a horseshoe to the horse?" + }, { "author": "Viktor", "quote": "Humanity... is self-corrupting." }, - { "author": "Viktor", "quote": "I am not the man I was, but who I wished to be." }, - { "author": "Viktor", "quote": "I am the only one with the means to cure suffering. But it is a lonely path." }, - { "author": "Viktor", "quote": "I chose to become this. Difficulty had no bearing nor did danger. It was... necessary." }, - { "author": "Viktor", "quote": "I did not know true... beauty, until the Arcane." }, + { + "author": "Viktor", + "quote": "I am not the man I was, but who I wished to be." + }, + { + "author": "Viktor", + "quote": "I am the only one with the means to cure suffering. But it is a lonely path." + }, + { + "author": "Viktor", + "quote": "I chose to become this. Difficulty had no bearing nor did danger. It was... necessary." + }, + { + "author": "Viktor", + "quote": "I did not know true... beauty, until the Arcane." + }, { "author": "Viktor", "quote": "I offer no choice, for there is none." }, - { "author": "Viktor", "quote": "I sense... trepidation. But one cannot grow if left unchallenged." }, - { "author": "Viktor", "quote": "Mankind clings to its past. Glorifies its present. And lives in dread of tomorrow." }, - { "author": "Viktor", "quote": "Passion double-crosses, subverts, divides." }, + { + "author": "Viktor", + "quote": "I sense... trepidation. But one cannot grow if left unchallenged." + }, + { + "author": "Viktor", + "quote": "Mankind clings to its past. Glorifies its present. And lives in dread of tomorrow." + }, + { + "author": "Viktor", + "quote": "Passion double-crosses, subverts, divides." + }, { "author": "Viktor", "quote": "Sentiment is incompatible with control." }, - { "author": "Viktor", "quote": "So much of what we value is inconsequential." }, - { "author": "Viktor", "quote": "They think humanity can survive with emotion? Survive as what? Creatures blinded by impulse?" }, - { "author": "Viktor", "quote": "To live with flaws, is to be subject to them." }, - { "author": "Viktor", "quote": "True change must be imposed, not offered." }, - { "author": "Viktor", "quote": "What I am doing is not torment. Torment is allowing the mind to corrupt the soul." }, + { + "author": "Viktor", + "quote": "So much of what we value is inconsequential." + }, + { + "author": "Viktor", + "quote": "They think humanity can survive with emotion? Survive as what? Creatures blinded by impulse?" + }, + { + "author": "Viktor", + "quote": "To live with flaws, is to be subject to them." + }, + { + "author": "Viktor", + "quote": "True change must be imposed, not offered." + }, + { + "author": "Viktor", + "quote": "What I am doing is not torment. Torment is allowing the mind to corrupt the soul." + }, { "author": "Volibear", "quote": "A thousand scars, what is one more?" }, - { "author": "Volibear", "quote": "The creations of mortals fail. The wild remains." }, - { "author": "Volibear", "quote": "The land slumbers, but it is not dead. With my roar, I wake it. With my thunder, I call it." }, - { "author": "Volibear", "quote": "They have forgotten the old ways. The old ways have not forgotten them." }, - { "author": "Volibear", "quote": "Warm-bloods rose on two legs... and forgot how to run." }, - { "author": "Xerath", "quote": "I am power incarnate! Who dares oppose me?" }, + { + "author": "Volibear", + "quote": "The creations of mortals fail. The wild remains." + }, + { + "author": "Volibear", + "quote": "The land slumbers, but it is not dead. With my roar, I wake it. With my thunder, I call it." + }, + { + "author": "Volibear", + "quote": "They have forgotten the old ways. The old ways have not forgotten them." + }, + { + "author": "Volibear", + "quote": "Warm-bloods rose on two legs... and forgot how to run." + }, + { + "author": "Xerath", + "quote": "I am power incarnate! Who dares oppose me?" + }, { "author": "Xerath", "quote": "I am the will of man, unbound by flesh." }, - { "author": "Xerath", "quote": "I see the forces that hold the universe together." }, + { + "author": "Xerath", + "quote": "I see the forces that hold the universe together." + }, { "author": "Xerath", "quote": "The secrets of magic are mine alone." }, - { "author": "Yone", "quote": "Long before blades and sorcery are needed, words... can save a soul." }, - { "author": "Yone", "quote": "Sleep is not for the weak, but for the blessed." }, - { "author": "Yone", "quote": "Sometimes, to save someone, you must fight them." } + { + "author": "Yone", + "quote": "Long before blades and sorcery are needed, words... can save a soul." + }, + { + "author": "Yone", + "quote": "Sleep is not for the weak, but for the blessed." + }, + { + "author": "Yone", + "quote": "Sometimes, to save someone, you must fight them." + } ] } diff --git a/database.ts b/database.ts index d2be0b4..cb36075 100644 --- a/database.ts +++ b/database.ts @@ -1,9 +1,9 @@ -import { Model, Sequelize, STRING } from "sequelize"; +import {Model, Sequelize, STRING} from 'sequelize'; -import "node:fs/promises"; -import { readFile } from "node:fs"; +import 'node:fs/promises'; +import {readFile} from 'node:fs'; -export const sql = new Sequelize("sqlite://./blitzcrank.sqlite"); +export const sql = new Sequelize('sqlite://./blitzcrank.sqlite'); export class GuildSetting extends Model { declare guildId: string; @@ -13,19 +13,19 @@ export class GuildSetting extends Model { GuildSetting.init( { - guildId: { type: STRING }, - key: { type: STRING }, - value: { type: STRING }, + guildId: {type: STRING}, + key: {type: STRING}, + value: {type: STRING}, }, - { sequelize: sql }, + {sequelize: sql}, ); export async function initDb() { try { await sql.authenticate(); - console.log("Connected to database"); + console.log('Connected to database'); } catch (error) { - console.error("Unable to connect to the database:", error); + console.error('Unable to connect to the database:', error); } } @@ -43,10 +43,10 @@ export async function readSettingsFromFile(path: string) { }); } } - GuildSetting.bulkCreate(toInsert, { updateOnDuplicate: ["value"] }); + GuildSetting.bulkCreate(toInsert, {updateOnDuplicate: ['value']}); } catch (error) { // TODO - console.log("Could not read settings:", error); + console.log('Could not read settings:', error); } }); } diff --git a/index.ts b/index.ts index c6195c1..279cb1a 100644 --- a/index.ts +++ b/index.ts @@ -1,11 +1,11 @@ -import type { Interaction } from "discord.js"; -import { Client, Events, GatewayIntentBits, MessageFlags } from "discord.js"; +import type {Interaction} from 'discord.js'; +import {Client, Events, GatewayIntentBits, MessageFlags} from 'discord.js'; import type { SlashCommandBuilder, SlashCommandOptionsOnlyBuilder, -} from "discord.js"; +} from 'discord.js'; -import { sql, GuildSetting, initDb } from "./database"; +import {sql, GuildSetting, initDb} from './database'; const BLITZCRANK_BANNER = ` ****++++++++++*+++ @@ -71,9 +71,9 @@ const client = new Client({ ], }); -import { Routes } from "discord.js"; -import { guildId, appId, token, remindersChannelId } from "./config.json"; -import { REST } from "discord.js"; +import {Routes} from 'discord.js'; +import {guildId, appId, token, remindersChannelId} from './config.json'; +import {REST} from 'discord.js'; const rest = new REST(); rest.setToken(token); @@ -84,26 +84,26 @@ interface Command { initialize: (any) => Promise; } -import { Collection } from "discord.js"; +import {Collection} from 'discord.js'; const commands = new Collection(); -import PingCommand from "./commands/calendar/ping"; -import RemindCommand from "./commands/calendar/remind"; -import QuoteCommand from "./commands/quotes/quote"; +import PingCommand from './commands/calendar/ping'; +import RemindCommand from './commands/calendar/remind'; +import QuoteCommand from './commands/quotes/quote'; console.debug(`${remindersChannelId}`); -commands.set("ping", PingCommand({ client: client, db: sql })); +commands.set('ping', PingCommand({client: client, db: sql})); commands.set( - "remind", + 'remind', RemindCommand({ client: client, db: sql, publicChannel: remindersChannelId, - responseMode: "public", + responseMode: 'public', }), ); -commands.set("quote", QuoteCommand({})); +commands.set('quote', QuoteCommand({})); async function syncCommands() { try { @@ -111,7 +111,7 @@ async function syncCommands() { const _data = await rest.put( Routes.applicationGuildCommands(appId, guildId), { - body: commands.mapValues((cmd) => cmd.data.toJSON()), + body: commands.mapValues(cmd => cmd.data.toJSON()), }, ); console.log(`Successfully reloaded slash commands`); @@ -135,7 +135,7 @@ client.on(Events.InteractionCreate, async (interaction: Interaction) => { console.error(error); if (interaction.replied || interaction.deferred) { await interaction.followUp({ - content: "There was an error while executing this command", + content: 'There was an error while executing this command', flags: MessageFlags.Ephemeral, }); } @@ -143,7 +143,7 @@ client.on(Events.InteractionCreate, async (interaction: Interaction) => { // TODO }); -client.once(Events.ClientReady, async (readyClient) => { +client.once(Events.ClientReady, async readyClient => { await syncCommands(); initDb(); // TODO GuildSetting.sync(); // TODO @@ -155,7 +155,7 @@ client.once(Events.ClientReady, async (readyClient) => { }); } // Print banner - for (const ln of BLITZCRANK_BANNER.split("\n")) { + for (const ln of BLITZCRANK_BANNER.split('\n')) { console.log(ln); } console.log(`Logged in as ${readyClient.user.tag}`);