/*\ title: $:/plugins/tiddlywiki/multiwikiserver/store/sql-tiddler-store.js type: application/javascript module-type: library Higher level functions to perform basic tiddler operations with a sqlite3 database. This class is largely a wrapper for the sql-tiddler-database.js class, adding the following functionality: * Validating requests (eg bag and recipe name constraints) * Synchronising bag and recipe names to the admin wiki * Handling large tiddlers as attachments \*/ (function() { /* Create a tiddler store. Options include: databasePath - path to the database file (can be ":memory:" to get a temporary database) adminWiki - reference to $tw.Wiki object used for configuration attachmentStore - reference to associated attachment store engine - wasm | better */ function SqlTiddlerStore(options) { options = options || {}; this.attachmentStore = options.attachmentStore; this.adminWiki = options.adminWiki || $tw.wiki; this.eventListeners = {}; // Hashmap by type of array of event listener functions this.eventOutstanding = {}; // Hashmap by type of boolean true of outstanding events // Create the database this.databasePath = options.databasePath || ":memory:"; var SqlTiddlerDatabase = require("$:/plugins/tiddlywiki/multiwikiserver/store/sql-tiddler-database.js").SqlTiddlerDatabase; this.sqlTiddlerDatabase = new SqlTiddlerDatabase({ databasePath: this.databasePath, engine: options.engine }); this.sqlTiddlerDatabase.createTables(); } SqlTiddlerStore.prototype.addEventListener = function(type,listener) { this.eventListeners[type] = this.eventListeners[type] || []; this.eventListeners[type].push(listener); }; SqlTiddlerStore.prototype.removeEventListener = function(type,listener) { const listeners = this.eventListeners[type]; if(listeners) { var p = listeners.indexOf(listener); if(p !== -1) { listeners.splice(p,1); } } }; SqlTiddlerStore.prototype.dispatchEvent = function(type /*, args */) { const self = this; if(!this.eventOutstanding[type]) { $tw.utils.nextTick(function() { self.eventOutstanding[type] = false; const args = Array.prototype.slice.call(arguments,1), listeners = self.eventListeners[type]; if(listeners) { for(var p=0; p 256) { return "Too long"; } // Removed ~ from this list temporarily if(allowPrivilegedCharacters) { if(!(/^[^\s\u00A0\x00-\x1F\x7F`!@#%^&*()+={}\[\];\'\"<>,\\\?]+$/g.test(name))) { return "Invalid character(s)"; } } else { if(!(/^[^\s\u00A0\x00-\x1F\x7F`!@#$%^&*()+={}\[\];:\'\"<>.,\/\\\?]+$/g.test(name))) { return "Invalid character(s)"; } } return null; }; /* Returns null if the argument is an array of valid bag/recipe names, or a string error message if not */ SqlTiddlerStore.prototype.validateItemNames = function(names,allowPrivilegedCharacters) { if(!$tw.utils.isArray(names)) { return "Not a valid array"; } var errors = []; for(const name of names) { const result = this.validateItemName(name,allowPrivilegedCharacters); if(result && errors.indexOf(result) === -1) { errors.push(result); } } if(errors.length === 0) { return null; } else { return errors.join("\n"); } }; SqlTiddlerStore.prototype.close = function() { this.sqlTiddlerDatabase.close(); this.sqlTiddlerDatabase = undefined; }; /* Given tiddler fields, tiddler_id and a bag_name, return the tiddler fields after the following process: - Apply the tiddler_id as the revision field - Apply the bag_name as the bag field */ SqlTiddlerStore.prototype.processOutgoingTiddler = function(tiddlerFields,tiddler_id,bag_name,attachment_blob) { if(attachment_blob !== null) { return $tw.utils.extend( {}, tiddlerFields, { text: undefined, _canonical_uri: `/bags/${$tw.utils.encodeURIComponentExtended(bag_name)}/tiddlers/${$tw.utils.encodeURIComponentExtended(tiddlerFields.title)}/blob` } ); } else { return tiddlerFields; } }; /* */ SqlTiddlerStore.prototype.processIncomingTiddler = function(tiddlerFields, existing_attachment_blob, existing_canonical_uri) { let attachmentSizeLimit = $tw.utils.parseNumber(this.adminWiki.getTiddlerText("$:/config/MultiWikiServer/AttachmentSizeLimit")); if(attachmentSizeLimit < 100 * 1024) { attachmentSizeLimit = 100 * 1024; } const attachmentsEnabled = this.adminWiki.getTiddlerText("$:/config/MultiWikiServer/EnableAttachments", "yes") === "yes"; const contentTypeInfo = $tw.config.contentTypeInfo[tiddlerFields.type || "text/vnd.tiddlywiki"]; const isBinary = !!contentTypeInfo && contentTypeInfo.encoding === "base64"; let shouldProcessAttachment = tiddlerFields.text && tiddlerFields.text.length <= attachmentSizeLimit; if(existing_attachment_blob) { const fileSize = this.attachmentStore.getAttachmentFileSize(existing_attachment_blob); if(fileSize <= attachmentSizeLimit) { const existingAttachmentMeta = this.attachmentStore.getAttachmentMetadata(existing_attachment_blob); const hasCanonicalField = !!tiddlerFields._canonical_uri; const skipAttachment = hasCanonicalField && (tiddlerFields._canonical_uri === (existingAttachmentMeta?._canonical_uri || existing_canonical_uri)); shouldProcessAttachment = !skipAttachment; } else { shouldProcessAttachment = false; } } if(attachmentsEnabled && isBinary && shouldProcessAttachment) { const attachment_blob = existing_attachment_blob || this.attachmentStore.saveAttachment({ text: tiddlerFields.text, type: tiddlerFields.type, reference: tiddlerFields.title, _canonical_uri: tiddlerFields._canonical_uri }); if(tiddlerFields?._canonical_uri) { delete tiddlerFields._canonical_uri; } return { tiddlerFields: Object.assign({}, tiddlerFields, { text: undefined }), attachment_blob: attachment_blob }; } else { return { tiddlerFields: tiddlerFields, attachment_blob: existing_attachment_blob }; } }; SqlTiddlerStore.prototype.saveTiddlersFromPath = function(tiddler_files_path,bag_name) { var self = this; this.sqlTiddlerDatabase.transaction(function() { // Clear out the bag self.deleteAllTiddlersInBag(bag_name); // Get the tiddlers var path = require("path"); var tiddlersFromPath = $tw.loadTiddlersFromPath(path.resolve($tw.boot.corePath,$tw.config.editionsPath,tiddler_files_path)); // Save the tiddlers for(const tiddlersFromFile of tiddlersFromPath) { for(const tiddler of tiddlersFromFile.tiddlers) { self.saveBagTiddler(tiddler,bag_name,null); } } }); self.dispatchEvent("change"); }; SqlTiddlerStore.prototype.listBags = function() { return this.sqlTiddlerDatabase.listBags(); }; /* Options include: allowPrivilegedCharacters - allows "$", ":" and "/" to appear in recipe name */ SqlTiddlerStore.prototype.createBag = function(bag_name,description,options) { options = options || {}; var self = this; return this.sqlTiddlerDatabase.transaction(function() { const validationBagName = self.validateItemName(bag_name,options.allowPrivilegedCharacters); if(validationBagName) { return {message: validationBagName}; } self.sqlTiddlerDatabase.createBag(bag_name,description); self.dispatchEvent("change"); return null; }); }; SqlTiddlerStore.prototype.listRecipes = function() { return this.sqlTiddlerDatabase.listRecipes(); }; /* Returns null on success, or {message:} on error Options include: allowPrivilegedCharacters - allows "$", ":" and "/" to appear in recipe name */ SqlTiddlerStore.prototype.createRecipe = function(recipe_name,bag_names,description,options) { bag_names = bag_names || []; description = description || ""; options = options || {}; const validationRecipeName = this.validateItemName(recipe_name,options.allowPrivilegedCharacters); if(validationRecipeName) { return {message: validationRecipeName}; } if(bag_names.length === 0) { return {message: "Recipes must contain at least one bag"}; } var self = this; return this.sqlTiddlerDatabase.transaction(function() { self.sqlTiddlerDatabase.createRecipe(recipe_name,bag_names,description); self.dispatchEvent("change"); return null; }); }; /* Returns {tiddler_id:} */ SqlTiddlerStore.prototype.saveBagTiddler = function(incomingTiddlerFields,bag_name) { let _canonical_uri; const existing_attachment_blob = this.sqlTiddlerDatabase.getBagTiddlerAttachmentBlob(incomingTiddlerFields.title,bag_name) if(existing_attachment_blob) { _canonical_uri = `/bags/${$tw.utils.encodeURIComponentExtended(bag_name)}/tiddlers/${$tw.utils.encodeURIComponentExtended(incomingTiddlerFields.title)}/blob` } const {tiddlerFields, attachment_blob} = this.processIncomingTiddler(incomingTiddlerFields,existing_attachment_blob,_canonical_uri); const result = this.sqlTiddlerDatabase.saveBagTiddler(tiddlerFields,bag_name,attachment_blob); this.dispatchEvent("change"); return result; }; /* Create a tiddler in a bag adopting the specified file as the attachment. The attachment file must be on the same disk as the attachment store Options include: filepath - filepath to the attachment file hash - string hash of the attachment file type - content type of file as uploaded Returns {tiddler_id:} */ SqlTiddlerStore.prototype.saveBagTiddlerWithAttachment = function(incomingTiddlerFields,bag_name,options) { const attachment_blob = this.attachmentStore.adoptAttachment(options.filepath,options.type,options.hash,options._canonical_uri); if(attachment_blob) { const result = this.sqlTiddlerDatabase.saveBagTiddler(incomingTiddlerFields,bag_name,attachment_blob); this.dispatchEvent("change"); return result; } else { return null; } }; /* Returns {tiddler_id:,bag_name:} */ SqlTiddlerStore.prototype.saveRecipeTiddler = function(incomingTiddlerFields,recipe_name) { const existing_attachment_blob = this.sqlTiddlerDatabase.getRecipeTiddlerAttachmentBlob(incomingTiddlerFields.title,recipe_name) const {tiddlerFields, attachment_blob} = this.processIncomingTiddler(incomingTiddlerFields,existing_attachment_blob,incomingTiddlerFields._canonical_uri); const result = this.sqlTiddlerDatabase.saveRecipeTiddler(tiddlerFields,recipe_name,attachment_blob); this.dispatchEvent("change"); return result; }; SqlTiddlerStore.prototype.deleteTiddler = function(title,bag_name) { const result = this.sqlTiddlerDatabase.deleteTiddler(title,bag_name); this.dispatchEvent("change"); return result; }; /* returns {tiddler_id:,tiddler:} */ SqlTiddlerStore.prototype.getBagTiddler = function(title,bag_name) { var tiddlerInfo = this.sqlTiddlerDatabase.getBagTiddler(title,bag_name); if(tiddlerInfo) { return Object.assign( {}, tiddlerInfo, { tiddler: this.processOutgoingTiddler(tiddlerInfo.tiddler,tiddlerInfo.tiddler_id,bag_name,tiddlerInfo.attachment_blob) }); } else { return null; } }; /* Get an attachment ready to stream. Returns null if there is an error or: tiddler_id: revision of tiddler stream: stream of file type: type of file Returns {tiddler_id:,bag_name:} */ SqlTiddlerStore.prototype.getBagTiddlerStream = function(title,bag_name) { const tiddlerInfo = this.sqlTiddlerDatabase.getBagTiddler(title,bag_name); if(tiddlerInfo) { if(tiddlerInfo.attachment_blob) { return $tw.utils.extend( {}, this.attachmentStore.getAttachmentStream(tiddlerInfo.attachment_blob), { tiddler_id: tiddlerInfo.tiddler_id, bag_name: bag_name } ); } else { const { Readable } = require('stream'); const stream = new Readable(); stream._read = function() { // Push data const type = tiddlerInfo.tiddler.type || "text/plain"; stream.push(tiddlerInfo.tiddler.text || "",($tw.config.contentTypeInfo[type] ||{encoding: "utf8"}).encoding); // Push null to indicate the end of the stream stream.push(null); }; return { tiddler_id: tiddlerInfo.tiddler_id, bag_name: bag_name, stream: stream, type: tiddlerInfo.tiddler.type || "text/plain" } } } else { return null; } }; /* Returns {bag_name:, tiddler: {fields}, tiddler_id:} */ SqlTiddlerStore.prototype.getRecipeTiddler = function(title,recipe_name) { var tiddlerInfo = this.sqlTiddlerDatabase.getRecipeTiddler(title,recipe_name); if(tiddlerInfo) { return Object.assign( {}, tiddlerInfo, { tiddler: this.processOutgoingTiddler(tiddlerInfo.tiddler,tiddlerInfo.tiddler_id,tiddlerInfo.bag_name,tiddlerInfo.attachment_blob) }); } else { return null; } }; /* Get the titles of the tiddlers in a bag. Returns an empty array for bags that do not exist */ SqlTiddlerStore.prototype.getBagTiddlers = function(bag_name) { return this.sqlTiddlerDatabase.getBagTiddlers(bag_name); }; /* Get the tiddler_id of the newest tiddler in a bag. Returns null for bags that do not exist */ SqlTiddlerStore.prototype.getBagLastTiddlerId = function(bag_name) { return this.sqlTiddlerDatabase.getBagLastTiddlerId(bag_name); }; /* Get the titles of the tiddlers in a recipe as {title:,bag_name:}. Returns null for recipes that do not exist */ SqlTiddlerStore.prototype.getRecipeTiddlers = function(recipe_name,options) { return this.sqlTiddlerDatabase.getRecipeTiddlers(recipe_name,options); }; /* Get the tiddler_id of the newest tiddler in a recipe. Returns null for recipes that do not exist */ SqlTiddlerStore.prototype.getRecipeLastTiddlerId = function(recipe_name) { return this.sqlTiddlerDatabase.getRecipeLastTiddlerId(recipe_name); }; SqlTiddlerStore.prototype.deleteAllTiddlersInBag = function(bag_name) { var self = this; return this.sqlTiddlerDatabase.transaction(function() { const result = self.sqlTiddlerDatabase.deleteAllTiddlersInBag(bag_name); self.dispatchEvent("change"); return result; }); }; /* Get the names of the bags in a recipe. Returns an empty array for recipes that do not exist */ SqlTiddlerStore.prototype.getRecipeBags = function(recipe_name) { return this.sqlTiddlerDatabase.getRecipeBags(recipe_name); }; exports.SqlTiddlerStore = SqlTiddlerStore; })();