mirror of
				https://github.com/zadam/trilium.git
				synced 2025-11-03 21:19:01 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			122 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			122 lines
		
	
	
		
			3.7 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 { 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", "mindMap" ].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 if (image.type === "mindMap") {
 | 
						|
        renderSvgAttachment(image, res, 'mindmap-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: Request) {
 | 
						|
    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
 | 
						|
};
 |