2024-03-11 00:25:44 +01:00
|
|
|
import { apiRoute, applyConfig } from "@api";
|
2023-11-29 00:54:39 +01:00
|
|
|
import { errorResponse, jsonResponse } from "@response";
|
|
|
|
|
import { encode } from "blurhash";
|
2024-03-11 00:57:26 +01:00
|
|
|
import { MediaBackendType } from "media-manager";
|
|
|
|
|
import type { MediaBackend } from "media-manager";
|
2024-04-07 07:30:49 +02:00
|
|
|
import sharp from "sharp";
|
|
|
|
|
import { attachmentToAPI, getUrl } from "~database/entities/Attachment";
|
2024-04-13 14:20:12 +02:00
|
|
|
import { db } from "~drizzle/db";
|
|
|
|
|
import { attachment } from "~drizzle/schema";
|
2024-04-14 10:16:03 +02:00
|
|
|
import { LocalMediaBackend, S3MediaBackend } from "media-manager";
|
2023-11-29 00:54:39 +01:00
|
|
|
|
2024-03-11 00:57:26 +01:00
|
|
|
export const meta = applyConfig({
|
2024-04-07 07:30:49 +02:00
|
|
|
allowedMethods: ["POST"],
|
|
|
|
|
ratelimits: {
|
|
|
|
|
max: 10,
|
|
|
|
|
duration: 60,
|
|
|
|
|
},
|
|
|
|
|
route: "/api/v1/media",
|
|
|
|
|
auth: {
|
|
|
|
|
required: true,
|
|
|
|
|
oauthPermissions: ["write:media"],
|
|
|
|
|
},
|
2023-11-29 00:54:39 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Upload new media
|
|
|
|
|
*/
|
2024-03-11 00:25:44 +01:00
|
|
|
export default apiRoute<{
|
2024-04-07 07:30:49 +02:00
|
|
|
file: File;
|
|
|
|
|
thumbnail?: File;
|
|
|
|
|
description?: string;
|
|
|
|
|
// TODO: Add focus
|
|
|
|
|
focus?: string;
|
2024-03-11 00:25:44 +01:00
|
|
|
}>(async (req, matchedRoute, extraData) => {
|
2024-04-07 07:30:49 +02:00
|
|
|
const { user } = extraData.auth;
|
|
|
|
|
|
|
|
|
|
if (!user) {
|
|
|
|
|
return errorResponse("Unauthorized", 401);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const { file, thumbnail, description } = extraData.parsedRequest;
|
|
|
|
|
|
|
|
|
|
if (!file) {
|
|
|
|
|
return errorResponse("No file provided", 400);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const config = await extraData.configManager.getConfig();
|
|
|
|
|
|
|
|
|
|
if (file.size > config.validation.max_media_size) {
|
|
|
|
|
return errorResponse(
|
|
|
|
|
`File too large, max size is ${config.validation.max_media_size} bytes`,
|
|
|
|
|
413,
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (
|
|
|
|
|
config.validation.enforce_mime_types &&
|
|
|
|
|
!config.validation.allowed_mime_types.includes(file.type)
|
|
|
|
|
) {
|
|
|
|
|
return errorResponse("Invalid file type", 415);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (
|
|
|
|
|
description &&
|
|
|
|
|
description.length > config.validation.max_media_description_size
|
|
|
|
|
) {
|
|
|
|
|
return errorResponse(
|
|
|
|
|
`Description too long, max length is ${config.validation.max_media_description_size} characters`,
|
|
|
|
|
413,
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const sha256 = new Bun.SHA256();
|
|
|
|
|
|
|
|
|
|
const isImage = file.type.startsWith("image/");
|
|
|
|
|
|
|
|
|
|
const metadata = isImage
|
|
|
|
|
? await sharp(await file.arrayBuffer()).metadata()
|
|
|
|
|
: null;
|
|
|
|
|
|
2024-04-09 15:18:04 +02:00
|
|
|
const blurhash = await new Promise<string | null>((resolve) => {
|
|
|
|
|
(async () =>
|
|
|
|
|
sharp(await file.arrayBuffer())
|
|
|
|
|
.raw()
|
|
|
|
|
.ensureAlpha()
|
|
|
|
|
.toBuffer((err, buffer) => {
|
|
|
|
|
if (err) {
|
|
|
|
|
resolve(null);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-09 15:51:22 +02:00
|
|
|
try {
|
|
|
|
|
resolve(
|
|
|
|
|
encode(
|
|
|
|
|
new Uint8ClampedArray(buffer),
|
|
|
|
|
metadata?.width ?? 0,
|
|
|
|
|
metadata?.height ?? 0,
|
|
|
|
|
4,
|
|
|
|
|
4,
|
|
|
|
|
) as string,
|
|
|
|
|
);
|
|
|
|
|
} catch {
|
|
|
|
|
resolve(null);
|
|
|
|
|
}
|
2024-04-09 15:18:04 +02:00
|
|
|
}))();
|
|
|
|
|
});
|
2024-04-07 07:30:49 +02:00
|
|
|
|
|
|
|
|
let url = "";
|
|
|
|
|
|
|
|
|
|
let mediaManager: MediaBackend;
|
|
|
|
|
|
|
|
|
|
switch (config.media.backend as MediaBackendType) {
|
|
|
|
|
case MediaBackendType.LOCAL:
|
|
|
|
|
mediaManager = new LocalMediaBackend(config);
|
|
|
|
|
break;
|
|
|
|
|
case MediaBackendType.S3:
|
|
|
|
|
mediaManager = new S3MediaBackend(config);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
// TODO: Replace with logger
|
|
|
|
|
throw new Error("Invalid media backend");
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-09 15:51:22 +02:00
|
|
|
const { path } = await mediaManager.addFile(file);
|
2024-04-07 07:30:49 +02:00
|
|
|
|
2024-04-09 15:51:22 +02:00
|
|
|
url = getUrl(path, config);
|
2024-04-07 07:30:49 +02:00
|
|
|
|
|
|
|
|
let thumbnailUrl = "";
|
|
|
|
|
|
|
|
|
|
if (thumbnail) {
|
2024-04-09 15:51:22 +02:00
|
|
|
const { path } = await mediaManager.addFile(thumbnail);
|
2024-04-07 07:30:49 +02:00
|
|
|
|
2024-04-09 15:51:22 +02:00
|
|
|
thumbnailUrl = getUrl(path, config);
|
2024-04-07 07:30:49 +02:00
|
|
|
}
|
|
|
|
|
|
2024-04-13 14:20:12 +02:00
|
|
|
const newAttachment = (
|
|
|
|
|
await db
|
|
|
|
|
.insert(attachment)
|
|
|
|
|
.values({
|
|
|
|
|
url,
|
|
|
|
|
thumbnailUrl,
|
|
|
|
|
sha256: sha256.update(await file.arrayBuffer()).digest("hex"),
|
|
|
|
|
mimeType: file.type,
|
|
|
|
|
description: description ?? "",
|
|
|
|
|
size: file.size,
|
|
|
|
|
blurhash: blurhash ?? undefined,
|
|
|
|
|
width: metadata?.width ?? undefined,
|
|
|
|
|
height: metadata?.height ?? undefined,
|
|
|
|
|
})
|
|
|
|
|
.returning()
|
|
|
|
|
)[0];
|
2024-04-07 07:30:49 +02:00
|
|
|
// TODO: Add job to process videos and other media
|
|
|
|
|
|
|
|
|
|
return jsonResponse(attachmentToAPI(newAttachment));
|
2024-03-11 00:25:44 +01:00
|
|
|
});
|