forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			1051 lines
		
	
	
	
		
			38 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			1051 lines
		
	
	
	
		
			38 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
/* Any copyright is dedicated to the Public Domain.
 | 
						|
 * http://creativecommons.org/publicdomain/zero/1.0/
 | 
						|
 */
 | 
						|
 | 
						|
const TESTROOT = "http://example.com/browser/toolkit/mozapps/extensions/test/xpinstall/";
 | 
						|
const TESTROOT2 = "http://example.org/browser/toolkit/mozapps/extensions/test/xpinstall/";
 | 
						|
const SECUREROOT = "https://example.com/browser/toolkit/mozapps/extensions/test/xpinstall/";
 | 
						|
const XPINSTALL_URL = "chrome://mozapps/content/xpinstall/xpinstallConfirm.xul";
 | 
						|
const PREF_INSTALL_REQUIREBUILTINCERTS = "extensions.install.requireBuiltInCerts";
 | 
						|
const PROGRESS_NOTIFICATION = "addon-progress";
 | 
						|
 | 
						|
ChromeUtils.import("resource://gre/modules/AppConstants.jsm");
 | 
						|
 | 
						|
var rootDir = getRootDirectory(gTestPath);
 | 
						|
var rootPath = rootDir.split("/");
 | 
						|
var chromeName = rootPath[0] + "//" + rootPath[2];
 | 
						|
var croot = chromeName + "/content/browser/toolkit/mozapps/extensions/test/xpinstall/";
 | 
						|
var jar = getJar(croot);
 | 
						|
if (jar) {
 | 
						|
  var tmpdir = extractJarToTmp(jar);
 | 
						|
  croot = "file://" + tmpdir.path + "/";
 | 
						|
}
 | 
						|
const CHROMEROOT = croot;
 | 
						|
 | 
						|
var gApp = document.getElementById("bundle_brand").getString("brandShortName");
 | 
						|
var gVersion = Services.appinfo.version;
 | 
						|
 | 
						|
function waitForTick() {
 | 
						|
  return new Promise(resolve => executeSoon(resolve));
 | 
						|
}
 | 
						|
 | 
						|
function getObserverTopic(aNotificationId) {
 | 
						|
  let topic = aNotificationId;
 | 
						|
  if (topic == "xpinstall-disabled")
 | 
						|
    topic = "addon-install-disabled";
 | 
						|
  else if (topic == "addon-progress")
 | 
						|
    topic = "addon-install-started";
 | 
						|
  else if (topic == "addon-install-restart")
 | 
						|
    topic = "addon-install-complete";
 | 
						|
  else if (topic == "addon-installed")
 | 
						|
    topic = "webextension-install-notify";
 | 
						|
  return topic;
 | 
						|
}
 | 
						|
 | 
						|
