317b1dfa0f
Don't start the connector pipe (Mac/Linux), don't listen for IPC on the command-line (Windows), and don't try to release Standalone's lock if DB is busy when Firefox starts.
518 lines
16 KiB
JavaScript
518 lines
16 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',
|
|
'isbn',
|
|
'utilities_internal',
|
|
'utilities_translate'
|
|
];
|
|
|
|
/** XPCOM files to be loaded only for local translation and DB access **/
|
|
const xpcomFilesLocal = [
|
|
'libraryTreeView',
|
|
'collectionTreeView',
|
|
'collectionTreeRow',
|
|
'annotate',
|
|
'api',
|
|
'attachments',
|
|
'cite',
|
|
'cookieSandbox',
|
|
'data/library',
|
|
'data/libraries',
|
|
'data/dataObject',
|
|
'data/dataObjects',
|
|
'data/dataObjectUtilities',
|
|
'data/cachedTypes',
|
|
'data/notes',
|
|
'data/item',
|
|
'data/items',
|
|
'data/collection',
|
|
'data/collections',
|
|
'data/feedItem',
|
|
'data/feedItems',
|
|
'data/feed',
|
|
'data/feeds',
|
|
'data/creators',
|
|
'data/group',
|
|
'data/groups',
|
|
'data/itemFields',
|
|
'data/relations',
|
|
'data/tags',
|
|
'db',
|
|
'duplicates',
|
|
'feedReader',
|
|
'fulltext',
|
|
'id',
|
|
'integration',
|
|
'itemTreeView',
|
|
'locateManager',
|
|
'mime',
|
|
'notifier',
|
|
'proxy',
|
|
'quickCopy',
|
|
'report',
|
|
'router',
|
|
'schema',
|
|
'search',
|
|
'server',
|
|
'style',
|
|
'sync',
|
|
'sync/syncAPIClient',
|
|
'sync/syncEngine',
|
|
'sync/syncExceptions',
|
|
'sync/syncEventListeners',
|
|
'sync/syncFullTextEngine',
|
|
'sync/syncLocal',
|
|
'sync/syncRunner',
|
|
'sync/syncUtilities',
|
|
'storage',
|
|
'storage/storageEngine',
|
|
'storage/storageLocal',
|
|
'storage/storageRequest',
|
|
'storage/storageResult',
|
|
'storage/storageUtilities',
|
|
'storage/streamListener',
|
|
'storage/zfs',
|
|
'storage/webdav',
|
|
'syncedSettings',
|
|
'timeline',
|
|
'uri',
|
|
'users',
|
|
'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;
|
|
},
|
|
|
|
/**
|
|
* Shuts down Zotero, calls a callback (that may return a promise),
|
|
* then reinitializes Zotero. Returns a promise that is resolved
|
|
* when this process completes.
|
|
*/
|
|
"reinit":function(cb, isConnector, options = {}) {
|
|
Services.obs.notifyObservers(zContext.Zotero, "zotero-before-reload", isConnector ? "connector" : "full");
|
|
return zContext.Zotero.shutdown().then(function() {
|
|
return cb ? cb() : false;
|
|
}).finally(function() {
|
|
makeZoteroContext(isConnector);
|
|
var o = {};
|
|
Object.assign(o, zInitOptions);
|
|
Object.assign(o, options);
|
|
zContext.Zotero.init(o);
|
|
});
|
|
}
|
|
};
|
|
|
|
/**
|
|
* 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-prereqs.js", zContext.Zotero.CiteProc);
|
|
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 (let xpcomFile of (isConnector ? xpcomFilesConnector : xpcomFilesLocal)) {
|
|
try {
|
|
subscriptLoader.loadSubScript("chrome://zotero/content/xpcom/" + xpcomFile + ".js", zContext);
|
|
}
|
|
catch (e) {
|
|
dump("Error loading " + xpcomFile + ".js\n\n");
|
|
dump(e + "\n\n");
|
|
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");
|
|
isFirstLoadThisSession = false;
|
|
});
|
|
}
|
|
else {
|
|
zContext.Zotero.debug("Already initialized");
|
|
}
|
|
this.wrappedJSObject = zContext.Zotero;
|
|
} catch(e) {
|
|
var msg = e instanceof Error
|
|
? e.name + ': ' + e.message + '\n' + e.fileName + ':' + e.lineNumber + '\n' + e.stack
|
|
: '' + e;
|
|
dump(msg + '\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
|
|
// TEMP: Disabled for 5.0 Beta
|
|
/*
|
|
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);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (cmdLine.handleFlag("ZoteroNoUserInput", false)) {
|
|
zInitOptions.noUserInput = true;
|
|
}
|
|
if (cmdLine.handleFlag("ZoteroSkipBundledFiles", false)) {
|
|
zInitOptions.skipBundledFiles = true;
|
|
}
|
|
},
|
|
|
|
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]);
|
|
}
|