feat: notify when naomi earns achievements
Node.js CI / Lint and Test (push) Successful in 48s

This commit is contained in:
2025-09-10 13:21:08 -07:00
parent bfaf757d3e
commit 996cde5e2f
6 changed files with 135 additions and 1 deletions
+1
View File
@@ -7,6 +7,7 @@
export const ids = {
channels: {
formSubmissions: "1410435042898874471",
gaming: "1385797656307175468",
general: "1385797320389431336",
menteeChat: "1400589073613062204",
mentorshipGoalForum: "1400629118110011526",
+4
View File
@@ -14,6 +14,7 @@ import { App } from "octokit";
import { ids } from "./config/ids.js";
import { handleMessageCreate } from "./events/handleMessageCreate.js";
import { cacheData } from "./modules/cacheData.js";
import { checkRetroAchievements } from "./modules/checkAchievements.js";
import { logMenteeJoin } from "./modules/logMenteeJoin.js";
import { logMenteeLeave } from "./modules/logMenteeLeave.js";
import {
@@ -72,6 +73,9 @@ amari.discord.once(Events.ClientReady, () => {
setInterval(() => {
amari.recentlyActiveChannels = new Set<string>();
}, 10 * 60 * 1000);
setInterval(() => {
void checkRetroAchievements(amari);
}, 10 * 60 * 1000);
});
amari.discord.on(Events.MessageCreate, (message) => {
+118
View File
@@ -0,0 +1,118 @@
/**
* @copyright NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import {
buildAuthorization,
getUserRecentAchievements,
type UserRecentAchievement,
} from "@retroachievements/api";
import {
MessageFlags,
TextDisplayBuilder,
ContainerBuilder,
SectionBuilder,
ThumbnailBuilder,
SeparatorBuilder,
SeparatorSpacingSize,
ActionRowBuilder,
ButtonBuilder,
ButtonStyle,
type MessageActionRowComponentBuilder,
} from "discord.js";
import { ids } from "../config/ids.js";
import type { Amari } from "../interfaces/amari.js";
const username = "naomilgbt";
const webApiKey = process.env.RA_KEY;
const constructComponents = (
achievement: UserRecentAchievement,
): Array<
TextDisplayBuilder |
ContainerBuilder |
ActionRowBuilder<MessageActionRowComponentBuilder>> => {
return [
new TextDisplayBuilder().
setContent("Naomi has unlocked a new achievement!"),
new ContainerBuilder().
setAccentColor(16_758_493).
addSectionComponents(
new SectionBuilder().
setThumbnailAccessory(
new ThumbnailBuilder().
setURL(`https://retroachievements.org/${achievement.badgeUrl}`),
).
addTextDisplayComponents(
new TextDisplayBuilder().setContent(`# ${achievement.title}`),
new TextDisplayBuilder().setContent(achievement.description),
),
).
addSeparatorComponents(
new SeparatorBuilder().setSpacing(SeparatorSpacingSize.Small).
setDivider(true),
).
addSectionComponents(
new SectionBuilder().
setThumbnailAccessory(
new ThumbnailBuilder().
setURL(`https://retroachievements.org${achievement.gameIcon}`),
).
addTextDisplayComponents(
new TextDisplayBuilder().setContent(
`-# From ${achievement.gameTitle} on ${achievement.consoleName}`,
),
new TextDisplayBuilder().setContent(
`-# Worth ${achievement.points.toString()} points!`,
),
),
),
new ActionRowBuilder<MessageActionRowComponentBuilder>().
addComponents(
new ButtonBuilder().
setStyle(ButtonStyle.Link).
setLabel("See her profile").
setURL("https://retroachievements.org/user/naomilgbt"),
),
];
};
/**
* Checks for RA achievements earned in the last 10 minutes.
* If achievements are found, formats them and sends to Discord.
* @param amari - Amari's instance.
*/
export const checkRetroAchievements = async(
amari: Amari,
): Promise<void> => {
if (webApiKey === undefined) {
return;
}
const auth = buildAuthorization({ username, webApiKey });
const recentAchievements = await getUserRecentAchievements(auth, {
recentMinutes: 10,
username: username,
});
if (recentAchievements.length <= 0) {
return;
}
const channel = amari.discord.channels.cache.get(ids.channels.gaming)
?? await amari.discord.channels.fetch(ids.channels.gaming);
if (channel?.isSendable() !== true) {
return;
}
await Promise.all(recentAchievements.map(async(achievement) => {
await channel.send({
components: constructComponents(achievement),
flags: [ MessageFlags.IsComponentsV2 ],
});
}));
};