/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ /* This code is loaded in every child process that is started by mochitest in * order to be used as a replacement for UniversalXPConnect */ /* import-globals-from specialpowersAPI.js */ /* globals addMessageListener, removeMessageListener, sendSyncMessage, sendAsyncMessage */ Components.utils.import("resource://gre/modules/Services.jsm"); function SpecialPowers(window) { this.window = Components.utils.getWeakReference(window); this._windowID = window.QueryInterface(Components.interfaces.nsIInterfaceRequestor) .getInterface(Components.interfaces.nsIDOMWindowUtils) .currentInnerWindowID; this._encounteredCrashDumpFiles = []; this._unexpectedCrashDumpFiles = { }; this._crashDumpDir = null; this.DOMWindowUtils = bindDOMWindowUtils(window); Object.defineProperty(this, "Components", { configurable: true, enumerable: true, get() { var win = this.window.get(); if (!win) return null; return getRawComponents(win); }}); this._pongHandlers = []; this._messageListener = this._messageReceived.bind(this); this._grandChildFrameMM = null; this._createFilesOnError = null; this._createFilesOnSuccess = null; this.SP_SYNC_MESSAGES = ["SPChromeScriptMessage", "SPLoadChromeScript", "SPImportInMainProcess", "SPObserverService", "SPPermissionManager", "SPPrefService", "SPProcessCrashService", "SPSetTestPluginEnabledState", "SPCleanUpSTSData"]; this.SP_ASYNC_MESSAGES = ["SpecialPowers.Focus", "SpecialPowers.Quit", "SpecialPowers.CreateFiles", "SpecialPowers.RemoveFiles", "SPPingService", "SPLoadExtension", "SPStartupExtension", "SPUnloadExtension", "SPExtensionMessage"]; addMessageListener("SPPingService", this._messageListener); addMessageListener("SpecialPowers.FilesCreated", this._messageListener); addMessageListener("SpecialPowers.FilesError", this._messageListener); let self = this; Services.obs.addObserver(function onInnerWindowDestroyed(subject, topic, data) { var id = subject.QueryInterface(Components.interfaces.nsISupportsPRUint64).data; if (self._windowID === id) { Services.obs.removeObserver(onInnerWindowDestroyed, "inner-window-destroyed"); try { removeMessageListener("SPPingService", self._messageListener); removeMessageListener("SpecialPowers.FilesCreated", self._messageListener); removeMessageListener("SpecialPowers.FilesError", self._messageListener); } catch (e) { // Ignore the exception which the message manager has been destroyed. if (e.result != Components.results.NS_ERROR_ILLEGAL_VALUE) { throw e; } } } }, "inner-window-destroyed"); } SpecialPowers.prototype = new SpecialPowersAPI(); SpecialPowers.prototype.toString = function() { return "[SpecialPowers]"; }; SpecialPowers.prototype.sanityCheck = function() { return "foo"; }; // This gets filled in in the constructor. SpecialPowers.prototype.DOMWindowUtils = undefined; SpecialPowers.prototype.Components = undefined; SpecialPowers.prototype.IsInNestedFrame = false; SpecialPowers.prototype._sendSyncMessage = function(msgname, msg) { if (this.SP_SYNC_MESSAGES.indexOf(msgname) == -1) { dump("TEST-INFO | specialpowers.js | Unexpected SP message: " + msgname + "\n"); } return sendSyncMessage(msgname, msg); }; SpecialPowers.prototype._sendAsyncMessage = function(msgname, msg) { if (this.SP_ASYNC_MESSAGES.indexOf(msgname) == -1) { dump("TEST-INFO | specialpowers.js | Unexpected SP message: " + msgname + "\n"); } sendAsyncMessage(msgname, msg); }; SpecialPowers.prototype._addMessageListener = function(msgname, listener) { addMessageListener(msgname, listener); sendAsyncMessage("SPPAddNestedMessageListener", { name: msgname }); }; SpecialPowers.prototype._removeMessageListener = function(msgname, listener) { removeMessageListener(msgname, listener); }; SpecialPowers.prototype.registerProcessCrashObservers = function() { addMessageListener("SPProcessCrashService", this._messageListener); sendSyncMessage("SPProcessCrashService", { op: "register-observer" }); }; SpecialPowers.prototype.unregisterProcessCrashObservers = function() { removeMessageListener("SPProcessCrashService", this._messageListener); sendSyncMessage("SPProcessCrashService", { op: "unregister-observer" }); }; SpecialPowers.prototype._messageReceived = function(aMessage) { switch (aMessage.name) { case "SPProcessCrashService": if (aMessage.json.type == "crash-observed") { for (let e of aMessage.json.dumpIDs) { this._encounteredCrashDumpFiles.push(e.id + "." + e.extension); } } break; case "SPPingService": if (aMessage.json.op == "pong") { var handler = this._pongHandlers.shift(); if (handler) { handler(); } if (this._grandChildFrameMM) { this._grandChildFrameMM.sendAsyncMessage("SPPingService", { op: "pong" }); } } break; case "SpecialPowers.FilesCreated": var createdHandler = this._createFilesOnSuccess; this._createFilesOnSuccess = null; this._createFilesOnError = null; if (createdHandler) { createdHandler(aMessage.data); } break; case "SpecialPowers.FilesError": var errorHandler = this._createFilesOnError; this._createFilesOnSuccess = null; this._createFilesOnError = null; if (errorHandler) { errorHandler(aMessage.data); } break; } return true; }; SpecialPowers.prototype.quit = function() { sendAsyncMessage("SpecialPowers.Quit", {}); }; // fileRequests is an array of file requests. Each file request is an object. // A request must have a field |name|, which gives the base of the name of the // file to be created in the profile directory. If the request has a |data| field // then that data will be written to the file. SpecialPowers.prototype.createFiles = function(fileRequests, onCreation, onError) { if (this._createFilesOnSuccess || this._createFilesOnError) { onError("Already waiting for SpecialPowers.createFiles() to finish."); return; } this._createFilesOnSuccess = onCreation; this._createFilesOnError = onError; sendAsyncMessage("SpecialPowers.CreateFiles", fileRequests); }; // Remove the files that were created using |SpecialPowers.createFiles()|. // This will be automatically called by |SimpleTest.finish()|. SpecialPowers.prototype.removeFiles = function() { sendAsyncMessage("SpecialPowers.RemoveFiles", {}); }; SpecialPowers.prototype.executeAfterFlushingMessageQueue = function(aCallback) { this._pongHandlers.push(aCallback); sendAsyncMessage("SPPingService", { op: "ping" }); }; SpecialPowers.prototype.nestedFrameSetup = function() { let self = this; Services.obs.addObserver(function onRemoteBrowserShown(subject, topic, data) { let frameLoader = subject; // get a ref to the app