temp disable submissions

This commit is contained in:
trafficlunar 2026-04-01 02:14:33 +01:00
parent 655af51766
commit 8608acae31
2 changed files with 235 additions and 240 deletions

View file

@ -75,266 +75,267 @@ export async function POST(request: NextRequest) {
const check = await rateLimit.handle(); const check = await rateLimit.handle();
if (check) return check; if (check) return check;
const response = await fetch(`${process.env.NEXT_PUBLIC_BASE_URL}/api/admin/can-submit`); return rateLimit.sendResponse({ error: "Submissions are temporarily disabled" }, 503);
const { value } = await response.json(); // const response = await fetch(`${process.env.NEXT_PUBLIC_BASE_URL}/api/admin/can-submit`);
if (!value) return rateLimit.sendResponse({ error: "Submissions are temporarily disabled" }, 503); // const { value } = await response.json();
// if (!value) return rateLimit.sendResponse({ error: "Submissions are temporarily disabled" }, 503);
// Parse tags and QR code as JSON // // Parse tags and QR code as JSON
const formData = await request.formData(); // const formData = await request.formData();
let rawTags: string[]; // let rawTags: string[];
let rawQrBytesRaw: string[]; // raw raw // let rawQrBytesRaw: string[]; // raw raw
try { // try {
rawTags = JSON.parse(formData.get("tags") as string); // rawTags = JSON.parse(formData.get("tags") as string);
rawQrBytesRaw = JSON.parse(formData.get("qrBytesRaw") as string); // rawQrBytesRaw = JSON.parse(formData.get("qrBytesRaw") as string);
} catch (error) { // } catch (error) {
Sentry.captureException(error, { // Sentry.captureException(error, {
extra: { stage: "submit-json-parse" }, // extra: { stage: "submit-json-parse" },
}); // });
return rateLimit.sendResponse({ error: "Invalid JSON in tags or QR code data" }, 400); // return rateLimit.sendResponse({ error: "Invalid JSON in tags or QR code data" }, 400);
} // }
// Minify instructions to save space and improve user experience // // Minify instructions to save space and improve user experience
let minifiedInstructions: Partial<SwitchMiiInstructions> | undefined; // let minifiedInstructions: Partial<SwitchMiiInstructions> | undefined;
if (formData.get("platform") === "SWITCH") // if (formData.get("platform") === "SWITCH")
minifiedInstructions = minifyInstructions(JSON.parse((formData.get("instructions") as string) ?? "{}") as SwitchMiiInstructions); // minifiedInstructions = minifyInstructions(JSON.parse((formData.get("instructions") as string) ?? "{}") as SwitchMiiInstructions);
// Parse and check all submission info // // Parse and check all submission info
const parsed = submitSchema.safeParse({ // const parsed = submitSchema.safeParse({
platform: formData.get("platform"), // platform: formData.get("platform"),
name: formData.get("name"), // name: formData.get("name"),
tags: rawTags, // tags: rawTags,
description: formData.get("description"), // description: formData.get("description"),
gender: formData.get("gender") ?? undefined, // ZOD MOMENT // gender: formData.get("gender") ?? undefined, // ZOD MOMENT
makeup: formData.get("makeup") ?? undefined, // makeup: formData.get("makeup") ?? undefined,
miiPortraitImage: formData.get("miiPortraitImage"), // miiPortraitImage: formData.get("miiPortraitImage"),
miiFeaturesImage: formData.get("miiFeaturesImage"), // miiFeaturesImage: formData.get("miiFeaturesImage"),
instructions: minifiedInstructions, // instructions: minifiedInstructions,
qrBytesRaw: rawQrBytesRaw, // qrBytesRaw: rawQrBytesRaw,
image1: formData.get("image1"), // image1: formData.get("image1"),
image2: formData.get("image2"), // image2: formData.get("image2"),
image3: formData.get("image3"), // image3: formData.get("image3"),
}); // });
if (!parsed.success) { // if (!parsed.success) {
const firstIssue = parsed.error.issues[0]; // const firstIssue = parsed.error.issues[0];
const path = firstIssue.path.length ? firstIssue.path.join(".") : "root"; // const path = firstIssue.path.length ? firstIssue.path.join(".") : "root";
const error = `${path}: ${firstIssue.message}`; // const error = `${path}: ${firstIssue.message}`;
const issues = parsed.error.issues; // const issues = parsed.error.issues;
const hasInstructionsErrors = issues.some((issue) => issue.path[0] === "instructions"); // const hasInstructionsErrors = issues.some((issue) => issue.path[0] === "instructions");
if (hasInstructionsErrors) { // if (hasInstructionsErrors) {
Sentry.captureException(error, { // Sentry.captureException(error, {
extra: { issues, rawInstructions: formData.get("instructions"), stage: "submit-instructions" }, // extra: { issues, rawInstructions: formData.get("instructions"), stage: "submit-instructions" },
}); // });
} // }
return rateLimit.sendResponse({ error }, 400); // return rateLimit.sendResponse({ error }, 400);
} // }
const { // const {
platform, // platform,
name: uncensoredName, // name: uncensoredName,
tags: uncensoredTags, // tags: uncensoredTags,
description: uncensoredDescription, // description: uncensoredDescription,
qrBytesRaw, // qrBytesRaw,
gender, // gender,
makeup, // makeup,
miiPortraitImage, // miiPortraitImage,
miiFeaturesImage, // miiFeaturesImage,
image1, // image1,
image2, // image2,
image3, // image3,
} = parsed.data; // } = parsed.data;
// Censor potential inappropriate words // // Censor potential inappropriate words
const name = profanity.censor(uncensoredName); // const name = profanity.censor(uncensoredName);
const tags = uncensoredTags.map((t) => profanity.censor(t)); // const tags = uncensoredTags.map((t) => profanity.censor(t));
const description = uncensoredDescription && profanity.censor(uncensoredDescription); // const description = uncensoredDescription && profanity.censor(uncensoredDescription);
// Validate image files // // Validate image files
const customImages: File[] = []; // const customImages: File[] = [];
for (const img of [image1, image2, image3]) { // for (const img of [image1, image2, image3]) {
if (!img) continue; // if (!img) continue;
const imageValidation = await validateImage(img); // const imageValidation = await validateImage(img);
if (imageValidation.valid) { // if (imageValidation.valid) {
customImages.push(img); // customImages.push(img);
} else { // } else {
return rateLimit.sendResponse({ error: `Failed to verify custom image: ${imageValidation.error}` }, imageValidation.status ?? 400); // return rateLimit.sendResponse({ error: `Failed to verify custom image: ${imageValidation.error}` }, imageValidation.status ?? 400);
} // }
} // }
// Check Mii portrait & features image (Switch) // // Check Mii portrait & features image (Switch)
if (platform === "SWITCH") { // if (platform === "SWITCH") {
const portraitValidation = await validateImage(miiPortraitImage); // const portraitValidation = await validateImage(miiPortraitImage);
const featuresValidation = await validateImage(miiFeaturesImage); // const featuresValidation = await validateImage(miiFeaturesImage);
if (!portraitValidation.valid) // if (!portraitValidation.valid)
return rateLimit.sendResponse({ error: `Failed to verify portrait: ${portraitValidation.error}` }, portraitValidation.status ?? 400); // return rateLimit.sendResponse({ error: `Failed to verify portrait: ${portraitValidation.error}` }, portraitValidation.status ?? 400);
if (!featuresValidation.valid) // if (!featuresValidation.valid)
return rateLimit.sendResponse({ error: `Failed to verify features: ${featuresValidation.error}` }, featuresValidation.status ?? 400); // return rateLimit.sendResponse({ error: `Failed to verify features: ${featuresValidation.error}` }, featuresValidation.status ?? 400);
} // }
const qrBytes = new Uint8Array(qrBytesRaw ?? []); // const qrBytes = new Uint8Array(qrBytesRaw ?? []);
// Convert QR code to JS (3DS) // // Convert QR code to JS (3DS)
let conversion: { mii: Mii; tomodachiLifeMii: ThreeDsTomodachiLifeMii } | undefined; // let conversion: { mii: Mii; tomodachiLifeMii: ThreeDsTomodachiLifeMii } | undefined;
if (platform === "THREE_DS") { // if (platform === "THREE_DS") {
try { // try {
conversion = convertQrCode(qrBytes); // conversion = convertQrCode(qrBytes);
} catch (error) { // } catch (error) {
Sentry.captureException(error, { extra: { stage: "qr-conversion" } }); // Sentry.captureException(error, { extra: { stage: "qr-conversion" } });
return rateLimit.sendResponse({ error: error instanceof Error ? error.message : String(error) }, 400); // return rateLimit.sendResponse({ error: error instanceof Error ? error.message : String(error) }, 400);
} // }
} // }
// Create Mii in database // // Create Mii in database
const miiRecord = await prisma.mii.create({ // const miiRecord = await prisma.mii.create({
data: { // data: {
userId: Number(session.user?.id), // userId: Number(session.user?.id),
platform, // platform,
name, // name,
tags, // tags,
description, // description,
gender: gender ?? "MALE", // gender: gender ?? "MALE",
// Automatically detect certain information if on 3DS // // Automatically detect certain information if on 3DS
...(platform === "THREE_DS" // ...(platform === "THREE_DS"
? conversion && { // ? conversion && {
firstName: conversion.tomodachiLifeMii.firstName, // firstName: conversion.tomodachiLifeMii.firstName,
lastName: conversion.tomodachiLifeMii.lastName, // lastName: conversion.tomodachiLifeMii.lastName,
gender: conversion.mii.gender == 0 ? MiiGender.MALE : MiiGender.FEMALE, // gender: conversion.mii.gender == 0 ? MiiGender.MALE : MiiGender.FEMALE,
islandName: conversion.tomodachiLifeMii.islandName, // islandName: conversion.tomodachiLifeMii.islandName,
allowedCopying: conversion.mii.allowCopying, // allowedCopying: conversion.mii.allowCopying,
} // }
: { // : {
instructions: minifiedInstructions, // instructions: minifiedInstructions,
makeup: makeup ?? "PARTIAL", // makeup: makeup ?? "PARTIAL",
}), // }),
}, // },
}); // });
// Ensure directories exist // // Ensure directories exist
const miiUploadsDirectory = path.join(uploadsDirectory, miiRecord.id.toString()); // const miiUploadsDirectory = path.join(uploadsDirectory, miiRecord.id.toString());
await fs.mkdir(miiUploadsDirectory, { recursive: true }); // await fs.mkdir(miiUploadsDirectory, { recursive: true });
try { // try {
let portraitBuffer: Buffer | undefined; // let portraitBuffer: Buffer | undefined;
// Download the image of the Mii (3DS) // // Download the image of the Mii (3DS)
if (platform === "THREE_DS") { // if (platform === "THREE_DS") {
const studioUrl = conversion?.mii.studioUrl({ width: 512 }); // const studioUrl = conversion?.mii.studioUrl({ width: 512 });
const studioResponse = await fetch(studioUrl!); // const studioResponse = await fetch(studioUrl!);
if (!studioResponse.ok) { // if (!studioResponse.ok) {
throw new Error(`Failed to fetch Mii image ${studioResponse.status}`); // throw new Error(`Failed to fetch Mii image ${studioResponse.status}`);
} // }
portraitBuffer = Buffer.from(await studioResponse.arrayBuffer()); // portraitBuffer = Buffer.from(await studioResponse.arrayBuffer());
} else if (platform === "SWITCH") { // } else if (platform === "SWITCH") {
portraitBuffer = Buffer.from(await miiPortraitImage.arrayBuffer()); // portraitBuffer = Buffer.from(await miiPortraitImage.arrayBuffer());
// Save features image // // Save features image
const featuresBuffer = Buffer.from(await miiFeaturesImage.arrayBuffer()); // const featuresBuffer = Buffer.from(await miiFeaturesImage.arrayBuffer());
const pngBuffer = await sharp(featuresBuffer) // const pngBuffer = await sharp(featuresBuffer)
.resize({ // .resize({
height: 800, // height: 800,
fit: "inside", // fit: "inside",
withoutEnlargement: true, // withoutEnlargement: true,
}) // })
.png({ quality: 85 }) // .png({ quality: 85 })
.toBuffer(); // .toBuffer();
const fileLocation = path.join(miiUploadsDirectory, "features.png"); // const fileLocation = path.join(miiUploadsDirectory, "features.png");
await fs.writeFile(fileLocation, pngBuffer); // await fs.writeFile(fileLocation, pngBuffer);
} // }
// Save portrait image // // Save portrait image
if (!portraitBuffer) throw Error("Mii portrait buffer not initialised"); // if (!portraitBuffer) throw Error("Mii portrait buffer not initialised");
const pngBuffer = await sharp(portraitBuffer) // const pngBuffer = await sharp(portraitBuffer)
.resize({ // .resize({
height: 500, // height: 500,
fit: "inside", // fit: "inside",
withoutEnlargement: true, // withoutEnlargement: true,
}) // })
.png({ quality: 85 }) // .png({ quality: 85 })
.toBuffer(); // .toBuffer();
const fileLocation = path.join(miiUploadsDirectory, "mii.png"); // const fileLocation = path.join(miiUploadsDirectory, "mii.png");
await fs.writeFile(fileLocation, pngBuffer); // await fs.writeFile(fileLocation, pngBuffer);
} catch (error) { // } catch (error) {
// Clean up if something went wrong // // Clean up if something went wrong
await prisma.mii.delete({ where: { id: miiRecord.id } }); // await prisma.mii.delete({ where: { id: miiRecord.id } });
console.error("Failed to download/store Mii portrait/features:", error); // console.error("Failed to download/store Mii portrait/features:", error);
Sentry.captureException(error, { extra: { miiId: miiRecord.id, stage: "studio-image-download" } }); // Sentry.captureException(error, { extra: { miiId: miiRecord.id, stage: "studio-image-download" } });
return rateLimit.sendResponse({ error: "Failed to download/store Mii portrait/features" }, 500); // return rateLimit.sendResponse({ error: "Failed to download/store Mii portrait/features" }, 500);
} // }
try { // try {
await generateMetadataImage(miiRecord, session.user?.name!); // await generateMetadataImage(miiRecord, session.user?.name!);
} catch (error) { // } catch (error) {
console.error("Failed to generate metadata image:", error); // console.error("Failed to generate metadata image:", error);
Sentry.captureException(error, { extra: { miiId: miiRecord.id, stage: "metadata-image-generation" } }); // Sentry.captureException(error, { extra: { miiId: miiRecord.id, stage: "metadata-image-generation" } });
} // }
if (platform === "THREE_DS") { // if (platform === "THREE_DS") {
try { // try {
// Generate a new QR code for aesthetic reasons // // Generate a new QR code for aesthetic reasons
const byteString = String.fromCharCode(...qrBytes); // const byteString = String.fromCharCode(...qrBytes);
const generatedCode = qrcode(0, "L"); // const generatedCode = qrcode(0, "L");
generatedCode.addData(byteString, "Byte"); // generatedCode.addData(byteString, "Byte");
generatedCode.make(); // generatedCode.make();
// Store QR code // // Store QR code
const codeDataUrl = generatedCode.createDataURL(); // const codeDataUrl = generatedCode.createDataURL();
const codeBase64 = codeDataUrl.replace(/^data:image\/gif;base64,/, ""); // const codeBase64 = codeDataUrl.replace(/^data:image\/gif;base64,/, "");
const codeBuffer = Buffer.from(codeBase64, "base64"); // const codeBuffer = Buffer.from(codeBase64, "base64");
// Compress and store // // Compress and store
const codePngBuffer = await sharp(codeBuffer).png({ quality: 85 }).toBuffer(); // const codePngBuffer = await sharp(codeBuffer).png({ quality: 85 }).toBuffer();
const codeFileLocation = path.join(miiUploadsDirectory, "qr-code.png"); // const codeFileLocation = path.join(miiUploadsDirectory, "qr-code.png");
await fs.writeFile(codeFileLocation, codePngBuffer); // await fs.writeFile(codeFileLocation, codePngBuffer);
} catch (error) { // } catch (error) {
// Clean up if something went wrong // // Clean up if something went wrong
await prisma.mii.delete({ where: { id: miiRecord.id } }); // await prisma.mii.delete({ where: { id: miiRecord.id } });
console.error("Error processing Mii files:", error); // console.error("Error processing Mii files:", error);
Sentry.captureException(error, { extra: { miiId: miiRecord.id, stage: "file-processing" } }); // Sentry.captureException(error, { extra: { miiId: miiRecord.id, stage: "file-processing" } });
return rateLimit.sendResponse({ error: "Failed to process and store Mii files" }, 500); // return rateLimit.sendResponse({ error: "Failed to process and store Mii files" }, 500);
} // }
} // }
// Compress and store user images // // Compress and store user images
try { // try {
await Promise.all( // await Promise.all(
customImages.map(async (image, index) => { // customImages.map(async (image, index) => {
const buffer = Buffer.from(await image.arrayBuffer()); // const buffer = Buffer.from(await image.arrayBuffer());
const pngBuffer = await sharp(buffer).resize({ height: 800, fit: "inside", withoutEnlargement: true }).png({ quality: 85 }).toBuffer(); // const pngBuffer = await sharp(buffer).resize({ height: 800, fit: "inside", withoutEnlargement: true }).png({ quality: 85 }).toBuffer();
const fileLocation = path.join(miiUploadsDirectory, `image${index}.png`); // const fileLocation = path.join(miiUploadsDirectory, `image${index}.png`);
await fs.writeFile(fileLocation, pngBuffer); // await fs.writeFile(fileLocation, pngBuffer);
}), // }),
); // );
// Update database to tell it how many images exist // // Update database to tell it how many images exist
await prisma.mii.update({ // await prisma.mii.update({
where: { // where: {
id: miiRecord.id, // id: miiRecord.id,
}, // },
data: { // data: {
imageCount: customImages.length, // imageCount: customImages.length,
}, // },
}); // });
} catch (error) { // } catch (error) {
console.error("Error storing user images:", error); // console.error("Error storing user images:", error);
Sentry.captureException(error, { extra: { miiId: miiRecord.id, stage: "user-image-storage" } }); // Sentry.captureException(error, { extra: { miiId: miiRecord.id, stage: "user-image-storage" } });
return rateLimit.sendResponse({ error: "Failed to store user images" }, 500); // return rateLimit.sendResponse({ error: "Failed to store user images" }, 500);
} // }
return rateLimit.sendResponse({ success: true, id: miiRecord.id }); // return rateLimit.sendResponse({ success: true, id: miiRecord.id });
} }

