| 
									
										
										
										
											2022-02-12 22:20:15 +01:00
										 |  |  | const fs = require("fs"); | 
					
						
							|  |  |  | const sanitize = require("sanitize-filename"); | 
					
						
							| 
									
										
										
										
											2024-02-16 22:44:12 +02:00
										 |  |  | const sql = require('./sql'); | 
					
						
							| 
									
										
										
										
											2023-11-22 19:34:48 +01:00
										 |  |  | const decryptService = require('./decrypt.js'); | 
					
						
							|  |  |  | const dataKeyService = require('./data_key.js'); | 
					
						
							|  |  |  | const extensionService = require('./extension.js'); | 
					
						
							| 
									
										
										
										
											2022-02-12 22:20:15 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | function dumpDocument(documentPath, targetPath, options) { | 
					
						
							|  |  |  |     const stats = { | 
					
						
							|  |  |  |         succeeded: 0, | 
					
						
							|  |  |  |         failed: 0, | 
					
						
							|  |  |  |         protected: 0, | 
					
						
							|  |  |  |         deleted: 0 | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     validatePaths(documentPath, targetPath); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     sql.openDatabase(documentPath); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const dataKey = dataKeyService.getDataKey(options.password); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const existingPaths = {}; | 
					
						
							|  |  |  |     const noteIdToPath = {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     dumpNote(targetPath, 'root'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     printDumpResults(stats, options); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function dumpNote(targetPath, noteId) { | 
					
						
							|  |  |  |         console.log(`Reading note '${noteId}'`); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-05 09:23:42 +02:00
										 |  |  |         let childTargetPath, noteRow, fileNameWithPath; | 
					
						
							| 
									
										
										
										
											2022-02-12 22:20:15 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         try { | 
					
						
							| 
									
										
										
										
											2023-06-05 09:23:42 +02:00
										 |  |  |             noteRow = sql.getRow("SELECT * FROM notes WHERE noteId = ?", [noteId]); | 
					
						
							| 
									
										
										
										
											2022-02-12 22:20:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-05 09:23:42 +02:00
										 |  |  |             if (noteRow.isDeleted) { | 
					
						
							| 
									
										
										
										
											2022-02-12 22:20:15 +01:00
										 |  |  |                 stats.deleted++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (!options.includeDeleted) { | 
					
						
							|  |  |  |                     console.log(`Note '${noteId}' is deleted and --include-deleted option is not used, skipping.`); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-05 09:23:42 +02:00
										 |  |  |             if (noteRow.isProtected) { | 
					
						
							| 
									
										
										
										
											2022-02-12 22:20:15 +01:00
										 |  |  |                 stats.protected++; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-05 09:23:42 +02:00
										 |  |  |                 noteRow.title = decryptService.decryptString(dataKey, noteRow.title); | 
					
						
							| 
									
										
										
										
											2022-02-12 22:20:15 +01:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-05 09:23:42 +02:00
										 |  |  |             let safeTitle = sanitize(noteRow.title); | 
					
						
							| 
									
										
										
										
											2022-02-12 22:20:15 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if (safeTitle.length > 20) { | 
					
						
							|  |  |  |                 safeTitle = safeTitle.substring(0, 20); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             childTargetPath = targetPath + '/' + safeTitle; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for (let i = 1; i < 100000 && childTargetPath in existingPaths; i++) { | 
					
						
							|  |  |  |                 childTargetPath = targetPath + '/' + safeTitle + '_' + i; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             existingPaths[childTargetPath] = true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-05 09:23:42 +02:00
										 |  |  |             if (noteRow.noteId in noteIdToPath) { | 
					
						
							|  |  |  |                 const message = `Note '${noteId}' has been already dumped to ${noteIdToPath[noteRow.noteId]}`; | 
					
						
							| 
									
										
										
										
											2022-02-12 22:20:15 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 console.log(message); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 fs.writeFileSync(childTargetPath, message); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-05 09:23:42 +02:00
										 |  |  |             let {content} = sql.getRow("SELECT content FROM blobs WHERE blobId = ?", [noteRow.blobId]); | 
					
						
							| 
									
										
										
										
											2022-02-12 22:20:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-05 09:23:42 +02:00
										 |  |  |             if (content !== null && noteRow.isProtected && dataKey) { | 
					
						
							| 
									
										
										
										
											2022-02-12 22:20:15 +01:00
										 |  |  |                 content = decryptService.decrypt(dataKey, content); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (isContentEmpty(content)) { | 
					
						
							|  |  |  |                 console.log(`Note '${noteId}' is empty, skipping.`); | 
					
						
							|  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2023-06-05 09:23:42 +02:00
										 |  |  |                 fileNameWithPath = extensionService.getFileName(noteRow, childTargetPath, safeTitle); | 
					
						
							| 
									
										
										
										
											2022-02-12 22:20:15 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 fs.writeFileSync(fileNameWithPath, content); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 stats.succeeded++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 console.log(`Dumped note '${noteId}' into ${fileNameWithPath} successfully.`); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             noteIdToPath[noteId] = childTargetPath; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         catch (e) { | 
					
						
							|  |  |  |             console.error(`DUMPERROR: Writing '${noteId}' failed with error '${e.message}':\n${e.stack}`); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             stats.failed++; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const childNoteIds = sql.getColumn("SELECT noteId FROM branches WHERE parentNoteId = ?", [noteId]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (childNoteIds.length > 0) { | 
					
						
							|  |  |  |             if (childTargetPath === fileNameWithPath) { | 
					
						
							|  |  |  |                 childTargetPath += '_dir'; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             try { | 
					
						
							|  |  |  |                 fs.mkdirSync(childTargetPath, {recursive: true}); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             catch (e) { | 
					
						
							|  |  |  |                 console.error(`DUMPERROR: Creating directory ${childTargetPath} failed with error '${e.message}'`); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for (const childNoteId of childNoteIds) { | 
					
						
							|  |  |  |                 dumpNote(childTargetPath, childNoteId); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function printDumpResults(stats, options) { | 
					
						
							|  |  |  |     console.log('\n----------------------- STATS -----------------------'); | 
					
						
							|  |  |  |     console.log('Successfully dumpted notes:   ', stats.succeeded.toString().padStart(5, ' ')); | 
					
						
							|  |  |  |     console.log('Protected notes:              ', stats.protected.toString().padStart(5, ' '), options.password ? '' : '(skipped)'); | 
					
						
							|  |  |  |     console.log('Failed notes:                 ', stats.failed.toString().padStart(5, ' ')); | 
					
						
							|  |  |  |     console.log('Deleted notes:                ', stats.deleted.toString().padStart(5, ' '), options.includeDeleted ? "(dumped)" : "(at least, skipped)"); | 
					
						
							|  |  |  |     console.log('-----------------------------------------------------'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!options.password && stats.protected > 0) { | 
					
						
							|  |  |  |         console.log("\nWARNING: protected notes are present in the document but no password has been provided. Protected notes have not been dumped."); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function isContentEmpty(content) { | 
					
						
							|  |  |  |     if (!content) { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (typeof content === "string") { | 
					
						
							|  |  |  |         return !content.trim() || content.trim() === '<p></p>'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (Buffer.isBuffer(content)) { | 
					
						
							|  |  |  |         return content.length === 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function validatePaths(documentPath, targetPath) { | 
					
						
							|  |  |  |     if (!fs.existsSync(documentPath)) { | 
					
						
							|  |  |  |         console.error(`Path to document '${documentPath}' has not been found. Run with --help to see usage.`); | 
					
						
							|  |  |  |         process.exit(1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!fs.existsSync(targetPath)) { | 
					
						
							|  |  |  |         const ret = fs.mkdirSync(targetPath, {recursive: true}); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!ret) { | 
					
						
							|  |  |  |             console.error(`Target path '${targetPath}' could not be created. Run with --help to see usage.`); | 
					
						
							|  |  |  |             process.exit(1); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module.exports = { | 
					
						
							|  |  |  |     dumpDocument | 
					
						
							|  |  |  | }; |