1 // Copyright 2014 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "chrome/browser/content_settings/permission_context_base.h"
7 #include "base/logging.h"
8 #include "base/prefs/pref_service.h"
9 #include "chrome/browser/content_settings/permission_bubble_request_impl.h"
10 #include "chrome/browser/content_settings/permission_context_uma_util.h"
11 #include "chrome/browser/content_settings/permission_queue_controller.h"
12 #include "chrome/browser/profiles/profile.h"
13 #include "chrome/browser/ui/website_settings/permission_bubble_manager.h"
14 #include "chrome/common/pref_names.h"
15 #include "components/content_settings/core/browser/content_settings_utils.h"
16 #include "components/content_settings/core/browser/host_content_settings_map.h"
17 #include "components/content_settings/core/common/permission_request_id.h"
18 #include "content/public/browser/browser_thread.h"
19 #include "content/public/browser/web_contents.h"
20 #include "net/base/net_util.h"
22 PermissionContextBase::PermissionContextBase(
24 const ContentSettingsType permission_type
)
26 permission_type_(permission_type
),
28 permission_queue_controller_
.reset(
29 new PermissionQueueController(profile_
, permission_type_
));
32 PermissionContextBase::~PermissionContextBase() {
33 DCHECK_CURRENTLY_ON(content::BrowserThread::UI
);
36 void PermissionContextBase::RequestPermission(
37 content::WebContents
* web_contents
,
38 const PermissionRequestID
& id
,
39 const GURL
& requesting_frame
,
41 const BrowserPermissionCallback
& callback
) {
42 DCHECK_CURRENTLY_ON(content::BrowserThread::UI
);
44 DecidePermission(web_contents
,
46 requesting_frame
.GetOrigin(),
47 web_contents
->GetLastCommittedURL().GetOrigin(),
52 ContentSetting
PermissionContextBase::GetPermissionStatus(
53 const GURL
& requesting_origin
,
54 const GURL
& embedding_origin
) const {
55 return profile_
->GetHostContentSettingsMap()->GetContentSetting(
56 requesting_origin
, embedding_origin
, permission_type_
, std::string());
59 void PermissionContextBase::ResetPermission(
60 const GURL
& requesting_origin
,
61 const GURL
& embedding_origin
) {
62 profile_
->GetHostContentSettingsMap()->SetContentSetting(
63 ContentSettingsPattern::FromURLNoWildcard(requesting_origin
),
64 ContentSettingsPattern::FromURLNoWildcard(embedding_origin
),
65 permission_type_
, std::string(), CONTENT_SETTING_DEFAULT
);
68 void PermissionContextBase::CancelPermissionRequest(
69 content::WebContents
* web_contents
,
70 const PermissionRequestID
& id
) {
71 DCHECK_CURRENTLY_ON(content::BrowserThread::UI
);
73 if (PermissionBubbleManager::Enabled()) {
74 PermissionBubbleRequest
* cancelling
=
75 pending_bubbles_
.get(id
.ToString());
76 if (cancelling
!= NULL
&& web_contents
!= NULL
&&
77 PermissionBubbleManager::FromWebContents(web_contents
) != NULL
) {
78 PermissionBubbleManager::FromWebContents(web_contents
)->
79 CancelRequest(cancelling
);
84 GetQueueController()->CancelInfoBarRequest(id
);
87 void PermissionContextBase::DecidePermission(
88 content::WebContents
* web_contents
,
89 const PermissionRequestID
& id
,
90 const GURL
& requesting_origin
,
91 const GURL
& embedding_origin
,
93 const BrowserPermissionCallback
& callback
) {
94 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI
));
96 if (!requesting_origin
.is_valid() || !embedding_origin
.is_valid()) {
98 << "Attempt to use " << content_settings::GetTypeName(permission_type_
)
99 << " from an invalid URL: " << requesting_origin
100 << "," << embedding_origin
101 << " (" << content_settings::GetTypeName(permission_type_
)
102 << " is not supported in popups)";
103 NotifyPermissionSet(id
, requesting_origin
, embedding_origin
, callback
,
104 false /* persist */, CONTENT_SETTING_BLOCK
);
108 // The Web MIDI API is not available for origin with non secure schemes.
109 // Access to the MIDI API is blocked.
110 // TODO(crbug.com/362214): Use a standard way to check the secure origin.
111 if (permission_type_
== CONTENT_SETTINGS_TYPE_MIDI_SYSEX
&&
112 !requesting_origin
.SchemeIsSecure() &&
113 !net::IsLocalhost(requesting_origin
.host())) {
114 NotifyPermissionSet(id
, requesting_origin
, embedding_origin
, callback
,
115 false /* persist */, CONTENT_SETTING_BLOCK
);
119 ContentSetting content_setting
=
120 profile_
->GetHostContentSettingsMap()
121 ->GetContentSettingAndMaybeUpdateLastUsage(
122 requesting_origin
, embedding_origin
, permission_type_
,
125 if (content_setting
== CONTENT_SETTING_ALLOW
||
126 content_setting
== CONTENT_SETTING_BLOCK
) {
127 NotifyPermissionSet(id
, requesting_origin
, embedding_origin
, callback
,
128 false /* persist */, content_setting
);
132 PermissionContextUmaUtil::PermissionRequested(
133 permission_type_
, requesting_origin
);
135 if (PermissionBubbleManager::Enabled()) {
136 if (pending_bubbles_
.get(id
.ToString()) != NULL
)
138 PermissionBubbleManager
* bubble_manager
=
139 PermissionBubbleManager::FromWebContents(web_contents
);
140 // TODO(mlamouri): sometimes |bubble_manager| is null. This check is meant
141 // to prevent crashes. See bug 457091.
144 scoped_ptr
<PermissionBubbleRequest
> request_ptr(
145 new PermissionBubbleRequestImpl(
146 requesting_origin
, user_gesture
, permission_type_
,
147 profile_
->GetPrefs()->GetString(prefs::kAcceptLanguages
),
148 base::Bind(&PermissionContextBase::PermissionDecided
,
149 weak_factory_
.GetWeakPtr(), id
, requesting_origin
,
150 embedding_origin
, callback
),
151 base::Bind(&PermissionContextBase::CleanUpBubble
,
152 weak_factory_
.GetWeakPtr(), id
)));
153 PermissionBubbleRequest
* request
= request_ptr
.get();
155 bool inserted
= pending_bubbles_
.add(
156 id
.ToString(), request_ptr
.Pass()).second
;
157 DCHECK(inserted
) << "Duplicate id " << id
.ToString();
158 bubble_manager
->AddRequest(request
);
162 // TODO(gbillock): Delete this and the infobar delegate when
163 // we're using only bubbles. crbug.com/337458
164 GetQueueController()->CreateInfoBarRequest(
165 id
, requesting_origin
, embedding_origin
,
166 base::Bind(&PermissionContextBase::PermissionDecided
,
167 weak_factory_
.GetWeakPtr(), id
, requesting_origin
,
168 embedding_origin
, callback
,
169 // the queue controller takes care of persisting the
174 void PermissionContextBase::PermissionDecided(
175 const PermissionRequestID
& id
,
176 const GURL
& requesting_origin
,
177 const GURL
& embedding_origin
,
178 const BrowserPermissionCallback
& callback
,
180 ContentSetting content_setting
) {
181 // Infobar persistance and its related UMA is tracked on the infobar
182 // controller directly.
183 if (PermissionBubbleManager::Enabled()) {
185 DCHECK(content_setting
== CONTENT_SETTING_ALLOW
||
186 content_setting
== CONTENT_SETTING_BLOCK
);
187 if (CONTENT_SETTING_ALLOW
)
188 PermissionContextUmaUtil::PermissionGranted(permission_type_
,
191 PermissionContextUmaUtil::PermissionDenied(permission_type_
,
194 DCHECK_EQ(content_setting
, CONTENT_SETTING_DEFAULT
);
195 PermissionContextUmaUtil::PermissionDismissed(permission_type_
,
200 NotifyPermissionSet(id
, requesting_origin
, embedding_origin
, callback
,
201 persist
, content_setting
);
204 PermissionQueueController
* PermissionContextBase::GetQueueController() {
205 return permission_queue_controller_
.get();
208 Profile
* PermissionContextBase::profile() const {
212 void PermissionContextBase::NotifyPermissionSet(
213 const PermissionRequestID
& id
,
214 const GURL
& requesting_origin
,
215 const GURL
& embedding_origin
,
216 const BrowserPermissionCallback
& callback
,
218 ContentSetting content_setting
) {
219 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI
));
222 UpdateContentSetting(requesting_origin
, embedding_origin
, content_setting
);
224 UpdateTabContext(id
, requesting_origin
,
225 content_setting
== CONTENT_SETTING_ALLOW
);
227 if (content_setting
== CONTENT_SETTING_DEFAULT
) {
229 profile_
->GetHostContentSettingsMap()->GetDefaultContentSetting(
230 permission_type_
, nullptr);
233 DCHECK_NE(content_setting
, CONTENT_SETTING_DEFAULT
);
234 callback
.Run(content_setting
);
237 void PermissionContextBase::CleanUpBubble(const PermissionRequestID
& id
) {
238 size_t success
= pending_bubbles_
.erase(id
.ToString());
239 DCHECK(success
== 1) << "Missing request " << id
.ToString();
242 void PermissionContextBase::UpdateContentSetting(
243 const GURL
& requesting_origin
,
244 const GURL
& embedding_origin
,
245 ContentSetting content_setting
) {
246 DCHECK_EQ(requesting_origin
, requesting_origin
.GetOrigin());
247 DCHECK_EQ(embedding_origin
, embedding_origin
.GetOrigin());
248 DCHECK(content_setting
== CONTENT_SETTING_ALLOW
||
249 content_setting
== CONTENT_SETTING_BLOCK
);
251 profile_
->GetHostContentSettingsMap()->SetContentSetting(
252 ContentSettingsPattern::FromURLNoWildcard(requesting_origin
),
253 ContentSettingsPattern::FromURLNoWildcard(embedding_origin
),
254 permission_type_
, std::string(), content_setting
);