2022-12-12 11:45:29 +01:00

452 lines
13 KiB
JavaScript

'use strict';
/**
* Upload.js service
*
* @description: A set of functions similar to controller's actions to avoid code duplication.
*/
const os = require('os');
const path = require('path');
const crypto = require('crypto');
const fs = require('fs');
const fse = require('fs-extra');
const _ = require('lodash');
const { extension } = require('mime-types');
const {
sanitize,
nameToSlug,
contentTypes: contentTypesUtils,
webhook: webhookUtils,
} = require('@strapi/utils');
const { NotFoundError } = require('@strapi/utils').errors;
const { MEDIA_UPDATE, MEDIA_CREATE, MEDIA_DELETE } = webhookUtils.webhookEvents;
const { ApplicationError } = require('@strapi/utils/lib/errors');
const { FILE_MODEL_UID } = require('../constants');
const { getService } = require('../utils');
const { bytesToKbytes } = require('../utils/file');
const { UPDATED_BY_ATTRIBUTE, CREATED_BY_ATTRIBUTE } = contentTypesUtils.constants;
const randomSuffix = () => crypto.randomBytes(5).toString('hex');
const generateFileName = (name) => {
const baseName = nameToSlug(name, { separator: '_', lowercase: false });
return `${baseName}_${randomSuffix()}`;
};
const sendMediaMetrics = (data) => {
if (_.has(data, 'caption') && !_.isEmpty(data.caption)) {
strapi.telemetry.send('didSaveMediaWithCaption');
}
if (_.has(data, 'alternativeText') && !_.isEmpty(data.alternativeText)) {
strapi.telemetry.send('didSaveMediaWithAlternativeText');
}
};
const createAndAssignTmpWorkingDirectoryToFiles = async (files) => {
const tmpWorkingDirectory = await fse.mkdtemp(path.join(os.tmpdir(), 'strapi-upload-'));
if (Array.isArray(files)) {
files.forEach((file) => {
file.tmpWorkingDirectory = tmpWorkingDirectory;
});
} else {
files.tmpWorkingDirectory = tmpWorkingDirectory;
}
return tmpWorkingDirectory;
};
module.exports = ({ strapi }) => ({
async emitEvent(event, data) {
const modelDef = strapi.getModel(FILE_MODEL_UID);
const sanitizedData = await sanitize.sanitizers.defaultSanitizeOutput(modelDef, data);
strapi.eventHub.emit(event, { media: sanitizedData });
},
async formatFileInfo({ filename, type, size }, fileInfo = {}, metas = {}) {
const fileService = getService('file');
let ext = path.extname(filename);
if (!ext) {
ext = `.${extension(type)}`;
}
const usedName = (fileInfo.name || filename).normalize();
const basename = path.basename(usedName, ext);
const entity = {
name: usedName,
alternativeText: fileInfo.alternativeText,
caption: fileInfo.caption,
folder: fileInfo.folder,
folderPath: await fileService.getFolderPath(fileInfo.folder),
hash: generateFileName(basename),
ext,
mime: type,
size: bytesToKbytes(size),
};
const { refId, ref, field } = metas;
if (refId && ref && field) {
entity.related = [
{
id: refId,
__type: ref,
__pivot: { field },
},
];
}
if (metas.path) {
entity.path = metas.path;
}
if (metas.tmpWorkingDirectory) {
entity.tmpWorkingDirectory = metas.tmpWorkingDirectory;
}
return entity;
},
async enhanceAndValidateFile(file, fileInfo = {}, metas = {}) {
const currentFile = await this.formatFileInfo(
{
filename: file.name,
type: file.type,
size: file.size,
},
fileInfo,
{
...metas,
tmpWorkingDirectory: file.tmpWorkingDirectory,
}
);
currentFile.getStream = () => fs.createReadStream(file.path);
const { optimize, isImage, isFaultyImage, isOptimizableImage } = strapi
.plugin('upload')
.service('image-manipulation');
if (await isImage(currentFile)) {
if (await isFaultyImage(currentFile)) {
throw new ApplicationError('File is not a valid image');
}
if (await isOptimizableImage(currentFile)) {
return optimize(currentFile);
}
}
return currentFile;
},
// TODO V5: remove enhanceFile
async enhanceFile(file, fileInfo = {}, metas = {}) {
process.emitWarning(
'[deprecated] In future versions, `enhanceFile` will be removed. Replace it with `enhanceAndValidateFile` instead.'
);
return this.enhanceAndValidateFile(file, fileInfo, metas);
},
async upload({ data, files }, { user } = {}) {
// create temporary folder to store files for stream manipulation
const tmpWorkingDirectory = await createAndAssignTmpWorkingDirectoryToFiles(files);
let uploadedFiles = [];
try {
const { fileInfo, ...metas } = data;
const fileArray = Array.isArray(files) ? files : [files];
const fileInfoArray = Array.isArray(fileInfo) ? fileInfo : [fileInfo];
const doUpload = async (file, fileInfo) => {
const fileData = await this.enhanceAndValidateFile(file, fileInfo, metas);
return this.uploadFileAndPersist(fileData, { user });
};
uploadedFiles = await Promise.all(
fileArray.map((file, idx) => doUpload(file, fileInfoArray[idx] || {}))
);
} finally {
// delete temporary folder
await fse.remove(tmpWorkingDirectory);
}
return uploadedFiles;
},
/**
* When uploading an image, an additional thubmnail is generated.
* Also, if there are responsive formats defined, another set of images will be generated too.
*
* @param {*} fileData
*/
async uploadImage(fileData) {
const { getDimensions, generateThumbnail, generateResponsiveFormats, isOptimizableImage } =
getService('image-manipulation');
// Store width and height of the original image
const { width, height } = await getDimensions(fileData);
// Make sure this is assigned before calling any upload
// That way it can mutate the width and height
_.assign(fileData, {
width,
height,
});
// For performance reasons, all uploads are wrapped in a single Promise.all
const uploadThumbnail = async (thumbnailFile) => {
await getService('provider').upload(thumbnailFile);
_.set(fileData, 'formats.thumbnail', thumbnailFile);
};
const uploadResponsiveFormat = async (format) => {
const { key, file } = format;
await getService('provider').upload(file);
_.set(fileData, ['formats', key], file);
};
const uploadPromises = [];
// Upload image
uploadPromises.push(getService('provider').upload(fileData));
// Generate & Upload thumbnail and responsive formats
if (await isOptimizableImage(fileData)) {
const thumbnailFile = await generateThumbnail(fileData);
if (thumbnailFile) {
uploadPromises.push(uploadThumbnail(thumbnailFile));
}
const formats = await generateResponsiveFormats(fileData);
if (Array.isArray(formats) && formats.length > 0) {
for (const format of formats) {
if (!format) continue;
uploadPromises.push(uploadResponsiveFormat(format));
}
}
}
// Wait for all uploads to finish
await Promise.all(uploadPromises);
},
/**
* Upload a file. If it is an image it will generate a thumbnail
* and responsive formats (if enabled).
*/
async uploadFileAndPersist(fileData, { user } = {}) {
const config = strapi.config.get('plugin.upload');
const { isImage } = getService('image-manipulation');
if (await isImage(fileData)) {
await this.uploadImage(fileData);
} else {
await getService('provider').upload(fileData);
}
_.set(fileData, 'provider', config.provider);
// Persist file(s)
return this.add(fileData, { user });
},
async updateFileInfo(id, { name, alternativeText, caption, folder }, { user } = {}) {
const dbFile = await this.findOne(id);
if (!dbFile) {
throw new NotFoundError();
}
const fileService = getService('file');
const newName = _.isNil(name) ? dbFile.name : name;
const newInfos = {
name: newName,
alternativeText: _.isNil(alternativeText) ? dbFile.alternativeText : alternativeText,
caption: _.isNil(caption) ? dbFile.caption : caption,
folder: _.isUndefined(folder) ? dbFile.folder : folder,
folderPath: _.isUndefined(folder) ? dbFile.path : await fileService.getFolderPath(folder),
};
return this.update(id, newInfos, { user });
},
async replace(id, { data, file }, { user } = {}) {
const config = strapi.config.get('plugin.upload');
const { isImage } = getService('image-manipulation');
const dbFile = await this.findOne(id);
if (!dbFile) {
throw new NotFoundError();
}
// create temporary folder to store files for stream manipulation
const tmpWorkingDirectory = await createAndAssignTmpWorkingDirectoryToFiles(file);
let fileData;
try {
const { fileInfo } = data;
fileData = await this.enhanceAndValidateFile(file, fileInfo);
// keep a constant hash and extension so the file url doesn't change when the file is replaced
_.assign(fileData, {
hash: dbFile.hash,
ext: dbFile.ext,
});
// execute delete function of the provider
if (dbFile.provider === config.provider) {
await strapi.plugin('upload').provider.delete(dbFile);
if (dbFile.formats) {
await Promise.all(
Object.keys(dbFile.formats).map((key) => {
return strapi.plugin('upload').provider.delete(dbFile.formats[key]);
})
);
}
}
// clear old formats
_.set(fileData, 'formats', {});
if (await isImage(fileData)) {
await this.uploadImage(fileData);
} else {
await getService('provider').upload(fileData);
}
_.set(fileData, 'provider', config.provider);
} finally {
// delete temporary folder
await fse.remove(tmpWorkingDirectory);
}
return this.update(id, fileData, { user });
},
async update(id, values, { user } = {}) {
const fileValues = { ...values };
if (user) {
fileValues[UPDATED_BY_ATTRIBUTE] = user.id;
}
sendMediaMetrics(fileValues);
const res = await strapi.entityService.update(FILE_MODEL_UID, id, { data: fileValues });
await this.emitEvent(MEDIA_UPDATE, res);
return res;
},
async add(values, { user } = {}) {
const fileValues = { ...values };
if (user) {
fileValues[UPDATED_BY_ATTRIBUTE] = user.id;
fileValues[CREATED_BY_ATTRIBUTE] = user.id;
}
sendMediaMetrics(fileValues);
const res = await strapi.query(FILE_MODEL_UID).create({ data: fileValues });
await this.emitEvent(MEDIA_CREATE, res);
return res;
},
findOne(id, populate) {
return strapi.entityService.findOne(FILE_MODEL_UID, id, { populate });
},
findMany(query) {
return strapi.entityService.findMany(FILE_MODEL_UID, query);
},
findPage(query) {
return strapi.entityService.findPage(FILE_MODEL_UID, query);
},
async remove(file) {
const config = strapi.config.get('plugin.upload');
// execute delete function of the provider
if (file.provider === config.provider) {
await strapi.plugin('upload').provider.delete(file);
if (file.formats) {
await Promise.all(
Object.keys(file.formats).map((key) => {
return strapi.plugin('upload').provider.delete(file.formats[key]);
})
);
}
}
const media = await strapi.query(FILE_MODEL_UID).findOne({
where: { id: file.id },
});
await this.emitEvent(MEDIA_DELETE, media);
return strapi.query(FILE_MODEL_UID).delete({ where: { id: file.id } });
},
async uploadToEntity(params, files) {
const { id, model, field } = params;
// create temporary folder to store files for stream manipulation
const tmpWorkingDirectory = await createAndAssignTmpWorkingDirectoryToFiles(files);
const arr = Array.isArray(files) ? files : [files];
const apiUploadFolderService = getService('api-upload-folder');
const apiUploadFolder = await apiUploadFolderService.getAPIUploadFolder();
try {
const enhancedFiles = await Promise.all(
arr.map((file) => {
return this.enhanceAndValidateFile(
file,
{ folder: apiUploadFolder.id },
{
refId: id,
ref: model,
field,
}
);
})
);
await Promise.all(enhancedFiles.map((file) => this.uploadFileAndPersist(file)));
} finally {
// delete temporary folder
await fse.remove(tmpWorkingDirectory);
}
},
getSettings() {
return strapi.store({ type: 'plugin', name: 'upload', key: 'settings' }).get();
},
setSettings(value) {
if (value.responsiveDimensions === true) {
strapi.telemetry.send('didEnableResponsiveDimensions');
} else {
strapi.telemetry.send('didDisableResponsiveDimensions');
}
return strapi.store({ type: 'plugin', name: 'upload', key: 'settings' }).set({ value });
},
});