mirror of
https://github.com/zadam/trilium.git
synced 2025-03-01 14:22:32 +01:00
121 lines
3.6 KiB
TypeScript
121 lines
3.6 KiB
TypeScript
"use strict";
|
|
|
|
import imageService from "../../services/image.js";
|
|
import becca from "../../becca/becca.js";
|
|
import fs from "fs";
|
|
import { Request, Response } from 'express';
|
|
import BNote from "../../becca/entities/bnote.js";
|
|
import BRevision from "../../becca/entities/brevision.js";
|
|
import { AppRequest } from '../route-interface';
|
|
import { RESOURCE_DIR } from "../../services/resource_dir.js";
|
|
|
|
function returnImageFromNote(req: Request, res: Response) {
|
|
const image = becca.getNote(req.params.noteId);
|
|
|
|
return returnImageInt(image, res);
|
|
}
|
|
|
|
function returnImageFromRevision(req: Request, res: Response) {
|
|
const image = becca.getRevision(req.params.revisionId);
|
|
|
|
return returnImageInt(image, res);
|
|
}
|
|
|
|
function returnImageInt(image: BNote | BRevision | null, res: Response) {
|
|
if (!image) {
|
|
res.set('Content-Type', 'image/png');
|
|
return res.send(fs.readFileSync(`${RESOURCE_DIR}/db/image-deleted.png`));
|
|
} else if (!["image", "canvas", "mermaid"].includes(image.type)) {
|
|
return res.sendStatus(400);
|
|
}
|
|
|
|
if (image.type === 'canvas') {
|
|
renderSvgAttachment(image, res, 'canvas-export.svg');
|
|
} else if (image.type === 'mermaid') {
|
|
renderSvgAttachment(image, res, 'mermaid-export.svg');
|
|
} else {
|
|
res.set('Content-Type', image.mime);
|
|
res.set("Cache-Control", "no-cache, no-store, must-revalidate");
|
|
res.send(image.getContent());
|
|
}
|
|
}
|
|
|
|
function renderSvgAttachment(image: BNote | BRevision, res: Response, attachmentName: string) {
|
|
let svg: string | Buffer = '<svg/>'
|
|
const attachment = image.getAttachmentByTitle(attachmentName);
|
|
|
|
if (attachment) {
|
|
svg = attachment.getContent();
|
|
} else {
|
|
// backwards compatibility, before attachments, the SVG was stored in the main note content as a separate key
|
|
const contentSvg = image.getJsonContentSafely()?.svg;
|
|
|
|
if (contentSvg) {
|
|
svg = contentSvg;
|
|
}
|
|
}
|
|
|
|
res.set('Content-Type', "image/svg+xml");
|
|
res.set("Cache-Control", "no-cache, no-store, must-revalidate");
|
|
res.send(svg);
|
|
}
|
|
|
|
|
|
function returnAttachedImage(req: Request, res: Response) {
|
|
const attachment = becca.getAttachment(req.params.attachmentId);
|
|
|
|
if (!attachment) {
|
|
res.set('Content-Type', 'image/png');
|
|
return res.send(fs.readFileSync(`${RESOURCE_DIR}/db/image-deleted.png`));
|
|
}
|
|
|
|
if (!["image"].includes(attachment.role)) {
|
|
return res.setHeader("Content-Type", "text/plain")
|
|
.status(400)
|
|
.send(`Attachment '${attachment.attachmentId}' has role '${attachment.role}', but 'image' was expected.`);
|
|
}
|
|
|
|
res.set('Content-Type', attachment.mime);
|
|
res.set("Cache-Control", "no-cache, no-store, must-revalidate");
|
|
res.send(attachment.getContent());
|
|
}
|
|
|
|
function updateImage(req: AppRequest) {
|
|
const {noteId} = req.params;
|
|
const {file} = req;
|
|
|
|
const note = becca.getNoteOrThrow(noteId);
|
|
|
|
if (!file) {
|
|
return {
|
|
uploaded: false,
|
|
message: `Missing image data.`
|
|
};
|
|
}
|
|
|
|
if (!["image/png", "image/jpeg", "image/gif", "image/webp", "image/svg+xml"].includes(file.mimetype)) {
|
|
return {
|
|
uploaded: false,
|
|
message: `Unknown image type: ${file.mimetype}`
|
|
};
|
|
}
|
|
|
|
if (typeof file.buffer === "string") {
|
|
return {
|
|
uploaded: false,
|
|
message: "Invalid image content."
|
|
};
|
|
}
|
|
|
|
imageService.updateImage(noteId, file.buffer, file.originalname);
|
|
|
|
return { uploaded: true };
|
|
}
|
|
|
|
export default {
|
|
returnImageFromNote,
|
|
returnImageFromRevision,
|
|
returnAttachedImage,
|
|
updateImage
|
|
};
|