mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-11-07 03:38:51 +02:00
Depends on D108226 Differential Revision: https://phabricator.services.mozilla.com/D108227
172 lines
5 KiB
JavaScript
172 lines
5 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/. */
|
|
|
|
"use strict";
|
|
|
|
XPCOMUtils.defineLazyModuleGetters(this, {
|
|
ExtensionPermissions: "resource://gre/modules/ExtensionPermissions.jsm",
|
|
});
|
|
|
|
var { ExtensionError } = ExtensionUtils;
|
|
|
|
XPCOMUtils.defineLazyPreferenceGetter(
|
|
this,
|
|
"promptsEnabled",
|
|
"extensions.webextOptionalPermissionPrompts"
|
|
);
|
|
|
|
function normalizePermissions(perms) {
|
|
perms = { ...perms };
|
|
perms.permissions = perms.permissions.filter(
|
|
perm => !perm.startsWith("internal:")
|
|
);
|
|
return perms;
|
|
}
|
|
|
|
this.permissions = class extends ExtensionAPI {
|
|
getAPI(context) {
|
|
let { extension } = context;
|
|
|
|
return {
|
|
permissions: {
|
|
async request(perms) {
|
|
let { permissions, origins } = perms;
|
|
|
|
let manifestPermissions =
|
|
context.extension.manifest.optional_permissions;
|
|
for (let perm of permissions) {
|
|
if (!manifestPermissions.includes(perm)) {
|
|
throw new ExtensionError(
|
|
`Cannot request permission ${perm} since it was not declared in optional_permissions`
|
|
);
|
|
}
|
|
}
|
|
|
|
let optionalOrigins = context.extension.optionalOrigins;
|
|
for (let origin of origins) {
|
|
if (!optionalOrigins.subsumes(new MatchPattern(origin))) {
|
|
throw new ExtensionError(
|
|
`Cannot request origin permission for ${origin} since it was not declared in optional_permissions`
|
|
);
|
|
}
|
|
}
|
|
|
|
if (promptsEnabled) {
|
|
permissions = permissions.filter(
|
|
perm => !context.extension.hasPermission(perm)
|
|
);
|
|
origins = origins.filter(
|
|
origin =>
|
|
!context.extension.allowedOrigins.subsumes(
|
|
new MatchPattern(origin)
|
|
)
|
|
);
|
|
|
|
if (!permissions.length && !origins.length) {
|
|
return true;
|
|
}
|
|
|
|
let browser = context.pendingEventBrowser || context.xulBrowser;
|
|
let allow = await new Promise(resolve => {
|
|
let subject = {
|
|
wrappedJSObject: {
|
|
browser,
|
|
name: context.extension.name,
|
|
icon: context.extension.iconURL,
|
|
permissions: { permissions, origins },
|
|
resolve,
|
|
},
|
|
};
|
|
Services.obs.notifyObservers(
|
|
subject,
|
|
"webextension-optional-permission-prompt"
|
|
);
|
|
});
|
|
if (!allow) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
await ExtensionPermissions.add(extension.id, perms, extension);
|
|
return true;
|
|
},
|
|
|
|
async getAll() {
|
|
let perms = normalizePermissions(context.extension.activePermissions);
|
|
delete perms.apis;
|
|
return perms;
|
|
},
|
|
|
|
async contains(permissions) {
|
|
for (let perm of permissions.permissions) {
|
|
if (!context.extension.hasPermission(perm)) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
for (let origin of permissions.origins) {
|
|
if (
|
|
!context.extension.allowedOrigins.subsumes(
|
|
new MatchPattern(origin)
|
|
)
|
|
) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
},
|
|
|
|
async remove(permissions) {
|
|
await ExtensionPermissions.remove(
|
|
extension.id,
|
|
permissions,
|
|
extension
|
|
);
|
|
return true;
|
|
},
|
|
|
|
onAdded: new EventManager({
|
|
context,
|
|
name: "permissions.onAdded",
|
|
register: fire => {
|
|
let callback = (event, change) => {
|
|
if (change.extensionId == extension.id && change.added) {
|
|
let perms = normalizePermissions(change.added);
|
|
if (perms.permissions.length || perms.origins.length) {
|
|
fire.async(perms);
|
|
}
|
|
}
|
|
};
|
|
|
|
extensions.on("change-permissions", callback);
|
|
return () => {
|
|
extensions.off("change-permissions", callback);
|
|
};
|
|
},
|
|
}).api(),
|
|
|
|
onRemoved: new EventManager({
|
|
context,
|
|
name: "permissions.onRemoved",
|
|
register: fire => {
|
|
let callback = (event, change) => {
|
|
if (change.extensionId == extension.id && change.removed) {
|
|
let perms = normalizePermissions(change.removed);
|
|
if (perms.permissions.length || perms.origins.length) {
|
|
fire.async(perms);
|
|
}
|
|
}
|
|
};
|
|
|
|
extensions.on("change-permissions", callback);
|
|
return () => {
|
|
extensions.off("change-permissions", callback);
|
|
};
|
|
},
|
|
}).api(),
|
|
},
|
|
};
|
|
}
|
|
};
|