Merge pull request #18 from mozilla/experiment-extension
(Squashed) experiment-extension branch
This commit is contained in:
commit
e534da7c88
8 changed files with 294 additions and 17 deletions
6
index.js
6
index.js
|
@ -2,12 +2,6 @@ const webExtension = require('sdk/webextension');
|
||||||
const {ContextualIdentityService} = require('resource://gre/modules/ContextualIdentityService.jsm');
|
const {ContextualIdentityService} = require('resource://gre/modules/ContextualIdentityService.jsm');
|
||||||
|
|
||||||
function handleWebExtensionMessage(message, sender, sendReply) {
|
function handleWebExtensionMessage(message, sender, sendReply) {
|
||||||
console.log(message);
|
|
||||||
if (message === 'get-identities') {
|
|
||||||
sendReply({
|
|
||||||
content: {identities: ContextualIdentityService.getIdentities()}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
webExtension.startup().then(api=> {
|
webExtension.startup().then(api=> {
|
||||||
|
|
27
webextension-experiment/LICENSE
Normal file
27
webextension-experiment/LICENSE
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
Copyright (c) 2010 - 2016, Mozilla Corporation
|
||||||
|
All rights reserved.
|
||||||
|
|
||||||
|
Redistribution and use in source and binary forms, with or without
|
||||||
|
modification, are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
|
* Redistributions of source code must retain the above copyright notice,
|
||||||
|
this list of conditions and the following disclaimer.
|
||||||
|
|
||||||
|
* Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
this list of conditions and the following disclaimer in the documentation
|
||||||
|
and/or other materials provided with the distribution.
|
||||||
|
|
||||||
|
* Neither the name of the Mozilla Corporation nor the names of its contributors
|
||||||
|
may be used to endorse or promote products derived from this software without
|
||||||
|
specific prior written permission.
|
||||||
|
|
||||||
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
||||||
|
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||||||
|
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||||
|
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||||
|
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||||
|
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||||
|
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
7
webextension-experiment/README.md
Normal file
7
webextension-experiment/README.md
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
|
||||||
|
## Webextension contextual identities API extension
|
||||||
|
|
||||||
|
This project contains the implementation of the Firefox
|
||||||
|
browser.contextualIdentites API.
|
||||||
|
|
||||||
|
Based on: https://bugzilla.mozilla.org/show_bug.cgi?id=1322856
|
118
webextension-experiment/api.js
Normal file
118
webextension-experiment/api.js
Normal file
|
@ -0,0 +1,118 @@
|
||||||
|
"use strict";
|
||||||
|
const {classes: Cc, interfaces: Ci, results: Cr, utils: Cu} = Components;
|
||||||
|
|
||||||
|
Cu.import("resource://gre/modules/Services.jsm");
|
||||||
|
const {XPCOMUtils} = Cu.import("resource://gre/modules/XPCOMUtils.jsm");
|
||||||
|
const CONTAINER_STORE = "firefox-container-";
|
||||||
|
|
||||||
|
XPCOMUtils.defineLazyModuleGetter(this, "ContextualIdentityService",
|
||||||
|
"resource://gre/modules/ContextualIdentityService.jsm");
|
||||||
|
|
||||||
|
function convert(identity) {
|
||||||
|
let result = {
|
||||||
|
name: ContextualIdentityService.getUserContextLabel(identity.userContextId),
|
||||||
|
icon: identity.icon,
|
||||||
|
color: identity.color,
|
||||||
|
cookieStoreId: getCookieStoreIdForContainer(identity.userContextId),
|
||||||
|
};
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getCookieStoreIdForContainer(containerId) {
|
||||||
|
return CONTAINER_STORE + containerId;
|
||||||
|
}
|
||||||
|
|
||||||
|
class API extends ExtensionAPI {
|
||||||
|
getAPI(context) {
|
||||||
|
let self = {
|
||||||
|
contextualIdentities: {
|
||||||
|
get(cookieStoreId) {
|
||||||
|
let containerId = getContainerForCookieStoreId(cookieStoreId);
|
||||||
|
if (!containerId) {
|
||||||
|
return Promise.resolve(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
let identity = ContextualIdentityService.getIdentityFromId(containerId);
|
||||||
|
return Promise.resolve(convert(identity));
|
||||||
|
},
|
||||||
|
|
||||||
|
query(details) {
|
||||||
|
let identities = [];
|
||||||
|
ContextualIdentityService.getIdentities().forEach(identity => {
|
||||||
|
if (details.name &&
|
||||||
|
ContextualIdentityService.getUserContextLabel(identity.userContextId) != details.name) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
identities.push(convert(identity));
|
||||||
|
});
|
||||||
|
|
||||||
|
return Promise.resolve(identities);
|
||||||
|
},
|
||||||
|
|
||||||
|
create(details) {
|
||||||
|
let identity = ContextualIdentityService.create(details.name,
|
||||||
|
details.icon,
|
||||||
|
details.color);
|
||||||
|
return Promise.resolve(convert(identity));
|
||||||
|
},
|
||||||
|
|
||||||
|
update(cookieStoreId, details) {
|
||||||
|
let containerId = getContainerForCookieStoreId(cookieStoreId);
|
||||||
|
if (!containerId) {
|
||||||
|
return Promise.resolve(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
let identity = ContextualIdentityService.getIdentityFromId(containerId);
|
||||||
|
if (!identity) {
|
||||||
|
return Promise.resolve(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (details.name !== null) {
|
||||||
|
identity.name = details.name;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (details.color !== null) {
|
||||||
|
identity.color = details.color;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (details.icon !== null) {
|
||||||
|
identity.icon = details.icon;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!ContextualIdentityService.update(identity.userContextId,
|
||||||
|
identity.name, identity.icon,
|
||||||
|
identity.color)) {
|
||||||
|
return Promise.resolve(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Promise.resolve(convert(identity));
|
||||||
|
},
|
||||||
|
|
||||||
|
remove(cookieStoreId) {
|
||||||
|
let containerId = getContainerForCookieStoreId(cookieStoreId);
|
||||||
|
if (!containerId) {
|
||||||
|
return Promise.resolve(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
let identity = ContextualIdentityService.getIdentityFromId(containerId);
|
||||||
|
if (!identity) {
|
||||||
|
return Promise.resolve(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
// We have to create the identity object before removing it.
|
||||||
|
let convertedIdentity = convert(identity);
|
||||||
|
|
||||||
|
if (!ContextualIdentityService.remove(identity.userContextId)) {
|
||||||
|
return Promise.resolve(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Promise.resolve(convertedIdentity);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
}
|
31
webextension-experiment/install.rdf
Normal file
31
webextension-experiment/install.rdf
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<RDF xmlns="http://www.w3.org/1999/02/22-rdf-syntax-ns#" xmlns:em="http://www.mozilla.org/2004/em-rdf#">
|
||||||
|
<Description about="urn:mozilla:install-manifest">
|
||||||
|
<em:id>contextualidentities@experiments.addons.mozilla.org</em:id>
|
||||||
|
<em:name>Experimental Contextual Identites API</em:name>
|
||||||
|
<em:type>256</em:type>
|
||||||
|
<em:version>0.1</em:version>
|
||||||
|
<em:description>Experimental Contextual Identities API</em:description>
|
||||||
|
<em:creator>groovecoder, baku and jkt</em:creator>
|
||||||
|
<em:multiprocessCompatible>true</em:multiprocessCompatible>
|
||||||
|
|
||||||
|
<!-- Firefox -->
|
||||||
|
<em:targetApplication>
|
||||||
|
<Description>
|
||||||
|
<em:id>{ec8030f7-c20a-464f-9b0e-13a3a9e97384}</em:id>
|
||||||
|
<em:minVersion>51.0</em:minVersion>
|
||||||
|
<em:maxVersion>*</em:maxVersion>
|
||||||
|
</Description>
|
||||||
|
</em:targetApplication>
|
||||||
|
|
||||||
|
<!-- for testing -->
|
||||||
|
<em:targetApplication>
|
||||||
|
<Description>
|
||||||
|
<em:id>xpcshell@tests.mozilla.org</em:id>
|
||||||
|
<em:minVersion>1</em:minVersion>
|
||||||
|
<em:maxVersion>2</em:maxVersion>
|
||||||
|
</Description>
|
||||||
|
</em:targetApplication>
|
||||||
|
|
||||||
|
</Description>
|
||||||
|
</RDF>
|
105
webextension-experiment/schema.json
Normal file
105
webextension-experiment/schema.json
Normal file
|
@ -0,0 +1,105 @@
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"namespace": "contextualIdentities",
|
||||||
|
"description": "Use the <code>browser.contextualIdentities</code> API to query and modify contextual identity, also called as containers.",
|
||||||
|
"permissions": ["contextualidentities"],
|
||||||
|
"types": [
|
||||||
|
{
|
||||||
|
"id": "ContextualIdentity",
|
||||||
|
"type": "object",
|
||||||
|
"description": "Represents information about a contextual identity.",
|
||||||
|
"properties": {
|
||||||
|
"name": {"type": "string", "description": "The name of the contextual identity."},
|
||||||
|
"icon": {"type": "string", "description": "The icon of the contextual identity."},
|
||||||
|
"color": {"type": "string", "description": "The color of the contextual identity."},
|
||||||
|
"cookieStoreId": {"type": "string", "description": "The cookie store ID of the contextual identity."}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"functions": [
|
||||||
|
{
|
||||||
|
"name": "get",
|
||||||
|
"type": "function",
|
||||||
|
"description": "Retrieves information about a single contextual identity.",
|
||||||
|
"async": true,
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"type": "string",
|
||||||
|
"name": "cookieStoreId",
|
||||||
|
"description": "The ID of the contextual identity cookie store. "
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "query",
|
||||||
|
"type": "function",
|
||||||
|
"description": "Retrieves all contextual identities",
|
||||||
|
"async": true,
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"type": "object",
|
||||||
|
"name": "details",
|
||||||
|
"description": "Information to filter the contextual identities being retrieved.",
|
||||||
|
"properties": {
|
||||||
|
"name": {"type": "string", "optional": true, "description": "Filters the contextual identity by name."}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "create",
|
||||||
|
"type": "function",
|
||||||
|
"description": "Creates a contextual identity with the given data.",
|
||||||
|
"async": true,
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"type": "object",
|
||||||
|
"name": "details",
|
||||||
|
"description": "Details about the contextual identity being created.",
|
||||||
|
"properties": {
|
||||||
|
"name": {"type": "string", "optional": false, "description": "The name of the contextual identity." },
|
||||||
|
"color": {"type": "string", "optional": false, "description": "The color of the contextual identity." },
|
||||||
|
"icon": {"type": "string", "optional": false, "description": "The icon of the contextual identity." }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "update",
|
||||||
|
"type": "function",
|
||||||
|
"description": "Updates a contextual identity with the given data.",
|
||||||
|
"async": true,
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"type": "string",
|
||||||
|
"name": "cookieStoreId",
|
||||||
|
"description": "The ID of the contextual identity cookie store. "
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "object",
|
||||||
|
"name": "details",
|
||||||
|
"description": "Details about the contextual identity being created.",
|
||||||
|
"properties": {
|
||||||
|
"name": {"type": "string", "optional": true, "description": "The name of the contextual identity." },
|
||||||
|
"color": {"type": "string", "optional": true, "description": "The color of the contextual identity." },
|
||||||
|
"icon": {"type": "string", "optional": true, "description": "The icon of the contextual identity." }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "remove",
|
||||||
|
"type": "function",
|
||||||
|
"description": "Deletes a contetual identity by its cookie Store ID.",
|
||||||
|
"async": true,
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"type": "string",
|
||||||
|
"name": "cookieStoreId",
|
||||||
|
"description": "The ID of the contextual identity cookie store. "
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
|
@ -1,12 +1,5 @@
|
||||||
const IDENTITY_L10NID_MATCH_INDEX = 1;
|
browser.contextualIdentities.query({}).then((identites) => {
|
||||||
|
identites.forEach((identity) => {
|
||||||
browser.runtime.sendMessage('get-identities').then(reply=> {
|
document.querySelector('.identities-list').innerHTML += `<li><a href="#">${identity.icon} ${identity.name}</a></li>`;
|
||||||
if (reply) {
|
});
|
||||||
reply.content.identities.forEach(identity=> {
|
|
||||||
const identityName = identity.l10nID.match(/userContext(\w*)\.label/)[IDENTITY_L10NID_MATCH_INDEX];
|
|
||||||
|
|
||||||
document.querySelector('.identities-list').innerHTML += `<li><a href="#">${identityName}</a></li>`;
|
|
||||||
});
|
|
||||||
console.log('response from sdk addon: ', reply.content);
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
|
@ -20,6 +20,8 @@
|
||||||
"homepage_url": "https://testpilot.firefox.com/",
|
"homepage_url": "https://testpilot.firefox.com/",
|
||||||
|
|
||||||
"permissions": [
|
"permissions": [
|
||||||
|
"experiments.contextualidentities",
|
||||||
|
"contextualidentities"
|
||||||
],
|
],
|
||||||
|
|
||||||
"browser_action": {
|
"browser_action": {
|
||||||
|
|
Loading…
Add table
Reference in a new issue