/g, '')
+ .replace(/<\/m:center>/g, '
')
+ //
+ .replace(//g, '')
+ .replace(/<\/m:underline>/g, '');
+};
+
+Zotero_Import_Mendeley.prototype._saveItems = async function (libraryID, json) {
+ var idMap = new Map();
+ await Zotero.DB.executeTransaction(async function () {
+ for (let itemJSON of json) {
+ let item = new Zotero.Item;
+ item.libraryID = libraryID;
+ if (itemJSON.key) {
+ item.key = itemJSON.key;
+ await item.loadPrimaryData();
+ }
+
+ // Remove external id before save
+ let toSave = Object.assign({}, itemJSON);
+ delete toSave.documentID;
+
+ item.fromJSON(toSave);
+ await item.save({
+ skipSelect: true,
+ skipDateModifiedUpdate: true
+ });
+ if (itemJSON.documentID) {
+ idMap.set(itemJSON.documentID, item.id);
+ }
+ }
+ }.bind(this));
+ return idMap;
+};
+
+/**
+ * Saves attachments and extracted annotations for a given document
+ */
+Zotero_Import_Mendeley.prototype._saveFilesAndAnnotations = async function (files, libraryID, parentItemID, annotations) {
+ for (let file of files) {
+ try {
+ if (!file.fileURL) continue;
+
+ let path = OS.Path.fromFileURI(file.fileURL);
+
+ let attachment;
+ if (await OS.File.exists(path)) {
+ let options = {
+ libraryID,
+ parentItemID,
+ file: path
+ };
+ // If file is in Mendeley downloads folder, import it
+ if (OS.Path.dirname(path).endsWith(OS.Path.join('Mendeley Desktop', 'Downloaded'))) {
+ attachment = await Zotero.Attachments.importFromFile(options);
+ }
+ // Otherwise link it
+ else {
+ attachment = await Zotero.Attachments.linkFromFile(options);
+ }
+ attachment.relations = {
+ 'mendeleyDB:fileHash': file.hash,
+ 'mendeleyDB:fileUUID': file.uuid
+ };
+ await attachment.saveTx({
+ skipSelect: true
+ });
+ }
+ else {
+ Zotero.warn(path + " not found -- not importing");
+ }
+
+ if (annotations) {
+ await this._saveAnnotations(
+ // We have annotations from all files for this document, so limit to just those on
+ // this file
+ annotations.filter(a => a.hash == file.hash),
+ parentItemID,
+ attachment ? attachment.id : null
+ );
+ }
+ }
+ catch (e) {
+ Zotero.logError(e);
+ }
+ }
+}
+
+Zotero_Import_Mendeley.prototype._saveAnnotations = async function (annotations, parentItemID, attachmentItemID) {
+ if (!annotations.length) return;
+ var noteStrings = [];
+ var parentItem = Zotero.Items.get(parentItemID);
+ var libraryID = parentItem.libraryID;
+ if (attachmentItemID) {
+ var attachmentItem = Zotero.Items.get(attachmentItemID);
+ var attachmentURIPath = Zotero.API.getLibraryPrefix(libraryID) + '/items/' + attachmentItem.key;
+ }
+
+ for (let annotation of annotations) {
+ if (!annotation.note || !annotation.note.trim()) continue;
+
+ let linkStr;
+ let linkText = `note on p. ${annotation.page}`;
+ if (attachmentItem) {
+ let url = `zotero://open-pdf/${attachmentURIPath}?page=${annotation.page}`;
+ linkStr = `${linkText}`;
+ }
+ else {
+ linkStr = linkText;
+ }
+
+ noteStrings.push(
+ Zotero.Utilities.text2html(annotation.note.trim())
+ + `(${linkStr})
`
+ );
+ }
+
+ if (!noteStrings.length) return;
+
+ let note = new Zotero.Item('note');
+ note.libraryID = libraryID;
+ note.parentItemID = parentItemID;
+ note.setNote('' + Zotero.getString('extractedAnnotations') + '
\n' + noteStrings.join('\n'));
+ return note.saveTx({
+ skipSelect: true
+ });
+};
diff --git a/chrome/content/zotero/import/mendeley/mendeleySchemaMap.js b/chrome/content/zotero/import/mendeley/mendeleySchemaMap.js
new file mode 100644
index 000000000..af8e929e3
--- /dev/null
+++ b/chrome/content/zotero/import/mendeley/mendeleySchemaMap.js
@@ -0,0 +1,102 @@
+var map = {
+ 83: {
+ itemTypes: {
+ Bill: "bill",
+ Book: "book",
+ BookSection: "bookSection",
+ Case: "case",
+ ComputerProgram: "computerProgram",
+ ConferenceProceedings: "conferencePaper",
+ EncyclopediaArticle: "encyclopediaArticle",
+ Film: "film",
+ Generic: "document",
+ JournalArticle: "journalArticle",
+ MagazineArticle: "magazineArticle",
+ NewspaperArticle: "newspaperArticle",
+ Patent: "patent",
+ Report: "report",
+ Statute: "statute",
+ TelevisionBroadcast: "tvBroadcast",
+ Thesis: "thesis",
+ WebPage: "webpage",
+ WorkingPaper: "report"
+ },
+ fields: {
+ id: "",
+ uuid: "",
+ reviewedArticle: "",
+ revisionNumber: "",
+ publisher: "publisher",
+ reprintEdition: "",
+ series: "seriesTitle",
+ seriesNumber: "seriesNumber",
+ sections: "section",
+ seriesEditor: "creator[seriesEditor]", // falls back to editor if necessary
+ owner: "",
+ pages: "func[pages]",
+ month: "", // handled explicitly
+ originalPublication: "",
+ publication: "publicationTitle",
+ publicLawNumber: "publicLawNumber",
+ pmid: "extra[PMID]",
+ sourceType: "",
+ session: "session",
+ shortTitle: "shortTitle",
+ volume: "volume",
+ year: "", // handled explicitly
+ userType: "type",
+ country: "place[country]",
+ dateAccessed: "accessDate",
+ committee: "committee",
+ counsel: "creator[counsel]",
+ doi: "DOI",
+ edition: "edition",
+ day: "", // handled explicitly
+ department: "",
+ citationKey: "citationKey", // put in Extra
+ city: "place[city]",
+ chapter: "",
+ codeSection: "section",
+ codeVolume: "codeVolume",
+ code: "code",
+ codeNumber: "codeNumber",
+ issue: "issue",
+ language: "language",
+ isbn: "ISBN",
+ issn: "ISSN",
+ length: "",
+ medium: "medium",
+ lastUpdate: "",
+ legalStatus: "legalStatus",
+ hideFromMendeleyWebIndex: "",
+ institution: "publisher",
+ genre: "genre",
+ internationalTitle: "",
+ internationalUserType: "",
+ internationalAuthor: "",
+ internationalNumber: "",
+ deletionPending: "",
+ favourite: "", // tag?
+ confirmed: "", // tag?
+ deduplicated: "",
+ read: "", // tag?
+ type: "", // item type handled separately
+ title: "title",
+ privacy: "",
+ applicationNumber: "applicationNumber",
+ arxivId: "extra[arXiv]",
+ advisor: "",
+ articleColumn: "",
+ modified: "func[fromUnixtime:dateModified]",
+ abstract: "abstractNote",
+ added: "func[fromUnixtime:dateAdded]",
+ note: "func[note]",
+ importer: ""
+ },
+ creatorTypes: {
+ DocumentAuthor: "author",
+ DocumentEditor: "editor",
+ DocumentTranslator: "translator"
+ }
+ }
+};
diff --git a/chrome/content/zotero/xpcom/api.js b/chrome/content/zotero/xpcom/api.js
index f3aa9b040..6d69991f9 100644
--- a/chrome/content/zotero/xpcom/api.js
+++ b/chrome/content/zotero/xpcom/api.js
@@ -154,7 +154,7 @@ Zotero.API = {
return 'groups/' + Zotero.Groups.getGroupIDFromLibraryID(libraryID);
default:
- throw new Error(`Invalid type '${type}`);
+ throw new Error(`Invalid type '${type}'`);
}
}
};
diff --git a/chrome/content/zotero/xpcom/data/collection.js b/chrome/content/zotero/xpcom/data/collection.js
index 3a0f5bf75..8d263aae3 100644
--- a/chrome/content/zotero/xpcom/data/collection.js
+++ b/chrome/content/zotero/xpcom/data/collection.js
@@ -698,8 +698,7 @@ Zotero.Collection.prototype.fromJSON = function (json) {
this.name = json.name;
this.parentKey = json.parentCollection ? json.parentCollection : false;
- // TODO
- //this.setRelations(json.relations);
+ this.setRelations(json.relations);
}
@@ -713,7 +712,7 @@ Zotero.Collection.prototype.toJSON = function (options = {}) {
obj.name = this.name;
obj.parentCollection = this.parentKey ? this.parentKey : false;
- obj.relations = {}; // TEMP
+ obj.relations = this.getRelations();
return this._postToJSON(env);
}
diff --git a/chrome/content/zotero/xpcom/data/dataObject.js b/chrome/content/zotero/xpcom/data/dataObject.js
index 342aeede2..4d2d29f34 100644
--- a/chrome/content/zotero/xpcom/data/dataObject.js
+++ b/chrome/content/zotero/xpcom/data/dataObject.js
@@ -285,7 +285,7 @@ Zotero.DataObject.prototype._setParentKey = function(key) {
/**
* Returns all relations of the object
*
- * @return {Object} - Object with predicates as keys and arrays of URIs as values
+ * @return {Object} - Object with predicates as keys and arrays of values
*/
Zotero.DataObject.prototype.getRelations = function () {
this._requireData('relations');
@@ -410,7 +410,7 @@ Zotero.DataObject.prototype.setRelations = function (newRelations) {
// Limit predicates to letters and colons for now
for (let p in newRelations) {
- if (!/[a-z]+:[a-z]+/.test(p)) {
+ if (!/^[a-z]+:[a-z]+$/i.test(p)) {
throw new Error(`Invalid relation predicate '${p}'`);
}
}
diff --git a/chrome/content/zotero/xpcom/data/item.js b/chrome/content/zotero/xpcom/data/item.js
index 4aecb5ebe..52d1f9c53 100644
--- a/chrome/content/zotero/xpcom/data/item.js
+++ b/chrome/content/zotero/xpcom/data/item.js
@@ -1051,8 +1051,7 @@ Zotero.Item.prototype.setCreator = function (orderIndex, data) {
var msg = "Creator type '" + Zotero.CreatorTypes.getName(data.creatorTypeID) + "' "
+ "isn't valid for " + Zotero.ItemTypes.getName(itemTypeID)
+ " -- changing to primary creator";
- Zotero.debug(msg, 2);
- Components.utils.reportError(msg);
+ Zotero.warn(msg);
data.creatorTypeID = Zotero.CreatorTypes.getPrimaryIDForType(itemTypeID);
}
diff --git a/chrome/content/zotero/xpcom/data/itemFields.js b/chrome/content/zotero/xpcom/data/itemFields.js
index 67bb7053f..c1ac1f126 100644
--- a/chrome/content/zotero/xpcom/data/itemFields.js
+++ b/chrome/content/zotero/xpcom/data/itemFields.js
@@ -260,6 +260,11 @@ Zotero.ItemFields = new function() {
throw new Error("Invalid field '" + baseField + '" for base field');
}
+ // If field isn't a base field, return it if it's valid for the type
+ if (!this.isBaseField(baseFieldID)) {
+ return this.isValidForType(baseFieldID, itemTypeID) ? baseFieldID : false;
+ }
+
return _baseTypeFields[itemTypeID][baseFieldID];
}
diff --git a/chrome/content/zotero/xpcom/data/items.js b/chrome/content/zotero/xpcom/data/items.js
index 7be2e13cd..c61192e21 100644
--- a/chrome/content/zotero/xpcom/data/items.js
+++ b/chrome/content/zotero/xpcom/data/items.js
@@ -315,7 +315,21 @@ Zotero.Items = function() {
item._clearChanged('itemData');
// Display titles
- item.updateDisplayTitle()
+ try {
+ item.updateDisplayTitle()
+ }
+ catch (e) {
+ // A few item types need creators to be loaded. Instead of making
+ // updateDisplayTitle() async and loading conditionally, just catch the error
+ // and load on demand
+ if (e instanceof Zotero.Exception.UnloadedDataException) {
+ yield item.loadDataType('creators');
+ item.updateDisplayTitle()
+ }
+ else {
+ throw e;
+ }
+ }
}
});
diff --git a/chrome/content/zotero/xpcom/data/relations.js b/chrome/content/zotero/xpcom/data/relations.js
index 3b259fda8..66c6dcbd0 100644
--- a/chrome/content/zotero/xpcom/data/relations.js
+++ b/chrome/content/zotero/xpcom/data/relations.js
@@ -32,7 +32,8 @@ Zotero.Relations = new function () {
this._namespaces = {
dc: 'http://purl.org/dc/elements/1.1/',
- owl: 'http://www.w3.org/2002/07/owl#'
+ owl: 'http://www.w3.org/2002/07/owl#',
+ mendeleyDB: 'http://zotero.org/namespaces/mendeleyDB#'
};
var _types = ['collection', 'item'];
diff --git a/chrome/content/zotero/xpcom/db.js b/chrome/content/zotero/xpcom/db.js
index 9450881d4..b3886b622 100644
--- a/chrome/content/zotero/xpcom/db.js
+++ b/chrome/content/zotero/xpcom/db.js
@@ -31,8 +31,8 @@
// the same database is accessed simultaneously by multiple Zotero instances.
const DB_LOCK_EXCLUSIVE = true;
-Zotero.DBConnection = function(dbName) {
- if (!dbName) {
+Zotero.DBConnection = function(dbNameOrPath) {
+ if (!dbNameOrPath) {
throw ('DB name not provided in Zotero.DBConnection()');
}
@@ -70,8 +70,18 @@ Zotero.DBConnection = function(dbName) {
return Zotero.Date.toUnixTimestamp(d);
});
- // Private members
- this._dbName = dbName;
+ // Absolute path to DB
+ 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._connection = null;
this._transactionID = null;
@@ -91,6 +101,14 @@ Zotero.DBConnection = function(dbName) {
this._dbIsCorrupt = 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
*/
Zotero.DBConnection.prototype.test = function () {
- return this._getConnectionAsync().return();
+ return this._getConnectionAsync().then(() => {});
}
Zotero.DBConnection.prototype.getAsyncStatement = Zotero.Promise.coroutine(function* (sql) {
@@ -818,9 +836,10 @@ Zotero.DBConnection.prototype.logQuery = function (sql, params = [], options) {
}
-Zotero.DBConnection.prototype.tableExists = Zotero.Promise.coroutine(function* (table) {
+Zotero.DBConnection.prototype.tableExists = Zotero.Promise.coroutine(function* (table, db) {
yield this._getConnectionAsync();
- var sql = "SELECT COUNT(*) FROM sqlite_master WHERE type='table' AND tbl_name=?";
+ var prefix = db ? db + '.' : '';
+ var sql = `SELECT COUNT(*) FROM ${prefix}sqlite_master WHERE type='table' AND tbl_name=?`;
var count = yield this.valueQueryAsync(sql, [table]);
return !!count;
});
@@ -879,7 +898,7 @@ Zotero.DBConnection.prototype.vacuum = function () {
// TEMP
Zotero.DBConnection.prototype.info = Zotero.Promise.coroutine(function* () {
var info = {};
- var pragmas = ['auto_vacuum', 'cache_size', 'locking_mode', 'page_size'];
+ var pragmas = ['auto_vacuum', 'cache_size', 'main.locking_mode', 'page_size'];
for (let p of pragmas) {
info[p] = yield Zotero.DB.valueQueryAsync(`PRAGMA ${p}`);
}
@@ -894,9 +913,13 @@ Zotero.DBConnection.prototype.integrityCheck = Zotero.Promise.coroutine(function
Zotero.DBConnection.prototype.checkException = function (e) {
+ if (this._externalDB) {
+ return true;
+ }
+
if (e.message.includes(this.DB_CORRUPTION_STRING)) {
// 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, '');
this._dbIsCorrupt = true;
@@ -947,6 +970,11 @@ Zotero.DBConnection.prototype.closeDatabase = Zotero.Promise.coroutine(function*
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"]
.getService(Components.interfaces.mozIStorageService);
@@ -980,27 +1008,21 @@ Zotero.DBConnection.prototype.backupDatabase = Zotero.Promise.coroutine(function
});
try {
- var corruptMarker = Zotero.File.pathToFile(
- Zotero.DataDirectory.getDatabase(this._dbName, 'is.corrupt')
- );
+ let corruptMarker = Zotero.File.pathToFile(this._dbPath + '.is.corrupt');
- if (this.skipBackup || Zotero.skipLoading) {
- this._debug("Skipping backup of database '" + this._dbName + "'", 1);
- return false;
- }
- else if (this._dbIsCorrupt || corruptMarker.exists()) {
+ if (this._dbIsCorrupt || corruptMarker.exists()) {
this._debug("Database '" + this._dbName + "' is marked as corrupt -- skipping backup", 1);
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
if (!suffix && !force) {
- var backupFile = Zotero.File.pathToFile(Zotero.DataDirectory.getDatabase(this._dbName, 'bak'));
- if (yield OS.File.exists(backupFile.path)) {
- var currentDBTime = (yield OS.File.stat(file.path)).lastModificationDate;
- var lastBackupTime = (yield OS.File.stat(backupFile.path)).lastModificationDate;
+ let backupFile = this._dbPath + '.bak';
+ if (yield OS.File.exists(backupFile)) {
+ let currentDBTime = (yield OS.File.stat(file.path)).lastModificationDate;
+ let lastBackupTime = (yield OS.File.stat(backupFile)).lastModificationDate;
if (currentDBTime == lastBackupTime) {
Zotero.debug("Database '" + this._dbName + "' hasn't changed -- skipping backup");
return;
@@ -1021,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
// 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)) {
try {
yield OS.File.remove(tmpFile);
@@ -1038,18 +1060,21 @@ Zotero.DBConnection.prototype.backupDatabase = Zotero.Promise.coroutine(function
// the lock is lost
try {
if (DB_LOCK_EXCLUSIVE) {
- yield this.queryAsync("PRAGMA 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) {
- Zotero.debug(e);
- Components.utils.reportError(e);
+ Zotero.logError(e);
return false;
}
finally {
if (DB_LOCK_EXCLUSIVE) {
- yield this.queryAsync("PRAGMA locking_mode=EXCLUSIVE", false, { inBackup: true });
+ yield this.queryAsync("PRAGMA main.locking_mode=EXCLUSIVE", false, { inBackup: true });
}
}
@@ -1080,7 +1105,7 @@ Zotero.DBConnection.prototype.backupDatabase = Zotero.Promise.coroutine(function
// Special backup
if (!suffix && numBackups > 1) {
// 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)) {
yield OS.File.remove(targetFile);
}
@@ -1090,12 +1115,8 @@ Zotero.DBConnection.prototype.backupDatabase = Zotero.Promise.coroutine(function
var targetNum = i;
var sourceNum = targetNum - 1;
- var targetFile = Zotero.DataDirectory.getDatabase(
- this._dbName, targetNum + '.bak'
- );
- var sourceFile = Zotero.DataDirectory.getDatabase(
- this._dbName, sourceNum ? sourceNum + '.bak' : 'bak'
- );
+ let targetFile = this._dbPath + '.' + targetNum + '.bak';
+ let sourceFile = this._dbPath + '.' + (sourceNum ? sourceNum + '.bak' : 'bak')
if (!(yield OS.File.exists(sourceFile))) {
continue;
@@ -1107,9 +1128,7 @@ Zotero.DBConnection.prototype.backupDatabase = Zotero.Promise.coroutine(function
}
}
- var backupFile = Zotero.DataDirectory.getDatabase(
- this._dbName, (suffix ? suffix + '.' : '') + 'bak'
- );
+ let backupFile = this._dbPath + '.' + (suffix ? suffix + '.' : '') + 'bak';
// Remove old backup file
if (yield OS.File.exists(backupFile)) {
@@ -1146,11 +1165,11 @@ Zotero.DBConnection.prototype._getConnection = function (options) {
/*
* 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 (this._backupPromise && this._backupPromise.isPending() && (!options || !options.inBackup)) {
Zotero.debug("Waiting for database backup to complete", 2);
- yield this._backupPromise;
+ await this._backupPromise;
}
if (this._connection) {
@@ -1161,48 +1180,50 @@ Zotero.DBConnection.prototype._getConnectionAsync = Zotero.Promise.coroutine(fun
}
this._debug("Asynchronously opening database '" + this._dbName + "'");
+ Zotero.debug(this._dbPath);
// Get the storage service
var store = Components.classes["@mozilla.org/storage/service;1"].
getService(Components.interfaces.mozIStorageService);
- var file = Zotero.File.pathToFile(Zotero.DataDirectory.getDatabase(this._dbName));
- var backupFile = Zotero.File.pathToFile(Zotero.DataDirectory.getDatabase(this._dbName, 'bak'));
-
- var fileName = this._dbName + '.sqlite';
+ var file = this._dbPath;
+ var backupFile = this._dbPath + '.bak';
+ var fileName = OS.Path.basename(file);
+ var corruptMarker = this._dbPath + '.is.corrupt';
catchBlock: try {
- var corruptMarker = Zotero.File.pathToFile(Zotero.DataDirectory.getDatabase(this._dbName, 'is.corrupt'));
- if (corruptMarker.exists()) {
+ if (await OS.File.exists(corruptMarker)) {
throw new Error(this.DB_CORRUPTION_STRING);
}
- this._connection = yield Zotero.Promise.resolve(this.Sqlite.openConnection({
- path: file.path
+ this._connection = await Zotero.Promise.resolve(this.Sqlite.openConnection({
+ path: file
}));
}
catch (e) {
+ // Don't deal with corrupted external dbs
+ if (this._externalDB) {
+ throw e;
+ }
+
Zotero.logError(e);
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!
- if (!backupFile.exists()) {
+ if (!await OS.File.exists(backupFile)) {
this._debug("No backup file for DB '" + this._dbName + "' exists", 1);
// Save damaged filed
this._debug('Saving damaged DB file with .damaged extension', 1);
- var damagedFile = Zotero.File.pathToFile(
- Zotero.DataDirectory.getDatabase(this._dbName, 'damaged')
- );
+ let damagedFile = this._dbPath + '.damaged';
Zotero.moveToUnique(file, damagedFile);
// Create new main database
- var file = Zotero.File.pathToFile(Zotero.DataDirectory.getDatabase(this._dbName));
this._connection = store.openDatabase(file);
- if (corruptMarker.exists()) {
- corruptMarker.remove(null);
+ if (await OS.File.exists(corruptMarker)) {
+ await OS.File.remove(corruptMarker);
}
Zotero.alert(
@@ -1215,24 +1236,21 @@ Zotero.DBConnection.prototype._getConnectionAsync = Zotero.Promise.coroutine(fun
// Save damaged file
this._debug('Saving damaged DB file with .damaged extension', 1);
- var damagedFile = Zotero.File.pathToFile(
- Zotero.DataDirectory.getDatabase(this._dbName, 'damaged')
- );
+ let damagedFile = this._dbPath + '.damaged';
Zotero.moveToUnique(file, damagedFile);
// Test the backup file
try {
Zotero.debug("Asynchronously opening DB connection");
- this._connection = yield Zotero.Promise.resolve(this.Sqlite.openConnection({
- path: backupFile.path
+ this._connection = await Zotero.Promise.resolve(this.Sqlite.openConnection({
+ path: backupFile
}));
}
// Can't open backup either
catch (e) {
// Create new main database
- var file = Zotero.File.pathToFile(Zotero.DataDirectory.getDatabase(this._dbName));
- this._connection = yield Zotero.Promise.resolve(this.Sqlite.openConnection({
- path: file.path
+ this._connection = await Zotero.Promise.resolve(this.Sqlite.openConnection({
+ path: file
}));
Zotero.alert(
@@ -1241,8 +1259,8 @@ Zotero.DBConnection.prototype._getConnectionAsync = Zotero.Promise.coroutine(fun
Zotero.getString('db.dbRestoreFailed', fileName)
);
- if (corruptMarker.exists()) {
- corruptMarker.remove(null);
+ if (await OS.File.exists(corruptMarker)) {
+ await OS.File.remove(corruptMarker);
}
break catchBlock;
@@ -1253,7 +1271,7 @@ Zotero.DBConnection.prototype._getConnectionAsync = Zotero.Promise.coroutine(fun
// Copy backup file to main DB file
this._debug("Restoring database '" + this._dbName + "' from backup file", 1);
try {
- backupFile.copyTo(backupFile.parent, fileName);
+ await OS.File.copy(backupFile, file);
}
catch (e) {
// TODO: deal with low disk space
@@ -1261,8 +1279,7 @@ Zotero.DBConnection.prototype._getConnectionAsync = Zotero.Promise.coroutine(fun
}
// Open restored database
- var file = OS.Path.join(Zotero.DataDirectory.dir, fileName);
- this._connection = yield Zotero.Promise.resolve(this.Sqlite.openConnection({
+ this._connection = await Zotero.Promise.resolve(this.Sqlite.openConnection({
path: file
}));
this._debug('Database restored', 1);
@@ -1271,13 +1288,13 @@ Zotero.DBConnection.prototype._getConnectionAsync = Zotero.Promise.coroutine(fun
Zotero.getString('general.warning'),
Zotero.getString('db.dbRestored', [
fileName,
- Zotero.Date.getFileDateString(backupFile),
- Zotero.Date.getFileTimeString(backupFile)
+ Zotero.Date.getFileDateString(Zotero.File.pathToFile(backupFile)),
+ Zotero.Date.getFileTimeString(Zotero.File.pathToFile(backupFile))
])
);
- if (corruptMarker.exists()) {
- corruptMarker.remove(null);
+ if (await OS.File.exists(corruptMarker)) {
+ await OS.File.remove(corruptMarker);
}
break catchBlock;
@@ -1287,44 +1304,36 @@ Zotero.DBConnection.prototype._getConnectionAsync = Zotero.Promise.coroutine(fun
throw (e);
}
- if (DB_LOCK_EXCLUSIVE) {
- yield this.queryAsync("PRAGMA locking_mode=EXCLUSIVE");
+ if (!this._externalDB) {
+ 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 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;
-});
+};
Zotero.DBConnection.prototype._debug = function (str, level) {
var prefix = this._dbName == 'zotero' ? '' : '[' + this._dbName + '] ';
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);
diff --git a/chrome/content/zotero/xpcom/mime.js b/chrome/content/zotero/xpcom/mime.js
index 77419507c..66b87cce2 100644
--- a/chrome/content/zotero/xpcom/mime.js
+++ b/chrome/content/zotero/xpcom/mime.js
@@ -26,7 +26,6 @@
Zotero.MIME = new function(){
this.isTextType = isTextType;
this.getPrimaryExtension = getPrimaryExtension;
- this.sniffForMIMEType = sniffForMIMEType;
this.sniffForBinary = sniffForBinary;
this.hasNativeHandler = hasNativeHandler;
this.hasInternalHandler = hasInternalHandler;
@@ -48,8 +47,9 @@ Zotero.MIME = new function(){
["\uFFFDPNG", 'image/png', 0],
["JFIF", 'image/jpeg'],
["FLV", "video/x-flv", 0],
- ["\u0000\u0000\u0001\u0000", "image/vnd.microsoft.icon", 0]
-
+ ["\u0000\u0000\u0001\u0000", "image/vnd.microsoft.icon", 0],
+ ["\u0053\u0051\u004C\u0069\u0074\u0065\u0020\u0066"
+ + "\u006F\u0072\u006D\u0061\u0074\u0020\u0033\u0000", "application/x-sqlite3", 0]
];
var _extensions = {
@@ -228,12 +228,12 @@ Zotero.MIME = new function(){
/*
* Searches string for magic numbers
*/
- function sniffForMIMEType(str){
- for (var i in _snifferEntries){
- var match = false;
+ this.sniffForMIMEType = function (str) {
+ for (let i in _snifferEntries) {
+ let match = false;
// If an offset is defined, match only from there
- if (typeof _snifferEntries[i][2] != 'undefined') {
- if (str.substr(i[2]).indexOf(_snifferEntries[i][0]) == 0) {
+ if (_snifferEntries[i][2] != undefined) {
+ if (str.substr(_snifferEntries[i][2]).indexOf(_snifferEntries[i][0]) == 0) {
match = true;
}
}
@@ -274,7 +274,7 @@ Zotero.MIME = new function(){
* ext is an optional file extension hint if data sniffing is unsuccessful
*/
this.getMIMETypeFromData = function (str, ext){
- var mimeType = sniffForMIMEType(str);
+ var mimeType = this.sniffForMIMEType(str);
if (mimeType){
Zotero.debug('Detected MIME type ' + mimeType);
return mimeType;
diff --git a/chrome/content/zotero/xpcom/sync/syncRunner.js b/chrome/content/zotero/xpcom/sync/syncRunner.js
index 1b4a5e48d..e6e3b5083 100644
--- a/chrome/content/zotero/xpcom/sync/syncRunner.js
+++ b/chrome/content/zotero/xpcom/sync/syncRunner.js
@@ -70,6 +70,7 @@ Zotero.Sync.Runner_Module = function (options = {}) {
var _enabled = false;
var _autoSyncTimer;
var _delaySyncUntil;
+ var _delayPromises = [];
var _firstInSession = true;
var _syncInProgress = false;
var _stopping = false;
@@ -148,6 +149,21 @@ Zotero.Sync.Runner_Module = function (options = {}) {
yield Zotero.Promise.delay(delay);
}
+ // If paused, wait until we're done
+ while (true) {
+ if (_delayPromises.some(p => p.isPending())) {
+ this.setSyncStatus(Zotero.getString('sync.status.waiting'));
+ Zotero.debug("Syncing is paused -- waiting to sync");
+ yield Zotero.Promise.all(_delayPromises);
+ // If more were added, continue
+ if (_delayPromises.some(p => p.isPending())) {
+ continue;
+ }
+ _delayPromises = [];
+ }
+ break;
+ }
+
// purgeDataObjects() starts a transaction, so if there's an active one then show a
// nice message and wait until there's not. Another transaction could still start
// before purgeDataObjects() and result in a wait timeout, but this should reduce the
@@ -958,6 +974,21 @@ Zotero.Sync.Runner_Module = function (options = {}) {
};
+ /**
+ * Delay syncs until the returned function is called
+ *
+ * @return {Function} - Resolve function
+ */
+ this.delayIndefinite = function () {
+ var resolve;
+ var promise = new Zotero.Promise(function () {
+ resolve = arguments[0];
+ });
+ _delayPromises.push(promise);
+ return resolve;
+ };
+
+
/**
* Trigger updating of the main sync icon, the sync error icon, and
* library-specific sync error icons across all windows
diff --git a/chrome/content/zotero/xpcom/zotero.js b/chrome/content/zotero/xpcom/zotero.js
index e15ea10ca..a0947cf36 100644
--- a/chrome/content/zotero/xpcom/zotero.js
+++ b/chrome/content/zotero/xpcom/zotero.js
@@ -877,6 +877,9 @@ Services.scriptloader.loadSubScript("resource://zotero/polyfill.js");
* Initializes the DB connection
*/
var _initDB = Zotero.Promise.coroutine(function* (haveReleasedLock) {
+ // Initialize main database connection
+ Zotero.DB = new Zotero.DBConnection('zotero');
+
try {
// Test read access
yield Zotero.DB.test();
diff --git a/chrome/content/zotero/zoteroPane.xul b/chrome/content/zotero/zoteroPane.xul
index 1d5817c32..af44a9dae 100644
--- a/chrome/content/zotero/zoteroPane.xul
+++ b/chrome/content/zotero/zoteroPane.xul
@@ -49,7 +49,7 @@
-
+
diff --git a/chrome/locale/en-US/zotero/zotero.dtd b/chrome/locale/en-US/zotero/zotero.dtd
index dd99f00cf..e16c2cb74 100644
--- a/chrome/locale/en-US/zotero/zotero.dtd
+++ b/chrome/locale/en-US/zotero/zotero.dtd
@@ -202,6 +202,14 @@
+
+
+
+
+
+
+
+
diff --git a/chrome/locale/en-US/zotero/zotero.properties b/chrome/locale/en-US/zotero/zotero.properties
index bfbe3ec44..57656b555 100644
--- a/chrome/locale/en-US/zotero/zotero.properties
+++ b/chrome/locale/en-US/zotero/zotero.properties
@@ -69,6 +69,7 @@ general.processing = Processing
general.submitted = Submitted
general.thanksForHelpingImprove = Thanks for helping to improve %S!
general.describeProblem = Briefly describe the problem:
+general.nMegabytes = %S MB
general.operationInProgress = A Zotero operation is currently in progress.
general.operationInProgress.waitUntilFinished = Please wait until it has finished.
@@ -688,10 +689,12 @@ fileInterface.importComplete = Import Complete
fileInterface.itemsWereImported = %1$S item was imported;%1$S items were imported
fileInterface.itemsExported = Exporting items…
fileInterface.import = Import
+fileInterface.chooseAppDatabaseToImport = Choose the %S database to import
fileInterface.export = Export
fileInterface.exportedItems = Exported Items
fileInterface.imported = Imported
fileInterface.unsupportedFormat = The selected file is not in a supported format.
+fileInterface.appDatabase = %S Database
fileInterface.viewSupportedFormats = View Supported Formats…
fileInterface.untitledBibliography = Untitled Bibliography
fileInterface.bibliographyHTMLTitle = Bibliography
@@ -1075,6 +1078,7 @@ rtfScan.rtf = Rich Text Format (.rtf)
rtfScan.saveTitle = Select a location in which to save the formatted file
rtfScan.scannedFileSuffix = (Scanned)
+extractedAnnotations = Extracted Annotations
file.accessError.theFileCannotBeCreated = The file '%S' cannot be created.
file.accessError.theFileCannotBeUpdated = The file '%S' cannot be updated.
diff --git a/chrome/skin/default/zotero/importWizard.css b/chrome/skin/default/zotero/importWizard.css
new file mode 100644
index 000000000..308fe0d42
--- /dev/null
+++ b/chrome/skin/default/zotero/importWizard.css
@@ -0,0 +1,44 @@
+.wizard-header-label {
+ font-size: 16px;
+ font-weight: bold;
+}
+
+/* Start */
+wizard[currentpageid="page-start"] .wizard-header-label {
+ padding-top: 24px;
+}
+
+wizard[currentpageid="page-start"] .wizard-page-box {
+ margin-top: -2px;
+ padding-top: 0;
+}
+
+radiogroup {
+ font-size: 14px;
+ margin-top: 4px;
+}
+
+radio {
+ padding-top: 5px;
+}
+
+/* File options */
+wizard[currentpageid="page-file-options"] .wizard-header {
+ display: none;
+}
+
+#file-options-header {
+ font-size: 15px;
+ font-weight: bold;
+ margin-bottom: 6px;
+}
+
+listbox, #result-description {
+ font-size: 13px;
+}
+
+#result-report-error {
+ margin-top: 13px;
+ margin-left: 0;
+ font-size: 13px;
+}
diff --git a/resource/tinymce/css/note-content.css b/resource/tinymce/css/note-content.css
index d95964524..66a4c7475 100644
--- a/resource/tinymce/css/note-content.css
+++ b/resource/tinymce/css/note-content.css
@@ -1,3 +1,36 @@
+h1 {
+ font-size: 1.6em;
+ padding-bottom: .2em;
+}
+
+h2 {
+ font-size: 1.4em;
+ font-weight: normal;
+ margin-top: 0;
+ padding-bottom: .2em;
+ border-bottom: 1px solid lightgray;
+}
+
+h3 {
+ font-size: 1.1em;
+}
+
+h4 {
+ font-size: 1em;
+ margin-bottom: 1em;
+}
+
+h5 {
+ font-size: .9em;
+ margin-bottom: 1em;
+}
+
+h6 {
+ font-size: .8em;
+ margin-top: 1.6em;
+ margin-bottom: 1.3em;
+}
+
blockquote {
margin-top: 1.5em;
margin-bottom: 1.5em;