Add support for databases in other directories
Previously you could use Zotero.DBConnection to open another database in the data directory, but not one stored elsewhere in the filesystem. This allows an absolute path to be passed instead. Various operations (backups, corrupt DB recovery, pragma commands) are disabled for external databases.
This commit is contained in:
parent
603388c79d
commit
f7e411d561
|
@ -31,8 +31,8 @@
|
||||||
// the same database is accessed simultaneously by multiple Zotero instances.
|
// the same database is accessed simultaneously by multiple Zotero instances.
|
||||||
const DB_LOCK_EXCLUSIVE = true;
|
const DB_LOCK_EXCLUSIVE = true;
|
||||||
|
|
||||||
Zotero.DBConnection = function(dbName) {
|
Zotero.DBConnection = function(dbNameOrPath) {
|
||||||
if (!dbName) {
|
if (!dbNameOrPath) {
|
||||||
throw ('DB name not provided in Zotero.DBConnection()');
|
throw ('DB name not provided in Zotero.DBConnection()');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -70,8 +70,18 @@ Zotero.DBConnection = function(dbName) {
|
||||||
return Zotero.Date.toUnixTimestamp(d);
|
return Zotero.Date.toUnixTimestamp(d);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Private members
|
// Absolute path to DB
|
||||||
this._dbName = dbName;
|
if (dbNameOrPath.startsWith('/') || (Zotero.isWin && dbNameOrPath.includes('\\'))) {
|
||||||
|
this._dbName = OS.Path.basename(dbNameOrPath).replace(/\.sqlite$/, '');
|
||||||
|
this._dbPath = dbNameOrPath;
|
||||||
|
this._externalDB = true;
|
||||||
|
}
|
||||||
|
// DB name in data directory
|
||||||
|
else {
|
||||||
|
this._dbName = dbNameOrPath;
|
||||||
|
this._dbPath = Zotero.DataDirectory.getDatabase(dbNameOrPath);
|
||||||
|
this._externalDB = false;
|
||||||
|
}
|
||||||
this._shutdown = false;
|
this._shutdown = false;
|
||||||
this._connection = null;
|
this._connection = null;
|
||||||
this._transactionID = null;
|
this._transactionID = null;
|
||||||
|
@ -91,6 +101,14 @@ Zotero.DBConnection = function(dbName) {
|
||||||
this._dbIsCorrupt = null
|
this._dbIsCorrupt = null
|
||||||
|
|
||||||
this._transactionPromise = null;
|
this._transactionPromise = null;
|
||||||
|
|
||||||
|
if (dbNameOrPath == 'zotero') {
|
||||||
|
this.IncompatibleVersionException = function (msg, dbClientVersion) {
|
||||||
|
this.message = msg;
|
||||||
|
this.dbClientVersion = dbClientVersion;
|
||||||
|
}
|
||||||
|
this.IncompatibleVersionException.prototype = Object.create(Error.prototype);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////
|
||||||
|
@ -105,7 +123,7 @@ Zotero.DBConnection = function(dbName) {
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
Zotero.DBConnection.prototype.test = function () {
|
Zotero.DBConnection.prototype.test = function () {
|
||||||
return this._getConnectionAsync().return();
|
return this._getConnectionAsync().then(() => {});
|
||||||
}
|
}
|
||||||
|
|
||||||
Zotero.DBConnection.prototype.getAsyncStatement = Zotero.Promise.coroutine(function* (sql) {
|
Zotero.DBConnection.prototype.getAsyncStatement = Zotero.Promise.coroutine(function* (sql) {
|
||||||
|
@ -895,9 +913,13 @@ Zotero.DBConnection.prototype.integrityCheck = Zotero.Promise.coroutine(function
|
||||||
|
|
||||||
|
|
||||||
Zotero.DBConnection.prototype.checkException = function (e) {
|
Zotero.DBConnection.prototype.checkException = function (e) {
|
||||||
|
if (this._externalDB) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
if (e.message.includes(this.DB_CORRUPTION_STRING)) {
|
if (e.message.includes(this.DB_CORRUPTION_STRING)) {
|
||||||
// Write corrupt marker to data directory
|
// Write corrupt marker to data directory
|
||||||
var file = Zotero.File.pathToFile(Zotero.DataDirectory.getDatabase(this._dbName, 'is.corrupt'));
|
var file = Zotero.File.pathToFile(this._dbPath + '.is.corrupt');
|
||||||
Zotero.File.putContents(file, '');
|
Zotero.File.putContents(file, '');
|
||||||
|
|
||||||
this._dbIsCorrupt = true;
|
this._dbIsCorrupt = true;
|
||||||
|
@ -948,6 +970,11 @@ Zotero.DBConnection.prototype.closeDatabase = Zotero.Promise.coroutine(function*
|
||||||
|
|
||||||
|
|
||||||
Zotero.DBConnection.prototype.backupDatabase = Zotero.Promise.coroutine(function* (suffix, force) {
|
Zotero.DBConnection.prototype.backupDatabase = Zotero.Promise.coroutine(function* (suffix, force) {
|
||||||
|
if (this.skipBackup || this._externalDB || Zotero.skipLoading) {
|
||||||
|
this._debug("Skipping backup of database '" + this._dbName + "'", 1);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
var storageService = Components.classes["@mozilla.org/storage/service;1"]
|
var storageService = Components.classes["@mozilla.org/storage/service;1"]
|
||||||
.getService(Components.interfaces.mozIStorageService);
|
.getService(Components.interfaces.mozIStorageService);
|
||||||
|
|
||||||
|
@ -981,27 +1008,21 @@ Zotero.DBConnection.prototype.backupDatabase = Zotero.Promise.coroutine(function
|
||||||
});
|
});
|
||||||
|
|
||||||
try {
|
try {
|
||||||
var corruptMarker = Zotero.File.pathToFile(
|
let corruptMarker = Zotero.File.pathToFile(this._dbPath + '.is.corrupt');
|
||||||
Zotero.DataDirectory.getDatabase(this._dbName, 'is.corrupt')
|
|
||||||
);
|
|
||||||
|
|
||||||
if (this.skipBackup || Zotero.skipLoading) {
|
if (this._dbIsCorrupt || corruptMarker.exists()) {
|
||||||
this._debug("Skipping backup of database '" + this._dbName + "'", 1);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
else if (this._dbIsCorrupt || corruptMarker.exists()) {
|
|
||||||
this._debug("Database '" + this._dbName + "' is marked as corrupt -- skipping backup", 1);
|
this._debug("Database '" + this._dbName + "' is marked as corrupt -- skipping backup", 1);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
var file = Zotero.File.pathToFile(Zotero.DataDirectory.getDatabase(this._dbName));
|
let file = this._dbPath;
|
||||||
|
|
||||||
// For standard backup, make sure last backup is old enough to replace
|
// For standard backup, make sure last backup is old enough to replace
|
||||||
if (!suffix && !force) {
|
if (!suffix && !force) {
|
||||||
var backupFile = Zotero.File.pathToFile(Zotero.DataDirectory.getDatabase(this._dbName, 'bak'));
|
let backupFile = this._dbPath + '.bak';
|
||||||
if (yield OS.File.exists(backupFile.path)) {
|
if (yield OS.File.exists(backupFile)) {
|
||||||
var currentDBTime = (yield OS.File.stat(file.path)).lastModificationDate;
|
let currentDBTime = (yield OS.File.stat(file.path)).lastModificationDate;
|
||||||
var lastBackupTime = (yield OS.File.stat(backupFile.path)).lastModificationDate;
|
let lastBackupTime = (yield OS.File.stat(backupFile)).lastModificationDate;
|
||||||
if (currentDBTime == lastBackupTime) {
|
if (currentDBTime == lastBackupTime) {
|
||||||
Zotero.debug("Database '" + this._dbName + "' hasn't changed -- skipping backup");
|
Zotero.debug("Database '" + this._dbName + "' hasn't changed -- skipping backup");
|
||||||
return;
|
return;
|
||||||
|
@ -1022,7 +1043,7 @@ Zotero.DBConnection.prototype.backupDatabase = Zotero.Promise.coroutine(function
|
||||||
|
|
||||||
// Copy via a temporary file so we don't run into disk space issues
|
// Copy via a temporary file so we don't run into disk space issues
|
||||||
// after deleting the old backup file
|
// after deleting the old backup file
|
||||||
var tmpFile = Zotero.DataDirectory.getDatabase(this._dbName, 'tmp');
|
var tmpFile = this._dbPath + '.tmp';
|
||||||
if (yield OS.File.exists(tmpFile)) {
|
if (yield OS.File.exists(tmpFile)) {
|
||||||
try {
|
try {
|
||||||
yield OS.File.remove(tmpFile);
|
yield OS.File.remove(tmpFile);
|
||||||
|
@ -1041,11 +1062,14 @@ Zotero.DBConnection.prototype.backupDatabase = Zotero.Promise.coroutine(function
|
||||||
if (DB_LOCK_EXCLUSIVE) {
|
if (DB_LOCK_EXCLUSIVE) {
|
||||||
yield this.queryAsync("PRAGMA main.locking_mode=NORMAL", false, { inBackup: true });
|
yield this.queryAsync("PRAGMA main.locking_mode=NORMAL", false, { inBackup: true });
|
||||||
}
|
}
|
||||||
storageService.backupDatabaseFile(file, OS.Path.basename(tmpFile), file.parent);
|
storageService.backupDatabaseFile(
|
||||||
|
Zotero.File.pathToFile(file),
|
||||||
|
OS.Path.basename(tmpFile),
|
||||||
|
Zotero.File.pathToFile(file).parent
|
||||||
|
);
|
||||||
}
|
}
|
||||||
catch (e) {
|
catch (e) {
|
||||||
Zotero.debug(e);
|
Zotero.logError(e);
|
||||||
Components.utils.reportError(e);
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
finally {
|
finally {
|
||||||
|
@ -1081,7 +1105,7 @@ Zotero.DBConnection.prototype.backupDatabase = Zotero.Promise.coroutine(function
|
||||||
// Special backup
|
// Special backup
|
||||||
if (!suffix && numBackups > 1) {
|
if (!suffix && numBackups > 1) {
|
||||||
// Remove oldest backup file
|
// Remove oldest backup file
|
||||||
var targetFile = Zotero.DataDirectory.getDatabase(this._dbName, (numBackups - 1) + '.bak');
|
let targetFile = this._dbPath + '.' + (numBackups - 1) + '.bak';
|
||||||
if (yield OS.File.exists(targetFile)) {
|
if (yield OS.File.exists(targetFile)) {
|
||||||
yield OS.File.remove(targetFile);
|
yield OS.File.remove(targetFile);
|
||||||
}
|
}
|
||||||
|
@ -1091,12 +1115,8 @@ Zotero.DBConnection.prototype.backupDatabase = Zotero.Promise.coroutine(function
|
||||||
var targetNum = i;
|
var targetNum = i;
|
||||||
var sourceNum = targetNum - 1;
|
var sourceNum = targetNum - 1;
|
||||||
|
|
||||||
var targetFile = Zotero.DataDirectory.getDatabase(
|
let targetFile = this._dbPath + '.' + targetNum + '.bak';
|
||||||
this._dbName, targetNum + '.bak'
|
let sourceFile = this._dbPath + '.' + (sourceNum ? sourceNum + '.bak' : 'bak')
|
||||||
);
|
|
||||||
var sourceFile = Zotero.DataDirectory.getDatabase(
|
|
||||||
this._dbName, sourceNum ? sourceNum + '.bak' : 'bak'
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!(yield OS.File.exists(sourceFile))) {
|
if (!(yield OS.File.exists(sourceFile))) {
|
||||||
continue;
|
continue;
|
||||||
|
@ -1108,9 +1128,7 @@ Zotero.DBConnection.prototype.backupDatabase = Zotero.Promise.coroutine(function
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var backupFile = Zotero.DataDirectory.getDatabase(
|
let backupFile = this._dbPath + '.' + (suffix ? suffix + '.' : '') + 'bak';
|
||||||
this._dbName, (suffix ? suffix + '.' : '') + 'bak'
|
|
||||||
);
|
|
||||||
|
|
||||||
// Remove old backup file
|
// Remove old backup file
|
||||||
if (yield OS.File.exists(backupFile)) {
|
if (yield OS.File.exists(backupFile)) {
|
||||||
|
@ -1147,11 +1165,11 @@ Zotero.DBConnection.prototype._getConnection = function (options) {
|
||||||
/*
|
/*
|
||||||
* Retrieve a link to the data store asynchronously
|
* Retrieve a link to the data store asynchronously
|
||||||
*/
|
*/
|
||||||
Zotero.DBConnection.prototype._getConnectionAsync = Zotero.Promise.coroutine(function* (options) {
|
Zotero.DBConnection.prototype._getConnectionAsync = async function (options) {
|
||||||
// If a backup is in progress, wait until it's done
|
// If a backup is in progress, wait until it's done
|
||||||
if (this._backupPromise && this._backupPromise.isPending() && (!options || !options.inBackup)) {
|
if (this._backupPromise && this._backupPromise.isPending() && (!options || !options.inBackup)) {
|
||||||
Zotero.debug("Waiting for database backup to complete", 2);
|
Zotero.debug("Waiting for database backup to complete", 2);
|
||||||
yield this._backupPromise;
|
await this._backupPromise;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this._connection) {
|
if (this._connection) {
|
||||||
|
@ -1162,48 +1180,50 @@ Zotero.DBConnection.prototype._getConnectionAsync = Zotero.Promise.coroutine(fun
|
||||||
}
|
}
|
||||||
|
|
||||||
this._debug("Asynchronously opening database '" + this._dbName + "'");
|
this._debug("Asynchronously opening database '" + this._dbName + "'");
|
||||||
|
Zotero.debug(this._dbPath);
|
||||||
|
|
||||||
// Get the storage service
|
// Get the storage service
|
||||||
var store = Components.classes["@mozilla.org/storage/service;1"].
|
var store = Components.classes["@mozilla.org/storage/service;1"].
|
||||||
getService(Components.interfaces.mozIStorageService);
|
getService(Components.interfaces.mozIStorageService);
|
||||||
|
|
||||||
var file = Zotero.File.pathToFile(Zotero.DataDirectory.getDatabase(this._dbName));
|
var file = this._dbPath;
|
||||||
var backupFile = Zotero.File.pathToFile(Zotero.DataDirectory.getDatabase(this._dbName, 'bak'));
|
var backupFile = this._dbPath + '.bak';
|
||||||
|
var fileName = OS.Path.basename(file);
|
||||||
var fileName = this._dbName + '.sqlite';
|
var corruptMarker = this._dbPath + '.is.corrupt';
|
||||||
|
|
||||||
catchBlock: try {
|
catchBlock: try {
|
||||||
var corruptMarker = Zotero.File.pathToFile(Zotero.DataDirectory.getDatabase(this._dbName, 'is.corrupt'));
|
if (await OS.File.exists(corruptMarker)) {
|
||||||
if (corruptMarker.exists()) {
|
|
||||||
throw new Error(this.DB_CORRUPTION_STRING);
|
throw new Error(this.DB_CORRUPTION_STRING);
|
||||||
}
|
}
|
||||||
this._connection = yield Zotero.Promise.resolve(this.Sqlite.openConnection({
|
this._connection = await Zotero.Promise.resolve(this.Sqlite.openConnection({
|
||||||
path: file.path
|
path: file
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
catch (e) {
|
catch (e) {
|
||||||
|
// Don't deal with corrupted external dbs
|
||||||
|
if (this._externalDB) {
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
|
||||||
Zotero.logError(e);
|
Zotero.logError(e);
|
||||||
|
|
||||||
if (e.message.includes(this.DB_CORRUPTION_STRING)) {
|
if (e.message.includes(this.DB_CORRUPTION_STRING)) {
|
||||||
this._debug("Database file '" + file.leafName + "' corrupted", 1);
|
this._debug(`Database file '${fileName}' corrupted`, 1);
|
||||||
|
|
||||||
// No backup file! Eek!
|
// No backup file! Eek!
|
||||||
if (!backupFile.exists()) {
|
if (!await OS.File.exists(backupFile)) {
|
||||||
this._debug("No backup file for DB '" + this._dbName + "' exists", 1);
|
this._debug("No backup file for DB '" + this._dbName + "' exists", 1);
|
||||||
|
|
||||||
// Save damaged filed
|
// Save damaged filed
|
||||||
this._debug('Saving damaged DB file with .damaged extension', 1);
|
this._debug('Saving damaged DB file with .damaged extension', 1);
|
||||||
var damagedFile = Zotero.File.pathToFile(
|
let damagedFile = this._dbPath + '.damaged';
|
||||||
Zotero.DataDirectory.getDatabase(this._dbName, 'damaged')
|
|
||||||
);
|
|
||||||
Zotero.moveToUnique(file, damagedFile);
|
Zotero.moveToUnique(file, damagedFile);
|
||||||
|
|
||||||
// Create new main database
|
// Create new main database
|
||||||
var file = Zotero.File.pathToFile(Zotero.DataDirectory.getDatabase(this._dbName));
|
|
||||||
this._connection = store.openDatabase(file);
|
this._connection = store.openDatabase(file);
|
||||||
|
|
||||||
if (corruptMarker.exists()) {
|
if (await OS.File.exists(corruptMarker)) {
|
||||||
corruptMarker.remove(null);
|
await OS.File.remove(corruptMarker);
|
||||||
}
|
}
|
||||||
|
|
||||||
Zotero.alert(
|
Zotero.alert(
|
||||||
|
@ -1216,24 +1236,21 @@ Zotero.DBConnection.prototype._getConnectionAsync = Zotero.Promise.coroutine(fun
|
||||||
|
|
||||||
// Save damaged file
|
// Save damaged file
|
||||||
this._debug('Saving damaged DB file with .damaged extension', 1);
|
this._debug('Saving damaged DB file with .damaged extension', 1);
|
||||||
var damagedFile = Zotero.File.pathToFile(
|
let damagedFile = this._dbPath + '.damaged';
|
||||||
Zotero.DataDirectory.getDatabase(this._dbName, 'damaged')
|
|
||||||
);
|
|
||||||
Zotero.moveToUnique(file, damagedFile);
|
Zotero.moveToUnique(file, damagedFile);
|
||||||
|
|
||||||
// Test the backup file
|
// Test the backup file
|
||||||
try {
|
try {
|
||||||
Zotero.debug("Asynchronously opening DB connection");
|
Zotero.debug("Asynchronously opening DB connection");
|
||||||
this._connection = yield Zotero.Promise.resolve(this.Sqlite.openConnection({
|
this._connection = await Zotero.Promise.resolve(this.Sqlite.openConnection({
|
||||||
path: backupFile.path
|
path: backupFile
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
// Can't open backup either
|
// Can't open backup either
|
||||||
catch (e) {
|
catch (e) {
|
||||||
// Create new main database
|
// Create new main database
|
||||||
var file = Zotero.File.pathToFile(Zotero.DataDirectory.getDatabase(this._dbName));
|
this._connection = await Zotero.Promise.resolve(this.Sqlite.openConnection({
|
||||||
this._connection = yield Zotero.Promise.resolve(this.Sqlite.openConnection({
|
path: file
|
||||||
path: file.path
|
|
||||||
}));
|
}));
|
||||||
|
|
||||||
Zotero.alert(
|
Zotero.alert(
|
||||||
|
@ -1242,8 +1259,8 @@ Zotero.DBConnection.prototype._getConnectionAsync = Zotero.Promise.coroutine(fun
|
||||||
Zotero.getString('db.dbRestoreFailed', fileName)
|
Zotero.getString('db.dbRestoreFailed', fileName)
|
||||||
);
|
);
|
||||||
|
|
||||||
if (corruptMarker.exists()) {
|
if (await OS.File.exists(corruptMarker)) {
|
||||||
corruptMarker.remove(null);
|
await OS.File.remove(corruptMarker);
|
||||||
}
|
}
|
||||||
|
|
||||||
break catchBlock;
|
break catchBlock;
|
||||||
|
@ -1254,7 +1271,7 @@ Zotero.DBConnection.prototype._getConnectionAsync = Zotero.Promise.coroutine(fun
|
||||||
// Copy backup file to main DB file
|
// Copy backup file to main DB file
|
||||||
this._debug("Restoring database '" + this._dbName + "' from backup file", 1);
|
this._debug("Restoring database '" + this._dbName + "' from backup file", 1);
|
||||||
try {
|
try {
|
||||||
backupFile.copyTo(backupFile.parent, fileName);
|
await OS.File.copy(backupFile, file);
|
||||||
}
|
}
|
||||||
catch (e) {
|
catch (e) {
|
||||||
// TODO: deal with low disk space
|
// TODO: deal with low disk space
|
||||||
|
@ -1262,8 +1279,7 @@ Zotero.DBConnection.prototype._getConnectionAsync = Zotero.Promise.coroutine(fun
|
||||||
}
|
}
|
||||||
|
|
||||||
// Open restored database
|
// Open restored database
|
||||||
var file = OS.Path.join(Zotero.DataDirectory.dir, fileName);
|
this._connection = await Zotero.Promise.resolve(this.Sqlite.openConnection({
|
||||||
this._connection = yield Zotero.Promise.resolve(this.Sqlite.openConnection({
|
|
||||||
path: file
|
path: file
|
||||||
}));
|
}));
|
||||||
this._debug('Database restored', 1);
|
this._debug('Database restored', 1);
|
||||||
|
@ -1272,13 +1288,13 @@ Zotero.DBConnection.prototype._getConnectionAsync = Zotero.Promise.coroutine(fun
|
||||||
Zotero.getString('general.warning'),
|
Zotero.getString('general.warning'),
|
||||||
Zotero.getString('db.dbRestored', [
|
Zotero.getString('db.dbRestored', [
|
||||||
fileName,
|
fileName,
|
||||||
Zotero.Date.getFileDateString(backupFile),
|
Zotero.Date.getFileDateString(Zotero.File.pathToFile(backupFile)),
|
||||||
Zotero.Date.getFileTimeString(backupFile)
|
Zotero.Date.getFileTimeString(Zotero.File.pathToFile(backupFile))
|
||||||
])
|
])
|
||||||
);
|
);
|
||||||
|
|
||||||
if (corruptMarker.exists()) {
|
if (await OS.File.exists(corruptMarker)) {
|
||||||
corruptMarker.remove(null);
|
await OS.File.remove(corruptMarker);
|
||||||
}
|
}
|
||||||
|
|
||||||
break catchBlock;
|
break catchBlock;
|
||||||
|
@ -1288,44 +1304,36 @@ Zotero.DBConnection.prototype._getConnectionAsync = Zotero.Promise.coroutine(fun
|
||||||
throw (e);
|
throw (e);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (DB_LOCK_EXCLUSIVE) {
|
if (!this._externalDB) {
|
||||||
yield this.queryAsync("PRAGMA main.locking_mode=EXCLUSIVE");
|
if (DB_LOCK_EXCLUSIVE) {
|
||||||
|
await this.queryAsync("PRAGMA main.locking_mode=EXCLUSIVE");
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
await this.queryAsync("PRAGMA main.locking_mode=NORMAL");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set page cache size to 8MB
|
||||||
|
let pageSize = await this.valueQueryAsync("PRAGMA page_size");
|
||||||
|
let cacheSize = 8192000 / pageSize;
|
||||||
|
await this.queryAsync("PRAGMA cache_size=" + cacheSize);
|
||||||
|
|
||||||
|
// Enable foreign key checks
|
||||||
|
await this.queryAsync("PRAGMA foreign_keys=true");
|
||||||
|
|
||||||
|
// Register idle observer for DB backup
|
||||||
|
Zotero.Schema.schemaUpdatePromise.then(() => {
|
||||||
|
Zotero.debug("Initializing DB backup idle observer");
|
||||||
|
var idleService = Components.classes["@mozilla.org/widget/idleservice;1"]
|
||||||
|
.getService(Components.interfaces.nsIIdleService);
|
||||||
|
idleService.addIdleObserver(this, 300);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
else {
|
|
||||||
yield this.queryAsync("PRAGMA main.locking_mode=NORMAL");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set page cache size to 8MB
|
|
||||||
var pageSize = yield this.valueQueryAsync("PRAGMA page_size");
|
|
||||||
var cacheSize = 8192000 / pageSize;
|
|
||||||
yield this.queryAsync("PRAGMA cache_size=" + cacheSize);
|
|
||||||
|
|
||||||
// Enable foreign key checks
|
|
||||||
yield this.queryAsync("PRAGMA foreign_keys=true");
|
|
||||||
|
|
||||||
// Register idle observer for DB backup
|
|
||||||
Zotero.Schema.schemaUpdatePromise.then(() => {
|
|
||||||
Zotero.debug("Initializing DB backup idle observer");
|
|
||||||
var idleService = Components.classes["@mozilla.org/widget/idleservice;1"]
|
|
||||||
.getService(Components.interfaces.nsIIdleService);
|
|
||||||
idleService.addIdleObserver(this, 300);
|
|
||||||
});
|
|
||||||
|
|
||||||
return this._connection;
|
return this._connection;
|
||||||
});
|
};
|
||||||
|
|
||||||
|
|
||||||
Zotero.DBConnection.prototype._debug = function (str, level) {
|
Zotero.DBConnection.prototype._debug = function (str, level) {
|
||||||
var prefix = this._dbName == 'zotero' ? '' : '[' + this._dbName + '] ';
|
var prefix = this._dbName == 'zotero' ? '' : '[' + this._dbName + '] ';
|
||||||
Zotero.debug(prefix + str, level);
|
Zotero.debug(prefix + str, level);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Initialize main database connection
|
|
||||||
Zotero.DB = new Zotero.DBConnection('zotero');
|
|
||||||
|
|
||||||
Zotero.DB.IncompatibleVersionException = function (msg, dbClientVersion) {
|
|
||||||
this.message = msg;
|
|
||||||
this.dbClientVersion = dbClientVersion;
|
|
||||||
}
|
|
||||||
Zotero.DB.IncompatibleVersionException.prototype = Object.create(Error.prototype);
|
|
||||||
|
|
|
@ -877,6 +877,9 @@ Services.scriptloader.loadSubScript("resource://zotero/polyfill.js");
|
||||||
* Initializes the DB connection
|
* Initializes the DB connection
|
||||||
*/
|
*/
|
||||||
var _initDB = Zotero.Promise.coroutine(function* (haveReleasedLock) {
|
var _initDB = Zotero.Promise.coroutine(function* (haveReleasedLock) {
|
||||||
|
// Initialize main database connection
|
||||||
|
Zotero.DB = new Zotero.DBConnection('zotero');
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// Test read access
|
// Test read access
|
||||||
yield Zotero.DB.test();
|
yield Zotero.DB.test();
|
||||||
|
|
Loading…
Reference in New Issue
Block a user