db0fa3c33e
Promise-based rewrite of most of the codebase, with asynchronous database and file access -- see https://github.com/zotero/zotero/issues/518 for details. WARNING: This includes backwards-incompatible schema changes. An incomplete list of other changes: - Schema overhaul - Replace main tables with new versions with updated schema - Enable real foreign key support and remove previous triggers - Don't use NULLs for local libraryID, which broke the UNIQUE index preventing object key duplication. All code (Zotero and third-party) using NULL for the local library will need to be updated to use 0 instead (already done for Zotero code) - Add 'compatibility' DB version that can be incremented manually to break DB compatibility with previous versions. 'userdata' upgrades will no longer automatically break compatibility. - Demote creators and tags from first-class objects to item properties - New API syncing properties - 'synced'/'version' properties to data objects - 'etag' to groups - 'version' to libraries - Create Zotero.DataObject that other objects inherit from - Consolidate data object loading into Zotero.DataObjects - Change object reloading so that only the loaded and changed parts of objects are reloaded, instead of reloading all data from the database (with some exceptions, including item primary data) - Items and collections now have .parentItem and .parentKey properties, replacing item.getSource() and item.getSourceKey() - New function Zotero.serial(fn), to wrap an async function such that all calls are run serially - New function Zotero.Utilities.Internal.forEachChunkAsync(arr, chunkSize, func) - Add tag selector loading message - Various API and name changes, since everything was breaking anyway Known broken things: - Syncing (will be completely rewritten for API syncing) - Translation architecture (needs promise-based rewrite) - Duplicates view - DB integrity check (from schema changes) - Dragging (may be difficult to fix) Lots of other big and little things are certainly broken, particularly with the UI, which can be affected by async code in all sorts of subtle ways.
472 lines
No EOL
15 KiB
JavaScript
472 lines
No EOL
15 KiB
JavaScript
/*
|
|
***** BEGIN LICENSE BLOCK *****
|
|
|
|
Copyright © 2009 Center for History and New Media
|
|
George Mason University, Fairfax, Virginia, USA
|
|
http://zotero.org
|
|
|
|
This file is part of Zotero.
|
|
|
|
Zotero is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU Affero General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
Zotero is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU Affero General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Affero General Public License
|
|
along with Zotero. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
Based on nsChromeExtensionHandler example code by Ed Anuff at
|
|
http://kb.mozillazine.org/Dev_:_Extending_the_Chrome_Protocol
|
|
|
|
***** END LICENSE BLOCK *****
|
|
*/
|
|
|
|
const Cc = Components.classes;
|
|
const Ci = Components.interfaces;
|
|
|
|
/** XPCOM files to be loaded for all modes **/
|
|
const xpcomFilesAll = [
|
|
'zotero',
|
|
'date',
|
|
'debug',
|
|
'error',
|
|
'file',
|
|
'http',
|
|
'mimeTypeHandler',
|
|
'openurl',
|
|
'ipc',
|
|
'progressWindow',
|
|
'translation/translate',
|
|
'translation/translate_firefox',
|
|
'translation/translator',
|
|
'translation/tlds',
|
|
'utilities',
|
|
'utilities_internal',
|
|
'utilities_translate'
|
|
];
|
|
|
|
/** XPCOM files to be loaded only for local translation and DB access **/
|
|
const xpcomFilesLocal = [
|
|
'libraryTreeView',
|
|
'collectionTreeView',
|
|
'annotate',
|
|
'attachments',
|
|
'cite',
|
|
'cookieSandbox',
|
|
'data_access',
|
|
'data/dataObject',
|
|
'data/dataObjects',
|
|
'data/dataObjectLoader',
|
|
'data/dataObjectUtilities',
|
|
'data/cachedTypes',
|
|
'data/notes',
|
|
'data/item',
|
|
'data/items',
|
|
'data/collection',
|
|
'data/collections',
|
|
'data/creators',
|
|
'data/group',
|
|
'data/groups',
|
|
'data/itemFields',
|
|
'data/libraries',
|
|
'data/relation',
|
|
'data/relations',
|
|
'data/tags',
|
|
'db',
|
|
'duplicates',
|
|
'fulltext',
|
|
'id',
|
|
'integration',
|
|
'itemTreeView',
|
|
'locateManager',
|
|
'mime',
|
|
'notifier',
|
|
'proxy',
|
|
'quickCopy',
|
|
'report',
|
|
'schema',
|
|
'search',
|
|
'server',
|
|
'style',
|
|
'sync',
|
|
'sync/syncEngine',
|
|
'sync/syncAPI',
|
|
'sync/syncLocal',
|
|
'storage',
|
|
'storage/streamListener',
|
|
'storage/queueManager',
|
|
'storage/queue',
|
|
'storage/request',
|
|
'storage/mode',
|
|
'storage/zfs',
|
|
'storage/webdav',
|
|
'syncedSettings',
|
|
'timeline',
|
|
'uri',
|
|
'translation/translate_item',
|
|
'translation/translators',
|
|
'server_connector'
|
|
];
|
|
|
|
/** XPCOM files to be loaded only for connector translation and DB access **/
|
|
const xpcomFilesConnector = [
|
|
'connector/translate_item',
|
|
'connector/translator',
|
|
'connector/connector',
|
|
'connector/connector_firefox',
|
|
'connector/cachedTypes',
|
|
'connector/repo',
|
|
'connector/typeSchemaData'
|
|
];
|
|
|
|
Components.utils.import("resource://gre/modules/XPCOMUtils.jsm");
|
|
Components.utils.import("resource://gre/modules/Services.jsm");
|
|
|
|
var instanceID = (new Date()).getTime();
|
|
var isFirstLoadThisSession = true;
|
|
var zContext = null;
|
|
var zInitOptions = {};
|
|
|
|
ZoteroContext = function() {}
|
|
ZoteroContext.prototype = {
|
|
/**
|
|
* Convenience method to replicate window.alert()
|
|
**/
|
|
// TODO: is this still used? if so, move to zotero.js
|
|
"alert":function alert(msg){
|
|
this.Zotero.debug("alert() is deprecated from Zotero XPCOM");
|
|
Cc["@mozilla.org/embedcomp/prompt-service;1"]
|
|
.getService(Ci.nsIPromptService)
|
|
.alert(null, "", msg);
|
|
},
|
|
|
|
/**
|
|
* Convenience method to replicate window.confirm()
|
|
**/
|
|
// TODO: is this still used? if so, move to zotero.js
|
|
"confirm":function confirm(msg){
|
|
this.Zotero.debug("confirm() is deprecated from Zotero XPCOM");
|
|
return Cc["@mozilla.org/embedcomp/prompt-service;1"]
|
|
.getService(Ci.nsIPromptService)
|
|
.confirm(null, "", msg);
|
|
},
|
|
|
|
"Cc":Cc,
|
|
"Ci":Ci,
|
|
|
|
/**
|
|
* Convenience method to replicate window.setTimeout()
|
|
**/
|
|
"setTimeout":function setTimeout(func, ms){
|
|
this.Zotero.setTimeout(func, ms);
|
|
},
|
|
|
|
/**
|
|
* Switches in or out of connector mode
|
|
*/
|
|
"switchConnectorMode":function(isConnector) {
|
|
if(isConnector !== this.isConnector) {
|
|
Services.obs.notifyObservers(zContext.Zotero, "zotero-before-reload", isConnector ? "connector" : "full");
|
|
zContext.Zotero.shutdown().then(function() {
|
|
// create a new zContext
|
|
makeZoteroContext(isConnector);
|
|
return zContext.Zotero.init(zInitOptions);
|
|
}).done();
|
|
}
|
|
|
|
return zContext;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* The class from which the Zotero global XPCOM context is constructed
|
|
*
|
|
* @constructor
|
|
* This runs when ZoteroService is first requested to load all applicable scripts and initialize
|
|
* Zotero. Calls to other XPCOM components must be in here rather than in top-level code, as other
|
|
* components may not have yet been initialized.
|
|
*/
|
|
function makeZoteroContext(isConnector) {
|
|
if(zContext) {
|
|
// Swap out old zContext
|
|
var oldzContext = zContext;
|
|
// Create new zContext
|
|
zContext = new ZoteroContext();
|
|
// Swap in old Zotero object, so that references don't break, but empty it
|
|
zContext.Zotero = oldzContext.Zotero;
|
|
for(var key in zContext.Zotero) delete zContext.Zotero[key];
|
|
} else {
|
|
zContext = new ZoteroContext();
|
|
zContext.Zotero = function() {};
|
|
}
|
|
|
|
var subscriptLoader = Cc["@mozilla.org/moz/jssubscript-loader;1"].getService(Ci.mozIJSSubScriptLoader);
|
|
|
|
// Load zotero.js first
|
|
subscriptLoader.loadSubScript("chrome://zotero/content/xpcom/" + xpcomFilesAll[0] + ".js", zContext);
|
|
|
|
// Load CiteProc into Zotero.CiteProc namespace
|
|
zContext.Zotero.CiteProc = {"Zotero":zContext.Zotero};
|
|
subscriptLoader.loadSubScript("chrome://zotero/content/xpcom/citeproc.js", zContext.Zotero.CiteProc);
|
|
|
|
// Load XRegExp object into Zotero.XRegExp
|
|
const xregexpFiles = [
|
|
/**Core functions**/
|
|
'xregexp',
|
|
|
|
/**Addons**/
|
|
'addons/build', //adds ability to "build regular expressions using named subpatterns, for readability and pattern reuse"
|
|
'addons/matchrecursive', //adds ability to "match recursive constructs using XRegExp pattern strings as left and right delimiters"
|
|
|
|
/**Unicode support**/
|
|
'addons/unicode/unicode-base', //required for all other unicode packages. Adds \p{Letter} category
|
|
|
|
//'addons/unicode/unicode-blocks', //adds support for all Unicode blocks (e.g. InArabic, InCyrillic_Extended_A, etc.)
|
|
'addons/unicode/unicode-categories', //adds support for all Unicode categories (e.g. Punctuation, Lowercase_Letter, etc.)
|
|
//'addons/unicode/unicode-properties', //adds Level 1 Unicode properties (e.g. Uppercase, White_Space, etc.)
|
|
//'addons/unicode/unicode-scripts' //adds support for all Unicode scripts (e.g. Gujarati, Cyrillic, etc.)
|
|
'addons/unicode/unicode-zotero' //adds support for some Unicode categories used in Zotero
|
|
];
|
|
for (var i=0; i<xregexpFiles.length; i++) {
|
|
subscriptLoader.loadSubScript("chrome://zotero/content/xpcom/xregexp/" + xregexpFiles[i] + ".js", zContext);
|
|
}
|
|
|
|
// Load remaining xpcomFiles
|
|
for (var i=1; i<xpcomFilesAll.length; i++) {
|
|
try {
|
|
subscriptLoader.loadSubScript("chrome://zotero/content/xpcom/" + xpcomFilesAll[i] + ".js", zContext);
|
|
}
|
|
catch (e) {
|
|
Components.utils.reportError("Error loading " + xpcomFilesAll[i] + ".js", zContext);
|
|
throw (e);
|
|
}
|
|
}
|
|
|
|
// Load xpcomFiles for specific mode
|
|
for each(var xpcomFile in (isConnector ? xpcomFilesConnector : xpcomFilesLocal)) {
|
|
try {
|
|
subscriptLoader.loadSubScript("chrome://zotero/content/xpcom/" + xpcomFile + ".js", zContext);
|
|
}
|
|
catch (e) {
|
|
Components.utils.reportError("Error loading " + xpcomFile + ".js", zContext);
|
|
throw (e);
|
|
}
|
|
}
|
|
|
|
// Load RDF files into Zotero.RDF.AJAW namespace (easier than modifying all of the references)
|
|
const rdfXpcomFiles = [
|
|
'rdf/init',
|
|
'rdf/uri',
|
|
'rdf/term',
|
|
'rdf/identity',
|
|
'rdf/match',
|
|
'rdf/n3parser',
|
|
'rdf/rdfparser',
|
|
'rdf/serialize'
|
|
];
|
|
zContext.Zotero.RDF = {Zotero:zContext.Zotero};
|
|
for (var i=0; i<rdfXpcomFiles.length; i++) {
|
|
subscriptLoader.loadSubScript("chrome://zotero/content/xpcom/" + rdfXpcomFiles[i] + ".js", zContext.Zotero.RDF);
|
|
}
|
|
|
|
if(isStandalone()) {
|
|
// If isStandalone, load standalone.js
|
|
subscriptLoader.loadSubScript("chrome://zotero/content/xpcom/standalone.js", zContext);
|
|
}
|
|
|
|
// load nsTransferable (query: do we still use this?)
|
|
subscriptLoader.loadSubScript("chrome://global/content/nsTransferable.js", zContext);
|
|
|
|
// add connector-related properties
|
|
zContext.Zotero.isConnector = isConnector;
|
|
zContext.Zotero.instanceID = instanceID;
|
|
zContext.Zotero.__defineGetter__("isFirstLoadThisSession", function() isFirstLoadThisSession);
|
|
};
|
|
|
|
/**
|
|
* The class representing the Zotero service, and affiliated XPCOM goop
|
|
*/
|
|
function ZoteroService() {
|
|
try {
|
|
var start = Date.now();
|
|
|
|
if(isFirstLoadThisSession) {
|
|
makeZoteroContext(false);
|
|
zContext.Zotero.init(zInitOptions)
|
|
.catch(function (e) {
|
|
dump(e + "\n\n");
|
|
Components.utils.reportError(e);
|
|
|
|
if (e === "ZOTERO_SHOULD_START_AS_CONNECTOR") {
|
|
// if Zotero should start as a connector, reload it
|
|
return zContext.Zotero.shutdown()
|
|
.then(function() {
|
|
makeZoteroContext(true);
|
|
return zContext.Zotero.init(zInitOptions);
|
|
})
|
|
}
|
|
else {
|
|
throw e;
|
|
}
|
|
})
|
|
.then(function () {
|
|
zContext.Zotero.debug("Initialized in "+(Date.now() - start)+" ms");
|
|
})
|
|
.done();
|
|
}
|
|
else {
|
|
zContext.Zotero.debug("Already initialized");
|
|
}
|
|
isFirstLoadThisSession = false;
|
|
this.wrappedJSObject = zContext.Zotero;
|
|
} catch(e) {
|
|
var msg = typeof e == 'string' ? e : e.name;
|
|
dump(e + "\n\n");
|
|
Components.utils.reportError(e);
|
|
throw e;
|
|
}
|
|
}
|
|
|
|
ZoteroService.prototype = {
|
|
contractID: '@zotero.org/Zotero;1',
|
|
classDescription: 'Zotero',
|
|
classID: Components.ID('{e4c61080-ec2d-11da-8ad9-0800200c9a66}'),
|
|
QueryInterface: XPCOMUtils.generateQI([Components.interfaces.nsISupports,
|
|
Components.interfaces.nsIProtocolHandler])
|
|
}
|
|
|
|
var _isStandalone = null;
|
|
/**
|
|
* Determine whether Zotero Standalone is running
|
|
*/
|
|
function isStandalone() {
|
|
if(_isStandalone === null) {
|
|
var appInfo = Components.classes["@mozilla.org/xre/app-info;1"].
|
|
getService(Components.interfaces.nsIXULAppInfo);
|
|
_isStandalone = appInfo.ID === 'zotero@chnm.gmu.edu';
|
|
}
|
|
return _isStandalone;
|
|
}
|
|
|
|
/**
|
|
* The class representing the Zotero command line handler
|
|
*/
|
|
function ZoteroCommandLineHandler() {}
|
|
ZoteroCommandLineHandler.prototype = {
|
|
/* nsICommandLineHandler */
|
|
handle : function(cmdLine) {
|
|
// Force debug output
|
|
if (cmdLine.handleFlag("zoterodebug", false)) {
|
|
zInitOptions.forceDebugLog = true;
|
|
}
|
|
|
|
// handler to open Zotero pane at startup in Zotero for Firefox
|
|
if (!isStandalone() && cmdLine.handleFlag("ZoteroPaneOpen", false)) {
|
|
zInitOptions.openPane = true;
|
|
}
|
|
|
|
// handler for Zotero integration commands
|
|
// this is typically used on Windows only, via WM_COPYDATA rather than the command line
|
|
var agent = cmdLine.handleFlagWithParam("ZoteroIntegrationAgent", false);
|
|
if(agent) {
|
|
// Don't open a new window
|
|
cmdLine.preventDefault = true;
|
|
|
|
var command = cmdLine.handleFlagWithParam("ZoteroIntegrationCommand", false);
|
|
var docId = cmdLine.handleFlagWithParam("ZoteroIntegrationDocument", false);
|
|
|
|
// Not quite sure why this is necessary to get the appropriate scoping
|
|
var Zotero = this.Zotero;
|
|
Zotero.setTimeout(function() { Zotero.Integration.execCommand(agent, command, docId) }, 0);
|
|
}
|
|
|
|
// handler for Windows IPC commands
|
|
var ipcParam = cmdLine.handleFlagWithParam("ZoteroIPC", false);
|
|
if(ipcParam) {
|
|
// Don't open a new window
|
|
cmdLine.preventDefault = true;
|
|
var Zotero = this.Zotero;
|
|
Zotero.setTimeout(function() { Zotero.IPC.parsePipeInput(ipcParam) }, 0);
|
|
}
|
|
|
|
// special handler for "zotero" URIs at the command line to prevent them from opening a new
|
|
// window
|
|
if(isStandalone()) {
|
|
var param = cmdLine.handleFlagWithParam("url", false);
|
|
if(param) {
|
|
var uri = cmdLine.resolveURI(param);
|
|
if(uri.schemeIs("zotero")) {
|
|
// Check for existing window and focus it
|
|
var wm = Components.classes["@mozilla.org/appshell/window-mediator;1"]
|
|
.getService(Components.interfaces.nsIWindowMediator);
|
|
var win = wm.getMostRecentWindow("navigator:browser");
|
|
if(win) {
|
|
win.focus();
|
|
Components.classes["@mozilla.org/network/protocol;1?name=zotero"]
|
|
.createInstance(Components.interfaces.nsIProtocolHandler).newChannel(uri);
|
|
}
|
|
} else {
|
|
this.Zotero.debug("Not handling URL: "+uri.spec);
|
|
}
|
|
}
|
|
|
|
var param = cmdLine.handleFlagWithParam("file", false);
|
|
if(param) {
|
|
var file = Components.classes["@mozilla.org/file/local;1"].
|
|
createInstance(Components.interfaces.nsILocalFile);
|
|
file.initWithPath(param);
|
|
|
|
if(file.leafName.substr(-4).toLowerCase() === ".csl"
|
|
|| file.leafName.substr(-8).toLowerCase() === ".csl.txt") {
|
|
// Install CSL file
|
|
this.Zotero.Styles.install(file);
|
|
} else {
|
|
// Ask before importing
|
|
var checkState = {"value":this.Zotero.Prefs.get('import.createNewCollection.fromFileOpenHandler')};
|
|
if(Components.classes["@mozilla.org/embedcomp/prompt-service;1"]
|
|
.getService(Components.interfaces.nsIPromptService)
|
|
.confirmCheck(null, this.Zotero.getString('ingester.importFile.title'),
|
|
this.Zotero.getString('ingester.importFile.text', [file.leafName]),
|
|
this.Zotero.getString('ingester.importFile.intoNewCollection'),
|
|
checkState)) {
|
|
// Perform file import in front window
|
|
var wm = Components.classes["@mozilla.org/appshell/window-mediator;1"]
|
|
.getService(Components.interfaces.nsIWindowMediator);
|
|
var browserWindow = wm.getMostRecentWindow("navigator:browser");
|
|
browserWindow.Zotero_File_Interface.importFile(file, checkState.value);
|
|
this.Zotero.Prefs.set('import.createNewCollection.fromFileOpenHandler', checkState.value);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
contractID: "@mozilla.org/commandlinehandler/general-startup;1?type=zotero",
|
|
classDescription: "Zotero Command Line Handler",
|
|
classID: Components.ID("{531828f8-a16c-46be-b9aa-14845c3b010f}"),
|
|
service: true,
|
|
_xpcom_categories: [{category:"command-line-handler", entry:"m-zotero"}],
|
|
QueryInterface: XPCOMUtils.generateQI([Components.interfaces.nsICommandLineHandler,
|
|
Components.interfaces.nsISupports])
|
|
};
|
|
|
|
ZoteroCommandLineHandler.prototype.__defineGetter__("Zotero", function() {
|
|
if(!zContext) new ZoteroService();
|
|
return zContext.Zotero;
|
|
});
|
|
|
|
/**
|
|
* XPCOMUtils.generateNSGetFactory was introduced in Mozilla 2 (Firefox 4).
|
|
* XPCOMUtils.generateNSGetModule is for Mozilla 1.9.2 (Firefox 3.6).
|
|
*/
|
|
if (XPCOMUtils.generateNSGetFactory) {
|
|
var NSGetFactory = XPCOMUtils.generateNSGetFactory([ZoteroService, ZoteroCommandLineHandler]);
|
|
} else {
|
|
var NSGetModule = XPCOMUtils.generateNSGetModule([ZoteroService, ZoteroCommandLineHandler]);
|
|
} |