| // Copyright 2013 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "chrome/browser/ui/webui/settings/reset_settings_handler.h" |
| |
| #include <string> |
| #include <utility> |
| |
| #include "base/functional/bind.h" |
| #include "base/functional/callback_helpers.h" |
| #include "base/metrics/histogram_macros.h" |
| #include "base/metrics/user_metrics.h" |
| #include "base/time/time.h" |
| #include "base/values.h" |
| #include "build/build_config.h" |
| #include "build/chromeos_buildflags.h" |
| #include "chrome/browser/browser_process.h" |
| #include "chrome/browser/google/google_brand.h" |
| #include "chrome/browser/lifetime/application_lifetime.h" |
| #include "chrome/browser/net/system_network_context_manager.h" |
| #include "chrome/browser/prefs/chrome_pref_service_factory.h" |
| #include "chrome/browser/profile_resetter/brandcode_config_fetcher.h" |
| #include "chrome/browser/profile_resetter/brandcoded_default_settings.h" |
| #include "chrome/browser/profile_resetter/profile_resetter.h" |
| #include "chrome/browser/profile_resetter/resettable_settings_snapshot.h" |
| #include "chrome/browser/profiles/profile.h" |
| #include "chrome/common/url_constants.h" |
| #include "chrome/grit/generated_resources.h" |
| #include "components/bookmarks/browser/bookmark_model.h" |
| #include "components/prefs/pref_service.h" |
| #include "content/public/browser/web_ui.h" |
| #include "content/public/browser/web_ui_data_source.h" |
| #include "services/network/public/mojom/url_loader_factory.mojom.h" |
| #include "ui/base/l10n/l10n_util.h" |
| |
| #if BUILDFLAG(IS_CHROMEOS_ASH) |
| #include "chrome/browser/bookmarks/bookmark_model_factory.h" |
| #include "chrome/browser/ui/webui/ash/settings/pref_names.h" |
| #endif // BUILDFLAG(IS_CHROMEOS_ASH) |
| |
| #if BUILDFLAG(IS_WIN) |
| #include "chrome/browser/profile_resetter/triggered_profile_resetter.h" |
| #include "chrome/browser/profile_resetter/triggered_profile_resetter_factory.h" |
| #endif // BUILDFLAG(IS_WIN) |
| |
| namespace settings { |
| |
| namespace { |
| |
| reset_report::ChromeResetReport::ResetRequestOrigin |
| ResetRequestOriginFromString(const std::string& request_origin) { |
| static const char kOriginUserClick[] = "userclick"; |
| static const char kOriginTriggeredReset[] = "triggeredreset"; |
| |
| if (request_origin == kOriginUserClick) |
| return reset_report::ChromeResetReport::RESET_REQUEST_ORIGIN_USER_CLICK; |
| if (request_origin == kOriginTriggeredReset) { |
| return reset_report::ChromeResetReport:: |
| RESET_REQUEST_ORIGIN_TRIGGERED_RESET; |
| } |
| if (!request_origin.empty()) |
| NOTREACHED(); |
| |
| return reset_report::ChromeResetReport::RESET_REQUEST_ORIGIN_UNKNOWN; |
| } |
| |
| } // namespace |
| |
| #if BUILDFLAG(IS_CHROMEOS_ASH) |
| // static |
| const char ResetSettingsHandler::kCctResetSettingsHash[] = "cct"; |
| |
| // static |
| void ResetSettingsHandler::RegisterLocalStatePrefs( |
| PrefRegistrySimple* registry) { |
| registry->RegisterBooleanPref(ash::settings::prefs::kSanitizeCompleted, |
| false); |
| } |
| #endif // BUILDFLAG(IS_CHROMEOS_ASH) |
| |
| // static |
| bool ResetSettingsHandler::ShouldShowResetProfileBanner(Profile* profile) { |
| const base::Time reset_time = chrome_prefs::GetResetTime(profile); |
| |
| // If there is no reset time, do not show the banner. |
| if (reset_time.is_null()) |
| return false; |
| |
| // Otherwise, only show the banner if it has been less than |kBannerShowTime| |
| // since reset. |
| static constexpr base::TimeDelta kBannerShowTime = base::Days(5); |
| const base::TimeDelta since_reset = base::Time::Now() - reset_time; |
| return since_reset < kBannerShowTime; |
| } |
| |
| ResetSettingsHandler::ResetSettingsHandler(Profile* profile) |
| : profile_(profile) { |
| google_brand::GetBrand(&brandcode_); |
| } |
| |
| ResetSettingsHandler::~ResetSettingsHandler() {} |
| |
| void ResetSettingsHandler::OnJavascriptDisallowed() { |
| callback_weak_ptr_factory_.InvalidateWeakPtrs(); |
| } |
| |
| void ResetSettingsHandler::RegisterMessages() { |
| web_ui()->RegisterMessageCallback( |
| "performResetProfileSettings", |
| base::BindRepeating(&ResetSettingsHandler::HandleResetProfileSettings, |
| base::Unretained(this))); |
| web_ui()->RegisterMessageCallback( |
| "onShowResetProfileDialog", |
| base::BindRepeating(&ResetSettingsHandler::OnShowResetProfileDialog, |
| base::Unretained(this))); |
| web_ui()->RegisterMessageCallback( |
| "getReportedSettings", |
| base::BindRepeating(&ResetSettingsHandler::HandleGetReportedSettings, |
| base::Unretained(this))); |
| web_ui()->RegisterMessageCallback( |
| "onHideResetProfileDialog", |
| base::BindRepeating(&ResetSettingsHandler::OnHideResetProfileDialog, |
| base::Unretained(this))); |
| web_ui()->RegisterMessageCallback( |
| "onHideResetProfileBanner", |
| base::BindRepeating(&ResetSettingsHandler::OnHideResetProfileBanner, |
| base::Unretained(this))); |
| web_ui()->RegisterMessageCallback( |
| "getTriggeredResetToolName", |
| base::BindRepeating( |
| &ResetSettingsHandler::HandleGetTriggeredResetToolName, |
| base::Unretained(this))); |
| #if BUILDFLAG(IS_CHROMEOS_ASH) |
| web_ui()->RegisterMessageCallback( |
| "performSanitizeSettings", |
| base::BindRepeating(&ResetSettingsHandler::HandleSanitizeSettings, |
| base::Unretained(this))); |
| web_ui()->RegisterMessageCallback( |
| "onShowSanitizeDialog", |
| base::BindRepeating(&ResetSettingsHandler::OnShowSanitizeDialog, |
| base::Unretained(this))); |
| #endif // BUILDFLAG(IS_CHROMEOS_ASH) |
| } |
| |
| void ResetSettingsHandler::HandleResetProfileSettings( |
| const base::Value::List& args) { |
| AllowJavascript(); |
| |
| CHECK_EQ(3U, args.size()); |
| const std::string& callback_id = args[0].GetString(); |
| const bool& send_settings = args[1].GetBool(); |
| std::string request_origin_string = args[2].GetString(); |
| reset_report::ChromeResetReport::ResetRequestOrigin request_origin = |
| ResetRequestOriginFromString(request_origin_string); |
| |
| DCHECK(brandcode_.empty() || config_fetcher_); |
| if (config_fetcher_ && config_fetcher_->IsActive()) { |
| // Reset once the prefs are fetched. |
| config_fetcher_->SetCallback(base::BindOnce( |
| &ResetSettingsHandler::ResetProfile, base::Unretained(this), |
| callback_id, send_settings, request_origin)); |
| } else { |
| ResetProfile(callback_id, send_settings, request_origin); |
| } |
| } |
| |
| void ResetSettingsHandler::OnResetProfileSettingsDone( |
| std::string callback_id, |
| bool send_feedback, |
| reset_report::ChromeResetReport::ResetRequestOrigin request_origin) { |
| ResolveJavascriptCallback(base::Value(callback_id), base::Value()); |
| if (send_feedback && setting_snapshot_) { |
| ResettableSettingsSnapshot current_snapshot(profile_); |
| int difference = setting_snapshot_->FindDifferentFields(current_snapshot); |
| if (difference) { |
| setting_snapshot_->Subtract(current_snapshot); |
| std::unique_ptr<reset_report::ChromeResetReport> report_proto = |
| SerializeSettingsReportToProto(*setting_snapshot_, difference); |
| if (report_proto) { |
| report_proto->set_reset_request_origin(request_origin); |
| SendSettingsFeedbackProto(*report_proto, profile_); |
| } |
| } |
| } |
| setting_snapshot_.reset(); |
| } |
| |
| void ResetSettingsHandler::HandleGetReportedSettings( |
| const base::Value::List& args) { |
| AllowJavascript(); |
| |
| CHECK_EQ(1U, args.size()); |
| const std::string& callback_id = args[0].GetString(); |
| |
| setting_snapshot_->RequestShortcuts( |
| base::BindOnce(&ResetSettingsHandler::OnGetReportedSettingsDone, |
| callback_weak_ptr_factory_.GetWeakPtr(), callback_id)); |
| } |
| |
| void ResetSettingsHandler::OnGetReportedSettingsDone(std::string callback_id) { |
| base::Value::List list = |
| GetReadableFeedbackForSnapshot(profile_, *setting_snapshot_); |
| ResolveJavascriptCallback(base::Value(callback_id), list); |
| } |
| |
| void ResetSettingsHandler::OnShowResetProfileDialog( |
| const base::Value::List& args) { |
| if (!GetResetter()->IsActive()) { |
| setting_snapshot_ = std::make_unique<ResettableSettingsSnapshot>(profile_); |
| } |
| FetchSettings(); |
| } |
| |
| void ResetSettingsHandler::OnHideResetProfileDialog( |
| const base::Value::List& args) { |
| if (!GetResetter()->IsActive()) |
| setting_snapshot_.reset(); |
| } |
| |
| void ResetSettingsHandler::OnHideResetProfileBanner( |
| const base::Value::List& args) { |
| chrome_prefs::ClearResetTime(profile_); |
| } |
| |
| void ResetSettingsHandler::FetchSettings() { |
| if (brandcode_.empty()) { |
| return; |
| } |
| config_fetcher_ = std::make_unique<BrandcodeConfigFetcher>( |
| g_browser_process->system_network_context_manager() |
| ->GetURLLoaderFactory(), |
| base::BindOnce(&ResetSettingsHandler::OnSettingsFetched, |
| base::Unretained(this)), |
| GURL("https://tools.google.com/service/update2"), brandcode_); |
| } |
| |
| void ResetSettingsHandler::OnSettingsFetched() { |
| DCHECK(config_fetcher_); |
| DCHECK(!config_fetcher_->IsActive()); |
| // The initial prefs is fetched. We are waiting for user pressing 'Reset'. |
| } |
| |
| void ResetSettingsHandler::ResetSettings( |
| ProfileResetter::ResettableFlags resettable_flags, |
| base::OnceClosure callback) { |
| CHECK(!GetResetter()->IsActive()); |
| |
| std::unique_ptr<BrandcodedDefaultSettings> default_settings; |
| if (config_fetcher_) { |
| DCHECK(!config_fetcher_->IsActive()); |
| default_settings = config_fetcher_->GetSettings(); |
| config_fetcher_.reset(); |
| } else { |
| DCHECK(brandcode_.empty()); |
| } |
| |
| // If failed to fetch BrandcodedDefaultSettings or this is an organic |
| // installation, use default settings. |
| if (!default_settings) |
| default_settings = std::make_unique<BrandcodedDefaultSettings>(); |
| |
| GetResetter()->Reset(resettable_flags, std::move(default_settings), |
| std::move(callback)); |
| } |
| |
| void ResetSettingsHandler::ResetProfile( |
| const std::string& callback_id, |
| bool send_settings, |
| reset_report::ChromeResetReport::ResetRequestOrigin request_origin) { |
| ResetSettings( |
| ProfileResetter::ALL, |
| base::BindOnce(&ResetSettingsHandler::OnResetProfileSettingsDone, |
| callback_weak_ptr_factory_.GetWeakPtr(), callback_id, |
| send_settings, request_origin)); |
| |
| base::RecordAction(base::UserMetricsAction("ResetProfile")); |
| } |
| |
| ProfileResetter* ResetSettingsHandler::GetResetter() { |
| if (!resetter_) |
| resetter_ = std::make_unique<ProfileResetter>(profile_); |
| return resetter_.get(); |
| } |
| |
| void ResetSettingsHandler::HandleGetTriggeredResetToolName( |
| const base::Value::List& args) { |
| AllowJavascript(); |
| |
| CHECK_EQ(1U, args.size()); |
| const base::Value& callback_id = args[0]; |
| |
| // Set up the localized strings for the triggered profile reset dialog. |
| // Custom reset tool names are supported on Windows only. |
| std::u16string reset_tool_name; |
| #if BUILDFLAG(IS_WIN) |
| Profile* profile = Profile::FromWebUI(web_ui()); |
| TriggeredProfileResetter* triggered_profile_resetter = |
| TriggeredProfileResetterFactory::GetForBrowserContext(profile); |
| // TriggeredProfileResetter instance will be nullptr for incognito profiles. |
| if (triggered_profile_resetter) { |
| reset_tool_name = triggered_profile_resetter->GetResetToolName(); |
| |
| // Now that a reset UI has been shown, don't trigger again for this profile. |
| triggered_profile_resetter->ClearResetTrigger(); |
| } |
| #endif // BUILDFLAG(IS_WIN) |
| |
| if (reset_tool_name.empty()) { |
| reset_tool_name = l10n_util::GetStringUTF16( |
| IDS_TRIGGERED_RESET_PROFILE_SETTINGS_DEFAULT_TOOL_NAME); |
| } |
| |
| base::Value string_value(reset_tool_name); |
| ResolveJavascriptCallback(callback_id, string_value); |
| } |
| |
| #if BUILDFLAG(IS_CHROMEOS_ASH) |
| void ResetSettingsHandler::OnShowSanitizeDialog(const base::Value::List& args) { |
| FetchSettings(); |
| } |
| |
| void ResetSettingsHandler::HandleSanitizeSettings( |
| const base::Value::List& args) { |
| DCHECK(brandcode_.empty() || config_fetcher_); |
| if (config_fetcher_ && config_fetcher_->IsActive()) { |
| // Reset once the prefs are fetched. |
| config_fetcher_->SetCallback(base::BindOnce( |
| &ResetSettingsHandler::SanitizeSettings, base::Unretained(this))); |
| } else { |
| SanitizeSettings(); |
| } |
| } |
| |
| namespace { |
| |
| std::u16string getBookmarkScriptsFolderName() { |
| return u"[Caution] Scripts"; |
| } |
| |
| void checkBookmarksFolder(std::vector<const bookmarks::BookmarkNode*>& jsnodes, |
| const bookmarks::BookmarkNode* node) { |
| if (node->GetTitledUrlNodeTitle() == getBookmarkScriptsFolderName()) { |
| return; |
| } |
| |
| for (const auto& child : node->children()) { |
| if (child->is_url()) { |
| const GURL u = child->GetTitledUrlNodeUrl(); |
| if (u.SchemeIs("javascript")) { |
| jsnodes.push_back(child.get()); |
| } |
| } else { |
| checkBookmarksFolder(jsnodes, child.get()); |
| } |
| } |
| } |
| |
| const bookmarks::BookmarkNode* getBookmarkScriptsFolder( |
| bookmarks::BookmarkModel* model) { |
| const std::u16string folder_name = getBookmarkScriptsFolderName(); |
| for (const auto& child : model->other_node()->children()) { |
| if (child->GetTitledUrlNodeTitle() == folder_name) { |
| return child.get(); |
| } |
| } |
| return model->AddFolder(model->other_node(), 0, folder_name); |
| } |
| |
| void sanitizeBookmarks(content::BrowserContext* profile) { |
| bookmarks::BookmarkModel* model = |
| BookmarkModelFactory::GetForBrowserContext(profile); |
| if (!model->loaded()) { |
| } else { |
| std::vector<const bookmarks::BookmarkNode*> jsnodes; |
| checkBookmarksFolder(jsnodes, model->root_node()); |
| if (jsnodes.size()) { |
| const bookmarks::BookmarkNode* scripts = |
| settings::getBookmarkScriptsFolder(model); |
| for (auto* const node : jsnodes) { |
| model->Move(node, scripts, 0); |
| } |
| } |
| } |
| } |
| |
| } // namespace |
| |
| void ResetSettingsHandler::SanitizeSettings() { |
| sanitizeBookmarks(profile_); |
| |
| ProfileResetter::ResettableFlags to_sanitize = |
| ProfileResetter::DEFAULT_SEARCH_ENGINE | ProfileResetter::HOMEPAGE | |
| ProfileResetter::CONTENT_SETTINGS | ProfileResetter::EXTENSIONS | |
| ProfileResetter::STARTUP_PAGES | ProfileResetter::PINNED_TABS | |
| ProfileResetter::SHORTCUTS | ProfileResetter::NTP_CUSTOMIZATIONS | |
| ProfileResetter::LANGUAGES; |
| // TODO(b/319446147): get send_feedback flag and pass it down |
| ResetSettings(to_sanitize, |
| base::BindOnce(&ResetSettingsHandler::OnSanitizeDone, |
| callback_weak_ptr_factory_.GetWeakPtr())); |
| |
| base::RecordAction(base::UserMetricsAction("Sanitize")); |
| } |
| |
| void ResetSettingsHandler::OnSanitizeDone() { |
| setting_snapshot_.reset(); |
| PrefService* prefs = g_browser_process->local_state(); |
| prefs->SetBoolean(ash::settings::prefs::kSanitizeCompleted, true); |
| prefs->CommitPendingWrite(); |
| chrome::AttemptRestart(); |
| } |
| |
| #endif // BUILDFLAG(IS_CHROMEOS_ASH) |
| |
| } // namespace settings |