| Index: chrome/browser/safe_browsing/safe_browsing_navigation_observer_browsertest.cc | 
| diff --git a/chrome/browser/safe_browsing/safe_browsing_navigation_observer_browsertest.cc b/chrome/browser/safe_browsing/safe_browsing_navigation_observer_browsertest.cc | 
| new file mode 100644 | 
| index 0000000000000000000000000000000000000000..dd7eb32f57217d98049bbb5a605dfc01b813de73 | 
| --- /dev/null | 
| +++ b/chrome/browser/safe_browsing/safe_browsing_navigation_observer_browsertest.cc | 
| @@ -0,0 +1,777 @@ | 
| +// Copyright 2016 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 "base/strings/stringprintf.h" | 
| +#include "chrome/browser/chrome_notification_types.h" | 
| +#include "chrome/browser/download/download_prefs.h" | 
| +#include "chrome/browser/profiles/profile.h" | 
| +#include "chrome/browser/safe_browsing/safe_browsing_navigation_observer.h" | 
| +#include "chrome/browser/safe_browsing/safe_browsing_navigation_observer_manager.h" | 
| +#include "chrome/browser/sessions/session_tab_helper.h" | 
| +#include "chrome/browser/ui/browser.h" | 
| +#include "chrome/browser/ui/tabs/tab_strip_model.h" | 
| +#include "chrome/common/pref_names.h" | 
| +#include "chrome/test/base/in_process_browser_test.h" | 
| +#include "chrome/test/base/ui_test_utils.h" | 
| +#include "components/prefs/pref_service.h" | 
| +#include "content/public/browser/browser_context.h" | 
| +#include "content/public/browser/download_item.h" | 
| +#include "content/public/browser/download_manager.h" | 
| +#include "content/public/browser/notification_service.h" | 
| +#include "content/public/test/browser_test_utils.h" | 
| +#include "content/public/test/test_navigation_observer.h" | 
| +#include "content/public/test/test_utils.h" | 
| +#include "net/dns/mock_host_resolver.h" | 
| +#include "net/test/embedded_test_server/embedded_test_server.h" | 
| +#include "url/gurl.h" | 
| +#include "url/url_canon.h" | 
| + | 
| +namespace safe_browsing { | 
| + | 
| +const char kSingleFrameTestURL[] = | 
| +    "/safe_browsing/download_protection/navigation_observer/" | 
| +    "navigation_observer_tests.html"; | 
| +const char kMultiFrameTestURL[] = | 
| +    "/safe_browsing/download_protection/navigation_observer/" | 
| +    "navigation_observer_multi_frame_tests.html"; | 
| +const char kRedirectURL[] = | 
| +    "/safe_browsing/download_protection/navigation_observer/redirect.html"; | 
| +const char kDownloadDataURL[] = | 
| +    "data:application/octet-stream;base64,a2poYWxrc2hkbGtoYXNka2xoYXNsa2RoYWxra" | 
| +    "GtoYWxza2hka2xzamFoZGxramhhc2xka2hhc2xrZGgKYXNrZGpoa2FzZGpoYWtzaGRrYXNoZGt" | 
| +    "oYXNrZGhhc2tkaGthc2hka2Foc2RraGFrc2hka2FzaGRraGFzCmFza2pkaGFrc2hkbSxjbmtza" | 
| +    "mFoZGtoYXNrZGhhc2tka2hrYXNkCjg3MzQ2ODEyNzQ2OGtqc2hka2FoZHNrZGhraApha3NqZGt" | 
| +    "hc2Roa3NkaGthc2hka2FzaGtkaAohISomXkAqJl4qYWhpZGFzeWRpeWlhc1xcb1wKa2Fqc2Roa" | 
| +    "2FzaGRrYXNoZGsKYWtzamRoc2tkaAplbmQK"; | 
| +const char kIframeDirectDownloadURL[] = | 
| +    "/safe_browsing/download_protection/navigation_observer/iframe.html"; | 
| +const char kIframeRetargetingURL[] = | 
| +    "/safe_browsing/download_protection/navigation_observer/" | 
| +    "iframe_retargeting.html"; | 
| +const char kDownloadItemURL[] = "/safe_browsing/download_protection/signed.exe"; | 
| + | 
| +// Test class to help create SafeBrowsingNavigationObservers for each | 
| +// WebContents before they are actually installed through AttachTabHelper. | 
| +class TestNavigationObserverManager | 
| +    : public SafeBrowsingNavigationObserverManager { | 
| + public: | 
| +  TestNavigationObserverManager() : SafeBrowsingNavigationObserverManager() { | 
| +    registrar_.Add(this, chrome::NOTIFICATION_TAB_ADDED, | 
| +                   content::NotificationService::AllSources()); | 
| +  } | 
| + | 
| +  void Observe(int type, | 
| +               const content::NotificationSource& source, | 
| +               const content::NotificationDetails& details) override { | 
| +    if (type == chrome::NOTIFICATION_TAB_ADDED) { | 
| +      content::WebContents* dest_content = | 
| +          content::Details<content::WebContents>(details).ptr(); | 
| +      DCHECK(dest_content); | 
| +      observer_list_.push_back( | 
| +          new SafeBrowsingNavigationObserver(dest_content, this)); | 
| +      DCHECK(observer_list_.back()); | 
| +    } else if (type == chrome::NOTIFICATION_RETARGETING) { | 
| +      RecordRetargeting(details); | 
| +    } | 
| +  } | 
| + | 
| + protected: | 
| +  ~TestNavigationObserverManager() override { observer_list_.clear(); } | 
| + | 
| + private: | 
| +  std::vector<SafeBrowsingNavigationObserver*> observer_list_; | 
| +}; | 
| + | 
| +class SBNavigationObserverBrowserTest : public InProcessBrowserTest { | 
| + public: | 
| +  SBNavigationObserverBrowserTest() {} | 
| + | 
| +  void SetUpOnMainThread() override { | 
| +    // Disable Safe Browsing service since it is irrelevant to this test. | 
| +    browser()->profile()->GetPrefs()->SetBoolean(prefs::kSafeBrowsingEnabled, | 
| +                                                 false); | 
| +    ASSERT_TRUE(embedded_test_server()->Start()); | 
| +    host_resolver()->AddRule("*", "127.0.0.1"); | 
| +    // Navigate to test page. | 
| +    ui_test_utils::NavigateToURL( | 
| +        browser(), embedded_test_server()->GetURL(kSingleFrameTestURL)); | 
| +    observer_manager_ = new TestNavigationObserverManager(); | 
| +    observer_ = new SafeBrowsingNavigationObserver( | 
| +        browser()->tab_strip_model()->GetActiveWebContents(), | 
| +        observer_manager_); | 
| +    ASSERT_TRUE(observer_); | 
| +    ASSERT_TRUE(InitialSetup()); | 
| +  } | 
| + | 
| +  bool InitialSetup() { | 
| +    if (!browser()) | 
| +      return false; | 
| + | 
| +    if (!downloads_directory_.CreateUniqueTempDir()) | 
| +      return false; | 
| + | 
| +    // Set up default download path. | 
| +    browser()->profile()->GetPrefs()->SetFilePath( | 
| +        prefs::kDownloadDefaultDirectory, downloads_directory_.GetPath()); | 
| +    browser()->profile()->GetPrefs()->SetFilePath( | 
| +        prefs::kSaveFileDefaultDirectory, downloads_directory_.GetPath()); | 
| +    browser()->profile()->GetPrefs()->SetBoolean(prefs::kPromptForDownload, | 
| +                                                 false); | 
| +    content::DownloadManager* manager = | 
| +        content::BrowserContext::GetDownloadManager(browser()->profile()); | 
| +    DownloadPrefs::FromDownloadManager(manager)->ResetAutoOpen(); | 
| +    manager->RemoveAllDownloads(); | 
| + | 
| +    return true; | 
| +  } | 
| + | 
| +  void TearDownOnMainThread() override { delete observer_; } | 
| + | 
| +  // Most test cases will trigger downloads, though we don't really care if | 
| +  // download completed or not. So we cancel downloads as soon as we record | 
| +  // all the navigation events we need. | 
| +  void CancelDownloads() { | 
| +    std::vector<content::DownloadItem*> download_items; | 
| +    content::DownloadManager* manager = | 
| +        content::BrowserContext::GetDownloadManager(browser()->profile()); | 
| +    manager->GetAllDownloads(&download_items); | 
| +    for (auto item : download_items) { | 
| +      if (!item->IsDone()) | 
| +        item->Cancel(true); | 
| +    } | 
| +  } | 
| + | 
| +  // This function needs javascript support, only works on | 
| +  // navigation_observer_tests.html and | 
| +  // navigation_observer_multi_frame_tests.html. | 
| +  void ClickTestLink(const char* test_name, | 
| +                     int number_of_navigations) { | 
| +    TabStripModel* tab_strip = browser()->tab_strip_model(); | 
| +    content::WebContents* current_web_contents = | 
| +        tab_strip->GetActiveWebContents(); | 
| +    ASSERT_TRUE(content::WaitForLoadStop(current_web_contents)); | 
| +    content::TestNavigationObserver navigation_observer( | 
| +      current_web_contents, | 
| +      number_of_navigations); | 
| +    navigation_observer.StartWatchingNewWebContents(); | 
| +    // Execute test. | 
| +    std::string script = base::StringPrintf("clickLink('%s');", test_name); | 
| +    ASSERT_TRUE(content::ExecuteScript(current_web_contents, script)); | 
| +    // Wait for navigations on current tab and new tab (if any) to finish. | 
| +    navigation_observer.Wait(); | 
| +    navigation_observer.StopWatchingNewWebContents(); | 
| +    // Cancel unfinished download if any. | 
| +    CancelDownloads(); | 
| +  } | 
| + | 
| +  void VerifyNavigationEvent(const GURL& expected_source_url, | 
| +                             const GURL& expected_source_main_frame_url, | 
| +                             const GURL& expected_original_request_url, | 
| +                             const GURL& expected_destination_url, | 
| +                             bool expected_is_user_initiated, | 
| +                             bool expected_has_committed, | 
| +                             bool expected_has_server_redirect, | 
| +                             const NavigationEvent& actual_nav_event) { | 
| +    EXPECT_EQ(expected_source_url, actual_nav_event.source_url); | 
| +    EXPECT_EQ(expected_source_main_frame_url, | 
| +              actual_nav_event.source_main_frame_url); | 
| +    EXPECT_EQ(expected_original_request_url, | 
| +              actual_nav_event.original_request_url); | 
| +    EXPECT_EQ(expected_destination_url, actual_nav_event.destination_url); | 
| +    EXPECT_EQ(expected_is_user_initiated, actual_nav_event.is_user_initiated); | 
| +    EXPECT_EQ(expected_has_committed, actual_nav_event.has_committed); | 
| +    EXPECT_EQ(expected_has_server_redirect, | 
| +              actual_nav_event.has_server_redirect); | 
| +  } | 
| + | 
| +  void VerifyHostToIpMap() { | 
| +    // Since all testing pages have the same host, there is only one entry in | 
| +    // host_to_ip_map_. | 
| +    SafeBrowsingNavigationObserverManager::HostToIpMap* actual_host_ip_map = | 
| +        host_to_ip_map(); | 
| +    ASSERT_EQ(std::size_t(1), actual_host_ip_map->size()); | 
| +    auto ip_list = | 
| +        actual_host_ip_map->at(embedded_test_server()->base_url().host()); | 
| +    ASSERT_EQ(std::size_t(1), ip_list.size()); | 
| +    EXPECT_EQ(embedded_test_server()->host_port_pair().host(), | 
| +              ip_list.back().ip); | 
| +  } | 
| + | 
| +  SafeBrowsingNavigationObserverManager::NavigationMap* navigation_map() { | 
| +    return observer_manager_->navigation_map(); | 
| +  } | 
| + | 
| +  SafeBrowsingNavigationObserverManager::HostToIpMap* host_to_ip_map() { | 
| +    return observer_manager_->host_to_ip_map(); | 
| +  } | 
| + | 
| + protected: | 
| +  SafeBrowsingNavigationObserverManager* observer_manager_; | 
| +  SafeBrowsingNavigationObserver* observer_; | 
| + | 
| + private: | 
| +  base::ScopedTempDir downloads_directory_; | 
| +}; | 
| + | 
| +// Click on a link and start download on the same page. | 
| +IN_PROC_BROWSER_TEST_F(SBNavigationObserverBrowserTest, DirectDownload) { | 
| +  ClickTestLink("direct_download", 1); | 
| +  GURL initial_url = embedded_test_server()->GetURL(kSingleFrameTestURL); | 
| +  GURL download_url = embedded_test_server()->GetURL(kDownloadItemURL); | 
| +  auto nav_map = navigation_map(); | 
| +  ASSERT_TRUE(nav_map); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(download_url).size()); | 
| +  // Since this test uses javascript to mimic clicking on a link (no actual user | 
| +  // gesture), and DidGetUserInteraction() does not respond to ExecuteScript(), | 
| +  // therefore is_user_initiated is false. | 
| +  VerifyNavigationEvent(initial_url,   // source_url | 
| +                        initial_url,   // source_main_frame_url | 
| +                        download_url,  // original_request_url | 
| +                        download_url,  // destination_url | 
| +                        false,         // is_user_initiated, | 
| +                        false,         // has_committed | 
| +                        false,         // has_server_redirect | 
| +                        nav_map->at(download_url).at(0)); | 
| +  VerifyHostToIpMap(); | 
| +} | 
| + | 
| +// Click on a link with rel="noreferrer" attribute, and start download on the | 
| +// same page. | 
| +IN_PROC_BROWSER_TEST_F(SBNavigationObserverBrowserTest, | 
| +                       DirectDownloadNoReferrer) { | 
| +  ClickTestLink("direct_download_noreferrer", 1); | 
| +  GURL initial_url = embedded_test_server()->GetURL(kSingleFrameTestURL); | 
| +  GURL download_url = embedded_test_server()->GetURL(kDownloadItemURL); | 
| +  auto nav_map = navigation_map(); | 
| +  ASSERT_TRUE(nav_map); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(download_url).size()); | 
| +  VerifyNavigationEvent(initial_url,   // source_url | 
| +                        initial_url,   // source_main_frame_url | 
| +                        download_url,  // original_request_url | 
| +                        download_url,  // destination_url | 
| +                        false,         // is_user_initiated, | 
| +                        false,         // has_committed | 
| +                        false,         // has_server_redirect | 
| +                        nav_map->at(download_url).at(0)); | 
| +  VerifyHostToIpMap(); | 
| +} | 
| + | 
| +// Click on a link with rel="noreferrer" attribute, and start download in a | 
| +// new tab using target=_blank. | 
| +IN_PROC_BROWSER_TEST_F(SBNavigationObserverBrowserTest, | 
| +                       DirectDownloadNoReferrerTargetBlank) { | 
| +  ClickTestLink("direct_download_noreferrer_target_blank", 1); | 
| +  GURL initial_url = embedded_test_server()->GetURL(kSingleFrameTestURL); | 
| +  GURL download_url = embedded_test_server()->GetURL(kDownloadItemURL); | 
| +  auto nav_map = navigation_map(); | 
| +  ASSERT_TRUE(nav_map); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->size()); | 
| +  ASSERT_EQ(std::size_t(2), nav_map->at(download_url).size()); | 
| +  // The first NavigationEvent was obtained from NOIFICATION_RETARGETING. | 
| +  // TODO(jialiul): After https://crbug.com/651895 is fixed, we'll no longer | 
| +  // listen to NOTIFICATION_RETARGETING, hence only one NavigationEvent will | 
| +  // be observed with the true initator URL. This applies to other new tab | 
| +  // download, and target blank download test cases too. | 
| +  VerifyNavigationEvent(initial_url,   // source_url | 
| +                        initial_url,   // source_main_frame_url | 
| +                        download_url,  // original_request_url | 
| +                        download_url,  // destination_url | 
| +                        false,         // is_user_initiated, | 
| +                        false,         // has_committed | 
| +                        false,         // has_server_redirect | 
| +                        nav_map->at(download_url).at(0)); | 
| +  // The second one is the actual navigation which triggers download. | 
| +  VerifyNavigationEvent(GURL(),        // source_url | 
| +                        GURL(),        // source_main_frame_url | 
| +                        download_url,  // original_request_url | 
| +                        download_url,  // destination_url | 
| +                        false,         // is_user_initiated, | 
| +                        false,         // has_committed | 
| +                        false,         // has_server_redirect | 
| +                        nav_map->at(download_url).at(1)); | 
| +  VerifyHostToIpMap(); | 
| +} | 
| + | 
| +// Click on a link which navigates to a page then redirects to a download using | 
| +// META HTTP-EQUIV="refresh". All transitions happen in the same tab. | 
| +IN_PROC_BROWSER_TEST_F(SBNavigationObserverBrowserTest, | 
| +                       SingleMetaRefreshRedirect) { | 
| +  ClickTestLink("single_meta_refresh_redirect", 2); | 
| +  GURL initial_url = embedded_test_server()->GetURL(kSingleFrameTestURL); | 
| +  GURL redirect_url = embedded_test_server()->GetURL(kRedirectURL); | 
| +  GURL download_url = embedded_test_server()->GetURL(kDownloadItemURL); | 
| +  auto nav_map = navigation_map(); | 
| +  ASSERT_TRUE(nav_map); | 
| +  // Since unlike server redirects client redirects commit and then generate a | 
| +  // second navigation, our observer records two NavigationEvents for this test. | 
| +  ASSERT_EQ(std::size_t(2), nav_map->size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(redirect_url).size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(download_url).size()); | 
| +  VerifyNavigationEvent(initial_url,   // source_url | 
| +                        initial_url,   // source_main_frame_url | 
| +                        redirect_url,  // original_request_url | 
| +                        redirect_url,  // destination_url | 
| +                        false,         // is_user_initiated, | 
| +                        true,          // has_committed | 
| +                        false,         // has_server_redirect | 
| +                        nav_map->at(redirect_url).at(0)); | 
| +  VerifyNavigationEvent(redirect_url,  // source_url | 
| +                        redirect_url,  // source_main_frame_url | 
| +                        download_url,  // original_request_url | 
| +                        download_url,  // destination_url | 
| +                        false,         // is_user_initiated, | 
| +                        false,         // has_committed | 
| +                        false,         // has_server_redirect | 
| +                        nav_map->at(download_url).at(0)); | 
| +  VerifyHostToIpMap(); | 
| +} | 
| + | 
| +// Click on a link which navigates to a page then redirects to a download using | 
| +// META HTTP-EQUIV="refresh". First navigation happens in target blank. | 
| +IN_PROC_BROWSER_TEST_F(SBNavigationObserverBrowserTest, | 
| +                       SingleMetaRefreshRedirectTargetBlank) { | 
| +  ClickTestLink("single_meta_refresh_redirect_target_blank", 2); | 
| +  GURL initial_url = embedded_test_server()->GetURL(kSingleFrameTestURL); | 
| +  GURL redirect_url = embedded_test_server()->GetURL(kRedirectURL); | 
| +  GURL download_url = embedded_test_server()->GetURL(kDownloadItemURL); | 
| +  auto nav_map = navigation_map(); | 
| +  ASSERT_TRUE(nav_map); | 
| +  ASSERT_EQ(std::size_t(2), nav_map->size()); | 
| +  ASSERT_EQ(std::size_t(2), nav_map->at(redirect_url).size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(download_url).size()); | 
| +  // TODO(jialiul): After https://crbug.com/651895 is fixed, we'll no longer | 
| +  // listen to NOTIFICATION_RETARGETING, hence only two NavigationEvents will | 
| +  // be observed with the true initator URL. | 
| +  VerifyNavigationEvent(initial_url,   // source_url | 
| +                        initial_url,   // source_main_frame_url | 
| +                        redirect_url,  // original_request_url | 
| +                        redirect_url,  // destination_url | 
| +                        false,         // is_user_initiated, | 
| +                        false,         // has_committed | 
| +                        false,         // has_server_redirect | 
| +                        nav_map->at(redirect_url).at(0)); | 
| +  VerifyNavigationEvent(GURL(),        // source_url | 
| +                        GURL(),        // source_main_frame_url | 
| +                        redirect_url,  // original_request_url | 
| +                        redirect_url,  // destination_url | 
| +                        false,         // is_user_initiated, | 
| +                        true,          // has_committed | 
| +                        false,         // has_server_redirect | 
| +                        nav_map->at(redirect_url).at(1)); | 
| +  VerifyNavigationEvent(redirect_url,  // source_url | 
| +                        redirect_url,  // source_main_frame_url | 
| +                        download_url,  // original_request_url | 
| +                        download_url,  // destination_url | 
| +                        false,         // is_user_initiated, | 
| +                        false,         // has_committed | 
| +                        false,         // has_server_redirect | 
| +                        nav_map->at(download_url).at(0)); | 
| +  VerifyHostToIpMap(); | 
| +} | 
| + | 
| +// Click on a link which redirects twice before reaching download using | 
| +// META HTTP-EQUIV="refresh". All transitions happen in the same tab. | 
| +IN_PROC_BROWSER_TEST_F(SBNavigationObserverBrowserTest, | 
| +                       MultiMetaRefreshRedirects) { | 
| +  ClickTestLink("multiple_meta_refresh_redirects", 3); | 
| +  GURL initial_url = embedded_test_server()->GetURL(kSingleFrameTestURL); | 
| +  GURL first_redirect_url = embedded_test_server()->GetURL( | 
| +      "/safe_browsing/download_protection/navigation_observer/" | 
| +      "double_redirect.html"); | 
| +  GURL second_redirect_url = embedded_test_server()->GetURL(kRedirectURL); | 
| +  GURL download_url = embedded_test_server()->GetURL(kDownloadItemURL); | 
| +  auto nav_map = navigation_map(); | 
| +  ASSERT_TRUE(nav_map); | 
| +  ASSERT_EQ(std::size_t(3), nav_map->size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(first_redirect_url).size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(second_redirect_url).size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(download_url).size()); | 
| +  VerifyNavigationEvent(initial_url,         // source_url | 
| +                        initial_url,         // source_main_frame_url | 
| +                        first_redirect_url,  // original_request_url | 
| +                        first_redirect_url,  // destination_url | 
| +                        false,               // is_user_initiated, | 
| +                        true,                // has_committed | 
| +                        false,               // has_server_redirect | 
| +                        nav_map->at(first_redirect_url).at(0)); | 
| +  VerifyNavigationEvent(first_redirect_url,   // source_url | 
| +                        first_redirect_url,   // source_main_frame_url | 
| +                        second_redirect_url,  // original_request_url | 
| +                        second_redirect_url,  // destination_url | 
| +                        false,                // is_user_initiated, | 
| +                        true,                 // has_committed | 
| +                        false,                // has_server_redirect | 
| +                        nav_map->at(second_redirect_url).at(0)); | 
| +  VerifyNavigationEvent(second_redirect_url,  // source_url | 
| +                        second_redirect_url,  // source_main_frame_url | 
| +                        download_url,         // original_request_url | 
| +                        download_url,         // destination_url | 
| +                        false,                // is_user_initiated, | 
| +                        false,                // has_committed | 
| +                        false,                // has_server_redirect | 
| +                        nav_map->at(download_url).at(0)); | 
| +  VerifyHostToIpMap(); | 
| +} | 
| + | 
| +// Click on a link which redirects to download using window.location. | 
| +// All transitions happen in the same tab. | 
| +IN_PROC_BROWSER_TEST_F(SBNavigationObserverBrowserTest, | 
| +                       WindowLocationRedirect) { | 
| +  ClickTestLink("window_location_redirection", 1); | 
| +  GURL initial_url = embedded_test_server()->GetURL(kSingleFrameTestURL); | 
| +  GURL download_url = embedded_test_server()->GetURL(kDownloadItemURL); | 
| +  auto nav_map = navigation_map(); | 
| +  ASSERT_TRUE(nav_map); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(download_url).size()); | 
| +  VerifyNavigationEvent(initial_url,   // source_url | 
| +                        initial_url,   // source_main_frame_url | 
| +                        download_url,  // original_request_url | 
| +                        download_url,  // destination_url | 
| +                        false,         // is_user_initiated, | 
| +                        false,         // has_committed | 
| +                        false,         // has_server_redirect | 
| +                        nav_map->at(download_url).at(0)); | 
| +} | 
| + | 
| +// Click on a link which redirects twice until it reaches download using a | 
| +// mixture of meta refresh and window.location. All transitions happen in the | 
| +// same tab. | 
| +IN_PROC_BROWSER_TEST_F(SBNavigationObserverBrowserTest, MixRedirects) { | 
| +  ClickTestLink("mix_redirects", 2); | 
| +  GURL redirect_url = embedded_test_server()->GetURL(kRedirectURL); | 
| +  GURL initial_url = embedded_test_server()->GetURL(kSingleFrameTestURL); | 
| +  GURL download_url = embedded_test_server()->GetURL(kDownloadItemURL); | 
| +  auto nav_map = navigation_map(); | 
| +  ASSERT_TRUE(nav_map); | 
| +  ASSERT_EQ(std::size_t(2), nav_map->size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(redirect_url).size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(download_url).size()); | 
| +  VerifyNavigationEvent(initial_url,   // source_url | 
| +                        initial_url,   // source_main_frame_url | 
| +                        redirect_url,  // original_request_url | 
| +                        redirect_url,  // destination_url | 
| +                        false,         // is_user_initiated, | 
| +                        true,          // has_committed | 
| +                        false,         // has_server_redirect | 
| +                        nav_map->at(redirect_url).at(0)); | 
| +  VerifyNavigationEvent(redirect_url,  // source_url | 
| +                        redirect_url,  // source_main_frame_url | 
| +                        download_url,  // original_request_url | 
| +                        download_url,  // destination_url | 
| +                        false,         // is_user_initiated, | 
| +                        false,         // has_committed | 
| +                        false,         // has_server_redirect | 
| +                        nav_map->at(download_url).at(0)); | 
| +  VerifyHostToIpMap(); | 
| +} | 
| + | 
| +// Use javascript to open download in a new tab. | 
| +IN_PROC_BROWSER_TEST_F(SBNavigationObserverBrowserTest, NewTabDownload) { | 
| +  ClickTestLink("new_tab_download", 2); | 
| +  GURL initial_url = embedded_test_server()->GetURL(kSingleFrameTestURL); | 
| +  GURL download_url = embedded_test_server()->GetURL(kDownloadItemURL); | 
| +  GURL blank_url = GURL(url::kAboutBlankURL); | 
| +  auto nav_map = navigation_map(); | 
| +  ASSERT_TRUE(nav_map); | 
| +  ASSERT_EQ(std::size_t(2), nav_map->size()); | 
| +  ASSERT_EQ(std::size_t(2), nav_map->at(blank_url).size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(download_url).size()); | 
| +  VerifyNavigationEvent(initial_url,  // source_url | 
| +                        initial_url,  // source_main_frame_url | 
| +                        blank_url,    // original_request_url | 
| +                        blank_url,    // destination_url | 
| +                        false,        // is_user_initiated, | 
| +                        false,        // has_committed | 
| +                        false,        // has_server_redirect | 
| +                        nav_map->at(blank_url).at(0)); | 
| +  // Source and target are at different tabs. | 
| +  EXPECT_NE(nav_map->at(blank_url).at(0).source_tab_id, | 
| +            nav_map->at(blank_url).at(0).target_tab_id); | 
| +  VerifyNavigationEvent(GURL(),     // source_url | 
| +                        GURL(),     // source_main_frame_url | 
| +                        blank_url,  // original_request_url | 
| +                        blank_url,  // destination_url | 
| +                        false,      // is_user_initiated, | 
| +                        false,      // has_committed | 
| +                        false,      // has_server_redirect | 
| +                        nav_map->at(blank_url).at(1)); | 
| +  EXPECT_EQ(nav_map->at(blank_url).at(1).source_tab_id, | 
| +            nav_map->at(blank_url).at(1).target_tab_id); | 
| +  VerifyNavigationEvent(blank_url,     // source_url | 
| +                        blank_url,     // source_main_frame_url | 
| +                        download_url,  // original_request_url | 
| +                        download_url,  // destination_url | 
| +                        false,         // is_user_initiated, | 
| +                        false,         // has_committed | 
| +                        false,         // has_server_redirect | 
| +                        nav_map->at(download_url).at(0)); | 
| +  EXPECT_EQ(nav_map->at(download_url).at(0).source_tab_id, | 
| +            nav_map->at(download_url).at(0).target_tab_id); | 
| +  VerifyHostToIpMap(); | 
| +} | 
| + | 
| +// Use javascript to open download in a new tab and download has a data url. | 
| +IN_PROC_BROWSER_TEST_F(SBNavigationObserverBrowserTest, | 
| +                       NewTabDownloadWithDataURL) { | 
| +  ClickTestLink("new_tab_download_with_data_url", 2); | 
| +  GURL initial_url = embedded_test_server()->GetURL(kSingleFrameTestURL); | 
| +  GURL download_url = GURL(kDownloadDataURL); | 
| +  GURL blank_url = GURL("about:blank"); | 
| +  auto nav_map = navigation_map(); | 
| +  ASSERT_TRUE(nav_map); | 
| +  ASSERT_EQ(std::size_t(2), nav_map->size()); | 
| +  ASSERT_EQ(std::size_t(2), nav_map->at(blank_url).size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(download_url).size()); | 
| +  // The first one comes from NOTIFICATION_RETARGETING. | 
| +  VerifyNavigationEvent(initial_url,  // source_url | 
| +                        initial_url,  // source_main_frame_url | 
| +                        blank_url,    // original_request_url | 
| +                        blank_url,    // destination_url | 
| +                        false,        // is_user_initiated, | 
| +                        false,        // has_committed | 
| +                        false,        // has_server_redirect | 
| +                        nav_map->at(blank_url).at(0)); | 
| +  // Source and target are at different tabs. | 
| +  EXPECT_FALSE(nav_map->at(blank_url).at(0).source_tab_id == | 
| +               nav_map->at(blank_url).at(0).target_tab_id); | 
| +  VerifyNavigationEvent(GURL(),     // source_url | 
| +                        GURL(),     // source_main_frame_url | 
| +                        blank_url,  // original_request_url | 
| +                        blank_url,  // destination_url | 
| +                        false,      // is_user_initiated, | 
| +                        false,      // has_committed | 
| +                        false,      // has_server_redirect | 
| +                        nav_map->at(blank_url).at(1)); | 
| +  EXPECT_EQ(nav_map->at(blank_url).at(1).source_tab_id, | 
| +            nav_map->at(blank_url).at(1).target_tab_id); | 
| +  VerifyNavigationEvent(blank_url,     // source_url | 
| +                        blank_url,     // source_main_frame_url | 
| +                        download_url,  // original_request_url | 
| +                        download_url,  // destination_url | 
| +                        false,         // is_user_initiated, | 
| +                        false,         // has_committed | 
| +                        false,         // has_server_redirect | 
| +                        nav_map->at(download_url).at(0)); | 
| +  EXPECT_TRUE(nav_map->at(download_url).at(0).source_tab_id == | 
| +              nav_map->at(download_url).at(0).target_tab_id); | 
| +  // Since data url does does not have IP, host_to_ip_map_ should be empty. | 
| +  EXPECT_EQ(std::size_t(0), host_to_ip_map()->size()); | 
| +} | 
| + | 
| +// TODO(jialiul): Need to figure out why this test is failing on Windows and | 
| +// flaky on other platforms. | 
| +#define MAYBE_DownloadViaHTML5FileApi DISABLED_DownloadViaHTML5FileApi | 
| +// Download via html5 file API. | 
| +IN_PROC_BROWSER_TEST_F(SBNavigationObserverBrowserTest, | 
| +                       MAYBE_DownloadViaHTML5FileApi) { | 
| +  ClickTestLink("html5_file_api", 1); | 
| +  GURL initial_url = embedded_test_server()->GetURL(kSingleFrameTestURL); | 
| +  std::string download_url_str = | 
| +      base::StringPrintf("filesystem:%stemporary/test.exe", | 
| +                         embedded_test_server()->base_url().spec().c_str()); | 
| +  GURL download_url = GURL(download_url_str); | 
| +  auto nav_map = navigation_map(); | 
| +  ASSERT_TRUE(nav_map); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(download_url).size()); | 
| +  VerifyNavigationEvent(initial_url,   // source_url | 
| +                        initial_url,   // source_main_frame_url | 
| +                        download_url,  // original_request_url | 
| +                        download_url,  // destination_url | 
| +                        false,         // is_user_initiated, | 
| +                        false,         // has_committed | 
| +                        false,         // has_server_redirect | 
| +                        nav_map->at(download_url).at(0)); | 
| +  VerifyHostToIpMap(); | 
| +} | 
| + | 
| +// Click a link in a subframe and start download. | 
| +IN_PROC_BROWSER_TEST_F(SBNavigationObserverBrowserTest, | 
| +                       SubFrameDirectDownload) { | 
| +  ui_test_utils::NavigateToURL( | 
| +      browser(), embedded_test_server()->GetURL(kMultiFrameTestURL)); | 
| +  std::string test_name = | 
| +      base::StringPrintf("%s', '%s", "iframe1", "iframe_direct_download"); | 
| +  ClickTestLink(test_name.c_str(), 1); | 
| +  GURL initial_url = embedded_test_server()->GetURL(kSingleFrameTestURL); | 
| +  GURL multi_frame_test_url = | 
| +      embedded_test_server()->GetURL(kMultiFrameTestURL); | 
| +  GURL iframe_url = embedded_test_server()->GetURL(kIframeDirectDownloadURL); | 
| +  GURL iframe_retargeting_url = | 
| +      embedded_test_server()->GetURL(kIframeRetargetingURL); | 
| +  GURL download_url = embedded_test_server()->GetURL(kDownloadItemURL); | 
| +  auto nav_map = navigation_map(); | 
| +  ASSERT_TRUE(nav_map); | 
| +  ASSERT_EQ(std::size_t(4), nav_map->size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(multi_frame_test_url).size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(iframe_url).size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(iframe_retargeting_url).size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(download_url).size()); | 
| +  VerifyNavigationEvent(initial_url,           // source_url | 
| +                        initial_url,           // source_main_frame_url | 
| +                        multi_frame_test_url,  // original_request_url | 
| +                        multi_frame_test_url,  // destination_url | 
| +                        true,                  // is_user_initiated, | 
| +                        true,                  // has_committed | 
| +                        false,                 // has_server_redirect | 
| +                        nav_map->at(multi_frame_test_url).at(0)); | 
| +  VerifyNavigationEvent(GURL(),                // source_url | 
| +                        multi_frame_test_url,  // source_main_frame_url | 
| +                        iframe_url,            // original_request_url | 
| +                        iframe_url,            // destination_url | 
| +                        false,                 // is_user_initiated, | 
| +                        true,                  // has_committed | 
| +                        false,                 // has_server_redirect | 
| +                        nav_map->at(iframe_url).at(0)); | 
| +  VerifyNavigationEvent(GURL(),                  // source_url | 
| +                        multi_frame_test_url,    // source_main_frame_url | 
| +                        iframe_retargeting_url,  // original_request_url | 
| +                        iframe_retargeting_url,  // destination_url | 
| +                        false,                   // is_user_initiated, | 
| +                        true,                    // has_committed | 
| +                        false,                   // has_server_redirect | 
| +                        nav_map->at(iframe_retargeting_url).at(0)); | 
| +  VerifyNavigationEvent(iframe_url,            // source_url | 
| +                        multi_frame_test_url,  // source_main_frame_url | 
| +                        download_url,          // original_request_url | 
| +                        download_url,          // destination_url | 
| +                        false,                 // is_user_initiated, | 
| +                        false,                 // has_committed | 
| +                        false,                 // has_server_redirect | 
| +                        nav_map->at(download_url).at(0)); | 
| +  VerifyHostToIpMap(); | 
| +} | 
| + | 
| +// Click a link in a subframe and open download in a new tab. | 
| +IN_PROC_BROWSER_TEST_F(SBNavigationObserverBrowserTest, | 
| +                       SubFrameNewTabDownload) { | 
| +  ui_test_utils::NavigateToURL( | 
| +      browser(), embedded_test_server()->GetURL(kMultiFrameTestURL)); | 
| +  std::string test_name = | 
| +      base::StringPrintf("%s', '%s", "iframe2", "iframe_new_tab_download"); | 
| +  ClickTestLink(test_name.c_str(), 2); | 
| +  GURL initial_url = embedded_test_server()->GetURL(kSingleFrameTestURL); | 
| +  GURL multi_frame_test_url = | 
| +      embedded_test_server()->GetURL(kMultiFrameTestURL); | 
| +  GURL iframe_url = embedded_test_server()->GetURL(kIframeDirectDownloadURL); | 
| +  GURL iframe_retargeting_url = | 
| +      embedded_test_server()->GetURL(kIframeRetargetingURL); | 
| +  GURL blank_url = GURL(url::kAboutBlankURL); | 
| +  GURL download_url = embedded_test_server()->GetURL(kDownloadItemURL); | 
| +  auto nav_map = navigation_map(); | 
| +  ASSERT_TRUE(nav_map); | 
| +  ASSERT_EQ(std::size_t(5), nav_map->size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(multi_frame_test_url).size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(iframe_url).size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(iframe_retargeting_url).size()); | 
| +  ASSERT_EQ(std::size_t(2), nav_map->at(blank_url).size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(download_url).size()); | 
| +  VerifyNavigationEvent(initial_url,           // source_url | 
| +                        initial_url,           // source_main_frame_url | 
| +                        multi_frame_test_url,  // original_request_url | 
| +                        multi_frame_test_url,  // destination_url | 
| +                        true,                  // is_user_initiated, | 
| +                        true,                  // has_committed | 
| +                        false,                 // has_server_redirect | 
| +                        nav_map->at(multi_frame_test_url).at(0)); | 
| +  VerifyNavigationEvent(GURL(),                // source_url | 
| +                        multi_frame_test_url,  // source_main_frame_url | 
| +                        iframe_url,            // original_request_url | 
| +                        iframe_url,            // destination_url | 
| +                        false,                 // is_user_initiated, | 
| +                        true,                  // has_committed | 
| +                        false,                 // has_server_redirect | 
| +                        nav_map->at(iframe_url).at(0)); | 
| +  VerifyNavigationEvent(GURL(),                  // source_url | 
| +                        multi_frame_test_url,    // source_main_frame_url | 
| +                        iframe_retargeting_url,  // original_request_url | 
| +                        iframe_retargeting_url,  // destination_url | 
| +                        false,                   // is_user_initiated, | 
| +                        true,                    // has_committed | 
| +                        false,                   // has_server_redirect | 
| +                        nav_map->at(iframe_retargeting_url).at(0)); | 
| +  VerifyNavigationEvent(iframe_retargeting_url,  // source_url | 
| +                        multi_frame_test_url,    // source_main_frame_url | 
| +                        blank_url,               // original_request_url | 
| +                        blank_url,               // destination_url | 
| +                        false,                   // is_user_initiated, | 
| +                        false,                   // has_committed | 
| +                        false,                   // has_server_redirect | 
| +                        nav_map->at(blank_url).at(0)); | 
| +  VerifyNavigationEvent(GURL(),     // source_url | 
| +                        GURL(),     // source_main_frame_url | 
| +                        blank_url,  // original_request_url | 
| +                        blank_url,  // destination_url | 
| +                        false,      // is_user_initiated, | 
| +                        false,      // has_committed | 
| +                        false,      // has_server_redirect | 
| +                        nav_map->at(blank_url).at(1)); | 
| +  VerifyNavigationEvent(blank_url,     // source_url | 
| +                        blank_url,     // source_main_frame_url | 
| +                        download_url,  // original_request_url | 
| +                        download_url,  // destination_url | 
| +                        false,         // is_user_initiated, | 
| +                        false,         // has_committed | 
| +                        false,         // has_server_redirect | 
| +                        nav_map->at(download_url).at(0)); | 
| +  VerifyHostToIpMap(); | 
| +} | 
| + | 
| +// Server-side redirect. | 
| +IN_PROC_BROWSER_TEST_F(SBNavigationObserverBrowserTest, ServerRedirect) { | 
| +  GURL initial_url = embedded_test_server()->GetURL(kSingleFrameTestURL); | 
| +  GURL download_url = embedded_test_server()->GetURL(kDownloadItemURL); | 
| +  GURL request_url = | 
| +      embedded_test_server()->GetURL("/server-redirect?" + download_url.spec()); | 
| +  ui_test_utils::NavigateToURL(browser(), request_url); | 
| +  CancelDownloads(); | 
| +  auto nav_map = navigation_map(); | 
| +  ASSERT_TRUE(nav_map); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->size()); | 
| +  ASSERT_EQ(std::size_t(1), nav_map->at(download_url).size()); | 
| +  VerifyNavigationEvent(initial_url,   // source_url | 
| +                        initial_url,   // source_main_frame_url | 
| +                        request_url,   // original_request_url | 
| +                        download_url,  // destination_url | 
| +                        true,          // is_user_initiated, | 
| +                        false,         // has_committed | 
| +                        true,          // has_server_redirect | 
| +                        nav_map->at(download_url).at(0)); | 
| +} | 
| + | 
| +// host_to_ip_map_ size should increase by one after a new navigation. | 
| +IN_PROC_BROWSER_TEST_F(SBNavigationObserverBrowserTest, AddIPMapping) { | 
| +  auto ip_map = host_to_ip_map(); | 
| +  std::string test_server_host(embedded_test_server()->base_url().host()); | 
| +  ip_map->insert( | 
| +      std::make_pair(test_server_host, std::vector<ResolvedIPAddress>())); | 
| +  ASSERT_EQ(std::size_t(0), ip_map->at(test_server_host).size()); | 
| +  ClickTestLink("direct_download", 1); | 
| +  EXPECT_EQ(std::size_t(1), ip_map->at(test_server_host).size()); | 
| +  EXPECT_EQ(embedded_test_server()->host_port_pair().host(), | 
| +            ip_map->at(test_server_host).back().ip); | 
| +} | 
| + | 
| +// If we have already seen an IP associated with a host, update its timestamp. | 
| +IN_PROC_BROWSER_TEST_F(SBNavigationObserverBrowserTest, IPListDedup) { | 
| +  auto ip_map = host_to_ip_map(); | 
| +  std::string test_server_host(embedded_test_server()->base_url().host()); | 
| +  ip_map->insert( | 
| +      std::make_pair(test_server_host, std::vector<ResolvedIPAddress>())); | 
| +  base::Time yesterday(base::Time::Now() - base::TimeDelta::FromDays(1)); | 
| +  ip_map->at(test_server_host) | 
| +      .push_back(ResolvedIPAddress( | 
| +          yesterday, embedded_test_server()->host_port_pair().host())); | 
| +  ASSERT_EQ(std::size_t(1), ip_map->at(test_server_host).size()); | 
| +  ClickTestLink("direct_download", 1); | 
| +  EXPECT_EQ(std::size_t(1), ip_map->at(test_server_host).size()); | 
| +  EXPECT_EQ(embedded_test_server()->host_port_pair().host(), | 
| +            ip_map->at(test_server_host).back().ip); | 
| +  EXPECT_NE(yesterday, ip_map->at(test_server_host).front().timestamp); | 
| +} | 
| + | 
| +}  // namespace safe_browsing | 
|  |