| OLD | NEW |
| (Empty) | |
| 1 // Copyright 2016 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. |
| 4 |
| 5 #include "chrome/browser/permissions/permission_decision_auto_blocker.h" |
| 6 |
| 7 #include <memory> |
| 8 |
| 9 #include "base/feature_list.h" |
| 10 #include "base/logging.h" |
| 11 #include "base/memory/ptr_util.h" |
| 12 #include "base/strings/string_number_conversions.h" |
| 13 #include "base/values.h" |
| 14 #include "chrome/browser/content_settings/host_content_settings_map_factory.h" |
| 15 #include "chrome/browser/permissions/permission_uma_util.h" |
| 16 #include "chrome/browser/permissions/permission_util.h" |
| 17 #include "chrome/common/chrome_features.h" |
| 18 #include "components/content_settings/core/browser/host_content_settings_map.h" |
| 19 #include "components/variations/variations_associated_data.h" |
| 20 #include "content/public/browser/permission_type.h" |
| 21 #include "url/gurl.h" |
| 22 |
| 23 namespace { |
| 24 |
| 25 // The default number of times that users may explicitly dismiss a permission |
| 26 // prompt from an origin before it is automatically blocked. |
| 27 const int kPromptDismissalsBeforeBlock = 3; |
| 28 |
| 29 std::unique_ptr<base::DictionaryValue> GetOriginDict( |
| 30 HostContentSettingsMap* settings, |
| 31 const GURL& origin_url) { |
| 32 std::unique_ptr<base::DictionaryValue> dict = |
| 33 base::DictionaryValue::From(settings->GetWebsiteSetting( |
| 34 origin_url, origin_url, |
| 35 CONTENT_SETTINGS_TYPE_PROMPT_NO_DECISION_COUNT, std::string(), |
| 36 nullptr)); |
| 37 if (!dict) |
| 38 return base::WrapUnique(new base::DictionaryValue()); |
| 39 |
| 40 return dict; |
| 41 } |
| 42 |
| 43 base::DictionaryValue* GetOrCreatePermissionDict( |
| 44 base::DictionaryValue* origin_dict, |
| 45 const std::string& permission) { |
| 46 base::DictionaryValue* permission_dict = nullptr; |
| 47 if (!origin_dict->GetDictionaryWithoutPathExpansion(permission, |
| 48 &permission_dict)) { |
| 49 permission_dict = new base::DictionaryValue(); |
| 50 origin_dict->SetWithoutPathExpansion(permission, |
| 51 base::WrapUnique(permission_dict)); |
| 52 } |
| 53 |
| 54 return permission_dict; |
| 55 } |
| 56 |
| 57 } // namespace |
| 58 |
| 59 // static |
| 60 const char PermissionDecisionAutoBlocker::kPromptDismissCountKey[] = |
| 61 "dismiss_count"; |
| 62 |
| 63 // static |
| 64 const char PermissionDecisionAutoBlocker::kPromptIgnoreCountKey[] = |
| 65 "ignore_count"; |
| 66 |
| 67 // static |
| 68 void PermissionDecisionAutoBlocker::RemoveCountsByUrl( |
| 69 Profile* profile, |
| 70 base::Callback<bool(const GURL& url)> filter) { |
| 71 HostContentSettingsMap* map = |
| 72 HostContentSettingsMapFactory::GetForProfile(profile); |
| 73 |
| 74 std::unique_ptr<ContentSettingsForOneType> settings( |
| 75 new ContentSettingsForOneType); |
| 76 map->GetSettingsForOneType(CONTENT_SETTINGS_TYPE_PROMPT_NO_DECISION_COUNT, |
| 77 std::string(), settings.get()); |
| 78 |
| 79 for (const auto& site : *settings) { |
| 80 GURL origin(site.primary_pattern.ToString()); |
| 81 |
| 82 if (origin.is_valid() && filter.Run(origin)) { |
| 83 map->SetWebsiteSettingDefaultScope( |
| 84 origin, GURL(), CONTENT_SETTINGS_TYPE_PROMPT_NO_DECISION_COUNT, |
| 85 std::string(), nullptr); |
| 86 } |
| 87 } |
| 88 } |
| 89 |
| 90 PermissionDecisionAutoBlocker::PermissionDecisionAutoBlocker(Profile* profile) |
| 91 : profile_(profile), |
| 92 prompt_dismissals_before_block_(kPromptDismissalsBeforeBlock) { |
| 93 UpdateFromVariations(); |
| 94 } |
| 95 |
| 96 int PermissionDecisionAutoBlocker::RecordIgnore( |
| 97 const GURL& url, |
| 98 content::PermissionType permission) { |
| 99 int current_ignore_count = |
| 100 RecordActionInWebsiteSettings(url, permission, kPromptIgnoreCountKey); |
| 101 |
| 102 PermissionUmaUtil::PermissionPromptIgnored(permission, current_ignore_count); |
| 103 |
| 104 return current_ignore_count; |
| 105 } |
| 106 |
| 107 bool PermissionDecisionAutoBlocker::ShouldChangeDismissalToBlock( |
| 108 const GURL& url, |
| 109 content::PermissionType permission) { |
| 110 int current_dismissal_count = |
| 111 RecordActionInWebsiteSettings(url, permission, kPromptDismissCountKey); |
| 112 |
| 113 PermissionUmaUtil::PermissionPromptDismissed(permission, |
| 114 current_dismissal_count); |
| 115 |
| 116 if (!base::FeatureList::IsEnabled(features::kBlockPromptsIfDismissedOften)) |
| 117 return false; |
| 118 |
| 119 return current_dismissal_count >= prompt_dismissals_before_block_; |
| 120 } |
| 121 |
| 122 int PermissionDecisionAutoBlocker::GetActionCountForTest( |
| 123 const GURL& url, |
| 124 content::PermissionType permission, |
| 125 const char* key) { |
| 126 HostContentSettingsMap* map = |
| 127 HostContentSettingsMapFactory::GetForProfile(profile_); |
| 128 std::unique_ptr<base::DictionaryValue> dict = GetOriginDict(map, url); |
| 129 |
| 130 base::DictionaryValue* permission_dict = GetOrCreatePermissionDict( |
| 131 dict.get(), PermissionUtil::GetPermissionString(permission)); |
| 132 |
| 133 int current_count = 0; |
| 134 permission_dict->GetInteger(key, ¤t_count); |
| 135 return current_count; |
| 136 } |
| 137 |
| 138 int PermissionDecisionAutoBlocker::RecordActionInWebsiteSettings( |
| 139 const GURL& url, |
| 140 content::PermissionType permission, |
| 141 const char* key) { |
| 142 HostContentSettingsMap* map = |
| 143 HostContentSettingsMapFactory::GetForProfile(profile_); |
| 144 std::unique_ptr<base::DictionaryValue> dict = GetOriginDict(map, url); |
| 145 |
| 146 base::DictionaryValue* permission_dict = GetOrCreatePermissionDict( |
| 147 dict.get(), PermissionUtil::GetPermissionString(permission)); |
| 148 |
| 149 int current_count = 0; |
| 150 permission_dict->GetInteger(key, ¤t_count); |
| 151 permission_dict->SetInteger(key, ++current_count); |
| 152 |
| 153 map->SetWebsiteSettingDefaultScope( |
| 154 url, GURL(), CONTENT_SETTINGS_TYPE_PROMPT_NO_DECISION_COUNT, |
| 155 std::string(), std::move(dict)); |
| 156 |
| 157 return current_count; |
| 158 } |
| 159 |
| 160 void PermissionDecisionAutoBlocker::UpdateFromVariations() { |
| 161 int prompt_dismissals = -1; |
| 162 std::string value = variations::GetVariationParamValueByFeature( |
| 163 features::kBlockPromptsIfDismissedOften, kPromptDismissCountKey); |
| 164 |
| 165 // If converting the value fails, stick with the default value. |
| 166 if (base::StringToInt(value, &prompt_dismissals) && prompt_dismissals > 0) |
| 167 prompt_dismissals_before_block_ = prompt_dismissals; |
| 168 } |
| OLD | NEW |