diff --git a/plugins/tiddlywiki/multiwikiserver/modules/routes/handlers/post-bag.js b/plugins/tiddlywiki/multiwikiserver/modules/routes/handlers/post-bag.js index dcf4166ff..c84c9eda5 100644 --- a/plugins/tiddlywiki/multiwikiserver/modules/routes/handlers/post-bag.js +++ b/plugins/tiddlywiki/multiwikiserver/modules/routes/handlers/post-bag.js @@ -13,52 +13,30 @@ description \*/ (function() { - /*jslint node: true, browser: true */ - /*global $tw: false */ - "use strict"; +/*jslint node: true, browser: true */ +/*global $tw: false */ +"use strict"; + +exports.method = "POST"; + +exports.path = /^\/bags$/; + +exports.bodyFormat = "www-form-urlencoded"; + +exports.csrfDisable = true; + +exports.useACL = true; + +exports.entityName = "bag" + +exports.handler = function(request,response,state) { + var server = state.server, + sqlTiddlerDatabase = server.sqlTiddlerDatabase; - exports.method = "POST"; - - exports.path = /^\/bags$/; - - exports.bodyFormat = "www-form-urlencoded"; - - exports.csrfDisable = true; - - exports.useACL = true; - - exports.entityName = "bag" - - exports.handler = function(request,response,state) { - var server = state.server, - sqlTiddlerDatabase = server.sqlTiddlerDatabase; - - // Handle DELETE request - if(state.data._method === "DELETE") { - if(state.data.bag_name) { - const result = $tw.mws.store.deleteBag(state.data.bag_name); - if(!result) { - state.sendResponse(302,{ - "Content-Type": "text/plain", - "Location": "/" - }); - } else { - state.sendResponse(400,{ - "Content-Type": "text/plain" - }, - result.message, - "utf8"); - } - } else { - state.sendResponse(400,{ - "Content-Type": "text/plain" - }); - } - return; - } - + // Handle DELETE request + if(state.data._method === "DELETE") { if(state.data.bag_name) { - const result = $tw.mws.store.createBag(state.data.bag_name,state.data.description); + const result = $tw.mws.store.deleteBag(state.data.bag_name); if(!result) { state.sendResponse(302,{ "Content-Type": "text/plain", @@ -76,7 +54,28 @@ description "Content-Type": "text/plain" }); } - }; + return; + } - }()); - \ No newline at end of file + if(state.data.bag_name) { + const result = $tw.mws.store.createBag(state.data.bag_name,state.data.description); + if(!result) { + state.sendResponse(302,{ + "Content-Type": "text/plain", + "Location": "/" + }); + } else { + state.sendResponse(400,{ + "Content-Type": "text/plain" + }, + result.message, + "utf8"); + } + } else { + state.sendResponse(400,{ + "Content-Type": "text/plain" + }); + } +}; + +}()); diff --git a/plugins/tiddlywiki/multiwikiserver/modules/routes/handlers/post-recipe.js b/plugins/tiddlywiki/multiwikiserver/modules/routes/handlers/post-recipe.js index adf11a14e..8304a12c3 100644 --- a/plugins/tiddlywiki/multiwikiserver/modules/routes/handlers/post-recipe.js +++ b/plugins/tiddlywiki/multiwikiserver/modules/routes/handlers/post-recipe.js @@ -14,57 +14,31 @@ bag_names: space separated list of bags \*/ (function() { - /*jslint node: true, browser: true */ - /*global $tw: false */ - "use strict"; +/*jslint node: true, browser: true */ +/*global $tw: false */ +"use strict"; + +exports.method = "POST"; + +exports.path = /^\/recipes$/; + +exports.bodyFormat = "www-form-urlencoded"; + +exports.csrfDisable = true; + +exports.useACL = true; + +exports.entityName = "recipe" + +exports.handler = function(request,response,state) { + var server = state.server, + sqlTiddlerDatabase = server.sqlTiddlerDatabase; - exports.method = "POST"; - - exports.path = /^\/recipes$/; - - exports.bodyFormat = "www-form-urlencoded"; - - exports.csrfDisable = true; - - exports.useACL = true; - - exports.entityName = "recipe" - - exports.handler = function(request,response,state) { - var server = state.server, - sqlTiddlerDatabase = server.sqlTiddlerDatabase; - - // Check and handle if this is a DELETE request - if(state.data._method === "DELETE") { - if(state.data.recipe_name && state.data.bag_names) { - const result = sqlTiddlerDatabase.deleteRecipe(state.data.recipe_name); - if(!result) { - state.sendResponse(302,{ - "Content-Type": "text/plain", - "Location": "/" - }); - } else { - state.sendResponse(400,{ - "Content-Type": "text/plain" - }, - result.message, - "utf8"); - } - } else { - state.sendResponse(400,{ - "Content-Type": "text/plain" - }); - } - return; - } - - // Handle POST request (original code) + // Check and handle if this is a DELETE request + if(state.data._method === "DELETE") { if(state.data.recipe_name && state.data.bag_names) { - const result = $tw.mws.store.createRecipe(state.data.recipe_name,$tw.utils.parseStringArray(state.data.bag_names),state.data.description); + const result = sqlTiddlerDatabase.deleteRecipe(state.data.recipe_name); if(!result) { - if(state.authenticatedUser) { - sqlTiddlerDatabase.assignRecipeToUser(state.data.recipe_name,state.authenticatedUser.user_id); - } state.sendResponse(302,{ "Content-Type": "text/plain", "Location": "/" @@ -81,6 +55,32 @@ bag_names: space separated list of bags "Content-Type": "text/plain" }); } - }; - - }()); \ No newline at end of file + return; + } + + // Handle POST request (original code) + if(state.data.recipe_name && state.data.bag_names) { + const result = $tw.mws.store.createRecipe(state.data.recipe_name,$tw.utils.parseStringArray(state.data.bag_names),state.data.description); + if(!result) { + if(state.authenticatedUser) { + sqlTiddlerDatabase.assignRecipeToUser(state.data.recipe_name,state.authenticatedUser.user_id); + } + state.sendResponse(302,{ + "Content-Type": "text/plain", + "Location": "/" + }); + } else { + state.sendResponse(400,{ + "Content-Type": "text/plain" + }, + result.message, + "utf8"); + } + } else { + state.sendResponse(400,{ + "Content-Type": "text/plain" + }); + } +}; + +}()); \ No newline at end of file diff --git a/plugins/tiddlywiki/multiwikiserver/modules/store/sql-tiddler-database.js b/plugins/tiddlywiki/multiwikiserver/modules/store/sql-tiddler-database.js index 19bd619cd..d92680858 100644 --- a/plugins/tiddlywiki/multiwikiserver/modules/store/sql-tiddler-database.js +++ b/plugins/tiddlywiki/multiwikiserver/modules/store/sql-tiddler-database.js @@ -12,1476 +12,1475 @@ Validation is for the most part left to the caller (function() { - /* - Create a tiddler store. Options include: - - databasePath - path to the database file (can be ":memory:" to get a temporary database) - engine - wasm | better - */ - function SqlTiddlerDatabase(options) { - options = options || {}; - const SqlEngine = require("$:/plugins/tiddlywiki/multiwikiserver/store/sql-engine.js").SqlEngine; - this.engine = new SqlEngine({ - databasePath: options.databasePath, - engine: options.engine - }); - this.entityTypeToTableMap = { - bag: { - table: "bags", - column: "bag_name" - }, - recipe: { - table: "recipes", - column: "recipe_name" - } - }; +/* +Create a tiddler store. Options include: + +databasePath - path to the database file (can be ":memory:" to get a temporary database) +engine - wasm | better +*/ +function SqlTiddlerDatabase(options) { + options = options || {}; + const SqlEngine = require("$:/plugins/tiddlywiki/multiwikiserver/store/sql-engine.js").SqlEngine; + this.engine = new SqlEngine({ + databasePath: options.databasePath, + engine: options.engine + }); + this.entityTypeToTableMap = { + bag: { + table: "bags", + column: "bag_name" + }, + recipe: { + table: "recipes", + column: "recipe_name" + } + }; +} + +SqlTiddlerDatabase.prototype.close = function() { + this.engine.close(); +}; + + +SqlTiddlerDatabase.prototype.transaction = function(fn) { + return this.engine.transaction(fn); +}; + +SqlTiddlerDatabase.prototype.createTables = function() { + this.engine.runStatements([` + -- Users table + CREATE TABLE IF NOT EXISTS users ( + user_id INTEGER PRIMARY KEY AUTOINCREMENT, + username TEXT UNIQUE NOT NULL, + email TEXT UNIQUE NOT NULL, + password TEXT NOT NULL, + created_at TEXT DEFAULT (datetime('now')), + last_login TEXT + ) + `,` + -- User Session table + CREATE TABLE IF NOT EXISTS sessions ( + user_id INTEGER NOT NULL, + session_id TEXT NOT NULL, + created_at TEXT NOT NULL, + last_accessed TEXT NOT NULL, + PRIMARY KEY (user_id), + FOREIGN KEY (user_id) REFERENCES users(user_id) + ) + `,` + -- Groups table + CREATE TABLE IF NOT EXISTS groups ( + group_id INTEGER PRIMARY KEY AUTOINCREMENT, + group_name TEXT UNIQUE NOT NULL, + description TEXT + ) + `,` + -- Roles table + CREATE TABLE IF NOT EXISTS roles ( + role_id INTEGER PRIMARY KEY AUTOINCREMENT, + role_name TEXT UNIQUE NOT NULL, + description TEXT + ) + `,` + -- Permissions table + CREATE TABLE IF NOT EXISTS permissions ( + permission_id INTEGER PRIMARY KEY AUTOINCREMENT, + permission_name TEXT UNIQUE NOT NULL, + description TEXT + ) + `,` + -- User-Group association table + CREATE TABLE IF NOT EXISTS user_groups ( + user_id INTEGER, + group_id INTEGER, + PRIMARY KEY (user_id, group_id), + FOREIGN KEY (user_id) REFERENCES users(user_id), + FOREIGN KEY (group_id) REFERENCES groups(group_id) + ) + `,` + -- User-Role association table + CREATE TABLE IF NOT EXISTS user_roles ( + user_id INTEGER, + role_id INTEGER, + PRIMARY KEY (user_id, role_id), + FOREIGN KEY (user_id) REFERENCES users(user_id), + FOREIGN KEY (role_id) REFERENCES roles(role_id) + ) + `,` + -- Group-Role association table + CREATE TABLE IF NOT EXISTS group_roles ( + group_id INTEGER, + role_id INTEGER, + PRIMARY KEY (group_id, role_id), + FOREIGN KEY (group_id) REFERENCES groups(group_id), + FOREIGN KEY (role_id) REFERENCES roles(role_id) + ) + `,` + -- Role-Permission association table + CREATE TABLE IF NOT EXISTS role_permissions ( + role_id INTEGER, + permission_id INTEGER, + PRIMARY KEY (role_id, permission_id), + FOREIGN KEY (role_id) REFERENCES roles(role_id), + FOREIGN KEY (permission_id) REFERENCES permissions(permission_id) + ) + `,` + -- Bags have names and access control settings + CREATE TABLE IF NOT EXISTS bags ( + bag_id INTEGER PRIMARY KEY AUTOINCREMENT, + bag_name TEXT UNIQUE NOT NULL, + accesscontrol TEXT NOT NULL, + description TEXT NOT NULL + ) + `,` + -- Recipes have names... + CREATE TABLE IF NOT EXISTS recipes ( + recipe_id INTEGER PRIMARY KEY AUTOINCREMENT, + recipe_name TEXT UNIQUE NOT NULL, + description TEXT NOT NULL, + owner_id INTEGER, + FOREIGN KEY (owner_id) REFERENCES users(user_id) + ) + `,` + -- ...and recipes also have an ordered list of bags + CREATE TABLE IF NOT EXISTS recipe_bags ( + recipe_id INTEGER NOT NULL, + bag_id INTEGER NOT NULL, + position INTEGER NOT NULL, + FOREIGN KEY (recipe_id) REFERENCES recipes(recipe_id) ON UPDATE CASCADE ON DELETE CASCADE, + FOREIGN KEY (bag_id) REFERENCES bags(bag_id) ON UPDATE CASCADE ON DELETE CASCADE, + UNIQUE (recipe_id, bag_id) + ) + `,` + -- Tiddlers are contained in bags and have titles + CREATE TABLE IF NOT EXISTS tiddlers ( + tiddler_id INTEGER PRIMARY KEY AUTOINCREMENT, + bag_id INTEGER NOT NULL, + title TEXT NOT NULL, + is_deleted BOOLEAN NOT NULL, + attachment_blob TEXT, -- null or the name of an attachment blob + FOREIGN KEY (bag_id) REFERENCES bags(bag_id) ON UPDATE CASCADE ON DELETE CASCADE, + UNIQUE (bag_id, title) + ) + `,` + -- Tiddlers also have unordered lists of fields, each of which has a name and associated value + CREATE TABLE IF NOT EXISTS fields ( + tiddler_id INTEGER, + field_name TEXT NOT NULL, + field_value TEXT NOT NULL, + FOREIGN KEY (tiddler_id) REFERENCES tiddlers(tiddler_id) ON UPDATE CASCADE ON DELETE CASCADE, + UNIQUE (tiddler_id, field_name) + ) + `,` + -- ACL table (using bag/recipe ids directly) + CREATE TABLE IF NOT EXISTS acl ( + acl_id INTEGER PRIMARY KEY AUTOINCREMENT, + entity_name TEXT NOT NULL, + entity_type TEXT NOT NULL CHECK (entity_type IN ('bag', 'recipe')), + role_id INTEGER, + permission_id INTEGER, + FOREIGN KEY (role_id) REFERENCES roles(role_id), + FOREIGN KEY (permission_id) REFERENCES permissions(permission_id) + ) + `,` + -- Indexes for performance (we can add more as needed based on query patterns) + CREATE INDEX IF NOT EXISTS idx_tiddlers_bag_id ON tiddlers(bag_id) + `,` + CREATE INDEX IF NOT EXISTS idx_fields_tiddler_id ON fields(tiddler_id) + `,` + CREATE INDEX IF NOT EXISTS idx_recipe_bags_recipe_id ON recipe_bags(recipe_id) + `,` + CREATE INDEX IF NOT EXISTS idx_acl_entity_id ON acl(entity_name) + `]); +}; + +SqlTiddlerDatabase.prototype.listBags = function() { + const rows = this.engine.runStatementGetAll(` + SELECT bag_name, bag_id, accesscontrol, description + FROM bags + ORDER BY bag_name + `); + return rows; +}; + +/* +Delete a recipe and its bag associations +*/ +SqlTiddlerDatabase.prototype.deleteRecipe = function(recipe_name) { + // Delete recipe_bags entries first (due to foreign key constraints) + this.engine.runStatement(` + DELETE FROM recipe_bags + WHERE recipe_id = ( + SELECT recipe_id + FROM recipes + WHERE recipe_name = $recipe_name + ) + `, { + $recipe_name: recipe_name + }); + + // Then delete the recipe itself + this.engine.runStatement(` + DELETE FROM recipes + WHERE recipe_name = $recipe_name + `, { + $recipe_name: recipe_name + }); +}; + +/* +Create or update a bag +Returns the bag_id of the bag +*/ +SqlTiddlerDatabase.prototype.createBag = function(bag_name,description,accesscontrol) { + accesscontrol = accesscontrol || ""; + // Run the queries + var bag = this.engine.runStatement(` + INSERT OR IGNORE INTO bags (bag_name, accesscontrol, description) + VALUES ($bag_name, '', '') + `,{ + $bag_name: bag_name + }); + const updateBags = this.engine.runStatement(` + UPDATE bags + SET accesscontrol = $accesscontrol, + description = $description + WHERE bag_name = $bag_name + `,{ + $bag_name: bag_name, + $accesscontrol: accesscontrol, + $description: description + }); + return updateBags.lastInsertRowid; +}; + +/* +Delete a bag and all its associated data +*/ +SqlTiddlerDatabase.prototype.deleteBag = function(bag_name) { + // Delete the bag (cascade will handle related records) + this.engine.runStatement(` + DELETE FROM bags + WHERE bag_name = $bag_name + `, { + $bag_name: bag_name + }); +}; + +/* +Returns array of {recipe_name:,recipe_id:,description:,bag_names: []} +*/ +SqlTiddlerDatabase.prototype.listRecipes = function() { + const rows = this.engine.runStatementGetAll(` + SELECT r.recipe_name, r.recipe_id, r.description, b.bag_name, rb.position + FROM recipes AS r + JOIN recipe_bags AS rb ON rb.recipe_id = r.recipe_id + JOIN bags AS b ON rb.bag_id = b.bag_id + ORDER BY r.recipe_name, rb.position + `); + const results = []; + let currentRecipeName = null, currentRecipeIndex = -1; + for(const row of rows) { + if(row.recipe_name !== currentRecipeName) { + currentRecipeName = row.recipe_name; + currentRecipeIndex += 1; + results.push({ + recipe_name: row.recipe_name, + recipe_id: row.recipe_id, + description: row.description, + bag_names: [] + }); + } + results[currentRecipeIndex].bag_names.push(row.bag_name); } - - SqlTiddlerDatabase.prototype.close = function() { - this.engine.close(); - }; - - - SqlTiddlerDatabase.prototype.transaction = function(fn) { - return this.engine.transaction(fn); - }; - - SqlTiddlerDatabase.prototype.createTables = function() { - this.engine.runStatements([` - -- Users table - CREATE TABLE IF NOT EXISTS users ( - user_id INTEGER PRIMARY KEY AUTOINCREMENT, - username TEXT UNIQUE NOT NULL, - email TEXT UNIQUE NOT NULL, - password TEXT NOT NULL, - created_at TEXT DEFAULT (datetime('now')), - last_login TEXT - ) - `,` - -- User Session table - CREATE TABLE IF NOT EXISTS sessions ( - user_id INTEGER NOT NULL, - session_id TEXT NOT NULL, - created_at TEXT NOT NULL, - last_accessed TEXT NOT NULL, - PRIMARY KEY (user_id), - FOREIGN KEY (user_id) REFERENCES users(user_id) - ) - `,` - -- Groups table - CREATE TABLE IF NOT EXISTS groups ( - group_id INTEGER PRIMARY KEY AUTOINCREMENT, - group_name TEXT UNIQUE NOT NULL, - description TEXT - ) - `,` - -- Roles table - CREATE TABLE IF NOT EXISTS roles ( - role_id INTEGER PRIMARY KEY AUTOINCREMENT, - role_name TEXT UNIQUE NOT NULL, - description TEXT - ) - `,` - -- Permissions table - CREATE TABLE IF NOT EXISTS permissions ( - permission_id INTEGER PRIMARY KEY AUTOINCREMENT, - permission_name TEXT UNIQUE NOT NULL, - description TEXT - ) - `,` - -- User-Group association table - CREATE TABLE IF NOT EXISTS user_groups ( - user_id INTEGER, - group_id INTEGER, - PRIMARY KEY (user_id, group_id), - FOREIGN KEY (user_id) REFERENCES users(user_id), - FOREIGN KEY (group_id) REFERENCES groups(group_id) - ) - `,` - -- User-Role association table - CREATE TABLE IF NOT EXISTS user_roles ( - user_id INTEGER, - role_id INTEGER, - PRIMARY KEY (user_id, role_id), - FOREIGN KEY (user_id) REFERENCES users(user_id), - FOREIGN KEY (role_id) REFERENCES roles(role_id) - ) - `,` - -- Group-Role association table - CREATE TABLE IF NOT EXISTS group_roles ( - group_id INTEGER, - role_id INTEGER, - PRIMARY KEY (group_id, role_id), - FOREIGN KEY (group_id) REFERENCES groups(group_id), - FOREIGN KEY (role_id) REFERENCES roles(role_id) - ) - `,` - -- Role-Permission association table - CREATE TABLE IF NOT EXISTS role_permissions ( - role_id INTEGER, - permission_id INTEGER, - PRIMARY KEY (role_id, permission_id), - FOREIGN KEY (role_id) REFERENCES roles(role_id), - FOREIGN KEY (permission_id) REFERENCES permissions(permission_id) - ) - `,` - -- Bags have names and access control settings - CREATE TABLE IF NOT EXISTS bags ( - bag_id INTEGER PRIMARY KEY AUTOINCREMENT, - bag_name TEXT UNIQUE NOT NULL, - accesscontrol TEXT NOT NULL, - description TEXT NOT NULL - ) - `,` - -- Recipes have names... - CREATE TABLE IF NOT EXISTS recipes ( - recipe_id INTEGER PRIMARY KEY AUTOINCREMENT, - recipe_name TEXT UNIQUE NOT NULL, - description TEXT NOT NULL, - owner_id INTEGER, - FOREIGN KEY (owner_id) REFERENCES users(user_id) - ) - `,` - -- ...and recipes also have an ordered list of bags - CREATE TABLE IF NOT EXISTS recipe_bags ( - recipe_id INTEGER NOT NULL, - bag_id INTEGER NOT NULL, - position INTEGER NOT NULL, - FOREIGN KEY (recipe_id) REFERENCES recipes(recipe_id) ON UPDATE CASCADE ON DELETE CASCADE, - FOREIGN KEY (bag_id) REFERENCES bags(bag_id) ON UPDATE CASCADE ON DELETE CASCADE, - UNIQUE (recipe_id, bag_id) - ) - `,` - -- Tiddlers are contained in bags and have titles - CREATE TABLE IF NOT EXISTS tiddlers ( - tiddler_id INTEGER PRIMARY KEY AUTOINCREMENT, - bag_id INTEGER NOT NULL, - title TEXT NOT NULL, - is_deleted BOOLEAN NOT NULL, - attachment_blob TEXT, -- null or the name of an attachment blob - FOREIGN KEY (bag_id) REFERENCES bags(bag_id) ON UPDATE CASCADE ON DELETE CASCADE, - UNIQUE (bag_id, title) - ) - `,` - -- Tiddlers also have unordered lists of fields, each of which has a name and associated value - CREATE TABLE IF NOT EXISTS fields ( - tiddler_id INTEGER, - field_name TEXT NOT NULL, - field_value TEXT NOT NULL, - FOREIGN KEY (tiddler_id) REFERENCES tiddlers(tiddler_id) ON UPDATE CASCADE ON DELETE CASCADE, - UNIQUE (tiddler_id, field_name) - ) - `,` - -- ACL table (using bag/recipe ids directly) - CREATE TABLE IF NOT EXISTS acl ( - acl_id INTEGER PRIMARY KEY AUTOINCREMENT, - entity_name TEXT NOT NULL, - entity_type TEXT NOT NULL CHECK (entity_type IN ('bag', 'recipe')), - role_id INTEGER, - permission_id INTEGER, - FOREIGN KEY (role_id) REFERENCES roles(role_id), - FOREIGN KEY (permission_id) REFERENCES permissions(permission_id) - ) - `,` - -- Indexes for performance (we can add more as needed based on query patterns) - CREATE INDEX IF NOT EXISTS idx_tiddlers_bag_id ON tiddlers(bag_id) - `,` - CREATE INDEX IF NOT EXISTS idx_fields_tiddler_id ON fields(tiddler_id) - `,` - CREATE INDEX IF NOT EXISTS idx_recipe_bags_recipe_id ON recipe_bags(recipe_id) - `,` - CREATE INDEX IF NOT EXISTS idx_acl_entity_id ON acl(entity_name) - `]); - }; - - SqlTiddlerDatabase.prototype.listBags = function() { - const rows = this.engine.runStatementGetAll(` - SELECT bag_name, bag_id, accesscontrol, description - FROM bags - ORDER BY bag_name - `); - return rows; - }; - - /* - Delete a recipe and its bag associations - */ - SqlTiddlerDatabase.prototype.deleteRecipe = function(recipe_name) { - // Delete recipe_bags entries first (due to foreign key constraints) - this.engine.runStatement(` - DELETE FROM recipe_bags - WHERE recipe_id = ( - SELECT recipe_id - FROM recipes + return results; +}; + +/* +Create or update a recipe +Returns the recipe_id of the recipe +*/ +SqlTiddlerDatabase.prototype.createRecipe = function(recipe_name,bag_names,description) { + // Run the queries + this.engine.runStatement(` + -- Delete existing recipe_bags entries for this recipe + DELETE FROM recipe_bags WHERE recipe_id = (SELECT recipe_id FROM recipes WHERE recipe_name = $recipe_name) + `,{ + $recipe_name: recipe_name + }); + const updateRecipes = this.engine.runStatement(` + -- Create the entry in the recipes table if required + INSERT OR REPLACE INTO recipes (recipe_name, description) + VALUES ($recipe_name, $description) + `,{ + $recipe_name: recipe_name, + $description: description + }); + this.engine.runStatement(` + INSERT INTO recipe_bags (recipe_id, bag_id, position) + SELECT r.recipe_id, b.bag_id, j.key as position + FROM recipes r + JOIN bags b + INNER JOIN json_each($bag_names) AS j ON j.value = b.bag_name + WHERE r.recipe_name = $recipe_name + `,{ + $recipe_name: recipe_name, + $bag_names: JSON.stringify(bag_names) + }); + + return updateRecipes.lastInsertRowid; +}; + +/* +Assign a recipe to a user +*/ +SqlTiddlerDatabase.prototype.assignRecipeToUser = function(recipe_name,user_id) { + this.engine.runStatement(` + UPDATE recipes SET owner_id = $user_id WHERE recipe_name = $recipe_name + `,{ + $recipe_name: recipe_name, + $user_id: user_id + }); +}; + +/* +Returns {tiddler_id:} +*/ +SqlTiddlerDatabase.prototype.saveBagTiddler = function(tiddlerFields,bag_name,attachment_blob) { + attachment_blob = attachment_blob || null; + // Update the tiddlers table + var info = this.engine.runStatement(` + INSERT OR REPLACE INTO tiddlers (bag_id, title, is_deleted, attachment_blob) + VALUES ( + (SELECT bag_id FROM bags WHERE bag_name = $bag_name), + $title, + FALSE, + $attachment_blob + ) + `,{ + $title: tiddlerFields.title, + $attachment_blob: attachment_blob, + $bag_name: bag_name + }); + // Update the fields table + this.engine.runStatement(` + INSERT OR REPLACE INTO fields (tiddler_id, field_name, field_value) + SELECT + t.tiddler_id, + json_each.key AS field_name, + json_each.value AS field_value + FROM ( + SELECT tiddler_id + FROM tiddlers + WHERE bag_id = ( + SELECT bag_id + FROM bags + WHERE bag_name = $bag_name + ) AND title = $title + ) AS t + JOIN json_each($field_values) AS json_each + `,{ + $title: tiddlerFields.title, + $bag_name: bag_name, + $field_values: JSON.stringify(Object.assign({},tiddlerFields,{title: undefined})) + }); + return { + tiddler_id: info.lastInsertRowid + } +}; + +/* +Returns {tiddler_id:,bag_name:} or null if the recipe is empty +*/ +SqlTiddlerDatabase.prototype.saveRecipeTiddler = function(tiddlerFields,recipe_name,attachment_blob) { + // Find the topmost bag in the recipe + var row = this.engine.runStatementGet(` + SELECT b.bag_name + FROM bags AS b + JOIN ( + SELECT rb.bag_id + FROM recipe_bags AS rb + WHERE rb.recipe_id = ( + SELECT recipe_id + FROM recipes WHERE recipe_name = $recipe_name ) - `, { - $recipe_name: recipe_name - }); - - // Then delete the recipe itself - this.engine.runStatement(` - DELETE FROM recipes - WHERE recipe_name = $recipe_name - `, { - $recipe_name: recipe_name - }); - }; - - /* - Create or update a bag - Returns the bag_id of the bag - */ - SqlTiddlerDatabase.prototype.createBag = function(bag_name,description,accesscontrol) { - accesscontrol = accesscontrol || ""; - // Run the queries - var bag = this.engine.runStatement(` - INSERT OR IGNORE INTO bags (bag_name, accesscontrol, description) - VALUES ($bag_name, '', '') - `,{ - $bag_name: bag_name - }); - const updateBags = this.engine.runStatement(` - UPDATE bags - SET accesscontrol = $accesscontrol, - description = $description - WHERE bag_name = $bag_name - `,{ - $bag_name: bag_name, - $accesscontrol: accesscontrol, - $description: description - }); - return updateBags.lastInsertRowid; - }; - - /* - Delete a bag and all its associated data - */ - SqlTiddlerDatabase.prototype.deleteBag = function(bag_name) { - // Delete the bag (cascade will handle related records) - this.engine.runStatement(` - DELETE FROM bags - WHERE bag_name = $bag_name - `, { - $bag_name: bag_name - }); - }; - - /* - Returns array of {recipe_name:,recipe_id:,description:,bag_names: []} - */ - SqlTiddlerDatabase.prototype.listRecipes = function() { - const rows = this.engine.runStatementGetAll(` - SELECT r.recipe_name, r.recipe_id, r.description, b.bag_name, rb.position - FROM recipes AS r - JOIN recipe_bags AS rb ON rb.recipe_id = r.recipe_id - JOIN bags AS b ON rb.bag_id = b.bag_id - ORDER BY r.recipe_name, rb.position - `); - const results = []; - let currentRecipeName = null, currentRecipeIndex = -1; - for(const row of rows) { - if(row.recipe_name !== currentRecipeName) { - currentRecipeName = row.recipe_name; - currentRecipeIndex += 1; - results.push({ - recipe_name: row.recipe_name, - recipe_id: row.recipe_id, - description: row.description, - bag_names: [] - }); - } - results[currentRecipeIndex].bag_names.push(row.bag_name); - } - return results; - }; - - /* - Create or update a recipe - Returns the recipe_id of the recipe - */ - SqlTiddlerDatabase.prototype.createRecipe = function(recipe_name,bag_names,description) { - // Run the queries - this.engine.runStatement(` - -- Delete existing recipe_bags entries for this recipe - DELETE FROM recipe_bags WHERE recipe_id = (SELECT recipe_id FROM recipes WHERE recipe_name = $recipe_name) - `,{ - $recipe_name: recipe_name - }); - const updateRecipes = this.engine.runStatement(` - -- Create the entry in the recipes table if required - INSERT OR REPLACE INTO recipes (recipe_name, description) - VALUES ($recipe_name, $description) - `,{ - $recipe_name: recipe_name, - $description: description - }); - this.engine.runStatement(` - INSERT INTO recipe_bags (recipe_id, bag_id, position) - SELECT r.recipe_id, b.bag_id, j.key as position - FROM recipes r - JOIN bags b - INNER JOIN json_each($bag_names) AS j ON j.value = b.bag_name - WHERE r.recipe_name = $recipe_name - `,{ - $recipe_name: recipe_name, - $bag_names: JSON.stringify(bag_names) - }); - - return updateRecipes.lastInsertRowid; - }; - - /* - Assign a recipe to a user - */ - SqlTiddlerDatabase.prototype.assignRecipeToUser = function(recipe_name,user_id) { - this.engine.runStatement(` - UPDATE recipes SET owner_id = $user_id WHERE recipe_name = $recipe_name - `,{ - $recipe_name: recipe_name, - $user_id: user_id - }); - }; - - /* - Returns {tiddler_id:} - */ - SqlTiddlerDatabase.prototype.saveBagTiddler = function(tiddlerFields,bag_name,attachment_blob) { - attachment_blob = attachment_blob || null; - // Update the tiddlers table - var info = this.engine.runStatement(` - INSERT OR REPLACE INTO tiddlers (bag_id, title, is_deleted, attachment_blob) - VALUES ( - (SELECT bag_id FROM bags WHERE bag_name = $bag_name), - $title, - FALSE, - $attachment_blob - ) - `,{ - $title: tiddlerFields.title, - $attachment_blob: attachment_blob, - $bag_name: bag_name - }); - // Update the fields table - this.engine.runStatement(` - INSERT OR REPLACE INTO fields (tiddler_id, field_name, field_value) - SELECT - t.tiddler_id, - json_each.key AS field_name, - json_each.value AS field_value - FROM ( - SELECT tiddler_id - FROM tiddlers - WHERE bag_id = ( - SELECT bag_id - FROM bags - WHERE bag_name = $bag_name - ) AND title = $title - ) AS t - JOIN json_each($field_values) AS json_each - `,{ - $title: tiddlerFields.title, - $bag_name: bag_name, - $field_values: JSON.stringify(Object.assign({},tiddlerFields,{title: undefined})) - }); - return { - tiddler_id: info.lastInsertRowid - } - }; - - /* - Returns {tiddler_id:,bag_name:} or null if the recipe is empty - */ - SqlTiddlerDatabase.prototype.saveRecipeTiddler = function(tiddlerFields,recipe_name,attachment_blob) { - // Find the topmost bag in the recipe - var row = this.engine.runStatementGet(` - SELECT b.bag_name - FROM bags AS b - JOIN ( - SELECT rb.bag_id - FROM recipe_bags AS rb - WHERE rb.recipe_id = ( - SELECT recipe_id - FROM recipes - WHERE recipe_name = $recipe_name - ) - ORDER BY rb.position DESC - LIMIT 1 - ) AS selected_bag - ON b.bag_id = selected_bag.bag_id - `,{ - $recipe_name: recipe_name - }); - if(!row) { - return null; - } - // Save the tiddler to the topmost bag - var info = this.saveBagTiddler(tiddlerFields,row.bag_name,attachment_blob); - return { - tiddler_id: info.tiddler_id, - bag_name: row.bag_name - }; - }; - - /* - Returns {tiddler_id:} of the delete marker - */ - SqlTiddlerDatabase.prototype.deleteTiddler = function(title,bag_name) { - // Delete the fields of this tiddler - this.engine.runStatement(` - DELETE FROM fields - WHERE tiddler_id IN ( - SELECT t.tiddler_id - FROM tiddlers AS t - INNER JOIN bags AS b ON t.bag_id = b.bag_id - WHERE b.bag_name = $bag_name AND t.title = $title - ) - `,{ - $title: title, - $bag_name: bag_name - }); - // Mark the tiddler itself as deleted - const rowDeleteMarker = this.engine.runStatement(` - INSERT OR REPLACE INTO tiddlers (bag_id, title, is_deleted, attachment_blob) - VALUES ( - (SELECT bag_id FROM bags WHERE bag_name = $bag_name), - $title, - TRUE, - NULL - ) - `,{ - $title: title, - $bag_name: bag_name - }); - return {tiddler_id: rowDeleteMarker.lastInsertRowid}; - }; - - /* - returns {tiddler_id:,tiddler:,attachment_blob:} - */ - SqlTiddlerDatabase.prototype.getBagTiddler = function(title,bag_name) { - const rowTiddler = this.engine.runStatementGet(` - SELECT t.tiddler_id, t.attachment_blob - FROM bags AS b - INNER JOIN tiddlers AS t ON b.bag_id = t.bag_id - WHERE t.title = $title AND b.bag_name = $bag_name AND t.is_deleted = FALSE - `,{ - $title: title, - $bag_name: bag_name - }); - if(!rowTiddler) { - return null; - } - const rows = this.engine.runStatementGetAll(` - SELECT field_name, field_value, tiddler_id - FROM fields - WHERE tiddler_id = $tiddler_id - `,{ - $tiddler_id: rowTiddler.tiddler_id - }); - if(rows.length === 0) { - return null; - } else { - return { - tiddler_id: rows[0].tiddler_id, - attachment_blob: rowTiddler.attachment_blob, - tiddler: rows.reduce((accumulator,value) => { - accumulator[value["field_name"]] = value.field_value; - return accumulator; - },{title: title}) - }; - } - }; - - /* - Returns {bag_name:, tiddler: {fields}, tiddler_id:, attachment_blob:} - */ - SqlTiddlerDatabase.prototype.getRecipeTiddler = function(title,recipe_name) { - const rowTiddlerId = this.engine.runStatementGet(` - SELECT t.tiddler_id, t.attachment_blob, b.bag_name - FROM bags AS b - INNER JOIN recipe_bags AS rb ON b.bag_id = rb.bag_id - INNER JOIN recipes AS r ON rb.recipe_id = r.recipe_id - INNER JOIN tiddlers AS t ON b.bag_id = t.bag_id - WHERE r.recipe_name = $recipe_name - AND t.title = $title - AND t.is_deleted = FALSE ORDER BY rb.position DESC LIMIT 1 - `,{ - $title: title, - $recipe_name: recipe_name - }); - if(!rowTiddlerId) { - return null; - } - // Get the fields - const rows = this.engine.runStatementGetAll(` - SELECT field_name, field_value - FROM fields - WHERE tiddler_id = $tiddler_id - `,{ - $tiddler_id: rowTiddlerId.tiddler_id - }); + ) AS selected_bag + ON b.bag_id = selected_bag.bag_id + `,{ + $recipe_name: recipe_name + }); + if(!row) { + return null; + } + // Save the tiddler to the topmost bag + var info = this.saveBagTiddler(tiddlerFields,row.bag_name,attachment_blob); + return { + tiddler_id: info.tiddler_id, + bag_name: row.bag_name + }; +}; + +/* +Returns {tiddler_id:} of the delete marker +*/ +SqlTiddlerDatabase.prototype.deleteTiddler = function(title,bag_name) { + // Delete the fields of this tiddler + this.engine.runStatement(` + DELETE FROM fields + WHERE tiddler_id IN ( + SELECT t.tiddler_id + FROM tiddlers AS t + INNER JOIN bags AS b ON t.bag_id = b.bag_id + WHERE b.bag_name = $bag_name AND t.title = $title + ) + `,{ + $title: title, + $bag_name: bag_name + }); + // Mark the tiddler itself as deleted + const rowDeleteMarker = this.engine.runStatement(` + INSERT OR REPLACE INTO tiddlers (bag_id, title, is_deleted, attachment_blob) + VALUES ( + (SELECT bag_id FROM bags WHERE bag_name = $bag_name), + $title, + TRUE, + NULL + ) + `,{ + $title: title, + $bag_name: bag_name + }); + return {tiddler_id: rowDeleteMarker.lastInsertRowid}; +}; + +/* +returns {tiddler_id:,tiddler:,attachment_blob:} +*/ +SqlTiddlerDatabase.prototype.getBagTiddler = function(title,bag_name) { + const rowTiddler = this.engine.runStatementGet(` + SELECT t.tiddler_id, t.attachment_blob + FROM bags AS b + INNER JOIN tiddlers AS t ON b.bag_id = t.bag_id + WHERE t.title = $title AND b.bag_name = $bag_name AND t.is_deleted = FALSE + `,{ + $title: title, + $bag_name: bag_name + }); + if(!rowTiddler) { + return null; + } + const rows = this.engine.runStatementGetAll(` + SELECT field_name, field_value, tiddler_id + FROM fields + WHERE tiddler_id = $tiddler_id + `,{ + $tiddler_id: rowTiddler.tiddler_id + }); + if(rows.length === 0) { + return null; + } else { return { - bag_name: rowTiddlerId.bag_name, - tiddler_id: rowTiddlerId.tiddler_id, - attachment_blob: rowTiddlerId.attachment_blob, + tiddler_id: rows[0].tiddler_id, + attachment_blob: rowTiddler.attachment_blob, tiddler: rows.reduce((accumulator,value) => { accumulator[value["field_name"]] = value.field_value; return accumulator; },{title: title}) }; - }; - - /* - Checks if a user has permission to access a recipe - */ - SqlTiddlerDatabase.prototype.hasRecipePermission = function(userId, recipeName, permissionName) { - // check if the user is the owner of the entity - const recipe = this.engine.runStatementGet(` - SELECT owner_id - FROM recipes - WHERE recipe_name = $recipe_name - `, { - $recipe_name: recipeName - }); - - if(recipe?.owner_id) { - return recipe.owner_id === userId; - } - return this.checkACLPermission(userId, "recipe", recipeName, permissionName) - }; - - /* - Checks if a user has permission to access a bag - */ - SqlTiddlerDatabase.prototype.hasBagPermission = function(userId, bagName, permissionName) { - return this.checkACLPermission(userId, "bag", bagName, permissionName) - }; - - SqlTiddlerDatabase.prototype.getACLByName = function(entityType, entityName, fetchAll) { - const entityInfo = this.entityTypeToTableMap[entityType]; - if (!entityInfo) { - throw new Error("Invalid entity type: " + entityType); - } - - // First, check if there's an ACL record for the entity and get the permission_id - var checkACLExistsQuery = ` - SELECT * - FROM acl - WHERE entity_type = $entity_type - AND entity_name = $entity_name - `; - - if (!fetchAll) { - checkACLExistsQuery += ' LIMIT 1' - } - - const aclRecord = this.engine[fetchAll ? 'runStatementGetAll' : 'runStatementGet'](checkACLExistsQuery, { - $entity_type: entityType, - $entity_name: entityName - }); - - return aclRecord; } - - SqlTiddlerDatabase.prototype.checkACLPermission = function(userId, entityType, entityName) { - // if the entityName starts with "$:/", we'll assume its a system bag/recipe, then grant the user permission - if(entityName.startsWith("$:/")) { - return true; - } - - const aclRecord = this.getACLByName(entityType, entityName); - - // If no ACL record exists, return true for hasPermission - if (!aclRecord) { - return true; - } - - // If ACL record exists, check for user permission using the retrieved permission_id - const checkPermissionQuery = ` - SELECT 1 - FROM users u - JOIN user_roles ur ON u.user_id = ur.user_id - JOIN roles r ON ur.role_id = r.role_id - JOIN acl a ON r.role_id = a.role_id - WHERE u.user_id = $user_id - AND a.entity_type = $entity_type - AND a.entity_name = $entity_name - AND a.permission_id = $permission_id - LIMIT 1 - `; - - const result = this.engine.runStatementGet(checkPermissionQuery, { - $user_id: userId, - $entity_type: entityType, - $entity_name: entityName, - $permission_id: aclRecord.permission_id - }); - - let hasPermission = result !== undefined; - - return hasPermission; - }; - - /** - * Returns the ACL records for an entity (bag or recipe) - */ - SqlTiddlerDatabase.prototype.getEntityAclRecords = function(entityName) { - const checkACLExistsQuery = ` - SELECT * - FROM acl - WHERE entity_name = $entity_name - `; - - const aclRecords = this.engine.runStatementGetAll(checkACLExistsQuery, { - $entity_name: entityName - }); - - return aclRecords - } - - /* - Get the entity by name - */ - SqlTiddlerDatabase.prototype.getEntityByName = function(entityType, entityName) { - const entityInfo = this.entityTypeToTableMap[entityType]; - if (entityInfo) { - return this.engine.runStatementGet(`SELECT * FROM ${entityInfo.table} WHERE ${entityInfo.column} = $entity_name`, { - $entity_name: entityName - }); - } +}; + +/* +Returns {bag_name:, tiddler: {fields}, tiddler_id:, attachment_blob:} +*/ +SqlTiddlerDatabase.prototype.getRecipeTiddler = function(title,recipe_name) { + const rowTiddlerId = this.engine.runStatementGet(` + SELECT t.tiddler_id, t.attachment_blob, b.bag_name + FROM bags AS b + INNER JOIN recipe_bags AS rb ON b.bag_id = rb.bag_id + INNER JOIN recipes AS r ON rb.recipe_id = r.recipe_id + INNER JOIN tiddlers AS t ON b.bag_id = t.bag_id + WHERE r.recipe_name = $recipe_name + AND t.title = $title + AND t.is_deleted = FALSE + ORDER BY rb.position DESC + LIMIT 1 + `,{ + $title: title, + $recipe_name: recipe_name + }); + if(!rowTiddlerId) { return null; } - - /* - Get the titles of the tiddlers in a bag. Returns an empty array for bags that do not exist - */ - SqlTiddlerDatabase.prototype.getBagTiddlers = function(bag_name) { - const rows = this.engine.runStatementGetAll(` - SELECT DISTINCT title, tiddler_id - FROM tiddlers - WHERE bag_id IN ( - SELECT bag_id - FROM bags - WHERE bag_name = $bag_name - ) - AND tiddlers.is_deleted = FALSE - ORDER BY title ASC - `,{ - $bag_name: bag_name - }); - return rows; + // Get the fields + const rows = this.engine.runStatementGetAll(` + SELECT field_name, field_value + FROM fields + WHERE tiddler_id = $tiddler_id + `,{ + $tiddler_id: rowTiddlerId.tiddler_id + }); + return { + bag_name: rowTiddlerId.bag_name, + tiddler_id: rowTiddlerId.tiddler_id, + attachment_blob: rowTiddlerId.attachment_blob, + tiddler: rows.reduce((accumulator,value) => { + accumulator[value["field_name"]] = value.field_value; + return accumulator; + },{title: title}) }; - - /* - Get the tiddler_id of the newest tiddler in a bag. Returns null for bags that do not exist - */ - SqlTiddlerDatabase.prototype.getBagLastTiddlerId = function(bag_name) { - const row = this.engine.runStatementGet(` - SELECT tiddler_id - FROM tiddlers - WHERE bag_id IN ( - SELECT bag_id - FROM bags - WHERE bag_name = $bag_name - ) - ORDER BY tiddler_id DESC - LIMIT 1 - `,{ - $bag_name: bag_name - }); - if(row) { - return row.tiddler_id; - } else { - return null; - } - }; - - /* - Get the metadata of the tiddlers in a recipe as an array [{title:,tiddler_id:,bag_name:,is_deleted:}], - sorted in ascending order of tiddler_id. - - Options include: - - limit: optional maximum number of results to return - last_known_tiddler_id: tiddler_id of the last known update. Only returns tiddlers that have been created, modified or deleted since - include_deleted: boolean, defaults to false - - Returns null for recipes that do not exist - */ - SqlTiddlerDatabase.prototype.getRecipeTiddlers = function(recipe_name,options) { - options = options || {}; - // Get the recipe ID - const rowsCheckRecipe = this.engine.runStatementGet(` - SELECT recipe_id FROM recipes WHERE recipes.recipe_name = $recipe_name - `,{ - $recipe_name: recipe_name - }); - if(!rowsCheckRecipe) { - return null; - } - const recipe_id = rowsCheckRecipe.recipe_id; - // Compose the query to get the tiddlers - const params = { - $recipe_id: recipe_id - } - if(options.limit) { - params.$limit = options.limit.toString(); - } - if(options.last_known_tiddler_id) { - params.$last_known_tiddler_id = options.last_known_tiddler_id; - } - const rows = this.engine.runStatementGetAll(` - SELECT title, tiddler_id, is_deleted, bag_name - FROM ( - SELECT t.title, t.tiddler_id, t.is_deleted, b.bag_name, MAX(rb.position) AS position - FROM bags AS b - INNER JOIN recipe_bags AS rb ON b.bag_id = rb.bag_id - INNER JOIN tiddlers AS t ON b.bag_id = t.bag_id - WHERE rb.recipe_id = $recipe_id - ${options.include_deleted ? "" : "AND t.is_deleted = FALSE"} - ${options.last_known_tiddler_id ? "AND tiddler_id > $last_known_tiddler_id" : ""} - GROUP BY t.title - ORDER BY t.title, tiddler_id DESC - ${options.limit ? "LIMIT $limit" : ""} - ) - `,params); - return rows; - }; - - /* - Get the tiddler_id of the newest tiddler in a recipe. Returns null for recipes that do not exist - */ - SqlTiddlerDatabase.prototype.getRecipeLastTiddlerId = function(recipe_name) { - const row = this.engine.runStatementGet(` - SELECT t.title, t.tiddler_id, b.bag_name, MAX(rb.position) AS position - FROM bags AS b - INNER JOIN recipe_bags AS rb ON b.bag_id = rb.bag_id - INNER JOIN recipes AS r ON rb.recipe_id = r.recipe_id - INNER JOIN tiddlers AS t ON b.bag_id = t.bag_id - WHERE r.recipe_name = $recipe_name - GROUP BY t.title - ORDER BY t.tiddler_id DESC - LIMIT 1 - `,{ - $recipe_name: recipe_name - }); - if(row) { - return row.tiddler_id; - } else { - return null; - } - }; - - SqlTiddlerDatabase.prototype.deleteAllTiddlersInBag = function(bag_name) { - // Delete the fields - this.engine.runStatement(` - DELETE FROM fields - WHERE tiddler_id IN ( - SELECT tiddler_id - FROM tiddlers - WHERE bag_id = (SELECT bag_id FROM bags WHERE bag_name = $bag_name) - AND is_deleted = FALSE - ) - `,{ - $bag_name: bag_name - }); - // Mark the tiddlers as deleted - this.engine.runStatement(` - UPDATE tiddlers - SET is_deleted = TRUE - WHERE bag_id = (SELECT bag_id FROM bags WHERE bag_name = $bag_name) - AND is_deleted = FALSE - `,{ - $bag_name: bag_name - }); - }; - - /* - Get the names of the bags in a recipe. Returns an empty array for recipes that do not exist - */ - SqlTiddlerDatabase.prototype.getRecipeBags = function(recipe_name) { - const rows = this.engine.runStatementGetAll(` - SELECT bags.bag_name - FROM bags - JOIN ( - SELECT rb.bag_id, rb.position as position - FROM recipe_bags AS rb - JOIN recipes AS r ON rb.recipe_id = r.recipe_id - WHERE r.recipe_name = $recipe_name - ORDER BY rb.position - ) AS bag_priority ON bags.bag_id = bag_priority.bag_id - ORDER BY position - `,{ - $recipe_name: recipe_name - }); - return rows.map(value => value.bag_name); - }; - - /* - Get the attachment value of a bag, if any exist - */ - SqlTiddlerDatabase.prototype.getBagTiddlerAttachmentBlob = function(title,bag_name) { - const row = this.engine.runStatementGet(` - SELECT t.attachment_blob - FROM bags AS b - INNER JOIN tiddlers AS t ON b.bag_id = t.bag_id - WHERE t.title = $title AND b.bag_name = $bag_name AND t.is_deleted = FALSE +}; + +/* +Checks if a user has permission to access a recipe +*/ +SqlTiddlerDatabase.prototype.hasRecipePermission = function(userId, recipeName, permissionName) { + // check if the user is the owner of the entity + const recipe = this.engine.runStatementGet(` + SELECT owner_id + FROM recipes + WHERE recipe_name = $recipe_name `, { - $title: title, - $bag_name: bag_name + $recipe_name: recipeName }); - return row ? row.attachment_blob : null; - }; + + if(recipe?.owner_id) { + return recipe.owner_id === userId; + } + return this.checkACLPermission(userId, "recipe", recipeName, permissionName) +}; + +/* +Checks if a user has permission to access a bag +*/ +SqlTiddlerDatabase.prototype.hasBagPermission = function(userId, bagName, permissionName) { + return this.checkACLPermission(userId, "bag", bagName, permissionName) +}; + +SqlTiddlerDatabase.prototype.getACLByName = function(entityType, entityName, fetchAll) { + const entityInfo = this.entityTypeToTableMap[entityType]; + if (!entityInfo) { + throw new Error("Invalid entity type: " + entityType); + } + + // First, check if there's an ACL record for the entity and get the permission_id + var checkACLExistsQuery = ` + SELECT * + FROM acl + WHERE entity_type = $entity_type + AND entity_name = $entity_name + `; + + if (!fetchAll) { + checkACLExistsQuery += ' LIMIT 1' + } + + const aclRecord = this.engine[fetchAll ? 'runStatementGetAll' : 'runStatementGet'](checkACLExistsQuery, { + $entity_type: entityType, + $entity_name: entityName + }); + + return aclRecord; +} + +SqlTiddlerDatabase.prototype.checkACLPermission = function(userId, entityType, entityName) { + // if the entityName starts with "$:/", we'll assume its a system bag/recipe, then grant the user permission + if(entityName.startsWith("$:/")) { + return true; + } + + const aclRecord = this.getACLByName(entityType, entityName); + + // If no ACL record exists, return true for hasPermission + if (!aclRecord) { + return true; + } + + // If ACL record exists, check for user permission using the retrieved permission_id + const checkPermissionQuery = ` + SELECT 1 + FROM users u + JOIN user_roles ur ON u.user_id = ur.user_id + JOIN roles r ON ur.role_id = r.role_id + JOIN acl a ON r.role_id = a.role_id + WHERE u.user_id = $user_id + AND a.entity_type = $entity_type + AND a.entity_name = $entity_name + AND a.permission_id = $permission_id + LIMIT 1 + `; + + const result = this.engine.runStatementGet(checkPermissionQuery, { + $user_id: userId, + $entity_type: entityType, + $entity_name: entityName, + $permission_id: aclRecord.permission_id + }); - /* - Get the attachment value of a recipe, if any exist - */ - SqlTiddlerDatabase.prototype.getRecipeTiddlerAttachmentBlob = function(title,recipe_name) { - const row = this.engine.runStatementGet(` - SELECT t.attachment_blob - FROM bags AS b - INNER JOIN recipe_bags AS rb ON b.bag_id = rb.bag_id - INNER JOIN recipes AS r ON rb.recipe_id = r.recipe_id - INNER JOIN tiddlers AS t ON b.bag_id = t.bag_id - WHERE r.recipe_name = $recipe_name AND t.title = $title AND t.is_deleted = FALSE - ORDER BY rb.position DESC - LIMIT 1 - `, { - $title: title, - $recipe_name: recipe_name - }); - return row ? row.attachment_blob : null; - }; - - // User CRUD operations - SqlTiddlerDatabase.prototype.createUser = function(username, email, password) { - const result = this.engine.runStatement(` - INSERT INTO users (username, email, password) - VALUES ($username, $email, $password) - `, { - $username: username, - $email: email, - $password: password - }); - return result.lastInsertRowid; - }; - - SqlTiddlerDatabase.prototype.getUser = function(userId) { - return this.engine.runStatementGet(` - SELECT * FROM users WHERE user_id = $userId - `, { - $userId: userId - }); - }; - - SqlTiddlerDatabase.prototype.getUserByUsername = function(username) { - return this.engine.runStatementGet(` - SELECT * FROM users WHERE username = $username - `, { - $username: username - }); - }; - - SqlTiddlerDatabase.prototype.getUserByEmail = function(email) { - return this.engine.runStatementGet(` - SELECT * FROM users WHERE email = $email - `, { - $email: email - }); - }; - - SqlTiddlerDatabase.prototype.listUsersByRoleId = function(roleId) { - return this.engine.runStatementGetAll(` - SELECT u.* - FROM users u - JOIN user_roles ur ON u.user_id = ur.user_id - WHERE ur.role_id = $roleId - ORDER BY u.username - `, { - $roleId: roleId - }); - }; - - SqlTiddlerDatabase.prototype.updateUser = function (userId, username, email, roleId) { - const existingUser = this.engine.runStatement(` - SELECT user_id FROM users - WHERE email = $email AND user_id != $userId - `, { - $email: email, - $userId: userId - }); - - if (existingUser.length > 0) { - return { - success: false, - message: "Email address already in use by another user." - }; - } - - try { - this.engine.transaction(() => { - // Update user information - this.engine.runStatement(` - UPDATE users - SET username = $username, email = $email - WHERE user_id = $userId - `, { - $userId: userId, - $username: username, - $email: email - }); - - if (roleId) { - // Remove all existing roles for the user - this.engine.runStatement(` - DELETE FROM user_roles - WHERE user_id = $userId - `, { - $userId: userId - }); - - // Add the new role - this.engine.runStatement(` - INSERT INTO user_roles (user_id, role_id) - VALUES ($userId, $roleId) - `, { - $userId: userId, - $roleId: roleId - }); - } - }); - - return { - success: true, - message: "User profile and role updated successfully." - }; - } catch (error) { - return { - success: false, - message: "Failed to update user profile: " + error.message - }; - } - }; - - SqlTiddlerDatabase.prototype.updateUserPassword = function (userId, newHash) { - try { - this.engine.runStatement(` - UPDATE users - SET password = $newHash - WHERE user_id = $userId - `, { - $userId: userId, - $newHash: newHash, - }); - - return { - success: true, - message: "Password updated successfully." - }; - } catch (error) { - return { - success: false, - message: "Failed to update password: " + error.message - }; - } - }; - - SqlTiddlerDatabase.prototype.deleteUser = function(userId) { - this.engine.runStatement(` - DELETE FROM users WHERE user_id = $userId - `, { - $userId: userId - }); - }; - - SqlTiddlerDatabase.prototype.listUsers = function() { - return this.engine.runStatementGetAll(` - SELECT * FROM users ORDER BY username - `); - }; - - SqlTiddlerDatabase.prototype.createOrUpdateUserSession = function(userId, sessionId) { - const currentTimestamp = new Date().toISOString(); - - // First, try to update an existing session - const updateResult = this.engine.runStatement(` - UPDATE sessions - SET session_id = $sessionId, last_accessed = $timestamp - WHERE user_id = $userId - `, { - $userId: userId, - $sessionId: sessionId, - $timestamp: currentTimestamp - }); - - // If no existing session was updated, create a new one - if (updateResult.changes === 0) { - this.engine.runStatement(` - INSERT INTO sessions (user_id, session_id, created_at, last_accessed) - VALUES ($userId, $sessionId, $timestamp, $timestamp) - `, { - $userId: userId, - $sessionId: sessionId, - $timestamp: currentTimestamp - }); - } - - return sessionId; - }; - - SqlTiddlerDatabase.prototype.findUserBySessionId = function(sessionId) { - // First, get the user_id from the sessions table - const sessionResult = this.engine.runStatementGet(` - SELECT user_id, last_accessed - FROM sessions - WHERE session_id = $sessionId - `, { - $sessionId: sessionId - }); - - if (!sessionResult) { - return null; // Session not found - } - - const lastAccessed = new Date(sessionResult.last_accessed); - const expirationTime = 24 * 60 * 60 * 1000; // 24 hours in milliseconds - if (new Date() - lastAccessed > expirationTime) { - // Session has expired - this.deleteSession(sessionId); - return null; - } - - // Update the last_accessed timestamp - const currentTimestamp = new Date().toISOString(); - this.engine.runStatement(` - UPDATE sessions - SET last_accessed = $timestamp - WHERE session_id = $sessionId - `, { - $sessionId: sessionId, - $timestamp: currentTimestamp - }); - - const userResult = this.engine.runStatementGet(` - SELECT * - FROM users - WHERE user_id = $userId - `, { - $userId: sessionResult.user_id - }); - - if (!userResult) { - return null; - } - - return userResult; - }; - - SqlTiddlerDatabase.prototype.deleteSession = function(sessionId) { - this.engine.runStatement(` - DELETE FROM sessions - WHERE session_id = $sessionId - `, { - $sessionId: sessionId - }); - }; - - SqlTiddlerDatabase.prototype.deleteUserSessions = function(userId) { - this.engine.runStatement(` - DELETE FROM sessions - WHERE user_id = $userId - `, { - $userId: userId - }); - }; - - // Set the user as an admin - SqlTiddlerDatabase.prototype.setUserAdmin = function(userId) { - var admin = this.getRoleByName("ADMIN"); - if(admin) { - this.addRoleToUser(userId, admin.role_id); - } - }; - - // Group CRUD operations - SqlTiddlerDatabase.prototype.createGroup = function(groupName, description) { - const result = this.engine.runStatement(` - INSERT INTO groups (group_name, description) - VALUES ($groupName, $description) - `, { - $groupName: groupName, - $description: description - }); - return result.lastInsertRowid; - }; - - SqlTiddlerDatabase.prototype.getGroup = function(groupId) { - return this.engine.runStatementGet(` - SELECT * FROM groups WHERE group_id = $groupId - `, { - $groupId: groupId - }); - }; - - SqlTiddlerDatabase.prototype.updateGroup = function(groupId, groupName, description) { - this.engine.runStatement(` - UPDATE groups - SET group_name = $groupName, description = $description - WHERE group_id = $groupId - `, { - $groupId: groupId, - $groupName: groupName, - $description: description - }); - }; - - SqlTiddlerDatabase.prototype.deleteGroup = function(groupId) { - this.engine.runStatement(` - DELETE FROM groups WHERE group_id = $groupId - `, { - $groupId: groupId - }); - }; - - SqlTiddlerDatabase.prototype.listGroups = function() { - return this.engine.runStatementGetAll(` - SELECT * FROM groups ORDER BY group_name - `); - }; - - // Role CRUD operations - SqlTiddlerDatabase.prototype.createRole = function(roleName, description) { - const result = this.engine.runStatement(` - INSERT OR IGNORE INTO roles (role_name, description) - VALUES ($roleName, $description) - `, { - $roleName: roleName, - $description: description - }); - return result.lastInsertRowid; - }; - - SqlTiddlerDatabase.prototype.getRole = function(roleId) { - return this.engine.runStatementGet(` - SELECT * FROM roles WHERE role_id = $roleId - `, { - $roleId: roleId - }); - }; - - SqlTiddlerDatabase.prototype.getRoleByName = function(roleName) { - return this.engine.runStatementGet(` - SELECT * FROM roles WHERE role_name = $roleName - `, { - $roleName: roleName + let hasPermission = result !== undefined; + + return hasPermission; +}; + +/** + * Returns the ACL records for an entity (bag or recipe) + */ +SqlTiddlerDatabase.prototype.getEntityAclRecords = function(entityName) { + const checkACLExistsQuery = ` + SELECT * + FROM acl + WHERE entity_name = $entity_name + `; + + const aclRecords = this.engine.runStatementGetAll(checkACLExistsQuery, { + $entity_name: entityName + }); + + return aclRecords +} + +/* +Get the entity by name +*/ +SqlTiddlerDatabase.prototype.getEntityByName = function(entityType, entityName) { + const entityInfo = this.entityTypeToTableMap[entityType]; + if (entityInfo) { + return this.engine.runStatementGet(`SELECT * FROM ${entityInfo.table} WHERE ${entityInfo.column} = $entity_name`, { + $entity_name: entityName }); } - - SqlTiddlerDatabase.prototype.updateRole = function(roleId, roleName, description) { - this.engine.runStatement(` - UPDATE roles - SET role_name = $roleName, description = $description - WHERE role_id = $roleId - `, { - $roleId: roleId, - $roleName: roleName, - $description: description - }); - }; - - SqlTiddlerDatabase.prototype.deleteRole = function(roleId) { - this.engine.runStatement(` - DELETE FROM roles WHERE role_id = $roleId - `, { - $roleId: roleId - }); - }; - - SqlTiddlerDatabase.prototype.listRoles = function() { - return this.engine.runStatementGetAll(` - SELECT * FROM roles ORDER BY role_name DESC - `); - }; - - // Permission CRUD operations - SqlTiddlerDatabase.prototype.createPermission = function(permissionName, description) { - const result = this.engine.runStatement(` - INSERT OR IGNORE INTO permissions (permission_name, description) - VALUES ($permissionName, $description) - `, { - $permissionName: permissionName, - $description: description - }); - return result.lastInsertRowid; - }; - - SqlTiddlerDatabase.prototype.getPermission = function(permissionId) { - return this.engine.runStatementGet(` - SELECT * FROM permissions WHERE permission_id = $permissionId - `, { - $permissionId: permissionId - }); - }; - - SqlTiddlerDatabase.prototype.getPermissionByName = function(permissionName) { - return this.engine.runStatementGet(` - SELECT * FROM permissions WHERE permission_name = $permissionName - `, { - $permissionName: permissionName - }); - }; - - SqlTiddlerDatabase.prototype.updatePermission = function(permissionId, permissionName, description) { - this.engine.runStatement(` - UPDATE permissions - SET permission_name = $permissionName, description = $description - WHERE permission_id = $permissionId - `, { - $permissionId: permissionId, - $permissionName: permissionName, - $description: description - }); - }; - - SqlTiddlerDatabase.prototype.deletePermission = function(permissionId) { - this.engine.runStatement(` - DELETE FROM permissions WHERE permission_id = $permissionId - `, { - $permissionId: permissionId - }); - }; - - SqlTiddlerDatabase.prototype.listPermissions = function() { - return this.engine.runStatementGetAll(` - SELECT * FROM permissions ORDER BY permission_name - `); - }; - - // ACL CRUD operations - SqlTiddlerDatabase.prototype.createACL = function(entityName, entityType, roleId, permissionId) { - if(!entityName.startsWith("$:/")) { - const result = this.engine.runStatement(` - INSERT OR IGNORE INTO acl (entity_name, entity_type, role_id, permission_id) - VALUES ($entityName, $entityType, $roleId, $permissionId) - `, - { - $entityName: entityName, - $entityType: entityType, - $roleId: roleId, - $permissionId: permissionId + return null; +} + +/* +Get the titles of the tiddlers in a bag. Returns an empty array for bags that do not exist +*/ +SqlTiddlerDatabase.prototype.getBagTiddlers = function(bag_name) { + const rows = this.engine.runStatementGetAll(` + SELECT DISTINCT title, tiddler_id + FROM tiddlers + WHERE bag_id IN ( + SELECT bag_id + FROM bags + WHERE bag_name = $bag_name + ) + AND tiddlers.is_deleted = FALSE + ORDER BY title ASC + `,{ + $bag_name: bag_name + }); + return rows; +}; + +/* +Get the tiddler_id of the newest tiddler in a bag. Returns null for bags that do not exist +*/ +SqlTiddlerDatabase.prototype.getBagLastTiddlerId = function(bag_name) { + const row = this.engine.runStatementGet(` + SELECT tiddler_id + FROM tiddlers + WHERE bag_id IN ( + SELECT bag_id + FROM bags + WHERE bag_name = $bag_name + ) + ORDER BY tiddler_id DESC + LIMIT 1 + `,{ + $bag_name: bag_name + }); + if(row) { + return row.tiddler_id; + } else { + return null; + } +}; + +/* +Get the metadata of the tiddlers in a recipe as an array [{title:,tiddler_id:,bag_name:,is_deleted:}], +sorted in ascending order of tiddler_id. + +Options include: + +limit: optional maximum number of results to return +last_known_tiddler_id: tiddler_id of the last known update. Only returns tiddlers that have been created, modified or deleted since +include_deleted: boolean, defaults to false + +Returns null for recipes that do not exist +*/ +SqlTiddlerDatabase.prototype.getRecipeTiddlers = function(recipe_name,options) { + options = options || {}; + // Get the recipe ID + const rowsCheckRecipe = this.engine.runStatementGet(` + SELECT recipe_id FROM recipes WHERE recipes.recipe_name = $recipe_name + `,{ + $recipe_name: recipe_name + }); + if(!rowsCheckRecipe) { + return null; + } + const recipe_id = rowsCheckRecipe.recipe_id; + // Compose the query to get the tiddlers + const params = { + $recipe_id: recipe_id + } + if(options.limit) { + params.$limit = options.limit.toString(); + } + if(options.last_known_tiddler_id) { + params.$last_known_tiddler_id = options.last_known_tiddler_id; + } + const rows = this.engine.runStatementGetAll(` + SELECT title, tiddler_id, is_deleted, bag_name + FROM ( + SELECT t.title, t.tiddler_id, t.is_deleted, b.bag_name, MAX(rb.position) AS position + FROM bags AS b + INNER JOIN recipe_bags AS rb ON b.bag_id = rb.bag_id + INNER JOIN tiddlers AS t ON b.bag_id = t.bag_id + WHERE rb.recipe_id = $recipe_id + ${options.include_deleted ? "" : "AND t.is_deleted = FALSE"} + ${options.last_known_tiddler_id ? "AND tiddler_id > $last_known_tiddler_id" : ""} + GROUP BY t.title + ORDER BY t.title, tiddler_id DESC + ${options.limit ? "LIMIT $limit" : ""} + ) + `,params); + return rows; +}; + +/* +Get the tiddler_id of the newest tiddler in a recipe. Returns null for recipes that do not exist +*/ +SqlTiddlerDatabase.prototype.getRecipeLastTiddlerId = function(recipe_name) { + const row = this.engine.runStatementGet(` + SELECT t.title, t.tiddler_id, b.bag_name, MAX(rb.position) AS position + FROM bags AS b + INNER JOIN recipe_bags AS rb ON b.bag_id = rb.bag_id + INNER JOIN recipes AS r ON rb.recipe_id = r.recipe_id + INNER JOIN tiddlers AS t ON b.bag_id = t.bag_id + WHERE r.recipe_name = $recipe_name + GROUP BY t.title + ORDER BY t.tiddler_id DESC + LIMIT 1 + `,{ + $recipe_name: recipe_name + }); + if(row) { + return row.tiddler_id; + } else { + return null; + } +}; + +SqlTiddlerDatabase.prototype.deleteAllTiddlersInBag = function(bag_name) { + // Delete the fields + this.engine.runStatement(` + DELETE FROM fields + WHERE tiddler_id IN ( + SELECT tiddler_id + FROM tiddlers + WHERE bag_id = (SELECT bag_id FROM bags WHERE bag_name = $bag_name) + AND is_deleted = FALSE + ) + `,{ + $bag_name: bag_name + }); + // Mark the tiddlers as deleted + this.engine.runStatement(` + UPDATE tiddlers + SET is_deleted = TRUE + WHERE bag_id = (SELECT bag_id FROM bags WHERE bag_name = $bag_name) + AND is_deleted = FALSE + `,{ + $bag_name: bag_name + }); +}; + +/* +Get the names of the bags in a recipe. Returns an empty array for recipes that do not exist +*/ +SqlTiddlerDatabase.prototype.getRecipeBags = function(recipe_name) { + const rows = this.engine.runStatementGetAll(` + SELECT bags.bag_name + FROM bags + JOIN ( + SELECT rb.bag_id, rb.position as position + FROM recipe_bags AS rb + JOIN recipes AS r ON rb.recipe_id = r.recipe_id + WHERE r.recipe_name = $recipe_name + ORDER BY rb.position + ) AS bag_priority ON bags.bag_id = bag_priority.bag_id + ORDER BY position + `,{ + $recipe_name: recipe_name + }); + return rows.map(value => value.bag_name); +}; + +/* +Get the attachment value of a bag, if any exist +*/ +SqlTiddlerDatabase.prototype.getBagTiddlerAttachmentBlob = function(title,bag_name) { + const row = this.engine.runStatementGet(` + SELECT t.attachment_blob + FROM bags AS b + INNER JOIN tiddlers AS t ON b.bag_id = t.bag_id + WHERE t.title = $title AND b.bag_name = $bag_name AND t.is_deleted = FALSE + `, { + $title: title, + $bag_name: bag_name + }); + return row ? row.attachment_blob : null; +}; + +/* +Get the attachment value of a recipe, if any exist +*/ +SqlTiddlerDatabase.prototype.getRecipeTiddlerAttachmentBlob = function(title,recipe_name) { + const row = this.engine.runStatementGet(` + SELECT t.attachment_blob + FROM bags AS b + INNER JOIN recipe_bags AS rb ON b.bag_id = rb.bag_id + INNER JOIN recipes AS r ON rb.recipe_id = r.recipe_id + INNER JOIN tiddlers AS t ON b.bag_id = t.bag_id + WHERE r.recipe_name = $recipe_name AND t.title = $title AND t.is_deleted = FALSE + ORDER BY rb.position DESC + LIMIT 1 + `, { + $title: title, + $recipe_name: recipe_name + }); + return row ? row.attachment_blob : null; +}; + +// User CRUD operations +SqlTiddlerDatabase.prototype.createUser = function(username, email, password) { + const result = this.engine.runStatement(` + INSERT INTO users (username, email, password) + VALUES ($username, $email, $password) + `, { + $username: username, + $email: email, + $password: password + }); + return result.lastInsertRowid; +}; + +SqlTiddlerDatabase.prototype.getUser = function(userId) { + return this.engine.runStatementGet(` + SELECT * FROM users WHERE user_id = $userId + `, { + $userId: userId + }); +}; + +SqlTiddlerDatabase.prototype.getUserByUsername = function(username) { + return this.engine.runStatementGet(` + SELECT * FROM users WHERE username = $username + `, { + $username: username + }); +}; + +SqlTiddlerDatabase.prototype.getUserByEmail = function(email) { + return this.engine.runStatementGet(` + SELECT * FROM users WHERE email = $email + `, { + $email: email + }); +}; + +SqlTiddlerDatabase.prototype.listUsersByRoleId = function(roleId) { + return this.engine.runStatementGetAll(` + SELECT u.* + FROM users u + JOIN user_roles ur ON u.user_id = ur.user_id + WHERE ur.role_id = $roleId + ORDER BY u.username + `, { + $roleId: roleId + }); +}; + +SqlTiddlerDatabase.prototype.updateUser = function (userId, username, email, roleId) { + const existingUser = this.engine.runStatement(` + SELECT user_id FROM users + WHERE email = $email AND user_id != $userId +`, { + $email: email, + $userId: userId + }); + + if (existingUser.length > 0) { + return { + success: false, + message: "Email address already in use by another user." + }; + } + + try { + this.engine.transaction(() => { + // Update user information + this.engine.runStatement(` + UPDATE users + SET username = $username, email = $email + WHERE user_id = $userId + `, { + $userId: userId, + $username: username, + $email: email }); - return result.lastInsertRowid; - } - }; - - SqlTiddlerDatabase.prototype.getACL = function(aclId) { - return this.engine.runStatementGet(` - SELECT * FROM acl WHERE acl_id = $aclId - `, { - $aclId: aclId + + if (roleId) { + // Remove all existing roles for the user + this.engine.runStatement(` + DELETE FROM user_roles + WHERE user_id = $userId + `, { + $userId: userId + }); + + // Add the new role + this.engine.runStatement(` + INSERT INTO user_roles (user_id, role_id) + VALUES ($userId, $roleId) + `, { + $userId: userId, + $roleId: roleId + }); + } }); - }; - - SqlTiddlerDatabase.prototype.updateACL = function(aclId, entityId, entityType, roleId, permissionId) { + + return { + success: true, + message: "User profile and role updated successfully." + }; + } catch (error) { + return { + success: false, + message: "Failed to update user profile: " + error.message + }; + } +}; + +SqlTiddlerDatabase.prototype.updateUserPassword = function (userId, newHash) { + try { this.engine.runStatement(` - UPDATE acl - SET entity_name = $entityId, entity_type = $entityType, - role_id = $roleId, permission_id = $permissionId - WHERE acl_id = $aclId - `, { - $aclId: aclId, - $entityId: entityId, - $entityType: entityType, - $roleId: roleId, - $permissionId: permissionId - }); - }; - - SqlTiddlerDatabase.prototype.deleteACL = function(aclId) { - this.engine.runStatement(` - DELETE FROM acl WHERE acl_id = $aclId - `, { - $aclId: aclId - }); - }; - - SqlTiddlerDatabase.prototype.listACLs = function() { - return this.engine.runStatementGetAll(` - SELECT * FROM acl ORDER BY entity_type, entity_name - `); - }; - - // Association management functions - SqlTiddlerDatabase.prototype.addUserToGroup = function(userId, groupId) { - this.engine.runStatement(` - INSERT OR IGNORE INTO user_groups (user_id, group_id) - VALUES ($userId, $groupId) - `, { - $userId: userId, - $groupId: groupId - }); - }; - - SqlTiddlerDatabase.prototype.isUserInGroup = function(userId, groupId) { - const result = this.engine.runStatementGet(` - SELECT 1 FROM user_groups - WHERE user_id = $userId AND group_id = $groupId - `, { - $userId: userId, - $groupId: groupId - }); - return result !== undefined; - }; - - SqlTiddlerDatabase.prototype.removeUserFromGroup = function(userId, groupId) { - this.engine.runStatement(` - DELETE FROM user_groups - WHERE user_id = $userId AND group_id = $groupId - `, { - $userId: userId, - $groupId: groupId - }); - }; - - SqlTiddlerDatabase.prototype.addRoleToUser = function(userId, roleId) { - this.engine.runStatement(` - INSERT OR IGNORE INTO user_roles (user_id, role_id) - VALUES ($userId, $roleId) - `, { - $userId: userId, - $roleId: roleId - }); - }; - - SqlTiddlerDatabase.prototype.removeRoleFromUser = function(userId, roleId) { - this.engine.runStatement(` - DELETE FROM user_roles - WHERE user_id = $userId AND role_id = $roleId - `, { - $userId: userId, - $roleId: roleId - }); - }; - - SqlTiddlerDatabase.prototype.addRoleToGroup = function(groupId, roleId) { - this.engine.runStatement(` - INSERT OR IGNORE INTO group_roles (group_id, role_id) - VALUES ($groupId, $roleId) - `, { - $groupId: groupId, - $roleId: roleId - }); - }; - - SqlTiddlerDatabase.prototype.removeRoleFromGroup = function(groupId, roleId) { - this.engine.runStatement(` - DELETE FROM group_roles - WHERE group_id = $groupId AND role_id = $roleId - `, { - $groupId: groupId, - $roleId: roleId - }); - }; - - SqlTiddlerDatabase.prototype.addPermissionToRole = function(roleId, permissionId) { - this.engine.runStatement(` - INSERT OR IGNORE INTO role_permissions (role_id, permission_id) - VALUES ($roleId, $permissionId) - `, { - $roleId: roleId, - $permissionId: permissionId - }); - }; - - SqlTiddlerDatabase.prototype.removePermissionFromRole = function(roleId, permissionId) { - this.engine.runStatement(` - DELETE FROM role_permissions - WHERE role_id = $roleId AND permission_id = $permissionId - `, { - $roleId: roleId, - $permissionId: permissionId - }); - }; - - SqlTiddlerDatabase.prototype.getUserRoles = function(userId) { - const query = ` - SELECT r.role_id, r.role_name - FROM user_roles ur - JOIN roles r ON ur.role_id = r.role_id - WHERE ur.user_id = $userId - LIMIT 1 - `; - - return this.engine.runStatementGet(query, { $userId: userId }); - }; - - SqlTiddlerDatabase.prototype.deleteUserRolesByRoleId = function(roleId) { - this.engine.runStatement(` - DELETE FROM user_roles - WHERE role_id = $roleId - `, { - $roleId: roleId - }); - }; - - SqlTiddlerDatabase.prototype.deleteUserRolesByUserId = function(userId) { - this.engine.runStatement(` - DELETE FROM user_roles + UPDATE users + SET password = $newHash WHERE user_id = $userId `, { - $userId: userId + $userId: userId, + $newHash: newHash, }); - }; - - SqlTiddlerDatabase.prototype.isRoleInUse = function(roleId) { - // Check if the role is assigned to any users - const userRoleCheck = this.engine.runStatementGet(` - SELECT 1 - FROM user_roles + + return { + success: true, + message: "Password updated successfully." + }; + } catch (error) { + return { + success: false, + message: "Failed to update password: " + error.message + }; + } +}; + +SqlTiddlerDatabase.prototype.deleteUser = function(userId) { + this.engine.runStatement(` + DELETE FROM users WHERE user_id = $userId + `, { + $userId: userId + }); +}; + +SqlTiddlerDatabase.prototype.listUsers = function() { + return this.engine.runStatementGetAll(` + SELECT * FROM users ORDER BY username + `); +}; + +SqlTiddlerDatabase.prototype.createOrUpdateUserSession = function(userId, sessionId) { + const currentTimestamp = new Date().toISOString(); + + // First, try to update an existing session + const updateResult = this.engine.runStatement(` + UPDATE sessions + SET session_id = $sessionId, last_accessed = $timestamp + WHERE user_id = $userId + `, { + $userId: userId, + $sessionId: sessionId, + $timestamp: currentTimestamp + }); + + // If no existing session was updated, create a new one + if (updateResult.changes === 0) { + this.engine.runStatement(` + INSERT INTO sessions (user_id, session_id, created_at, last_accessed) + VALUES ($userId, $sessionId, $timestamp, $timestamp) + `, { + $userId: userId, + $sessionId: sessionId, + $timestamp: currentTimestamp + }); + } + + return sessionId; +}; + +SqlTiddlerDatabase.prototype.findUserBySessionId = function(sessionId) { + // First, get the user_id from the sessions table + const sessionResult = this.engine.runStatementGet(` + SELECT user_id, last_accessed + FROM sessions + WHERE session_id = $sessionId + `, { + $sessionId: sessionId + }); + + if (!sessionResult) { + return null; // Session not found + } + + const lastAccessed = new Date(sessionResult.last_accessed); + const expirationTime = 24 * 60 * 60 * 1000; // 24 hours in milliseconds + if (new Date() - lastAccessed > expirationTime) { + // Session has expired + this.deleteSession(sessionId); + return null; + } + + // Update the last_accessed timestamp + const currentTimestamp = new Date().toISOString(); + this.engine.runStatement(` + UPDATE sessions + SET last_accessed = $timestamp + WHERE session_id = $sessionId + `, { + $sessionId: sessionId, + $timestamp: currentTimestamp + }); + + const userResult = this.engine.runStatementGet(` + SELECT * + FROM users + WHERE user_id = $userId + `, { + $userId: sessionResult.user_id + }); + + if (!userResult) { + return null; + } + + return userResult; +}; + +SqlTiddlerDatabase.prototype.deleteSession = function(sessionId) { + this.engine.runStatement(` + DELETE FROM sessions + WHERE session_id = $sessionId + `, { + $sessionId: sessionId + }); +}; + +SqlTiddlerDatabase.prototype.deleteUserSessions = function(userId) { + this.engine.runStatement(` + DELETE FROM sessions + WHERE user_id = $userId + `, { + $userId: userId + }); +}; + +// Set the user as an admin +SqlTiddlerDatabase.prototype.setUserAdmin = function(userId) { + var admin = this.getRoleByName("ADMIN"); + if(admin) { + this.addRoleToUser(userId, admin.role_id); + } +}; + +// Group CRUD operations +SqlTiddlerDatabase.prototype.createGroup = function(groupName, description) { + const result = this.engine.runStatement(` + INSERT INTO groups (group_name, description) + VALUES ($groupName, $description) + `, { + $groupName: groupName, + $description: description + }); + return result.lastInsertRowid; +}; + +SqlTiddlerDatabase.prototype.getGroup = function(groupId) { + return this.engine.runStatementGet(` + SELECT * FROM groups WHERE group_id = $groupId + `, { + $groupId: groupId + }); +}; + +SqlTiddlerDatabase.prototype.updateGroup = function(groupId, groupName, description) { + this.engine.runStatement(` + UPDATE groups + SET group_name = $groupName, description = $description + WHERE group_id = $groupId + `, { + $groupId: groupId, + $groupName: groupName, + $description: description + }); +}; + +SqlTiddlerDatabase.prototype.deleteGroup = function(groupId) { + this.engine.runStatement(` + DELETE FROM groups WHERE group_id = $groupId + `, { + $groupId: groupId + }); +}; + +SqlTiddlerDatabase.prototype.listGroups = function() { + return this.engine.runStatementGetAll(` + SELECT * FROM groups ORDER BY group_name + `); +}; + +// Role CRUD operations +SqlTiddlerDatabase.prototype.createRole = function(roleName, description) { + const result = this.engine.runStatement(` + INSERT OR IGNORE INTO roles (role_name, description) + VALUES ($roleName, $description) + `, { + $roleName: roleName, + $description: description + }); + return result.lastInsertRowid; +}; + +SqlTiddlerDatabase.prototype.getRole = function(roleId) { + return this.engine.runStatementGet(` + SELECT * FROM roles WHERE role_id = $roleId + `, { + $roleId: roleId + }); +}; + +SqlTiddlerDatabase.prototype.getRoleByName = function(roleName) { + return this.engine.runStatementGet(` + SELECT * FROM roles WHERE role_name = $roleName + `, { + $roleName: roleName + }); +} + +SqlTiddlerDatabase.prototype.updateRole = function(roleId, roleName, description) { + this.engine.runStatement(` + UPDATE roles + SET role_name = $roleName, description = $description WHERE role_id = $roleId + `, { + $roleId: roleId, + $roleName: roleName, + $description: description + }); +}; + +SqlTiddlerDatabase.prototype.deleteRole = function(roleId) { + this.engine.runStatement(` + DELETE FROM roles WHERE role_id = $roleId + `, { + $roleId: roleId + }); +}; + +SqlTiddlerDatabase.prototype.listRoles = function() { + return this.engine.runStatementGetAll(` + SELECT * FROM roles ORDER BY role_name DESC + `); +}; + +// Permission CRUD operations +SqlTiddlerDatabase.prototype.createPermission = function(permissionName, description) { + const result = this.engine.runStatement(` + INSERT OR IGNORE INTO permissions (permission_name, description) + VALUES ($permissionName, $description) + `, { + $permissionName: permissionName, + $description: description + }); + return result.lastInsertRowid; +}; + +SqlTiddlerDatabase.prototype.getPermission = function(permissionId) { + return this.engine.runStatementGet(` + SELECT * FROM permissions WHERE permission_id = $permissionId + `, { + $permissionId: permissionId + }); +}; + +SqlTiddlerDatabase.prototype.getPermissionByName = function(permissionName) { + return this.engine.runStatementGet(` + SELECT * FROM permissions WHERE permission_name = $permissionName + `, { + $permissionName: permissionName + }); +}; + +SqlTiddlerDatabase.prototype.updatePermission = function(permissionId, permissionName, description) { + this.engine.runStatement(` + UPDATE permissions + SET permission_name = $permissionName, description = $description + WHERE permission_id = $permissionId + `, { + $permissionId: permissionId, + $permissionName: permissionName, + $description: description + }); +}; + +SqlTiddlerDatabase.prototype.deletePermission = function(permissionId) { + this.engine.runStatement(` + DELETE FROM permissions WHERE permission_id = $permissionId + `, { + $permissionId: permissionId + }); +}; + +SqlTiddlerDatabase.prototype.listPermissions = function() { + return this.engine.runStatementGetAll(` + SELECT * FROM permissions ORDER BY permission_name + `); +}; + +// ACL CRUD operations +SqlTiddlerDatabase.prototype.createACL = function(entityName, entityType, roleId, permissionId) { + if(!entityName.startsWith("$:/")) { + const result = this.engine.runStatement(` + INSERT OR IGNORE INTO acl (entity_name, entity_type, role_id, permission_id) + VALUES ($entityName, $entityType, $roleId, $permissionId) + `, + { + $entityName: entityName, + $entityType: entityType, + $roleId: roleId, + $permissionId: permissionId + }); + return result.lastInsertRowid; + } +}; + +SqlTiddlerDatabase.prototype.getACL = function(aclId) { + return this.engine.runStatementGet(` + SELECT * FROM acl WHERE acl_id = $aclId + `, { + $aclId: aclId + }); +}; + +SqlTiddlerDatabase.prototype.updateACL = function(aclId, entityId, entityType, roleId, permissionId) { + this.engine.runStatement(` + UPDATE acl + SET entity_name = $entityId, entity_type = $entityType, + role_id = $roleId, permission_id = $permissionId + WHERE acl_id = $aclId + `, { + $aclId: aclId, + $entityId: entityId, + $entityType: entityType, + $roleId: roleId, + $permissionId: permissionId + }); +}; + +SqlTiddlerDatabase.prototype.deleteACL = function(aclId) { + this.engine.runStatement(` + DELETE FROM acl WHERE acl_id = $aclId + `, { + $aclId: aclId + }); +}; + +SqlTiddlerDatabase.prototype.listACLs = function() { + return this.engine.runStatementGetAll(` + SELECT * FROM acl ORDER BY entity_type, entity_name + `); +}; + +// Association management functions +SqlTiddlerDatabase.prototype.addUserToGroup = function(userId, groupId) { + this.engine.runStatement(` + INSERT OR IGNORE INTO user_groups (user_id, group_id) + VALUES ($userId, $groupId) + `, { + $userId: userId, + $groupId: groupId + }); +}; + +SqlTiddlerDatabase.prototype.isUserInGroup = function(userId, groupId) { + const result = this.engine.runStatementGet(` + SELECT 1 FROM user_groups + WHERE user_id = $userId AND group_id = $groupId + `, { + $userId: userId, + $groupId: groupId + }); + return result !== undefined; +}; + +SqlTiddlerDatabase.prototype.removeUserFromGroup = function(userId, groupId) { + this.engine.runStatement(` + DELETE FROM user_groups + WHERE user_id = $userId AND group_id = $groupId + `, { + $userId: userId, + $groupId: groupId + }); +}; + +SqlTiddlerDatabase.prototype.addRoleToUser = function(userId, roleId) { + this.engine.runStatement(` + INSERT OR IGNORE INTO user_roles (user_id, role_id) + VALUES ($userId, $roleId) + `, { + $userId: userId, + $roleId: roleId + }); +}; + +SqlTiddlerDatabase.prototype.removeRoleFromUser = function(userId, roleId) { + this.engine.runStatement(` + DELETE FROM user_roles + WHERE user_id = $userId AND role_id = $roleId + `, { + $userId: userId, + $roleId: roleId + }); +}; + +SqlTiddlerDatabase.prototype.addRoleToGroup = function(groupId, roleId) { + this.engine.runStatement(` + INSERT OR IGNORE INTO group_roles (group_id, role_id) + VALUES ($groupId, $roleId) + `, { + $groupId: groupId, + $roleId: roleId + }); +}; + +SqlTiddlerDatabase.prototype.removeRoleFromGroup = function(groupId, roleId) { + this.engine.runStatement(` + DELETE FROM group_roles + WHERE group_id = $groupId AND role_id = $roleId + `, { + $groupId: groupId, + $roleId: roleId + }); +}; + +SqlTiddlerDatabase.prototype.addPermissionToRole = function(roleId, permissionId) { + this.engine.runStatement(` + INSERT OR IGNORE INTO role_permissions (role_id, permission_id) + VALUES ($roleId, $permissionId) + `, { + $roleId: roleId, + $permissionId: permissionId + }); +}; + +SqlTiddlerDatabase.prototype.removePermissionFromRole = function(roleId, permissionId) { + this.engine.runStatement(` + DELETE FROM role_permissions + WHERE role_id = $roleId AND permission_id = $permissionId + `, { + $roleId: roleId, + $permissionId: permissionId + }); +}; + +SqlTiddlerDatabase.prototype.getUserRoles = function(userId) { + const query = ` + SELECT r.role_id, r.role_name + FROM user_roles ur + JOIN roles r ON ur.role_id = r.role_id + WHERE ur.user_id = $userId LIMIT 1 - `, { - $roleId: roleId - }); + `; - if(userRoleCheck) { - return true; - } - - // Check if the role is used in any ACLs - const aclRoleCheck = this.engine.runStatementGet(` - SELECT 1 - FROM acl + return this.engine.runStatementGet(query, { $userId: userId }); +}; + +SqlTiddlerDatabase.prototype.deleteUserRolesByRoleId = function(roleId) { + this.engine.runStatement(` + DELETE FROM user_roles WHERE role_id = $roleId - LIMIT 1 - `, { + `, { $roleId: roleId - }); - - if(aclRoleCheck) { - return true; - } - - // If we've reached this point, the role is not in use - return false; - }; - - SqlTiddlerDatabase.prototype.getRoleById = function(roleId) { - const role = this.engine.runStatementGet(` - SELECT role_id, role_name, description - FROM roles - WHERE role_id = $roleId - `, { - $roleId: roleId - }); - - return role; - }; - - exports.SqlTiddlerDatabase = SqlTiddlerDatabase; - - })(); - \ No newline at end of file + }); +}; + +SqlTiddlerDatabase.prototype.deleteUserRolesByUserId = function(userId) { + this.engine.runStatement(` + DELETE FROM user_roles + WHERE user_id = $userId + `, { + $userId: userId + }); +}; + +SqlTiddlerDatabase.prototype.isRoleInUse = function(roleId) { + // Check if the role is assigned to any users + const userRoleCheck = this.engine.runStatementGet(` + SELECT 1 + FROM user_roles + WHERE role_id = $roleId + LIMIT 1 + `, { + $roleId: roleId + }); + + if(userRoleCheck) { + return true; + } + + // Check if the role is used in any ACLs + const aclRoleCheck = this.engine.runStatementGet(` + SELECT 1 + FROM acl + WHERE role_id = $roleId + LIMIT 1 + `, { + $roleId: roleId + }); + + if(aclRoleCheck) { + return true; + } + + // If we've reached this point, the role is not in use + return false; +}; + +SqlTiddlerDatabase.prototype.getRoleById = function(roleId) { + const role = this.engine.runStatementGet(` + SELECT role_id, role_name, description + FROM roles + WHERE role_id = $roleId + `, { + $roleId: roleId + }); + + return role; +}; + +exports.SqlTiddlerDatabase = SqlTiddlerDatabase; + +})(); diff --git a/plugins/tiddlywiki/multiwikiserver/modules/store/sql-tiddler-store.js b/plugins/tiddlywiki/multiwikiserver/modules/store/sql-tiddler-store.js index cf75b1e51..1c1dc6442 100644 --- a/plugins/tiddlywiki/multiwikiserver/modules/store/sql-tiddler-store.js +++ b/plugins/tiddlywiki/multiwikiserver/modules/store/sql-tiddler-store.js @@ -15,464 +15,464 @@ This class is largely a wrapper for the sql-tiddler-database.js class, adding th (function() { - /* - Create a tiddler store. Options include: +/* +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; +}; + +/* +Delete a recipe. Returns null on success, or {message:} on error +*/ +SqlTiddlerStore.prototype.deleteRecipe = function(recipe_name) { + var self = this; + return this.sqlTiddlerDatabase.transaction(function() { + // Check if recipe exists + const recipes = self.sqlTiddlerDatabase.listRecipes(); + const recipeExists = recipes.some(recipe => recipe.recipe_name === recipe_name); + if(!recipeExists) { + return {message: "Recipe does not exist"}; + } + + // Delete the recipe + self.sqlTiddlerDatabase.deleteRecipe(recipe_name); + self.dispatchEvent("change"); + return null; + }); +}; + +/* +Delete a bag. Returns null on success, or {message:} on error +*/ +SqlTiddlerStore.prototype.deleteBag = function(bag_name) { + var self = this; + return this.sqlTiddlerDatabase.transaction(function() { + // Check if bag exists + const bags = self.sqlTiddlerDatabase.listBags(); + const bagExists = bags.some(bag => bag.bag_name === bag_name); + if(!bagExists) { + return {message: "Bag does not exist"}; + } + + // Delete the bag + self.sqlTiddlerDatabase.deleteBag(bag_name); + self.dispatchEvent("change"); + 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 ? 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 + }); - 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(); + if(tiddlerFields && tiddlerFields._canonical_uri) { + delete tiddlerFields._canonical_uri; } - SqlTiddlerStore.prototype.addEventListener = function(type,listener) { - this.eventListeners[type] = this.eventListeners[type] || []; - this.eventListeners[type].push(listener); + return { + tiddlerFields: Object.assign({}, tiddlerFields, { text: undefined }), + attachment_blob: attachment_blob }; - - 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); + } 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); } } - }; - - 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)"; - } + }); + 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; - }; - - /* - Delete a recipe. Returns null on success, or {message:} on error - */ - SqlTiddlerStore.prototype.deleteRecipe = function(recipe_name) { - var self = this; - return this.sqlTiddlerDatabase.transaction(function() { - // Check if recipe exists - const recipes = self.sqlTiddlerDatabase.listRecipes(); - const recipeExists = recipes.some(recipe => recipe.recipe_name === recipe_name); - if(!recipeExists) { - return {message: "Recipe does not exist"}; - } - - // Delete the recipe - self.sqlTiddlerDatabase.deleteRecipe(recipe_name); - self.dispatchEvent("change"); - return null; - }); - }; - - /* - Delete a bag. Returns null on success, or {message:} on error - */ - SqlTiddlerStore.prototype.deleteBag = function(bag_name) { - var self = this; - return this.sqlTiddlerDatabase.transaction(function() { - // Check if bag exists - const bags = self.sqlTiddlerDatabase.listBags(); - const bagExists = bags.some(bag => bag.bag_name === bag_name); - if(!bagExists) { - return {message: "Bag does not exist"}; - } - - // Delete the bag - self.sqlTiddlerDatabase.deleteBag(bag_name); - self.dispatchEvent("change"); - 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) { + }); +}; + +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( {}, - tiddlerFields, + this.attachmentStore.getAttachmentStream(tiddlerInfo.attachment_blob), { - text: undefined, - _canonical_uri: `/bags/${$tw.utils.encodeURIComponentExtended(bag_name)}/tiddlers/${$tw.utils.encodeURIComponentExtended(tiddlerFields.title)}/blob` + tiddler_id: tiddlerInfo.tiddler_id, + bag_name: bag_name } ); } 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 ? 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 && 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); - } + 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"); - }; - - 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; - - })(); \ No newline at end of file + }); +}; + +/* +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; + +})(); \ No newline at end of file