Notes/src/services/image.js

171 lines
5.1 KiB
JavaScript
Raw Normal View History

2018-02-11 00:18:59 -05:00
"use strict";
2021-06-29 22:15:57 +02:00
const becca = require('../becca/becca');
const log = require('./log');
const protectedSessionService = require('./protected_session');
2018-11-08 11:08:16 +01:00
const noteService = require('./notes');
const optionService = require('./options');
const sql = require('./sql');
2018-02-11 00:18:59 -05:00
const jimp = require('jimp');
const imageType = require('image-type');
const sanitizeFilename = require('sanitize-filename');
2021-06-29 22:15:57 +02:00
const noteRevisionService = require('./note_revisions');
2020-03-25 11:28:44 +01:00
const isSvg = require('is-svg');
const isAnimated = require('is-animated');
2018-02-11 00:18:59 -05:00
async function processImage(uploadBuffer, originalName, shrinkImageSwitch) {
const compressImages = optionService.getOptionBool("compressImages");
2020-03-25 11:28:44 +01:00
const origImageFormat = getImageType(uploadBuffer);
2019-07-10 20:38:27 +02:00
if (origImageFormat && ["webp", "svg", "gif"].includes(origImageFormat.ext)) {
2019-07-10 20:38:27 +02:00
// JIMP does not support webp at the moment: https://github.com/oliver-moran/jimp/issues/144
shrinkImageSwitch = false;
}
else if (isAnimated(uploadBuffer)) {
// recompression of animated images will make them static
shrinkImageSwitch = false;
}
2019-07-10 20:38:27 +02:00
const finalImageBuffer = (compressImages && shrinkImageSwitch) ? await shrinkImage(uploadBuffer, originalName) : uploadBuffer;
2018-02-11 00:18:59 -05:00
2020-03-25 11:28:44 +01:00
const imageFormat = getImageType(finalImageBuffer);
2018-02-11 00:18:59 -05:00
2019-11-08 22:34:30 +01:00
return {
buffer: finalImageBuffer,
imageFormat
};
}
2020-03-25 11:28:44 +01:00
function getImageType(buffer) {
if (isSvg(buffer)) {
return {
ext: 'svg'
}
}
else {
2020-11-17 22:35:20 +01:00
return imageType(buffer) || "jpg"; // optimistic JPG default
2020-03-25 11:28:44 +01:00
}
}
2020-03-25 18:21:55 +01:00
function getImageMimeFromExtension(ext) {
ext = ext.toLowerCase();
return 'image/' + (ext === 'svg' ? 'svg+xml' : ext);
}
2020-06-20 12:31:38 +02:00
function updateImage(noteId, uploadBuffer, originalName) {
2020-04-08 11:07:38 +02:00
log.info(`Updating image ${noteId}: ${originalName}`);
2021-05-02 11:23:58 +02:00
const note = becca.getNote(noteId);
2019-11-08 22:34:30 +01:00
2020-06-20 12:31:38 +02:00
noteRevisionService.createNoteRevision(note);
noteRevisionService.protectNoteRevisions(note);
2019-11-08 22:34:30 +01:00
2020-06-20 12:31:38 +02:00
note.setLabel('originalFileName', originalName);
2019-11-08 23:09:57 +01:00
// resizing images asynchronously since JIMP does not support sync operation
processImage(uploadBuffer, originalName, true).then(({buffer, imageFormat}) => {
sql.transactional(() => {
note.mime = getImageMimeFromExtension(imageFormat.ext);
note.save();
note.setContent(buffer);
})
});
2019-11-08 22:34:30 +01:00
}
function saveImage(parentNoteId, uploadBuffer, originalName, shrinkImageSwitch, trimFilename = false) {
2020-04-08 11:07:38 +02:00
log.info(`Saving image ${originalName}`);
if (trimFilename && originalName.length > 40) {
// https://github.com/zadam/trilium/issues/2307
originalName = "image";
}
2018-02-11 00:18:59 -05:00
const fileName = sanitizeFilename(originalName);
2021-05-02 11:23:58 +02:00
const parentNote = becca.getNote(parentNoteId);
2019-11-08 22:34:30 +01:00
2020-06-20 12:31:38 +02:00
const {note} = noteService.createNewNote({
2019-11-16 11:09:52 +01:00
parentNoteId,
title: fileName,
2018-11-08 11:08:16 +01:00
type: 'image',
mime: 'unknown',
content: '',
2019-11-16 11:09:52 +01:00
isProtected: parentNote.isProtected && protectedSessionService.isProtectedSessionAvailable()
2018-11-08 11:08:16 +01:00
});
2018-02-11 00:18:59 -05:00
2020-06-20 12:31:38 +02:00
note.addLabel('originalFileName', originalName);
2019-11-16 11:09:52 +01:00
// resizing images asynchronously since JIMP does not support sync operation
processImage(uploadBuffer, originalName, shrinkImageSwitch).then(({buffer, imageFormat}) => {
sql.transactional(() => {
note.mime = getImageMimeFromExtension(imageFormat.ext);
if (!originalName.includes(".")) {
originalName += "." + imageFormat.ext;
note.setLabel('originalFileName', originalName);
note.title = sanitizeFilename(originalName);
}
note.save();
note.setContent(buffer);
})
});
return {
fileName,
2019-02-25 21:22:57 +01:00
note,
2018-11-08 11:08:16 +01:00
noteId: note.noteId,
2019-11-08 23:09:57 +01:00
url: `api/images/${note.noteId}/${fileName}`
};
2018-02-11 00:18:59 -05:00
}
2019-03-03 20:41:03 +01:00
async function shrinkImage(buffer, originalName) {
2020-06-20 12:31:38 +02:00
const jpegQuality = optionService.getOptionInt('imageJpegQuality');
let finalImageBuffer;
try {
finalImageBuffer = await resize(buffer, jpegQuality);
}
catch (e) {
log.error("Failed to resize image '" + originalName + "'\nStack: " + e.stack);
finalImageBuffer = buffer;
2019-02-24 12:24:28 +01:00
}
// if resizing did not help with size then save the original
// (can happen when e.g. resizing PNG into JPEG)
if (finalImageBuffer.byteLength >= buffer.byteLength) {
finalImageBuffer = buffer;
}
2019-02-24 12:24:28 +01:00
return finalImageBuffer;
}
async function resize(buffer, quality) {
2020-06-20 12:31:38 +02:00
const imageMaxWidthHeight = optionService.getOptionInt('imageMaxWidthHeight');
const image = await jimp.read(buffer);
2018-02-11 00:18:59 -05:00
if (image.bitmap.width > image.bitmap.height && image.bitmap.width > imageMaxWidthHeight) {
image.resize(imageMaxWidthHeight, jimp.AUTO);
2018-02-11 00:18:59 -05:00
}
else if (image.bitmap.height > imageMaxWidthHeight) {
image.resize(jimp.AUTO, imageMaxWidthHeight);
2018-02-11 00:18:59 -05:00
}
image.quality(quality);
2018-02-11 00:18:59 -05:00
// when converting PNG to JPG we lose alpha channel, this is replaced by white to match Trilium white background
image.background(0xFFFFFFFF);
return await image.getBufferAsync(jimp.MIME_JPEG);
2018-02-11 00:18:59 -05:00
}
module.exports = {
2019-11-08 22:34:30 +01:00
saveImage,
updateImage
2020-06-20 12:31:38 +02:00
};