| 
									
										
										
										
											2018-01-28 23:16:50 -05:00
										 |  |  | "use strict"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-29 18:34:59 -05:00
										 |  |  | const Entity = require('./entity'); | 
					
						
							| 
									
										
										
										
											2018-08-13 10:59:31 +02:00
										 |  |  | const Attribute = require('./attribute'); | 
					
						
							| 
									
										
										
										
											2018-04-20 00:12:01 -04:00
										 |  |  | const protectedSessionService = require('../services/protected_session'); | 
					
						
							| 
									
										
										
										
											2018-03-31 10:51:37 -04:00
										 |  |  | const repository = require('../services/repository'); | 
					
						
							| 
									
										
										
										
											2018-04-02 20:46:46 -04:00
										 |  |  | const dateUtils = require('../services/date_utils'); | 
					
						
							| 
									
										
										
										
											2018-01-28 23:16:50 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-29 18:34:59 -05:00
										 |  |  | class Note extends Entity { | 
					
						
							| 
									
										
										
										
											2018-01-29 23:35:36 -05:00
										 |  |  |     static get tableName() { return "notes"; } | 
					
						
							| 
									
										
										
										
											2018-01-30 20:12:19 -05:00
										 |  |  |     static get primaryKeyName() { return "noteId"; } | 
					
						
							| 
									
										
										
										
											2018-08-12 20:04:48 +02:00
										 |  |  |     static get hashedProperties() { return ["noteId", "title", "content", "type", "isProtected", "isDeleted"]; } | 
					
						
							| 
									
										
										
										
											2018-01-29 23:35:36 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-31 10:51:37 -04:00
										 |  |  |     constructor(row) { | 
					
						
							|  |  |  |         super(row); | 
					
						
							| 
									
										
										
										
											2018-01-29 20:57:55 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-07 11:38:00 +02:00
										 |  |  |         this.isProtected = !!this.isProtected; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-08 12:27:10 -04:00
										 |  |  |         // check if there's noteId, otherwise this is a new entity which wasn't encrypted yet
 | 
					
						
							|  |  |  |         if (this.isProtected && this.noteId) { | 
					
						
							| 
									
										
										
										
											2018-04-20 00:12:01 -04:00
										 |  |  |             protectedSessionService.decryptNote(this); | 
					
						
							| 
									
										
										
										
											2018-01-29 23:35:36 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-11 00:10:11 -04:00
										 |  |  |         this.setContent(this.content); | 
					
						
							| 
									
										
										
										
											2018-01-29 20:57:55 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-08 08:31:19 -04:00
										 |  |  |     setContent(content) { | 
					
						
							|  |  |  |         this.content = content; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-11 00:10:11 -04:00
										 |  |  |         try { | 
					
						
							| 
									
										
										
										
											2018-04-08 08:31:19 -04:00
										 |  |  |             this.jsonContent = JSON.parse(this.content); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-04-11 00:10:11 -04:00
										 |  |  |         catch(e) {} | 
					
						
							| 
									
										
										
										
											2018-04-08 08:31:19 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-01 09:26:02 +02:00
										 |  |  |     isRoot() { | 
					
						
							|  |  |  |         return this.noteId === 'root'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-29 23:17:44 -05:00
										 |  |  |     isJson() { | 
					
						
							| 
									
										
										
										
											2018-03-25 23:25:17 -04:00
										 |  |  |         return this.mime === "application/json"; | 
					
						
							| 
									
										
										
										
											2018-01-29 23:17:44 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-18 10:47:02 -05:00
										 |  |  |     isJavaScript() { | 
					
						
							| 
									
										
										
										
											2018-03-01 22:30:06 -05:00
										 |  |  |         return (this.type === "code" || this.type === "file") | 
					
						
							| 
									
										
										
										
											2018-03-05 22:08:45 -05:00
										 |  |  |             && (this.mime.startsWith("application/javascript") || this.mime === "application/x-javascript"); | 
					
						
							| 
									
										
										
										
											2018-02-18 10:47:02 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-04 21:05:14 -05:00
										 |  |  |     isHtml() { | 
					
						
							| 
									
										
										
										
											2018-05-26 19:27:47 -04:00
										 |  |  |         return (this.type === "code" || this.type === "file" || this.type === "render") && this.mime === "text/html"; | 
					
						
							| 
									
										
										
										
											2018-03-04 21:05:14 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 23:09:36 -05:00
										 |  |  |     getScriptEnv() { | 
					
						
							|  |  |  |         if (this.isHtml() || (this.isJavaScript() && this.mime.endsWith('env=frontend'))) { | 
					
						
							|  |  |  |             return "frontend"; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-07 00:17:18 -05:00
										 |  |  |         if (this.type === 'render') { | 
					
						
							|  |  |  |             return "frontend"; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 23:09:36 -05:00
										 |  |  |         if (this.isJavaScript() && this.mime.endsWith('env=backend')) { | 
					
						
							|  |  |  |             return "backend"; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return null; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-07 13:33:10 +02:00
										 |  |  |     async getOwnedAttributes() { | 
					
						
							|  |  |  |         return await repository.getEntities(`SELECT * FROM attributes WHERE isDeleted = 0 AND noteId = ?`, [this.noteId]); | 
					
						
							| 
									
										
										
										
											2018-01-28 23:16:50 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-07 13:33:10 +02:00
										 |  |  |     async getAttributes() { | 
					
						
							|  |  |  |         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 | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         SELECT attributes.* FROM attributes JOIN tree ON attributes.noteId = tree.noteId  | 
					
						
							|  |  |  |         WHERE attributes.isDeleted = 0 AND (attributes.isInheritable = 1 OR attributes.noteId = ?) | 
					
						
							|  |  |  |         ORDER BY level, noteId, position`, [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; | 
					
						
							| 
									
										
										
										
											2018-03-03 09:11:41 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-07 13:33:10 +02:00
										 |  |  |         return filteredAttributes; | 
					
						
							| 
									
										
										
										
											2018-03-03 09:11:41 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-24 22:02:26 -04:00
										 |  |  |     async hasLabel(name) { | 
					
						
							| 
									
										
										
										
											2018-08-07 13:33:10 +02:00
										 |  |  |         return !!await this.getLabel(name); | 
					
						
							| 
									
										
										
										
											2018-03-04 22:09:51 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-24 22:02:26 -04:00
										 |  |  |     // WARNING: this doesn't take into account the possibility to have multi-valued labels!
 | 
					
						
							|  |  |  |     async getLabel(name) { | 
					
						
							| 
									
										
										
										
											2018-08-07 13:33:10 +02:00
										 |  |  |         const attributes = await this.getAttributes(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return attributes.find(attr => attr.type === 'label' && attr.name === name); | 
					
						
							| 
									
										
										
										
											2018-01-29 17:41:59 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-13 10:59:31 +02:00
										 |  |  |     async getLabelValue(name) { | 
					
						
							|  |  |  |         const label = await this.getLabel(name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return label ? label.value : null; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-13 13:53:08 +02:00
										 |  |  |     async toggleLabel(enabled, name, value = "") { | 
					
						
							|  |  |  |         if (enabled) { | 
					
						
							|  |  |  |             await this.setLabel(name, value); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             await this.removeLabel(name, value); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-13 10:59:31 +02:00
										 |  |  |     async setLabel(name, value = "") { | 
					
						
							| 
									
										
										
										
											2018-08-13 13:53:08 +02:00
										 |  |  |         const attributes = await this.getOwnedAttributes(); | 
					
						
							|  |  |  |         let label = attributes.find(attr => attr.type === 'label' && attr.value === value); | 
					
						
							| 
									
										
										
										
											2018-08-13 10:59:31 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (!label) { | 
					
						
							|  |  |  |             label = new Attribute({ | 
					
						
							|  |  |  |                 noteId: this.noteId, | 
					
						
							|  |  |  |                 type: 'label', | 
					
						
							| 
									
										
										
										
											2018-08-13 13:53:08 +02:00
										 |  |  |                 name: name, | 
					
						
							|  |  |  |                 value: value | 
					
						
							| 
									
										
										
										
											2018-08-13 10:59:31 +02:00
										 |  |  |             }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-13 13:53:08 +02:00
										 |  |  |             await label.save(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-08-13 10:59:31 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-13 13:53:08 +02:00
										 |  |  |     async removeLabel(name, value = "") { | 
					
						
							|  |  |  |         const attributes = await this.getOwnedAttributes(); | 
					
						
							| 
									
										
										
										
											2018-08-13 10:59:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-13 13:53:08 +02:00
										 |  |  |         for (const attribute of attributes) { | 
					
						
							|  |  |  |             if (attribute.type === 'label' && (!value || value === attribute.value)) { | 
					
						
							|  |  |  |                 attribute.isDeleted = true; | 
					
						
							|  |  |  |                 await attribute.save(); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2018-08-13 10:59:31 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-29 18:34:59 -05:00
										 |  |  |     async getRevisions() { | 
					
						
							| 
									
										
										
										
											2018-03-31 10:51:37 -04:00
										 |  |  |         return await repository.getEntities("SELECT * FROM note_revisions WHERE noteId = ?", [this.noteId]); | 
					
						
							| 
									
										
										
										
											2018-01-29 18:34:59 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-31 22:15:06 -04:00
										 |  |  |     async getNoteImages() { | 
					
						
							|  |  |  |         return await repository.getEntities("SELECT * FROM note_images WHERE noteId = ? AND isDeleted = 0", [this.noteId]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-31 23:08:22 -04:00
										 |  |  |     async getBranches() { | 
					
						
							| 
									
										
										
										
											2018-03-31 10:51:37 -04:00
										 |  |  |         return await repository.getEntities("SELECT * FROM branches WHERE isDeleted = 0 AND noteId = ?", [this.noteId]); | 
					
						
							| 
									
										
										
										
											2018-01-28 23:16:50 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-29 23:35:36 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-31 23:08:22 -04:00
										 |  |  |     async getChildNote(name) { | 
					
						
							| 
									
										
										
										
											2018-03-31 10:51:37 -04:00
										 |  |  |         return await repository.getEntity(`
 | 
					
						
							| 
									
										
										
										
											2018-02-24 14:42:52 -05:00
										 |  |  |           SELECT notes.*  | 
					
						
							| 
									
										
										
										
											2018-03-24 21:39:15 -04:00
										 |  |  |           FROM branches  | 
					
						
							| 
									
										
										
										
											2018-02-24 14:42:52 -05:00
										 |  |  |             JOIN notes USING(noteId)  | 
					
						
							|  |  |  |           WHERE notes.isDeleted = 0 | 
					
						
							| 
									
										
										
										
											2018-03-24 21:39:15 -04:00
										 |  |  |                 AND branches.isDeleted = 0 | 
					
						
							|  |  |  |                 AND branches.parentNoteId = ? | 
					
						
							| 
									
										
										
										
											2018-02-24 14:42:52 -05:00
										 |  |  |                 AND notes.title = ?`, [this.noteId, name]);
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-31 23:08:22 -04:00
										 |  |  |     async getChildNotes() { | 
					
						
							| 
									
										
										
										
											2018-03-31 10:51:37 -04:00
										 |  |  |         return await repository.getEntities(`
 | 
					
						
							| 
									
										
										
										
											2018-02-24 14:42:52 -05:00
										 |  |  |           SELECT notes.*  | 
					
						
							| 
									
										
										
										
											2018-03-24 21:39:15 -04:00
										 |  |  |           FROM branches  | 
					
						
							| 
									
										
										
										
											2018-02-24 14:42:52 -05:00
										 |  |  |             JOIN notes USING(noteId)  | 
					
						
							|  |  |  |           WHERE notes.isDeleted = 0 | 
					
						
							| 
									
										
										
										
											2018-03-24 21:39:15 -04:00
										 |  |  |                 AND branches.isDeleted = 0 | 
					
						
							|  |  |  |                 AND branches.parentNoteId = ? | 
					
						
							|  |  |  |           ORDER BY branches.notePosition`, [this.noteId]);
 | 
					
						
							| 
									
										
										
										
											2018-02-24 14:42:52 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-31 23:08:22 -04:00
										 |  |  |     async getChildBranches() { | 
					
						
							|  |  |  |         return await repository.getEntities(`
 | 
					
						
							|  |  |  |           SELECT branches.*  | 
					
						
							|  |  |  |           FROM branches  | 
					
						
							|  |  |  |           WHERE branches.isDeleted = 0 | 
					
						
							|  |  |  |                 AND branches.parentNoteId = ? | 
					
						
							|  |  |  |           ORDER BY branches.notePosition`, [this.noteId]);
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     async getParentNotes() { | 
					
						
							| 
									
										
										
										
											2018-03-31 10:51:37 -04:00
										 |  |  |         return await repository.getEntities(`
 | 
					
						
							| 
									
										
										
										
											2018-02-24 14:42:52 -05:00
										 |  |  |           SELECT parent_notes.*  | 
					
						
							|  |  |  |           FROM  | 
					
						
							| 
									
										
										
										
											2018-03-24 21:39:15 -04:00
										 |  |  |             branches AS child_tree  | 
					
						
							| 
									
										
										
										
											2018-02-24 14:42:52 -05:00
										 |  |  |             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]);
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-29 23:35:36 -05:00
										 |  |  |     beforeSaving() { | 
					
						
							| 
									
										
										
										
											2018-04-11 00:10:11 -04:00
										 |  |  |         if (this.isJson() && this.jsonContent) { | 
					
						
							| 
									
										
										
										
											2018-03-31 22:15:06 -04:00
										 |  |  |             this.content = JSON.stringify(this.jsonContent, null, '\t'); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-29 23:35:36 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (this.isProtected) { | 
					
						
							| 
									
										
										
										
											2018-04-20 00:12:01 -04:00
										 |  |  |             protectedSessionService.encryptNote(this); | 
					
						
							| 
									
										
										
										
											2018-01-29 23:35:36 -05:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-03-31 22:15:06 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-01 17:38:24 -04:00
										 |  |  |         if (!this.isDeleted) { | 
					
						
							|  |  |  |             this.isDeleted = false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-31 22:15:06 -04:00
										 |  |  |         if (!this.dateCreated) { | 
					
						
							| 
									
										
										
										
											2018-04-02 20:46:46 -04:00
										 |  |  |             this.dateCreated = dateUtils.nowDate(); | 
					
						
							| 
									
										
										
										
											2018-03-31 22:15:06 -04:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-12 20:04:48 +02:00
										 |  |  |         super.beforeSaving(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (this.isChanged) { | 
					
						
							|  |  |  |             this.dateModified = dateUtils.nowDate(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-29 23:35:36 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-28 23:16:50 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module.exports = Note; |