diff --git a/browser/base/content/browser-trackingprotection.js b/browser/base/content/browser-trackingprotection.js new file mode 100644 index 000000000000..bc02c1fbdf99 --- /dev/null +++ b/browser/base/content/browser-trackingprotection.js @@ -0,0 +1,96 @@ +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. + +let TrackingProtection = { + PREF_ENABLED: "privacy.trackingprotection.enabled", + + init() { + let $ = selector => document.querySelector(selector); + this.container = $("#tracking-protection-container"); + this.content = $("#tracking-protection-content"); + + this.updateEnabled(); + Services.prefs.addObserver(this.PREF_ENABLED, this, false); + + this.enabledHistogram.add(this.enabled); + }, + + uninit() { + Services.prefs.removeObserver(this.PREF_ENABLED, this); + }, + + observe() { + this.updateEnabled(); + }, + + updateEnabled() { + this.enabled = Services.prefs.getBoolPref(this.PREF_ENABLED); + this.container.hidden = !this.enabled; + }, + + get enabledHistogram() { + return Services.telemetry.getHistogramById("TRACKING_PROTECTION_ENABLED"); + }, + + get eventsHistogram() { + return Services.telemetry.getHistogramById("TRACKING_PROTECTION_EVENTS"); + }, + + onSecurityChange(state) { + if (!this.enabled) { + return; + } + + let { + STATE_BLOCKED_TRACKING_CONTENT, STATE_LOADED_TRACKING_CONTENT + } = Ci.nsIWebProgressListener; + + if (state & STATE_BLOCKED_TRACKING_CONTENT) { + this.content.setAttribute("block-active", true); + this.content.removeAttribute("block-disabled"); + } else if (state & STATE_LOADED_TRACKING_CONTENT) { + this.content.setAttribute("block-disabled", true); + this.content.removeAttribute("block-active"); + } else { + this.content.removeAttribute("block-disabled"); + this.content.removeAttribute("block-active"); + } + + // Telemetry for state change. + this.eventsHistogram.add(0); + }, + + disableForCurrentPage() { + // Convert document URI into the format used by + // nsChannelClassifier::ShouldEnableTrackingProtection. + // Any scheme turned into https is correct. + let normalizedUrl = Services.io.newURI( + "https://" + gBrowser.selectedBrowser.currentURI.hostPort, + null, null); + + // Add the current host in the 'trackingprotection' consumer of + // the permission manager using a normalized URI. This effectively + // places this host on the tracking protection allowlist. + Services.perms.add(normalizedUrl, + "trackingprotection", Services.perms.ALLOW_ACTION); + + // Telemetry for disable protection. + this.eventsHistogram.add(1); + + BrowserReload(); + }, + + enableForCurrentPage() { + // Remove the current host from the 'trackingprotection' consumer + // of the permission manager. This effectively removes this host + // from the tracking protection allowlist. + Services.perms.remove(gBrowser.selectedBrowser.currentURI.host, + "trackingprotection"); + + // Telemetry for enable protection. + this.eventsHistogram.add(2); + + BrowserReload(); + }, +}; diff --git a/browser/base/content/browser.js b/browser/base/content/browser.js index 7f4d699cb0a7..0d79962b725e 100644 --- a/browser/base/content/browser.js +++ b/browser/base/content/browser.js @@ -280,6 +280,7 @@ let gInitialPages = [ #include browser-social.js #include browser-tabview.js #include browser-thumbnails.js +#include browser-trackingprotection.js #ifdef MOZ_DATA_REPORTING #include browser-data-submission-info-bar.js @@ -964,6 +965,7 @@ var gBrowserInit = { BrowserOnClick.init(); DevEdition.init(); AboutPrivateBrowsingListener.init(); + TrackingProtection.init(); let mm = window.getGroupMessageManager("browsers"); mm.loadFrameScript("chrome://browser/content/tab-content.js", true); @@ -1446,12 +1448,6 @@ var gBrowserInit = { } }, 5000); - // Telemetry for tracking protection. - let tpEnabled = gPrefService - .getBoolPref("privacy.trackingprotection.enabled"); - Services.telemetry.getHistogramById("TRACKING_PROTECTION_ENABLED") - .add(tpEnabled); - PanicButtonNotifier.init(); }); this.delayedStartupFinished = true; @@ -1534,6 +1530,8 @@ var gBrowserInit = { DevEdition.uninit(); + TrackingProtection.uninit(); + gMenuButtonUpdateBadge.uninit(); ReadingListUI.uninit(); @@ -4383,6 +4381,7 @@ var XULBrowserWindow = { uri = Services.uriFixup.createExposableURI(uri); } catch (e) {} gIdentityHandler.checkIdentity(this._state, uri); + TrackingProtection.onSecurityChange(this._state); }, // simulate all change notifications after switching tabs @@ -6781,7 +6780,7 @@ var gIdentityHandler = { nsIWebProgressListener.STATE_BLOCKED_TRACKING_CONTENT | nsIWebProgressListener.STATE_LOADED_TRACKING_CONTENT)) { this.showBadContentDoorhanger(state); - } else if (gPrefService.getBoolPref("privacy.trackingprotection.enabled")) { + } else if (TrackingProtection.enabled) { // We didn't show the shield Services.telemetry.getHistogramById("TRACKING_PROTECTION_SHIELD") .add(0); diff --git a/browser/base/content/test/general/browser_trackingUI_1.js b/browser/base/content/test/general/browser_trackingUI_1.js index 00555c9c549f..9c67f39c9e85 100644 --- a/browser/base/content/test/general/browser_trackingUI_1.js +++ b/browser/base/content/test/general/browser_trackingUI_1.js @@ -2,118 +2,100 @@ * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ -// Test that the Tracking Protection Doorhanger appears -// and has the correct state when tracking content is blocked (Bug 1043801) +// Test that the Tracking Protection section is visible in the Control Center +// and has the correct state for the cases when: +// * A page with no tracking elements is loaded. +// * A page with tracking elements is loaded and they are blocked. +// * A page with tracking elements is loaded and they are not blocked. +// See also Bugs 1175327 and 1043801. -var PREF = "privacy.trackingprotection.enabled"; -var BENIGN_PAGE = "http://tracking.example.org/browser/browser/base/content/test/general/benignPage.html"; -var TRACKING_PAGE = "http://tracking.example.org/browser/browser/base/content/test/general/trackingPage.html"; +let PREF = "privacy.trackingprotection.enabled"; +let BENIGN_PAGE = "http://tracking.example.org/browser/browser/base/content/test/general/benignPage.html"; +let TRACKING_PAGE = "http://tracking.example.org/browser/browser/base/content/test/general/trackingPage.html"; +let TrackingProtection = null; -function testBenignPage(gTestBrowser) -{ - // Make sure the doorhanger does NOT appear - var notification = PopupNotifications.getNotification("bad-content", gTestBrowser); - is(notification, null, "Tracking Content Doorhanger did NOT appear when protection was ON and tracking was NOT present"); +registerCleanupFunction(function() { + TrackingProtection = null; + Services.prefs.clearUserPref(PREF); + gBrowser.removeCurrentTab(); +}); + +function hidden(sel) { + let win = gBrowser.ownerGlobal; + let el = win.document.querySelector(sel); + let display = win.getComputedStyle(el).getPropertyValue("display", null); + return display === "none"; } -function* testTrackingPage(gTestBrowser) -{ - // Make sure the doorhanger appears - var notification = PopupNotifications.getNotification("bad-content", gTestBrowser); - isnot(notification, null, "Tracking Content Doorhanger did appear when protection was ON and tracking was present"); - notification.reshow(); - var notificationElement = PopupNotifications.panel.firstChild; +function testBenignPage() { + ok (!TrackingProtection.content.hasAttribute("block-disabled"), "blocking not disabled"); + ok (!TrackingProtection.content.hasAttribute("block-active"), "blocking is not active"); - // Wait for the method to be attached after showing the popup - yield promiseWaitForCondition(() => { - return notificationElement.disableTrackingContentProtection; - }); - - // Make sure the state of the doorhanger includes blocking tracking elements - ok(notificationElement.isTrackingContentBlocked, - "Tracking Content is being blocked"); - - // Make sure the notification has no trackingblockdisabled attribute - ok(!notificationElement.hasAttribute("trackingblockdisabled"), - "Doorhanger must have no trackingblockdisabled attribute"); + // Make sure that the no tracking elements message appears + ok (!hidden("#tracking-not-detected"), "labelNoTracking is visible"); + ok (hidden("#tracking-loaded"), "labelTrackingLoaded is hidden"); + ok (hidden("#tracking-blocked"), "labelTrackingBlocked is hidden"); } -function* testTrackingPageWhitelisted(gTestBrowser) -{ - // Make sure the doorhanger appears - var notification = PopupNotifications.getNotification("bad-content", gTestBrowser); - isnot(notification, null, "Tracking Content Doorhanger did appear when protection was ON and tracking was present but white-listed"); - notification.reshow(); - var notificationElement = PopupNotifications.panel.firstChild; +function testTrackingPage() { + ok (!TrackingProtection.content.hasAttribute("block-disabled"), "blocking not disabled"); + ok (TrackingProtection.content.hasAttribute("block-active"), "blocking is active"); - // Wait for the method to be attached after showing the popup - yield promiseWaitForCondition(() => { - return notificationElement.disableTrackingContentProtection; - }); - - var notificationElement = PopupNotifications.panel.firstChild; - - // Make sure the state of the doorhanger does NOT include blocking tracking elements - ok(!notificationElement.isTrackingContentBlocked, - "Tracking Content is NOT being blocked"); - - // Make sure the notification has the trackingblockdisabled attribute set to true - is(notificationElement.getAttribute("trackingblockdisabled"), "true", - "Doorhanger must have [trackingblockdisabled='true'] attribute"); + // Make sure that the blocked tracking elements message appears + ok (hidden("#tracking-not-detected"), "labelNoTracking is hidden"); + ok (hidden("#tracking-loaded"), "labelTrackingLoaded is hidden"); + ok (!hidden("#tracking-blocked"), "labelTrackingBlocked is visible"); } -function testTrackingPageOFF(gTestBrowser) -{ - // Make sure the doorhanger does NOT appear - var notification = PopupNotifications.getNotification("bad-content", gTestBrowser); - is(notification, null, "Tracking Content Doorhanger did NOT appear when protection was OFF and tracking was present"); -} +function testTrackingPageWhitelisted() { + ok (TrackingProtection.content.hasAttribute("block-disabled"), "blocking is disabled"); + ok (!TrackingProtection.content.hasAttribute("block-active"), "blocking is not active"); -function testBenignPageOFF(gTestBrowser) -{ - // Make sure the doorhanger does NOT appear - var notification = PopupNotifications.getNotification("bad-content", gTestBrowser); - is(notification, null, "Tracking Content Doorhanger did NOT appear when protection was OFF and tracking was NOT present"); + // Make sure that the blocked tracking elements message appears + ok (hidden("#tracking-not-detected"), "labelNoTracking is hidden"); + ok (!hidden("#tracking-loaded"), "labelTrackingLoaded is visible"); + ok (hidden("#tracking-blocked"), "labelTrackingBlocked is hidden"); } add_task(function* () { - registerCleanupFunction(function() { - Services.prefs.clearUserPref(PREF); - gBrowser.removeCurrentTab(); - }); - yield updateTrackingProtectionDatabase(); let tab = gBrowser.selectedTab = gBrowser.addTab(); - // Enable Tracking Protection + TrackingProtection = gBrowser.ownerGlobal.TrackingProtection; + ok (TrackingProtection, "Functionality is attached to the browser window"); + is (TrackingProtection.enabled, Services.prefs.getBoolPref(PREF), + "The initial enabled value is based on the default pref value"); + + info("Enable Tracking Protection"); Services.prefs.setBoolPref(PREF, true); + ok (TrackingProtection.enabled, "Functionality is enabled after setting the pref"); - // Point tab to a test page NOT containing tracking elements + info("Point tab to a test page NOT containing tracking elements"); yield promiseTabLoadEvent(tab, BENIGN_PAGE); - testBenignPage(gBrowser.getBrowserForTab(tab)); + testBenignPage(); - // Point tab to a test page containing tracking elements + info("Point tab to a test page containing tracking elements"); yield promiseTabLoadEvent(tab, TRACKING_PAGE); - // Tracking content must be blocked - yield testTrackingPage(gBrowser.getBrowserForTab(tab)); + info("Tracking content must be blocked"); + testTrackingPage(); - // Disable Tracking Content Protection for the page (which reloads the page) - PopupNotifications.panel.firstChild.disableTrackingContentProtection(); + info("Disable Tracking Content Protection for the page (which reloads the page)"); + TrackingProtection.disableForCurrentPage(); - // Wait for tab to reload following tracking-protection page white-listing + info("Wait for tab to reload following tracking-protection page white-listing"); yield promiseTabLoadEvent(tab); - // Tracking content must be white-listed (NOT blocked) - yield testTrackingPageWhitelisted(gBrowser.getBrowserForTab(tab)); + info("Tracking content must be white-listed (NOT blocked)"); + testTrackingPageWhitelisted(); - // Re-enable Tracking Content Protection for the page (which reloads the page) - PopupNotifications.panel.firstChild.enableTrackingContentProtection(); + info("Re-enable Tracking Content Protection for the page (which reloads the page)"); + TrackingProtection.enableForCurrentPage(); - // Wait for tab to reload following tracking-protection page white-listing + info("Wait for tab to reload following tracking-protection page white-listing"); yield promiseTabLoadEvent(tab); - // Tracking content must be blocked - yield testTrackingPage(gBrowser.getBrowserForTab(tab)); + info("Tracking content must be blocked"); + testTrackingPage(); }); diff --git a/browser/base/content/test/general/browser_trackingUI_2.js b/browser/base/content/test/general/browser_trackingUI_2.js index a466bb9d4c5c..556558c88268 100644 --- a/browser/base/content/test/general/browser_trackingUI_2.js +++ b/browser/base/content/test/general/browser_trackingUI_2.js @@ -2,45 +2,48 @@ * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ -// Test that the Tracking Protection Doorhanger does not ever appear -// when the feature is off (Bug 1043801) +// Test that the Tracking Protection section is never visible in the +// Control Center when the feature is off. +// See also Bugs 1175327 and 1043801. -var PREF = "privacy.trackingprotection.enabled"; -var BENIGN_PAGE = "http://tracking.example.org/browser/browser/base/content/test/general/benignPage.html"; -var TRACKING_PAGE = "http://tracking.example.org/browser/browser/base/content/test/general/trackingPage.html"; +let PREF = "privacy.trackingprotection.enabled"; +let BENIGN_PAGE = "http://tracking.example.org/browser/browser/base/content/test/general/benignPage.html"; +let TRACKING_PAGE = "http://tracking.example.org/browser/browser/base/content/test/general/trackingPage.html"; +let TrackingProtection = null; -function testTrackingPageOFF(gTestBrowser) -{ - // Make sure the doorhanger does NOT appear - var notification = PopupNotifications.getNotification("bad-content", gTestBrowser); - is(notification, null, "Tracking Content Doorhanger did NOT appear when protection was OFF and tracking was present"); +registerCleanupFunction(function() { + TrackingProtection = null; + Services.prefs.clearUserPref(PREF); + gBrowser.removeCurrentTab(); +}); + +function testTrackingPageOFF() { + ok (TrackingProtection.container.hidden, "The container is hidden"); } -function testBenignPageOFF(gTestBrowser) -{ - // Make sure the doorhanger does NOT appear - var notification = PopupNotifications.getNotification("bad-content", gTestBrowser); - is(notification, null, "Tracking Content Doorhanger did NOT appear when protection was OFF and tracking was NOT present"); +function testBenignPageOFF() { + ok (TrackingProtection.container.hidden, "The container is hidden"); } add_task(function* () { - registerCleanupFunction(function() { - Services.prefs.clearUserPref(PREF); - gBrowser.removeCurrentTab(); - }); - yield updateTrackingProtectionDatabase(); let tab = gBrowser.selectedTab = gBrowser.addTab(); - // Disable Tracking Protection + TrackingProtection = gBrowser.ownerGlobal.TrackingProtection; + ok (TrackingProtection, "Functionality is attached to the browser window"); + is (TrackingProtection.enabled, Services.prefs.getBoolPref(PREF), + "The initial enabled value is based on the default pref value"); + + info ("Disable Tracking Protection"); Services.prefs.setBoolPref(PREF, false); + ok (!TrackingProtection.enabled, "Functionality is disabled after setting the pref"); - // Point tab to a test page containing tracking elements + info ("Point tab to a test page containing tracking elements"); yield promiseTabLoadEvent(tab, TRACKING_PAGE); - testTrackingPageOFF(gBrowser.getBrowserForTab(tab)); + testTrackingPageOFF(); - // Point tab to a test page NOT containing tracking elements + info ("Point tab to a test page NOT containing tracking elements"); yield promiseTabLoadEvent(tab, BENIGN_PAGE); - testBenignPageOFF(gBrowser.getBrowserForTab(tab)); + testBenignPageOFF(); }); diff --git a/browser/base/content/urlbarBindings.xml b/browser/base/content/urlbarBindings.xml index 519101caaafb..b537739befe6 100644 --- a/browser/base/content/urlbarBindings.xml +++ b/browser/base/content/urlbarBindings.xml @@ -2377,10 +2377,6 @@ file, You can obtain one at http://mozilla.org/MPL/2.0/. Services.urlFormatter.formatURLPref("app.support.baseURL") + "tracking-protection"; } - if (Services.prefs.getBoolPref("privacy.trackingprotection.enabled")) { - let histogram = Services.telemetry.getHistogramById("TRACKING_PROTECTION_EVENTS"); - histogram.add(0); - } ]]> @@ -2429,10 +2421,6 @@ file, You can obtain one at http://mozilla.org/MPL/2.0/. // from the tracking protection allowlist. Services.perms.remove(gBrowser.selectedBrowser.currentURI.host, "trackingprotection"); - // Telemetry for enable protection - let histogram = Services.telemetry.getHistogramById( - "TRACKING_PROTECTION_EVENTS"); - histogram.add(2); BrowserReload(); ]]> diff --git a/browser/components/controlcenter/content/panel.inc.xul b/browser/components/controlcenter/content/panel.inc.xul index d91bac14b3e8..c8e97c29aa66 100644 --- a/browser/components/controlcenter/content/panel.inc.xul +++ b/browser/components/controlcenter/content/panel.inc.xul @@ -36,6 +36,42 @@ oncommand="gIdentityHandler.showSubView('security', this)"/> + + + + + + + + + + + + + diff --git a/browser/components/preferences/in-content/tests/browser_privacypane_5.js b/browser/components/preferences/in-content/tests/browser_privacypane_5.js index 1a2b572d2d59..4f549bb3430a 100644 --- a/browser/components/preferences/in-content/tests/browser_privacypane_5.js +++ b/browser/components/preferences/in-content/tests/browser_privacypane_5.js @@ -21,7 +21,7 @@ function test() { ]; if (AppConstants.NIGHTLY_BUILD) - tests.push(test_locbar_suggestion_retention("searches", true)), + tests.push(test_locbar_suggestion_retention("searches", true)); run_test_subset(tests); } diff --git a/browser/components/preferences/tests/browser_privacypane_5.js b/browser/components/preferences/tests/browser_privacypane_5.js index 1a2b572d2d59..4f549bb3430a 100644 --- a/browser/components/preferences/tests/browser_privacypane_5.js +++ b/browser/components/preferences/tests/browser_privacypane_5.js @@ -21,7 +21,7 @@ function test() { ]; if (AppConstants.NIGHTLY_BUILD) - tests.push(test_locbar_suggestion_retention("searches", true)), + tests.push(test_locbar_suggestion_retention("searches", true)); run_test_subset(tests); } diff --git a/browser/devtools/debugger/debugger-controller.js b/browser/devtools/debugger/debugger-controller.js index 54ae20dc2c24..5010eb685941 100644 --- a/browser/devtools/debugger/debugger-controller.js +++ b/browser/devtools/debugger/debugger-controller.js @@ -455,6 +455,7 @@ function Workers() { this._onWorkerListChanged = this._onWorkerListChanged.bind(this); this._onWorkerFreeze = this._onWorkerFreeze.bind(this); this._onWorkerThaw = this._onWorkerThaw.bind(this); + this._onWorkerSelect = this._onWorkerSelect.bind(this); } Workers.prototype = { @@ -476,6 +477,10 @@ Workers.prototype = { }, _updateWorkerList: function () { + if (!this._tabClient.listWorkers) { + return; + } + this._tabClient.listWorkers((response) => { let workerActors = new Set(); for (let worker of response.workers) { @@ -516,6 +521,13 @@ Workers.prototype = { _onWorkerThaw: function (type, packet) { let workerClient = this._workerClients.get(packet.from); DebuggerView.Workers.addWorker(packet.from, workerClient.url); + }, + + _onWorkerSelect: function (workerActor) { + let workerClient = this._workerClients.get(workerActor); + gDevTools.showToolbox(devtools.TargetFactory.forWorker(workerClient), + "jsdebugger", + devtools.Toolbox.HostType.WINDOW); } }; diff --git a/browser/devtools/debugger/views/workers-view.js b/browser/devtools/debugger/views/workers-view.js index 808cac63f719..3b7b9db24d89 100644 --- a/browser/devtools/debugger/views/workers-view.js +++ b/browser/devtools/debugger/views/workers-view.js @@ -3,7 +3,9 @@ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ "use strict"; -function WorkersView() {} +function WorkersView() { + this._onWorkerSelect = this._onWorkerSelect.bind(this); +} WorkersView.prototype = Heritage.extend(WidgetMethods, { initialize: function () { @@ -17,6 +19,7 @@ WorkersView.prototype = Heritage.extend(WidgetMethods, { showArrows: true, }); this.emptyText = L10N.getStr("noWorkersText"); + this.widget.addEventListener("select", this._onWorkerSelect, false); }, addWorker: function (actor, name) { @@ -30,6 +33,13 @@ WorkersView.prototype = Heritage.extend(WidgetMethods, { removeWorker: function (actor) { this.remove(this.getItemByValue(actor)); + }, + + _onWorkerSelect: function () { + if (this.selectedItem !== null) { + DebuggerController.Workers._onWorkerSelect(this.selectedItem.value); + this.selectedItem = null; + } } }); diff --git a/browser/devtools/framework/target.js b/browser/devtools/framework/target.js index ee9cb1ac0b12..13c9177860d4 100644 --- a/browser/devtools/framework/target.js +++ b/browser/devtools/framework/target.js @@ -60,6 +60,15 @@ exports.TargetFactory = { return targetPromise; }, + forWorker: function TF_forWorker(workerClient) { + let target = targets.get(workerClient); + if (target == null) { + target = new WorkerTarget(workerClient); + targets.set(workerClient, target); + } + return target; + }, + /** * Creating a target for a tab that is being closed is a problem because it * allows a leak as a result of coming after the close event which normally @@ -799,3 +808,64 @@ WindowTarget.prototype = { return 'WindowTarget:' + this.window; }, }; + +function WorkerTarget(workerClient) { + EventEmitter.decorate(this); + this._workerClient = workerClient; +} + +/** + * A WorkerTarget represents a worker. Unlike TabTarget, which can represent + * either a local or remote tab, WorkerTarget always represents a remote worker. + * Moreover, unlike TabTarget, which is constructed with a placeholder object + * for remote tabs (from which a TabClient can then be lazily obtained), + * WorkerTarget is constructed with a WorkerClient directly. + * + * The reason for this is that in order to get notifications when a worker + * closes/freezes/thaws, the UI needs to attach to each worker anyway, so by + * the time a WorkerTarget for a given worker is created, a WorkerClient for + * that worker will already be available. Consequently, there is no need to + * obtain a WorkerClient lazily. + * + * WorkerClient is designed to mimic the interface of TabClient as closely as + * possible. This allows us to debug workers as if they were ordinary tabs, + * requiring only minimal changes to the rest of the frontend. + */ +WorkerTarget.prototype = { + get isRemote() { + return true; + }, + + get isTabActor() { + return true; + }, + + get form() { + return { + from: this._workerClient.actor, + type: "attached", + isFrozen: this._workerClient.isFrozen, + url: this._workerClient.url + }; + }, + + get activeTab() { + return this._workerClient; + }, + + get client() { + return this._workerClient.client; + }, + + destroy: function () {}, + + hasActor: function (name) { + return false; + }, + + getTrait: function (name) { + return undefined; + }, + + makeRemote: function () {} +}; diff --git a/browser/locales/en-US/chrome/browser/browser.dtd b/browser/locales/en-US/chrome/browser/browser.dtd index d34cc4a4caac..2c8089b2711a 100644 --- a/browser/locales/en-US/chrome/browser/browser.dtd +++ b/browser/locales/en-US/chrome/browser/browser.dtd @@ -768,6 +768,15 @@ you can use these alternative items. Otherwise, their values should be empty. - + + + + + + + + + diff --git a/browser/modules/NewTabURL.jsm b/browser/modules/NewTabURL.jsm index 9e78efaae095..f673bbc48564 100644 --- a/browser/modules/NewTabURL.jsm +++ b/browser/modules/NewTabURL.jsm @@ -10,6 +10,8 @@ let Cu = Components.utils; this.EXPORTED_SYMBOLS = [ "NewTabURL" ]; +Components.utils.import("resource://gre/modules/Services.jsm"); + this.NewTabURL = { _url: "about:newtab", _overridden: false, @@ -25,10 +27,12 @@ this.NewTabURL = { override: function(newURL) { this._url = newURL; this._overridden = true; + Services.obs.notifyObservers(null, "newtab-url-changed", this._url); }, reset: function() { this._url = "about:newtab"; this._overridden = false; + Services.obs.notifyObservers(null, "newtab-url-changed", this._url); } }; diff --git a/browser/modules/test/xpcshell/test_NewTabURL.js b/browser/modules/test/xpcshell/test_NewTabURL.js index e5ccd9ecfc79..aad233729d1d 100644 --- a/browser/modules/test/xpcshell/test_NewTabURL.js +++ b/browser/modules/test/xpcshell/test_NewTabURL.js @@ -4,14 +4,30 @@ "use strict"; Components.utils.import("resource:///modules/NewTabURL.jsm"); +Components.utils.import("resource://gre/modules/Services.jsm"); -function run_test() { +add_task(function* () { Assert.equal(NewTabURL.get(), "about:newtab", "Default newtab URL should be about:newtab"); let url = "http://example.com/"; + let notificationPromise = promiseNewtabURLNotification(url); NewTabURL.override(url); + yield notificationPromise; Assert.ok(NewTabURL.overridden, "Newtab URL should be overridden"); Assert.equal(NewTabURL.get(), url, "Newtab URL should be the custom URL"); + + notificationPromise = promiseNewtabURLNotification("about:newtab"); NewTabURL.reset(); + yield notificationPromise; Assert.ok(!NewTabURL.overridden, "Newtab URL should not be overridden"); Assert.equal(NewTabURL.get(), "about:newtab", "Newtab URL should be the about:newtab"); +}); + +function promiseNewtabURLNotification(aNewURL) { + return new Promise(resolve => { + Services.obs.addObserver(function observer(aSubject, aTopic, aData) { + Services.obs.removeObserver(observer, aTopic); + Assert.equal(aData, aNewURL, "Data for newtab-url-changed notification should be new URL."); + resolve(); + }, "newtab-url-changed", false); + }); } diff --git a/browser/themes/linux/jar.mn b/browser/themes/linux/jar.mn index 6c09a4b34905..b4a65d5e91a3 100644 --- a/browser/themes/linux/jar.mn +++ b/browser/themes/linux/jar.mn @@ -150,6 +150,8 @@ browser.jar: skin/classic/browser/controlcenter/conn-secure-dv.svg (../shared/controlcenter/conn-secure-dv.svg) skin/classic/browser/controlcenter/conn-secure-ev.svg (../shared/controlcenter/conn-secure-ev.svg) skin/classic/browser/controlcenter/permissions.svg (../shared/controlcenter/permissions.svg) + skin/classic/browser/controlcenter/tracking-protection.svg (../shared/controlcenter/tracking-protection.svg) + skin/classic/browser/controlcenter/tracking-protection-disabled.svg (../shared/controlcenter/tracking-protection-disabled.svg) skin/classic/browser/customizableui/background-noise-toolbar.png (customizableui/background-noise-toolbar.png) skin/classic/browser/customizableui/customize-illustration.png (../shared/customizableui/customize-illustration.png) skin/classic/browser/customizableui/customize-illustration-rtl.png (../shared/customizableui/customize-illustration-rtl.png) diff --git a/browser/themes/osx/jar.mn b/browser/themes/osx/jar.mn index e5c7749c10e8..5d4d0389c1f9 100644 --- a/browser/themes/osx/jar.mn +++ b/browser/themes/osx/jar.mn @@ -194,6 +194,8 @@ browser.jar: skin/classic/browser/controlcenter/conn-secure-dv.svg (../shared/controlcenter/conn-secure-dv.svg) skin/classic/browser/controlcenter/conn-secure-ev.svg (../shared/controlcenter/conn-secure-ev.svg) skin/classic/browser/controlcenter/permissions.svg (../shared/controlcenter/permissions.svg) + skin/classic/browser/controlcenter/tracking-protection.svg (../shared/controlcenter/tracking-protection.svg) + skin/classic/browser/controlcenter/tracking-protection-disabled.svg (../shared/controlcenter/tracking-protection-disabled.svg) skin/classic/browser/customizableui/background-noise-toolbar.png (customizableui/background-noise-toolbar.png) skin/classic/browser/customizableui/customize-titleBar-toggle.png (customizableui/customize-titleBar-toggle.png) skin/classic/browser/customizableui/customize-titleBar-toggle@2x.png (customizableui/customize-titleBar-toggle@2x.png) diff --git a/browser/themes/shared/controlcenter/panel.inc.css b/browser/themes/shared/controlcenter/panel.inc.css index ca456a1e3846..371b6bdbec46 100644 --- a/browser/themes/shared/controlcenter/panel.inc.css +++ b/browser/themes/shared/controlcenter/panel.inc.css @@ -55,7 +55,8 @@ #identity-popup-securityView, #identity-popup-security-content, -#identity-popup-permissions-content { +#identity-popup-permissions-content, +#tracking-protection-content { padding: 0.75em 0 1em; -moz-padding-start: calc(2em + 24px); -moz-padding-end: 1em; @@ -66,7 +67,8 @@ #identity-popup-securityView:-moz-locale-dir(rtl), #identity-popup-security-content:-moz-locale-dir(rtl), -#identity-popup-permissions-content:-moz-locale-dir(rtl) { +#identity-popup-permissions-content:-moz-locale-dir(rtl), +#tracking-protection-content:-moz-locale-dir(rtl) { background-position: calc(100% - 1em) 1em; } @@ -199,6 +201,30 @@ margin-top: 1em; } +/* TRACKING PROTECTION */ + +#tracking-protection-content { + background-image: url("chrome://browser/skin/controlcenter/tracking-protection.svg"); +} + +#tracking-protection-content[block-disabled] { + background-image: url("chrome://browser/skin/controlcenter/tracking-protection-disabled.svg"); +} + +#tracking-actions { + margin: 1em 0 0; +} + +#tracking-protection-content[block-active] > #tracking-not-detected, +#tracking-protection-content[block-disabled] > #tracking-not-detected, +#tracking-protection-content:not([block-active]) > #tracking-blocked, +#tracking-protection-content:not([block-active]) #tracking-action-unblock, +#tracking-protection-content:not([block-disabled]) > #tracking-loaded, +#tracking-protection-content:not([block-disabled]) #tracking-action-block, +#tracking-protection-content:not([block-active]):not([block-disabled]) > #tracking-actions { + display: none; +} + /* PERMISSIONS */ #identity-popup-permissions-content { diff --git a/browser/themes/shared/controlcenter/tracking-protection-disabled.svg b/browser/themes/shared/controlcenter/tracking-protection-disabled.svg new file mode 100644 index 000000000000..955dfe23d0c3 --- /dev/null +++ b/browser/themes/shared/controlcenter/tracking-protection-disabled.svg @@ -0,0 +1,24 @@ + + + + + + + + + + + + + + + + + + diff --git a/browser/themes/shared/controlcenter/tracking-protection.svg b/browser/themes/shared/controlcenter/tracking-protection.svg new file mode 100644 index 000000000000..bf3f7806967f --- /dev/null +++ b/browser/themes/shared/controlcenter/tracking-protection.svg @@ -0,0 +1,22 @@ + + + + + + + + + + + + + + + + diff --git a/browser/themes/windows/jar.mn b/browser/themes/windows/jar.mn index df3b34dcd658..31abf45210e1 100644 --- a/browser/themes/windows/jar.mn +++ b/browser/themes/windows/jar.mn @@ -197,6 +197,8 @@ browser.jar: skin/classic/browser/controlcenter/conn-secure-dv.svg (../shared/controlcenter/conn-secure-dv.svg) skin/classic/browser/controlcenter/conn-secure-ev.svg (../shared/controlcenter/conn-secure-ev.svg) skin/classic/browser/controlcenter/permissions.svg (../shared/controlcenter/permissions.svg) + skin/classic/browser/controlcenter/tracking-protection.svg (../shared/controlcenter/tracking-protection.svg) + skin/classic/browser/controlcenter/tracking-protection-disabled.svg (../shared/controlcenter/tracking-protection-disabled.svg) skin/classic/browser/customizableui/background-noise-toolbar.png (customizableui/background-noise-toolbar.png) skin/classic/browser/customizableui/customizeFavicon.ico (../shared/customizableui/customizeFavicon.ico) skin/classic/browser/customizableui/customize-illustration.png (../shared/customizableui/customize-illustration.png) diff --git a/mobile/android/base/android-services.mozbuild b/mobile/android/base/android-services.mozbuild index ac6af72c16d5..bec1fac3b0e3 100644 --- a/mobile/android/base/android-services.mozbuild +++ b/mobile/android/base/android-services.mozbuild @@ -863,6 +863,7 @@ sync_java_files = [ 'fxa/activities/FxAccountStatusFragment.java', 'fxa/activities/FxAccountUpdateCredentialsActivity.java', 'fxa/activities/FxAccountVerifiedAccountActivity.java', + 'fxa/activities/PicassoPreferenceIconTarget.java', 'fxa/authenticator/AccountPickler.java', 'fxa/authenticator/AndroidFxAccount.java', 'fxa/authenticator/FxAccountAuthenticator.java', diff --git a/mobile/android/base/fxa/FxAccountConstants.java b/mobile/android/base/fxa/FxAccountConstants.java index c3e3d063741c..b5aed319cb77 100644 --- a/mobile/android/base/fxa/FxAccountConstants.java +++ b/mobile/android/base/fxa/FxAccountConstants.java @@ -24,7 +24,7 @@ public class FxAccountConstants { public static final String STAGE_PROFILE_SERVER_ENDPOINT = "https://latest.dev.lcip.org/profile/v1"; // Action to update on cached profile information. - public static final String ACCOUNT_PROFILE_AVATAR_UPDATED_ACTION = "org.mozilla.gecko.fxa.profile.cached"; + public static final String ACCOUNT_PROFILE_JSON_UPDATED_ACTION = "org.mozilla.gecko.fxa.profile.JSON.updated"; // You must be at least 13 years old, on the day of creation, to create a Firefox Account. public static final int MINIMUM_AGE_TO_CREATE_AN_ACCOUNT = 13; diff --git a/mobile/android/base/fxa/activities/FxAccountStatusFragment.java b/mobile/android/base/fxa/activities/FxAccountStatusFragment.java index e90766a99aac..709163cbb2b3 100644 --- a/mobile/android/base/fxa/activities/FxAccountStatusFragment.java +++ b/mobile/android/base/fxa/activities/FxAccountStatusFragment.java @@ -50,6 +50,8 @@ import android.text.TextUtils; import android.text.format.DateUtils; import android.widget.Toast; +import com.squareup.picasso.Picasso; +import com.squareup.picasso.Target; /** * A fragment that displays the status of an AndroidFxAccount. @@ -140,13 +142,11 @@ public class FxAccountStatusFragment // Runnable to update last synced time. protected Runnable lastSyncedTimeUpdateRunnable; - // Runnable to retry fetching profile information. - protected Runnable profileFetchRunnable; - // Broadcast Receiver to update profile Information. protected FxAccountProfileInformationReceiver accountProfileInformationReceiver; protected final InnerSyncStatusDelegate syncStatusDelegate = new InnerSyncStatusDelegate(); + private Target profileAvatarTarget; protected Preference ensureFindPreference(String key) { Preference preference = findPreference(key); @@ -485,6 +485,18 @@ public class FxAccountStatusFragment // register/unregister calls. FxAccountSyncStatusHelper.getInstance().startObserving(syncStatusDelegate); + if (AppConstants.MOZ_ANDROID_FIREFOX_ACCOUNT_PROFILES) { + // Register a local broadcast receiver to get profile cached notification. + final IntentFilter intentFilter = new IntentFilter(); + intentFilter.addAction(FxAccountConstants.ACCOUNT_PROFILE_JSON_UPDATED_ACTION); + accountProfileInformationReceiver = new FxAccountProfileInformationReceiver(); + LocalBroadcastManager.getInstance(getActivity()).registerReceiver(accountProfileInformationReceiver, intentFilter); + + // profilePreference is set during onCreate, so it's definitely not null here. + final float cornerRadius = getResources().getDimension(R.dimen.fxaccount_profile_image_width) / 2; + profileAvatarTarget = new PicassoPreferenceIconTarget(getResources(), profilePreference, cornerRadius); + } + refresh(); } @@ -498,14 +510,15 @@ public class FxAccountStatusFragment handler.removeCallbacks(lastSyncedTimeUpdateRunnable); } - if (profileFetchRunnable != null) { - handler.removeCallbacks(profileFetchRunnable); - } - // Focus lost, unregister broadcast receiver. if (accountProfileInformationReceiver != null) { LocalBroadcastManager.getInstance(getActivity()).unregisterReceiver(accountProfileInformationReceiver); } + + if (profileAvatarTarget != null) { + Picasso.with(getActivity()).cancelRequest(profileAvatarTarget); + profileAvatarTarget = null; + } } protected void hardRefresh() { @@ -606,53 +619,60 @@ public class FxAccountStatusFragment return; } - final ExtendedJSONObject cachedProfileJSON = fxAccount.getCachedProfileJSON(); - if (cachedProfileJSON != null) { - // Update profile information from the cached Json. - updateProfileInformation(cachedProfileJSON); + final ExtendedJSONObject profileJSON = fxAccount.getProfileJSON(); + if (profileJSON == null) { + // Update the profile title with email as the fallback. + // Profile icon by default use the default avatar as the fallback. + profilePreference.setTitle(fxAccount.getEmail()); return; } - // Update the profile title with email as the fallback. - // Profile icon by default use the default avatar as the fallback. - profilePreference.setTitle(fxAccount.getEmail()); - - // Register a local broadcast receiver to get profile cached notification. - final IntentFilter intentFilter = new IntentFilter(); - intentFilter.addAction(FxAccountConstants.ACCOUNT_PROFILE_AVATAR_UPDATED_ACTION); - accountProfileInformationReceiver = new FxAccountProfileInformationReceiver(); - LocalBroadcastManager.getInstance(getActivity()).registerReceiver(accountProfileInformationReceiver, intentFilter); - - // Fetch the profile from the server. - fxAccount.maybeUpdateProfileJSON(false); - - // Schedule an runnable to retry fetching profile. - profileFetchRunnable = new ProfileFetchUpdateRunnable(); - handler.postDelayed(profileFetchRunnable, PROFILE_FETCH_RETRY_INTERVAL_IN_MILLISECONDS); + updateProfileInformation(profileJSON); } /** * Update profile information from json on UI thread. * - * @param profileJson json fetched from server. + * @param profileJSON json fetched from server. */ - protected void updateProfileInformation(final ExtendedJSONObject profileJson) { - // Remove the scheduled runnable for fetching the profile information. - if (profileFetchRunnable != null) { - handler.removeCallbacks(profileFetchRunnable); + protected void updateProfileInformation(final ExtendedJSONObject profileJSON) { + // View changes must always be done on UI thread. + ThreadUtils.assertOnUiThread(); + + FxAccountUtils.pii(LOG_TAG, "Profile JSON is: " + profileJSON.toJSONString()); + + final String userName = profileJSON.getString(FxAccountConstants.KEY_PROFILE_JSON_USERNAME); + // Update the profile username and email if available. + if (!TextUtils.isEmpty(userName)) { + profilePreference.setTitle(userName); + profilePreference.setSummary(fxAccount.getEmail()); + } else { + profilePreference.setTitle(fxAccount.getEmail()); } - // Read the profile information from json and Update the UI elements. - ThreadUtils.postToUiThread(new Runnable() { - @Override - public void run() { - // Icon update from java is not supported prior to API 11, skip the avatar update for older device. - if (AppConstants.Versions.feature11Plus) { - profilePreference.setIcon(getResources().getDrawable(R.drawable.sync_avatar_default)); - } - profilePreference.setTitle(fxAccount.getAndroidAccount().name); - } - }); + // Icon update from java is not supported prior to API 11, skip the avatar image fetch and update for older device. + if (!AppConstants.Versions.feature11Plus) { + Logger.info(LOG_TAG, "Skipping profile image fetch for older pre-API 11 devices."); + return; + } + + // Avatar URI empty, skip profile image fetch. + final String avatarURI = profileJSON.getString(FxAccountConstants.KEY_PROFILE_JSON_AVATAR); + if (TextUtils.isEmpty(avatarURI)) { + Logger.info(LOG_TAG, "AvatarURI is empty, skipping profile image fetch."); + return; + } + + // Using noPlaceholder would avoid a pop of the default image, but it's not available in the version of Picasso + // we ship in the tree. + Picasso + .with(getActivity()) + .load(avatarURI) + .centerInside() + .resizeDimen(R.dimen.fxaccount_profile_image_width, R.dimen.fxaccount_profile_image_height) + .placeholder(R.drawable.sync_avatar_default) + .error(R.drawable.sync_avatar_default) + .into(profileAvatarTarget); } private void scheduleAndUpdateLastSyncedTime() { @@ -830,26 +850,24 @@ public class FxAccountStatusFragment } } - /** - * The Runnable that schedules a future to fetch profile information. - */ - protected class ProfileFetchUpdateRunnable implements Runnable { - @Override - public void run() { - updateProfileInformation(); - } - } - /** * Broadcast receiver to receive updates for the cached profile action. */ public class FxAccountProfileInformationReceiver extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { - if (intent.getAction().equals(FxAccountConstants.ACCOUNT_PROFILE_AVATAR_UPDATED_ACTION)) { - // We should have a cached profile json here. - updateProfileInformation(fxAccount.getCachedProfileJSON()); + if (!intent.getAction().equals(FxAccountConstants.ACCOUNT_PROFILE_JSON_UPDATED_ACTION)) { + return; } + + Logger.info(LOG_TAG, "Profile avatar cache update action broadcast received."); + // Update the UI from cached profile json on the main thread. + getActivity().runOnUiThread(new Runnable() { + @Override + public void run() { + updateProfileInformation(); + } + }); } } diff --git a/mobile/android/base/fxa/activities/PicassoPreferenceIconTarget.java b/mobile/android/base/fxa/activities/PicassoPreferenceIconTarget.java new file mode 100644 index 000000000000..bc15f085ae8e --- /dev/null +++ b/mobile/android/base/fxa/activities/PicassoPreferenceIconTarget.java @@ -0,0 +1,76 @@ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +package org.mozilla.gecko.fxa.activities; + +import android.content.res.Resources; +import android.graphics.Bitmap; +import android.graphics.drawable.BitmapDrawable; +import android.graphics.drawable.Drawable; +import android.preference.Preference; +import android.support.v4.graphics.drawable.RoundedBitmapDrawable; +import android.support.v4.graphics.drawable.RoundedBitmapDrawableFactory; +import com.squareup.picasso.Picasso; +import com.squareup.picasso.Target; +import org.mozilla.gecko.AppConstants; + +/** + * A Picasso Target that updates a preference icon. + * + * Nota bene: Android grew support for updating preference icons programatically + * only in API 11. This class silently ignores requests before API 11. + */ +public class PicassoPreferenceIconTarget implements Target { + private final Preference preference; + private final Resources resources; + private final float cornerRadius; + + public PicassoPreferenceIconTarget(Resources resources, Preference preference) { + this(resources, preference, 0); + } + + public PicassoPreferenceIconTarget(Resources resources, Preference preference, float cornerRadius) { + this.resources = resources; + this.preference = preference; + this.cornerRadius = cornerRadius; + } + + @Override + public void onBitmapLoaded(Bitmap bitmap, Picasso.LoadedFrom from) { + // Updating icons from Java is not supported prior to API 11. + if (!AppConstants.Versions.feature11Plus) { + return; + } + + final Drawable drawable; + if (cornerRadius > 0) { + final RoundedBitmapDrawable roundedBitmapDrawable; + roundedBitmapDrawable = RoundedBitmapDrawableFactory.create(resources, bitmap); + roundedBitmapDrawable.setCornerRadius(cornerRadius); + roundedBitmapDrawable.setAntiAlias(true); + drawable = roundedBitmapDrawable; + } else { + drawable = new BitmapDrawable(resources, bitmap); + } + preference.setIcon(drawable); + } + + @Override + public void onBitmapFailed(Drawable errorDrawable) { + // Updating icons from Java is not supported prior to API 11. + if (!AppConstants.Versions.feature11Plus) { + return; + } + preference.setIcon(errorDrawable); + } + + @Override + public void onPrepareLoad(Drawable placeHolderDrawable) { + // Updating icons from Java is not supported prior to API 11. + if (!AppConstants.Versions.feature11Plus) { + return; + } + preference.setIcon(placeHolderDrawable); + } +} diff --git a/mobile/android/base/fxa/authenticator/AndroidFxAccount.java b/mobile/android/base/fxa/authenticator/AndroidFxAccount.java index c056f211ed57..5232b7676ecf 100644 --- a/mobile/android/base/fxa/authenticator/AndroidFxAccount.java +++ b/mobile/android/base/fxa/authenticator/AndroidFxAccount.java @@ -69,12 +69,12 @@ public class AndroidFxAccount { public static final String ACCOUNT_KEY_TOKEN_SERVER = "tokenServerURI"; // Sync-specific. public static final String ACCOUNT_KEY_DESCRIPTOR = "descriptor"; - public static final String ACCOUNT_KEY_PROFILE_AVATAR = "avatar"; public static final int CURRENT_BUNDLE_VERSION = 2; public static final String BUNDLE_KEY_BUNDLE_VERSION = "version"; public static final String BUNDLE_KEY_STATE_LABEL = "stateLabel"; public static final String BUNDLE_KEY_STATE = "state"; + public static final String BUNDLE_KEY_PROFILE_JSON = "profile"; // Account authentication token type for fetching account profile. public static final String PROFILE_OAUTH_TOKEN_TYPE = "oauth::profile"; @@ -105,13 +105,6 @@ public class AndroidFxAccount { } private static final String PREF_KEY_LAST_SYNCED_TIMESTAMP = "lastSyncedTimestamp"; - public static final String PREF_KEY_LAST_PROFILE_FETCH_TIME = "lastProfilefetchTime"; - public static final String PREF_KEY_NUMBER_OF_PROFILE_FETCH = "numProfileFetch"; - - // Max wait time between successful profile avatar network fetch. - public static final long PROFILE_FETCH_RETRY_BACKOFF_DELTA_IN_MILLISECONDS = 24 * 60 * 60 * 1000; - // Max attempts allowed for retrying profile avatar network fetch. - public static final int MAX_PROFILE_FETCH_RETRIES = 5; protected final Context context; protected final AccountManager accountManager; @@ -127,7 +120,6 @@ public class AndroidFxAccount { */ protected static final ConcurrentHashMap perAccountBundleCache = new ConcurrentHashMap<>(); - private ExtendedJSONObject profileJson; public static void invalidateCaches() { perAccountBundleCache.clear(); @@ -667,39 +659,17 @@ public class AndroidFxAccount { return intent; } - private void setLastProfileFetchTimestampAndAttempts(long now, int attempts) { - try { - getSyncPrefs().edit().putLong(PREF_KEY_LAST_PROFILE_FETCH_TIME, now).commit(); - getSyncPrefs().edit().putInt(PREF_KEY_NUMBER_OF_PROFILE_FETCH, attempts); - } catch (Exception e) { - Logger.warn(LOG_TAG, "Got exception setting last profile fetch time & attempts; ignoring.", e); - } - } - - private long getLastProfileFetchTimestamp() { - final long neverFetched = -1L; - try { - return getSyncPrefs().getLong(PREF_KEY_LAST_PROFILE_FETCH_TIME, neverFetched); - } catch (Exception e) { - Logger.warn(LOG_TAG, "Got exception getting last profile fetch time; ignoring.", e); - return neverFetched; - } - } - - private int getNumberOfProfileFetch() { - final int neverFetched = 0; - try { - return getSyncPrefs().getInt(PREF_KEY_NUMBER_OF_PROFILE_FETCH, neverFetched); - } catch (Exception e) { - Logger.warn(LOG_TAG, "Got exception getting number of profile fetch; ignoring.", e); - return neverFetched; - } - } - - private boolean canScheduleProfileFetch() { - final int attempts = getNumberOfProfileFetch(); - final long delta = System.currentTimeMillis() - getLastProfileFetchTimestamp(); - return delta > PROFILE_FETCH_RETRY_BACKOFF_DELTA_IN_MILLISECONDS || attempts < MAX_PROFILE_FETCH_RETRIES; + /** + * Create an intent announcing that the profile JSON attached to this Firefox Account has been updated. + *

+ * It is not guaranteed that the profile JSON has changed. + * + * @return Intent to broadcast. + */ + private Intent makeProfileJSONUpdatedIntent() { + final Intent intent = new Intent(); + intent.setAction(FxAccountConstants.ACCOUNT_PROFILE_JSON_UPDATED_ACTION); + return intent; } public void setLastSyncedTimestamp(long now) { @@ -755,60 +725,31 @@ public class AndroidFxAccount { ContentResolver.setIsSyncable(account, BrowserContract.READING_LIST_AUTHORITY, 1); } - // Helper function to create intent for profile avatar updated event. - private Intent getProfileAvatarUpdatedIntent() { - final Intent profileCachedIntent = new Intent(); - profileCachedIntent.setAction(FxAccountConstants.ACCOUNT_PROFILE_AVATAR_UPDATED_ACTION); - return profileCachedIntent; - } - /** - * Returns the cached profile JSON object if available or null. + * Returns the current profile JSON if available, or null. * - * @return profile JSON Object. + * @return profile JSON object. */ - public ExtendedJSONObject getCachedProfileJSON() { - if (profileJson == null) { - // Try to retrieve and parse the json string from account manager. - final String profileJsonString = accountManager.getUserData(account, ACCOUNT_KEY_PROFILE_AVATAR); - if (profileJsonString != null) { - Logger.info(LOG_TAG, "Cached Profile information retrieved from AccountManager."); - try { - profileJson = ExtendedJSONObject.parseJSONObject(profileJsonString); - } catch (Exception e) { - Logger.error(LOG_TAG, "Failed to parse profile json; ignoring.", e); - } - } + public ExtendedJSONObject getProfileJSON() { + final String profileString = getBundleData(BUNDLE_KEY_PROFILE_JSON); + if (profileString == null) { + return null; } - return profileJson; + + try { + return new ExtendedJSONObject(profileString); + } catch (Exception e) { + Logger.error(LOG_TAG, "Failed to parse profile JSON; ignoring and returning null.", e); + } + return null; } /** - * Fetches the profile json from the server and updates the local cache. - * + * Fetch the profile JSON associated to the underlying Firefox Account from the server and update the local store. *

- * On successful fetch and cache, LocalBroadcastManager is used to notify the receivers asynchronously. - *

- * - * @param isForceFetch boolean to isForceFetch fetch from the server. + * The LocalBroadcastManager is used to notify the receivers asynchronously after a successful fetch. */ - public void maybeUpdateProfileJSON(final boolean isForceFetch) { - final ExtendedJSONObject profileJson = getCachedProfileJSON(); - final Intent profileAvatarUpdatedIntent = getProfileAvatarUpdatedIntent(); - - if (!isForceFetch && profileJson != null && !profileJson.keySet().isEmpty()) { - // Second line of defense, cache may have been updated in between. - Logger.info(LOG_TAG, "Profile already cached."); - LocalBroadcastManager.getInstance(context).sendBroadcast(profileAvatarUpdatedIntent); - return; - } - - if (!isForceFetch && !canScheduleProfileFetch()) { - // Rate limiting repeated attempts to fetch the profile information. - Logger.info(LOG_TAG, "Too many attempts to fetch the profile information."); - return; - } - + public void fetchProfileJSON() { ThreadUtils.postToBackgroundThread(new Runnable() { @Override public void run() { @@ -828,24 +769,15 @@ public class AndroidFxAccount { final Intent intent = new Intent(context, FxAccountProfileService.class); intent.putExtra(FxAccountProfileService.KEY_AUTH_TOKEN, authToken); intent.putExtra(FxAccountProfileService.KEY_PROFILE_SERVER_URI, getProfileServerURI()); - intent.putExtra(FxAccountProfileService.KEY_RESULT_RECEIVER, new ProfileResultReceiver(profileAvatarUpdatedIntent)); + intent.putExtra(FxAccountProfileService.KEY_RESULT_RECEIVER, new ProfileResultReceiver(new Handler())); context.startService(intent); - - // Update the profile fetch time and attempts, resetting the attempts if last fetch was over a day old. - final int attempts = getNumberOfProfileFetch(); - final long now = System.currentTimeMillis(); - final long delta = now - getLastProfileFetchTimestamp(); - setLastProfileFetchTimestampAndAttempts(now, delta < PROFILE_FETCH_RETRY_BACKOFF_DELTA_IN_MILLISECONDS ? attempts + 1 : 1); } }); } private class ProfileResultReceiver extends ResultReceiver { - private final Intent profileAvatarUpdatedIntent; - - public ProfileResultReceiver(Intent broadcastIntent) { - super(new Handler()); - this.profileAvatarUpdatedIntent = broadcastIntent; + public ProfileResultReceiver(Handler handler) { + super(handler); } @Override @@ -853,21 +785,17 @@ public class AndroidFxAccount { super.onReceiveResult(resultCode, bundle); switch (resultCode) { case Activity.RESULT_OK: - try { - final String resultData = bundle.getString(FxAccountProfileService.KEY_RESULT_STRING); - profileJson = ExtendedJSONObject.parseJSONObject(resultData); - accountManager.setUserData(account, ACCOUNT_KEY_PROFILE_AVATAR, resultData); - Logger.pii(LOG_TAG, "Profile fetch successful." + resultData); - LocalBroadcastManager.getInstance(context).sendBroadcast(profileAvatarUpdatedIntent); - } catch (Exception e) { - Logger.error(LOG_TAG, "Failed to parse profile json; ignoring.", e); - } + final String resultData = bundle.getString(FxAccountProfileService.KEY_RESULT_STRING); + updateBundleValues(BUNDLE_KEY_PROFILE_JSON, resultData); + Logger.info(LOG_TAG, "Profile JSON fetch succeeeded!"); + FxAccountUtils.pii(LOG_TAG, "Profile JSON fetch returned: " + resultData); + LocalBroadcastManager.getInstance(context).sendBroadcast(makeDeletedAccountIntent()); break; case Activity.RESULT_CANCELED: - Logger.warn(LOG_TAG, "Failed to fetch profile; ignoring."); + Logger.warn(LOG_TAG, "Failed to fetch profile JSON; ignoring."); break; default: - Logger.warn(LOG_TAG, "Invalid Result code received; ignoring."); + Logger.warn(LOG_TAG, "Invalid result code received; ignoring."); break; } } diff --git a/mobile/android/base/fxa/sync/FxAccountProfileService.java b/mobile/android/base/fxa/sync/FxAccountProfileService.java index 723b08219bef..1548c3ff0c9b 100644 --- a/mobile/android/base/fxa/sync/FxAccountProfileService.java +++ b/mobile/android/base/fxa/sync/FxAccountProfileService.java @@ -11,6 +11,7 @@ import android.os.Bundle; import android.os.ResultReceiver; import org.mozilla.gecko.background.common.log.Logger; +import org.mozilla.gecko.background.fxa.FxAccountUtils; import org.mozilla.gecko.background.fxa.oauth.FxAccountAbstractClient; import org.mozilla.gecko.background.fxa.oauth.FxAccountAbstractClientException; import org.mozilla.gecko.background.fxa.profile.FxAccountProfileClient10; @@ -37,6 +38,11 @@ public class FxAccountProfileService extends IntentService { final String profileServerURI = intent.getStringExtra(KEY_PROFILE_SERVER_URI); final ResultReceiver resultReceiver = intent.getParcelableExtra(KEY_RESULT_RECEIVER); + if (resultReceiver == null) { + Logger.warn(LOG_TAG, "Result receiver must not be null; ignoring intent."); + return; + } + if (authToken == null || authToken.length() == 0) { Logger.warn(LOG_TAG, "Invalid Auth Token"); sendResult("Invalid Auth Token", resultReceiver, Activity.RESULT_CANCELED); @@ -66,7 +72,7 @@ public class FxAccountProfileService extends IntentService { @Override public void handleSuccess(ExtendedJSONObject result) { if (result != null){ - Logger.pii(LOG_TAG, "Profile Server response : " + result.toJSONString()); + FxAccountUtils.pii(LOG_TAG, "Profile server return profile: " + result.toJSONString()); sendResult(result.toJSONString(), resultReceiver, Activity.RESULT_OK); } } diff --git a/mobile/android/base/fxa/sync/FxAccountSyncAdapter.java b/mobile/android/base/fxa/sync/FxAccountSyncAdapter.java index fef11a114ec0..c6ac45fd892d 100644 --- a/mobile/android/base/fxa/sync/FxAccountSyncAdapter.java +++ b/mobile/android/base/fxa/sync/FxAccountSyncAdapter.java @@ -14,6 +14,7 @@ import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.LinkedBlockingQueue; +import org.mozilla.gecko.AppConstants; import org.mozilla.gecko.background.common.log.Logger; import org.mozilla.gecko.background.fxa.FxAccountUtils; import org.mozilla.gecko.background.fxa.SkewHandler; @@ -532,6 +533,12 @@ public class FxAccountSyncAdapter extends AbstractThreadedSyncAdapter { final KeyBundle syncKeyBundle = married.getSyncKeyBundle(); final String clientState = married.getClientState(); syncWithAssertion(audience, assertion, tokenServerEndpointURI, tokenBackoffHandler, sharedPrefs, syncKeyBundle, clientState, sessionCallback, extras, fxAccount); + + if (AppConstants.MOZ_ANDROID_FIREFOX_ACCOUNT_PROFILES) { + // Force fetch the profile avatar information. + Logger.info(LOG_TAG, "Fetching profile avatar information."); + fxAccount.fetchProfileJSON(); + } } catch (Exception e) { syncDelegate.handleError(e); return; diff --git a/mobile/android/base/locales/en-US/android_strings.dtd b/mobile/android/base/locales/en-US/android_strings.dtd index d803c4660286..0e696e7d64cb 100644 --- a/mobile/android/base/locales/en-US/android_strings.dtd +++ b/mobile/android/base/locales/en-US/android_strings.dtd @@ -426,9 +426,9 @@ size. --> - - + diff --git a/mobile/android/base/resources/values/fxaccount_dimens.xml b/mobile/android/base/resources/values/fxaccount_dimens.xml index 0298be25065e..f355141b4d47 100644 --- a/mobile/android/base/resources/values/fxaccount_dimens.xml +++ b/mobile/android/base/resources/values/fxaccount_dimens.xml @@ -25,4 +25,9 @@ 16dp 0x02000000 + + + 48dp + + 48dp diff --git a/mobile/android/base/strings.xml.in b/mobile/android/base/strings.xml.in index 36de7c036a2c..cbb6eaadbed9 100644 --- a/mobile/android/base/strings.xml.in +++ b/mobile/android/base/strings.xml.in @@ -326,7 +326,7 @@ &site_settings_clear; &site_settings_no_settings; - &reading_list_added2; + &reading_list_added3; &reading_list_removed; &reading_list_remove; &reading_list_duplicate; diff --git a/mobile/android/modules/HomeProvider.jsm b/mobile/android/modules/HomeProvider.jsm index 7d3022742029..77e15aad977c 100644 --- a/mobile/android/modules/HomeProvider.jsm +++ b/mobile/android/modules/HomeProvider.jsm @@ -5,8 +5,6 @@ "use strict"; -/*globals gSyncCheckIntervalSecs, gUpdateTimerManager, Sqlite, DB_PATH */ - this.EXPORTED_SYMBOLS = [ "HomeProvider" ]; const { utils: Cu, classes: Cc, interfaces: Ci } = Components; @@ -127,18 +125,18 @@ function syncTimerCallback(timer) { } } -let HomeStorage = function(datasetId) { +this.HomeStorage = function(datasetId) { this.datasetId = datasetId; }; -let ValidationError = function(message) { +this.ValidationError = function(message) { this.name = "ValidationError"; this.message = message; }; ValidationError.prototype = new Error(); ValidationError.prototype.constructor = ValidationError; -let HomeProvider = Object.freeze({ +this.HomeProvider = Object.freeze({ ValidationError: ValidationError, /** diff --git a/mobile/locales/en-US/searchplugins/amazondotcom.xml b/mobile/locales/en-US/searchplugins/amazondotcom.xml index d52ff8633ff5..479abc36d93e 100644 --- a/mobile/locales/en-US/searchplugins/amazondotcom.xml +++ b/mobile/locales/en-US/searchplugins/amazondotcom.xml @@ -5,7 +5,7 @@ Amazon.com ISO-8859-1 - + diff --git a/mobile/locales/en-US/searchplugins/bing.xml b/mobile/locales/en-US/searchplugins/bing.xml index e503048b1769..f90bf97da216 100644 --- a/mobile/locales/en-US/searchplugins/bing.xml +++ b/mobile/locales/en-US/searchplugins/bing.xml @@ -4,7 +4,7 @@ Bing - + diff --git a/mobile/locales/en-US/searchplugins/duckduckgo.xml b/mobile/locales/en-US/searchplugins/duckduckgo.xml index 97ca332799bd..9e96ae21a005 100644 --- a/mobile/locales/en-US/searchplugins/duckduckgo.xml +++ b/mobile/locales/en-US/searchplugins/duckduckgo.xml @@ -5,7 +5,7 @@ DuckDuckGo UTF-8 - + diff --git a/mobile/locales/en-US/searchplugins/google.xml b/mobile/locales/en-US/searchplugins/google.xml index 63a14176e54b..f563a69b4379 100644 --- a/mobile/locales/en-US/searchplugins/google.xml +++ b/mobile/locales/en-US/searchplugins/google.xml @@ -5,7 +5,7 @@ Google UTF-8 - + diff --git a/mobile/locales/en-US/searchplugins/twitter.xml b/mobile/locales/en-US/searchplugins/twitter.xml index da5eb759054c..1334c4726f65 100644 --- a/mobile/locales/en-US/searchplugins/twitter.xml +++ b/mobile/locales/en-US/searchplugins/twitter.xml @@ -4,7 +4,7 @@ Twitter -  +  diff --git a/mobile/locales/en-US/searchplugins/wikipedia.xml b/mobile/locales/en-US/searchplugins/wikipedia.xml index bd22b1ea3c66..85753c0227ee 100644 --- a/mobile/locales/en-US/searchplugins/wikipedia.xml +++ b/mobile/locales/en-US/searchplugins/wikipedia.xml @@ -5,7 +5,7 @@ Wikipedia UTF-8 - + diff --git a/mobile/locales/en-US/searchplugins/yahoo.xml b/mobile/locales/en-US/searchplugins/yahoo.xml index cceee8748034..6fa29bf2018d 100644 --- a/mobile/locales/en-US/searchplugins/yahoo.xml +++ b/mobile/locales/en-US/searchplugins/yahoo.xml @@ -5,7 +5,7 @@ Yahoo UTF-8 - + diff --git a/python/mozbuild/mozbuild/artifacts.py b/python/mozbuild/mozbuild/artifacts.py index fb0e2bce8c51..49b6612f2327 100644 --- a/python/mozbuild/mozbuild/artifacts.py +++ b/python/mozbuild/mozbuild/artifacts.py @@ -325,7 +325,6 @@ class Artifacts(object): n = os.path.join(distdir, 'bin', os.path.basename(info.filename)) fh = FileAvoidWrite(n, mode='r') shutil.copyfileobj(zf.open(info), fh) - fh.write(zf.open(info).read()) file_existed, file_updated = fh.close() self.log(logging.INFO, 'artifact', {'updating': 'Updating' if file_updated else 'Not updating', 'filename': n}, diff --git a/toolkit/devtools/client/dbg-client.jsm b/toolkit/devtools/client/dbg-client.jsm index 4c36ae72daa5..52693e79b5eb 100644 --- a/toolkit/devtools/client/dbg-client.jsm +++ b/toolkit/devtools/client/dbg-client.jsm @@ -511,7 +511,8 @@ DebuggerClient.prototype = { executeSoon(() => aOnResponse({ from: workerClient.actor, type: "attached", - isFrozen: workerClient.isFrozen + isFrozen: workerClient.isFrozen, + url: workerClient.url }, workerClient)); return; } @@ -1379,6 +1380,8 @@ function WorkerClient(aClient, aForm) { this.addListener("close", this._onClose); this.addListener("freeze", this._onFreeze); this.addListener("thaw", this._onThaw); + + this.traits = {}; } WorkerClient.prototype = { @@ -1454,6 +1457,10 @@ WorkerClient.prototype = { this._isFrozen = false; }, + reconfigure: function () { + return Promise.resolve(); + }, + events: ["close", "freeze", "thaw"] };