mirror of
https://github.com/strapi/strapi.git
synced 2025-07-21 16:10:18 +00:00
403 lines
10 KiB
JavaScript
403 lines
10 KiB
JavaScript
'use strict';
|
|
|
|
/**
|
|
* Upload.js service
|
|
*
|
|
* @description: A set of functions similar to controller's actions to avoid code duplication.
|
|
*/
|
|
|
|
const fs = require('fs');
|
|
const path = require('path');
|
|
const crypto = require('crypto');
|
|
const util = require('util');
|
|
const _ = require('lodash');
|
|
const {
|
|
nameToSlug,
|
|
contentTypes: contentTypesUtils,
|
|
sanitizeEntity,
|
|
webhook: webhookUtils,
|
|
} = require('@strapi/utils');
|
|
|
|
const { MEDIA_UPDATE, MEDIA_CREATE, MEDIA_DELETE } = webhookUtils.webhookEvents;
|
|
|
|
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 combineFilters = params => {
|
|
// FIXME: until we support boolean operators for querying we need to make mime_ncontains use AND instead of OR
|
|
if (_.has(params, 'mime_ncontains') && Array.isArray(params.mime_ncontains)) {
|
|
params._where = params.mime_ncontains.map(val => ({ mime_ncontains: val }));
|
|
delete params.mime_ncontains;
|
|
}
|
|
};
|
|
|
|
module.exports = ({ strapi }) => ({
|
|
emitEvent(event, data) {
|
|
const modelDef = strapi.getModel('file', 'upload');
|
|
strapi.eventHub.emit(event, { media: sanitizeEntity(data, { model: modelDef }) });
|
|
},
|
|
|
|
formatFileInfo({ filename, type, size }, fileInfo = {}, metas = {}) {
|
|
const ext = path.extname(filename);
|
|
const basename = path.basename(fileInfo.name || filename, ext);
|
|
|
|
const usedName = fileInfo.name || filename;
|
|
|
|
const entity = {
|
|
name: usedName,
|
|
alternativeText: fileInfo.alternativeText,
|
|
caption: fileInfo.caption,
|
|
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;
|
|
}
|
|
|
|
return entity;
|
|
},
|
|
|
|
async enhanceFile(file, fileInfo = {}, metas = {}) {
|
|
let readBuffer;
|
|
try {
|
|
readBuffer = await util.promisify(fs.readFile)(file.path);
|
|
} catch (e) {
|
|
if (e.code === 'ERR_FS_FILE_TOO_LARGE') {
|
|
throw strapi.errors.entityTooLarge('FileTooBig', {
|
|
errors: [
|
|
{
|
|
id: 'Upload.status.sizeLimit',
|
|
message: `${file.name} file is bigger than the limit size!`,
|
|
values: { file: file.name },
|
|
},
|
|
],
|
|
});
|
|
}
|
|
throw e;
|
|
}
|
|
|
|
const { optimize } = strapi.plugins.upload.services['image-manipulation'];
|
|
|
|
const { buffer, info } = await optimize(readBuffer);
|
|
|
|
const formattedFile = this.formatFileInfo(
|
|
{
|
|
filename: file.name,
|
|
type: file.type,
|
|
size: file.size,
|
|
},
|
|
fileInfo,
|
|
metas
|
|
);
|
|
|
|
return _.assign(formattedFile, info, {
|
|
buffer,
|
|
});
|
|
},
|
|
|
|
async upload({ data, files }, { user } = {}) {
|
|
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.enhanceFile(file, fileInfo, metas);
|
|
|
|
return this.uploadFileAndPersist(fileData, { user });
|
|
};
|
|
|
|
return await Promise.all(
|
|
fileArray.map((file, idx) => doUpload(file, fileInfoArray[idx] || {}))
|
|
);
|
|
},
|
|
|
|
async uploadFileAndPersist(fileData, { user } = {}) {
|
|
const config = strapi.config.get('plugin.upload');
|
|
|
|
const {
|
|
getDimensions,
|
|
generateThumbnail,
|
|
generateResponsiveFormats,
|
|
} = strapi.plugins.upload.services['image-manipulation'];
|
|
|
|
await strapi.plugins.upload.provider.upload(fileData);
|
|
|
|
const thumbnailFile = await generateThumbnail(fileData);
|
|
if (thumbnailFile) {
|
|
await strapi.plugins.upload.provider.upload(thumbnailFile);
|
|
delete thumbnailFile.buffer;
|
|
_.set(fileData, 'formats.thumbnail', thumbnailFile);
|
|
}
|
|
|
|
const formats = await generateResponsiveFormats(fileData);
|
|
if (Array.isArray(formats) && formats.length > 0) {
|
|
for (const format of formats) {
|
|
if (!format) continue;
|
|
|
|
const { key, file } = format;
|
|
|
|
await strapi.plugins.upload.provider.upload(file);
|
|
delete file.buffer;
|
|
|
|
_.set(fileData, ['formats', key], file);
|
|
}
|
|
}
|
|
|
|
const { width, height } = await getDimensions(fileData.buffer);
|
|
|
|
delete fileData.buffer;
|
|
|
|
_.assign(fileData, {
|
|
provider: config.provider,
|
|
width,
|
|
height,
|
|
});
|
|
|
|
return this.add(fileData, { user });
|
|
},
|
|
|
|
async updateFileInfo(id, { name, alternativeText, caption }, { user } = {}) {
|
|
const dbFile = await this.findOne({ id });
|
|
|
|
if (!dbFile) {
|
|
throw strapi.errors.notFound('file not found');
|
|
}
|
|
|
|
const newInfos = {
|
|
name: _.isNil(name) ? dbFile.name : name,
|
|
alternativeText: _.isNil(alternativeText) ? dbFile.alternativeText : alternativeText,
|
|
caption: _.isNil(caption) ? dbFile.caption : caption,
|
|
};
|
|
|
|
return this.update({ id }, newInfos, { user });
|
|
},
|
|
|
|
async replace(id, { data, file }, { user } = {}) {
|
|
const config = strapi.plugins.upload.config;
|
|
|
|
const {
|
|
getDimensions,
|
|
generateThumbnail,
|
|
generateResponsiveFormats,
|
|
} = strapi.plugins.upload.services['image-manipulation'];
|
|
|
|
const dbFile = await this.findOne({ id });
|
|
|
|
if (!dbFile) {
|
|
throw strapi.errors.notFound('file not found');
|
|
}
|
|
|
|
const { fileInfo } = data;
|
|
const fileData = await this.enhanceFile(file, fileInfo);
|
|
|
|
// keep a constant hash
|
|
_.assign(fileData, {
|
|
hash: dbFile.hash,
|
|
ext: dbFile.ext,
|
|
});
|
|
|
|
// execute delete function of the provider
|
|
if (dbFile.provider === config.provider) {
|
|
await strapi.plugins.upload.provider.delete(dbFile);
|
|
|
|
if (dbFile.formats) {
|
|
await Promise.all(
|
|
Object.keys(dbFile.formats).map(key => {
|
|
return strapi.plugins.upload.provider.delete(dbFile.formats[key]);
|
|
})
|
|
);
|
|
}
|
|
}
|
|
|
|
await strapi.plugins.upload.provider.upload(fileData);
|
|
|
|
// clear old formats
|
|
_.set(fileData, 'formats', {});
|
|
|
|
const thumbnailFile = await generateThumbnail(fileData);
|
|
if (thumbnailFile) {
|
|
await strapi.plugins.upload.provider.upload(thumbnailFile);
|
|
delete thumbnailFile.buffer;
|
|
_.set(fileData, 'formats.thumbnail', thumbnailFile);
|
|
}
|
|
|
|
const formats = await generateResponsiveFormats(fileData);
|
|
if (Array.isArray(formats) && formats.length > 0) {
|
|
for (const format of formats) {
|
|
if (!format) continue;
|
|
|
|
const { key, file } = format;
|
|
|
|
await strapi.plugins.upload.provider.upload(file);
|
|
delete file.buffer;
|
|
|
|
_.set(fileData, ['formats', key], file);
|
|
}
|
|
}
|
|
|
|
const { width, height } = await getDimensions(fileData.buffer);
|
|
delete fileData.buffer;
|
|
|
|
_.assign(fileData, {
|
|
provider: config.provider,
|
|
width,
|
|
height,
|
|
});
|
|
|
|
return this.update({ id }, fileData, { user });
|
|
},
|
|
|
|
async update(params, values, { user } = {}) {
|
|
const fileValues = { ...values };
|
|
if (user) {
|
|
fileValues[UPDATED_BY_ATTRIBUTE] = user.id;
|
|
}
|
|
sendMediaMetrics(fileValues);
|
|
|
|
//
|
|
const res = await strapi
|
|
.query('plugin::upload.file')
|
|
.update({ where: params, data: fileValues });
|
|
|
|
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('plugin::upload.file').create({ data: fileValues });
|
|
|
|
this.emitEvent(MEDIA_CREATE, res);
|
|
|
|
return res;
|
|
},
|
|
|
|
findOne(params, populate) {
|
|
return strapi.query('plugin::upload.file').findOne({ where: params, populate });
|
|
},
|
|
|
|
fetchAll(params) {
|
|
combineFilters(params);
|
|
return strapi.query('plugin::upload.file').findMany({ ...params });
|
|
},
|
|
|
|
count(params) {
|
|
combineFilters(params);
|
|
return strapi.query('plugin::upload.file').count({ ...params });
|
|
},
|
|
|
|
async remove(file) {
|
|
const config = strapi.plugins.upload.config;
|
|
|
|
// execute delete function of the provider
|
|
if (file.provider === config.provider) {
|
|
await strapi.plugins.upload.provider.delete(file);
|
|
|
|
if (file.formats) {
|
|
await Promise.all(
|
|
Object.keys(file.formats).map(key => {
|
|
return strapi.plugins.upload.provider.delete(file.formats[key]);
|
|
})
|
|
);
|
|
}
|
|
}
|
|
|
|
const media = await strapi.query('plugin::upload.file').findOne({
|
|
where: { id: file.id },
|
|
});
|
|
|
|
this.emitEvent(MEDIA_DELETE, media);
|
|
|
|
return strapi.query('plugin::upload.file').delete({ where: { id: file.id } });
|
|
},
|
|
|
|
async uploadToEntity(params, files) {
|
|
const { id, model, field } = params;
|
|
|
|
const arr = Array.isArray(files) ? files : [files];
|
|
const enhancedFiles = await Promise.all(
|
|
arr.map(file => {
|
|
return this.enhanceFile(
|
|
file,
|
|
{},
|
|
{
|
|
refId: id,
|
|
ref: model,
|
|
field,
|
|
}
|
|
);
|
|
})
|
|
);
|
|
|
|
await Promise.all(enhancedFiles.map(file => this.uploadFileAndPersist(file)));
|
|
},
|
|
|
|
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 });
|
|
},
|
|
});
|