gecko-dev/toolkit/components/extensions/ext-management.js

285 строки
8.7 KiB
JavaScript

/* -*- Mode: indent-tabs-mode: nil; js-indent-level: 2 -*- */
/* vim: set sts=2 sw=2 et tw=80: */
"use strict";
// The ext-* files are imported into the same scopes.
/* import-globals-from ext-toolkit.js */
XPCOMUtils.defineLazyGetter(this, "strBundle", function() {
const stringSvc = Cc["@mozilla.org/intl/stringbundle;1"].getService(Ci.nsIStringBundleService);
return stringSvc.createBundle("chrome://global/locale/extensions.properties");
});
XPCOMUtils.defineLazyModuleGetter(this, "AddonManager",
"resource://gre/modules/AddonManager.jsm");
XPCOMUtils.defineLazyServiceGetter(this, "promptService",
"@mozilla.org/embedcomp/prompt-service;1",
"nsIPromptService");
XPCOMUtils.defineLazyGetter(this, "GlobalManager", () => {
const {GlobalManager} = Cu.import("resource://gre/modules/Extension.jsm", {});
return GlobalManager;
});
var {
ExtensionError,
} = ExtensionUtils;
const _ = (key, ...args) => {
if (args.length) {
return strBundle.formatStringFromName(key, args, args.length);
}
return strBundle.GetStringFromName(key);
};
const installType = addon => {
if (addon.temporarilyInstalled) {
return "development";
} else if (addon.foreignInstall) {
return "sideload";
} else if (addon.isSystem) {
return "other";
}
return "normal";
};
const getExtensionInfoForAddon = (extension, addon) => {
let extInfo = {
id: addon.id,
name: addon.name,
description: addon.description || "",
version: addon.version,
mayDisable: !!(addon.permissions & AddonManager.PERM_CAN_DISABLE),
enabled: addon.isActive,
optionsUrl: addon.optionsURL || "",
installType: installType(addon),
type: addon.type,
};
if (extension) {
let m = extension.manifest;
let hostPerms = extension.whiteListedHosts.patterns.map(matcher => matcher.pattern);
extInfo.permissions = Array.from(extension.permissions).filter(perm => {
return !hostPerms.includes(perm);
});
extInfo.hostPermissions = hostPerms;
extInfo.shortName = m.short_name || "";
if (m.icons) {
extInfo.icons = Object.keys(m.icons).map(key => {
return {size: Number(key), url: m.icons[key]};
});
}
}
if (!addon.isActive) {
extInfo.disabledReason = "unknown";
}
if (addon.homepageURL) {
extInfo.homepageUrl = addon.homepageURL;
}
if (addon.updateURL) {
extInfo.updateUrl = addon.updateURL;
}
return extInfo;
};
const listenerMap = new WeakMap();
// Some management APIs are intentionally limited.
const allowedTypes = ["theme", "extension"];
function checkAllowedAddon(addon) {
if (addon.isSystem || addon.isAPIExtension) {
return false;
}
if (addon.type == "extension" && !addon.isWebExtension) {
return false;
}
return allowedTypes.includes(addon.type);
}
class AddonListener extends ExtensionUtils.EventEmitter {
constructor() {
super();
AddonManager.addAddonListener(this);
}
release() {
AddonManager.removeAddonListener(this);
}
getExtensionInfo(addon) {
let ext = addon.isWebExtension && GlobalManager.extensionMap.get(addon.id);
return getExtensionInfoForAddon(ext, addon);
}
onEnabled(addon) {
if (!checkAllowedAddon(addon)) {
return;
}
this.emit("onEnabled", this.getExtensionInfo(addon));
}
onDisabled(addon) {
if (!checkAllowedAddon(addon)) {
return;
}
this.emit("onDisabled", this.getExtensionInfo(addon));
}
onInstalled(addon) {
if (!checkAllowedAddon(addon)) {
return;
}
this.emit("onInstalled", this.getExtensionInfo(addon));
}
onUninstalled(addon) {
if (!checkAllowedAddon(addon)) {
return;
}
this.emit("onUninstalled", this.getExtensionInfo(addon));
}
}
let addonListener;
const getManagementListener = (extension, context) => {
if (!listenerMap.has(extension)) {
if (!addonListener) {
addonListener = new AddonListener();
}
listenerMap.set(extension, {});
context.callOnClose({
close: () => {
listenerMap.delete(extension);
if (listenerMap.length === 0) {
addonListener.release();
addonListener = null;
}
},
});
}
return addonListener;
};
this.management = class extends ExtensionAPI {
getAPI(context) {
let {extension} = context;
return {
management: {
async get(id) {
let addon = await AddonManager.getAddonByID(id);
if (!addon) {
throw new ExtensionError(`No such addon ${id}`);
}
if (!checkAllowedAddon(addon)) {
throw new ExtensionError("get not allowed for this addon");
}
// If the extension is enabled get it and use it for more data.
let ext = GlobalManager.extensionMap.get(addon.id);
return getExtensionInfoForAddon(ext, addon);
},
async getAll() {
let addons = await AddonManager.getAddonsByTypes(allowedTypes);
return addons.filter(checkAllowedAddon).map(addon => {
// If the extension is enabled get it and use it for more data.
let ext = GlobalManager.extensionMap.get(addon.id);
return getExtensionInfoForAddon(ext, addon);
});
},
async getSelf() {
let addon = await AddonManager.getAddonByID(extension.id);
return getExtensionInfoForAddon(extension, addon);
},
async uninstallSelf(options) {
if (options && options.showConfirmDialog) {
let message = _("uninstall.confirmation.message", extension.name);
if (options.dialogMessage) {
message = `${options.dialogMessage}\n${message}`;
}
let title = _("uninstall.confirmation.title", extension.name);
let buttonFlags = promptService.BUTTON_POS_0 * promptService.BUTTON_TITLE_IS_STRING +
promptService.BUTTON_POS_1 * promptService.BUTTON_TITLE_IS_STRING;
let button0Title = _("uninstall.confirmation.button-0.label");
let button1Title = _("uninstall.confirmation.button-1.label");
let response = promptService.confirmEx(null, title, message, buttonFlags, button0Title, button1Title, null, null, {value: 0});
if (response == 1) {
throw new ExtensionError("User cancelled uninstall of extension");
}
}
let addon = await AddonManager.getAddonByID(extension.id);
let canUninstall = Boolean(addon.permissions & AddonManager.PERM_CAN_UNINSTALL);
if (!canUninstall) {
throw new ExtensionError("The add-on cannot be uninstalled");
}
addon.uninstall();
},
async setEnabled(id, enabled) {
let addon = await AddonManager.getAddonByID(id);
if (!addon) {
throw new ExtensionError(`No such addon ${id}`);
}
if (addon.type !== "theme") {
throw new ExtensionError("setEnabled applies only to theme addons");
}
if (addon.isSystem) {
throw new ExtensionError("setEnabled cannot be used with a system addon");
}
addon.userDisabled = !enabled;
},
onDisabled: new EventManager(context, "management.onDisabled", fire => {
let listener = (event, data) => {
fire.async(data);
};
getManagementListener(extension, context).on("onDisabled", listener);
return () => {
getManagementListener(extension, context).off("onDisabled", listener);
};
}).api(),
onEnabled: new EventManager(context, "management.onEnabled", fire => {
let listener = (event, data) => {
fire.async(data);
};
getManagementListener(extension, context).on("onEnabled", listener);
return () => {
getManagementListener(extension, context).off("onEnabled", listener);
};
}).api(),
onInstalled: new EventManager(context, "management.onInstalled", fire => {
let listener = (event, data) => {
fire.async(data);
};
getManagementListener(extension, context).on("onInstalled", listener);
return () => {
getManagementListener(extension, context).off("onInstalled", listener);
};
}).api(),
onUninstalled: new EventManager(context, "management.onUninstalled", fire => {
let listener = (event, data) => {
fire.async(data);
};
getManagementListener(extension, context).on("onUninstalled", listener);
return () => {
getManagementListener(extension, context).off("onUninstalled", listener);
};
}).api(),
},
};
}
};