// Copyright (c) 2012 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. // // Implementation of the SafeBrowsingBlockingPage class. #include "chrome/browser/safe_browsing/safe_browsing_blocking_page.h" #include #include "base/bind.h" #include "base/command_line.h" #include "base/i18n/rtl.h" #include "base/lazy_instance.h" #include "base/metrics/field_trial.h" #include "base/metrics/histogram.h" #include "base/prefs/pref_service.h" #include "base/strings/string_number_conversions.h" #include "base/strings/string_piece.h" #include "base/strings/stringprintf.h" #include "base/strings/utf_string_conversions.h" #include "base/time/time.h" #include "base/values.h" #include "chrome/browser/browser_process.h" #include "chrome/browser/profiles/profile.h" #include "chrome/browser/renderer_preferences_util.h" #include "chrome/browser/safe_browsing/malware_details.h" #include "chrome/browser/safe_browsing/ui_manager.h" #include "chrome/browser/tab_contents/tab_util.h" #include "chrome/common/chrome_switches.h" #include "chrome/common/pref_names.h" #include "chrome/common/url_constants.h" #include "chrome/grit/generated_resources.h" #include "chrome/grit/locale_settings.h" #include "components/google/core/browser/google_util.h" #include "content/public/browser/browser_thread.h" #include "content/public/browser/interstitial_page.h" #include "content/public/browser/navigation_controller.h" #include "content/public/browser/user_metrics.h" #include "content/public/browser/web_contents.h" #include "content/public/common/renderer_preferences.h" #include "grit/browser_resources.h" #include "net/base/escape.h" #include "ui/base/l10n/l10n_util.h" using base::UserMetricsAction; using content::BrowserThread; using content::InterstitialPage; using content::OpenURLParams; using content::Referrer; using content::WebContents; namespace { // For malware interstitial pages, we link the problematic URL to Google's // diagnostic page. #if defined(GOOGLE_CHROME_BUILD) const char kSbDiagnosticUrl[] = "http://safebrowsing.clients.google.com/safebrowsing/diagnostic?site=%s&client=googlechrome"; #else const char kSbDiagnosticUrl[] = "http://safebrowsing.clients.google.com/safebrowsing/diagnostic?site=%s&client=chromium"; #endif // URL for malware and phishing, V2. const char kLearnMoreMalwareUrlV2[] = "https://www.google.com/transparencyreport/safebrowsing/"; const char kLearnMorePhishingUrlV2[] = "https://www.google.com/transparencyreport/safebrowsing/"; const char kPrivacyLinkHtml[] = "%s"; // After a malware interstitial where the user opted-in to the report // but clicked "proceed anyway", we delay the call to // MalwareDetails::FinishCollection() by this much time (in // milliseconds). const int64 kMalwareDetailsProceedDelayMilliSeconds = 3000; // The commands returned by the page when the user performs an action. const char kDoReportCommand[] = "doReport"; const char kDontReportCommand[] = "dontReport"; const char kExpandedSeeMoreCommand[] = "expandedSeeMore"; const char kLearnMoreCommand[] = "learnMore2"; const char kProceedCommand[] = "proceed"; const char kShowDiagnosticCommand[] = "showDiagnostic"; const char kShowPrivacyCommand[] = "showPrivacy"; const char kTakeMeBackCommand[] = "takeMeBack"; // Other constants used to communicate with the JavaScript. const char kBoxChecked[] = "boxchecked"; const char kDisplayCheckBox[] = "displaycheckbox"; // Constants for the Experience Sampling instrumentation. const char kEventNameMalware[] = "safebrowsing_interstitial_"; const char kEventNameHarmful[] = "harmful_interstitial_"; const char kEventNamePhishing[] = "phishing_interstitial_"; const char kEventNameOther[] = "safebrowsing_other_interstitial_"; base::LazyInstance g_unsafe_resource_map = LAZY_INSTANCE_INITIALIZER; } // namespace // static SafeBrowsingBlockingPageFactory* SafeBrowsingBlockingPage::factory_ = NULL; // The default SafeBrowsingBlockingPageFactory. Global, made a singleton so we // don't leak it. class SafeBrowsingBlockingPageFactoryImpl : public SafeBrowsingBlockingPageFactory { public: SafeBrowsingBlockingPage* CreateSafeBrowsingPage( SafeBrowsingUIManager* ui_manager, WebContents* web_contents, const SafeBrowsingBlockingPage::UnsafeResourceList& unsafe_resources) override { return new SafeBrowsingBlockingPage(ui_manager, web_contents, unsafe_resources); } private: friend struct base::DefaultLazyInstanceTraits< SafeBrowsingBlockingPageFactoryImpl>; SafeBrowsingBlockingPageFactoryImpl() { } DISALLOW_COPY_AND_ASSIGN(SafeBrowsingBlockingPageFactoryImpl); }; static base::LazyInstance g_safe_browsing_blocking_page_factory_impl = LAZY_INSTANCE_INITIALIZER; // static const void* SafeBrowsingBlockingPage::kTypeForTesting = &SafeBrowsingBlockingPage::kTypeForTesting; SafeBrowsingBlockingPage::SafeBrowsingBlockingPage( SafeBrowsingUIManager* ui_manager, WebContents* web_contents, const UnsafeResourceList& unsafe_resources) : SecurityInterstitialPage(web_contents, unsafe_resources[0].url), malware_details_proceed_delay_ms_( kMalwareDetailsProceedDelayMilliSeconds), ui_manager_(ui_manager), report_loop_(NULL), is_main_frame_load_blocked_(IsMainPageLoadBlocked(unsafe_resources)), unsafe_resources_(unsafe_resources), proceeded_(false) { bool malware = false; bool harmful = false; bool phishing = false; for (UnsafeResourceList::const_iterator iter = unsafe_resources_.begin(); iter != unsafe_resources_.end(); ++iter) { const UnsafeResource& resource = *iter; SBThreatType threat_type = resource.threat_type; if (threat_type == SB_THREAT_TYPE_URL_MALWARE || threat_type == SB_THREAT_TYPE_CLIENT_SIDE_MALWARE_URL) { malware = true; } else if (threat_type == SB_THREAT_TYPE_URL_UNWANTED) { harmful = true; } else { DCHECK(threat_type == SB_THREAT_TYPE_URL_PHISHING || threat_type == SB_THREAT_TYPE_CLIENT_SIDE_PHISHING_URL); phishing = true; } } DCHECK(phishing || malware || harmful); if (malware) interstitial_reason_ = SB_REASON_MALWARE; else if (harmful) interstitial_reason_ = SB_REASON_HARMFUL; else interstitial_reason_ = SB_REASON_PHISHING; // This must be done after calculating |interstitial_reason_| above. // Use same prefix for UMA as for Rappor. metrics_helper_.reset(new SecurityInterstitialMetricsHelper( web_contents, request_url(), GetMetricPrefix(), GetMetricPrefix(), SecurityInterstitialMetricsHelper::REPORT_RAPPOR, GetSamplingEventName())); metrics_helper_->RecordUserDecision(SecurityInterstitialMetricsHelper::SHOW); metrics_helper_->RecordUserInteraction( SecurityInterstitialMetricsHelper::TOTAL_VISITS); if (IsPrefEnabled(prefs::kSafeBrowsingProceedAnywayDisabled)) { metrics_helper_->RecordUserDecision( SecurityInterstitialMetricsHelper::PROCEEDING_DISABLED); } if (!is_main_frame_load_blocked_) { navigation_entry_index_to_remove_ = web_contents->GetController().GetLastCommittedEntryIndex(); } else { navigation_entry_index_to_remove_ = -1; } // Start computing malware details. They will be sent only // if the user opts-in on the blocking page later. // If there's more than one malicious resources, it means the user // clicked through the first warning, so we don't prepare additional // reports. if (unsafe_resources.size() == 1 && unsafe_resources[0].threat_type == SB_THREAT_TYPE_URL_MALWARE && malware_details_.get() == NULL && CanShowMalwareDetailsOption()) { malware_details_ = MalwareDetails::NewMalwareDetails( ui_manager_, web_contents, unsafe_resources[0]); } } bool SafeBrowsingBlockingPage::CanShowMalwareDetailsOption() { return (!web_contents()->GetBrowserContext()->IsOffTheRecord() && web_contents()->GetURL().SchemeIs(url::kHttpScheme)); } SafeBrowsingBlockingPage::~SafeBrowsingBlockingPage() { } void SafeBrowsingBlockingPage::CommandReceived(const std::string& cmd) { std::string command(cmd); // Make a local copy so we can modify it. // The Jasonified response has quotes, remove them. if (command.length() > 1 && command[0] == '"') { command = command.substr(1, command.length() - 2); } if (command == "pageLoadComplete") { // content::WaitForRenderFrameReady sends this message when the page // load completes. Ignore it. return; } if (command == kDoReportCommand) { SetReportingPreference(true); return; } if (command == kDontReportCommand) { SetReportingPreference(false); return; } if (command == kLearnMoreCommand) { // User pressed "Learn more". metrics_helper_->RecordUserInteraction( SecurityInterstitialMetricsHelper::SHOW_LEARN_MORE); GURL learn_more_url( interstitial_reason_ == SB_REASON_PHISHING ? kLearnMorePhishingUrlV2 : kLearnMoreMalwareUrlV2); learn_more_url = google_util::AppendGoogleLocaleParam( learn_more_url, g_browser_process->GetApplicationLocale()); OpenURLParams params(learn_more_url, Referrer(), CURRENT_TAB, ui::PAGE_TRANSITION_LINK, false); web_contents()->OpenURL(params); return; } if (command == kShowPrivacyCommand) { // User pressed "Safe Browsing privacy policy". metrics_helper_->RecordUserInteraction( SecurityInterstitialMetricsHelper::SHOW_PRIVACY_POLICY); GURL privacy_url( l10n_util::GetStringUTF8(IDS_SAFE_BROWSING_PRIVACY_POLICY_URL)); privacy_url = google_util::AppendGoogleLocaleParam( privacy_url, g_browser_process->GetApplicationLocale()); OpenURLParams params(privacy_url, Referrer(), CURRENT_TAB, ui::PAGE_TRANSITION_LINK, false); web_contents()->OpenURL(params); return; } bool proceed_blocked = false; if (command == kProceedCommand) { if (IsPrefEnabled(prefs::kSafeBrowsingProceedAnywayDisabled)) { proceed_blocked = true; } else { metrics_helper_->RecordUserDecision( SecurityInterstitialMetricsHelper::PROCEED); interstitial_page()->Proceed(); // |this| has been deleted after Proceed() returns. return; } } if (command == kTakeMeBackCommand || proceed_blocked) { // Don't record the user action here because there are other ways of // triggering DontProceed, like clicking the back button. if (is_main_frame_load_blocked_) { // If the load is blocked, we want to close the interstitial and discard // the pending entry. interstitial_page()->DontProceed(); // |this| has been deleted after DontProceed() returns. return; } // Otherwise the offending entry has committed, and we need to go back or // to a safe page. We will close the interstitial when that page commits. if (web_contents()->GetController().CanGoBack()) { web_contents()->GetController().GoBack(); } else { web_contents()->GetController().LoadURL( GURL(chrome::kChromeUINewTabURL), content::Referrer(), ui::PAGE_TRANSITION_AUTO_TOPLEVEL, std::string()); } return; } // The "report error" and "show diagnostic" commands can have a number // appended to them, which is the index of the element they apply to. size_t element_index = 0; size_t colon_index = command.find(':'); if (colon_index != std::string::npos) { DCHECK(colon_index < command.size() - 1); int result_int = 0; bool result = base::StringToInt(base::StringPiece(command.begin() + colon_index + 1, command.end()), &result_int); command = command.substr(0, colon_index); if (result) element_index = static_cast(result_int); } if (element_index >= unsafe_resources_.size()) { NOTREACHED(); return; } std::string bad_url_spec = unsafe_resources_[element_index].url.spec(); if (command == kShowDiagnosticCommand) { // We're going to take the user to Google's SafeBrowsing diagnostic page. metrics_helper_->RecordUserInteraction( SecurityInterstitialMetricsHelper::SHOW_DIAGNOSTIC); std::string diagnostic = base::StringPrintf(kSbDiagnosticUrl, net::EscapeQueryParamValue(bad_url_spec, true).c_str()); GURL diagnostic_url(diagnostic); diagnostic_url = google_util::AppendGoogleLocaleParam( diagnostic_url, g_browser_process->GetApplicationLocale()); DCHECK(unsafe_resources_[element_index].threat_type == SB_THREAT_TYPE_URL_MALWARE || unsafe_resources_[element_index].threat_type == SB_THREAT_TYPE_CLIENT_SIDE_MALWARE_URL || unsafe_resources_[element_index].threat_type == SB_THREAT_TYPE_URL_UNWANTED); OpenURLParams params( diagnostic_url, Referrer(), CURRENT_TAB, ui::PAGE_TRANSITION_LINK, false); web_contents()->OpenURL(params); return; } if (command == kExpandedSeeMoreCommand) { metrics_helper_->RecordUserInteraction( SecurityInterstitialMetricsHelper::SHOW_ADVANCED); return; } NOTREACHED() << "Unexpected command: " << command; } void SafeBrowsingBlockingPage::OverrideRendererPrefs( content::RendererPreferences* prefs) { Profile* profile = Profile::FromBrowserContext( web_contents()->GetBrowserContext()); renderer_preferences_util::UpdateFromSystemSettings( prefs, profile, web_contents()); } void SafeBrowsingBlockingPage::SetReportingPreference(bool report) { Profile* profile = Profile::FromBrowserContext( web_contents()->GetBrowserContext()); PrefService* pref = profile->GetPrefs(); pref->SetBoolean(prefs::kSafeBrowsingExtendedReportingEnabled, report); UMA_HISTOGRAM_BOOLEAN("SB2.SetExtendedReportingEnabled", report); } void SafeBrowsingBlockingPage::OnProceed() { proceeded_ = true; // Send the malware details, if we opted to. FinishMalwareDetails(malware_details_proceed_delay_ms_); NotifySafeBrowsingUIManager(ui_manager_, unsafe_resources_, true); // Check to see if some new notifications of unsafe resources have been // received while we were showing the interstitial. UnsafeResourceMap* unsafe_resource_map = GetUnsafeResourcesMap(); UnsafeResourceMap::iterator iter = unsafe_resource_map->find(web_contents()); SafeBrowsingBlockingPage* blocking_page = NULL; if (iter != unsafe_resource_map->end() && !iter->second.empty()) { // Build an interstitial for all the unsafe resources notifications. // Don't show it now as showing an interstitial while an interstitial is // already showing would cause DontProceed() to be invoked. blocking_page = factory_->CreateSafeBrowsingPage(ui_manager_, web_contents(), iter->second); unsafe_resource_map->erase(iter); } // Now that this interstitial is gone, we can show the new one. if (blocking_page) blocking_page->Show(); } const void* SafeBrowsingBlockingPage::GetTypeForTesting() const { return SafeBrowsingBlockingPage::kTypeForTesting; } bool SafeBrowsingBlockingPage::ShouldCreateNewNavigation() const { return is_main_frame_load_blocked_; } void SafeBrowsingBlockingPage::OnDontProceed() { // We could have already called Proceed(), in which case we must not notify // the SafeBrowsingUIManager again, as the client has been deleted. if (proceeded_) return; if (!IsPrefEnabled(prefs::kSafeBrowsingProceedAnywayDisabled)) { metrics_helper_->RecordUserDecision( SecurityInterstitialMetricsHelper::DONT_PROCEED); } // Send the malware details, if we opted to. FinishMalwareDetails(0); // No delay NotifySafeBrowsingUIManager(ui_manager_, unsafe_resources_, false); // The user does not want to proceed, clear the queued unsafe resources // notifications we received while the interstitial was showing. UnsafeResourceMap* unsafe_resource_map = GetUnsafeResourcesMap(); UnsafeResourceMap::iterator iter = unsafe_resource_map->find(web_contents()); if (iter != unsafe_resource_map->end() && !iter->second.empty()) { NotifySafeBrowsingUIManager(ui_manager_, iter->second, false); unsafe_resource_map->erase(iter); } // We don't remove the navigation entry if the tab is being destroyed as this // would trigger a navigation that would cause trouble as the render view host // for the tab has by then already been destroyed. We also don't delete the // current entry if it has been committed again, which is possible on a page // that had a subresource warning. int last_committed_index = web_contents()->GetController().GetLastCommittedEntryIndex(); if (navigation_entry_index_to_remove_ != -1 && navigation_entry_index_to_remove_ != last_committed_index && !web_contents()->IsBeingDestroyed()) { CHECK(web_contents()->GetController().RemoveEntryAtIndex( navigation_entry_index_to_remove_)); navigation_entry_index_to_remove_ = -1; } } void SafeBrowsingBlockingPage::FinishMalwareDetails(int64 delay_ms) { if (malware_details_.get() == NULL) return; // Not all interstitials have malware details (eg phishing). DCHECK_EQ(interstitial_reason_, SB_REASON_MALWARE); const bool enabled = IsPrefEnabled(prefs::kSafeBrowsingExtendedReportingEnabled); UMA_HISTOGRAM_BOOLEAN("SB2.ExtendedReportingIsEnabled", enabled); if (enabled) { // Finish the malware details collection, send it over. BrowserThread::PostDelayedTask( BrowserThread::IO, FROM_HERE, base::Bind(&MalwareDetails::FinishCollection, malware_details_.get()), base::TimeDelta::FromMilliseconds(delay_ms)); } } bool SafeBrowsingBlockingPage::IsPrefEnabled(const char* pref) { Profile* profile = Profile::FromBrowserContext(web_contents()->GetBrowserContext()); return profile->GetPrefs()->GetBoolean(pref); } // static void SafeBrowsingBlockingPage::NotifySafeBrowsingUIManager( SafeBrowsingUIManager* ui_manager, const UnsafeResourceList& unsafe_resources, bool proceed) { BrowserThread::PostTask( BrowserThread::IO, FROM_HERE, base::Bind(&SafeBrowsingUIManager::OnBlockingPageDone, ui_manager, unsafe_resources, proceed)); } // static SafeBrowsingBlockingPage::UnsafeResourceMap* SafeBrowsingBlockingPage::GetUnsafeResourcesMap() { return g_unsafe_resource_map.Pointer(); } // static SafeBrowsingBlockingPage* SafeBrowsingBlockingPage::CreateBlockingPage( SafeBrowsingUIManager* ui_manager, WebContents* web_contents, const UnsafeResource& unsafe_resource) { std::vector resources; resources.push_back(unsafe_resource); // Set up the factory if this has not been done already (tests do that // before this method is called). if (!factory_) factory_ = g_safe_browsing_blocking_page_factory_impl.Pointer(); return factory_->CreateSafeBrowsingPage(ui_manager, web_contents, resources); } // static void SafeBrowsingBlockingPage::ShowBlockingPage( SafeBrowsingUIManager* ui_manager, const UnsafeResource& unsafe_resource) { DVLOG(1) << __FUNCTION__ << " " << unsafe_resource.url.spec(); WebContents* web_contents = tab_util::GetWebContentsByID( unsafe_resource.render_process_host_id, unsafe_resource.render_view_id); InterstitialPage* interstitial = InterstitialPage::GetInterstitialPage(web_contents); if (interstitial && !unsafe_resource.is_subresource) { // There is already an interstitial showing and we are about to display a // new one for the main frame. Just hide the current one, it is now // irrelevent interstitial->DontProceed(); interstitial = NULL; } if (!interstitial) { // There are no interstitial currently showing in that tab, go ahead and // show this interstitial. SafeBrowsingBlockingPage* blocking_page = CreateBlockingPage(ui_manager, web_contents, unsafe_resource); blocking_page->Show(); return; } // This is an interstitial for a page's resource, let's queue it. UnsafeResourceMap* unsafe_resource_map = GetUnsafeResourcesMap(); (*unsafe_resource_map)[web_contents].push_back(unsafe_resource); } // static bool SafeBrowsingBlockingPage::IsMainPageLoadBlocked( const UnsafeResourceList& unsafe_resources) { // Client-side phishing detection interstitials never block the main frame // load, since they happen after the page is finished loading. if (unsafe_resources[0].threat_type == SB_THREAT_TYPE_CLIENT_SIDE_PHISHING_URL) { return false; } // Otherwise, check the threat type. return unsafe_resources.size() == 1 && !unsafe_resources[0].is_subresource; } std::string SafeBrowsingBlockingPage::GetMetricPrefix() const { switch (interstitial_reason_) { case SB_REASON_MALWARE: return "malware"; case SB_REASON_HARMFUL: return "harmful"; case SB_REASON_PHISHING: return "phishing"; } NOTREACHED(); return std::string(); } std::string SafeBrowsingBlockingPage::GetSamplingEventName() const { switch (interstitial_reason_) { case SB_REASON_MALWARE: return kEventNameMalware; case SB_REASON_HARMFUL: return kEventNameHarmful; case SB_REASON_PHISHING: return kEventNamePhishing; default: return kEventNameOther; } } void SafeBrowsingBlockingPage::PopulateInterstitialStrings( base::DictionaryValue* load_time_data) { CHECK(load_time_data); CHECK(!unsafe_resources_.empty()); load_time_data->SetString("type", "SAFEBROWSING"); load_time_data->SetString( "tabTitle", l10n_util::GetStringUTF16(IDS_SAFEBROWSING_V3_TITLE)); load_time_data->SetString( "openDetails", l10n_util::GetStringUTF16(IDS_SAFEBROWSING_V3_OPEN_DETAILS_BUTTON)); load_time_data->SetString( "closeDetails", l10n_util::GetStringUTF16(IDS_SAFEBROWSING_V3_CLOSE_DETAILS_BUTTON)); load_time_data->SetString( "primaryButtonText", l10n_util::GetStringUTF16(IDS_SAFEBROWSING_OVERRIDABLE_SAFETY_BUTTON)); load_time_data->SetBoolean( "overridable", !IsPrefEnabled(prefs::kSafeBrowsingProceedAnywayDisabled)); switch (interstitial_reason_) { case SB_REASON_MALWARE: PopulateMalwareLoadTimeData(load_time_data); break; case SB_REASON_HARMFUL: PopulateHarmfulLoadTimeData(load_time_data); break; case SB_REASON_PHISHING: PopulatePhishingLoadTimeData(load_time_data); break; } } void SafeBrowsingBlockingPage::PopulateExtendedReportingOption( base::DictionaryValue* load_time_data) { // Only show checkbox if !(HTTPS || incognito-mode). const bool show = CanShowMalwareDetailsOption(); load_time_data->SetBoolean(kDisplayCheckBox, show); if (!show) return; const std::string privacy_link = base::StringPrintf( kPrivacyLinkHtml, l10n_util::GetStringUTF8( IDS_SAFE_BROWSING_PRIVACY_POLICY_PAGE).c_str()); load_time_data->SetString( "optInLink", l10n_util::GetStringFUTF16(IDS_SAFE_BROWSING_MALWARE_REPORTING_AGREE, base::UTF8ToUTF16(privacy_link))); load_time_data->SetBoolean( kBoxChecked, IsPrefEnabled(prefs::kSafeBrowsingExtendedReportingEnabled)); } void SafeBrowsingBlockingPage::PopulateMalwareLoadTimeData( base::DictionaryValue* load_time_data) { load_time_data->SetBoolean("phishing", false); load_time_data->SetString( "heading", l10n_util::GetStringUTF16(IDS_MALWARE_V3_HEADING)); load_time_data->SetString( "primaryParagraph", l10n_util::GetStringFUTF16( IDS_MALWARE_V3_PRIMARY_PARAGRAPH, GetFormattedHostName())); load_time_data->SetString( "explanationParagraph", is_main_frame_load_blocked_ ? l10n_util::GetStringFUTF16( IDS_MALWARE_V3_EXPLANATION_PARAGRAPH, GetFormattedHostName()) : l10n_util::GetStringFUTF16( IDS_MALWARE_V3_EXPLANATION_PARAGRAPH_SUBRESOURCE, base::UTF8ToUTF16(web_contents()->GetURL().host()), GetFormattedHostName())); load_time_data->SetString( "finalParagraph", l10n_util::GetStringUTF16(IDS_MALWARE_V3_PROCEED_PARAGRAPH)); PopulateExtendedReportingOption(load_time_data); } void SafeBrowsingBlockingPage::PopulateHarmfulLoadTimeData( base::DictionaryValue* load_time_data) { load_time_data->SetBoolean("phishing", false); load_time_data->SetString( "heading", l10n_util::GetStringUTF16(IDS_HARMFUL_V3_HEADING)); load_time_data->SetString( "primaryParagraph", l10n_util::GetStringFUTF16( IDS_HARMFUL_V3_PRIMARY_PARAGRAPH, GetFormattedHostName())); load_time_data->SetString( "explanationParagraph", l10n_util::GetStringFUTF16( IDS_HARMFUL_V3_EXPLANATION_PARAGRAPH, GetFormattedHostName())); load_time_data->SetString( "finalParagraph", l10n_util::GetStringUTF16(IDS_HARMFUL_V3_PROCEED_PARAGRAPH)); PopulateExtendedReportingOption(load_time_data); } void SafeBrowsingBlockingPage::PopulatePhishingLoadTimeData( base::DictionaryValue* load_time_data) { load_time_data->SetBoolean("phishing", true); load_time_data->SetString( "heading", l10n_util::GetStringUTF16(IDS_PHISHING_V3_HEADING)); load_time_data->SetString( "primaryParagraph", l10n_util::GetStringFUTF16( IDS_PHISHING_V3_PRIMARY_PARAGRAPH, GetFormattedHostName())); load_time_data->SetString( "explanationParagraph", l10n_util::GetStringFUTF16(IDS_PHISHING_V3_EXPLANATION_PARAGRAPH, GetFormattedHostName())); load_time_data->SetString( "finalParagraph", l10n_util::GetStringUTF16(IDS_PHISHING_V3_PROCEED_PARAGRAPH)); PopulateExtendedReportingOption(load_time_data); }