Source code

Revision control

Copy as Markdown

Other Tools

Test Info: Warnings

/* Any copyright is dedicated to the Public Domain.
*/
// This verifies that add-on update checks work
// The test extension uses an insecure update url.
Services.prefs.setBoolPref(PREF_EM_CHECK_UPDATE_SECURITY, false);
// This test uses add-on versions that follow the toolkit version but we
// started to encourage the use of a simpler format in Bug 1793925. We disable
// the pref below to avoid install errors.
Services.prefs.setBoolPref(
"extensions.webextensions.warnings-as-errors",
false
);
const updateFile = "test_update.json";
const profileDir = gProfD.clone();
profileDir.append("extensions");
const ADDONS = {
test_update: {
id: "addon1@tests.mozilla.org",
version: "2.0",
name: "Test 1",
},
test_update8: {
id: "addon8@tests.mozilla.org",
version: "2.0",
name: "Test 8",
},
test_update12: {
id: "addon12@tests.mozilla.org",
version: "2.0",
name: "Test 12",
},
test_install2_1: {
id: "addon2@tests.mozilla.org",
version: "2.0",
name: "Real Test 2",
},
test_install2_2: {
id: "addon2@tests.mozilla.org",
version: "3.0",
name: "Real Test 3",
},
};
var testserver = createHttpServer({ hosts: ["example.com"] });
testserver.registerDirectory("/data/", do_get_file("data"));
const XPIS = {};
add_task(async function setup() {
createAppInfo("xpcshell@tests.mozilla.org", "XPCShell", "1", "1");
Services.locale.requestedLocales = ["fr-FR"];
for (let [name, info] of Object.entries(ADDONS)) {
XPIS[name] = createTempWebExtensionFile({
manifest: {
name: info.name,
version: info.version,
browser_specific_settings: { gecko: { id: info.id } },
},
});
testserver.registerFile(`/addons/${name}.xpi`, XPIS[name]);
}
AddonTestUtils.updateReason = AddonManager.UPDATE_WHEN_USER_REQUESTED;
await promiseStartupManager();
});
// Verify that an update is available and can be installed.
add_task(async function test_apply_update() {
await promiseInstallWebExtension({
manifest: {
name: "Test Addon 1",
version: "1.0",
browser_specific_settings: {
gecko: {
id: "addon1@tests.mozilla.org",
update_url: `http://example.com/data/${updateFile}`,
},
},
},
});
let a1 = await AddonManager.getAddonByID("addon1@tests.mozilla.org");
notEqual(a1, null);
equal(a1.version, "1.0");
equal(a1.applyBackgroundUpdates, AddonManager.AUTOUPDATE_DEFAULT);
equal(a1.releaseNotesURI, null);
notEqual(a1.syncGUID, null);
let originalSyncGUID = a1.syncGUID;
await expectEvents(
{
ignorePlugins: true,
addonEvents: {
"addon1@tests.mozilla.org": [
{
event: "onPropertyChanged",
properties: ["applyBackgroundUpdates"],
},
],
},
},
async () => {
a1.applyBackgroundUpdates = AddonManager.AUTOUPDATE_DISABLE;
}
);
a1.applyBackgroundUpdates = AddonManager.AUTOUPDATE_DISABLE;
let install;
await expectEvents(
{
ignorePlugins: true,
installEvents: [{ event: "onNewInstall" }],
},
async () => {
({ updateAvailable: install } =
await AddonTestUtils.promiseFindAddonUpdates(a1));
}
);
let installs = await AddonManager.getAllInstalls();
equal(installs.length, 1);
equal(installs[0], install);
equal(install.name, a1.name);
equal(install.version, "2.0");
equal(install.state, AddonManager.STATE_AVAILABLE);
equal(install.existingAddon, a1);
equal(install.releaseNotesURI.spec, "http://example.com/updateInfo.xhtml");
// Verify that another update check returns the same AddonInstall
let { updateAvailable: install2 } =
await AddonTestUtils.promiseFindAddonUpdates(a1);
installs = await AddonManager.getAllInstalls();
equal(installs.length, 1);
equal(installs[0], install);
equal(install2, install);
await expectEvents(
{
ignorePlugins: true,
installEvents: [
{ event: "onDownloadStarted" },
{ event: "onDownloadEnded", returnValue: false },
],
},
() => {
install.install();
}
);
equal(install.state, AddonManager.STATE_DOWNLOADED);
// Continue installing the update.
// Verify that another update check returns no new update
let { updateAvailable } = await AddonTestUtils.promiseFindAddonUpdates(
install.existingAddon
);
ok(
!updateAvailable,
"Should find no available updates when one is already downloading"
);
installs = await AddonManager.getAllInstalls();
equal(installs.length, 1);
equal(installs[0], install);
await expectEvents(
{
ignorePlugins: true,
addonEvents: {
"addon1@tests.mozilla.org": [
{ event: "onInstalling" },
{ event: "onInstalled" },
],
},
installEvents: [
{ event: "onInstallStarted" },
{ event: "onInstallEnded" },
],
},
() => {
install.install();
}
);
await AddonTestUtils.loadAddonsList(true);
// Grab the current time so we can check the mtime of the add-on below
// without worrying too much about how long other tests take.
let startupTime = Date.now();
ok(isExtensionInBootstrappedList(profileDir, "addon1@tests.mozilla.org"));
a1 = await AddonManager.getAddonByID("addon1@tests.mozilla.org");
notEqual(a1, null);
equal(a1.version, "2.0");
ok(isExtensionInBootstrappedList(profileDir, a1.id));
equal(a1.applyBackgroundUpdates, AddonManager.AUTOUPDATE_DISABLE);
equal(a1.releaseNotesURI.spec, "http://example.com/updateInfo.xhtml");
notEqual(a1.syncGUID, null);
equal(originalSyncGUID, a1.syncGUID);
// Make sure that the extension lastModifiedTime was updated.
let testFile = getAddonFile(a1);
let difference = testFile.lastModifiedTime - startupTime;
Assert.less(Math.abs(difference), MAX_TIME_DIFFERENCE);
await a1.uninstall();
});
// Check that an update check finds compatibility updates and applies them
add_task(async function test_compat_update() {
await promiseInstallWebExtension({
manifest: {
name: "Test Addon 2",
version: "1.0",
browser_specific_settings: {
gecko: {
id: "addon2@tests.mozilla.org",
update_url: "http://example.com/data/" + updateFile,
strict_max_version: "0",
},
},
},
});
let a2 = await AddonManager.getAddonByID("addon2@tests.mozilla.org");
notEqual(a2, null);
ok(a2.isActive);
ok(a2.isCompatible);
ok(!a2.appDisabled);
ok(a2.isCompatibleWith("0", "0"));
let result = await AddonTestUtils.promiseFindAddonUpdates(a2);
ok(result.compatibilityUpdate, "Should have seen a compatibility update");
ok(!result.updateAvailable, "Should not have seen a version update");
ok(a2.isCompatible);
ok(!a2.appDisabled);
ok(a2.isActive);
await promiseRestartManager();
a2 = await AddonManager.getAddonByID("addon2@tests.mozilla.org");
notEqual(a2, null);
ok(a2.isActive);
ok(a2.isCompatible);
ok(!a2.appDisabled);
await a2.uninstall();
});
// Checks that we see no compatibility information when there is none.
add_task(async function test_no_compat() {
gAppInfo.platformVersion = "5";
await promiseRestartManager("5");
await promiseInstallWebExtension({
manifest: {
name: "Test Addon 3",
browser_specific_settings: {
gecko: {
id: "addon3@tests.mozilla.org",
update_url: `http://example.com/data/${updateFile}`,
strict_min_version: "5",
},
},
},
});
gAppInfo.platformVersion = "1";
await promiseRestartManager("1");
let a3 = await AddonManager.getAddonByID("addon3@tests.mozilla.org");
notEqual(a3, null);
ok(!a3.isActive);
ok(!a3.isCompatible);
ok(a3.appDisabled);
ok(a3.isCompatibleWith("5", "5"));
ok(!a3.isCompatibleWith("2", "2"));
let result = await AddonTestUtils.promiseFindAddonUpdates(a3);
ok(
!result.compatibilityUpdate,
"Should not have seen a compatibility update"
);
ok(!result.updateAvailable, "Should not have seen a version update");
});
// Checks that compatibility info for future apps are detected but don't make
// the item compatibile.
add_task(async function test_future_compat() {
let a3 = await AddonManager.getAddonByID("addon3@tests.mozilla.org");
notEqual(a3, null);
ok(!a3.isActive);
ok(!a3.isCompatible);
ok(a3.appDisabled);
ok(a3.isCompatibleWith("5", "5"));
ok(!a3.isCompatibleWith("2", "2"));
let result = await AddonTestUtils.promiseFindAddonUpdates(
a3,
undefined,
"3.0",
"3.0"
);
ok(result.compatibilityUpdate, "Should have seen a compatibility update");
ok(!result.updateAvailable, "Should not have seen a version update");
ok(!a3.isActive);
ok(!a3.isCompatible);
ok(a3.appDisabled);
await promiseRestartManager();
a3 = await AddonManager.getAddonByID("addon3@tests.mozilla.org");
notEqual(a3, null);
ok(!a3.isActive);
ok(!a3.isCompatible);
ok(a3.appDisabled);
await a3.uninstall();
});
// Test that background update checks work
add_task(async function test_background_update() {
await promiseInstallWebExtension({
manifest: {
name: "Test Addon 1",
version: "1.0",
browser_specific_settings: {
gecko: {
id: "addon1@tests.mozilla.org",
update_url: `http://example.com/data/${updateFile}`,
strict_min_version: "1",
strict_max_version: "1",
},
},
},
});
function checkInstall(install) {
notEqual(install.existingAddon, null);
equal(install.existingAddon.id, "addon1@tests.mozilla.org");
}
await expectEvents(
{
ignorePlugins: true,
addonEvents: {
"addon1@tests.mozilla.org": [
{ event: "onInstalling" },
{ event: "onInstalled" },
],
},
installEvents: [
{ event: "onNewInstall" },
{ event: "onDownloadStarted" },
{ event: "onDownloadEnded", callback: checkInstall },
{ event: "onInstallStarted" },
{ event: "onInstallEnded" },
],
},
() => {
AddonManagerPrivate.backgroundUpdateCheck();
}
);
let a1 = await AddonManager.getAddonByID("addon1@tests.mozilla.org");
notEqual(a1, null);
equal(a1.version, "2.0");
equal(a1.releaseNotesURI.spec, "http://example.com/updateInfo.xhtml");
await a1.uninstall();
});
const STATE_BLOCKED = Ci.nsIBlocklistService.STATE_BLOCKED;
const PARAMS =
"?" +
[
"req_version=%REQ_VERSION%",
"item_id=%ITEM_ID%",
"item_version=%ITEM_VERSION%",
"item_maxappversion=%ITEM_MAXAPPVERSION%",
"item_status=%ITEM_STATUS%",
"app_id=%APP_ID%",
"app_version=%APP_VERSION%",
"current_app_version=%CURRENT_APP_VERSION%",
"app_os=%APP_OS%",
"app_abi=%APP_ABI%",
"app_locale=%APP_LOCALE%",
"update_type=%UPDATE_TYPE%",
].join("&");
const PARAM_ADDONS = {
"addon1@tests.mozilla.org": {
manifest: {
name: "Test Addon 1",
version: "5.0",
browser_specific_settings: {
gecko: {
id: "addon1@tests.mozilla.org",
strict_min_version: "1",
strict_max_version: "2",
},
},
},
params: {
item_version: "5.0",
item_maxappversion: "2",
item_status: "userEnabled",
app_version: "1",
update_type: "97",
},
updateType: [AddonManager.UPDATE_WHEN_USER_REQUESTED],
},
"addon2@tests.mozilla.org": {
manifest: {
name: "Test Addon 2",
version: "67.0.5b1",
browser_specific_settings: {
gecko: {
id: "addon2@tests.mozilla.org",
update_url: "http://example.com/data/param_test.json" + PARAMS,
strict_min_version: "0",
strict_max_version: "3",
},
},
},
initialState: {
userDisabled: true,
},
params: {
item_version: "67.0.5b1",
item_maxappversion: "3",
item_status: "userDisabled",
app_version: "1",
update_type: "49",
},
updateType: [AddonManager.UPDATE_WHEN_ADDON_INSTALLED],
compatOnly: true,
},
"addon3@tests.mozilla.org": {
manifest: {
name: "Test Addon 3",
version: "1.3+",
browser_specific_settings: {
gecko: {
id: "addon3@tests.mozilla.org",
},
},
},
params: {
item_version: "1.3+",
item_status: "userEnabled",
app_version: "1",
update_type: "112",
},
updateType: [AddonManager.UPDATE_WHEN_PERIODIC_UPDATE],
},
"addon4@tests.mozilla.org": {
manifest: {
name: "Test Addon 4",
version: "0.5ab6",
browser_specific_settings: {
gecko: {
id: "addon4@tests.mozilla.org",
strict_min_version: "1",
strict_max_version: "5",
},
},
},
params: {
item_version: "0.5ab6",
item_maxappversion: "5",
item_status: "userEnabled",
app_version: "2",
update_type: "98",
},
updateType: [AddonManager.UPDATE_WHEN_NEW_APP_DETECTED, "2"],
},
"addon5@tests.mozilla.org": {
manifest: {
name: "Test Addon 5",
version: "1.0",
browser_specific_settings: {
gecko: {
id: "addon5@tests.mozilla.org",
strict_min_version: "1",
strict_max_version: "1",
},
},
},
params: {
item_version: "1.0",
item_maxappversion: "1",
item_status: "userEnabled",
app_version: "1",
update_type: "35",
},
updateType: [AddonManager.UPDATE_WHEN_NEW_APP_INSTALLED],
compatOnly: true,
},
"addon6@tests.mozilla.org": {
manifest: {
name: "Test Addon 6",
version: "1.0",
browser_specific_settings: {
gecko: {
id: "addon6@tests.mozilla.org",
strict_min_version: "1",
strict_max_version: "1",
},
},
},
params: {
item_version: "1.0",
item_maxappversion: "1",
item_status: "userEnabled",
app_version: "1",
update_type: "99",
},
updateType: [AddonManager.UPDATE_WHEN_NEW_APP_INSTALLED],
},
"blocklist2@tests.mozilla.org": {
manifest: {
name: "Test Addon 1",
version: "5.0",
browser_specific_settings: {
gecko: {
id: "blocklist2@tests.mozilla.org",
strict_min_version: "1",
strict_max_version: "2",
},
},
},
params: {
item_version: "5.0",
item_maxappversion: "2",
item_status: "userEnabled,blocklisted",
app_version: "1",
update_type: "97",
},
updateType: [AddonManager.UPDATE_WHEN_USER_REQUESTED],
blocklistState: STATE_BLOCKED,
},
};
const PARAM_IDS = Object.keys(PARAM_ADDONS);
// Verify the parameter escaping in update urls.
add_task(async function test_params() {
let blocked = [];
for (let [id, options] of Object.entries(PARAM_ADDONS)) {
if (options.blocklistState == STATE_BLOCKED) {
blocked.push(`${id}:${options.manifest.version}`);
}
}
let extensionsMLBF = [{ stash: { blocked, unblocked: [] }, stash_time: 0 }];
await AddonTestUtils.loadBlocklistRawData({ extensionsMLBF });
for (let [id, options] of Object.entries(PARAM_ADDONS)) {
await promiseInstallWebExtension({ manifest: options.manifest });
if (options.initialState) {
let addon = await AddonManager.getAddonByID(id);
await setInitialState(addon, options.initialState);
}
}
let resultsPromise = new Promise(resolve => {
let results = new Map();
testserver.registerPathHandler("/data/param_test.json", function (request) {
let params = new URLSearchParams(request.queryString);
let itemId = params.get("item_id");
ok(
!results.has(itemId),
`Should not see a duplicate request for item ${itemId}`
);
results.set(itemId, params);
if (results.size === PARAM_IDS.length) {
resolve(results);
}
request.setStatusLine(null, 500, "Server Error");
});
});
let addons = await getAddons(PARAM_IDS);
for (let [id, options] of Object.entries(PARAM_ADDONS)) {
// Having an onUpdateAvailable callback in the listener automagically adds
// UPDATE_TYPE_NEWVERSION to the update type flags in the request.
let listener = options.compatOnly ? {} : { onUpdateAvailable() {} };
addons.get(id).findUpdates(listener, ...options.updateType);
}
let baseParams = {
req_version: "2",
app_id: "xpcshell@tests.mozilla.org",
current_app_version: "1",
app_os: "XPCShell",
app_abi: "noarch-spidermonkey",
app_locale: "fr-FR",
};
let results = await resultsPromise;
for (let [id, options] of Object.entries(PARAM_ADDONS)) {
info(`Checking update params for ${id}`);
let expected = Object.assign({}, baseParams, options.params);
let params = results.get(id);
for (let [prop, value] of Object.entries(expected)) {
equal(params.get(prop), value, `Expected value for ${prop}`);
}
}
for (let [, addon] of await getAddons(PARAM_IDS)) {
await addon.uninstall();
}
});
// Tests that if a manifest claims compatibility then the add-on will be
// seen as compatible regardless of what the update payload says.
add_task(async function test_manifest_compat() {
await promiseInstallWebExtension({
manifest: {
name: "Test Addon 1",
version: "5.0",
browser_specific_settings: {
gecko: {
id: "addon4@tests.mozilla.org",
update_url: `http://example.com/data/${updateFile}`,
strict_min_version: "0",
strict_max_version: "1",
},
},
},
});
let a4 = await AddonManager.getAddonByID("addon4@tests.mozilla.org");
ok(a4.isActive, "addon4 is active");
ok(a4.isCompatible, "addon4 is compatible");
// Test that a normal update check won't decrease a targetApplication's
// maxVersion but an update check for a new application will.
await AddonTestUtils.promiseFindAddonUpdates(
a4,
AddonManager.UPDATE_WHEN_PERIODIC_UPDATE
);
ok(a4.isActive, "addon4 is active");
ok(a4.isCompatible, "addon4 is compatible");
await AddonTestUtils.promiseFindAddonUpdates(
a4,
AddonManager.UPDATE_WHEN_NEW_APP_INSTALLED
);
ok(!a4.isActive, "addon4 is not active");
ok(!a4.isCompatible, "addon4 is not compatible");
await a4.uninstall();
});
// Test that the background update check doesn't update an add-on that isn't
// allowed to update automatically.
add_task(async function test_no_auto_update() {
// Have an add-on there that will be updated so we see some events from it
await promiseInstallWebExtension({
manifest: {
name: "Test Addon 1",
version: "1.0",
browser_specific_settings: {
gecko: {
id: "addon1@tests.mozilla.org",
update_url: `http://example.com/data/${updateFile}`,
},
},
},
});
await promiseInstallWebExtension({
manifest: {
name: "Test Addon 8",
version: "1.0",
browser_specific_settings: {
gecko: {
id: "addon8@tests.mozilla.org",
update_url: `http://example.com/data/${updateFile}`,
},
},
},
});
let a8 = await AddonManager.getAddonByID("addon8@tests.mozilla.org");
a8.applyBackgroundUpdates = AddonManager.AUTOUPDATE_DISABLE;
// The background update check will find updates for both add-ons but only
// proceed to install one of them.
let listener;
await new Promise(resolve => {
listener = {
onNewInstall(aInstall) {
let id = aInstall.existingAddon.id;
ok(
id == "addon1@tests.mozilla.org" || id == "addon8@tests.mozilla.org",
"Saw unexpected onNewInstall for " + id
);
},
onDownloadStarted(aInstall) {
equal(aInstall.existingAddon.id, "addon1@tests.mozilla.org");
},
onDownloadEnded(aInstall) {
equal(aInstall.existingAddon.id, "addon1@tests.mozilla.org");
},
onDownloadFailed() {
ok(false, "Should not have seen onDownloadFailed event");
},
onDownloadCancelled() {
ok(false, "Should not have seen onDownloadCancelled event");
},
onInstallStarted(aInstall) {
equal(aInstall.existingAddon.id, "addon1@tests.mozilla.org");
},
onInstallEnded(aInstall) {
equal(aInstall.existingAddon.id, "addon1@tests.mozilla.org");
resolve();
},
onInstallFailed() {
ok(false, "Should not have seen onInstallFailed event");
},
onInstallCancelled() {
ok(false, "Should not have seen onInstallCancelled event");
},
};
AddonManager.addInstallListener(listener);
AddonManagerPrivate.backgroundUpdateCheck();
});
AddonManager.removeInstallListener(listener);
let a1;
[a1, a8] = await AddonManager.getAddonsByIDs([
"addon1@tests.mozilla.org",
"addon8@tests.mozilla.org",
]);
notEqual(a1, null);
equal(a1.version, "2.0");
await a1.uninstall();
notEqual(a8, null);
equal(a8.version, "1.0");
await a8.uninstall();
});
// Test that the update check returns nothing for addons in locked install
// locations.
add_task(async function run_test_locked_install() {
const lockedDir = gProfD.clone();
lockedDir.append("locked_extensions");
registerDirectory("XREAppFeat", lockedDir);
await promiseShutdownManager();
let xpi = await createTempWebExtensionFile({
manifest: {
name: "Test Addon 13",
version: "1.0",
browser_specific_settings: {
gecko: {
id: "addon13@tests.mozilla.org",
},
},
},
});
xpi.copyTo(lockedDir, "addon13@tests.mozilla.org.xpi");
let validAddons = { system: ["addon13@tests.mozilla.org"] };
await overrideBuiltIns(validAddons);
await promiseStartupManager();
let a13 = await AddonManager.getAddonByID("addon13@tests.mozilla.org");
notEqual(a13, null);
let result = await AddonTestUtils.promiseFindAddonUpdates(a13);
ok(
!result.compatibilityUpdate,
"Should not have seen a compatibility update"
);
ok(!result.updateAvailable, "Should not have seen a version update");
let installs = await AddonManager.getAllInstalls();
equal(installs.length, 0);
});