async function waitForProgressNotification(aPanelOpen = false, aExpectedCount = 1) {
 | 
						|
  let notificationId = PROGRESS_NOTIFICATION;
 | 
						|
  info("Waiting for " + notificationId + " notification");
 | 
						|
 | 
						|
  let topic = getObserverTopic(notificationId);
 | 
						|
 | 
						|
  let observerPromise = new Promise(resolve => {
 | 
						|
    Services.obs.addObserver(function observer(aSubject, aTopic, aData) {
 | 
						|
      // Ignore the progress notification unless that is the notification we want
 | 
						|
      if (notificationId != PROGRESS_NOTIFICATION &&
 | 
						|
          aTopic == getObserverTopic(PROGRESS_NOTIFICATION)) {
 | 
						|
            return;
 | 
						|
      }
 | 
						|
      Services.obs.removeObserver(observer, topic);
 | 
						|
      resolve();
 | 
						|
    }, topic);
 | 
						|
  });
 | 
						|
 | 
						|
  let panelEventPromise;
 | 
						|
  if (aPanelOpen) {
 | 
						|
    panelEventPromise = Promise.resolve();
 | 
						|
  } else {
 | 
						|
    panelEventPromise = new Promise(resolve => {
 | 
						|
      PopupNotifications.panel.addEventListener("popupshowing", function() {
 | 
						|
        resolve();
 | 
						|
      }, {once: true});
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
  await observerPromise;
 | 
						|
  await panelEventPromise;
 | 
						|
  await waitForTick();
 | 
						|
 | 
						|
  info("Saw a notification");
 | 
						|
  ok(PopupNotifications.isPanelOpen, "Panel should be open");
 | 
						|
  is(PopupNotifications.panel.childNodes.length, aExpectedCount, "Should be the right number of notifications");
 | 
						|
  if (PopupNotifications.panel.childNodes.length) {
 | 
						|
    let nodes = Array.from(PopupNotifications.panel.childNodes);
 | 
						|
    let notification = nodes.find(n => n.id == notificationId + "-notification");
 | 
						|
    ok(notification, `Should have seen the right notification`);
 | 
						|
    ok(notification.button.hasAttribute("disabled"),
 | 
						|
       "The install button should be disabled");
 | 
						|
  }
 | 
						|
 | 
						|
  return PopupNotifications.panel;
 | 
						|
}
 | 
						|
 | 
						|
async function waitForNotification(aId, aExpectedCount = 1) {
 | 
						|
  info("Waiting for " + aId + " notification");
 | 
						|
 | 
						|
  let topic = getObserverTopic(aId);
 | 
						|
 | 
						|
  let observerPromise = new Promise(resolve => {
 | 
						|
    Services.obs.addObserver(function observer(aSubject, aTopic, aData) {
 | 
						|
      // Ignore the progress notification unless that is the notification we want
 | 
						|
      if (aId != PROGRESS_NOTIFICATION &&
 | 
						|
          aTopic == getObserverTopic(PROGRESS_NOTIFICATION)) {
 | 
						|
            return;
 | 
						|
      }
 | 
						|
      Services.obs.removeObserver(observer, topic);
 | 
						|
      resolve();
 | 
						|
    }, topic);
 | 
						|
  });
 | 
						|
 | 
						|
  let panelEventPromise = new Promise(resolve => {
 | 
						|
    PopupNotifications.panel.addEventListener("PanelUpdated", function eventListener(e) {
 | 
						|
      // Skip notifications that are not the one that we are supposed to be looking for
 | 
						|
      if (!e.detail.includes(aId)) {
 | 
						|
        return;
 | 
						|
      }
 | 
						|
      PopupNotifications.panel.removeEventListener("PanelUpdated", eventListener);
 | 
						|
      resolve();
 | 
						|
    });
 | 
						|
  });
 | 
						|
 | 
						|
  await observerPromise;
 | 
						|
  await panelEventPromise;
 | 
						|
  await waitForTick();
 | 
						|
 | 
						|
  info("Saw a " + aId + " notification");
 | 
						|
  ok(PopupNotifications.isPanelOpen, "Panel should be open");
 | 
						|
  is(PopupNotifications.panel.childNodes.length, aExpectedCount, "Should be the right number of notifications");
 | 
						|
  if (PopupNotifications.panel.childNodes.length) {
 | 
						|
    let nodes = Array.from(PopupNotifications.panel.childNodes);
 | 
						|
    let notification = nodes.find(n => n.id == aId + "-notification");
 | 
						|
    ok(notification, "Should have seen the " + aId + " notification");
 | 
						|
  }
 | 
						|
 | 
						|
  return PopupNotifications.panel;
 | 
						|
}
 | 
						|
 | 
						|
function waitForNotificationClose() {
 | 
						|
  return new Promise(resolve => {
 | 
						|
    info("Waiting for notification to close");
 | 
						|
    PopupNotifications.panel.addEventListener("popuphidden", function() {
 | 
						|
      resolve();
 | 
						|
    }, {once: true});
 | 
						|
  });
 | 
						|
}
 | 
						|
 | 
						|
async function waitForInstallDialog() {
 | 
						|
  if (Services.prefs.getBoolPref("xpinstall.customConfirmationUI", false)) {
 | 
						|
    let panel = await waitForNotification("addon-install-confirmation");
 | 
						|
    return panel.childNodes[0];
 | 
						|
  }
 | 
						|
 | 
						|
  info("Waiting for install dialog");
 | 
						|
 | 
						|
  let window = await new Promise(resolve => {
 | 
						|
    Services.wm.addListener({
 | 
						|
      onOpenWindow(aXULWindow) {
 | 
						|
        Services.wm.removeListener(this);
 | 
						|
        resolve(aXULWindow);
 | 
						|
      },
 | 
						|
      onCloseWindow(aXULWindow) {
 | 
						|
      },
 | 
						|
    });
 | 
						|
  });
 | 
						|
  info("Install dialog opened, waiting for focus");
 | 
						|
 | 
						|
  let domwindow = window.QueryInterface(Ci.nsIInterfaceRequestor)
 | 
						|
                        .getInterface(Ci.nsIDOMWindow);
 | 
						|
  await new Promise(resolve => {
 | 
						|
    waitForFocus(function() {
 | 
						|
      resolve();
 | 
						|
    }, domwindow);
 | 
						|
  });
 | 
						|
  info("Saw install dialog");
 | 
						|
  is(domwindow.document.location.href, XPINSTALL_URL, "Should have seen the right window open");
 | 
						|
 | 
						|
  // Override the countdown timer on the accept button
 | 
						|
  let button = domwindow.document.documentElement.getButton("accept");
 | 
						|
  button.disabled = false;
 | 
						|
 | 
						|
  return null;
 | 
						|
}
 | 
						|
 | 
						|
function removeTabAndWaitForNotificationClose() {
 | 
						|
  let closePromise = waitForNotificationClose();
 | 
						|
  BrowserTestUtils.removeTab(gBrowser.selectedTab);
 | 
						|
  return closePromise;
 | 
						|
}
 | 
						|
 | 
						|
function acceptInstallDialog(installDialog) {
 | 
						|
  if (Services.prefs.getBoolPref("xpinstall.customConfirmationUI", false)) {
 | 
						|
    installDialog.button.click();
 | 
						|
  } else {
 | 
						|
    let win = Services.wm.getMostRecentWindow("Addons:Install");
 | 
						|
    win.document.documentElement.acceptDialog();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
function cancelInstallDialog(installDialog) {
 | 
						|
  if (Services.prefs.getBoolPref("xpinstall.customConfirmationUI", false)) {
 | 
						|
    installDialog.secondaryButton.click();
 | 
						|
  } else {
 | 
						|
    let win = Services.wm.getMostRecentWindow("Addons:Install");
 | 
						|
    win.document.documentElement.cancelDialog();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
async function waitForSingleNotification(aCallback) {
 | 
						|
  while (PopupNotifications.panel.childNodes.length != 1) {
 | 
						|
    await new Promise(resolve => executeSoon(resolve));
 | 
						|
 | 
						|
    info("Waiting for single notification");
 | 
						|
    // Notification should never close while we wait
 | 
						|
    ok(PopupNotifications.isPanelOpen, "Notification should still be open");
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
function setupRedirect(aSettings) {
 | 
						|
  var url = "https://example.com/browser/toolkit/mozapps/extensions/test/xpinstall/redirect.sjs?mode=setup";
 | 
						|
  for (var name in aSettings) {
 | 
						|
    url += "&" + name + "=" + aSettings[name];
 | 
						|
  }
 | 
						|
 | 
						|
  var req = new XMLHttpRequest();
 | 
						|
  req.open("GET", url, false);
 | 
						|
  req.send(null);
 | 
						|
}
 | 
						|
 | 
						|
function getInstalls() {
 | 
						|
  return new Promise(resolve => {
 | 
						|
    AddonManager.getAllInstalls(installs => resolve(installs));
 | 
						|
  });
 | 
						|
}
 | 
						|
 | 
						|
var TESTS = [
 | 
						|
async function test_disabledInstall() {
 | 
						|
  Services.prefs.setBoolPref("xpinstall.enabled", false);
 | 
						|
 | 
						|
  let notificationPromise = waitForNotification("xpinstall-disabled");
 | 
						|
  let triggers = encodeURIComponent(JSON.stringify({
 | 
						|
    "XPI": "amosigned.xpi"
 | 
						|
  }));
 | 
						|
  BrowserTestUtils.openNewForegroundTab(gBrowser, TESTROOT + "installtrigger.html?" + triggers);
 | 
						|
  let panel = await notificationPromise;
 | 
						|
 | 
						|
  let notification = panel.childNodes[0];
 | 
						|
  is(notification.button.label, "Enable", "Should have seen the right button");
 | 
						|
  is(notification.getAttribute("label"),
 | 
						|
     "Software installation is currently disabled. Click Enable and try again.");
 | 
						|
 | 
						|
  let closePromise = waitForNotificationClose();
 | 
						|
  // Click on Enable
 | 
						|
  EventUtils.synthesizeMouseAtCenter(notification.button, {});
 | 
						|
  await closePromise;
 | 
						|
 | 
						|
  try {
 | 
						|
    ok(Services.prefs.getBoolPref("xpinstall.enabled"), "Installation should be enabled");
 | 
						|
  } catch (e) {
 | 
						|
    ok(false, "xpinstall.enabled should be set");
 | 
						|
  }
 | 
						|
 | 
						|
  BrowserTestUtils.removeTab(gBrowser.selectedTab);
 | 
						|
  let installs = await getInstalls();
 | 
						|
  is(installs.length, 0, "Shouldn't be any pending installs");
 | 
						|
},
 | 
						|
 | 
						|
async function test_blockedInstall() {
 | 
						|
  let notificationPromise = waitForNotification("addon-install-blocked");
 | 
						|
  let triggers = encodeURIComponent(JSON.stringify({
 | 
						|
    "XPI": "amosigned.xpi"
 | 
						|
  }));
 | 
						|
  BrowserTestUtils.openNewForegroundTab(gBrowser, TESTROOT + "installtrigger.html?" + triggers);
 | 
						|
  let panel = await notificationPromise;
 | 
						|
 | 
						|
  let notification = panel.childNodes[0];
 | 
						|
  is(notification.button.label, "Allow", "Should have seen the right button");
 | 
						|
  is(notification.getAttribute("origin"), "example.com",
 | 
						|
     "Should have seen the right origin host");
 | 
						|
  is(notification.getAttribute("label"),
 | 
						|
     gApp + " prevented this site from asking you to install software on your computer.",
 | 
						|
     "Should have seen the right message");
 | 
						|
 | 
						|
  let dialogPromise = waitForInstallDialog();
 | 
						|
  // Click on Allow
 | 
						|
  EventUtils.synthesizeMouse(notification.button, 20, 10, {});
 | 
						|
  // Notification should have changed to progress notification
 | 
						|
  ok(PopupNotifications.isPanelOpen, "Notification should still be open");
 | 
						|
  notification = panel.childNodes[0];
 | 
						|
  is(notification.id, "addon-progress-notification", "Should have seen the progress notification");
 | 
						|
  let installDialog = await dialogPromise;
 | 
						|
 | 
						|
  notificationPromise = waitForNotification("addon-install-restart");
 | 
						|
  acceptInstallDialog(installDialog);
 | 
						|
  panel = await notificationPromise;
 | 
						|
 | 
						|
  notification = panel.childNodes[0];
 | 
						|
  is(notification.button.label, "Restart Now", "Should have seen the right button");
 | 
						|
  is(notification.getAttribute("label"),
 | 
						|
     "XPI Test will be installed after you restart " + gApp + ".",
 | 
						|
     "Should have seen the right message");
 | 
						|
 | 
						|
  let installs = await getInstalls();
 | 
						|
  is(installs.length, 1, "Should be one pending install");
 | 
						|
  installs[0].cancel();
 | 
						|
  await removeTabAndWaitForNotificationClose();
 | 
						|
},
 | 
						|
 | 
						|
async function test_whitelistedInstall() {
 | 
						|
  let originalTab = gBrowser.selectedTab;
 | 
						|
  let tab;
 | 
						|
  gBrowser.selectedTab = originalTab;
 | 
						|
  let pm = Services.perms;
 | 
						|
  pm.add(makeURI("http://example.com/"), "install", pm.ALLOW_ACTION);
 | 
						|
 | 
						|
  let progressPromise = waitForProgressNotification();
 | 
						|
  let dialogPromise = waitForInstallDialog();
 | 
						|
  let triggers = encodeURIComponent(JSON.stringify({
 | 
						|
    "XPI": "amosigned.xpi"
 | 
						|
  }));
 | 
						|
  BrowserTestUtils.openNewForegroundTab(gBrowser, TESTROOT + "installtrigger.html?"
 | 
						|
                                                + triggers).then(newTab => tab = newTab);
 | 
						|
  await progressPromise;
 | 
						|
  let installDialog = await dialogPromise;
 | 
						|
  await BrowserTestUtils.waitForCondition(() => !!tab, "tab should be present");
 | 
						|
 | 
						|
  is(gBrowser.selectedTab, tab,
 | 
						|
     "tab selected in response to the addon-install-confirmation notification");
 | 
						|
 | 
						|
  let notificationPromise = waitForNotification("addon-install-restart");
 | 
						|
  acceptInstallDialog(installDialog);
 | 
						|
  let panel = await notificationPromise;
 | 
						|
 | 
						|
  let notification = panel.childNodes[0];
 | 
						|
  is(notification.button.label, "Restart Now", "Should have seen the right button");
 | 
						|
  is(notification.getAttribute("label"),
 | 
						|
     "XPI Test will be installed after you restart " + gApp + ".",
 | 
						|
     "Should have seen the right message");
 | 
						|
 | 
						|
  let installs = await getInstalls();
 | 
						|
  is(installs.length, 1, "Should be one pending install");
 | 
						|
  installs[0].cancel();
 | 
						|
 | 
						|
  Services.perms.remove(makeURI("http://example.com/"), "install");
 | 
						|
  await removeTabAndWaitForNotificationClose();
 | 
						|
},
 | 
						|
 | 
						|
async function test_failedDownload() {
 | 
						|
  let pm = Services.perms;
 | 
						|
  pm.add(makeURI("http://example.com/"), "install", pm.ALLOW_ACTION);
 | 
						|
 | 
						|
  let progressPromise = waitForProgressNotification();
 | 
						|
  let failPromise = waitForNotification("addon-install-failed");
 | 
						|
  let triggers = encodeURIComponent(JSON.stringify({
 | 
						|
    "XPI": "missing.xpi"
 | 
						|
  }));
 | 
						|
  BrowserTestUtils.openNewForegroundTab(gBrowser, TESTROOT + "installtrigger.html?" + triggers);
 | 
						|
  await progressPromise;
 | 
						|
  let panel = await failPromise;
 | 
						|
 | 
						|
  let notification = panel.childNodes[0];
 | 
						|
  is(notification.getAttribute("label"),
 | 
						|
     "The add-on could not be downloaded because of a connection failure.",
 | 
						|
     "Should have seen the right message");
 | 
						|
 | 
						|
  Services.perms.remove(makeURI("http://example.com/"), "install");
 | 
						|
  await removeTabAndWaitForNotificationClose();
 | 
						|
},
 | 
						|
 | 
						|
async function test_corruptFile() {
 | 
						|
  let pm = Services.perms;
 | 
						|
  pm.add(makeURI("http://example.com/"), "install", pm.ALLOW_ACTION);
 | 
						|
 | 
						|
  let progressPromise = waitForProgressNotification();
 | 
						|
  let failPromise = waitForNotification("addon-install-failed");
 | 
						|
  let triggers = encodeURIComponent(JSON.stringify({
 | 
						|
    "XPI": "corrupt.xpi"
 | 
						|
  }));
 | 
						|
  BrowserTestUtils.openNewForegroundTab(gBrowser, TESTROOT + "installtrigger.html?" + triggers);
 | 
						|
  await progressPromise;
 | 
						|
  let panel = await failPromise;
 | 
						|
 | 
						|
  let notification = panel.childNodes[0];
 | 
						|
  is(notification.getAttribute("label"),
 | 
						|
     "The add-on downloaded from this site could not be installed " +
 | 
						|
     "because it appears to be corrupt.",
 | 
						|
     "Should have seen the right message");
 | 
						|
 | 
						|
  Services.perms.remove(makeURI("http://example.com/"), "install");
 | 
						|
  await removeTabAndWaitForNotificationClose();
 | 
						|
},
 | 
						|
 | 
						|
async function test_incompatible() {
 | 
						|
  let pm = Services.perms;
 | 
						|
  pm.add(makeURI("http://example.com/"), "install", pm.ALLOW_ACTION);
 | 
						|
 | 
						|
  let progressPromise = waitForProgressNotification();
 | 
						|
  let failPromise = waitForNotification("addon-install-failed");
 | 
						|
  let triggers = encodeURIComponent(JSON.stringify({
 | 
						|
    "XPI": "incompatible.xpi"
 | 
						|
  }));
 | 
						|
  BrowserTestUtils.openNewForegroundTab(gBrowser, TESTROOT + "installtrigger.html?" + triggers);
 | 
						|
  await progressPromise;
 | 
						|
  let panel = await failPromise;
 | 
						|
 | 
						|
  let notification = panel.childNodes[0];
 | 
						|
  is(notification.getAttribute("label"),
 | 
						|
     "XPI Test could not be installed because it is not compatible with " +
 | 
						|
     gApp + " " + gVersion + ".",
 | 
						|
     "Should have seen the right message");
 | 
						|
 | 
						|
  Services.perms.remove(makeURI("http://example.com/"), "install");
 | 
						|
  await removeTabAndWaitForNotificationClose();
 | 
						|
},
 | 
						|
 | 
						|
async function test_restartless() {
 | 
						|
  let pm = Services.perms;
 | 
						|
  pm.add(makeURI("http://example.com/"), "install", pm.ALLOW_ACTION);
 | 
						|
 | 
						|
  let progressPromise = waitForProgressNotification();
 | 
						|
  let dialogPromise = waitForInstallDialog();
 | 
						|
  let triggers = encodeURIComponent(JSON.stringify({
 | 
						|
    "XPI": "restartless.xpi"
 | 
						|
  }));
 | 
						|
  gBrowser.selectedTab = BrowserTestUtils.addTab(gBrowser);
 | 
						|
  gBrowser.loadURI(TESTROOT + "installtrigger.html?" + triggers);
 | 
						|
  await progressPromise;
 | 
						|
  let installDialog = await dialogPromise;
 | 
						|
 | 
						|
  let notificationPromise = waitForNotification("addon-installed");
 | 
						|
  acceptInstallDialog(installDialog);
 | 
						|
  await notificationPromise;
 | 
						|
 | 
						|
  let installs = await getInstalls();
 | 
						|
  is(installs.length, 0, "Should be no pending installs");
 | 
						|
 | 
						|
  let addon = await new Promise(resolve => {
 | 
						|
    AddonManager.getAddonByID("restartless-xpi@tests.mozilla.org", result => {
 | 
						|
      resolve(result);
 | 
						|
    });
 | 
						|
  });
 | 
						|
  addon.uninstall();
 | 
						|
 | 
						|
  Services.perms.remove(makeURI("http://example.com/"), "install");
 | 
						|
  await removeTabAndWaitForNotificationClose(gBrowser.selectedTab);
 | 
						|
},
 | 
						|
 | 
						|
async function test_sequential() {
 | 
						|
  // This test is only relevant if using the new doorhanger UI
 | 
						|
  // TODO: this subtest is disabled until multiple notification prompts are
 | 
						|
  // reworked in bug 1188152
 | 
						|
  if (true || !Services.prefs.getBoolPref("xpinstall.customConfirmationUI", false)) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  let pm = Services.perms;
 | 
						|
  pm.add(makeURI("http://example.com/"), "install", pm.ALLOW_ACTION);
 | 
						|
 | 
						|
  let progressPromise = waitForProgressNotification();
 | 
						|
  let dialogPromise = waitForInstallDialog();
 | 
						|
  let triggers = encodeURIComponent(JSON.stringify({
 | 
						|
    "Restartless XPI": "restartless.xpi"
 | 
						|
  }));
 | 
						|
  BrowserTestUtils.openNewForegroundTab(gBrowser, TESTROOT + "installtrigger.html?" + triggers);
 | 
						|
  await progressPromise;
 | 
						|
  let installDialog = await dialogPromise;
 | 
						|
 | 
						|
  // Should see the right add-on
 | 
						|
  let container = document.getElementById("addon-install-confirmation-content");
 | 
						|
  is(container.childNodes.length, 1, "Should be one item listed");
 | 
						|
  is(container.childNodes[0].firstChild.getAttribute("value"), "XPI Test", "Should have the right add-on");
 | 
						|
 | 
						|
  progressPromise = waitForProgressNotification(true, 2);
 | 
						|
  triggers = encodeURIComponent(JSON.stringify({
 | 
						|
    "Theme XPI": "theme.xpi"
 | 
						|
  }));
 | 
						|
  gBrowser.loadURI(TESTROOT + "installtrigger.html?" + triggers);
 | 
						|
  await progressPromise;
 | 
						|
 | 
						|
  // Should still have the right add-on in the confirmation notification
 | 
						|
  is(container.childNodes.length, 1, "Should be one item listed");
 | 
						|
  is(container.childNodes[0].firstChild.getAttribute("value"), "XPI Test", "Should have the right add-on");
 | 
						|
 | 
						|
  // Wait for the install to complete, we won't see a new confirmation
 | 
						|
  // notification
 | 
						|
  await new Promise(resolve => {
 | 
						|
    Services.obs.addObserver(function observer() {
 | 
						|
      Services.obs.removeObserver(observer, "addon-install-confirmation");
 | 
						|
      resolve();
 | 
						|
    }, "addon-install-confirmation");
 | 
						|
  });
 | 
						|
 | 
						|
  // Make sure browser-addons.js executes first
 | 
						|
  await new Promise(resolve => executeSoon(resolve));
 | 
						|
 | 
						|
  // Should have dropped the progress notification
 | 
						|
  is(PopupNotifications.panel.childNodes.length, 1, "Should be the right number of notifications");
 | 
						|
  is(PopupNotifications.panel.childNodes[0].id, "addon-install-confirmation-notification",
 | 
						|
     "Should only be showing one install confirmation");
 | 
						|
 | 
						|
  // Should still have the right add-on in the confirmation notification
 | 
						|
  is(container.childNodes.length, 1, "Should be one item listed");
 | 
						|
  is(container.childNodes[0].firstChild.getAttribute("value"), "XPI Test", "Should have the right add-on");
 | 
						|
 | 
						|
  cancelInstallDialog(installDialog);
 | 
						|
 | 
						|
  ok(PopupNotifications.isPanelOpen, "Panel should still be open");
 | 
						|
  is(PopupNotifications.panel.childNodes.length, 1, "Should be the right number of notifications");
 | 
						|
  is(PopupNotifications.panel.childNodes[0].id, "addon-install-confirmation-notification",
 | 
						|
     "Should still have an install confirmation open");
 | 
						|
 | 
						|
  // Should have the next add-on's confirmation dialog
 | 
						|
  is(container.childNodes.length, 1, "Should be one item listed");
 | 
						|
  is(container.childNodes[0].firstChild.getAttribute("value"), "Theme Test", "Should have the right add-on");
 | 
						|
 | 
						|
  Services.perms.remove(makeURI("http://example.com"), "install");
 | 
						|
  let closePromise = waitForNotificationClose();
 | 
						|
  cancelInstallDialog(installDialog);
 | 
						|
  await closePromise;
 | 
						|
 | 
						|
  BrowserTestUtils.removeTab(gBrowser.selectedTab);
 | 
						|
},
 | 
						|
 | 
						|
async function test_allUnverified() {
 | 
						|
  // This test is only relevant if using the new doorhanger UI and allowing
 | 
						|
  // unsigned add-ons
 | 
						|
  if (!Services.prefs.getBoolPref("xpinstall.customConfirmationUI", false) ||
 | 
						|
      Services.prefs.getBoolPref("xpinstall.signatures.required", true) ||
 | 
						|
      AppConstants.MOZ_REQUIRE_SIGNING) {
 | 
						|
        return;
 | 
						|
  }
 | 
						|
  let pm = Services.perms;
 | 
						|
  pm.add(makeURI("http://example.com/"), "install", pm.ALLOW_ACTION);
 | 
						|
 | 
						|
  let progressPromise = waitForProgressNotification();
 | 
						|
  let dialogPromise = waitForInstallDialog();
 | 
						|
  let triggers = encodeURIComponent(JSON.stringify({
 | 
						|
    "Extension XPI": "restartless-unsigned.xpi"
 | 
						|
  }));
 | 
						|
  BrowserTestUtils.openNewForegroundTab(gBrowser, TESTROOT + "installtrigger.html?" + triggers);
 | 
						|
  await progressPromise;
 | 
						|
  let installDialog = await dialogPromise;
 | 
						|
 | 
						|
  let notification = document.getElementById("addon-install-confirmation-notification");
 | 
						|
  let message = notification.getAttribute("label");
 | 
						|
  is(message, "Caution: This site would like to install an unverified add-on in " + gApp + ". Proceed at your own risk.");
 | 
						|
 | 
						|
  let container = document.getElementById("addon-install-confirmation-content");
 | 
						|
  is(container.childNodes.length, 1, "Should be one item listed");
 | 
						|
  is(container.childNodes[0].firstChild.getAttribute("value"), "XPI Test", "Should have the right add-on");
 | 
						|
  is(container.childNodes[0].childNodes.length, 1, "Shouldn't have the unverified marker");
 | 
						|
 | 
						|
  let notificationPromise = waitForNotification("addon-installed");
 | 
						|
  acceptInstallDialog(installDialog);
 | 
						|
  await notificationPromise;
 | 
						|
 | 
						|
  let addon = await new Promise(resolve => {
 | 
						|
    AddonManager.getAddonByID("restartless-xpi@tests.mozilla.org", function(result) {
 | 
						|
      resolve(result);
 | 
						|
    });
 | 
						|
  });
 | 
						|
  addon.uninstall();
 | 
						|
 | 
						|
  Services.perms.remove(makeURI("http://example.com/"), "install");
 | 
						|
  await removeTabAndWaitForNotificationClose();
 | 
						|
},
 | 
						|
 | 
						|
async function test_url() {
 | 
						|
  let progressPromise = waitForProgressNotification();
 | 
						|
  let dialogPromise = waitForInstallDialog();
 | 
						|
  gBrowser.selectedTab = BrowserTestUtils.addTab(gBrowser, "about:blank");
 | 
						|
  await BrowserTestUtils.browserLoaded(gBrowser.selectedBrowser);
 | 
						|
  gBrowser.loadURI(TESTROOT + "amosigned.xpi");
 | 
						|
  await progressPromise;
 | 
						|
  let installDialog = await dialogPromise;
 | 
						|
 | 
						|
  let notificationPromise = waitForNotification("addon-install-restart");
 | 
						|
  acceptInstallDialog(installDialog);
 | 
						|
  let panel = await notificationPromise;
 | 
						|
 | 
						|
  let notification = panel.childNodes[0];
 | 
						|
  is(notification.button.label, "Restart Now", "Should have seen the right button");
 | 
						|
  is(notification.getAttribute("label"),
 | 
						|
     "XPI Test will be installed after you restart " + gApp + ".",
 | 
						|
     "Should have seen the right message");
 | 
						|
 | 
						|
  let installs = await getInstalls();
 | 
						|
  is(installs.length, 1, "Should be one pending install");
 | 
						|
  installs[0].cancel();
 | 
						|
 | 
						|
  await removeTabAndWaitForNotificationClose();
 | 
						|
},
 | 
						|
 | 
						|
async function test_localFile() {
 | 
						|
  let cr = Cc["@mozilla.org/chrome/chrome-registry;1"]
 | 
						|
             .getService(Ci.nsIChromeRegistry);
 | 
						|
  let path;
 | 
						|
  try {
 | 
						|
    path = cr.convertChromeURL(makeURI(CHROMEROOT + "corrupt.xpi")).spec;
 | 
						|
  } catch (ex) {
 | 
						|
    path = CHROMEROOT + "corrupt.xpi";
 | 
						|
  }
 | 
						|
 | 
						|
  let failPromise = new Promise(resolve => {
 | 
						|
    Services.obs.addObserver(function observer() {
 | 
						|
      Services.obs.removeObserver(observer, "addon-install-failed");
 | 
						|
      resolve();
 | 
						|
    }, "addon-install-failed");
 | 
						|
  });
 | 
						|
  gBrowser.selectedTab = BrowserTestUtils.addTab(gBrowser, "about:blank");
 | 
						|
  await BrowserTestUtils.browserLoaded(gBrowser.selectedBrowser);
 | 
						|
  gBrowser.loadURI(path);
 | 
						|
  await failPromise;
 | 
						|
 | 
						|
  // Wait for the browser code to add the failure notification
 | 
						|
  await waitForSingleNotification();
 | 
						|
 | 
						|
  let notification = PopupNotifications.panel.childNodes[0];
 | 
						|
  is(notification.id, "addon-install-failed-notification", "Should have seen the install fail");
 | 
						|
  is(notification.getAttribute("label"),
 | 
						|
     "This add-on could not be installed because it appears to be corrupt.",
 | 
						|
     "Should have seen the right message");
 | 
						|
 | 
						|
  await removeTabAndWaitForNotificationClose();
 | 
						|
},
 | 
						|
 | 
						|
async function test_tabClose() {
 | 
						|
  if (!Services.prefs.getBoolPref("xpinstall.customConfirmationUI", false)) {
 | 
						|
    info("Test skipped due to xpinstall.customConfirmationUI being false.");
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  let progressPromise = waitForProgressNotification();
 | 
						|
  let dialogPromise = waitForInstallDialog();
 | 
						|
  gBrowser.selectedTab = BrowserTestUtils.addTab(gBrowser, "about:blank");
 | 
						|
  await BrowserTestUtils.browserLoaded(gBrowser.selectedBrowser);
 | 
						|
  gBrowser.loadURI(TESTROOT + "amosigned.xpi");
 | 
						|
  await progressPromise;
 | 
						|
  await dialogPromise;
 | 
						|
 | 
						|
  let installs = await getInstalls();
 | 
						|
  is(installs.length, 1, "Should be one pending install");
 | 
						|
 | 
						|
  await removeTabAndWaitForNotificationClose(gBrowser.selectedTab);
 | 
						|
 | 
						|
  installs = await getInstalls();
 | 
						|
  is(installs.length, 0, "Should be no pending install since the tab is closed");
 | 
						|
},
 | 
						|
 | 
						|
// Add-ons should be cancelled and the install notification destroyed when
 | 
						|
// navigating to a new origin
 | 
						|
async function test_tabNavigate() {
 | 
						|
  if (!Services.prefs.getBoolPref("xpinstall.customConfirmationUI", false)) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  let pm = Services.perms;
 | 
						|
  pm.add(makeURI("http://example.com/"), "install", pm.ALLOW_ACTION);
 | 
						|
 | 
						|
  let progressPromise = waitForProgressNotification();
 | 
						|
  let dialogPromise = waitForInstallDialog();
 | 
						|
  let triggers = encodeURIComponent(JSON.stringify({
 | 
						|
    "Extension XPI": "amosigned.xpi"
 | 
						|
  }));
 | 
						|
  BrowserTestUtils.openNewForegroundTab(gBrowser, TESTROOT + "installtrigger.html?" + triggers);
 | 
						|
  await progressPromise;
 | 
						|
  await dialogPromise;
 | 
						|
 | 
						|
  let closePromise = waitForNotificationClose();
 | 
						|
  let loadPromise = BrowserTestUtils.browserLoaded(gBrowser.selectedBrowser);
 | 
						|
  gBrowser.loadURI("about:blank");
 | 
						|
  await closePromise;
 | 
						|
 | 
						|
  // At the point of closing notification, AddonManager hasn't yet removed
 | 
						|
  // pending installs.  It removes them in onLocationChange listener, and
 | 
						|
  // the notification is also closed in another onLocationChange listener,
 | 
						|
  // before AddonManager's one.  Wait for next tick to ensure all
 | 
						|
  // onLocationChange listeners are performed.
 | 
						|
  await waitForTick();
 | 
						|
 | 
						|
  let installs = await getInstalls();
 | 
						|
  is(installs.length, 0, "Should be no pending install");
 | 
						|
 | 
						|
  Services.perms.remove(makeURI("http://example.com/"), "install");
 | 
						|
  await loadPromise;
 | 
						|
 | 
						|
  await removeTabAndWaitForNotificationClose();
 | 
						|
},
 | 
						|
 | 
						|
async function test_urlBar() {
 | 
						|
  let progressPromise = waitForProgressNotification();
 | 
						|
  let dialogPromise = waitForInstallDialog();
 | 
						|
 | 
						|
  gBrowser.selectedTab = BrowserTestUtils.addTab(gBrowser, "about:blank");
 | 
						|
  await BrowserTestUtils.browserLoaded(gBrowser.selectedBrowser);
 | 
						|
  gURLBar.value = TESTROOT + "amosigned.xpi";
 | 
						|
  gURLBar.focus();
 | 
						|
  EventUtils.synthesizeKey("KEY_Enter");
 | 
						|
 | 
						|
  await progressPromise;
 | 
						|
  let installDialog = await dialogPromise;
 | 
						|
 | 
						|
  let notificationPromise = waitForNotification("addon-install-restart");
 | 
						|
  acceptInstallDialog(installDialog);
 | 
						|
  let panel = await notificationPromise;
 | 
						|
 | 
						|
  let notification = panel.childNodes[0];
 | 
						|
  is(notification.button.label, "Restart Now", "Should have seen the right button");
 | 
						|
  is(notification.getAttribute("label"),
 | 
						|
     "XPI Test will be installed after you restart " + gApp + ".",
 | 
						|
     "Should have seen the right message");
 | 
						|
 | 
						|
  let installs = await getInstalls();
 | 
						|
  is(installs.length, 1, "Should be one pending install");
 | 
						|
  installs[0].cancel();
 | 
						|
 | 
						|
  await removeTabAndWaitForNotificationClose();
 | 
						|
},
 | 
						|
 | 
						|
async function test_wrongHost() {
 | 
						|
  let requestedUrl = TESTROOT2 + "enabled.html";
 | 
						|
  gBrowser.selectedTab = BrowserTestUtils.addTab(gBrowser);
 | 
						|
 | 
						|
  let loadedPromise = BrowserTestUtils.browserLoaded(gBrowser.selectedBrowser, false, requestedUrl);
 | 
						|
  gBrowser.loadURI(TESTROOT2 + "enabled.html");
 | 
						|
  await loadedPromise;
 | 
						|
 | 
						|
  let progressPromise = waitForProgressNotification();
 | 
						|
  let notificationPromise = waitForNotification("addon-install-failed");
 | 
						|
  gBrowser.loadURI(TESTROOT + "corrupt.xpi");
 | 
						|
  await progressPromise;
 | 
						|
  let panel = await notificationPromise;
 | 
						|
 | 
						|
  let notification = panel.childNodes[0];
 | 
						|
  is(notification.getAttribute("label"),
 | 
						|
     "The add-on downloaded from this site could not be installed " +
 | 
						|
     "because it appears to be corrupt.",
 | 
						|
     "Should have seen the right message");
 | 
						|
 | 
						|
  await removeTabAndWaitForNotificationClose();
 | 
						|
},
 | 
						|
 | 
						|
async function test_reload() {
 | 
						|
  let pm = Services.perms;
 | 
						|
  pm.add(makeURI("http://example.com/"), "install", pm.ALLOW_ACTION);
 | 
						|
 | 
						|
  let progressPromise = waitForProgressNotification();
 | 
						|
  let dialogPromise = waitForInstallDialog();
 | 
						|
  let triggers = encodeURIComponent(JSON.stringify({
 | 
						|
    "Unsigned XPI": "amosigned.xpi"
 | 
						|
  }));
 | 
						|
  BrowserTestUtils.openNewForegroundTab(gBrowser, TESTROOT + "installtrigger.html?" + triggers);
 | 
						|
  await progressPromise;
 | 
						|
  let installDialog = await dialogPromise;
 | 
						|
 | 
						|
  let notificationPromise = waitForNotification("addon-install-restart");
 | 
						|
  acceptInstallDialog(installDialog);
 | 
						|
  let panel = await notificationPromise;
 | 
						|
 | 
						|
  let notification = panel.childNodes[0];
 | 
						|
  is(notification.button.label, "Restart Now", "Should have seen the right button");
 | 
						|
  is(notification.getAttribute("label"),
 | 
						|
     "XPI Test will be installed after you restart " + gApp + ".",
 | 
						|
     "Should have seen the right message");
 | 
						|
 | 
						|
  function testFail() {
 | 
						|
    ok(false, "Reloading should not have hidden the notification");
 | 
						|
  }
 | 
						|
  PopupNotifications.panel.addEventListener("popuphiding", testFail);
 | 
						|
  let requestedUrl = TESTROOT2 + "enabled.html";
 | 
						|
  let loadedPromise = BrowserTestUtils.browserLoaded(gBrowser.selectedBrowser, false, requestedUrl);
 | 
						|
  gBrowser.loadURI(TESTROOT2 + "enabled.html");
 | 
						|
  await loadedPromise;
 | 
						|
  PopupNotifications.panel.removeEventListener("popuphiding", testFail);
 | 
						|
 | 
						|
  let installs = await getInstalls();
 | 
						|
  is(installs.length, 1, "Should be one pending install");
 | 
						|
  installs[0].cancel();
 | 
						|
 | 
						|
  Services.perms.remove(makeURI("http://example.com/"), "install");
 | 
						|
  await removeTabAndWaitForNotificationClose();
 | 
						|
},
 | 
						|
 | 
						|
async function test_theme() {
 | 
						|
  let pm = Services.perms;
 | 
						|
  pm.add(makeURI("http://example.com/"), "install", pm.ALLOW_ACTION);
 | 
						|
 | 
						|
  let progressPromise = waitForProgressNotification();
 | 
						|
  let dialogPromise = waitForInstallDialog();
 | 
						|
  let triggers = encodeURIComponent(JSON.stringify({
 | 
						|
    "Theme XPI": "theme.xpi"
 | 
						|
  }));
 | 
						|
  BrowserTestUtils.openNewForegroundTab(gBrowser, TESTROOT + "installtrigger.html?" + triggers);
 | 
						|
  await progressPromise;
 | 
						|
  let installDialog = await dialogPromise;
 | 
						|
 | 
						|
  let notificationPromise = waitForNotification("addon-install-restart");
 | 
						|
  acceptInstallDialog(installDialog);
 | 
						|
  let panel = await notificationPromise;
 | 
						|
 | 
						|
  let notification = panel.childNodes[0];
 | 
						|
  is(notification.button.label, "Restart Now", "Should have seen the right button");
 | 
						|
  is(notification.getAttribute("label"),
 | 
						|
     "Theme Test will be installed after you restart " + gApp + ".",
 | 
						|
     "Should have seen the right message");
 | 
						|
 | 
						|
  let addon = await new Promise(resolve => {
 | 
						|
    AddonManager.getAddonByID("{972ce4c6-7e08-4474-a285-3208198ce6fd}", function(result) {
 | 
						|
      resolve(result);
 | 
						|
    });
 | 
						|
  });
 | 
						|
  ok(addon.userDisabled, "Should be switching away from the default theme.");
 | 
						|
  // Undo the pending theme switch
 | 
						|
  addon.userDisabled = false;
 | 
						|
 | 
						|
  addon = await new Promise(resolve => {
 | 
						|
    AddonManager.getAddonByID("theme-xpi@tests.mozilla.org", function(result) {
 | 
						|
      resolve(result);
 | 
						|
    });
 | 
						|
  });
 | 
						|
  isnot(addon, null, "Test theme will have been installed");
 | 
						|
  addon.uninstall();
 | 
						|
 | 
						|
  Services.perms.remove(makeURI("http://example.com/"), "install");
 | 
						|
  await removeTabAndWaitForNotificationClose();
 | 
						|
},
 | 
						|
 | 
						|
async function test_renotifyBlocked() {
 | 
						|
  let notificationPromise = waitForNotification("addon-install-blocked");
 | 
						|
  let triggers = encodeURIComponent(JSON.stringify({
 | 
						|
    "XPI": "amosigned.xpi"
 | 
						|
  }));
 | 
						|
  BrowserTestUtils.openNewForegroundTab(gBrowser, TESTROOT + "installtrigger.html?" + triggers);
 | 
						|
  let panel = await notificationPromise;
 | 
						|
 | 
						|
  let closePromise = waitForNotificationClose();
 | 
						|
  // hide the panel (this simulates the user dismissing it)
 | 
						|
  panel.hidePopup();
 | 
						|
  await closePromise;
 | 
						|
 | 
						|
  info("Timeouts after this probably mean bug 589954 regressed");
 | 
						|
 | 
						|
  await new Promise(resolve => executeSoon(resolve));
 | 
						|
 | 
						|
  notificationPromise = waitForNotification("addon-install-blocked");
 | 
						|
  gBrowser.loadURI(TESTROOT + "installtrigger.html?" + triggers);
 | 
						|
  await notificationPromise;
 | 
						|
 | 
						|
  let installs = await getInstalls();
 | 
						|
  is(installs.length, 2, "Should be two pending installs");
 | 
						|
 | 
						|
  await removeTabAndWaitForNotificationClose(gBrowser.selectedTab);
 | 
						|
 | 
						|
  installs = await getInstalls();
 | 
						|
  is(installs.length, 0, "Should have cancelled the installs");
 | 
						|
},
 | 
						|
 | 
						|
async function test_renotifyInstalled() {
 | 
						|
  let pm = Services.perms;
 | 
						|
  pm.add(makeURI("http://example.com/"), "install", pm.ALLOW_ACTION);
 | 
						|
 | 
						|
  let progressPromise = waitForProgressNotification();
 | 
						|
  let dialogPromise = waitForInstallDialog();
 | 
						|
  let triggers = encodeURIComponent(JSON.stringify({
 | 
						|
    "XPI": "amosigned.xpi"
 | 
						|
  }));
 | 
						|
  BrowserTestUtils.openNewForegroundTab(gBrowser, TESTROOT + "installtrigger.html?" + triggers);
 | 
						|
  await progressPromise;
 | 
						|
  let installDialog = await dialogPromise;
 | 
						|
 | 
						|
  // Wait for the complete notification
 | 
						|
  let notificationPromise = waitForNotification("addon-install-restart");
 | 
						|
  acceptInstallDialog(installDialog);
 | 
						|
  let panel = await notificationPromise;
 | 
						|
 | 
						|
  let closePromise = waitForNotificationClose();
 | 
						|
  // hide the panel (this simulates the user dismissing it)
 | 
						|
  panel.hidePopup();
 | 
						|
  await closePromise;
 | 
						|
 | 
						|
  // Install another
 | 
						|
  await new Promise(resolve => executeSoon(resolve));
 | 
						|
 | 
						|
  progressPromise = waitForProgressNotification();
 | 
						|
  dialogPromise = waitForInstallDialog();
 | 
						|
  gBrowser.loadURI(TESTROOT + "installtrigger.html?" + triggers);
 | 
						|
  await progressPromise;
 | 
						|
  installDialog = await dialogPromise;
 | 
						|
 | 
						|
  info("Timeouts after this probably mean bug 589954 regressed");
 | 
						|
 | 
						|
  // Wait for the complete notification
 | 
						|
  notificationPromise = waitForNotification("addon-install-restart");
 | 
						|
  acceptInstallDialog(installDialog);
 | 
						|
  await notificationPromise;
 | 
						|
 | 
						|
  let installs = await getInstalls();
 | 
						|
  is(installs.length, 1, "Should be one pending installs");
 | 
						|
  installs[0].cancel();
 | 
						|
 | 
						|
  Services.perms.remove(makeURI("http://example.com/"), "install");
 | 
						|
  await removeTabAndWaitForNotificationClose();
 | 
						|
},
 | 
						|
 | 
						|
async function test_cancel() {
 | 
						|
  let pm = Services.perms;
 | 
						|
  pm.add(makeURI("http://example.com/"), "install", pm.ALLOW_ACTION);
 | 
						|
 | 
						|
  let notificationPromise = waitForNotification(PROGRESS_NOTIFICATION);
 | 
						|
  let triggers = encodeURIComponent(JSON.stringify({
 | 
						|
    "XPI": "slowinstall.sjs?file=amosigned.xpi"
 | 
						|
  }));
 | 
						|
  BrowserTestUtils.openNewForegroundTab(gBrowser, TESTROOT + "installtrigger.html?" + triggers);
 | 
						|
  let panel = await notificationPromise;
 | 
						|
 | 
						|
  let notification = panel.childNodes[0];
 | 
						|
  // Close the notification
 | 
						|
  let anchor = document.getElementById("addons-notification-icon");
 | 
						|
  anchor.click();
 | 
						|
  // Reopen the notification
 | 
						|
  anchor.click();
 | 
						|
 | 
						|
  ok(PopupNotifications.isPanelOpen, "Notification should still be open");
 | 
						|
  is(PopupNotifications.panel.childNodes.length, 1, "Should be only one notification");
 | 
						|
  notification = panel.childNodes[0];
 | 
						|
  is(notification.id, "addon-progress-notification", "Should have seen the progress notification");
 | 
						|
 | 
						|
  // Cancel the download
 | 
						|
  let install = notification.notification.options.installs[0];
 | 
						|
  let cancelledPromise = new Promise(resolve => {
 | 
						|
    install.addListener({
 | 
						|
      onDownloadCancelled() {
 | 
						|
        install.removeListener(this);
 | 
						|
        resolve();
 | 
						|
      }
 | 
						|
    });
 | 
						|
  });
 | 
						|
  EventUtils.synthesizeMouseAtCenter(notification.secondaryButton, {});
 | 
						|
  await cancelledPromise;
 | 
						|
 | 
						|
  await new Promise(resolve => executeSoon(resolve));
 | 
						|
 | 
						|
  ok(!PopupNotifications.isPanelOpen, "Notification should be closed");
 | 
						|
 | 
						|
  let installs = await getInstalls();
 | 
						|
  is(installs.length, 0, "Should be no pending install");
 | 
						|
 | 
						|
  Services.perms.remove(makeURI("http://example.com/"), "install");
 | 
						|
  BrowserTestUtils.removeTab(gBrowser.selectedTab);
 | 
						|
},
 | 
						|
 | 
						|
async function test_failedSecurity() {
 | 
						|
  Services.prefs.setBoolPref(PREF_INSTALL_REQUIREBUILTINCERTS, false);
 | 
						|
  setupRedirect({
 | 
						|
    "Location": TESTROOT + "amosigned.xpi"
 | 
						|
  });
 | 
						|
 | 
						|
  let notificationPromise = waitForNotification("addon-install-blocked");
 | 
						|
  let triggers = encodeURIComponent(JSON.stringify({
 | 
						|
    "XPI": "redirect.sjs?mode=redirect"
 | 
						|
  }));
 | 
						|
  BrowserTestUtils.openNewForegroundTab(gBrowser, SECUREROOT + "installtrigger.html?" + triggers);
 | 
						|
  let panel = await notificationPromise;
 | 
						|
 | 
						|
  let notification = panel.childNodes[0];
 | 
						|
  // Click on Allow
 | 
						|
  EventUtils.synthesizeMouse(notification.button, 20, 10, {});
 | 
						|
 | 
						|
  // Notification should have changed to progress notification
 | 
						|
  ok(PopupNotifications.isPanelOpen, "Notification should still be open");
 | 
						|
  is(PopupNotifications.panel.childNodes.length, 1, "Should be only one notification");
 | 
						|
  notification = panel.childNodes[0];
 | 
						|
  is(notification.id, "addon-progress-notification", "Should have seen the progress notification");
 | 
						|
 | 
						|
  // Wait for it to fail
 | 
						|
  await new Promise(resolve => {
 | 
						|
    Services.obs.addObserver(function observer() {
 | 
						|
      Services.obs.removeObserver(observer, "addon-install-failed");
 | 
						|
      resolve();
 | 
						|
    }, "addon-install-failed");
 | 
						|
  });
 | 
						|
 | 
						|
  // Allow the browser code to add the failure notification and then wait
 | 
						|
  // for the progress notification to dismiss itself
 | 
						|
  await waitForSingleNotification();
 | 
						|
  is(PopupNotifications.panel.childNodes.length, 1, "Should be only one notification");
 | 
						|
  notification = panel.childNodes[0];
 | 
						|
  is(notification.id, "addon-install-failed-notification", "Should have seen the install fail");
 | 
						|
 | 
						|
  Services.prefs.setBoolPref(PREF_INSTALL_REQUIREBUILTINCERTS, true);
 | 
						|
  await removeTabAndWaitForNotificationClose();
 | 
						|
}
 | 
						|
];
 | 
						|
 | 
						|
var gTestStart = null;
 | 
						|
 | 
						|
var XPInstallObserver = {
 | 
						|
  observe(aSubject, aTopic, aData) {
 | 
						|
    var installInfo = aSubject.wrappedJSObject;
 | 
						|
    info("Observed " + aTopic + " for " + installInfo.installs.length + " installs");
 | 
						|
    installInfo.installs.forEach(function(aInstall) {
 | 
						|
      info("Install of " + aInstall.sourceURI.spec + " was in state " + aInstall.state);
 | 
						|
    });
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
add_task(async function() {
 | 
						|
  requestLongerTimeout(4);
 | 
						|
 | 
						|
  Services.prefs.setBoolPref("extensions.logging.enabled", true);
 | 
						|
  Services.prefs.setBoolPref("extensions.strictCompatibility", true);
 | 
						|
  Services.prefs.setBoolPref("extensions.install.requireSecureOrigin", false);
 | 
						|
  Services.prefs.setIntPref("security.dialog_enable_delay", 0);
 | 
						|
 | 
						|
  Services.obs.addObserver(XPInstallObserver, "addon-install-started");
 | 
						|
  Services.obs.addObserver(XPInstallObserver, "addon-install-blocked");
 | 
						|
  Services.obs.addObserver(XPInstallObserver, "addon-install-failed");
 | 
						|
 | 
						|
  registerCleanupFunction(function() {
 | 
						|
    // Make sure no more test parts run in case we were timed out
 | 
						|
    TESTS = [];
 | 
						|
 | 
						|
    AddonManager.getAllInstalls(function(aInstalls) {
 | 
						|
      aInstalls.forEach(function(aInstall) {
 | 
						|
        aInstall.cancel();
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    Services.prefs.clearUserPref("extensions.logging.enabled");
 | 
						|
    Services.prefs.clearUserPref("extensions.strictCompatibility");
 | 
						|
    Services.prefs.clearUserPref("extensions.install.requireSecureOrigin");
 | 
						|
    Services.prefs.clearUserPref("security.dialog_enable_delay");
 | 
						|
 | 
						|
    Services.obs.removeObserver(XPInstallObserver, "addon-install-started");
 | 
						|
    Services.obs.removeObserver(XPInstallObserver, "addon-install-blocked");
 | 
						|
    Services.obs.removeObserver(XPInstallObserver, "addon-install-failed");
 | 
						|
  });
 | 
						|
 | 
						|
  for (let i = 0; i < TESTS.length; ++i) {
 | 
						|
    if (gTestStart)
 | 
						|
      info("Test part took " + (Date.now() - gTestStart) + "ms");
 | 
						|
 | 
						|
    ok(!PopupNotifications.isPanelOpen, "Notification should be closed");
 | 
						|
 | 
						|
    let installs = await new Promise(resolve => {
 | 
						|
      AddonManager.getAllInstalls(function(aInstalls) {
 | 
						|
        resolve(aInstalls);
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    is(installs.length, 0, "Should be no active installs");
 | 
						|
    info("Running " + TESTS[i].name);
 | 
						|
    gTestStart = Date.now();
 | 
						|
    await TESTS[i]();
 | 
						|
  }
 | 
						|
});
 |