diff options
Diffstat (limited to 'mobile/android/modules/geckoview/GeckoViewConsole.jsm')
-rw-r--r-- | mobile/android/modules/geckoview/GeckoViewConsole.jsm | 193 |
1 files changed, 193 insertions, 0 deletions
diff --git a/mobile/android/modules/geckoview/GeckoViewConsole.jsm b/mobile/android/modules/geckoview/GeckoViewConsole.jsm new file mode 100644 index 0000000000..faf4d04841 --- /dev/null +++ b/mobile/android/modules/geckoview/GeckoViewConsole.jsm @@ -0,0 +1,193 @@ +/* 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/. */ +"use strict"; + +var EXPORTED_SYMBOLS = ["GeckoViewConsole"]; + +const { XPCOMUtils } = ChromeUtils.import( + "resource://gre/modules/XPCOMUtils.jsm" +); +const { GeckoViewUtils } = ChromeUtils.import( + "resource://gre/modules/GeckoViewUtils.jsm" +); + +XPCOMUtils.defineLazyModuleGetters(this, { + Services: "resource://gre/modules/Services.jsm", +}); + +const { debug, warn } = GeckoViewUtils.initLogging("Console"); + +const LOG_EVENT_TOPIC = "console-api-log-event"; + +var GeckoViewConsole = { + _isEnabled: false, + + get enabled() { + return this._isEnabled; + }, + + set enabled(aVal) { + debug`enabled = ${aVal}`; + if (!!aVal === this._isEnabled) { + return; + } + + this._isEnabled = !!aVal; + if (this._isEnabled) { + Services.obs.addObserver(this, LOG_EVENT_TOPIC); + } else { + Services.obs.removeObserver(this, LOG_EVENT_TOPIC); + } + }, + + observe(aSubject, aTopic, aData) { + switch (aTopic) { + case "nsPref:changed": + this.enabled = Services.prefs.getBoolPref(aData, false); + break; + case LOG_EVENT_TOPIC: + this._handleConsoleMessage(aSubject); + break; + } + }, + + _handleConsoleMessage(aMessage) { + aMessage = aMessage.wrappedJSObject; + + const mappedArguments = Array.from( + aMessage.arguments, + this.formatResult, + this + ); + const joinedArguments = mappedArguments.join(" "); + + if (aMessage.level == "error" || aMessage.level == "warn") { + const flag = + aMessage.level == "error" + ? Ci.nsIScriptError.errorFlag + : Ci.nsIScriptError.warningFlag; + const consoleMsg = Cc["@mozilla.org/scripterror;1"].createInstance( + Ci.nsIScriptError + ); + consoleMsg.init( + joinedArguments, + null, + null, + 0, + 0, + flag, + "content javascript" + ); + Services.console.logMessage(consoleMsg); + } else if (aMessage.level == "trace") { + const bundle = Services.strings.createBundle( + "chrome://browser/locale/browser.properties" + ); + const args = aMessage.arguments; + const filename = this.abbreviateSourceURL(args[0].filename); + const functionName = + args[0].functionName || + bundle.GetStringFromName("stacktrace.anonymousFunction"); + const lineNumber = args[0].lineNumber; + + let body = bundle.formatStringFromName("stacktrace.outputMessage", [ + filename, + functionName, + lineNumber, + ]); + body += "\n"; + args.forEach(function(aFrame) { + const functionName = + aFrame.functionName || + bundle.GetStringFromName("stacktrace.anonymousFunction"); + body += + " " + + aFrame.filename + + " :: " + + functionName + + " :: " + + aFrame.lineNumber + + "\n"; + }); + + Services.console.logStringMessage(body); + } else if (aMessage.level == "time" && aMessage.arguments) { + const bundle = Services.strings.createBundle( + "chrome://browser/locale/browser.properties" + ); + const body = bundle.formatStringFromName("timer.start", [ + aMessage.arguments.name, + ]); + Services.console.logStringMessage(body); + } else if (aMessage.level == "timeEnd" && aMessage.arguments) { + const bundle = Services.strings.createBundle( + "chrome://browser/locale/browser.properties" + ); + const body = bundle.formatStringFromName("timer.end", [ + aMessage.arguments.name, + aMessage.arguments.duration, + ]); + Services.console.logStringMessage(body); + } else if ( + ["group", "groupCollapsed", "groupEnd"].includes(aMessage.level) + ) { + // Do nothing yet + } else { + Services.console.logStringMessage(joinedArguments); + } + }, + + getResultType(aResult) { + let type = aResult === null ? "null" : typeof aResult; + if (type == "object" && aResult.constructor && aResult.constructor.name) { + type = aResult.constructor.name; + } + return type.toLowerCase(); + }, + + formatResult(aResult) { + let output = ""; + const type = this.getResultType(aResult); + switch (type) { + case "string": + case "boolean": + case "date": + case "error": + case "number": + case "regexp": + output = aResult.toString(); + break; + case "null": + case "undefined": + output = type; + break; + default: + output = aResult.toString(); + break; + } + + return output; + }, + + abbreviateSourceURL(aSourceURL) { + // Remove any query parameters. + const hookIndex = aSourceURL.indexOf("?"); + if (hookIndex > -1) { + aSourceURL = aSourceURL.substring(0, hookIndex); + } + + // Remove a trailing "/". + if (aSourceURL[aSourceURL.length - 1] == "/") { + aSourceURL = aSourceURL.substring(0, aSourceURL.length - 1); + } + + // Remove all but the last path component. + const slashIndex = aSourceURL.lastIndexOf("/"); + if (slashIndex > -1) { + aSourceURL = aSourceURL.substring(slashIndex + 1); + } + + return aSourceURL; + }, +}; |