mirror of
https://github.com/strapi/strapi.git
synced 2025-07-19 15:06:11 +00:00
170 lines
4.3 KiB
JavaScript
170 lines
4.3 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 crypto = require('crypto');
|
|
const _ = require('lodash');
|
|
const toArray = require('stream-to-array');
|
|
const uuid = require('uuid/v4');
|
|
|
|
function niceHash(buffer) {
|
|
return crypto
|
|
.createHash('sha256')
|
|
.update(buffer)
|
|
.digest('base64')
|
|
.replace(/=/g, '')
|
|
.replace(/\//g, '-')
|
|
.replace(/\+/, '_');
|
|
}
|
|
|
|
module.exports = {
|
|
bufferize: async files => {
|
|
if (_.isEmpty(files) === 0) {
|
|
throw 'Missing files.';
|
|
}
|
|
|
|
// files is always an array to map on
|
|
files = _.isArray(files) ? files : [files];
|
|
|
|
const createBuffer = async stream => {
|
|
const parts = await toArray(fs.createReadStream(stream.path));
|
|
const buffers = parts.map(part =>
|
|
_.isBuffer(part) ? part : Buffer.from(part)
|
|
);
|
|
|
|
const buffer = Buffer.concat(buffers);
|
|
|
|
return {
|
|
tmpPath: stream.path,
|
|
name: stream.name,
|
|
sha256: niceHash(buffer),
|
|
hash: uuid().replace(/-/g, ''),
|
|
ext:
|
|
stream.name.split('.').length > 1
|
|
? `.${_.last(stream.name.split('.'))}`
|
|
: '',
|
|
buffer,
|
|
mime: stream.type,
|
|
size: (stream.size / 1000).toFixed(2),
|
|
};
|
|
};
|
|
|
|
// transform all files in buffer
|
|
return Promise.all(files.map(stream => createBuffer(stream)));
|
|
},
|
|
|
|
upload(files, config) {
|
|
// Get upload provider settings to configure the provider to use.
|
|
const provider = _.find(strapi.plugins.upload.config.providers, {
|
|
provider: config.provider,
|
|
});
|
|
|
|
if (!provider) {
|
|
throw new Error(
|
|
`The provider package isn't installed. Please run \`npm install strapi-provider-upload-${config.provider}\``
|
|
);
|
|
}
|
|
|
|
const actions = provider.init(config);
|
|
|
|
// upload a single file
|
|
const uploadFile = async file => {
|
|
await actions.upload(file);
|
|
|
|
// Remove buffer to don't save it.
|
|
delete file.buffer;
|
|
file.provider = provider.provider;
|
|
|
|
const res = await strapi.plugins['upload'].services.upload.add(file);
|
|
|
|
// Remove temp file
|
|
fs.unlinkSync(file.tmpPath);
|
|
return res;
|
|
};
|
|
|
|
// Execute upload function of the provider for all files.
|
|
return Promise.all(files.map(file => uploadFile(file)));
|
|
},
|
|
|
|
add(values) {
|
|
return strapi.query('file', 'upload').create(values);
|
|
},
|
|
|
|
fetch(params) {
|
|
return strapi.query('file', 'upload').findOne({
|
|
id: params.id,
|
|
});
|
|
},
|
|
|
|
fetchAll(params) {
|
|
return strapi.query('file', 'upload').find(params);
|
|
},
|
|
|
|
count(params) {
|
|
return strapi.query('file', 'upload').count(params);
|
|
},
|
|
|
|
async remove(file, config) {
|
|
// get upload provider settings to configure the provider to use
|
|
const provider = _.cloneDeep(
|
|
_.find(strapi.plugins.upload.config.providers, {
|
|
provider: config.provider,
|
|
})
|
|
);
|
|
_.assign(provider, config);
|
|
const actions = provider.init(config);
|
|
|
|
// execute delete function of the provider
|
|
if (file.provider === provider.provider) {
|
|
await actions.delete(file);
|
|
}
|
|
|
|
return strapi.query('file', 'upload').delete({ id: file.id });
|
|
},
|
|
|
|
async uploadToEntity(params, files, source) {
|
|
// Retrieve provider settings from database.
|
|
const config = await strapi
|
|
.store({
|
|
environment: strapi.config.environment,
|
|
type: 'plugin',
|
|
name: 'upload',
|
|
})
|
|
.get({ key: 'provider' });
|
|
|
|
const model = strapi.getModel(params.model, source);
|
|
|
|
// Asynchronous upload.
|
|
return await Promise.all(
|
|
Object.keys(files).map(async attribute => {
|
|
// Bufferize files per attribute.
|
|
const buffers = await this.bufferize(files[attribute]);
|
|
const enhancedFiles = buffers.map(file => {
|
|
const details = model.attributes[attribute];
|
|
|
|
// Add related information to be able to make
|
|
// the relationships later.
|
|
file[details.via] = [
|
|
{
|
|
refId: params.id,
|
|
ref: params.model,
|
|
source,
|
|
field: attribute,
|
|
},
|
|
];
|
|
|
|
return file;
|
|
});
|
|
|
|
// Make upload async.
|
|
return this.upload(enhancedFiles, config);
|
|
})
|
|
);
|
|
},
|
|
};
|