/*\ title: $:/plugins/tiddlywiki/multiwikiserver/store/sql-tiddler-database.js type: application/javascript module-type: library Low level SQL functions to store and retrieve tiddlers in a SQLite database. This class is intended to encapsulate all the SQL queries used to access the database. 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" } }; } 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 (session_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; }; /* 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; }; /* 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, r.owner_id, 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, owner_id: row.owner_id, 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 }); 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}) }; }; /* Checks if a user has permission to access a recipe */ SqlTiddlerDatabase.prototype.hasRecipePermission = function(userId, recipeName, permissionName) { try { // 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 && recipe?.owner_id === userId) { return true; } else { var permission = this.checkACLPermission(userId, "recipe", recipeName, permissionName, recipe?.owner_id) return permission; } } catch (error) { console.error(error) return false } }; /* 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 acl.*, permissions.permission_name FROM acl LEFT JOIN permissions ON acl.permission_id = permissions.permission_id WHERE acl.entity_type = $entity_type AND acl.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, permissionName, ownerId) { try { // if the entityName starts with "$:/", we'll assume its a system bag/recipe, then grant the user permission if(entityName.startsWith("$:/")) { return true; } const aclRecords = this.getACLByName(entityType, entityName, true); const aclRecord = aclRecords.find(record => record.permission_name === permissionName); // If no ACL record exists, return true for hasPermission if ((!aclRecord && !ownerId && aclRecords.length === 0) || ((!!aclRecord && !!ownerId) && ownerId === userId)) { return true; } // If ACL record exists, check for user permission using the retrieved permission_id const checkPermissionQuery = ` SELECT * 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; } catch (error) { console.error(error); return false } }; /** * 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 }); } 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 }); 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.createUserSession = function(userId, sessionId) { const currentTimestamp = new Date().toISOString(); 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 `; 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 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; })();