blob: acd3b41603f34341374982cdcf68dc8f0ec24416 [file] [log] [blame]
pineafan63fc5e22022-08-04 22:04:10 +01001import fetch from "node-fetch";
Skyler Grey32179982023-03-07 23:59:06 +00002import { writeFileSync } from "fs";
pineafan63fc5e22022-08-04 22:04:10 +01003import generateFileName from "../utils/temp/generateFileName.js";
4import Tesseract from "node-tesseract-ocr";
5import type Discord from "discord.js";
pineafan3a02ea32022-08-11 21:35:04 +01006import client from "../utils/client.js";
TheCodedProfb5e9d552023-01-29 15:43:26 -05007import { createHash } from "crypto";
Skyler Grey32179982023-03-07 23:59:06 +00008import * as nsfwjs from "nsfwjs";
Skyler Grey0a4846c2023-03-08 00:32:01 +00009import ClamScan from "clamscan";
Skyler Grey62da9bf2023-03-08 00:11:00 +000010import * as tf from "@tensorflow/tfjs-node";
pineafan6de4da52023-03-07 20:43:44 +000011import EmojiEmbed from "../utils/generateEmojiEmbed.js";
12import getEmojiByName from "../utils/getEmojiByName.js";
13import { ActionRowBuilder, ButtonBuilder, ButtonStyle } from "discord.js";
Skyler Greyd1157312023-03-08 10:07:38 +000014import config from "../config/main.js";
Skyler Grey0d885222023-03-08 21:46:37 +000015import GIFEncoder from "gifencoder";
16import gm_var from 'gm';
17const gm = gm_var.subClass({ imageMagick: '7+' });
pineafan813bdf42022-07-24 10:39:10 +010018
Skyler Grey75ea9172022-08-06 10:22:23 +010019interface NSFWSchema {
20 nsfw: boolean;
TheCodedProf5b53a8c2023-02-03 15:40:26 -050021 errored?: boolean;
Skyler Grey75ea9172022-08-06 10:22:23 +010022}
23interface MalwareSchema {
Skyler Grey0d885222023-03-08 21:46:37 +000024 malware: boolean;
TheCodedProf5b53a8c2023-02-03 15:40:26 -050025 errored?: boolean;
Skyler Grey75ea9172022-08-06 10:22:23 +010026}
pineafan813bdf42022-07-24 10:39:10 +010027
Skyler Grey14432712023-03-07 23:40:50 +000028const nsfw_model = await nsfwjs.load();
Skyler Greyd1157312023-03-08 10:07:38 +000029const clamscanner = await new ClamScan().init({
30 clamdscan: {
31 socket: config.clamavSocket
32 }
33});
TheCodedProfd8ef1f32023-03-06 19:15:18 -050034
Skyler Grey0d885222023-03-08 21:46:37 +000035export async function testNSFW(attachment: {
36 url: string;
37 local: string;
38 height: number | null;
39 width: number | null;
40}): Promise<NSFWSchema> {
41 const [fileStream, hash] = await streamAttachment(attachment.url);
Skyler Greyda16adf2023-03-05 10:22:12 +000042 const alreadyHaveCheck = await client.database.scanCache.read(hash);
Skyler Grey0d885222023-03-08 21:46:37 +000043 if (alreadyHaveCheck && ("nsfw" in alreadyHaveCheck!)) {
44 return { nsfw: alreadyHaveCheck.nsfw }
45 };
TheCodedProfd8ef1f32023-03-06 19:15:18 -050046
Skyler Grey0d885222023-03-08 21:46:37 +000047 const image = gm(fileStream).command('convert').in('-')
TheCodedProfd8ef1f32023-03-06 19:15:18 -050048
Skyler Grey0d885222023-03-08 21:46:37 +000049 const encoder = new GIFEncoder(attachment.width ?? 1024, attachment.height ?? 1024);
50
51
52 // const array = new Uint8Array(fileStream);
53 const img = tf.node.decodeImage(array) as tf.Tensor3D;
54
55 const predictions = (await nsfw_model.classify(img, 1))[0]!;
56 console.log(2, predictions);
TheCodedProfd8ef1f32023-03-06 19:15:18 -050057
Skyler Greyd1157312023-03-08 10:07:38 +000058 const nsfw = predictions.className === "Hentai" || predictions.className === "Porn";
59 await client.database.scanCache.write(hash, "nsfw", nsfw);
60
61 return { nsfw };
pineafan813bdf42022-07-24 10:39:10 +010062}
63
pineafan02ba0232022-07-24 22:16:15 +010064export async function testMalware(link: string): Promise<MalwareSchema> {
Skyler Grey0a4846c2023-03-08 00:32:01 +000065 const [fileName, hash] = await saveAttachment(link);
Skyler Greyda16adf2023-03-05 10:22:12 +000066 const alreadyHaveCheck = await client.database.scanCache.read(hash);
Skyler Grey0d885222023-03-08 21:46:37 +000067 if (alreadyHaveCheck?.malware !== undefined) return { malware: alreadyHaveCheck.malware };
Skyler Greyd1157312023-03-08 10:07:38 +000068 let malware;
Skyler Grey0a4846c2023-03-08 00:32:01 +000069 try {
Skyler Greyd1157312023-03-08 10:07:38 +000070 malware = (await clamscanner.scanFile(fileName)).isInfected;
Skyler Grey0a4846c2023-03-08 00:32:01 +000071 } catch (e) {
Skyler Grey0d885222023-03-08 21:46:37 +000072 return { malware: true };
Skyler Grey0a4846c2023-03-08 00:32:01 +000073 }
Skyler Greyf4f21c42023-03-08 14:36:29 +000074 await client.database.scanCache.write(hash, "malware", malware);
Skyler Grey0d885222023-03-08 21:46:37 +000075 return { malware };
pineafan3a02ea32022-08-11 21:35:04 +010076}
77
78export async function testLink(link: string): Promise<{ safe: boolean; tags: string[] }> {
Skyler Greyda16adf2023-03-05 10:22:12 +000079 const alreadyHaveCheck = await client.database.scanCache.read(link);
Skyler Grey0d885222023-03-08 21:46:37 +000080 if (alreadyHaveCheck?.bad_link !== undefined)
81 return { safe: alreadyHaveCheck.bad_link, tags: alreadyHaveCheck.tags ?? [] };
82 return { safe: true, tags: [] };
83 // const scanned: { safe?: boolean; tags?: string[] } = {}
84 // await client.database.scanCache.write(link, "bad_link", scanned.safe ?? true, scanned.tags ?? []);
85 // return {
86 // safe: scanned.safe ?? true,
87 // tags: scanned.tags ?? []
88 // };
pineafan813bdf42022-07-24 10:39:10 +010089}
90
Skyler Grey0d885222023-03-08 21:46:37 +000091export async function streamAttachment(link: string): Promise<[Buffer, string]> {
Skyler Greyda16adf2023-03-05 10:22:12 +000092 const image = await (await fetch(link)).arrayBuffer();
Skyler Grey32179982023-03-07 23:59:06 +000093 const enc = new TextDecoder("utf-8");
Skyler Grey0d885222023-03-08 21:46:37 +000094 const buf = Buffer.from(image);
95 return [buf, createHash("sha512").update(enc.decode(image), "base64").digest("base64")];
Skyler Grey32179982023-03-07 23:59:06 +000096}
97
98export async function saveAttachment(link: string): Promise<[string, string]> {
99 const image = await (await fetch(link)).arrayBuffer();
Skyler Greyf4f21c42023-03-08 14:36:29 +0000100 const fileName = await generateFileName(link.split("/").pop()!.split(".").pop()!);
TheCodedProf5b53a8c2023-02-03 15:40:26 -0500101 const enc = new TextDecoder("utf-8");
Skyler Grey0d885222023-03-08 21:46:37 +0000102 writeFileSync(fileName, new DataView(image));
Skyler Grey32179982023-03-07 23:59:06 +0000103 return [fileName, createHash("sha512").update(enc.decode(image), "base64").digest("base64")];
pineafan813bdf42022-07-24 10:39:10 +0100104}
105
pineafan813bdf42022-07-24 10:39:10 +0100106const linkTypes = {
Skyler Grey75ea9172022-08-06 10:22:23 +0100107 PHISHING: "Links designed to trick users into clicking on them.",
108 DATING: "Dating sites.",
109 TRACKERS: "Websites that store or track personal information.",
110 ADVERTISEMENTS: "Websites only for ads.",
Skyler Grey11236ba2022-08-08 21:13:33 +0100111 FACEBOOK: "Facebook pages. (Facebook has a number of dangerous trackers. Read more on /privacy)",
Skyler Grey75ea9172022-08-06 10:22:23 +0100112 AMP: "AMP pages. (AMP is a technology that allows websites to be served by Google. Read more on /privacy)",
pineafan813bdf42022-07-24 10:39:10 +0100113 "FACEBOOK TRACKERS": "Websites that include trackers from Facebook.",
Skyler Grey11236ba2022-08-08 21:13:33 +0100114 "IP GRABBERS": "Websites that store your IP address, which shows your approximate location.",
Skyler Grey75ea9172022-08-06 10:22:23 +0100115 PORN: "Websites that include pornography.",
116 GAMBLING: "Gambling sites, often scams.",
Skyler Grey11236ba2022-08-08 21:13:33 +0100117 MALWARE: "Websites which download files designed to break or slow down your device.",
Skyler Grey75ea9172022-08-06 10:22:23 +0100118 PIRACY: "Sites which include illegally downloaded material.",
Skyler Grey11236ba2022-08-08 21:13:33 +0100119 RANSOMWARE: "Websites which download a program that can steal your data and make you pay to get it back.",
Skyler Grey75ea9172022-08-06 10:22:23 +0100120 REDIRECTS: "Sites like bit.ly which could redirect to a malicious site.",
121 SCAMS: "Sites which are designed to trick you into doing something.",
122 TORRENT: "Websites that download torrent files.",
123 HATE: "Websites that spread hate towards groups or individuals.",
124 JUNK: "Websites that are designed to make you waste time."
pineafan63fc5e22022-08-04 22:04:10 +0100125};
pineafan813bdf42022-07-24 10:39:10 +0100126export { linkTypes };
127
pineafan63fc5e22022-08-04 22:04:10 +0100128export async function LinkCheck(message: Discord.Message): Promise<string[]> {
Skyler Grey75ea9172022-08-06 10:22:23 +0100129 const links =
130 message.content.match(
131 /https?:\/\/(www\.)?[-a-zA-Z0-9@:%._+~#=]{1,256}\.[a-zA-Z0-9()]{1,6}\b([-a-zA-Z0-9()@:%_+.~#?&//=]*)/gi
132 ) ?? [];
133 const detections: { tags: string[]; safe: boolean }[] = [];
134 const promises: Promise<void>[] = links.map(async (element) => {
pineafan63fc5e22022-08-04 22:04:10 +0100135 let returned;
pineafan813bdf42022-07-24 10:39:10 +0100136 try {
Skyler Grey11236ba2022-08-08 21:13:33 +0100137 if (element.match(/https?:\/\/[a-zA-Z]+\.?discord(app)?\.(com|net)\/?/)) return; // Also matches discord.net, not enough of a bug
pineafan63fc5e22022-08-04 22:04:10 +0100138 returned = await testLink(element);
139 } catch {
Skyler Grey75ea9172022-08-06 10:22:23 +0100140 detections.push({ tags: [], safe: true });
pineafan63fc5e22022-08-04 22:04:10 +0100141 return;
142 }
Skyler Grey75ea9172022-08-06 10:22:23 +0100143 detections.push({ tags: returned.tags, safe: returned.safe });
pineafan813bdf42022-07-24 10:39:10 +0100144 });
145 await Promise.all(promises);
Skyler Grey75ea9172022-08-06 10:22:23 +0100146 const detectionsTypes = detections
147 .map((element) => {
Skyler Grey11236ba2022-08-08 21:13:33 +0100148 const type = Object.keys(linkTypes).find((type) => element.tags.includes(type));
Skyler Grey75ea9172022-08-06 10:22:23 +0100149 if (type) return type;
150 // if (!element.safe) return "UNSAFE"
151 return undefined;
152 })
153 .filter((element) => element !== undefined);
pineafan63fc5e22022-08-04 22:04:10 +0100154 return detectionsTypes as string[];
pineafan813bdf42022-07-24 10:39:10 +0100155}
156
Skyler Grey0d885222023-03-08 21:46:37 +0000157export async function NSFWCheck(element: {
158 url: string;
159 local: string;
160 height: number | null;
161 width: number | null;
162}): Promise<boolean> {
pineafan813bdf42022-07-24 10:39:10 +0100163 try {
TheCodedProfb5e9d552023-01-29 15:43:26 -0500164 return (await testNSFW(element)).nsfw;
Skyler Grey0d885222023-03-08 21:46:37 +0000165 } catch (e) {
166 console.log(e)
pineafan63fc5e22022-08-04 22:04:10 +0100167 return false;
pineafan813bdf42022-07-24 10:39:10 +0100168 }
169}
170
Skyler Grey11236ba2022-08-08 21:13:33 +0100171export async function SizeCheck(element: { height: number | null; width: number | null }): Promise<boolean> {
pineafan63fc5e22022-08-04 22:04:10 +0100172 if (element.height === null || element.width === null) return true;
173 if (element.height < 20 || element.width < 20) return false;
174 return true;
pineafan813bdf42022-07-24 10:39:10 +0100175}
176
pineafan63fc5e22022-08-04 22:04:10 +0100177export async function MalwareCheck(element: string): Promise<boolean> {
pineafan813bdf42022-07-24 10:39:10 +0100178 try {
Skyler Grey0d885222023-03-08 21:46:37 +0000179 return (await testMalware(element)).malware;
pineafan813bdf42022-07-24 10:39:10 +0100180 } catch {
pineafan63fc5e22022-08-04 22:04:10 +0100181 return true;
pineafan813bdf42022-07-24 10:39:10 +0100182 }
183}
184
pineafan1e462ab2023-03-07 21:34:06 +0000185export function TestString(
186 string: string,
187 soft: string[],
188 strict: string[],
189 enabled?: boolean
190): { word: string; type: string } | null {
pineafan6de4da52023-03-07 20:43:44 +0000191 if (!enabled) return null;
Skyler Grey75ea9172022-08-06 10:22:23 +0100192 for (const word of strict) {
pineafan813bdf42022-07-24 10:39:10 +0100193 if (string.toLowerCase().includes(word)) {
Skyler Grey75ea9172022-08-06 10:22:23 +0100194 return { word: word, type: "strict" };
pineafan813bdf42022-07-24 10:39:10 +0100195 }
196 }
Skyler Grey75ea9172022-08-06 10:22:23 +0100197 for (const word of soft) {
198 for (const word2 of string.match(/[a-z]+/gi) ?? []) {
pineafane23c4ec2022-07-27 21:56:27 +0100199 if (word2 === word) {
pineafan6de4da52023-03-07 20:43:44 +0000200 return { word: word, type: "soft" };
pineafan813bdf42022-07-24 10:39:10 +0100201 }
202 }
203 }
pineafan63fc5e22022-08-04 22:04:10 +0100204 return null;
pineafan813bdf42022-07-24 10:39:10 +0100205}
206
pineafan63fc5e22022-08-04 22:04:10 +0100207export async function TestImage(url: string): Promise<string | null> {
Skyler Grey75ea9172022-08-06 10:22:23 +0100208 const text = await Tesseract.recognize(url, {
209 lang: "eng",
210 oem: 1,
211 psm: 3
212 });
pineafan813bdf42022-07-24 10:39:10 +0100213 return text;
214}
pineafan6de4da52023-03-07 20:43:44 +0000215
Skyler Grey0d885222023-03-08 21:46:37 +0000216export async function doMemberChecks(member: Discord.GuildMember): Promise<void> {
pineafan6de4da52023-03-07 20:43:44 +0000217 if (member.user.bot) return;
Skyler Grey0d885222023-03-08 21:46:37 +0000218 console.log("Checking member " + member.user.tag)
219 const guild = member.guild;
pineafan6de4da52023-03-07 20:43:44 +0000220 const guildData = await client.database.guilds.read(guild.id);
221 if (!guildData.logging.staff.channel) return;
pineafan1e462ab2023-03-07 21:34:06 +0000222 const [loose, strict] = [guildData.filters.wordFilter.words.loose, guildData.filters.wordFilter.words.strict];
Skyler Grey0d885222023-03-08 21:46:37 +0000223 console.log(1, loose, strict)
pineafan6de4da52023-03-07 20:43:44 +0000224 // Does the username contain filtered words
225 const usernameCheck = TestString(member.user.username, loose, strict, guildData.filters.wordFilter.enabled);
Skyler Grey0d885222023-03-08 21:46:37 +0000226 console.log(2, usernameCheck)
pineafan6de4da52023-03-07 20:43:44 +0000227 // Does the nickname contain filtered words
228 const nicknameCheck = TestString(member.nickname ?? "", loose, strict, guildData.filters.wordFilter.enabled);
Skyler Grey0d885222023-03-08 21:46:37 +0000229 console.log(3, nicknameCheck)
pineafan6de4da52023-03-07 20:43:44 +0000230 // Does the profile picture contain filtered words
pineafan1e462ab2023-03-07 21:34:06 +0000231 const avatarTextCheck = TestString(
232 (await TestImage(member.user.displayAvatarURL({ forceStatic: true }))) ?? "",
233 loose,
234 strict,
235 guildData.filters.wordFilter.enabled
236 );
Skyler Grey0d885222023-03-08 21:46:37 +0000237 console.log(4, avatarTextCheck)
pineafan6de4da52023-03-07 20:43:44 +0000238 // Is the profile picture NSFW
Skyler Grey0d885222023-03-08 21:46:37 +0000239 const avatar = member.displayAvatarURL({ extension: "png", size: 1024, forceStatic: true });
pineafan1e462ab2023-03-07 21:34:06 +0000240 const avatarCheck =
Skyler Grey0d885222023-03-08 21:46:37 +0000241 guildData.filters.images.NSFW && (await NSFWCheck({url: avatar, local: "", height: 1024, width: 1024}));
242 console.log(5, avatarCheck)
pineafan6de4da52023-03-07 20:43:44 +0000243 // Does the username contain an invite
Skyler Grey14432712023-03-07 23:40:50 +0000244 const inviteCheck = guildData.filters.invite.enabled && /discord\.gg\/[a-zA-Z0-9]+/gi.test(member.user.username);
Skyler Grey0d885222023-03-08 21:46:37 +0000245 console.log(6, inviteCheck)
pineafan6de4da52023-03-07 20:43:44 +0000246 // Does the nickname contain an invite
pineafan1e462ab2023-03-07 21:34:06 +0000247 const nicknameInviteCheck =
Skyler Grey14432712023-03-07 23:40:50 +0000248 guildData.filters.invite.enabled && /discord\.gg\/[a-zA-Z0-9]+/gi.test(member.nickname ?? "");
Skyler Grey0d885222023-03-08 21:46:37 +0000249 console.log(7, nicknameInviteCheck)
pineafan1e462ab2023-03-07 21:34:06 +0000250 if (
251 usernameCheck !== null ||
252 nicknameCheck !== null ||
253 avatarCheck ||
254 inviteCheck ||
255 nicknameInviteCheck ||
256 avatarTextCheck !== null
257 ) {
pineafan6de4da52023-03-07 20:43:44 +0000258 const infractions = [];
259 if (usernameCheck !== null) {
260 infractions.push(`Username contains a ${usernameCheck.type}ly filtered word (${usernameCheck.word})`);
pineafan1e462ab2023-03-07 21:34:06 +0000261 }
262 if (nicknameCheck !== null) {
pineafan6de4da52023-03-07 20:43:44 +0000263 infractions.push(`Nickname contains a ${nicknameCheck.type}ly filtered word (${nicknameCheck.word})`);
pineafan1e462ab2023-03-07 21:34:06 +0000264 }
265 if (avatarCheck) {
pineafan6de4da52023-03-07 20:43:44 +0000266 infractions.push("Profile picture is NSFW");
pineafan1e462ab2023-03-07 21:34:06 +0000267 }
268 if (inviteCheck) {
pineafan6de4da52023-03-07 20:43:44 +0000269 infractions.push("Username contains an invite");
pineafan1e462ab2023-03-07 21:34:06 +0000270 }
271 if (nicknameInviteCheck) {
pineafan6de4da52023-03-07 20:43:44 +0000272 infractions.push("Nickname contains an invite");
pineafan1e462ab2023-03-07 21:34:06 +0000273 }
274 if (avatarTextCheck !== null) {
275 infractions.push(
276 `Profile picture contains a ${avatarTextCheck.type}ly filtered word: ${avatarTextCheck.word}`
277 );
pineafan6de4da52023-03-07 20:43:44 +0000278 }
279 if (infractions.length === 0) return;
280 // This is bad - Warn in the staff notifications channel
281 const filter = getEmojiByName("ICONS.FILTER");
282 const channel = guild.channels.cache.get(guildData.logging.staff.channel) as Discord.TextChannel;
283 const embed = new EmojiEmbed()
284 .setTitle("Member Flagged")
285 .setEmoji("ICONS.FLAGS.RED")
286 .setStatus("Danger")
pineafan1e462ab2023-03-07 21:34:06 +0000287 .setDescription(
288 `**Member:** ${member.user.username} (<@${member.user.id}>)\n\n` +
289 infractions.map((element) => `${filter} ${element}`).join("\n")
290 );
pineafan6de4da52023-03-07 20:43:44 +0000291 await channel.send({
292 embeds: [embed],
pineafan1e462ab2023-03-07 21:34:06 +0000293 components: [
294 new ActionRowBuilder<ButtonBuilder>().addComponents(
295 ...[
296 new ButtonBuilder()
297 .setCustomId(`mod:warn:${member.user.id}`)
298 .setLabel("Warn")
299 .setStyle(ButtonStyle.Primary),
300 new ButtonBuilder()
301 .setCustomId(`mod:mute:${member.user.id}`)
302 .setLabel("Mute")
303 .setStyle(ButtonStyle.Primary),
304 new ButtonBuilder()
305 .setCustomId(`mod:kick:${member.user.id}`)
306 .setLabel("Kick")
307 .setStyle(ButtonStyle.Danger),
308 new ButtonBuilder()
309 .setCustomId(`mod:ban:${member.user.id}`)
310 .setLabel("Ban")
311 .setStyle(ButtonStyle.Danger)
312 ].concat(
313 usernameCheck !== null || nicknameCheck !== null || avatarTextCheck !== null
314 ? [
315 new ButtonBuilder()
316 .setCustomId(`mod:nickname:${member.user.id}`)
317 .setLabel("Change Name")
318 .setStyle(ButtonStyle.Primary)
319 ]
320 : []
321 )
322 )
323 ]
pineafan6de4da52023-03-07 20:43:44 +0000324 });
325 }
pineafan1e462ab2023-03-07 21:34:06 +0000326}