2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								<!DOCTYPE html>  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								< html  lang = "en" >  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								< head >  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    < meta  charset = "utf-8" > 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    < title > JSDoc: Source: entities/note.js< / title > 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    < script  src = "scripts/prettify/prettify.js" >  < / script > 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    < script  src = "scripts/prettify/lang-css.js" >  < / script > 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <!-- [if lt IE 9]>
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      < script  src = "//html5shiv.googlecode.com/svn/trunk/html5.js" > < / script > 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <![endif]--> 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    < link  type = "text/css"  rel = "stylesheet"  href = "styles/prettify-tomorrow.css" > 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    < link  type = "text/css"  rel = "stylesheet"  href = "styles/jsdoc-default.css" > 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								< / head >  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								< body >  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								< div  id = "main" >  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    < h1  class = "page-title" > Source: entities/note.js< / h1 > 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    < section > 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        < article > 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            < pre  class = "prettyprint source linenums" > < code > "use strict";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								const Entity = require('./entity');
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								const Attribute = require('./attribute');
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								const protectedSessionService = require('../services/protected_session');
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								const repository = require('../services/repository');
							 
						 
					
						
							
								
									
										
										
										
											2018-12-22 22:28:49 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								const sql = require('../services/sql');
							 
						 
					
						
							
								
									
										
										
										
											2019-03-27 21:04:25 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								const utils = require('../services/utils');
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								const dateUtils = require('../services/date_utils');
							 
						 
					
						
							
								
									
										
										
										
											2019-03-27 21:04:25 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								const syncTableService = require('../services/sync_table');
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								const LABEL = 'label';
							 
						 
					
						
							
								
									
										
										
										
											2018-11-13 12:50:08 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								const LABEL_DEFINITION = 'label-definition';
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								const RELATION = 'relation';
							 
						 
					
						
							
								
									
										
										
										
											2018-11-13 12:50:08 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								const RELATION_DEFINITION = 'relation-definition';
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-03-12 21:42:27 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								const STRING_MIME_TYPES = ["application/x-javascript"];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								/**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								 * This represents a Note which is a central object in the Trilium Notes project.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								 *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								 * @property {string} noteId - primary key
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								 * @property {string} type - one of "text", "code", "file" or "render"
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								 * @property {string} mime - MIME type, e.g. "text/html"
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								 * @property {string} title - note title
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								 * @property {boolean} isProtected - true if note is protected
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								 * @property {boolean} isDeleted - true if note is deleted
							 
						 
					
						
							
								
									
										
										
										
											2019-03-14 20:49:10 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								 * @property {string} dateCreated - local date time (with offset)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								 * @property {string} dateModified - local date time (with offset)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								 * @property {string} utcDateCreated
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								 * @property {string} utcDateModified
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								 *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								 * @extends Entity
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								 */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								class Note extends Entity {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    static get entityName() { return "notes"; }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    static get primaryKeyName() { return "noteId"; }
							 
						 
					
						
							
								
									
										
										
										
											2019-02-20 22:24:51 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    static get hashedProperties() { return ["noteId", "title", "type", "isProtected", "isDeleted"]; }
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param row - object containing database row from "notes" table
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    constructor(row) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        super(row);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.isProtected = !!this.isProtected;
							 
						 
					
						
							
								
									
										
										
										
											2018-11-08 23:49:44 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /* true if content (meaning any kind of potentially encrypted content) is either not encrypted
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         * or encrypted, but with available protected session (so effectively decrypted) */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.isContentAvailable = true;
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // check if there's noteId, otherwise this is a new entity which wasn't encrypted yet
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (this.isProtected & &  this.noteId) {
							 
						 
					
						
							
								
									
										
										
										
											2018-11-08 23:49:44 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            this.isContentAvailable = protectedSessionService.isProtectedSessionAvailable();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-02-20 22:24:51 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            if (this.isContentAvailable) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                protectedSessionService.decryptNote(this);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            else {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                this.title = "[protected]";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-04-14 12:24:48 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /*
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Note content has quite special handling - it's not a separate entity, but a lazily loaded
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * part of Note entity with it's own sync. Reasons behind this hybrid design has been:
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * - content can be quite large and it's not necessary to load it / fill memory for any note access even if we don't need a content, especially for bulk operations like search
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * - changes in the note metadata or title should not trigger note content sync (so we keep separate utcDateModified and sync rows)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * - but to the user note content and title changes are one and the same - single dateModified (so all changes must go through Note and content is not a separate entity)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-03-27 21:04:25 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /** @returns {Promise< *>} */
							 
						 
					
						
							
								
									
										
										
										
											2019-04-14 12:24:48 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    async getContent(silentNotFoundError = false) {
							 
						 
					
						
							
								
									
										
										
										
											2019-03-27 21:04:25 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        if (this.content === undefined) {
							 
						 
					
						
							
								
									
										
										
										
											2019-04-14 12:24:48 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            const res = await sql.getRow(`SELECT content, hash FROM note_contents WHERE noteId = ?`, [this.noteId]);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (!res) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                if (silentNotFoundError) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    return undefined;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                else {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    throw new Error("Cannot find note content for noteId=" + this.noteId);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            this.content = res.content;
							 
						 
					
						
							
								
									
										
										
										
											2019-02-20 22:24:51 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-03-27 21:04:25 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            if (this.isProtected) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                if (this.isContentAvailable) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    protectedSessionService.decryptNoteContent(this);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                else {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    this.content = "";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
									
										
										
										
											2019-02-20 22:24:51 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (this.isStringNote()) {
							 
						 
					
						
							
								
									
										
										
										
											2019-03-27 21:04:25 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                this.content = this.content === null
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    ? ""
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    : this.content.toString("UTF-8");
							 
						 
					
						
							
								
									
										
										
										
											2019-02-20 22:24:51 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-03-27 21:04:25 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return this.content;
							 
						 
					
						
							
								
									
										
										
										
											2019-02-20 22:24:51 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /** @returns {Promise< *>} */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getJsonContent() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const content = await this.getContent();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return JSON.parse(content);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /** @returns {Promise} */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async setContent(content) {
							 
						 
					
						
							
								
									
										
										
										
											2019-04-14 12:24:48 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        // force updating note itself so that dateChanged is represented correctly even for the content
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.forcedChange = true;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        await this.save();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-03-27 21:04:25 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        this.content = content;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const pojo = {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            noteId: this.noteId,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            content: content,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            utcDateModified: dateUtils.utcNowDateTime(),
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            hash: utils.hash(this.noteId + "|" + content)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        };
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (this.isProtected) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (this.isContentAvailable) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                protectedSessionService.encryptNoteContent(pojo);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            else {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                throw new Error(`Cannot update content of noteId=${this.noteId} since we're out of protected session.`);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
									
										
										
										
											2019-02-20 22:24:51 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-03-27 21:04:25 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        await sql.upsert("note_contents", "noteId", pojo);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        await syncTableService.addNoteContentSync(this.noteId);
							 
						 
					
						
							
								
									
										
										
										
											2019-02-20 22:24:51 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /** @returns {Promise} */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async setJsonContent(content) {
							 
						 
					
						
							
								
									
										
										
										
											2019-03-12 21:42:27 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        await this.setContent(JSON.stringify(content, null, '\t'));
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /** @returns {boolean} true if this note is the root of the note tree. Root note has "root" noteId */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    isRoot() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.noteId === 'root';
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /** @returns {boolean} true if this note is of application/json content type */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    isJson() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.mime === "application/json";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /** @returns {boolean} true if this note is JavaScript (code or attachment) */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    isJavaScript() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return (this.type === "code" || this.type === "file")
							 
						 
					
						
							
								
									
										
										
										
											2018-12-22 22:28:49 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            & &  (this.mime.startsWith("application/javascript")
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                || this.mime === "application/x-javascript"
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                || this.mime === "text/javascript");
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /** @returns {boolean} true if this note is HTML */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    isHtml() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return (this.type === "code" || this.type === "file" || this.type === "render") & &  this.mime === "text/html";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-02-20 22:24:51 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /** @returns {boolean} true if the note has string content (not binary) */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    isStringNote() {
							 
						 
					
						
							
								
									
										
										
										
											2019-03-12 21:42:27 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return ["text", "code", "relation-map", "search"].includes(this.type)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            || this.mime.startsWith('text/')
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            || STRING_MIME_TYPES.includes(this.mime);
							 
						 
					
						
							
								
									
										
										
										
											2019-02-20 22:24:51 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    /** @returns {string} JS script environment - either "frontend" or "backend" */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getScriptEnv() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (this.isHtml() || (this.isJavaScript() & &  this.mime.endsWith('env=frontend'))) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return "frontend";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (this.type === 'render') {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return "frontend";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (this.isJavaScript() & &  this.mime.endsWith('env=backend')) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return "backend";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Attribute[]>} attributes belonging to this specific note (excludes inherited attributes)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getOwnedAttributes() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return await repository.getEntities(`SELECT * FROM attributes WHERE isDeleted = 0 AND noteId = ?`, [this.noteId]);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2018-11-08 23:49:44 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Attribute[]>} relations targetting this specific note
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getTargetRelations() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return await repository.getEntities("SELECT * FROM attributes WHERE type = 'relation' AND isDeleted = 0 AND value = ?", [this.noteId]);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2018-09-03 16:05:28 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [name] - attribute name to filter
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Attribute[]>} all note's attributes, including inherited ones
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getAttributes(name) {
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        if (!this.__attributeCache) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            await this.loadAttributesToCache();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2018-09-03 16:05:28 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        if (name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return this.__attributeCache.filter(attr => attr.name === name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return this.__attributeCache;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2018-09-03 16:05:28 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [name] - label name to filter
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Attribute[]>} all note's labels (attributes with type label), including inherited ones
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getLabels(name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return (await this.getAttributes(name)).filter(attr => attr.type === LABEL);
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2018-11-13 12:50:08 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [name] - label name to filter
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Attribute[]>} all note's label definitions, including inherited ones
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getLabelDefinitions(name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return (await this.getAttributes(name)).filter(attr => attr.type === LABEL_DEFINITION);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2018-09-03 16:05:28 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [name] - relation name to filter
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Attribute[]>} all note's relations (attributes with type relation), including inherited ones
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getRelations(name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return (await this.getAttributes(name)).filter(attr => attr.type === RELATION);
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2018-11-13 12:50:08 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [name] - relation name to filter
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Attribute[]>} all note's relation definitions including inherited ones
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getRelationDefinitions(name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return (await this.getAttributes(name)).filter(attr => attr.type === RELATION_DEFINITION);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Clear note's attributes cache to force fresh reload for next attribute request.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Cache is note instance scoped.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    invalidateAttributeCache() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.__attributeCache = null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /** @returns {Promise< void>} */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async loadAttributesToCache() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const attributes = await repository.getEntities(`
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            WITH RECURSIVE
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            tree(noteId, level) AS (
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                SELECT ?, 0
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                UNION
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                SELECT branches.parentNoteId, tree.level + 1 FROM branches
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    JOIN tree ON branches.noteId = tree.noteId
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    JOIN notes ON notes.noteId = branches.parentNoteId
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                WHERE notes.isDeleted = 0
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                  AND branches.isDeleted = 0
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            ),
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            treeWithAttrs(noteId, level) AS (
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                SELECT * FROM tree
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                UNION
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                SELECT attributes.value, treeWithAttrs.level + 1 FROM attributes
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                     JOIN treeWithAttrs ON treeWithAttrs.noteId = attributes.noteId
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                WHERE attributes.isDeleted = 0
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                  AND attributes.type = 'relation'
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                  AND attributes.name = 'template'
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                  AND (attributes.noteId = ? OR attributes.isInheritable = 1)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                )
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            SELECT attributes.* FROM attributes JOIN treeWithAttrs ON attributes.noteId = treeWithAttrs.noteId
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            WHERE attributes.isDeleted = 0 AND (attributes.isInheritable = 1 OR attributes.noteId = ?)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            ORDER BY level, noteId, position`, [this.noteId, this.noteId, this.noteId]);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // attributes are ordered so that "closest" attributes are first
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // we order by noteId so that attributes from same note stay together. Actual noteId ordering doesn't matter.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const filteredAttributes = attributes.filter((attr, index) => {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (attr.isDefinition()) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                const firstDefinitionIndex = attributes.findIndex(el => el.type === attr.type & &  el.name === attr.name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                // keep only if this element is the first definition for this type &  name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                return firstDefinitionIndex === index;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            else {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                const definitionAttr = attributes.find(el => el.type === attr.type + '-definition' & &  el.name === attr.name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                if (!definitionAttr) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    return true;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                const definition = definitionAttr.value;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                if (definition.multiplicityType === 'multivalue') {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    return true;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                else {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    const firstAttrIndex = attributes.findIndex(el => el.type === attr.type & &  el.name === attr.name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    // in case of single-valued attribute we'll keep it only if it's first (closest)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    return firstAttrIndex === index;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        });
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for (const attr of filteredAttributes) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            attr.isOwned = attr.noteId === this.noteId;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.__attributeCache = filteredAttributes;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} type - attribute type (label, relation, etc.)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - attribute name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< boolean>} true if note has an attribute with given type and name (including inherited)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async hasAttribute(type, name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return !!await this.getAttribute(type, name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} type - attribute type (label, relation, etc.)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - attribute name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Attribute>} attribute of given type and name. If there's more such attributes, first is  returned. Returns null if there's no such attribute belonging to this note.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getAttribute(type, name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const attributes = await this.getAttributes();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return attributes.find(attr => attr.type === type & &  attr.name === name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} type - attribute type (label, relation, etc.)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - attribute name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< string>} attribute value of given type and name or null if no such attribute exists.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getAttributeValue(type, name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const attr = await this.getAttribute(type, name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return attr ? attr.value : null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Based on enabled, attribute is either set or removed.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} type - attribute type ('relation', 'label' etc.)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {boolean} enabled - toggle On or Off
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - attribute name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - attribute value (optional)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< void>}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async toggleAttribute(type, enabled, name, value) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (enabled) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            await this.setAttribute(type, name, value);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            await this.removeAttribute(type, name, value);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Creates given attribute name-value pair if it doesn't exist.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} type - attribute type (label, relation, etc.)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - attribute name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - attribute value (optional)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< void>}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async setAttribute(type, name, value) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const attributes = await this.getOwnedAttributes();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        let attr = attributes.find(attr => attr.type === type & &  (value === undefined || attr.value === value));
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (!attr) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            attr = new Attribute({
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                noteId: this.noteId,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                type: type,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                name: name,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                value: value !== undefined ? value : ""
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            });
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            await attr.save();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            this.invalidateAttributeCache();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Removes given attribute name-value pair if it exists.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} type - attribute type (label, relation, etc.)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - attribute name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - attribute value (optional)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< void>}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async removeAttribute(type, name, value) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const attributes = await this.getOwnedAttributes();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for (const attribute of attributes) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (attribute.type === type & &  (value === undefined || value === attribute.value)) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                attribute.isDeleted = true;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                await attribute.save();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                this.invalidateAttributeCache();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - label name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< boolean>} true if label exists (including inherited)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async hasLabel(name) { return await this.hasAttribute(LABEL, name); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - relation name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< boolean>} true if relation exists (including inherited)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async hasRelation(name) { return await this.hasAttribute(RELATION, name); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - label name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Attribute>} label if it exists, null otherwise
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getLabel(name) { return await this.getAttribute(LABEL, name); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - relation name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Attribute>} relation if it exists, null otherwise
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getRelation(name) { return await this.getAttribute(RELATION, name); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - label name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< string>} label value if label exists, null otherwise
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getLabelValue(name) { return await this.getAttributeValue(LABEL, name); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - relation name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< string>} relation value if relation exists, null otherwise
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getRelationValue(name) { return await this.getAttributeValue(RELATION, name); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2018-12-22 22:28:49 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Note>|null} target note of the relation or null (if target is empty or note was not found)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getRelationTarget(name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const relation = await this.getRelation(name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return relation ? await repository.getNote(relation.value) : null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Based on enabled, label is either set or removed.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {boolean} enabled - toggle On or Off
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - label name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - label value (optional)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< void>}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async toggleLabel(enabled, name, value) { return await this.toggleAttribute(LABEL, enabled, name, value); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Based on enabled, relation is either set or removed.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {boolean} enabled - toggle On or Off
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - relation name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - relation value (noteId)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< void>}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async toggleRelation(enabled, name, value) { return await this.toggleAttribute(RELATION, enabled, name, value); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Create label name-value pair if it doesn't exist yet.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - label name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - label value
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< void>}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async setLabel(name, value) { return await this.setAttribute(LABEL, name, value); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Create relation name-value pair if it doesn't exist yet.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - relation name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - relation value (noteId)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< void>}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async setRelation(name, value) { return await this.setAttribute(RELATION, name, value); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Remove label name-value pair, if it exists.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - label name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - label value
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< void>}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async removeLabel(name, value) { return await this.removeAttribute(LABEL, name, value); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Remove relation name-value pair, if it exists.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - relation name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - relation value (noteId)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< void>}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async removeRelation(name, value) { return await this.removeAttribute(RELATION, name, value); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
									
										
										
										
											2018-12-22 22:28:49 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								     * @return {Promise< string[]>} return list of all descendant noteIds of this note. Returning just noteIds because number of notes can be huge. Includes also this note's noteId
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
									
										
										
										
											2018-12-22 22:28:49 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    async getDescendantNoteIds() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return await sql.getColumn(`
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            WITH RECURSIVE
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            tree(noteId) AS (
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                SELECT ?
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                UNION
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                SELECT branches.noteId FROM branches
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    JOIN tree ON branches.parentNoteId = tree.noteId
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    JOIN notes ON notes.noteId = branches.noteId
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                WHERE notes.isDeleted = 0
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                  AND branches.isDeleted = 0
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            )
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            SELECT noteId FROM tree`, [this.noteId]);
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
									
										
										
										
											2018-12-22 22:28:49 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								     * Finds descendant notes with given attribute name and value. Only own attributes are considered, not inherited ones
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} type - attribute type (label, relation, etc.)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - attribute name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - attribute value
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Note[]>}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
									
										
										
										
											2018-12-22 22:28:49 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    async getDescendantNotesWithAttribute(type, name, value) {
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        const params = [this.noteId, name];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        let valueCondition = "";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (value !== undefined) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            params.push(value);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            valueCondition = " AND attributes.value = ?";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const notes = await repository.getEntities(`
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            WITH RECURSIVE
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            tree(noteId) AS (
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                SELECT ?
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                UNION
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                SELECT branches.noteId FROM branches
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    JOIN tree ON branches.parentNoteId = tree.noteId
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    JOIN notes ON notes.noteId = branches.noteId
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                WHERE notes.isDeleted = 0
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                  AND branches.isDeleted = 0
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            )
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            SELECT notes.* FROM notes 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            JOIN tree ON tree.noteId = notes.noteId
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            JOIN attributes ON attributes.noteId = notes.noteId
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            WHERE attributes.isDeleted = 0 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								              AND attributes.name = ?
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								              ${valueCondition} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            ORDER BY noteId, position`, params);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return notes;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
									
										
										
										
											2018-12-22 22:28:49 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								     * Finds descendant notes with given label name and value. Only own labels are considered, not inherited ones
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - label name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - label value
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Note[]>}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
									
										
										
										
											2018-12-22 22:28:49 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    async getDescendantNotesWithLabel(name, value) { return await this.getDescendantNotesWithAttribute(LABEL, name, value); }
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
									
										
										
										
											2018-12-22 22:28:49 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								     * Finds descendant notes with given relation name and value. Only own relations are considered, not inherited ones
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - relation name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - relation value
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Note[]>}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
									
										
										
										
											2018-12-22 22:28:49 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    async getDescendantNotesWithRelation(name, value) { return await this.getDescendantNotesWithAttribute(RELATION, name, value); }
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Returns note revisions of this note.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< NoteRevision[]>}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getRevisions() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return await repository.getEntities("SELECT * FROM note_revisions WHERE noteId = ?", [this.noteId]);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
									
										
										
										
											2018-11-15 13:58:14 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								     * Get list of links coming out of this note.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
									
										
										
										
											2018-11-08 23:49:44 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								     * @returns {Promise< Link[]>}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getLinks() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return await repository.getEntities("SELECT * FROM links WHERE noteId = ? AND isDeleted = 0", [this.noteId]);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2018-11-15 13:58:14 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Get list of links targetting this note.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Link[]>}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getTargetLinks() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return await repository.getEntities("SELECT * FROM links WHERE targetNoteId = ? AND isDeleted = 0", [this.noteId]);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2018-11-08 23:49:44 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Return all links from this note, including deleted ones.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Link[]>}
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
									
										
										
										
											2018-11-08 23:49:44 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    async getLinksWithDeleted() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return await repository.getEntities("SELECT * FROM links WHERE noteId = ?", [this.noteId]);
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Branch[]>}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getBranches() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return await repository.getEntities("SELECT * FROM branches WHERE isDeleted = 0 AND noteId = ?", [this.noteId]);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2018-09-03 16:05:28 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {boolean} - true if note has children
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async hasChildren() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return (await this.getChildNotes()).length > 0;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Note[]>} child notes of this note
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getChildNotes() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return await repository.getEntities(`
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          SELECT notes.* 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          FROM branches 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            JOIN notes USING(noteId) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          WHERE notes.isDeleted = 0
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                AND branches.isDeleted = 0
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                AND branches.parentNoteId = ?
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          ORDER BY branches.notePosition`, [this.noteId]);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Branch[]>} child branches of this note
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getChildBranches() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return await repository.getEntities(`
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          SELECT branches.* 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          FROM branches 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          WHERE branches.isDeleted = 0
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                AND branches.parentNoteId = ?
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          ORDER BY branches.notePosition`, [this.noteId]);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Promise< Note[]>} parent notes of this note (note can have multiple parents because of cloning)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    async getParentNotes() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return await repository.getEntities(`
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          SELECT parent_notes.* 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          FROM 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            branches AS child_tree 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            JOIN notes AS parent_notes ON parent_notes.noteId = child_tree.parentNoteId 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          WHERE child_tree.noteId = ?
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                AND child_tree.isDeleted = 0
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                AND parent_notes.isDeleted = 0`, [this.noteId]);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    beforeSaving() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (!this.isDeleted) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            this.isDeleted = false;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (!this.dateCreated) {
							 
						 
					
						
							
								
									
										
										
										
											2019-03-14 20:49:10 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            this.dateCreated = dateUtils.localNowDateTime();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (!this.utcDateCreated) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            this.utcDateCreated = dateUtils.utcNowDateTime();
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        super.beforeSaving();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (this.isChanged) {
							 
						 
					
						
							
								
									
										
										
										
											2019-03-14 20:49:10 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            this.dateModified = dateUtils.localNowDateTime();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            this.utcDateModified = dateUtils.utcNowDateTime();
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
									
										
										
										
											2018-12-22 22:28:49 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // cannot be static!
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    updatePojo(pojo) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (pojo.isProtected) {
							 
						 
					
						
							
								
									
										
										
										
											2019-02-20 22:24:51 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            if (this.isContentAvailable) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                protectedSessionService.encryptNote(pojo);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            else {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                // updating protected note outside of protected session means we will keep original ciphertexts
							 
						 
					
						
							
								
									
										
										
										
											2019-03-27 21:04:25 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                delete pojo.title;
							 
						 
					
						
							
								
									
										
										
										
											2019-02-20 22:24:51 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
									
										
										
										
											2018-12-22 22:28:49 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        delete pojo.isContentAvailable;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        delete pojo.__attributeCache;
							 
						 
					
						
							
								
									
										
										
										
											2019-03-27 21:04:25 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        delete pojo.content;
							 
						 
					
						
							
								
									
										
										
										
											2019-04-14 12:24:48 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        delete pojo.contentHash;
							 
						 
					
						
							
								
									
										
										
										
											2018-12-22 22:28:49 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								module.exports = Note;< / code > < / pre > 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        < / article > 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    < / section > 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								< / div >  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								< nav >  
						 
					
						
							
								
									
										
										
										
											2019-03-27 21:04:25 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    < h2 > < a  href = "index.html" > Home< / a > < / h2 > < h3 > Classes< / h3 > < ul > < li > < a  href = "ApiToken.html" > ApiToken< / a > < / li > < li > < a  href = "Attribute.html" > Attribute< / a > < / li > < li > < a  href = "BackendScriptApi.html" > BackendScriptApi< / a > < / li > < li > < a  href = "Branch.html" > Branch< / a > < / li > < li > < a  href = "Entity.html" > Entity< / a > < / li > < li > < a  href = "Link.html" > Link< / a > < / li > < li > < a  href = "Note.html" > Note< / a > < / li > < li > < a  href = "NoteRevision.html" > NoteRevision< / a > < / li > < li > < a  href = "Option.html" > Option< / a > < / li > < li > < a  href = "RecentNote.html" > RecentNote< / a > < / li > < / ul > < h3 > < a  href = "global.html" > Global< / a > < / h3 > 
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								< / nav >  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								< br  class = "clear" >  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								< footer >  
						 
					
						
							
								
									
										
										
										
											2018-11-19 21:03:43 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    Documentation generated by < a  href = "https://github.com/jsdoc3/jsdoc" > JSDoc 3.5.5< / a > 
							 
						 
					
						
							
								
									
										
										
										
											2018-08-29 20:44:35 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								< / footer >  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								< script >  prettyPrint ( ) ;  < / script >  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								< script  src = "scripts/linenumber.js" >  < / script >  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								< / body >  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								< / html >