2019-09-13 20:02:41 +00:00
|
|
|
const fetch = require("node-fetch");
|
2020-11-20 21:16:52 +00:00
|
|
|
const url = require("url");
|
2021-01-05 02:53:34 +00:00
|
|
|
const { getType } = require("./image.js");
|
2020-11-05 21:40:18 +00:00
|
|
|
const execPromise = require("util").promisify(require("child_process").exec);
|
2019-09-13 20:02:41 +00:00
|
|
|
|
2020-11-20 21:16:52 +00:00
|
|
|
const tenorURLs = [
|
|
|
|
"tenor.com",
|
|
|
|
"www.tenor.com"
|
|
|
|
];
|
|
|
|
const giphyURLs = [
|
|
|
|
"giphy.com",
|
|
|
|
"www.giphy.com"
|
|
|
|
];
|
|
|
|
const imgurURLs = [
|
|
|
|
"imgur.com",
|
|
|
|
"www.imgur.com",
|
|
|
|
"i.imgur.com"
|
|
|
|
];
|
2021-01-18 02:40:52 +00:00
|
|
|
const gfycatURLs = [
|
|
|
|
"gfycat.com",
|
|
|
|
"www.gfycat.com",
|
|
|
|
"thumbs.gfycat.com",
|
|
|
|
"giant.gfycat.com"
|
|
|
|
];
|
2020-11-20 21:16:52 +00:00
|
|
|
|
2021-01-05 02:53:34 +00:00
|
|
|
const formats = ["image/jpeg", "image/png", "image/webp", "image/gif"];
|
|
|
|
|
2020-10-18 21:53:35 +00:00
|
|
|
// gets the proper image paths
|
|
|
|
const getImage = async (image, image2, gifv = false) => {
|
2019-09-13 20:02:41 +00:00
|
|
|
try {
|
2020-10-19 23:48:43 +00:00
|
|
|
const payload = {
|
|
|
|
url: image2,
|
|
|
|
path: image
|
|
|
|
};
|
2020-10-18 21:53:35 +00:00
|
|
|
if (gifv) {
|
2020-11-20 21:16:52 +00:00
|
|
|
const host = url.parse(image2).host;
|
|
|
|
if (tenorURLs.includes(host)) {
|
2020-11-05 21:40:18 +00:00
|
|
|
if (process.env.TENOR !== "") {
|
|
|
|
const data = await fetch(`https://api.tenor.com/v1/gifs?ids=${image2.split("-").pop()}&key=${process.env.TENOR}`);
|
|
|
|
const json = await data.json();
|
|
|
|
payload.path = json.results[0].media[0].gif.url;
|
|
|
|
} else {
|
|
|
|
const delay = (await execPromise(`ffprobe -v 0 -of csv=p=0 -select_streams v:0 -show_entries stream=r_frame_rate ${image}`)).stdout.replace("\n", "");
|
|
|
|
payload.delay = (100 / delay.split("/")[0]) * delay.split("/")[1];
|
|
|
|
}
|
2020-11-20 21:16:52 +00:00
|
|
|
} else if (giphyURLs.includes(host)) {
|
2020-10-19 23:48:43 +00:00
|
|
|
payload.path = `https://media0.giphy.com/media/${image2.split("-").pop()}/giphy.gif`;
|
2020-11-20 21:16:52 +00:00
|
|
|
} else if (imgurURLs.includes(host)) {
|
2020-10-19 23:48:43 +00:00
|
|
|
payload.path = image.replace(".mp4", ".gif");
|
2021-01-18 02:40:52 +00:00
|
|
|
} else if (gfycatURLs.includes(host)) {
|
|
|
|
payload.path = `https://thumbs.gfycat.com/${image.split("/").pop().split(".mp4")[0]}-size_restricted.gif`;
|
2020-10-18 21:53:35 +00:00
|
|
|
}
|
2020-10-20 01:24:53 +00:00
|
|
|
payload.type = "image/gif";
|
2021-01-05 02:53:34 +00:00
|
|
|
} else {
|
|
|
|
payload.type = await getType(payload.path);
|
|
|
|
if (!payload.type || !formats.includes(payload.type)) return;
|
2019-09-13 20:02:41 +00:00
|
|
|
}
|
2020-10-18 21:53:35 +00:00
|
|
|
return payload;
|
2019-09-13 20:02:41 +00:00
|
|
|
} catch (error) {
|
2020-07-17 00:53:44 +00:00
|
|
|
if (error.name === "AbortError") {
|
|
|
|
throw Error("Timed out");
|
|
|
|
} else {
|
|
|
|
throw error;
|
|
|
|
}
|
2019-09-13 20:02:41 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2020-08-16 16:48:37 +00:00
|
|
|
const checkImages = async (message) => {
|
|
|
|
let type;
|
|
|
|
// first check the embeds
|
|
|
|
if (message.embeds.length !== 0) {
|
2020-10-18 21:53:35 +00:00
|
|
|
// embeds can vary in types, we check for tenor gifs first
|
2020-08-16 16:48:37 +00:00
|
|
|
if (message.embeds[0].type === "gifv") {
|
2020-10-18 21:53:35 +00:00
|
|
|
type = await getImage(message.embeds[0].video.url, message.embeds[0].url, true);
|
|
|
|
// then we check for other image types
|
2021-01-06 22:10:31 +00:00
|
|
|
} else if ((message.embeds[0].type === "video" || message.embeds[0].type === "image") && message.embeds[0].thumbnail) {
|
2020-10-18 21:53:35 +00:00
|
|
|
type = await getImage(message.embeds[0].thumbnail.proxy_url, message.embeds[0].thumbnail.url);
|
|
|
|
// finally we check both possible image fields for "generic" embeds
|
|
|
|
} else if (message.embeds[0].type === "rich") {
|
|
|
|
if (message.embeds[0].thumbnail) {
|
|
|
|
type = await getImage(message.embeds[0].thumbnail.proxy_url, message.embeds[0].thumbnail.url);
|
|
|
|
} else if (message.embeds[0].image) {
|
|
|
|
type = await getImage(message.embeds[0].image.proxy_url, message.embeds[0].image.url);
|
|
|
|
}
|
2020-08-16 16:48:37 +00:00
|
|
|
}
|
|
|
|
// then check the attachments
|
2020-10-18 21:53:35 +00:00
|
|
|
} else if (message.attachments.length !== 0 && message.attachments[0].width) {
|
|
|
|
type = await getImage(message.attachments[0].proxy_url, message.attachments[0].url);
|
2020-08-16 16:48:37 +00:00
|
|
|
}
|
|
|
|
// if the file is an image then return it
|
|
|
|
return type ? type : false;
|
|
|
|
};
|
|
|
|
|
2019-09-13 20:02:41 +00:00
|
|
|
// this checks for the latest message containing an image and returns the url of the image
|
Class commands, improved sharding, and many other changes (#88)
* Load commands recursively
* Sort commands
* Missed a couple of spots
* missed even more spots apparently
* Ported commands in "fun" category to new class-based format, added babel eslint plugin
* Ported general commands, removed old/unneeded stuff, replaced moment with day, many more fixes I lost track of
* Missed a spot
* Removed unnecessary abort-controller package, add deprecation warning for mongo database
* Added imagereload, clarified premature end message
* Fixed docker-compose path issue, added total bot uptime to stats, more fixes for various parts
* Converted image commands into classes, fixed reload, ignore another WS event, cleaned up command handler and image runner
* Converted music/soundboard commands to class format
* Cleanup unnecessary logs
* awful tag command class port
* I literally somehow just learned that you can leave out the constructor in classes
* Pass client directly to commands/events, cleaned up command handler
* Migrated bot to eris-sharder, fixed some error handling stuff
* Remove unused modules
* Fixed type returning
* Switched back to Eris stable
* Some fixes and cleanup
* might wanna correct this
* Implement image command ratelimiting
* Added Bot token prefix, added imagestats, added running endpoint to API
2021-04-12 16:16:12 +00:00
|
|
|
module.exports = async (client, cmdMessage) => {
|
2020-08-16 16:48:37 +00:00
|
|
|
// we start by checking the current message for images
|
|
|
|
const result = await checkImages(cmdMessage);
|
|
|
|
if (result !== false) return result;
|
2020-11-26 15:31:24 +00:00
|
|
|
// if there aren't any in the current message then check if there's a reply
|
|
|
|
if (cmdMessage.messageReference) {
|
2020-12-26 18:17:10 +00:00
|
|
|
const replyMessage = await client.getMessage(cmdMessage.messageReference.channelID, cmdMessage.messageReference.messageID);
|
|
|
|
if (replyMessage) {
|
|
|
|
const replyResult = await checkImages(replyMessage);
|
|
|
|
if (replyResult !== false) return replyResult;
|
2020-11-26 15:31:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// if there aren't any replies then iterate over the last few messages in the channel
|
2019-09-13 20:02:41 +00:00
|
|
|
const messages = await cmdMessage.channel.getMessages();
|
|
|
|
// iterate over each message
|
|
|
|
for (const message of messages) {
|
2020-08-16 16:48:37 +00:00
|
|
|
const result = await checkImages(message);
|
|
|
|
if (result === false) {
|
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
return result;
|
2019-09-13 20:02:41 +00:00
|
|
|
}
|
|
|
|
}
|
2020-02-26 17:29:13 +00:00
|
|
|
};
|