diff --git a/config.js b/config.js index 13b7925e5..6c319b1d2 100644 --- a/config.js +++ b/config.js @@ -29,6 +29,7 @@ module.exports = { BASE_API: "https://image-api.strangebot.xyz", }, MUSIC: { + IDLE_TIME: 60, // Time in seconds before the bot disconnects from the voice channel MAX_SEARCH_RESULTS: 5, NODES: [ { diff --git a/dashboard/utils.js b/dashboard/utils.js index 09371f21f..926b72cae 100644 --- a/dashboard/utils.js +++ b/dashboard/utils.js @@ -11,9 +11,9 @@ async function fetchGuild(guildID, client, guilds) { async function fetchUser(userData, client, query) { if (userData.guilds) { userData.guilds.forEach((guild) => { - const perms = new Discord.Permissions(BigInt(guild.permissions)); - if (perms.has("MANAGE_GUILD")) { - guild.admin = true; + if (guild.permissions) { + const perms = new Discord.Permissions(BigInt(guild.permissions)); + if (perms.has("MANAGE_GUILD")) guild.admin = true; } guild.settingsUrl = client.guilds.cache.get(guild.id) ? `/manage/${guild.id}/` diff --git a/src/commands/admin/greeting/farewell.js b/src/commands/admin/greeting/farewell.js index 8a91de47c..3d771aae4 100644 --- a/src/commands/admin/greeting/farewell.js +++ b/src/commands/admin/greeting/farewell.js @@ -303,10 +303,7 @@ async function setDescription(settings, desc) { async function setThumbnail(settings, status) { settings.farewell.embed.thumbnail = status.toUpperCase() === "ON" ? true : false; - - settings.farewell.embed.thumbnail = null; await settings.save(); - return "Configuration saved! Farewell message updated"; } diff --git a/src/commands/admin/greeting/welcome.js b/src/commands/admin/greeting/welcome.js index 6059ab9a5..b8c8f8d0c 100644 --- a/src/commands/admin/greeting/welcome.js +++ b/src/commands/admin/greeting/welcome.js @@ -303,10 +303,7 @@ async function setDescription(settings, desc) { async function setThumbnail(settings, status) { settings.welcome.embed.thumbnail = status.toUpperCase() === "ON" ? true : false; - - settings.welcome.embed.thumbnail = null; await settings.save(); - return "Configuration saved! Welcome message updated"; } diff --git a/src/commands/admin/ticket/ticket.js b/src/commands/admin/ticket/ticket.js index 9dda5783a..6e5ea3296 100644 --- a/src/commands/admin/ticket/ticket.js +++ b/src/commands/admin/ticket/ticket.js @@ -181,7 +181,7 @@ module.exports = class Ticket extends Command { if (args.length < 2) return message.reply("Please provide a channel where ticket logs must be sent"); const target = getMatchingChannel(message.guild, args[1]); if (target.length === 0) return message.reply("Could not find any matching channel"); - response = await setupLogChannel(message, target); + response = await setupLogChannel(message, target[0]); } // Set limit diff --git a/src/commands/fun/facts.js b/src/commands/fun/facts.js index 8a29644ae..1cee7933c 100644 --- a/src/commands/fun/facts.js +++ b/src/commands/fun/facts.js @@ -17,6 +17,7 @@ module.exports = class FactCommand extends Command { command: { enabled: true, usage: "", + aliases: ["fact"], minArgsCount: 1, }, slashCommand: { @@ -61,10 +62,12 @@ async function getFact(user, choice) { const response = await getJson(`${BASE_URL}/${choice}`); if (!response.success) return MESSAGES.API_ERROR; - const imageUrl = response.data?.fact; + const fact = response.data?.fact; + const imageUrl = response.data?.image; const embed = new MessageEmbed() .setColor(EMBED_COLORS.TRANSPARENT) - .setImage(imageUrl) + .setThumbnail(imageUrl) + .setDescription(fact) .setFooter(`Requested by ${user.tag}`); return { embeds: [embed] }; diff --git a/src/commands/information/ping.js b/src/commands/information/ping.js index 9b29fe19a..05d1bc389 100644 --- a/src/commands/information/ping.js +++ b/src/commands/information/ping.js @@ -6,9 +6,9 @@ module.exports = class PingCommand extends Command { super(client, { name: "ping", description: "shows the current ping from the bot to the discord servers", + category: "INFORMATION", command: { enabled: true, - category: "INFORMATION", }, slashCommand: { enabled: true, diff --git a/src/commands/utility/help.js b/src/commands/utility/help.js index 1e1bb8395..8b3afd689 100644 --- a/src/commands/utility/help.js +++ b/src/commands/utility/help.js @@ -171,21 +171,21 @@ const waiter = (msg, userId, prefix) => { arrEmbeds = prefix ? getMsgCategoryEmbeds(msg.client, cat, prefix) : getSlashCategoryEmbeds(msg.client, cat); currentPage = 0; buttonsRow.components.forEach((button) => button.setDisabled(arrEmbeds.length > 1 ? false : true)); - await msg.edit({ embeds: [arrEmbeds[currentPage]], components: [menuRow, buttonsRow] }); + msg.editable && (await msg.edit({ embeds: [arrEmbeds[currentPage]], components: [menuRow, buttonsRow] })); break; } case "previousBtn": if (currentPage !== 0) { --currentPage; - await msg.edit({ embeds: [arrEmbeds[currentPage]], components: [menuRow, buttonsRow] }); + msg.editable && (await msg.edit({ embeds: [arrEmbeds[currentPage]], components: [menuRow, buttonsRow] })); } break; case "nextBtn": if (currentPage < arrEmbeds.length - 1) { currentPage++; - await msg.edit({ embeds: [arrEmbeds[currentPage]], components: [menuRow, buttonsRow] }); + msg.editable && (await msg.edit({ embeds: [arrEmbeds[currentPage]], components: [menuRow, buttonsRow] })); } break; } @@ -193,7 +193,7 @@ const waiter = (msg, userId, prefix) => { collector.on("end", () => { if (cache[`${msg.guildId}|${userId}`]) delete cache[`${msg.guildId}|${userId}`]; - return msg.edit({ components: [] }); + return msg.editable && msg.edit({ components: [] }); }); }; diff --git a/src/events/guild/guildDelete.js b/src/events/guild/guildDelete.js index 53620cdce..dc51c57cf 100644 --- a/src/events/guild/guildDelete.js +++ b/src/events/guild/guildDelete.js @@ -6,7 +6,6 @@ const { getSettings } = require("@schemas/Guild"); * @param {import('discord.js').Guild} guild */ module.exports = async (client, guild) => { - if (!guild.members.cache.has(guild.ownerId)) await guild.fetchOwner({ cache: true }); client.logger.log(`Guild Left: ${guild.name} Members: ${guild.memberCount}`); const settings = await getSettings(guild); @@ -15,13 +14,14 @@ module.exports = async (client, guild) => { if (!client.joinLeaveWebhook) return; + const owner = await client.users.fetch(guild.ownerId); const embed = new MessageEmbed() .setTitle("Guild Left") .setThumbnail(guild.iconURL()) .setColor(client.config.EMBED_COLORS.ERROR) .addField("Name", guild.name, false) .addField("ID", guild.id, false) - .addField("Owner", `${client.users.cache.get(guild.ownerId).tag} [\`${guild.ownerId}\`]`, false) + .addField("Owner", `${owner.username}#${owner.discriminator} [\`${owner.id}\`]`, false) .addField("Members", `\`\`\`yaml\n${guild.memberCount}\`\`\``, false) .setFooter(`Guild #${client.guilds.cache.size}`); diff --git a/src/events/ready.js b/src/events/ready.js index 282647f82..ff62f3951 100644 --- a/src/events/ready.js +++ b/src/events/ready.js @@ -1,6 +1,7 @@ const { counterHandler, inviteHandler } = require("@src/handlers"); const { cacheReactionRoles } = require("@schemas/Message"); const { getSettings } = require("@schemas/Guild"); +const { updateCounterChannels } = require("@src/handlers/counter"); /** * @param {import('@src/structures').BotClient} client @@ -38,6 +39,8 @@ module.exports = async (client) => { inviteHandler.cacheGuildInvites(guild); } } + + setInterval(() => updateCounterChannels(client), 10 * 60 * 1000); }; /** diff --git a/src/events/voice/voiceStateUpdate.js b/src/events/voice/voiceStateUpdate.js index 3b9b0c7f1..a47600849 100644 --- a/src/events/voice/voiceStateUpdate.js +++ b/src/events/voice/voiceStateUpdate.js @@ -1,3 +1,5 @@ +const { MUSIC } = require("@root/config"); + /** * @param {import('@src/structures').BotClient} client * @param {import('discord.js').VoiceState} oldState @@ -10,12 +12,12 @@ module.exports = async (client, oldState, newState) => { if (oldState.channelId !== guild.me.voice.channelId || newState.channel) return; // otherwise, check how many people are in the channel now - if (!oldState.channel.members.size - 1) { + if (oldState.channel.members.size === 1) { setTimeout(() => { // if 1 (you), wait 1 minute if (!oldState.channel.members.size - 1) // if there's still 1 member, client.musicManager.get(guild.id) && client.musicManager.get(guild.id).destroy(); - }, 1 * 60 * 1000); // (check and disconnect after 1 min) + }, MUSIC.IDLE_TIME * 1000); } }; diff --git a/src/helpers/logger.js b/src/helpers/logger.js index 57355111e..4cb017531 100644 --- a/src/helpers/logger.js +++ b/src/helpers/logger.js @@ -15,10 +15,13 @@ simpleLogger.setLevel("debug"); const errorWebhook = process.env.ERROR_LOGS ? new WebhookClient({ url: process.env.ERROR_LOGS }) : undefined; const sendWebhook = (content, err) => { + if (!content && !err) return; + const errString = err?.stack || err; + const embed = new MessageEmbed() .setColor(config.EMBED_COLORS.ERROR) .setAuthor(err?.name || "Error") - .setDescription("```js\n" + err?.stack || err + "```"); + .setDescription("```js\n" + (errString.length > 4096 ? `${errString.substr(0, 4000)}...` : errString) + "\n```"); if (err?.description) embed.addField("Description", content); if (err?.message) embed.addField("Message", err?.message); diff --git a/src/structures/Command.js b/src/structures/Command.js index 3ec6e0f90..76d321334 100644 --- a/src/structures/Command.js +++ b/src/structures/Command.js @@ -158,7 +158,7 @@ class Command { try { await this.messageRun(message, args, invoke, prefix); } catch (ex) { - await message.reply("Oops! An error occurred while running the command"); + await message.channel.send("Oops! An error occurred while running the command"); this.client.logger.error("messageRun", ex); } finally { this.applyCooldown(message.author.id); diff --git a/src/utils/botUtils.js b/src/utils/botUtils.js index 7cfe93c77..6da91656f 100644 --- a/src/utils/botUtils.js +++ b/src/utils/botUtils.js @@ -75,7 +75,7 @@ async function sendMessage(channel, content, seconds) { if (content.embeds && content.embeds.length > 0) perms.push("EMBED_LINKS"); if (channel.type !== "DM" && !channel.permissionsFor(channel.guild.me).has(perms)) return; try { - if (!seconds) return channel.send(content); + if (!seconds) return await channel.send(content); const reply = await channel.send(content); setTimeout(() => reply.deletable && reply.delete().catch((ex) => {}), seconds * 1000); } catch (ex) { @@ -92,7 +92,7 @@ async function safeDM(user, message, seconds) { if (!user || !message) return; try { const dm = await user.createDM(); - if (!seconds) return dm.send(message); + if (!seconds) return await dm.send(message); const reply = await dm.send(message); setTimeout(() => reply.deletable && reply.delete().catch((ex) => {}), seconds * 1000); } catch (ex) {