mirror of
https://github.com/plankanban/planka.git
synced 2025-12-17 17:23:25 +03:00
143 lines
3.5 KiB
JavaScript
143 lines
3.5 KiB
JavaScript
/*!
|
|
* Copyright (c) 2024 PLANKA Software GmbH
|
|
* Licensed under the Fair Use License: https://github.com/plankanban/planka/blob/master/LICENSE.md
|
|
*/
|
|
|
|
const fsPromises = require('fs').promises;
|
|
const { rimraf } = require('rimraf');
|
|
const { getEncoding } = require('istextorbinary');
|
|
const mime = require('mime');
|
|
const sharp = require('sharp');
|
|
|
|
const filenamify = require('../../../utils/filenamify');
|
|
const { MAX_SIZE_TO_GET_ENCODING } = require('../../../constants');
|
|
|
|
module.exports = {
|
|
inputs: {
|
|
file: {
|
|
type: 'json',
|
|
required: true,
|
|
},
|
|
},
|
|
|
|
async fn(inputs) {
|
|
const fileManager = sails.hooks['file-manager'].getInstance();
|
|
|
|
const filename = filenamify(inputs.file.filename);
|
|
const mimeType = mime.getType(filename);
|
|
const { size } = inputs.file;
|
|
|
|
const { id: uploadedFileId } = await UploadedFile.qm.createOne({
|
|
mimeType,
|
|
size,
|
|
type: UploadedFile.Types.ATTACHMENT,
|
|
});
|
|
|
|
const dirPathSegment = `${sails.config.custom.attachmentsPathSegment}/${uploadedFileId}`;
|
|
|
|
let buffer;
|
|
let encoding = null;
|
|
|
|
if (size <= MAX_SIZE_TO_GET_ENCODING) {
|
|
try {
|
|
buffer = await fsPromises.readFile(inputs.file.fd);
|
|
} catch (error) {
|
|
/* empty */
|
|
}
|
|
|
|
if (buffer) {
|
|
encoding = getEncoding(buffer);
|
|
}
|
|
}
|
|
|
|
const filePath = await fileManager.move(
|
|
inputs.file.fd,
|
|
`${dirPathSegment}/${filename}`,
|
|
inputs.file.type,
|
|
);
|
|
|
|
const data = {
|
|
uploadedFileId,
|
|
filename,
|
|
mimeType,
|
|
size,
|
|
encoding,
|
|
image: null,
|
|
};
|
|
|
|
if (!['image/svg+xml', 'application/pdf'].includes(mimeType)) {
|
|
let image = sharp(buffer || filePath, {
|
|
animated: true,
|
|
});
|
|
|
|
let metadata;
|
|
try {
|
|
metadata = await image.metadata();
|
|
} catch (error) {
|
|
/* empty */
|
|
}
|
|
|
|
if (metadata) {
|
|
let { width, pageHeight: height = metadata.height } = metadata;
|
|
if (metadata.orientation && metadata.orientation > 4) {
|
|
[image, width, height] = [image.rotate(), height, width];
|
|
}
|
|
|
|
const thumbnailsPathSegment = `${dirPathSegment}/thumbnails`;
|
|
const thumbnailsExtension = metadata.format === 'jpeg' ? 'jpg' : metadata.format;
|
|
|
|
try {
|
|
const outside360Buffer = await image
|
|
.resize(360, 360, {
|
|
fit: 'outside',
|
|
withoutEnlargement: true,
|
|
})
|
|
.png({
|
|
quality: 75,
|
|
force: false,
|
|
})
|
|
.toBuffer();
|
|
|
|
await fileManager.save(
|
|
`${thumbnailsPathSegment}/outside-360.${thumbnailsExtension}`,
|
|
outside360Buffer,
|
|
inputs.file.type,
|
|
);
|
|
|
|
const outside720Buffer = await image
|
|
.resize(720, 720, {
|
|
fit: 'outside',
|
|
withoutEnlargement: true,
|
|
})
|
|
.png({
|
|
quality: 75,
|
|
force: false,
|
|
})
|
|
.toBuffer();
|
|
|
|
await fileManager.save(
|
|
`${thumbnailsPathSegment}/outside-720.${thumbnailsExtension}`,
|
|
outside720Buffer,
|
|
inputs.file.type,
|
|
);
|
|
|
|
data.image = {
|
|
width,
|
|
height,
|
|
thumbnailsExtension,
|
|
};
|
|
} catch (error) {
|
|
sails.log.warn(error.stack);
|
|
await fileManager.deleteDir(thumbnailsPathSegment);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!filePath) {
|
|
await rimraf(inputs.file.fd);
|
|
}
|
|
|
|
return data;
|
|
},
|
|
};
|