| // Copyright 2014 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "chrome/browser/push_messaging/push_messaging_permission_context.h" |
| |
| #include "chrome/browser/content_settings/host_content_settings_map_factory.h" |
| #include "chrome/browser/notifications/notification_permission_context.h" |
| #include "chrome/browser/notifications/notification_permission_context_factory.h" |
| #include "chrome/browser/permissions/permission_request_id.h" |
| #include "chrome/browser/permissions/permission_uma_util.h" |
| #include "chrome/browser/profiles/profile.h" |
| #include "components/content_settings/core/browser/host_content_settings_map.h" |
| #include "content/public/browser/browser_thread.h" |
| #include "content/public/browser/permission_type.h" |
| #include "content/public/browser/web_contents.h" |
| #include "content/public/browser/web_contents_delegate.h" |
| #include "content/public/common/origin_util.h" |
| |
| PushMessagingPermissionContext::PushMessagingPermissionContext(Profile* profile) |
| : PermissionContextBase(profile, |
| content::PermissionType::PUSH_MESSAGING, |
| CONTENT_SETTINGS_TYPE_PUSH_MESSAGING), |
| profile_(profile), |
| weak_factory_ui_thread_(this) {} |
| |
| PushMessagingPermissionContext::~PushMessagingPermissionContext() {} |
| |
| ContentSetting PushMessagingPermissionContext::GetPermissionStatus( |
| const GURL& requesting_origin, |
| const GURL& embedding_origin) const { |
| // It's possible for this to return CONTENT_SETTING_BLOCK in cases where |
| // HostContentSettingsMap::GetContentSetting returns CONTENT_SETTING_ALLOW. |
| // TODO(johnme): This is likely to break assumptions made elsewhere, so we |
| // should try to remove this quirk. |
| #if defined(ENABLE_NOTIFICATIONS) |
| if (requesting_origin != embedding_origin) |
| return CONTENT_SETTING_BLOCK; |
| |
| ContentSetting push_content_setting = |
| PermissionContextBase::GetPermissionStatus(requesting_origin, |
| embedding_origin); |
| |
| NotificationPermissionContext* notification_context = |
| NotificationPermissionContextFactory::GetForProfile(profile_); |
| DCHECK(notification_context); |
| |
| ContentSetting notifications_permission = |
| notification_context->GetPermissionStatus(requesting_origin, |
| embedding_origin); |
| |
| if (notifications_permission == CONTENT_SETTING_BLOCK || |
| push_content_setting == CONTENT_SETTING_BLOCK) { |
| return CONTENT_SETTING_BLOCK; |
| } |
| if (notifications_permission == CONTENT_SETTING_ASK || |
| push_content_setting == CONTENT_SETTING_ASK) { |
| return CONTENT_SETTING_ASK; |
| } |
| DCHECK_EQ(CONTENT_SETTING_ALLOW, notifications_permission); |
| DCHECK_EQ(CONTENT_SETTING_ALLOW, push_content_setting); |
| return CONTENT_SETTING_ALLOW; |
| #else |
| return CONTENT_SETTING_BLOCK; |
| #endif |
| } |
| |
| // Unlike other permissions, push is decided by the following algorithm |
| // - You need to request it from a top level domain |
| // - You need to have notification permission granted. |
| // - You need to not have push permission explicitly blocked. |
| // - If those 3 things are true it is granted without prompting. |
| // This is done to avoid double prompting for notifications and push. |
| void PushMessagingPermissionContext::DecidePermission( |
| content::WebContents* web_contents, |
| const PermissionRequestID& id, |
| const GURL& requesting_origin, |
| const GURL& embedding_origin, |
| const BrowserPermissionCallback& callback) { |
| DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
| #if defined(ENABLE_NOTIFICATIONS) |
| if (requesting_origin != embedding_origin) { |
| NotifyPermissionSet(id, requesting_origin, embedding_origin, callback, |
| false /* persist */, CONTENT_SETTING_BLOCK); |
| return; |
| } |
| |
| NotificationPermissionContext* notification_context = |
| NotificationPermissionContextFactory::GetForProfile(profile_); |
| DCHECK(notification_context); |
| |
| notification_context->RequestPermission( |
| web_contents, id, requesting_origin, |
| base::Bind(&PushMessagingPermissionContext::DecidePushPermission, |
| weak_factory_ui_thread_.GetWeakPtr(), id, requesting_origin, |
| embedding_origin, callback)); |
| #else |
| NotifyPermissionSet(id, requesting_origin, embedding_origin, callback, |
| false /* persist */, CONTENT_SETTING_BLOCK); |
| #endif |
| } |
| |
| bool PushMessagingPermissionContext::IsRestrictedToSecureOrigins() const { |
| return true; |
| } |
| |
| void PushMessagingPermissionContext::DecidePushPermission( |
| const PermissionRequestID& id, |
| const GURL& requesting_origin, |
| const GURL& embedding_origin, |
| const BrowserPermissionCallback& callback, |
| ContentSetting notification_content_setting) { |
| DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
| ContentSetting push_content_setting = |
| HostContentSettingsMapFactory::GetForProfile(profile_) |
| ->GetContentSettingAndMaybeUpdateLastUsage( |
| requesting_origin, embedding_origin, content_settings_type(), |
| std::string()); |
| |
| if (push_content_setting == CONTENT_SETTING_BLOCK) { |
| DVLOG(1) << "Push permission was explicitly blocked."; |
| PermissionUmaUtil::PermissionDenied(permission_type(), requesting_origin); |
| NotifyPermissionSet(id, requesting_origin, embedding_origin, callback, |
| true /* persist */, CONTENT_SETTING_BLOCK); |
| return; |
| } |
| |
| if (notification_content_setting != CONTENT_SETTING_ALLOW) { |
| DVLOG(1) << "Notification permission has not been granted."; |
| NotifyPermissionSet(id, requesting_origin, embedding_origin, callback, |
| false /* persist */, notification_content_setting); |
| return; |
| } |
| |
| PermissionUmaUtil::PermissionGranted(permission_type(), requesting_origin); |
| NotifyPermissionSet(id, requesting_origin, embedding_origin, callback, |
| true /* persist */, CONTENT_SETTING_ALLOW); |
| } |