diff --git a/src/plugins/corruptMp4s.ts b/src/plugins/corruptMp4s.ts
new file mode 100644
index 00000000..aefff671
--- /dev/null
+++ b/src/plugins/corruptMp4s.ts
@@ -0,0 +1,104 @@
+/*
+ * Vencord, a modification for Discord's desktop app
+ * Copyright (c) 2022 Vendicated and contributors
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see .
+*/
+
+import { ApplicationCommandOptionType, sendBotMessage } from "../api/Commands";
+import { findOption } from "../api/Commands/commandHelpers";
+import { ApplicationCommandInputType } from "../api/Commands/types";
+import { Devs } from "../utils/constants";
+import definePlugin from "../utils/types";
+import { findByCode, findByProps } from "../webpack";
+
+const DRAFT_TYPE = 0;
+
+export default definePlugin({
+ name: "CorruptMp4s",
+ description: "Create corrupt mp4s with extremely high or negative duration",
+ authors: [Devs.Ven],
+ commands: [{
+ name: "corrupt",
+ description: "Create a corrupt mp4 with extremely high or negative duration",
+ inputType: ApplicationCommandInputType.BUILT_IN,
+ options: [
+ {
+ name: "mp4",
+ description: "the video to corrupt",
+ type: ApplicationCommandOptionType.ATTACHMENT,
+ required: true
+ },
+ {
+ name: "kind",
+ description: "the kind of corruption",
+ type: ApplicationCommandOptionType.STRING,
+ choices: [
+ {
+ name: "infinite",
+ value: "infinite",
+ label: "Very high duration"
+ },
+ {
+ name: "negative",
+ value: "negative",
+ label: "Negative duration"
+ }
+ ]
+ }
+ ],
+ execute: async (args, ctx) => {
+ const UploadStore = findByProps("getUploads");
+ const upload = UploadStore.getUploads(ctx.channel.id, DRAFT_TYPE)[0];
+
+ const video = upload?.item?.file as File | undefined;
+
+ if (video?.type !== "video/mp4")
+ return void sendBotMessage(ctx.channel.id, {
+ content: "Please upload a mp4 file"
+ });
+
+ const corruption = findOption(args, "kind", "negative");
+
+ const buf = new Uint8Array(await video.arrayBuffer());
+ let found = false;
+
+ // adapted from https://github.com/GeopJr/exorcism/blob/c9a12d77ccbcb49c987b385eafae250906efc297/src/App.svelte#L41-L48
+ for (let i = 0; i < buf.length; i++) {
+ if (buf[i] === 0x6d && buf[i + 1] === 0x76 && buf[i + 2] === 0x68 && buf[i + 3] === 0x64) {
+ let start = i + 18;
+ buf[start++] = 0x00;
+ buf[start++] = 0x01;
+ buf[start++] = corruption === "negative" ? 0xff : 0x7f;
+ buf[start++] = 0xff;
+ buf[start++] = 0xff;
+ buf[start++] = corruption === "negative" ? 0xf0 : 0xff;
+ found = true;
+ break;
+ }
+ }
+
+ if (!found) {
+ return void sendBotMessage(ctx.channel.id, {
+ content: "Could not find signature. Is this even a mp4?"
+ });
+ }
+
+ const newName = video.name.replace(/\.mp4$/i, ".corrupt.mp4");
+ const promptToUpload = findByCode("UPLOAD_FILE_LIMIT_ERROR");
+ const file = new File([buf], newName, { type: "video/mp4" });
+ setImmediate(() => promptToUpload([file], ctx.channel, DRAFT_TYPE));
+ }
+ }]
+});
diff --git a/src/plugins/petpet.ts b/src/plugins/petpet.ts
index 59b4e56b..7a794fd4 100644
--- a/src/plugins/petpet.ts
+++ b/src/plugins/petpet.ts
@@ -16,7 +16,7 @@
* along with this program. If not, see .
*/
-import { ApplicationCommandInputType, ApplicationCommandOptionType, Argument, CommandContext, findOption } from "../api/Commands";
+import { ApplicationCommandInputType, ApplicationCommandOptionType, Argument, CommandContext, findOption, sendBotMessage } from "../api/Commands";
import { Devs } from "../utils/constants";
import { getGifEncoder } from "../utils/dependencies";
import { lazyWebpack, makeLazy } from "../utils/misc";
@@ -132,8 +132,9 @@ export default definePlugin({
var url = await resolveImage(opts, cmdCtx, noServerPfp);
if (!url) throw "No Image specified!";
} catch (err) {
- // Todo make this send a clyde message once that PR is done
- console.log(err);
+ sendBotMessage(cmdCtx.channel.id, {
+ content: String(err),
+ });
return;
}