payload
Version:
Node, React and MongoDB Headless CMS and Application Framework
228 lines (227 loc) • 29.1 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
Object.defineProperty(exports, "generateFileData", {
enumerable: true,
get: function() {
return generateFileData;
}
});
const _filetype = require("file-type");
const _fs = /*#__PURE__*/ _interop_require_default(require("fs"));
const _mkdirp = /*#__PURE__*/ _interop_require_default(require("mkdirp"));
const _path = /*#__PURE__*/ _interop_require_default(require("path"));
const _sanitizefilename = /*#__PURE__*/ _interop_require_default(require("sanitize-filename"));
const _sharp = /*#__PURE__*/ _interop_require_default(require("sharp"));
const _errors = require("../errors");
const _canResizeImage = /*#__PURE__*/ _interop_require_default(require("./canResizeImage"));
const _cropImage = /*#__PURE__*/ _interop_require_default(require("./cropImage"));
const _getFileByPath = /*#__PURE__*/ _interop_require_default(require("./getFileByPath"));
const _getExternalFile = require("./getExternalFile");
const _getImageSize = /*#__PURE__*/ _interop_require_default(require("./getImageSize"));
const _getSafeFilename = /*#__PURE__*/ _interop_require_default(require("./getSafeFilename"));
const _imageResizer = /*#__PURE__*/ _interop_require_default(require("./imageResizer"));
const _isImage = /*#__PURE__*/ _interop_require_default(require("./isImage"));
function _interop_require_default(obj) {
return obj && obj.__esModule ? obj : {
default: obj
};
}
const generateFileData = async ({ collection: { config: collectionConfig }, config, data, overwriteExistingFiles, req, throwOnMissingFile })=>{
if (!collectionConfig.upload) {
return {
data,
files: []
};
}
let file = req.files?.file || undefined;
const { uploadEdits } = req.query || {};
const { disableLocalStorage, formatOptions, imageSizes, resizeOptions, staticDir, trimOptions } = collectionConfig.upload;
let staticPath = staticDir;
if (staticDir.indexOf('/') !== 0) {
staticPath = _path.default.resolve(config.paths.configDir, staticDir);
}
if (!file && uploadEdits && data) {
const { filename, url } = data;
try {
if (url && url.startsWith('/') && !disableLocalStorage) {
const filePath = `${staticPath}/${filename}`;
const response = await (0, _getFileByPath.default)(filePath);
file = response;
overwriteExistingFiles = true;
} else if (filename && url) {
file = await (0, _getExternalFile.getExternalFile)({
req,
data: data
});
overwriteExistingFiles = true;
}
} catch (err) {
throw new _errors.FileUploadError(req.t);
}
}
if (!file) {
if (throwOnMissingFile) throw new _errors.MissingFile(req.t);
return {
data,
files: []
};
}
if (!disableLocalStorage) {
_mkdirp.default.sync(staticPath);
}
let newData = data;
const filesToSave = [];
const fileData = {};
const fileIsAnimated = file.mimetype === 'image/gif' || file.mimetype === 'image/webp';
const cropData = typeof uploadEdits === 'object' && 'crop' in uploadEdits ? uploadEdits.crop : undefined;
try {
const fileSupportsResize = (0, _canResizeImage.default)(file.mimetype);
let fsSafeName;
let sharpFile;
let dimensions;
let fileBuffer;
let ext;
let mime;
const fileHasAdjustments = fileSupportsResize && Boolean(resizeOptions || formatOptions || trimOptions || file.tempFilePath);
const sharpOptions = {};
if (fileIsAnimated) sharpOptions.animated = true;
if (fileHasAdjustments) {
if (file.tempFilePath) {
sharpFile = (0, _sharp.default)(file.tempFilePath, sharpOptions).rotate() // pass rotate() to auto-rotate based on EXIF data. https://github.com/payloadcms/payload/pull/3081
;
} else {
sharpFile = (0, _sharp.default)(file.data, sharpOptions).rotate() // pass rotate() to auto-rotate based on EXIF data. https://github.com/payloadcms/payload/pull/3081
;
}
if (resizeOptions) {
sharpFile = sharpFile.resize(resizeOptions);
}
if (formatOptions) {
sharpFile = sharpFile.toFormat(formatOptions.format, formatOptions.options);
}
if (trimOptions) {
sharpFile = sharpFile.trim(trimOptions);
}
}
if ((0, _isImage.default)(file.mimetype)) {
dimensions = await (0, _getImageSize.default)(file);
fileData.width = dimensions.width;
fileData.height = dimensions.height;
}
if (sharpFile) {
const metadata = await sharpFile.metadata();
fileBuffer = await sharpFile.toBuffer({
resolveWithObject: true
});
({ ext, mime } = await (0, _filetype.fromBuffer)(fileBuffer.data) // This is getting an incorrect gif height back.
);
fileData.width = fileBuffer.info.width;
fileData.height = fileBuffer.info.height;
fileData.filesize = fileBuffer.info.size;
// Animated GIFs + WebP aggregate the height from every frame, so we need to use divide by number of pages
if (metadata.pages) {
fileData.height = fileBuffer.info.height / metadata.pages;
fileData.filesize = fileBuffer.data.length;
}
} else {
mime = file.mimetype;
fileData.filesize = file.size;
if (file.name.includes('.')) {
ext = file.name.split('.').pop();
} else {
ext = '';
}
}
// Adust SVG mime type. fromBuffer modifies it.
if (mime === 'application/xml' && ext === 'svg') mime = 'image/svg+xml';
fileData.mimeType = mime;
const baseFilename = (0, _sanitizefilename.default)(file.name.substring(0, file.name.lastIndexOf('.')) || file.name);
fsSafeName = `${baseFilename}${ext ? `.${ext}` : ''}`;
if (!overwriteExistingFiles) {
fsSafeName = await (0, _getSafeFilename.default)({
collectionSlug: collectionConfig.slug,
desiredFilename: fsSafeName,
req,
staticPath
});
}
fileData.filename = fsSafeName;
let fileForResize = file;
if (cropData) {
const { data: croppedImage, info } = await (0, _cropImage.default)({
cropData,
dimensions,
file
});
filesToSave.push({
buffer: croppedImage,
path: `${staticPath}/${fsSafeName}`
});
fileForResize = {
...file,
data: croppedImage,
size: info.size
};
fileData.width = info.width;
fileData.height = info.height;
fileData.filesize = info.size;
if (file.tempFilePath) {
await _fs.default.promises.writeFile(file.tempFilePath, croppedImage) // write fileBuffer to the temp path
;
} else {
req.files.file = fileForResize;
}
} else {
filesToSave.push({
buffer: fileBuffer?.data || file.data,
path: `${staticPath}/${fsSafeName}`
});
// If using temp files and the image is being resized, write the file to the temp path
if (fileBuffer?.data || file.data.length > 0) {
if (file.tempFilePath) {
await _fs.default.promises.writeFile(file.tempFilePath, fileBuffer?.data || file.data) // write fileBuffer to the temp path
;
} else {
// Assign the _possibly modified_ file to the request object
req.files.file = {
...file,
data: fileBuffer?.data || file.data,
size: fileBuffer?.info.size
};
}
}
}
if (Array.isArray(imageSizes) && fileSupportsResize) {
req.payloadUploadSizes = {};
const { sizeData, sizesToSave } = await (0, _imageResizer.default)({
config: collectionConfig,
dimensions: !cropData ? dimensions : {
...dimensions,
height: fileData.height,
width: fileData.width
},
file: fileForResize,
mimeType: fileData.mimeType,
req,
savedFilename: fsSafeName || file.name,
staticPath
});
fileData.sizes = sizeData;
filesToSave.push(...sizesToSave);
}
} catch (err) {
console.error(err);
throw new _errors.FileUploadError(req.t);
}
newData = {
...newData,
...fileData
};
return {
data: newData,
files: filesToSave
};
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/uploads/generateFileData.ts"],"sourcesContent":["import type { UploadedFile } from 'express-fileupload'\nimport type { OutputInfo, Sharp, SharpOptions } from 'sharp'\n\nimport { fromBuffer } from 'file-type'\nimport fs from 'fs'\nimport mkdirp from 'mkdirp'\nimport path from 'path'\nimport sanitize from 'sanitize-filename'\nimport sharp from 'sharp'\n\nimport type { Collection } from '../collections/config/types'\nimport type { SanitizedConfig } from '../config/types'\nimport type { PayloadRequest } from '../express/types'\nimport type { FileData, FileToSave, ProbedImageSize } from './types'\n\nimport { FileUploadError, MissingFile } from '../errors'\nimport canResizeImage from './canResizeImage'\nimport cropImage from './cropImage'\nimport getFileByPath from './getFileByPath'\nimport { getExternalFile } from './getExternalFile'\nimport getImageSize from './getImageSize'\nimport getSafeFileName from './getSafeFilename'\nimport resizeAndTransformImageSizes from './imageResizer'\nimport isImage from './isImage'\n\ntype Args<T> = {\n  collection: Collection\n  config: SanitizedConfig\n  data: T\n  overwriteExistingFiles?: boolean\n  req: PayloadRequest\n  throwOnMissingFile?: boolean\n}\n\ntype Result<T> = Promise<{\n  data: T\n  files: FileToSave[]\n}>\n\nexport const generateFileData = async <T>({\n  collection: { config: collectionConfig },\n  config,\n  data,\n  overwriteExistingFiles,\n  req,\n  throwOnMissingFile,\n}: Args<T>): Result<T> => {\n  if (!collectionConfig.upload) {\n    return {\n      data,\n      files: [],\n    }\n  }\n\n  let file = req.files?.file || undefined\n  const { uploadEdits } = req.query || {}\n\n  const { disableLocalStorage, formatOptions, imageSizes, resizeOptions, staticDir, trimOptions } =\n    collectionConfig.upload\n\n  let staticPath = staticDir\n  if (staticDir.indexOf('/') !== 0) {\n    staticPath = path.resolve(config.paths.configDir, staticDir)\n  }\n\n  if (!file && uploadEdits && data) {\n    const { filename, url } = data as FileData\n\n    try {\n      if (url && url.startsWith('/') && !disableLocalStorage) {\n        const filePath = `${staticPath}/${filename}`\n        const response = await getFileByPath(filePath)\n        file = response as UploadedFile\n        overwriteExistingFiles = true\n      } else if (filename && url) {\n        file = (await getExternalFile({ req, data: data as FileData })) as UploadedFile\n        overwriteExistingFiles = true\n      }\n    } catch (err) {\n      throw new FileUploadError(req.t)\n    }\n  }\n\n  if (!file) {\n    if (throwOnMissingFile) throw new MissingFile(req.t)\n\n    return {\n      data,\n      files: [],\n    }\n  }\n\n  if (!disableLocalStorage) {\n    mkdirp.sync(staticPath)\n  }\n\n  let newData = data\n  const filesToSave: FileToSave[] = []\n  const fileData: Partial<FileData> = {}\n  const fileIsAnimated = file.mimetype === 'image/gif' || file.mimetype === 'image/webp'\n  const cropData =\n    typeof uploadEdits === 'object' && 'crop' in uploadEdits ? uploadEdits.crop : undefined\n\n  try {\n    const fileSupportsResize = canResizeImage(file.mimetype)\n    let fsSafeName: string\n    let sharpFile: Sharp | undefined\n    let dimensions: ProbedImageSize | undefined\n    let fileBuffer: { data: Buffer; info: OutputInfo }\n    let ext\n    let mime: string\n    const fileHasAdjustments =\n      fileSupportsResize &&\n      Boolean(resizeOptions || formatOptions || trimOptions || file.tempFilePath)\n\n    const sharpOptions: SharpOptions = {}\n\n    if (fileIsAnimated) sharpOptions.animated = true\n\n    if (fileHasAdjustments) {\n      if (file.tempFilePath) {\n        sharpFile = sharp(file.tempFilePath, sharpOptions).rotate() // pass rotate() to auto-rotate based on EXIF data. https://github.com/payloadcms/payload/pull/3081\n      } else {\n        sharpFile = sharp(file.data, sharpOptions).rotate() // pass rotate() to auto-rotate based on EXIF data. https://github.com/payloadcms/payload/pull/3081\n      }\n\n      if (resizeOptions) {\n        sharpFile = sharpFile.resize(resizeOptions)\n      }\n      if (formatOptions) {\n        sharpFile = sharpFile.toFormat(formatOptions.format, formatOptions.options)\n      }\n      if (trimOptions) {\n        sharpFile = sharpFile.trim(trimOptions)\n      }\n    }\n\n    if (isImage(file.mimetype)) {\n      dimensions = await getImageSize(file)\n      fileData.width = dimensions.width\n      fileData.height = dimensions.height\n    }\n\n    if (sharpFile) {\n      const metadata = await sharpFile.metadata()\n      fileBuffer = await sharpFile.toBuffer({ resolveWithObject: true })\n      ;({ ext, mime } = await fromBuffer(fileBuffer.data)) // This is getting an incorrect gif height back.\n      fileData.width = fileBuffer.info.width\n      fileData.height = fileBuffer.info.height\n      fileData.filesize = fileBuffer.info.size\n\n      // Animated GIFs + WebP aggregate the height from every frame, so we need to use divide by number of pages\n      if (metadata.pages) {\n        fileData.height = fileBuffer.info.height / metadata.pages\n        fileData.filesize = fileBuffer.data.length\n      }\n    } else {\n      mime = file.mimetype\n      fileData.filesize = file.size\n\n      if (file.name.includes('.')) {\n        ext = file.name.split('.').pop()\n      } else {\n        ext = ''\n      }\n    }\n\n    // Adust SVG mime type. fromBuffer modifies it.\n    if (mime === 'application/xml' && ext === 'svg') mime = 'image/svg+xml'\n    fileData.mimeType = mime\n\n    const baseFilename = sanitize(file.name.substring(0, file.name.lastIndexOf('.')) || file.name)\n    fsSafeName = `${baseFilename}${ext ? `.${ext}` : ''}`\n\n    if (!overwriteExistingFiles) {\n      fsSafeName = await getSafeFileName({\n        collectionSlug: collectionConfig.slug,\n        desiredFilename: fsSafeName,\n        req,\n        staticPath,\n      })\n    }\n\n    fileData.filename = fsSafeName\n    let fileForResize = file\n\n    if (cropData) {\n      const { data: croppedImage, info } = await cropImage({ cropData, dimensions, file })\n\n      filesToSave.push({\n        buffer: croppedImage,\n        path: `${staticPath}/${fsSafeName}`,\n      })\n\n      fileForResize = {\n        ...file,\n        data: croppedImage,\n        size: info.size,\n      }\n      fileData.width = info.width\n      fileData.height = info.height\n      fileData.filesize = info.size\n\n      if (file.tempFilePath) {\n        await fs.promises.writeFile(file.tempFilePath, croppedImage) // write fileBuffer to the temp path\n      } else {\n        req.files.file = fileForResize\n      }\n    } else {\n      filesToSave.push({\n        buffer: fileBuffer?.data || file.data,\n        path: `${staticPath}/${fsSafeName}`,\n      })\n\n      // If using temp files and the image is being resized, write the file to the temp path\n      if (fileBuffer?.data || file.data.length > 0) {\n        if (file.tempFilePath) {\n          await fs.promises.writeFile(file.tempFilePath, fileBuffer?.data || file.data) // write fileBuffer to the temp path\n        } else {\n          // Assign the _possibly modified_ file to the request object\n          req.files.file = {\n            ...file,\n            data: fileBuffer?.data || file.data,\n            size: fileBuffer?.info.size,\n          }\n        }\n      }\n    }\n\n    if (Array.isArray(imageSizes) && fileSupportsResize) {\n      req.payloadUploadSizes = {}\n      const { sizeData, sizesToSave } = await resizeAndTransformImageSizes({\n        config: collectionConfig,\n        dimensions: !cropData\n          ? dimensions\n          : {\n              ...dimensions,\n              height: fileData.height,\n              width: fileData.width,\n            },\n        file: fileForResize,\n        mimeType: fileData.mimeType,\n        req,\n        savedFilename: fsSafeName || file.name,\n        staticPath,\n      })\n\n      fileData.sizes = sizeData\n      filesToSave.push(...sizesToSave)\n    }\n  } catch (err) {\n    console.error(err)\n    throw new FileUploadError(req.t)\n  }\n\n  newData = {\n    ...newData,\n    ...fileData,\n  }\n\n  return {\n    data: newData,\n    files: filesToSave,\n  }\n}\n"],"names":["generateFileData","collection","config","collectionConfig","data","overwriteExistingFiles","req","throwOnMissingFile","upload","files","file","undefined","uploadEdits","query","disableLocalStorage","formatOptions","imageSizes","resizeOptions","staticDir","trimOptions","staticPath","indexOf","path","resolve","paths","configDir","filename","url","startsWith","filePath","response","getFileByPath","getExternalFile","err","FileUploadError","t","MissingFile","mkdirp","sync","newData","filesToSave","fileData","fileIsAnimated","mimetype","cropData","crop","fileSupportsResize","canResizeImage","fsSafeName","sharpFile","dimensions","fileBuffer","ext","mime","fileHasAdjustments","Boolean","tempFilePath","sharpOptions","animated","sharp","rotate","resize","toFormat","format","options","trim","isImage","getImageSize","width","height","metadata","toBuffer","resolveWithObject","fromBuffer","info","filesize","size","pages","length","name","includes","split","pop","mimeType","baseFilename","sanitize","substring","lastIndexOf","getSafeFileName","collectionSlug","slug","desiredFilename","fileForResize","croppedImage","cropImage","push","buffer","fs","promises","writeFile","Array","isArray","payloadUploadSizes","sizeData","sizesToSave","resizeAndTransformImageSizes","savedFilename","sizes","console","error"],"mappings":"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"}