зеркало из https://github.com/mozilla/gecko-dev.git
264 строки
8.6 KiB
JavaScript
264 строки
8.6 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/. */
|
|
|
|
Components.utils.import("resource://gre/modules/Services.jsm");
|
|
|
|
/**
|
|
* Special Powers Exception - used to throw exceptions nicely
|
|
**/
|
|
function SpecialPowersException(aMsg) {
|
|
this.message = aMsg;
|
|
this.name = "SpecialPowersException";
|
|
}
|
|
|
|
SpecialPowersException.prototype.toString = function() {
|
|
return this.name + ': "' + this.message + '"';
|
|
};
|
|
|
|
function SpecialPowersObserverAPI() {
|
|
this._crashDumpDir = null;
|
|
this._processCrashObserversRegistered = false;
|
|
}
|
|
|
|
function parseKeyValuePairs(text) {
|
|
var lines = text.split('\n');
|
|
var data = {};
|
|
for (let i = 0; i < lines.length; i++) {
|
|
if (lines[i] == '')
|
|
continue;
|
|
|
|
// can't just .split() because the value might contain = characters
|
|
let eq = lines[i].indexOf('=');
|
|
if (eq != -1) {
|
|
let [key, value] = [lines[i].substring(0, eq),
|
|
lines[i].substring(eq + 1)];
|
|
if (key && value)
|
|
data[key] = value.replace(/\\n/g, "\n").replace(/\\\\/g, "\\");
|
|
}
|
|
}
|
|
return data;
|
|
}
|
|
|
|
function parseKeyValuePairsFromFile(file) {
|
|
var fstream = Cc["@mozilla.org/network/file-input-stream;1"].
|
|
createInstance(Ci.nsIFileInputStream);
|
|
fstream.init(file, -1, 0, 0);
|
|
var is = Cc["@mozilla.org/intl/converter-input-stream;1"].
|
|
createInstance(Ci.nsIConverterInputStream);
|
|
is.init(fstream, "UTF-8", 1024, Ci.nsIConverterInputStream.DEFAULT_REPLACEMENT_CHARACTER);
|
|
var str = {};
|
|
var contents = '';
|
|
while (is.readString(4096, str) != 0) {
|
|
contents += str.value;
|
|
}
|
|
is.close();
|
|
fstream.close();
|
|
return parseKeyValuePairs(contents);
|
|
}
|
|
|
|
SpecialPowersObserverAPI.prototype = {
|
|
|
|
_observe: function(aSubject, aTopic, aData) {
|
|
switch(aTopic) {
|
|
case "plugin-crashed":
|
|
case "ipc:content-shutdown":
|
|
function addDumpIDToMessage(propertyName) {
|
|
var id = aSubject.getPropertyAsAString(propertyName);
|
|
if (id) {
|
|
message.dumpIDs.push({id: id, extension: "dmp"});
|
|
message.dumpIDs.push({id: id, extension: "extra"});
|
|
}
|
|
}
|
|
|
|
var message = { type: "crash-observed", dumpIDs: [] };
|
|
aSubject = aSubject.QueryInterface(Ci.nsIPropertyBag2);
|
|
if (aTopic == "plugin-crashed") {
|
|
addDumpIDToMessage("pluginDumpID");
|
|
addDumpIDToMessage("browserDumpID");
|
|
|
|
let pluginID = aSubject.getPropertyAsAString("pluginDumpID");
|
|
let extra = this._getExtraData(pluginID);
|
|
if (extra && ("additional_minidumps" in extra)) {
|
|
let dumpNames = extra.additional_minidumps.split(',');
|
|
for (let name of dumpNames) {
|
|
message.dumpIDs.push({id: pluginID + "-" + name, extension: "dmp"});
|
|
}
|
|
}
|
|
} else { // ipc:content-shutdown
|
|
addDumpIDToMessage("dumpID");
|
|
}
|
|
this._sendAsyncMessage("SPProcessCrashService", message);
|
|
break;
|
|
}
|
|
},
|
|
|
|
_getCrashDumpDir: function() {
|
|
if (!this._crashDumpDir) {
|
|
this._crashDumpDir = Services.dirsvc.get("ProfD", Ci.nsIFile);
|
|
this._crashDumpDir.append("minidumps");
|
|
}
|
|
return this._crashDumpDir;
|
|
},
|
|
|
|
_getExtraData: function(dumpId) {
|
|
let extraFile = this._getCrashDumpDir().clone();
|
|
extraFile.append(dumpId + ".extra");
|
|
if (!extraFile.exists()) {
|
|
return null;
|
|
}
|
|
return parseKeyValuePairsFromFile(extraFile);
|
|
},
|
|
|
|
_deleteCrashDumpFiles: function(aFilenames) {
|
|
var crashDumpDir = this._getCrashDumpDir();
|
|
if (!crashDumpDir.exists()) {
|
|
return false;
|
|
}
|
|
|
|
var success = aFilenames.length != 0;
|
|
aFilenames.forEach(function(crashFilename) {
|
|
var file = crashDumpDir.clone();
|
|
file.append(crashFilename);
|
|
if (file.exists()) {
|
|
file.remove(false);
|
|
} else {
|
|
success = false;
|
|
}
|
|
});
|
|
return success;
|
|
},
|
|
|
|
_findCrashDumpFiles: function(aToIgnore) {
|
|
var crashDumpDir = this._getCrashDumpDir();
|
|
var entries = crashDumpDir.exists() && crashDumpDir.directoryEntries;
|
|
if (!entries) {
|
|
return [];
|
|
}
|
|
|
|
var crashDumpFiles = [];
|
|
while (entries.hasMoreElements()) {
|
|
var file = entries.getNext().QueryInterface(Ci.nsIFile);
|
|
var path = String(file.path);
|
|
if (path.match(/\.(dmp|extra)$/) && !aToIgnore[path]) {
|
|
crashDumpFiles.push(path);
|
|
}
|
|
}
|
|
return crashDumpFiles.concat();
|
|
},
|
|
|
|
_getURI: function (url) {
|
|
return Services.io.newURI(url, null, null);
|
|
},
|
|
|
|
/**
|
|
* messageManager callback function
|
|
* This will get requests from our API in the window and process them in chrome for it
|
|
**/
|
|
_receiveMessageAPI: function(aMessage) {
|
|
switch(aMessage.name) {
|
|
case "SPPrefService":
|
|
var prefs = Services.prefs;
|
|
var prefType = aMessage.json.prefType.toUpperCase();
|
|
var prefName = aMessage.json.prefName;
|
|
var prefValue = "prefValue" in aMessage.json ? aMessage.json.prefValue : null;
|
|
|
|
if (aMessage.json.op == "get") {
|
|
if (!prefName || !prefType)
|
|
throw new SpecialPowersException("Invalid parameters for get in SPPrefService");
|
|
} else if (aMessage.json.op == "set") {
|
|
if (!prefName || !prefType || prefValue === null)
|
|
throw new SpecialPowersException("Invalid parameters for set in SPPrefService");
|
|
} else if (aMessage.json.op == "clear") {
|
|
if (!prefName)
|
|
throw new SpecialPowersException("Invalid parameters for clear in SPPrefService");
|
|
} else {
|
|
throw new SpecialPowersException("Invalid operation for SPPrefService");
|
|
}
|
|
|
|
// Now we make the call
|
|
switch(prefType) {
|
|
case "BOOL":
|
|
if (aMessage.json.op == "get")
|
|
return(prefs.getBoolPref(prefName));
|
|
else
|
|
return(prefs.setBoolPref(prefName, prefValue));
|
|
case "INT":
|
|
if (aMessage.json.op == "get")
|
|
return(prefs.getIntPref(prefName));
|
|
else
|
|
return(prefs.setIntPref(prefName, prefValue));
|
|
case "CHAR":
|
|
if (aMessage.json.op == "get")
|
|
return(prefs.getCharPref(prefName));
|
|
else
|
|
return(prefs.setCharPref(prefName, prefValue));
|
|
case "COMPLEX":
|
|
if (aMessage.json.op == "get")
|
|
return(prefs.getComplexValue(prefName, prefValue[0]));
|
|
else
|
|
return(prefs.setComplexValue(prefName, prefValue[0], prefValue[1]));
|
|
case "":
|
|
if (aMessage.json.op == "clear") {
|
|
prefs.clearUserPref(prefName);
|
|
return;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case "SPProcessCrashService":
|
|
switch (aMessage.json.op) {
|
|
case "register-observer":
|
|
this._addProcessCrashObservers();
|
|
break;
|
|
case "unregister-observer":
|
|
this._removeProcessCrashObservers();
|
|
break;
|
|
case "delete-crash-dump-files":
|
|
return this._deleteCrashDumpFiles(aMessage.json.filenames);
|
|
case "find-crash-dump-files":
|
|
return this._findCrashDumpFiles(aMessage.json.crashDumpFilesToIgnore);
|
|
default:
|
|
throw new SpecialPowersException("Invalid operation for SPProcessCrashService");
|
|
}
|
|
break;
|
|
|
|
case "SPPermissionManager":
|
|
let msg = aMessage.json;
|
|
|
|
let secMan = Services.scriptSecurityManager;
|
|
let principal = secMan.getAppCodebasePrincipal(this._getURI(msg.url), msg.appId, msg.isInBrowserElement);
|
|
|
|
switch (msg.op) {
|
|
case "add":
|
|
Services.perms.addFromPrincipal(principal, msg.type, msg.permission);
|
|
break;
|
|
case "remove":
|
|
Services.perms.removeFromPrincipal(principal, msg.type);
|
|
break;
|
|
case "has":
|
|
let hasPerm = Services.perms.testPermissionFromPrincipal(principal, msg.type);
|
|
if (hasPerm == Ci.nsIPermissionManager.ALLOW_ACTION)
|
|
return true;
|
|
return false;
|
|
break;
|
|
case "test":
|
|
let testPerm = Services.perms.testPermissionFromPrincipal(principal, msg.type, msg.value);
|
|
if (testPerm == msg.value) {
|
|
return true;
|
|
}
|
|
return false;
|
|
break;
|
|
default:
|
|
throw new SpecialPowersException("Invalid operation for " +
|
|
"SPPermissionManager");
|
|
}
|
|
break;
|
|
|
|
default:
|
|
throw new SpecialPowersException("Unrecognized Special Powers API");
|
|
}
|
|
}
|
|
};
|
|
|