зеркало из https://github.com/mozilla/gecko-dev.git
187 строки
6.3 KiB
JavaScript
187 строки
6.3 KiB
JavaScript
/* 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.CC = Components.Constructor;
|
|
this.Cc = Components.classes;
|
|
this.Ci = Components.interfaces;
|
|
this.Cu = Components.utils;
|
|
this.Cr = Components.results;
|
|
|
|
const MARIONETTE_CONTRACTID = "@mozilla.org/marionette;1";
|
|
const MARIONETTE_CID = Components.ID("{786a1369-dca5-4adc-8486-33d23c88010a}");
|
|
const MARIONETTE_ENABLED_PREF = 'marionette.defaultPrefs.enabled';
|
|
const MARIONETTE_FORCELOCAL_PREF = 'marionette.force-local';
|
|
const MARIONETTE_LOG_PREF = 'marionette.logging';
|
|
|
|
this.ServerSocket = CC("@mozilla.org/network/server-socket;1",
|
|
"nsIServerSocket",
|
|
"initSpecialConnection");
|
|
|
|
Cu.import("resource://gre/modules/XPCOMUtils.jsm");
|
|
Cu.import("resource://gre/modules/Services.jsm");
|
|
Cu.import("resource://gre/modules/FileUtils.jsm");
|
|
Cu.import("resource://gre/modules/Log.jsm");
|
|
|
|
let loader = Cc["@mozilla.org/moz/jssubscript-loader;1"]
|
|
.getService(Ci.mozIJSSubScriptLoader);
|
|
|
|
function MarionetteComponent() {
|
|
this._loaded = false;
|
|
this.observerService = Services.obs;
|
|
|
|
// set up the logger
|
|
this.logger = Log.repository.getLogger("Marionette");
|
|
this.logger.level = Log.Level["Trace"];
|
|
let dumper = false;
|
|
#ifdef DEBUG
|
|
dumper = true;
|
|
#endif
|
|
#ifdef MOZ_B2G
|
|
dumper = true;
|
|
#endif
|
|
try {
|
|
if (dumper || Services.prefs.getBoolPref(MARIONETTE_LOG_PREF)) {
|
|
let formatter = new Log.BasicFormatter();
|
|
this.logger.addAppender(new Log.DumpAppender(formatter));
|
|
}
|
|
}
|
|
catch(e) {}
|
|
}
|
|
|
|
MarionetteComponent.prototype = {
|
|
classDescription: "Marionette component",
|
|
classID: MARIONETTE_CID,
|
|
contractID: MARIONETTE_CONTRACTID,
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsICommandLineHandler, Ci.nsIObserver]),
|
|
_xpcom_categories: [{category: "command-line-handler", entry: "b-marionette"},
|
|
{category: "profile-after-change", service: true}],
|
|
appName: Services.appinfo.name,
|
|
enabled: false,
|
|
finalUiStartup: false,
|
|
_marionetteServer: null,
|
|
|
|
onSocketAccepted: function mc_onSocketAccepted(aSocket, aTransport) {
|
|
this.logger.info("onSocketAccepted for Marionette dummy socket");
|
|
},
|
|
|
|
onStopListening: function mc_onStopListening(aSocket, status) {
|
|
this.logger.info("onStopListening for Marionette dummy socket, code " + status);
|
|
aSocket.close();
|
|
},
|
|
|
|
// Check cmdLine argument for --marionette
|
|
handle: function mc_handle(cmdLine) {
|
|
// If the CLI is there then lets do work otherwise nothing to see
|
|
if (cmdLine.handleFlag("marionette", false)) {
|
|
this.enabled = true;
|
|
this.logger.info("marionette enabled via command-line");
|
|
this.init();
|
|
}
|
|
},
|
|
|
|
observe: function mc_observe(aSubject, aTopic, aData) {
|
|
switch (aTopic) {
|
|
case "profile-after-change":
|
|
// Using final-ui-startup as the xpcom category doesn't seem to work,
|
|
// so we wait for that by adding an observer here.
|
|
this.observerService.addObserver(this, "final-ui-startup", false);
|
|
#ifdef ENABLE_MARIONETTE
|
|
let enabledPref = false;
|
|
try {
|
|
enabledPref = Services.prefs.getBoolPref(MARIONETTE_ENABLED_PREF);
|
|
} catch(e) {}
|
|
if (enabledPref) {
|
|
this.enabled = true;
|
|
this.logger.info("marionette enabled via build flag and pref");
|
|
|
|
// We want to suppress the modal dialog that's shown
|
|
// when starting up in safe-mode to enable testing.
|
|
if (Services.appinfo.inSafeMode) {
|
|
this.observerService.addObserver(this, "domwindowopened", false);
|
|
}
|
|
}
|
|
#endif
|
|
break;
|
|
case "final-ui-startup":
|
|
this.finalUiStartup = true;
|
|
this.observerService.removeObserver(this, aTopic);
|
|
this.observerService.addObserver(this, "xpcom-shutdown", false);
|
|
this.init();
|
|
break;
|
|
case "domwindowopened":
|
|
this.observerService.removeObserver(this, aTopic);
|
|
this._suppressSafeModeDialog(aSubject);
|
|
break;
|
|
case "xpcom-shutdown":
|
|
this.observerService.removeObserver(this, "xpcom-shutdown");
|
|
this.uninit();
|
|
break;
|
|
}
|
|
},
|
|
|
|
_suppressSafeModeDialog: function mc_suppressSafeModeDialog(aWindow) {
|
|
// Wait for the modal dialog to finish loading.
|
|
aWindow.addEventListener("load", function onLoad() {
|
|
aWindow.removeEventListener("load", onLoad);
|
|
|
|
if (aWindow.document.getElementById("safeModeDialog")) {
|
|
aWindow.setTimeout(() => {
|
|
// Accept the dialog to start in safe-mode.
|
|
aWindow.document.documentElement.getButton("accept").click();
|
|
});
|
|
}
|
|
});
|
|
},
|
|
|
|
init: function mc_init() {
|
|
if (!this._loaded && this.enabled && this.finalUiStartup) {
|
|
this._loaded = true;
|
|
|
|
let marionette_forcelocal = this.appName == 'B2G' ? false : true;
|
|
try {
|
|
marionette_forcelocal = Services.prefs.getBoolPref(MARIONETTE_FORCELOCAL_PREF);
|
|
}
|
|
catch(e) {}
|
|
Services.prefs.setBoolPref(MARIONETTE_FORCELOCAL_PREF, marionette_forcelocal);
|
|
|
|
if (!marionette_forcelocal) {
|
|
// See bug 800138. Because the first socket that opens with
|
|
// force-local=false fails, we open a dummy socket that will fail.
|
|
// keepWhenOffline=true so that it still work when offline (local).
|
|
// This allows the following attempt by Marionette to open a socket
|
|
// to succeed.
|
|
let insaneSacrificialGoat = new ServerSocket(666, Ci.nsIServerSocket.KeepWhenOffline, 4);
|
|
insaneSacrificialGoat.asyncListen(this);
|
|
}
|
|
|
|
let port;
|
|
try {
|
|
port = Services.prefs.getIntPref('marionette.defaultPrefs.port');
|
|
}
|
|
catch(e) {
|
|
port = 2828;
|
|
}
|
|
try {
|
|
loader.loadSubScript("chrome://marionette/content/marionette-server.js");
|
|
let forceLocal = Services.prefs.getBoolPref(MARIONETTE_FORCELOCAL_PREF);
|
|
this._marionetteServer = new MarionetteServer(port, forceLocal);
|
|
this.logger.info("Marionette server ready");
|
|
}
|
|
catch(e) {
|
|
this.logger.error('exception: ' + e.name + ', ' + e.message);
|
|
}
|
|
}
|
|
},
|
|
|
|
uninit: function mc_uninit() {
|
|
if (this._marionetteServer) {
|
|
this._marionetteServer.closeListener();
|
|
}
|
|
this._loaded = false;
|
|
},
|
|
|
|
};
|
|
|
|
this.NSGetFactory = XPCOMUtils.generateNSGetFactory([MarionetteComponent]);
|