View file

@ -31,21 +31,15 @@ export default async function SubmitPage() {
if (activePunishment) redirect("/off-the-island"); if (activePunishment) redirect("/off-the-island");
// Check if submissions are disabled // Check if submissions are disabled
let value: boolean | null = true; // let value: boolean | null = true;
try { // try {
const response = await fetch(`${process.env.NEXT_PUBLIC_BASE_URL}/api/admin/can-submit`); // const response = await fetch(`${process.env.NEXT_PUBLIC_BASE_URL}/api/admin/can-submit`);
value = await response.json(); // value = await response.json();
} catch (error) { // } catch (error) {
return <p>An error occurred!</p>; // return <p>An error occurred!</p>;
} // }
try {
const response = await fetch(`${process.env.NEXT_PUBLIC_BASE_URL}/api/admin/can-submit`);
value = await response.json();
} catch (error) {
return <p>An error occurred!</p>;
}
if (!value) // if (!value)
return ( return (
<div className="grow flex items-center justify-center"> <div className="grow flex items-center justify-center">
<div className="bg-amber-50 border-2 border-amber-500 rounded-2xl shadow-lg p-8 max-w-xs w-full text-center flex flex-col"> <div className="bg-amber-50 border-2 border-amber-500 rounded-2xl shadow-lg p-8 max-w-xs w-full text-center flex flex-col">
@ -59,5 +53,5 @@ export default async function SubmitPage() {
</div> </div>
); );
return <SubmitForm />; // return <SubmitForm />;
} }