jormungandr-bite/packages/backend/src/misc/emoji-meta.ts

51 lines
1.3 KiB
TypeScript
Raw Normal View History

2023-05-19 20:26:13 -06:00
import probeImageSize from "probe-image-size";
import { Mutex, withTimeout } from "async-mutex";
2023-05-19 20:26:13 -06:00
import { FILE_TYPE_BROWSERSAFE } from "@/const.js";
import Logger from "@/services/logger.js";
import { Cache } from "./cache.js";
2023-05-19 20:26:13 -06:00
export type Size = {
width: number;
height: number;
};
const cache = new Cache<boolean>(1000 * 60 * 10); // once every 10 minutes for the same url
const mutex = withTimeout(new Mutex(), 1000);
2023-05-19 20:26:13 -06:00
export async function getEmojiSize(url: string): Promise<Size> {
const logger = new Logger("emoji");
2023-05-19 20:26:13 -06:00
await mutex.runExclusive(() => {
const attempted = cache.get(url);
2023-05-19 20:26:13 -06:00
if (!attempted) {
cache.set(url, true);
} else {
logger.warn(`Attempt limit exceeded: ${url}`);
throw new Error("Too many attempts");
2023-05-19 20:26:13 -06:00
}
});
2023-05-19 20:26:13 -06:00
try {
logger.info(`Retrieving emoji size from ${url}`);
2023-05-19 20:26:13 -06:00
const { width, height, mime } = await probeImageSize(url, {
timeout: 5000,
});
if (!(mime.startsWith("image/") && FILE_TYPE_BROWSERSAFE.includes(mime))) {
throw new Error("Unsupported image type");
2023-05-19 20:26:13 -06:00
}
return { width, height };
} catch (e) {
throw new Error(`Unable to retrieve metadata: ${e}`);
2023-05-19 20:26:13 -06:00
}
}
export function getNormalSize(
{ width, height }: Size,
orientation?: number,
): Size {
return (orientation || 0) >= 5
? { width: height, height: width }
: { width, height };
}