Files
PteroStats/handlers/application.js

265 lines
13 KiB
JavaScript

require("dotenv").config();
const { Client, GatewayIntentBits, EmbedBuilder, time, ActivityType, ActionRowBuilder, ButtonBuilder, ButtonStyle } = require("discord.js");
const fs = require("node:fs");
const cliColor = require("cli-color");
const path = require("node:path");
const config = require("./configuration.js");
const convertUnits = require("./convertUnits.js");
const getStats = require("./getStats.js");
const webhook = require("./webhook.js");
const uptimeFormatter = require("./uptimeFormatter.js");
module.exports = function App() {
console.log(cliColor.cyanBright("[PteroStats] ") + cliColor.green("Starting app..."));
const client = new Client({
intents: [GatewayIntentBits.Guilds]
});
async function startGetStatus() {
try {
const results = await getStats();
if (results.isPanelDown) webhook(
new EmbedBuilder()
.setTitle("Panel Online")
.setColor("57F287")
.setDescription(`Panel is back online`)
)
createMessage({
panel: true,
uptime: results.uptime,
nodes: results.nodes,
servers: results.servers,
users: results.users,
});
} catch (error) {
if (config.log_error) console.error(error)
console.log(cliColor.cyanBright("[PteroStats] ") + cliColor.redBright("Panel is currently offline."));
fs.readFile(path.join(__dirname, "../cache.json"), (err, data) => {
if (err) {
createMessage({ cache: false, panel: false });
return console.log(cliColor.cyanBright("[PteroStats] ") + cliColor.redBright("Last cache was not found!"));
}
try {
const results = JSON.parse(data);
if (results.uptime) webhook(
new EmbedBuilder()
.setTitle("Panel Offline")
.setColor("ED4245")
.setDescription(`Panel is currently offline`)
);
results.uptime = false;
fs.writeFileSync("cache.json", JSON.stringify(results, null, 2), "utf8");
createMessage({
cache: true,
panel: false,
nodes: results.nodes,
servers: results.servers,
users: results.users,
});
} catch {
createMessage({ cache: false, panel: false });
console.log(cliColor.cyanBright("[PteroStats] ") + cliColor.redBright("Something went wrong with cache data..."));
}
});
}
}
client.once("ready", () => {
console.log(cliColor.cyanBright("[PteroStats] ") + cliColor.green(`${cliColor.blueBright(client.user.tag)} is online!`));
if (config.presence.enable) {
if (config.presence.text && config.presence.type) {
switch (config.presence.type.toLowerCase()) {
case "playing":
config.presence.type = ActivityType.Playing;
break;
case "listening":
config.presence.type = ActivityType.Listening;
break;
case "competing":
config.presence.type = ActivityType.Competing;
break;
default:
config.presence.type = ActivityType.Watching;
}
client.user.setActivity(config.presence.text, {
type: config.presence.type,
});
}
if (config.presence.status) {
if (!["idle", "online", "dnd", "invisible"].includes(
config.presence.status.toLowerCase()
))
config.presence.status = "online";
client.user.setStatus(config.presence.status);
}
}
startGetStatus();
});
async function createMessage({ cache, panel, uptime, nodes, servers, users }) {
let embed = new EmbedBuilder()
.setAuthor({
name: config.embed.nodes.author.name || null,
iconURL: config.embed.nodes.author.icon || null
})
.setDescription(config.embed.nodes.description || null)
.setTitle(config.embed.nodes.title || null)
.setColor(config.embed.nodes.color || null)
.setURL(config.embed.nodes.url || null)
.setThumbnail(config.embed.nodes.thumbnail || null)
let embeds = [embed];
if (config.nodes_settings.details) {
nodes.forEach((node, index) => {
if (index % 25 === 0 && index !== 0) {
embed = new EmbedBuilder().setColor(config.embed.nodes.color);
if (embeds.length < 9) embeds.push(embed);
}
embed.addFields({
name: `${node.attributes.name} - ${node.status ? config.status.online : config.status.offline}`,
value:
"```\n" +
(config.nodes_settings.host ? `Host : ${node.attributes.fqdn}\n` : "") +
`Memory : ${convertUnits(node.attributes.allocated_resources.memory, node.attributes.memory, config.nodes_settings.unit)}\n` +
`Disk : ${convertUnits(node.attributes.allocated_resources.disk, node.attributes.disk, config.nodes_settings.unit)}` +
(node.attributes?.allocated_resources?.cpu ? `\nCPU : ${node.attributes?.allocated_resources?.cpu || 0}%` : "") +
(config.nodes_settings.servers ? `\nServers: ${node.attributes.relationships.servers}${config.nodes_settings.allocations_as_max_servers ? ` / ${node.attributes.relationships.allocations}` : ""}` : "") +
(config.nodes_settings.uptime ? `\nUptime : ${node.uptime ? uptimeFormatter(Date.now() - node.uptime) : "N/A"}` : "") +
"```"
});
});
} else {
embeds[0].setDescription((embed.data.description ? (embed.data.description + "\n\n") : "") + nodes.map(node => `**${node.attributes.name}** - ${node.status ? config.status.online : config.status.offline}`).join("\n"));
}
let panelEmbed = new EmbedBuilder()
.setAuthor({
name: config.embed.panel.author.name || null,
iconURL: config.embed.panel.author.icon || null
})
.setColor(config.embed.panel.color || null)
.setTitle(config.embed.panel.title || null)
.setURL(config.embed.panel.url || null)
.setTimestamp(config.embed.panel.timestamp ? new Date() : null)
.setFooter({
text: config.embed.panel.footer.text || null,
iconURL: config.embed.panel.footer.icon || null
})
.setThumbnail(config.embed.panel.thumbnail || null)
.setImage(config.embed.panel.image || null)
.setDescription(
config.embed.panel.description
.replace(
"{{time}}",
time(new Date(Date.now() + (config.refresh * 1000) + 1000), "R")
) || null
)
.addFields({
name: `Panel - ${panel ? config.status.online : config.status.offline}`,
value:
"```\n" +
(config.panel_settings.host ? `Host : ${new URL(process.env.PanelURL).host}\n` : "") +
`Nodes : ${nodes.length}\n` +
(config.panel_settings.servers ? `Servers: ${servers || "Unknown"}\n` : "") +
(config.panel_settings.users ? `Users : ${users || "Unknown"}\n` : "") +
(config.panel_settings.uptime ? `Uptime : ${uptime ? uptimeFormatter(Date.now() - uptime) : "N/A"}\n` : "") +
"```"
});
if (config.panel_settings.status) embeds.unshift(panelEmbed);
embeds[embeds.length - 1]
.setTimestamp(config.embed.nodes.timestamp ? new Date() : null)
.setFooter({
text: config.embed.nodes.footer.text || null,
iconURL: config.embed.nodes.footer.icon || null
})
.setImage(config.embed.nodes.image || null)
if (!cache && !panel) {
embeds[embeds.length - 1].setDescription(
embeds[embeds.length - 1].data.description ? embeds[embeds.length - 1].data.description + "\n\nThere are no nodes to be display!" : "There are no nodes to be display!"
);
}
const components = []
if (config.button.enable) {
for (const row of ["row1", "row2", "row3", "row4", "row5"]) {
const buttons = config.button[row]?.slice(0, 5).filter(button => button.label && button.url);
if (buttons && buttons.length > 0) {
components.push(
new ActionRowBuilder().addComponents(
buttons.map(button =>
new ButtonBuilder()
.setLabel(button.label)
.setURL(button.url)
.setStyle(ButtonStyle.Link)
)
)
);
}
}
}
try {
const channel = await client.channels.fetch(process.env?.DiscordChannel);
const messages = await channel.messages.fetch({ limit: 10 });
const botMessage = messages.find(msg => msg.author.id === client.user.id);
console.log(cliColor.cyanBright("[PteroStats] ") + cliColor.green(`Panel stats successfully posted to the ${cliColor.blueBright(channel.name)} channel!`));
setTimeout(() => startGetStatus(), config.refresh * 1000);
if (botMessage) {
await botMessage.edit({ content: config.message.content || null, embeds, components });
} else {
await channel.send({ content: config.message.content || null, embeds, components });
}
} catch (error) {
handleDiscordError(error);
}
}
function handleDiscordError(error) {
try {
if (error.rawError?.code === 429) {
console.log(cliColor.cyanBright("[PteroStats] ") + cliColor.redBright("Error 429 | Your IP has been rate limited by either Discord or your website. If it's a rate limit with Discord, you must wait. If it's a issue with your website, consider whitelisting your server IP."));
} else if (error.rawError?.code === 403) {
console.log(cliColor.cyanBright("[PteroStats] ") + cliColor.redBright("FORBIDDEN | The channel ID you provided is incorrect. Please double check you have the right ID. If you're not sure, read our documentation: \n>>https://github.com/HirziDevs/PteroStats#getting-channel-id<<"));
} else if (error.code === "ENOTFOUND") {
console.log(cliColor.cyanBright("[PteroStats] ") + cliColor.redBright("ENOTFOUND | DNS Error. Ensure your network connection and DNS server are functioning correctly."));
} else if (error.rawError?.code === 50001) {
console.log(cliColor.cyanBright("[PteroStats] ") + cliColor.redBright("Discord Error | Your discord bot doesn't have access to see/send message/edit message in the channel!"));
} else if (error.rawError?.errors && Object?.values(error.rawError.errors)[0]?._errors[0]?.code === "MAX_EMBED_SIZE_EXCEEDED") {
console.log(cliColor.cyanBright("[PteroStats] ") + cliColor.redBright("Discord Error | Embed message limit exceeded! Please limit or decrease the nodes that need to be shown in the config!"));
} else if (error.rawError?.errors && Object?.values(error.rawError.errors)[0]?._errors[0]?.code) {
console.log(Object.values(error.rawError.errors)[0]._errors[0].message);
} else {
console.error(cliColor.cyanBright("[PteroStats] ") + cliColor.redBright("Discord Error"), error);
}
process.exit();
} catch (err) {
console.error(error)
process.exit();
}
}
try {
client.login(process.env?.DiscordBotToken);
} catch {
console.log(cliColor.cyanBright("[PteroStats] ") + cliColor.redBright("Discord Error | Invalid Discord Bot Token! Make sure you have the correct token in the config!"));
process.exit();
}
}