2023-11-26 21:33:46 +01:00
|
|
|
import * as fs from "node:fs";
|
2023-12-05 08:12:10 +01:00
|
|
|
import * as crypto from "node:crypto";
|
2023-01-13 05:40:33 +01:00
|
|
|
import * as stream from "node:stream";
|
|
|
|
import * as util from "node:util";
|
|
|
|
import { fileTypeFromFile } from "file-type";
|
2023-12-05 08:12:10 +01:00
|
|
|
import probeImageSize from "probe-image-size";
|
2023-01-13 05:40:33 +01:00
|
|
|
import isSvg from "is-svg";
|
|
|
|
import sharp from "sharp";
|
2023-12-05 08:12:10 +01:00
|
|
|
import { encode } from "blurhash";
|
2024-02-21 14:23:24 +01:00
|
|
|
import { inspect } from "node:util";
|
2020-01-12 08:40:58 +01:00
|
|
|
|
2020-04-11 11:28:40 +02:00
|
|
|
const pipeline = util.promisify(stream.pipeline);
|
|
|
|
|
2020-01-12 08:40:58 +01:00
|
|
|
export type FileInfo = {
|
|
|
|
size: number;
|
|
|
|
md5: string;
|
|
|
|
type: {
|
|
|
|
mime: string;
|
|
|
|
ext: string | null;
|
|
|
|
};
|
|
|
|
width?: number;
|
|
|
|
height?: number;
|
2021-12-03 03:19:28 +01:00
|
|
|
orientation?: number;
|
2020-07-18 17:24:07 +02:00
|
|
|
blurhash?: string;
|
2020-01-12 08:40:58 +01:00
|
|
|
warnings: string[];
|
|
|
|
};
|
|
|
|
|
|
|
|
const TYPE_OCTET_STREAM = {
|
2023-01-13 05:40:33 +01:00
|
|
|
mime: "application/octet-stream",
|
2021-12-09 15:58:30 +01:00
|
|
|
ext: null,
|
2020-01-12 08:40:58 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
const TYPE_SVG = {
|
2023-01-13 05:40:33 +01:00
|
|
|
mime: "image/svg+xml",
|
|
|
|
ext: "svg",
|
2020-01-12 08:40:58 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get file information
|
|
|
|
*/
|
2024-02-15 17:08:05 +01:00
|
|
|
export async function getFileInfo(path: string): Promise<FileInfo> {
|
2020-01-12 08:40:58 +01:00
|
|
|
const warnings = [] as string[];
|
|
|
|
|
|
|
|
const size = await getFileSize(path);
|
|
|
|
const md5 = await calcHash(path);
|
|
|
|
|
|
|
|
let type = await detectType(path);
|
|
|
|
|
|
|
|
// image dimensions
|
|
|
|
let width: number | undefined;
|
|
|
|
let height: number | undefined;
|
2021-12-03 03:19:28 +01:00
|
|
|
let orientation: number | undefined;
|
2020-01-12 08:40:58 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
if (
|
|
|
|
[
|
|
|
|
"image/jpeg",
|
|
|
|
"image/gif",
|
|
|
|
"image/png",
|
|
|
|
"image/apng",
|
|
|
|
"image/webp",
|
|
|
|
"image/bmp",
|
|
|
|
"image/tiff",
|
|
|
|
"image/svg+xml",
|
|
|
|
"image/vnd.adobe.photoshop",
|
|
|
|
"image/avif",
|
|
|
|
].includes(type.mime)
|
|
|
|
) {
|
|
|
|
const imageSize = await detectImageSize(path).catch((e) => {
|
2024-02-21 14:23:24 +01:00
|
|
|
warnings.push(`detectImageSize failed:\n${inspect(e)}`);
|
2020-01-12 08:40:58 +01:00
|
|
|
return undefined;
|
|
|
|
});
|
|
|
|
|
|
|
|
// うまく判定できない画像は octet-stream にする
|
|
|
|
if (!imageSize) {
|
2023-01-13 05:40:33 +01:00
|
|
|
warnings.push("cannot detect image dimensions");
|
2020-01-12 08:40:58 +01:00
|
|
|
type = TYPE_OCTET_STREAM;
|
2023-01-13 05:40:33 +01:00
|
|
|
} else if (imageSize.wUnits === "px") {
|
2020-01-12 08:40:58 +01:00
|
|
|
width = imageSize.width;
|
|
|
|
height = imageSize.height;
|
2021-12-03 03:19:28 +01:00
|
|
|
orientation = imageSize.orientation;
|
2020-01-12 08:40:58 +01:00
|
|
|
|
|
|
|
// 制限を超えている画像は octet-stream にする
|
|
|
|
if (imageSize.width > 16383 || imageSize.height > 16383) {
|
2023-01-13 05:40:33 +01:00
|
|
|
warnings.push("image dimensions exceeds limits");
|
2020-01-12 08:40:58 +01:00
|
|
|
type = TYPE_OCTET_STREAM;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
warnings.push(`unsupported unit type: ${imageSize.wUnits}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-18 17:24:07 +02:00
|
|
|
let blurhash: string | undefined;
|
2020-01-12 08:40:58 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
if (
|
|
|
|
[
|
|
|
|
"image/jpeg",
|
|
|
|
"image/gif",
|
|
|
|
"image/png",
|
|
|
|
"image/apng",
|
|
|
|
"image/webp",
|
|
|
|
"image/svg+xml",
|
|
|
|
"image/avif",
|
|
|
|
].includes(type.mime)
|
|
|
|
) {
|
|
|
|
blurhash = await getBlurhash(path).catch((e) => {
|
2024-02-21 14:23:24 +01:00
|
|
|
warnings.push(`getBlurhash failed:\n${inspect(e)}`);
|
2020-01-12 08:40:58 +01:00
|
|
|
return undefined;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
size,
|
|
|
|
md5,
|
|
|
|
type,
|
|
|
|
width,
|
|
|
|
height,
|
2021-12-03 03:19:28 +01:00
|
|
|
orientation,
|
2020-07-18 17:24:07 +02:00
|
|
|
blurhash,
|
2020-01-12 08:40:58 +01:00
|
|
|
warnings,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Detect MIME Type and extension
|
|
|
|
*/
|
2021-12-25 17:42:06 +01:00
|
|
|
export async function detectType(path: string): Promise<{
|
|
|
|
mime: string;
|
|
|
|
ext: string | null;
|
|
|
|
}> {
|
2020-01-12 08:40:58 +01:00
|
|
|
// Check 0 byte
|
|
|
|
const fileSize = await getFileSize(path);
|
|
|
|
if (fileSize === 0) {
|
|
|
|
return TYPE_OCTET_STREAM;
|
|
|
|
}
|
|
|
|
|
2022-03-07 15:51:34 +01:00
|
|
|
const type = await fileTypeFromFile(path);
|
2020-01-12 08:40:58 +01:00
|
|
|
|
|
|
|
if (type) {
|
|
|
|
// XMLはSVGかもしれない
|
2023-01-13 05:40:33 +01:00
|
|
|
if (type.mime === "application/xml" && (await checkSvg(path))) {
|
2020-01-12 08:40:58 +01:00
|
|
|
return TYPE_SVG;
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
mime: type.mime,
|
2021-12-09 15:58:30 +01:00
|
|
|
ext: type.ext,
|
2020-01-12 08:40:58 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
// 種類が不明でもSVGかもしれない
|
|
|
|
if (await checkSvg(path)) {
|
|
|
|
return TYPE_SVG;
|
|
|
|
}
|
|
|
|
|
|
|
|
// それでも種類が不明なら application/octet-stream にする
|
|
|
|
return TYPE_OCTET_STREAM;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check the file is SVG or not
|
|
|
|
*/
|
|
|
|
export async function checkSvg(path: string) {
|
|
|
|
try {
|
|
|
|
const size = await getFileSize(path);
|
|
|
|
if (size > 1 * 1024 * 1024) return false;
|
|
|
|
return isSvg(fs.readFileSync(path));
|
|
|
|
} catch {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get file size
|
|
|
|
*/
|
|
|
|
export async function getFileSize(path: string): Promise<number> {
|
2020-04-11 11:28:40 +02:00
|
|
|
const getStat = util.promisify(fs.stat);
|
|
|
|
return (await getStat(path)).size;
|
2020-01-12 08:40:58 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Calculate MD5 hash
|
|
|
|
*/
|
|
|
|
async function calcHash(path: string): Promise<string> {
|
2023-01-13 05:40:33 +01:00
|
|
|
const hash = crypto.createHash("md5").setEncoding("hex");
|
2020-04-11 11:28:40 +02:00
|
|
|
await pipeline(fs.createReadStream(path), hash);
|
|
|
|
return hash.read();
|
2020-01-12 08:40:58 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Detect dimensions of image
|
|
|
|
*/
|
|
|
|
async function detectImageSize(path: string): Promise<{
|
|
|
|
width: number;
|
|
|
|
height: number;
|
|
|
|
wUnits: string;
|
|
|
|
hUnits: string;
|
2021-12-03 03:19:28 +01:00
|
|
|
orientation?: number;
|
2020-01-12 08:40:58 +01:00
|
|
|
}> {
|
|
|
|
const readable = fs.createReadStream(path);
|
|
|
|
const imageSize = await probeImageSize(readable);
|
|
|
|
readable.destroy();
|
|
|
|
return imageSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Calculate average color of image
|
|
|
|
*/
|
2020-07-18 17:24:07 +02:00
|
|
|
function getBlurhash(path: string): Promise<string> {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
sharp(path)
|
|
|
|
.raw()
|
|
|
|
.ensureAlpha()
|
2023-01-13 05:40:33 +01:00
|
|
|
.resize(64, 64, { fit: "inside" })
|
2020-07-18 17:24:07 +02:00
|
|
|
.toBuffer((err, buffer, { width, height }) => {
|
|
|
|
if (err) return reject(err);
|
2020-10-17 13:12:00 +02:00
|
|
|
|
|
|
|
let hash;
|
|
|
|
|
|
|
|
try {
|
2023-07-14 04:06:57 +02:00
|
|
|
hash = encode(new Uint8ClampedArray(buffer), width, height, 7, 7);
|
2020-10-17 13:12:00 +02:00
|
|
|
} catch (e) {
|
|
|
|
return reject(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
resolve(hash);
|
2020-07-18 17:24:07 +02:00
|
|
|
});
|
|
|
|
});
|
2020-01-12 08:40:58 +01:00
|
|
|
}
|