UNPKG

payload

Version:

Node, React and MongoDB Headless CMS and Application Framework

228 lines (227 loc) • 29.1 kB
"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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"}