From ba0d11300444af30048873bffcf13a99da1464f6 Mon Sep 17 00:00:00 2001 From: Alexei Date: Tue, 5 Sep 2023 11:07:49 -0400 Subject: [PATCH 1/2] Avoid unnecessarily reopening the welcome page Because of idle background process termination with event pages and service workers. --- doc/permissions.md | 3 +++ src/js/background.js | 55 ++++++++++++++++++++++++++++++++++++++++- src/manifest.json | 1 + src/skin/js/firstRun.js | 17 +++++++------ 4 files changed, 67 insertions(+), 9 deletions(-) diff --git a/doc/permissions.md b/doc/permissions.md index ee82a13d4d..d566b3851c 100644 --- a/doc/permissions.md +++ b/doc/permissions.md @@ -26,3 +26,6 @@ These permissions allow Privacy Badger to use the WebRequest and WebRequestBlock ## Tabs Privacy Badger needs access to the tabs API so that the extension can detect which tab is active and which tabs are simply present in the background. The extension icon, badge and popup update to reflect the state of Privacy Badger. This often requires knowing the tab's URL. For example, updating the icon requires the URL in order to determine whether Privacy Badger should be shown as disabled on that tab. Privacy Badger also uses the tabs API for miscellaneous tasks such as opening or switching to the already open new user welcome page. + +## Alarms +Privacy Badger uses the Alarms API to temporarily ensure the background process does not get terminated as idle when Privacy Badger needs it to perform some longer running asynchronous task. This workaround is used to help reopen the welcome page when it appears that the extension has been restarted because of interaction with the Private/Incognito browsing permision prompt, but not because of idle background process termination. diff --git a/src/js/background.js b/src/js/background.js index 76f1066043..102c64bcc4 100644 --- a/src/js/background.js +++ b/src/js/background.js @@ -344,6 +344,55 @@ Badger.prototype = { }); }, + /** + * If the background process is an event page or a service worker, + * it can get terminated while the user is still on the welcome page. + * + * When the user spends >= 30s on the welcome page, the background process + * will get terminated and another welcome page will unexpectedly appear + * following any user action that restarts the background process. + * + * (We reopen the welcome page via firstRunTimerFinished, our workaround + * for restoring the welcome page when Firefox restarts the extension + * in response to interaction with the private browsing permission hanger.) + * + * Since extension alarm events reset the idle timer in both Firefox and + * Chrome, let's periodically set an immediately firing alarm to keep + * the background process running as long as the welcome page stays open. + */ + keepBackgroundAliveForWelcomePage: function () { + let ALARM_NAME = "welcome-page-keepalive", + INTERVAL = 10000; // 10 secs + + // wait a bit and create an alarm that will reset the idle timer + // + // we use setTimeout to set an immediately firing alarm + // because the alarms API has a minimum resolution of one minute, + // but we want to trigger an extension event after several seconds + setTimeout(function () { + chrome.alarms.create(ALARM_NAME, { delayInMinutes: 0 }); + }, INTERVAL); + + chrome.alarms.onAlarm.addListener(alarm => { + if (alarm.name != ALARM_NAME) { + return; + } + + // if the welcome page is still open + chrome.tabs.query({ + url: chrome.runtime.getURL("/skin/firstRun.html") + }, function (tabs) { + if (!tabs.length) { + return; + } + // wait and create another alarm + setTimeout(function () { + chrome.alarms.create(ALARM_NAME, { delayInMinutes: 0 }); + }, INTERVAL); + }); + }); + }, + initWelcomePage: function () { let self = this, privateStore = self.getPrivateSettings(); @@ -365,10 +414,14 @@ Badger.prototype = { }, showWelcomePage: function () { - let settings = this.getSettings(); + let self = this, + settings = self.getSettings(); + if (settings.getItem("showIntroPage")) { chrome.tabs.create({ url: chrome.runtime.getURL("/skin/firstRun.html") + }, function () { + self.keepBackgroundAliveForWelcomePage(); }); } else { // don't remind users to look at the intro page either diff --git a/src/manifest.json b/src/manifest.json index 7d0ef530a6..e2a670a96a 100644 --- a/src/manifest.json +++ b/src/manifest.json @@ -11,6 +11,7 @@ }, "incognito": "spanning", "permissions": [ + "alarms", "tabs", "http://*/*", "https://*/*", diff --git a/src/skin/js/firstRun.js b/src/skin/js/firstRun.js index 381eec1a6a..ef508e38b6 100644 --- a/src/skin/js/firstRun.js +++ b/src/skin/js/firstRun.js @@ -4,14 +4,15 @@ $(function () { $(".scroll-it").smoothScroll(); $(window).scroll(function () { - if (!already_set) { - if ($(window).scrollTop() > 400) { - already_set = true; - chrome.runtime.sendMessage({ - type: "updateSettings", - data: { seenComic: true } - }); - } + if (already_set) { + return; + } + if ($(window).scrollTop() > 400) { + already_set = true; + chrome.runtime.sendMessage({ + type: "updateSettings", + data: { seenComic: true } + }); } }); }); From b05fc017559c246165ab1203aa5818e503e045ba Mon Sep 17 00:00:00 2001 From: Alexei Date: Tue, 5 Sep 2023 16:29:23 -0400 Subject: [PATCH 2/2] Fix background process keepalive for Chrome --- src/js/background.js | 73 +++++++++++++++++++++++++++++++------------- 1 file changed, 52 insertions(+), 21 deletions(-) diff --git a/src/js/background.js b/src/js/background.js index 102c64bcc4..7672a21512 100644 --- a/src/js/background.js +++ b/src/js/background.js @@ -43,6 +43,13 @@ function Badger(from_qunit) { self.isFirstRun = false; self.isUpdate = false; + (function () { + let manifestJson = chrome.runtime.getManifest(); + self.manifestVersion = manifestJson.manifest_version; + self.isEventPage = (utils.hasOwn(manifestJson.background, "persistent") && + manifestJson.background.persistent === false); + }()); + self.firstPartyDomainPotentiallyRequired = testCookiesFirstPartyDomain(); self.widgetList = []; @@ -356,39 +363,63 @@ Badger.prototype = { * for restoring the welcome page when Firefox restarts the extension * in response to interaction with the private browsing permission hanger.) * - * Since extension alarm events reset the idle timer in both Firefox and - * Chrome, let's periodically set an immediately firing alarm to keep + * Let's periodically call a low-overhead, no-side effects API to keep * the background process running as long as the welcome page stays open. + * + * While extension alarm events reset the idle timer in both Firefox and + * Chrome, Chrome enforces a minimum resolution of one minute. However, + * since most extension API calls also reset the idle timer in Chrome, + * simply looking up whether the welcome page is still open is enough + * to reset the idle timer. */ keepBackgroundAliveForWelcomePage: function () { let ALARM_NAME = "welcome-page-keepalive", INTERVAL = 10000; // 10 secs - // wait a bit and create an alarm that will reset the idle timer - // - // we use setTimeout to set an immediately firing alarm - // because the alarms API has a minimum resolution of one minute, - // but we want to trigger an extension event after several seconds - setTimeout(function () { - chrome.alarms.create(ALARM_NAME, { delayInMinutes: 0 }); - }, INTERVAL); + if (badger.manifestVersion == 2 && !badger.isEventPage) { + return; // noop + } + + function getWelcomeTab(callback) { + chrome.tabs.query({ + url: chrome.runtime.getURL("/skin/firstRun.html") + }, function (tabs) { + callback(tabs[0]); + }); + } + + function workaroundForChrome() { + setTimeout(function () { + getWelcomeTab(function (tab) { + if (tab) { + workaroundForChrome(); + } + }); + }, INTERVAL); + } + + if (!badger.isEventPage) { + workaroundForChrome(); + return; + } + + // create an alarm that will reset the idle timer in Firefox + chrome.alarms.create(ALARM_NAME, { + when: Date.now() + INTERVAL + }); chrome.alarms.onAlarm.addListener(alarm => { if (alarm.name != ALARM_NAME) { return; } - - // if the welcome page is still open - chrome.tabs.query({ - url: chrome.runtime.getURL("/skin/firstRun.html") - }, function (tabs) { - if (!tabs.length) { - return; + getWelcomeTab(function (tab) { + // if the welcome page is still open + if (tab) { + // create another alarm + chrome.alarms.create(ALARM_NAME, { + when: Date.now() + INTERVAL + }); } - // wait and create another alarm - setTimeout(function () { - chrome.alarms.create(ALARM_NAME, { delayInMinutes: 0 }); - }, INTERVAL); }); }); },