406 lines
10 KiB
JavaScript
Raw Normal View History

'use strict';
/**
* Upload.js service
*
* @description: A set of functions similar to controller's actions to avoid code duplication.
*/
2018-02-19 14:26:20 +01:00
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,
2021-04-29 13:51:12 +02:00
} = 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;
}
};
2021-07-08 11:20:13 +02:00
module.exports = ({ strapi }) => ({
2021-07-08 18:15:32 +02:00
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),
};
2021-07-28 21:03:32 +02:00
const { refId, ref, field } = metas;
if (refId && ref && field) {
entity.related = [
{
2021-07-28 21:03:32 +02:00
__type: ref,
id: refId,
// refId,
// ref,
// source,
// field,
},
];
2018-02-19 14:26:20 +01:00
}
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,
});
2018-02-19 14:26:20 +01:00
},
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);
2018-03-07 14:18:15 +01:00
return this.uploadFileAndPersist(fileData, { user });
};
return await Promise.all(
fileArray.map((file, idx) => doUpload(file, fileInfoArray[idx] || {}))
);
},
async uploadFileAndPersist(fileData, { user } = {}) {
const config = strapi.plugins.upload.config;
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 } = {}) {
2021-07-08 22:07:52 +02:00
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'];
2021-07-08 22:07:52 +02:00
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);
2021-07-08 18:15:32 +02:00
//
const res = await strapi
2021-08-06 18:09:49 +02:00
.query('plugin::upload.file')
2021-07-08 18:15:32 +02:00
.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);
2021-08-06 18:09:49 +02:00
const res = await strapi.query('plugin::upload.file').create({ data: fileValues });
2021-07-08 18:15:32 +02:00
this.emitEvent(MEDIA_CREATE, res);
return res;
},
2021-07-08 22:07:52 +02:00
findOne(params, populate) {
2021-08-06 18:09:49 +02:00
return strapi.query('plugin::upload.file').findOne({ where: params, populate });
},
2021-07-28 21:03:32 +02:00
fetchAll(params) {
combineFilters(params);
2021-08-06 18:09:49 +02:00
return strapi.query('plugin::upload.file').findMany({ ...params });
},
count(params) {
combineFilters(params);
2021-08-06 18:09:49 +02:00
return strapi.query('plugin::upload.file').count({ ...params });
2018-02-19 19:54:45 +01:00
},
async remove(file) {
const config = strapi.plugins.upload.config;
2018-02-21 17:18:33 +01:00
// 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]);
})
);
}
2018-03-07 14:18:15 +01:00
}
2018-02-19 16:00:37 +01:00
2021-08-06 18:09:49 +02:00
const media = await strapi.query('plugin::upload.file').findOne({
2021-07-08 21:53:30 +02:00
where: { id: file.id },
2019-12-17 20:59:57 +01:00
});
2021-07-08 18:15:32 +02:00
this.emitEvent(MEDIA_DELETE, media);
2019-12-17 20:59:57 +01:00
2021-08-06 18:09:49 +02:00
return strapi.query('plugin::upload.file').delete({ where: { id: file.id } });
},
2021-07-28 21:03:32 +02:00
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 });
},
2021-07-08 11:20:13 +02:00
});