mirror of
https://github.com/Jermolene/TiddlyWiki5
synced 2025-01-25 08:26:52 +00:00
19e341c760
Available in Node.js 22.50 - see https://nodejs.org/api/sqlite.html Given the Node.js release timeline, I think it will be reasonable to switch to the integrated Node.js sqlite as the default, which will avoid all the npm install kerfuffle too.
142 lines
3.8 KiB
JavaScript
142 lines
3.8 KiB
JavaScript
/*\
|
|
title: $:/plugins/tiddlywiki/multiwikiserver/store/sql-engine.js
|
|
type: application/javascript
|
|
module-type: library
|
|
|
|
Low level functions to work with the SQLite engine, either via better-sqlite3 or node-sqlite3-wasm.
|
|
|
|
This class is intended to encapsulate all engine-specific logic.
|
|
|
|
\*/
|
|
|
|
(function() {
|
|
|
|
/*
|
|
Create a database engine. Options include:
|
|
|
|
databasePath - path to the database file (can be ":memory:" or missing to get a temporary database)
|
|
engine - wasm | better
|
|
*/
|
|
function SqlEngine(options) {
|
|
options = options || {};
|
|
// Initialise transaction mechanism
|
|
this.transactionDepth = 0;
|
|
// Initialise the statement cache
|
|
this.statements = Object.create(null); // Hashmap by SQL text of statement objects
|
|
// Choose engine
|
|
this.engine = options.engine || "node"; // node | wasm | better
|
|
// Create the database file directories if needed
|
|
if(options.databasePath) {
|
|
$tw.utils.createFileDirectories(options.databasePath);
|
|
}
|
|
// Create the database
|
|
const databasePath = options.databasePath || ":memory:";
|
|
let Database;
|
|
switch(this.engine) {
|
|
case "node":
|
|
({ DatabaseSync: Database } = require("node:sqlite"));
|
|
break;
|
|
case "wasm":
|
|
({ Database } = require("node-sqlite3-wasm"));
|
|
break;
|
|
case "better":
|
|
Database = require("better-sqlite3");
|
|
break;
|
|
}
|
|
this.db = new Database(databasePath,{
|
|
verbose: undefined && console.log
|
|
});
|
|
// Turn on WAL mode for better-sqlite3
|
|
if(this.engine === "better") {
|
|
// See https://github.com/WiseLibs/better-sqlite3/blob/master/docs/performance.md
|
|
this.db.pragma("journal_mode = WAL");
|
|
}
|
|
}
|
|
|
|
SqlEngine.prototype.close = function() {
|
|
for(const sql in this.statements) {
|
|
if(this.statements[sql].finalize) {
|
|
this.statements[sql].finalize();
|
|
}
|
|
}
|
|
this.statements = Object.create(null);
|
|
this.db.close();
|
|
this.db = undefined;
|
|
};
|
|
|
|
SqlEngine.prototype.normaliseParams = function(params) {
|
|
params = params || {};
|
|
const result = Object.create(null);
|
|
for(const paramName in params) {
|
|
if(this.engine !== "wasm" && paramName.startsWith("$")) {
|
|
result[paramName.slice(1)] = params[paramName];
|
|
} else {
|
|
result[paramName] = params[paramName];
|
|
}
|
|
}
|
|
return result;
|
|
};
|
|
|
|
SqlEngine.prototype.prepareStatement = function(sql) {
|
|
if(!(sql in this.statements)) {
|
|
this.statements[sql] = this.db.prepare(sql);
|
|
}
|
|
return this.statements[sql];
|
|
};
|
|
|
|
SqlEngine.prototype.runStatement = function(sql,params) {
|
|
params = this.normaliseParams(params);
|
|
const statement = this.prepareStatement(sql);
|
|
return statement.run(params);
|
|
};
|
|
|
|
SqlEngine.prototype.runStatementGet = function(sql,params) {
|
|
params = this.normaliseParams(params);
|
|
const statement = this.prepareStatement(sql);
|
|
return statement.get(params);
|
|
};
|
|
|
|
SqlEngine.prototype.runStatementGetAll = function(sql,params) {
|
|
params = this.normaliseParams(params);
|
|
const statement = this.prepareStatement(sql);
|
|
return statement.all(params);
|
|
};
|
|
|
|
SqlEngine.prototype.runStatements = function(sqlArray) {
|
|
for(const sql of sqlArray) {
|
|
this.runStatement(sql);
|
|
}
|
|
};
|
|
|
|
/*
|
|
Execute the given function in a transaction, committing if successful but rolling back if an error occurs. Returns whatever the given function returns.
|
|
|
|
Calls to this function can be safely nested, but only the topmost call will actually take place in a transaction.
|
|
|
|
TODO: better-sqlite3 provides its own transaction method which we should be using if available
|
|
*/
|
|
SqlEngine.prototype.transaction = function(fn) {
|
|
const alreadyInTransaction = this.transactionDepth > 0;
|
|
this.transactionDepth++;
|
|
try {
|
|
if(alreadyInTransaction) {
|
|
return fn();
|
|
} else {
|
|
this.runStatement(`BEGIN TRANSACTION`);
|
|
try {
|
|
var result = fn();
|
|
this.runStatement(`COMMIT TRANSACTION`);
|
|
} catch(e) {
|
|
this.runStatement(`ROLLBACK TRANSACTION`);
|
|
throw(e);
|
|
}
|
|
return result;
|
|
}
|
|
} finally {
|
|
this.transactionDepth--;
|
|
}
|
|
};
|
|
|
|
exports.SqlEngine = SqlEngine;
|
|
|
|
})(); |