blob: bf3303cd0cf6e0e9cce3e3e05aadfe53b932e9a2 [file] [log] [blame]
[email protected]c333e792012-01-06 16:57:391// Copyright (c) 2012 The Chromium Authors. All rights reserved.
[email protected]f9c292b32011-09-13 16:14:052// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
[email protected]d9ede582012-08-14 19:21:385#include "chrome/browser/extensions/navigation_observer.h"
[email protected]f9c292b32011-09-13 16:14:056
Sebastien Marchandf1349f52019-01-25 03:16:417#include "base/bind.h"
[email protected]f9c292b32011-09-13 16:14:058#include "chrome/browser/extensions/extension_service.h"
9#include "chrome/browser/profiles/profile.h"
[email protected]cdcb1dee2012-01-04 00:46:2010#include "content/public/browser/navigation_controller.h"
[email protected]ad23a092011-12-28 07:02:0411#include "content/public/browser/navigation_entry.h"
[email protected]ad50def52011-10-19 23:17:0712#include "content/public/browser/notification_service.h"
[email protected]c333e792012-01-06 16:57:3913#include "content/public/browser/notification_types.h"
[email protected]dccba4f82014-05-29 00:52:5614#include "extensions/browser/extension_prefs.h"
[email protected]59b0e602014-01-30 00:41:2415#include "extensions/browser/extension_system.h"
[email protected]f9c292b32011-09-13 16:14:0516
[email protected]c5eed492012-01-04 17:07:5017using content::NavigationController;
[email protected]10f417c52011-12-28 21:04:2318using content::NavigationEntry;
19
[email protected]d9ede582012-08-14 19:21:3820namespace extensions {
21
rdevlin.cronin4b184ca2017-01-13 01:35:4422namespace {
23// Whether to repeatedly prompt for the same extension id.
24bool g_repeat_prompting = false;
25}
26
Evan Stade75872a62019-09-06 21:17:3827NavigationObserver::NavigationObserver(Profile* profile) : profile_(profile) {
[email protected]f9c292b32011-09-13 16:14:0528 RegisterForNotifications();
lazyboy56a9b9182016-04-19 22:13:4429 extension_registry_observer_.Add(ExtensionRegistry::Get(profile));
[email protected]f9c292b32011-09-13 16:14:0530}
31
[email protected]d9ede582012-08-14 19:21:3832NavigationObserver::~NavigationObserver() {}
[email protected]f9c292b32011-09-13 16:14:0533
[email protected]d9ede582012-08-14 19:21:3834void NavigationObserver::Observe(int type,
35 const content::NotificationSource& source,
36 const content::NotificationDetails& details) {
thestigf0ed1962016-06-13 21:14:2537 DCHECK_EQ(content::NOTIFICATION_NAV_ENTRY_COMMITTED, type);
[email protected]f9c292b32011-09-13 16:14:0538
[email protected]c5eed492012-01-04 17:07:5039 NavigationController* controller =
40 content::Source<NavigationController>(source).ptr();
[email protected]f9c292b32011-09-13 16:14:0541 if (!profile_->IsSameProfile(
[email protected]a26023822011-12-29 00:23:5542 Profile::FromBrowserContext(controller->GetBrowserContext())))
[email protected]f9c292b32011-09-13 16:14:0543 return;
44
45 PromptToEnableExtensionIfNecessary(controller);
46}
47
rdevlin.cronin4b184ca2017-01-13 01:35:4448// static
49void NavigationObserver::SetAllowedRepeatedPromptingForTesting(bool allowed) {
50 g_repeat_prompting = allowed;
51}
52
[email protected]d9ede582012-08-14 19:21:3853void NavigationObserver::RegisterForNotifications() {
[email protected]f9c292b32011-09-13 16:14:0554 registrar_.Add(this, content::NOTIFICATION_NAV_ENTRY_COMMITTED,
[email protected]ad50def52011-10-19 23:17:0755 content::NotificationService::AllSources());
[email protected]f9c292b32011-09-13 16:14:0556}
57
[email protected]d9ede582012-08-14 19:21:3858void NavigationObserver::PromptToEnableExtensionIfNecessary(
[email protected]c5eed492012-01-04 17:07:5059 NavigationController* nav_controller) {
[email protected]f9c292b32011-09-13 16:14:0560 // Bail out if we're already running a prompt.
61 if (!in_progress_prompt_extension_id_.empty())
62 return;
63
[email protected]afe9aba2013-08-16 20:31:3464 NavigationEntry* nav_entry = nav_controller->GetVisibleEntry();
[email protected]f9c292b32011-09-13 16:14:0565 if (!nav_entry)
66 return;
67
Charles Reis56a84aa672017-10-26 20:10:4268 const GURL& url = nav_entry->GetURL();
nasko5ff4eea2017-05-25 02:51:0069
rdevlin.cronin4b184ca2017-01-13 01:35:4470 // NOTE: We only consider chrome-extension:// urls, and deliberately don't
71 // consider hosted app urls. This is because it's really annoying to visit the
72 // site associated with a hosted app (like calendar.google.com or
73 // drive.google.com) and have it repeatedly prompt you to re-enable an item.
74 // Visiting a chrome-extension:// url is a much stronger signal, and, without
75 // the item enabled, we won't show anything.
76 // TODO(devlin): While true, I still wonder how useful this is. We should get
77 // metrics.
78 if (!url.SchemeIs(kExtensionScheme))
79 return;
80
81 const Extension* extension = ExtensionRegistry::Get(profile_)
82 ->disabled_extensions()
83 .GetExtensionOrAppByURL(url);
[email protected]f9c292b32011-09-13 16:14:0584 if (!extension)
85 return;
86
87 // Try not to repeatedly prompt the user about the same extension.
rdevlin.cronin4b184ca2017-01-13 01:35:4488 if (!prompted_extensions_.insert(extension->id()).second &&
89 !g_repeat_prompting) {
[email protected]f9c292b32011-09-13 16:14:0590 return;
rdevlin.cronin4b184ca2017-01-13 01:35:4491 }
[email protected]f9c292b32011-09-13 16:14:0592
[email protected]7c82539c2014-02-19 06:09:1793 ExtensionPrefs* extension_prefs = ExtensionPrefs::Get(profile_);
rdevlin.cronin4b184ca2017-01-13 01:35:4494 // TODO(devlin): Why do we only consider extensions that escalate permissions?
95 // Maybe because it's the only one we have a good prompt for?
[email protected]f9c292b32011-09-13 16:14:0596 if (extension_prefs->DidExtensionEscalatePermissions(extension->id())) {
97 // Keep track of the extension id and nav controller we're prompting for.
rdevlin.cronin41593052016-01-08 01:40:1298 // These must be reset in OnInstallPromptDone.
[email protected]f9c292b32011-09-13 16:14:0599 in_progress_prompt_extension_id_ = extension->id();
100 in_progress_prompt_navigation_controller_ = nav_controller;
101
[email protected]619f86182012-07-03 21:30:18102 extension_install_prompt_.reset(
[email protected]91e51d612012-10-21 23:03:05103 new ExtensionInstallPrompt(nav_controller->GetWebContents()));
rdevlin.croninf84cab72015-12-12 03:45:23104 ExtensionInstallPrompt::PromptType type =
105 ExtensionInstallPrompt::GetReEnablePromptTypeForExtension(profile_,
106 extension);
107 extension_install_prompt_->ShowDialog(
rdevlin.cronin41593052016-01-08 01:40:12108 base::Bind(&NavigationObserver::OnInstallPromptDone,
109 weak_factory_.GetWeakPtr()),
110 extension, nullptr,
Jinho Bangb5216cec2018-01-17 19:43:11111 std::make_unique<ExtensionInstallPrompt::Prompt>(type),
rdevlin.croninf84cab72015-12-12 03:45:23112 ExtensionInstallPrompt::GetDefaultShowDialogCallback());
[email protected]f9c292b32011-09-13 16:14:05113 }
114}
115
rdevlin.cronin41593052016-01-08 01:40:12116void NavigationObserver::OnInstallPromptDone(
117 ExtensionInstallPrompt::Result result) {
lazyboy56a9b9182016-04-19 22:13:44118 // The extension was already uninstalled.
119 if (in_progress_prompt_extension_id_.empty())
120 return;
121
David Bertoni58c113a2019-08-02 19:53:26122 ExtensionRegistry* extension_registry = ExtensionRegistry::Get(profile_);
123 const Extension* extension = extension_registry->GetExtensionById(
124 in_progress_prompt_extension_id_, ExtensionRegistry::COMPATIBILITY);
lazyboy56a9b9182016-04-19 22:13:44125 CHECK(extension);
[email protected]f9c292b32011-09-13 16:14:05126
rdevlin.cronin41593052016-01-08 01:40:12127 if (result == ExtensionInstallPrompt::Result::ACCEPTED) {
128 NavigationController* nav_controller =
129 in_progress_prompt_navigation_controller_;
rdevlin.cronin41593052016-01-08 01:40:12130 CHECK(nav_controller);
[email protected]f9c292b32011-09-13 16:14:05131
David Bertoni58c113a2019-08-02 19:53:26132 ExtensionService* extension_service =
133 ExtensionSystem::Get(profile_)->extension_service();
rdevlin.cronin41593052016-01-08 01:40:12134 // Grant permissions, re-enable the extension, and then reload the tab.
135 extension_service->GrantPermissionsAndEnableExtension(extension);
toyoshim6142d96f2016-12-19 09:07:25136 nav_controller->Reload(content::ReloadType::NORMAL, true);
rdevlin.cronin41593052016-01-08 01:40:12137 } else {
rdevlin.cronin4b184ca2017-01-13 01:35:44138 // TODO(devlin): These metrics aren't very useful, since they're lumped in
139 // with the same for re-enabling/canceling when the extension first gets
140 // disabled, which is likely significantly more common (though impossible to
141 // tell). We need to separate these.
rdevlin.cronin41593052016-01-08 01:40:12142 std::string histogram_name =
lazyboy56a9b9182016-04-19 22:13:44143 result == ExtensionInstallPrompt::Result::USER_CANCELED
rdevlin.cronin41593052016-01-08 01:40:12144 ? "ReEnableCancel"
145 : "ReEnableAbort";
146 ExtensionService::RecordPermissionMessagesHistogram(extension,
147 histogram_name.c_str());
148 }
149
David Bertoni58c113a2019-08-02 19:53:26150 in_progress_prompt_extension_id_.clear();
rdevlin.cronin41593052016-01-08 01:40:12151 in_progress_prompt_navigation_controller_ = nullptr;
152 extension_install_prompt_.reset();
[email protected]f9c292b32011-09-13 16:14:05153}
[email protected]d9ede582012-08-14 19:21:38154
lazyboy56a9b9182016-04-19 22:13:44155void NavigationObserver::OnExtensionUninstalled(
156 content::BrowserContext* browser_context,
157 const Extension* extension,
158 UninstallReason reason) {
159 if (in_progress_prompt_extension_id_.empty() ||
160 in_progress_prompt_extension_id_ != extension->id()) {
161 return;
162 }
163
164 in_progress_prompt_extension_id_ = std::string();
165 in_progress_prompt_navigation_controller_ = nullptr;
166 extension_install_prompt_.reset();
167}
168
[email protected]d9ede582012-08-14 19:21:38169} // namespace extensions