/* eslint max-len: ["error", 80] */
"use strict";
loadTestSubscript("head_disco.js");
// The response to the discovery API, as documented at:
// https://addons-server.readthedocs.io/en/latest/topics/api/discovery.html
//
// The test is designed to easily verify whether the discopane works with the
// latest AMO API, by replacing API_RESPONSE_FILE's content with latest AMO API
// response, e.g. from https://addons.allizom.org/api/v4/discovery/?lang=en-US
// The response must contain at least one theme, and one extension.
const API_RESPONSE_FILE = PathUtils.join(
Services.dirsvc.get("CurWorkD", Ci.nsIFile).path,
// Trim empty component from splitting with trailing slash.
...RELATIVE_DIR.split("/").filter(c => c.length),
"discovery",
"api_response.json"
);
const AMO_TEST_HOST = "rewritten-for-testing.addons.allizom.org";
const ArrayBufferInputStream = Components.Constructor(
"@mozilla.org/io/arraybuffer-input-stream;1",
"nsIArrayBufferInputStream",
"setData"
);
const amoServer = AddonTestUtils.createHttpServer({ hosts: [AMO_TEST_HOST] });
amoServer.registerFile(
"/png",
FileUtils.getFile(
"CurWorkD",
`${RELATIVE_DIR}discovery/small-1x1.png`.split("/")
)
);
amoServer.registerPathHandler("/dummy", (request, response) => {
response.write("Dummy");
});
// `result` is an element in the `results` array from AMO's discovery API,
// stored in API_RESPONSE_FILE.
function getTestExpectationFromApiResult(result) {
return {
typeIsTheme: result.addon.type === "statictheme",
addonName: result.addon.name,
authorName: result.addon.authors[0].name,
editorialBody: result.description_text,
dailyUsers: result.addon.average_daily_users,
rating: result.addon.ratings.average,
};
}
// A helper to declare a response to discovery API requests.
class DiscoveryAPIHandler {
constructor(responseText) {
this.setResponseText(responseText);
this.requestCount = 0;
// Overwrite the previous discovery response handler.
amoServer.registerPathHandler("/discoapi", this);
}
setResponseText(responseText) {
this.responseBody = new TextEncoder().encode(responseText).buffer;
}
// Suspend discovery API requests until unblockResponses is called.
blockNextResponses() {
this._unblockPromise = new Promise(resolve => {
this.unblockResponses = resolve;
});
}
unblockResponses(responseText) {
throw new Error("You need to call blockNextResponses first!");
}
// nsIHttpRequestHandler::handle
async handle(request, response) {
++this.requestCount;
response.setHeader("Cache-Control", "no-cache", false);
response.processAsync();
await this._unblockPromise;
let body = this.responseBody;
let binStream = new ArrayBufferInputStream(body, 0, body.byteLength);
response.bodyOutputStream.writeFrom(binStream, body.byteLength);
response.finish();
}
}
// Retrieve the list of visible action elements inside a document or container.
function getVisibleActions(documentOrElement) {
return Array.from(documentOrElement.querySelectorAll("[action]")).filter(
elem =>
elem.getAttribute("action") !== "page-options" &&
elem.offsetWidth &&
elem.offsetHeight
);
}
function getActionName(actionElement) {
return actionElement.getAttribute("action");
}
function getCardByAddonId(win, addonId) {
for (let card of win.document.querySelectorAll("recommended-addon-card")) {
if (card.addonId === addonId) {
return card;
}
}
return null;
}
// Switch to a different view so we can switch back to the discopane later.
async function switchToNonDiscoView(win) {
// Listeners registered while the discopane was the active view continue to be
// active when the view switches to the extensions list, because both views
// share the same document.
win.gViewController.loadView("addons://list/extension");
await wait_for_view_load(win);
ok(
win.document.querySelector("addon-list"),
"Should be at the extension list view"
);
}
// Switch to the discopane and wait until it has fully rendered, including any
// cards from the discovery API.
async function switchToDiscoView(win) {
is(
getDiscoveryElement(win),
null,
"Cannot switch to discopane when the discopane is already shown"
);
win.gViewController.loadView("addons://discover/");
await wait_for_view_load(win);
await promiseDiscopaneUpdate(win);
}
// Wait until all images in the DOM have successfully loaded.
// There must be at least one `` in the document.
// Returns the number of loaded images.
async function waitForAllImagesLoaded(win) {
let imgs = Array.from(
win.document.querySelectorAll("discovery-pane img[src]")
);
function areAllImagesLoaded() {
let loadCount = imgs.filter(img => img.naturalWidth).length;
info(`Loaded ${loadCount} out of ${imgs.length} images`);
return loadCount === imgs.length;
}
if (!areAllImagesLoaded()) {
await promiseEvent(win.document, "load", true, areAllImagesLoaded);
}
return imgs.length;
}
// Install an add-on by clicking on the card.
// The promise resolves once the card has been updated.
async function testCardInstall(card) {
Assert.deepEqual(
getVisibleActions(card).map(getActionName),
["install-addon"],
"Should have an Install button before install"
);
let installButton =
card.querySelector("[data-l10n-id='install-extension-button']") ||
card.querySelector("[data-l10n-id='install-theme-button']");
let updatePromise = promiseEvent(card, "disco-card-updated");
installButton.click();
await updatePromise;
Assert.deepEqual(
getVisibleActions(card).map(getActionName),
["manage-addon"],
"Should have a Manage button after install"
);
}
// Uninstall the add-on (not via the card, since it has no uninstall button).
// The promise resolves once the card has been updated.
async function testAddonUninstall(card) {
Assert.deepEqual(
getVisibleActions(card).map(getActionName),
["manage-addon"],
"Should have a Manage button before uninstall"
);
let addon = await AddonManager.getAddonByID(card.addonId);
let updatePromise = promiseEvent(card, "disco-card-updated");
await addon.uninstall();
await updatePromise;
Assert.deepEqual(
getVisibleActions(card).map(getActionName),
["install-addon"],
"Should have an Install button after uninstall"
);
}
add_setup(async function () {
await SpecialPowers.pushPrefEnv({
set: [
[
"extensions.getAddons.discovery.api_url",
`http://${AMO_TEST_HOST}/discoapi`,
],
// Disable non-discopane recommendations to avoid unexpected discovery
// API requests.
["extensions.htmlaboutaddons.recommendations.enabled", false],
// Disable the telemetry client ID (and its associated UI warning).
// browser_html_discover_view_clientid.js covers this functionality.
["browser.discovery.enabled", false],
],
});
});
// Test that the discopane can be loaded and that meaningful results are shown.
// This relies on response data from the AMO API, stored in API_RESPONSE_FILE.
add_task(async function discopane_with_real_api_data() {
const apiText = await readAPIResponseFixture(
AMO_TEST_HOST,
API_RESPONSE_FILE
);
let apiHandler = new DiscoveryAPIHandler(apiText);
const apiResultArray = JSON.parse(apiText).results;
ok(apiResultArray.length, `Mock has ${apiResultArray.length} results`);
apiHandler.blockNextResponses();
let win = await loadInitialView("discover");
Assert.deepEqual(
getVisibleActions(win.document).map(getActionName),
[],
"The AMO button should be invisible when the AMO API hasn't responded"
);
apiHandler.unblockResponses();
await promiseDiscopaneUpdate(win);
let actionElements = getVisibleActions(win.document);
Assert.deepEqual(
actionElements.map(getActionName),
[
// Expecting an install button for every result.
...new Array(apiResultArray.length).fill("install-addon"),
"open-amo",
],
"All add-on cards should be rendered, with AMO button at the end."
);
let imgCount = await waitForAllImagesLoaded(win);
is(imgCount, apiResultArray.length, "Expected an image for every result");
// Check that the cards have the expected content.
let cards = Array.from(
win.document.querySelectorAll("recommended-addon-card")
);
is(cards.length, apiResultArray.length, "Every API result has a card");
for (let [i, card] of cards.entries()) {
let expectations = getTestExpectationFromApiResult(apiResultArray[i]);
info(`Expectations for card ${i}: ${JSON.stringify(expectations)}`);
let checkContent = (selector, expectation) => {
let text = card.querySelector(selector).textContent;
is(text, expectation, `Content of selector "${selector}"`);
};
checkContent(".disco-addon-name", expectations.addonName);
await win.document.l10n.translateFragment(card);
checkContent(
".disco-addon-author [data-l10n-name='author']",
expectations.authorName
);
let amoListingLink = card.querySelector(".disco-addon-author a");
ok(
amoListingLink.search.includes("utm_source=firefox-browser"),
`Listing link should have attribution parameter, url=${amoListingLink}`
);
let actions = getVisibleActions(card);
is(actions.length, 1, "Card should only have one install button");
let installButton = actions[0];
if (expectations.typeIsTheme) {
// Theme button + screenshot
ok(
installButton.matches("[data-l10n-id='install-theme-button'"),
"Has theme install button"
);
ok(
card.querySelector(".card-heading-image").offsetWidth,
"Preview image must be visible"
);
} else {
// Extension button + extended description.
ok(
installButton.matches("[data-l10n-id='install-extension-button'"),
"Has extension install button"
);
checkContent(".disco-description-main", expectations.editorialBody);
let ratingElem = card.querySelector("five-star-rating");
if (expectations.rating) {
is(ratingElem.rating, expectations.rating, "Expected rating value");
ok(ratingElem.offsetWidth, "Rating element is visible");
} else {
is(ratingElem.offsetWidth, 0, "Rating element is not visible");
}
let userCountElem = card.querySelector(".disco-user-count");
if (expectations.dailyUsers) {
Assert.deepEqual(
win.document.l10n.getAttributes(userCountElem),
{ id: "user-count", args: { dailyUsers: expectations.dailyUsers } },
"Card count should be rendered"
);
} else {
is(userCountElem.offsetWidth, 0, "User count element is not visible");
}
}
}
is(apiHandler.requestCount, 1, "Discovery API should be fetched once");
await closeView(win);
});
// Test whether extensions and themes can be installed from the discopane.
// Also checks that items in the list do not change position after installation,
// and that they are shown at the bottom of the list when the discopane is
// reopened.
add_task(async function install_from_discopane() {
const apiText = await readAPIResponseFixture(
AMO_TEST_HOST,
API_RESPONSE_FILE
);
const apiResultArray = JSON.parse(apiText).results;
let getAddonIdByAMOAddonType = type =>
apiResultArray.find(r => r.addon.type === type).addon.guid;
const FIRST_EXTENSION_ID = getAddonIdByAMOAddonType("extension");
const FIRST_THEME_ID = getAddonIdByAMOAddonType("statictheme");
let apiHandler = new DiscoveryAPIHandler(apiText);
let win = await loadInitialView("discover");
await promiseDiscopaneUpdate(win);
await waitForAllImagesLoaded(win);
// Test extension install.
let installExtensionPromise = promiseAddonInstall(amoServer, {
manifest: {
name: "My Awesome Add-on",
description: "Test extension install button",
browser_specific_settings: { gecko: { id: FIRST_EXTENSION_ID } },
permissions: [""],
},
});
await testCardInstall(getCardByAddonId(win, FIRST_EXTENSION_ID));
await installExtensionPromise;
// Test theme install.
let installThemePromise = promiseAddonInstall(amoServer, {
manifest: {
name: "My Fancy Theme",
description: "Test theme install button",
browser_specific_settings: { gecko: { id: FIRST_THEME_ID } },
theme: {
colors: {
tab_selected: "red",
},
},
},
});
let promiseThemeChange = promiseObserved("lightweight-theme-styling-update");
await testCardInstall(getCardByAddonId(win, FIRST_THEME_ID));
await installThemePromise;
await promiseThemeChange;
// After installing, the cards should have manage buttons instead of install
// buttons. The cards should still be at the top of the pane (and not be
// moved to the bottom).
Assert.deepEqual(
getVisibleActions(win.document).map(getActionName),
[
"manage-addon",
"manage-addon",
...new Array(apiResultArray.length - 2).fill("install-addon"),
"open-amo",
],
"The Install buttons should be replaced with Manage buttons"
);
// End of the testing installation from a card.
// Click on the Manage button to verify that it does something useful,
// and in order to be able to force the discovery pane to be rendered again.
let loaded = waitForViewLoad(win);
getCardByAddonId(win, FIRST_EXTENSION_ID)
.querySelector("[action='manage-addon']")
.click();
await loaded;
{
let addonCard = win.document.querySelector(
`addon-card[addon-id="${FIRST_EXTENSION_ID}"]`
);
ok(addonCard, "Add-on details should be shown");
ok(addonCard.expanded, "The card should have been expanded");
// TODO bug 1540253: Check that the "recommended" badge is visible.
}
// Now we are going to force an updated rendering and check that the cards are
// in the expected order, and then test uninstallation of the above add-ons.
await switchToDiscoView(win);
await waitForAllImagesLoaded(win);
Assert.deepEqual(
getVisibleActions(win.document).map(getActionName),
[
...new Array(apiResultArray.length - 2).fill("install-addon"),
"manage-addon",
"manage-addon",
"open-amo",
],
"Already-installed add-ons should be rendered at the end of the list"
);
promiseThemeChange = promiseObserved("lightweight-theme-styling-update");
await testAddonUninstall(getCardByAddonId(win, FIRST_THEME_ID));
await promiseThemeChange;
await testAddonUninstall(getCardByAddonId(win, FIRST_EXTENSION_ID));
is(apiHandler.requestCount, 1, "Discovery API should be fetched once");
await closeView(win);
});
// Tests that the page is able to switch views while the discopane is loading,
// without inadvertently replacing the page when the request finishes.
add_task(async function discopane_navigate_while_loading() {
let apiHandler = new DiscoveryAPIHandler(`{"results": []}`);
apiHandler.blockNextResponses();
let win = await loadInitialView("discover");
let updatePromise = promiseDiscopaneUpdate(win);
let didUpdateDiscopane = false;
updatePromise.then(() => {
didUpdateDiscopane = true;
});
// Switch views while the request is pending.
await switchToNonDiscoView(win);
is(
didUpdateDiscopane,
false,
"discopane should still not be updated because the request is blocked"
);
is(
getDiscoveryElement(win),
null,
"Discopane should be removed after switching to the extension list"
);
// Release pending requests, to verify that completing the request will not
// cause changes to the visible view. The updatePromise will still resolve
// though, because the event is dispatched to the removed ``.
apiHandler.unblockResponses();
await updatePromise;
ok(
win.document.querySelector("addon-list"),
"Should still be at the extension list view"
);
is(
getDiscoveryElement(win),
null,
"Discopane should not be in the document when it is not the active view"
);
is(apiHandler.requestCount, 1, "Discovery API should be fetched once");
await closeView(win);
});
// Tests that invalid responses are handled correctly and not cached.
// Also verifies that the response is cached as long as the page is active,
// but not when the page is fully reloaded.
add_task(async function discopane_cache_api_responses() {
const INVALID_RESPONSE_BODY = `{"This is some": invalid} JSON`;
let apiHandler = new DiscoveryAPIHandler(INVALID_RESPONSE_BODY);
let expectedErrMsg;
try {
JSON.parse(INVALID_RESPONSE_BODY);
ok(false, "JSON.parse should have thrown");
} catch (e) {
expectedErrMsg = e.message;
}
let invalidResponseHandledPromise = new Promise(resolve => {
Services.console.registerListener(function listener(msg) {
if (msg.message.includes(expectedErrMsg)) {
resolve();
Services.console.unregisterListener(listener);
}
});
});
let win = await loadInitialView("discover"); // Request #1
await promiseDiscopaneUpdate(win);
info("Waiting for expected error");
await invalidResponseHandledPromise;
is(apiHandler.requestCount, 1, "Discovery API should be fetched once");
Assert.deepEqual(
getVisibleActions(win.document).map(getActionName),
["open-amo"],
"The AMO button should be visible even when the response was invalid"
);
// Change to a valid response, so that the next response will be cached.
apiHandler.setResponseText(`{"results": []}`);
await switchToNonDiscoView(win);
await switchToDiscoView(win); // Request #2
is(
apiHandler.requestCount,
2,
"Should fetch new data because an invalid response should not be cached"
);
await switchToNonDiscoView(win);
await switchToDiscoView(win);
await closeView(win);
is(
apiHandler.requestCount,
2,
"The previous response was valid and should have been reused"
);
// Now open a new about:addons page and verify that a new API request is sent.
let anotherWin = await loadInitialView("discover");
await promiseDiscopaneUpdate(anotherWin);
await closeView(anotherWin);
is(apiHandler.requestCount, 3, "discovery API should be requested again");
});
add_task(async function discopane_no_cookies() {
let requestPromise = new Promise(resolve => {
amoServer.registerPathHandler("/discoapi", resolve);
});
Services.cookies.add(
AMO_TEST_HOST,
"/",
"name",
"value",
false,
false,
false,
Date.now() / 1000 + 600,
{},
Ci.nsICookie.SAMESITE_NONE,
Ci.nsICookie.SCHEME_HTTP
);
let win = await loadInitialView("discover");
let request = await requestPromise;
ok(!request.hasHeader("Cookie"), "discovery API should not receive cookies");
await closeView(win);
});
// The CSP of about:addons whitelists http:, but not data:, hence we are
// loading a little red data: image which gets blocked by the CSP.
add_task(async function csp_img_src() {
const RED_DATA_IMAGE =
"" +
"AHElEQVQI12P4//8/w38GIAXDIBKE0DHxgljNBAAO9TXL0Y4OHwAAAABJRU5ErkJggg==";
// Minimal API response to get the image in recommended-addon-card to render.
const DUMMY_EXTENSION_ID = "dummy-csp@extensionid";
const apiResponse = {
results: [
{
addon: {
guid: DUMMY_EXTENSION_ID,
type: "extension",
authors: [
{
name: "Some CSP author",
},
],
url: `http://${AMO_TEST_HOST}/dummy`,
icon_url: RED_DATA_IMAGE,
},
},
],
};
let apiHandler = new DiscoveryAPIHandler(JSON.stringify(apiResponse));
apiHandler.blockNextResponses();
let win = await loadInitialView("discover");
let cspPromise = new Promise(resolve => {
win.addEventListener("securitypolicyviolation", e => {
// non http(s) loads only report the scheme
is(e.blockedURI, "data", "CSP: blocked URI");
is(e.violatedDirective, "img-src", "CSP: violated directive");
resolve();
});
});
apiHandler.unblockResponses();
await cspPromise;
await closeView(win);
});
add_task(async function checkDiscopaneNotice() {
await SpecialPowers.pushPrefEnv({
set: [
["browser.discovery.enabled", true],
// Enabling the Data Upload pref may upload data.
// Point data reporting services to localhost so the data doesn't escape.
["toolkit.telemetry.server", "https://localhost:1337"],
["telemetry.fog.test.localhost_port", -1],
["datareporting.healthreport.uploadEnabled", true],
["extensions.htmlaboutaddons.recommendations.enabled", true],
["extensions.recommendations.hideNotice", false],
],
});
let win = await loadInitialView("extension");
let messageBar = win.document.querySelector("message-bar.discopane-notice");
ok(messageBar, "Recommended notice should exist in extensions view");
await switchToDiscoView(win);
messageBar = win.document.querySelector("message-bar.discopane-notice");
ok(messageBar, "Recommended notice should exist in disco view");
messageBar.closeButton.click();
messageBar = win.document.querySelector("message-bar.discopane-notice");
ok(!messageBar, "Recommended notice should not exist in disco view");
await switchToNonDiscoView(win);
messageBar = win.document.querySelector("message-bar.discopane-notice");
ok(!messageBar, "Recommended notice should not exist in extensions view");
await closeView(win);
});