2018-02-08 12:01:06 +01:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Upload.js service
|
|
|
|
*
|
|
|
|
* @description: A set of functions similar to controller's actions to avoid code duplication.
|
|
|
|
*/
|
|
|
|
|
2022-01-04 19:21:05 +01:00
|
|
|
const os = require('os');
|
2020-03-03 15:54:59 +01:00
|
|
|
const path = require('path');
|
2018-09-17 15:50:13 +08:00
|
|
|
const crypto = require('crypto');
|
2022-01-05 17:36:21 +01:00
|
|
|
const fs = require('fs');
|
|
|
|
const fse = require('fs-extra');
|
2020-10-27 11:27:17 +01:00
|
|
|
const _ = require('lodash');
|
2022-08-05 15:50:11 +02:00
|
|
|
const { extension } = require('mime-types');
|
2020-08-31 16:23:43 +02:00
|
|
|
const {
|
2021-11-08 15:52:42 +01:00
|
|
|
sanitize,
|
2020-08-31 16:23:43 +02:00
|
|
|
nameToSlug,
|
|
|
|
contentTypes: contentTypesUtils,
|
|
|
|
webhook: webhookUtils,
|
2021-04-29 13:51:12 +02:00
|
|
|
} = require('@strapi/utils');
|
2022-08-03 20:39:44 +02:00
|
|
|
const { NotFoundError } = require('@strapi/utils').errors;
|
2020-10-27 11:27:17 +01:00
|
|
|
|
2020-08-31 16:23:43 +02:00
|
|
|
const { MEDIA_UPDATE, MEDIA_CREATE, MEDIA_DELETE } = webhookUtils.webhookEvents;
|
2020-04-03 13:35:33 +02:00
|
|
|
|
2022-08-04 12:56:58 +02:00
|
|
|
const { ApplicationError } = require('@strapi/utils/lib/errors');
|
2022-05-13 16:10:18 +02:00
|
|
|
const { FILE_MODEL_UID } = require('../constants');
|
2021-08-19 22:27:00 +02:00
|
|
|
const { getService } = require('../utils');
|
2020-03-06 18:41:48 +01:00
|
|
|
const { bytesToKbytes } = require('../utils/file');
|
2020-03-03 15:54:59 +01:00
|
|
|
|
2020-08-18 17:09:21 +02:00
|
|
|
const { UPDATED_BY_ATTRIBUTE, CREATED_BY_ATTRIBUTE } = contentTypesUtils.constants;
|
|
|
|
|
2020-03-03 15:54:59 +01:00
|
|
|
const randomSuffix = () => crypto.randomBytes(5).toString('hex');
|
2020-08-18 17:09:21 +02:00
|
|
|
|
2022-08-08 23:33:39 +02:00
|
|
|
const generateFileName = (name) => {
|
2020-05-13 13:17:25 +02:00
|
|
|
const baseName = nameToSlug(name, { separator: '_', lowercase: false });
|
2020-03-03 15:54:59 +01:00
|
|
|
|
|
|
|
return `${baseName}_${randomSuffix()}`;
|
|
|
|
};
|
2018-09-17 15:50:13 +08:00
|
|
|
|
2022-08-08 23:33:39 +02:00
|
|
|
const sendMediaMetrics = (data) => {
|
2020-04-27 16:08:55 +02:00
|
|
|
if (_.has(data, 'caption') && !_.isEmpty(data.caption)) {
|
|
|
|
strapi.telemetry.send('didSaveMediaWithCaption');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_.has(data, 'alternativeText') && !_.isEmpty(data.alternativeText)) {
|
|
|
|
strapi.telemetry.send('didSaveMediaWithAlternativeText');
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-08-08 23:33:39 +02:00
|
|
|
const createAndAssignTmpWorkingDirectoryToFiles = async (files) => {
|
2022-03-01 14:52:31 +01:00
|
|
|
const tmpWorkingDirectory = await fse.mkdtemp(path.join(os.tmpdir(), 'strapi-upload-'));
|
2022-02-28 17:44:18 +01:00
|
|
|
|
2022-08-08 23:33:39 +02:00
|
|
|
if (Array.isArray(files)) {
|
|
|
|
files.forEach((file) => {
|
|
|
|
file.tmpWorkingDirectory = tmpWorkingDirectory;
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
files.tmpWorkingDirectory = tmpWorkingDirectory;
|
|
|
|
}
|
2022-02-28 17:44:18 +01:00
|
|
|
|
|
|
|
return tmpWorkingDirectory;
|
|
|
|
};
|
|
|
|
|
2021-07-08 11:20:13 +02:00
|
|
|
module.exports = ({ strapi }) => ({
|
2021-11-05 10:36:10 +01:00
|
|
|
async emitEvent(event, data) {
|
2022-05-13 16:10:18 +02:00
|
|
|
const modelDef = strapi.getModel(FILE_MODEL_UID);
|
2021-11-10 17:08:54 +01:00
|
|
|
const sanitizedData = await sanitize.sanitizers.defaultSanitizeOutput(modelDef, data);
|
2021-11-05 10:36:10 +01:00
|
|
|
|
|
|
|
strapi.eventHub.emit(event, { media: sanitizedData });
|
2021-07-08 18:15:32 +02:00
|
|
|
},
|
|
|
|
|
2022-03-30 16:26:09 +02:00
|
|
|
async formatFileInfo({ filename, type, size }, fileInfo = {}, metas = {}) {
|
2022-04-06 11:36:13 +02:00
|
|
|
const fileService = getService('file');
|
|
|
|
|
2022-08-08 10:38:42 +02:00
|
|
|
let ext = path.extname(filename);
|
|
|
|
if (!ext) {
|
2022-08-11 16:34:48 +02:00
|
|
|
ext = `.${extension(type)}`;
|
2022-08-08 09:37:37 +02:00
|
|
|
}
|
2020-06-25 11:13:42 +03:00
|
|
|
const basename = path.basename(fileInfo.name || filename, ext);
|
|
|
|
const usedName = fileInfo.name || filename;
|
2020-03-03 15:54:59 +01:00
|
|
|
|
|
|
|
const entity = {
|
|
|
|
name: usedName,
|
|
|
|
alternativeText: fileInfo.alternativeText,
|
|
|
|
caption: fileInfo.caption,
|
2022-03-30 16:26:09 +02:00
|
|
|
folder: fileInfo.folder,
|
2022-04-12 16:32:05 +02:00
|
|
|
folderPath: await fileService.getFolderPath(fileInfo.folder),
|
2020-06-25 11:13:42 +03:00
|
|
|
hash: generateFileName(basename),
|
2020-03-03 15:54:59 +01:00
|
|
|
ext,
|
|
|
|
mime: type,
|
2020-03-06 18:41:48 +01:00
|
|
|
size: bytesToKbytes(size),
|
2020-03-03 15:54:59 +01:00
|
|
|
};
|
|
|
|
|
2021-07-28 21:03:32 +02:00
|
|
|
const { refId, ref, field } = metas;
|
2020-03-03 15:54:59 +01:00
|
|
|
|
|
|
|
if (refId && ref && field) {
|
|
|
|
entity.related = [
|
|
|
|
{
|
2021-07-28 21:03:32 +02:00
|
|
|
id: refId,
|
2021-08-17 18:08:42 +02:00
|
|
|
__type: ref,
|
|
|
|
__pivot: { field },
|
2020-03-03 15:54:59 +01:00
|
|
|
},
|
|
|
|
];
|
2018-02-19 14:26:20 +01:00
|
|
|
}
|
|
|
|
|
2020-03-03 15:54:59 +01:00
|
|
|
if (metas.path) {
|
|
|
|
entity.path = metas.path;
|
|
|
|
}
|
|
|
|
|
2022-02-16 12:17:18 +01:00
|
|
|
if (metas.tmpWorkingDirectory) {
|
|
|
|
entity.tmpWorkingDirectory = metas.tmpWorkingDirectory;
|
|
|
|
}
|
|
|
|
|
2020-03-03 15:54:59 +01:00
|
|
|
return entity;
|
|
|
|
},
|
|
|
|
|
2022-08-04 12:56:58 +02:00
|
|
|
async enhanceAndValidateFile(file, fileInfo = {}, metas = {}) {
|
2022-03-30 16:26:09 +02:00
|
|
|
const currentFile = await this.formatFileInfo(
|
2020-03-03 15:54:59 +01:00
|
|
|
{
|
|
|
|
filename: file.name,
|
|
|
|
type: file.type,
|
|
|
|
size: file.size,
|
|
|
|
},
|
|
|
|
fileInfo,
|
2022-02-16 12:17:18 +01:00
|
|
|
{
|
|
|
|
...metas,
|
|
|
|
tmpWorkingDirectory: file.tmpWorkingDirectory,
|
|
|
|
}
|
2020-03-03 15:54:59 +01:00
|
|
|
);
|
2022-01-04 19:21:05 +01:00
|
|
|
currentFile.getStream = () => fs.createReadStream(file.path);
|
2019-08-01 08:52:35 +02:00
|
|
|
|
2022-08-04 12:56:58 +02:00
|
|
|
const { optimize, isImage, isFaultyImage, isOptimizableImage } = strapi
|
|
|
|
.plugin('upload')
|
|
|
|
.service('image-manipulation');
|
2022-01-04 19:21:05 +01:00
|
|
|
|
2022-08-04 12:56:58 +02:00
|
|
|
if (await isImage(currentFile)) {
|
|
|
|
if (await isFaultyImage(currentFile)) {
|
|
|
|
throw new ApplicationError('File is not a valid image');
|
|
|
|
}
|
2022-08-04 17:16:43 +02:00
|
|
|
if (await isOptimizableImage(currentFile)) {
|
|
|
|
return optimize(currentFile);
|
2022-08-04 12:56:58 +02:00
|
|
|
}
|
2022-02-08 15:24:32 +01:00
|
|
|
}
|
2022-08-04 17:16:43 +02:00
|
|
|
return currentFile;
|
2018-02-19 14:26:20 +01:00
|
|
|
},
|
|
|
|
|
2022-08-04 12:56:58 +02:00
|
|
|
// 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);
|
|
|
|
},
|
|
|
|
|
2020-09-08 15:18:32 +02:00
|
|
|
async upload({ data, files }, { user } = {}) {
|
2022-01-04 19:21:05 +01:00
|
|
|
// create temporary folder to store files for stream manipulation
|
2022-03-01 14:52:31 +01:00
|
|
|
const tmpWorkingDirectory = await createAndAssignTmpWorkingDirectoryToFiles(files);
|
2022-02-28 17:44:18 +01:00
|
|
|
|
2022-01-04 19:21:05 +01:00
|
|
|
let uploadedFiles = [];
|
2018-02-19 15:41:26 +01:00
|
|
|
|
2022-01-04 19:21:05 +01:00
|
|
|
try {
|
|
|
|
const { fileInfo, ...metas } = data;
|
2018-02-19 15:41:26 +01:00
|
|
|
|
2022-01-04 19:21:05 +01:00
|
|
|
const fileArray = Array.isArray(files) ? files : [files];
|
|
|
|
const fileInfoArray = Array.isArray(fileInfo) ? fileInfo : [fileInfo];
|
2018-03-07 14:18:15 +01:00
|
|
|
|
2022-01-04 19:21:05 +01:00
|
|
|
const doUpload = async (file, fileInfo) => {
|
2022-08-04 12:56:58 +02:00
|
|
|
const fileData = await this.enhanceAndValidateFile(file, fileInfo, metas);
|
2022-02-16 12:17:18 +01:00
|
|
|
return this.uploadFileAndPersist(fileData, { user });
|
2022-01-04 19:21:05 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
uploadedFiles = await Promise.all(
|
|
|
|
fileArray.map((file, idx) => doUpload(file, fileInfoArray[idx] || {}))
|
|
|
|
);
|
|
|
|
} finally {
|
|
|
|
// delete temporary folder
|
2022-02-16 12:17:18 +01:00
|
|
|
await fse.remove(tmpWorkingDirectory);
|
2022-01-04 19:21:05 +01:00
|
|
|
}
|
2019-02-04 13:17:59 -03:00
|
|
|
|
2022-01-04 19:21:05 +01:00
|
|
|
return uploadedFiles;
|
2020-03-05 13:51:15 +01:00
|
|
|
},
|
|
|
|
|
2022-07-30 10:14:37 +02:00
|
|
|
/**
|
2022-09-19 16:31:29 +02:00
|
|
|
* When uploading an image, an additional thumbnail is generated.
|
2022-08-01 12:14:13 +02:00
|
|
|
* Also, if there are responsive formats defined, another set of images will be generated too.
|
|
|
|
*
|
|
|
|
* @param {*} fileData
|
2022-07-30 10:14:37 +02:00
|
|
|
*/
|
2022-08-01 12:14:13 +02:00
|
|
|
async uploadImage(fileData) {
|
2022-08-08 23:33:39 +02:00
|
|
|
const { getDimensions, generateThumbnail, generateResponsiveFormats, isOptimizableImage } =
|
|
|
|
getService('image-manipulation');
|
2020-03-05 17:48:07 +01:00
|
|
|
|
2022-08-03 20:39:44 +02:00
|
|
|
// Store width and height of the original image
|
|
|
|
const { width, height } = await getDimensions(fileData);
|
|
|
|
|
2022-08-05 13:12:59 +02:00
|
|
|
// Make sure this is assigned before calling any upload
|
2022-08-03 20:39:44 +02:00
|
|
|
// That way it can mutate the width and height
|
|
|
|
_.assign(fileData, {
|
|
|
|
width,
|
|
|
|
height,
|
|
|
|
});
|
|
|
|
|
2022-09-20 15:23:45 +02:00
|
|
|
// For performance reasons, all uploads are wrapped in a single Promise.all
|
2022-08-11 16:59:02 +02:00
|
|
|
const uploadThumbnail = async (thumbnailFile) => {
|
2022-08-05 13:12:59 +02:00
|
|
|
await getService('provider').upload(thumbnailFile);
|
|
|
|
_.set(fileData, 'formats.thumbnail', thumbnailFile);
|
|
|
|
};
|
|
|
|
|
2022-08-11 16:59:02 +02:00
|
|
|
const uploadResponsiveFormat = async (format) => {
|
2022-08-05 13:12:59 +02:00
|
|
|
const { key, file } = format;
|
|
|
|
await getService('provider').upload(file);
|
|
|
|
_.set(fileData, ['formats', key], file);
|
|
|
|
};
|
|
|
|
|
2022-09-19 16:31:29 +02:00
|
|
|
const uploadPromises = [];
|
2022-08-03 20:39:44 +02:00
|
|
|
|
2022-09-20 15:23:45 +02:00
|
|
|
// Upload image
|
|
|
|
uploadPromises.push(getService('provider').upload(fileData));
|
|
|
|
|
2022-08-05 13:12:59 +02:00
|
|
|
// Generate & Upload thumbnail and responsive formats
|
2022-08-03 20:39:44 +02:00
|
|
|
if (await isOptimizableImage(fileData)) {
|
|
|
|
const thumbnailFile = await generateThumbnail(fileData);
|
|
|
|
if (thumbnailFile) {
|
2022-08-05 13:12:59 +02:00
|
|
|
uploadPromises.push(uploadThumbnail(thumbnailFile));
|
2022-08-03 20:39:44 +02:00
|
|
|
}
|
2020-03-05 17:48:07 +01:00
|
|
|
|
2022-08-03 20:39:44 +02:00
|
|
|
const formats = await generateResponsiveFormats(fileData);
|
|
|
|
if (Array.isArray(formats) && formats.length > 0) {
|
2022-09-19 16:31:29 +02:00
|
|
|
formats.forEach((format) => {
|
|
|
|
if (!format) return;
|
2022-08-05 13:12:59 +02:00
|
|
|
uploadPromises.push(uploadResponsiveFormat(format));
|
2022-09-19 16:31:29 +02:00
|
|
|
});
|
2020-03-20 18:32:29 +01:00
|
|
|
}
|
2022-02-08 15:24:32 +01:00
|
|
|
}
|
2022-08-05 13:12:59 +02:00
|
|
|
// Wait for all uploads to finish
|
|
|
|
await Promise.all(uploadPromises);
|
2022-08-01 12:14:13 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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');
|
|
|
|
|
2022-09-20 15:23:45 +02:00
|
|
|
await getService('provider').checkFileSize(fileData);
|
|
|
|
|
2022-08-01 12:14:13 +02:00
|
|
|
if (await isImage(fileData)) {
|
|
|
|
await this.uploadImage(fileData);
|
|
|
|
} else {
|
|
|
|
await getService('provider').upload(fileData);
|
|
|
|
}
|
2022-06-16 17:38:14 +01:00
|
|
|
|
2022-03-21 23:22:47 -04:00
|
|
|
_.set(fileData, 'provider', config.provider);
|
2020-03-05 13:51:15 +01:00
|
|
|
|
2022-07-30 10:14:37 +02:00
|
|
|
// Persist file(s)
|
2020-08-31 16:23:43 +02:00
|
|
|
return this.add(fileData, { user });
|
2020-03-24 12:21:51 +01:00
|
|
|
},
|
2020-03-05 13:51:15 +01:00
|
|
|
|
2022-03-30 16:26:09 +02:00
|
|
|
async updateFileInfo(id, { name, alternativeText, caption, folder }, { user } = {}) {
|
2021-09-24 15:40:02 +02:00
|
|
|
const dbFile = await this.findOne(id);
|
2020-03-24 12:21:51 +01:00
|
|
|
|
|
|
|
if (!dbFile) {
|
2021-10-27 18:54:58 +02:00
|
|
|
throw new NotFoundError();
|
2020-03-24 12:21:51 +01:00
|
|
|
}
|
|
|
|
|
2022-04-06 11:36:13 +02:00
|
|
|
const fileService = getService('file');
|
|
|
|
|
2022-03-30 16:26:09 +02:00
|
|
|
const newName = _.isNil(name) ? dbFile.name : name;
|
2020-03-27 16:00:26 +01:00
|
|
|
const newInfos = {
|
2022-03-30 16:26:09 +02:00
|
|
|
name: newName,
|
2020-03-24 12:21:51 +01:00
|
|
|
alternativeText: _.isNil(alternativeText) ? dbFile.alternativeText : alternativeText,
|
|
|
|
caption: _.isNil(caption) ? dbFile.caption : caption,
|
2022-03-30 16:26:09 +02:00
|
|
|
folder: _.isUndefined(folder) ? dbFile.folder : folder,
|
2022-04-12 16:32:05 +02:00
|
|
|
folderPath: _.isUndefined(folder) ? dbFile.path : await fileService.getFolderPath(folder),
|
2020-03-27 16:00:26 +01:00
|
|
|
};
|
2020-03-24 12:21:51 +01:00
|
|
|
|
2021-09-20 18:50:48 +02:00
|
|
|
return this.update(id, newInfos, { user });
|
2018-02-19 15:41:26 +01:00
|
|
|
},
|
|
|
|
|
2020-09-08 15:18:32 +02:00
|
|
|
async replace(id, { data, file }, { user } = {}) {
|
2021-08-19 22:27:00 +02:00
|
|
|
const config = strapi.config.get('plugin.upload');
|
2020-03-04 10:31:32 +01:00
|
|
|
|
2022-08-01 12:14:13 +02:00
|
|
|
const { isImage } = getService('image-manipulation');
|
2020-03-05 17:58:56 +01:00
|
|
|
|
2021-09-24 15:40:02 +02:00
|
|
|
const dbFile = await this.findOne(id);
|
2020-03-05 13:51:15 +01:00
|
|
|
if (!dbFile) {
|
2021-10-27 18:54:58 +02:00
|
|
|
throw new NotFoundError();
|
2020-03-05 13:51:15 +01:00
|
|
|
}
|
|
|
|
|
2022-02-16 12:17:18 +01:00
|
|
|
// create temporary folder to store files for stream manipulation
|
2022-03-01 14:52:31 +01:00
|
|
|
const tmpWorkingDirectory = await createAndAssignTmpWorkingDirectoryToFiles(file);
|
2022-02-28 17:44:18 +01:00
|
|
|
|
2022-02-16 12:17:18 +01:00
|
|
|
let fileData;
|
2020-03-04 10:31:32 +01:00
|
|
|
|
2022-02-16 12:17:18 +01:00
|
|
|
try {
|
|
|
|
const { fileInfo } = data;
|
2022-08-04 12:56:58 +02:00
|
|
|
fileData = await this.enhanceAndValidateFile(file, fileInfo);
|
2020-03-04 10:31:32 +01:00
|
|
|
|
2022-03-30 16:26:09 +02:00
|
|
|
// keep a constant hash and extension so the file url doesn't change when the file is replaced
|
2022-02-16 12:17:18 +01:00
|
|
|
_.assign(fileData, {
|
|
|
|
hash: dbFile.hash,
|
|
|
|
ext: dbFile.ext,
|
|
|
|
});
|
2020-03-26 15:52:56 +01:00
|
|
|
|
2022-02-16 12:17:18 +01:00
|
|
|
// execute delete function of the provider
|
|
|
|
if (dbFile.provider === config.provider) {
|
|
|
|
await strapi.plugin('upload').provider.delete(dbFile);
|
2022-02-08 15:24:32 +01:00
|
|
|
|
2022-02-16 12:17:18 +01:00
|
|
|
if (dbFile.formats) {
|
|
|
|
await Promise.all(
|
2022-08-08 23:33:39 +02:00
|
|
|
Object.keys(dbFile.formats).map((key) => {
|
2022-02-16 12:17:18 +01:00
|
|
|
return strapi.plugin('upload').provider.delete(dbFile.formats[key]);
|
|
|
|
})
|
|
|
|
);
|
|
|
|
}
|
2022-02-08 15:24:32 +01:00
|
|
|
}
|
2020-03-05 17:58:56 +01:00
|
|
|
|
2022-02-16 12:17:18 +01:00
|
|
|
// clear old formats
|
|
|
|
_.set(fileData, 'formats', {});
|
2020-03-08 20:03:45 +01:00
|
|
|
|
2022-06-16 17:38:14 +01:00
|
|
|
if (await isImage(fileData)) {
|
2022-08-01 12:14:13 +02:00
|
|
|
await this.uploadImage(fileData);
|
|
|
|
} else {
|
|
|
|
await getService('provider').upload(fileData);
|
2022-02-16 12:17:18 +01:00
|
|
|
}
|
2022-08-01 12:14:13 +02:00
|
|
|
|
2022-03-21 23:22:47 -04:00
|
|
|
_.set(fileData, 'provider', config.provider);
|
2022-02-16 12:17:18 +01:00
|
|
|
} finally {
|
|
|
|
// delete temporary folder
|
|
|
|
await fse.remove(tmpWorkingDirectory);
|
2022-02-08 15:24:32 +01:00
|
|
|
}
|
2020-03-04 10:31:32 +01:00
|
|
|
|
2021-09-20 18:50:48 +02:00
|
|
|
return this.update(id, fileData, { user });
|
2020-03-04 10:31:32 +01:00
|
|
|
},
|
|
|
|
|
2021-09-20 18:50:48 +02:00
|
|
|
async update(id, values, { user } = {}) {
|
2020-08-31 16:23:43 +02:00
|
|
|
const fileValues = { ...values };
|
|
|
|
if (user) {
|
|
|
|
fileValues[UPDATED_BY_ATTRIBUTE] = user.id;
|
|
|
|
}
|
|
|
|
sendMediaMetrics(fileValues);
|
2020-04-27 16:08:55 +02:00
|
|
|
|
2022-05-13 16:10:18 +02:00
|
|
|
const res = await strapi.entityService.update(FILE_MODEL_UID, id, { data: fileValues });
|
2021-07-08 18:15:32 +02:00
|
|
|
|
2021-11-05 10:36:10 +01:00
|
|
|
await this.emitEvent(MEDIA_UPDATE, res);
|
2021-07-08 18:15:32 +02:00
|
|
|
|
2020-03-24 12:21:51 +01:00
|
|
|
return res;
|
2020-03-04 10:31:32 +01:00
|
|
|
},
|
|
|
|
|
2020-09-08 15:18:32 +02:00
|
|
|
async add(values, { user } = {}) {
|
2020-08-31 16:23:43 +02:00
|
|
|
const fileValues = { ...values };
|
|
|
|
if (user) {
|
|
|
|
fileValues[UPDATED_BY_ATTRIBUTE] = user.id;
|
|
|
|
fileValues[CREATED_BY_ATTRIBUTE] = user.id;
|
|
|
|
}
|
|
|
|
sendMediaMetrics(fileValues);
|
2020-04-27 16:08:55 +02:00
|
|
|
|
2022-05-13 16:10:18 +02:00
|
|
|
const res = await strapi.query(FILE_MODEL_UID).create({ data: fileValues });
|
2021-07-08 18:15:32 +02:00
|
|
|
|
2021-11-05 10:36:10 +01:00
|
|
|
await this.emitEvent(MEDIA_CREATE, res);
|
2021-07-08 18:15:32 +02:00
|
|
|
|
2020-03-24 12:21:51 +01:00
|
|
|
return res;
|
2018-02-19 15:41:26 +01:00
|
|
|
},
|
|
|
|
|
2021-09-24 15:40:02 +02:00
|
|
|
findOne(id, populate) {
|
2022-05-13 16:10:18 +02:00
|
|
|
return strapi.entityService.findOne(FILE_MODEL_UID, id, { populate });
|
2018-02-19 15:41:26 +01:00
|
|
|
},
|
|
|
|
|
2021-10-07 17:23:42 +02:00
|
|
|
findMany(query) {
|
2022-05-13 16:10:18 +02:00
|
|
|
return strapi.entityService.findMany(FILE_MODEL_UID, query);
|
2020-03-25 19:17:46 +01:00
|
|
|
},
|
|
|
|
|
2021-10-07 17:23:42 +02:00
|
|
|
findPage(query) {
|
2022-05-13 16:10:18 +02:00
|
|
|
return strapi.entityService.findPage(FILE_MODEL_UID, query);
|
2021-10-07 17:23:42 +02:00
|
|
|
},
|
|
|
|
|
2020-02-26 19:38:23 +01:00
|
|
|
async remove(file) {
|
2021-08-19 22:27:00 +02:00
|
|
|
const config = strapi.config.get('plugin.upload');
|
2020-02-26 19:38:23 +01:00
|
|
|
|
2018-02-21 17:18:33 +01:00
|
|
|
// execute delete function of the provider
|
2020-02-26 19:38:23 +01:00
|
|
|
if (file.provider === config.provider) {
|
2021-08-19 22:27:00 +02:00
|
|
|
await strapi.plugin('upload').provider.delete(file);
|
2020-03-05 17:48:07 +01:00
|
|
|
|
2020-03-08 20:03:45 +01:00
|
|
|
if (file.formats) {
|
|
|
|
await Promise.all(
|
2022-08-08 23:33:39 +02:00
|
|
|
Object.keys(file.formats).map((key) => {
|
2021-08-19 22:27:00 +02:00
|
|
|
return strapi.plugin('upload').provider.delete(file.formats[key]);
|
2020-03-08 20:03:45 +01:00
|
|
|
})
|
|
|
|
);
|
2020-03-05 17:48:07 +01:00
|
|
|
}
|
2018-03-07 14:18:15 +01:00
|
|
|
}
|
2018-02-19 16:00:37 +01:00
|
|
|
|
2022-05-13 16:10:18 +02:00
|
|
|
const media = await strapi.query(FILE_MODEL_UID).findOne({
|
2021-07-08 21:53:30 +02:00
|
|
|
where: { id: file.id },
|
2019-12-17 20:59:57 +01:00
|
|
|
});
|
|
|
|
|
2021-11-05 10:36:10 +01:00
|
|
|
await this.emitEvent(MEDIA_DELETE, media);
|
2019-12-17 20:59:57 +01:00
|
|
|
|
2022-05-13 16:10:18 +02:00
|
|
|
return strapi.query(FILE_MODEL_UID).delete({ where: { id: file.id } });
|
2018-02-28 12:33:32 +01:00
|
|
|
},
|
|
|
|
|
2021-07-28 21:03:32 +02:00
|
|
|
async uploadToEntity(params, files) {
|
2020-03-03 15:54:59 +01:00
|
|
|
const { id, model, field } = params;
|
|
|
|
|
2022-02-28 17:44:18 +01:00
|
|
|
// create temporary folder to store files for stream manipulation
|
2022-03-01 14:52:31 +01:00
|
|
|
const tmpWorkingDirectory = await createAndAssignTmpWorkingDirectoryToFiles(files);
|
2022-02-28 17:44:18 +01:00
|
|
|
|
2020-03-03 15:54:59 +01:00
|
|
|
const arr = Array.isArray(files) ? files : [files];
|
2020-03-16 17:41:16 +01:00
|
|
|
|
2022-05-03 20:33:09 +02:00
|
|
|
const apiUploadFolderService = getService('api-upload-folder');
|
|
|
|
|
|
|
|
const apiUploadFolder = await apiUploadFolderService.getAPIUploadFolder();
|
|
|
|
|
2022-02-28 17:44:18 +01:00
|
|
|
try {
|
|
|
|
const enhancedFiles = await Promise.all(
|
2022-08-08 23:33:39 +02:00
|
|
|
arr.map((file) => {
|
2022-08-04 12:56:58 +02:00
|
|
|
return this.enhanceAndValidateFile(
|
2022-02-28 17:44:18 +01:00
|
|
|
file,
|
2022-05-03 20:33:09 +02:00
|
|
|
{ folder: apiUploadFolder.id },
|
2022-02-28 17:44:18 +01:00
|
|
|
{
|
|
|
|
refId: id,
|
|
|
|
ref: model,
|
|
|
|
field,
|
|
|
|
}
|
|
|
|
);
|
|
|
|
})
|
|
|
|
);
|
|
|
|
|
2022-08-08 23:33:39 +02:00
|
|
|
await Promise.all(enhancedFiles.map((file) => this.uploadFileAndPersist(file)));
|
2022-02-28 17:44:18 +01:00
|
|
|
} finally {
|
|
|
|
// delete temporary folder
|
|
|
|
await fse.remove(tmpWorkingDirectory);
|
|
|
|
}
|
2018-09-10 16:05:00 +08:00
|
|
|
},
|
2020-03-03 15:54:59 +01:00
|
|
|
|
2020-03-08 18:43:50 +01:00
|
|
|
getSettings() {
|
2021-09-13 12:03:12 +02:00
|
|
|
return strapi.store({ type: 'plugin', name: 'upload', key: 'settings' }).get();
|
2020-03-08 12:59:21 +01:00
|
|
|
},
|
|
|
|
|
2020-03-08 18:43:50 +01:00
|
|
|
setSettings(value) {
|
2020-04-27 16:08:55 +02:00
|
|
|
if (value.responsiveDimensions === true) {
|
|
|
|
strapi.telemetry.send('didEnableResponsiveDimensions');
|
|
|
|
} else {
|
|
|
|
strapi.telemetry.send('didDisableResponsiveDimensions');
|
|
|
|
}
|
|
|
|
|
2021-09-13 12:03:12 +02:00
|
|
|
return strapi.store({ type: 'plugin', name: 'upload', key: 'settings' }).set({ value });
|
2020-02-24 17:39:15 +01:00
|
|
|
},
|
2021-07-08 11:20:13 +02:00
|
|
|
});
|