1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147
|
// Copyright 2021 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "base/containers/adapters.h"
#include "base/containers/circular_deque.h"
#include "build/build_config.h"
#include "chrome/browser/breadcrumbs/breadcrumb_manager_tab_helper.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/ssl/cert_verifier_browser_test.h"
#include "chrome/browser/ui/browser.h"
#include "chrome/common/chrome_switches.h"
#include "chrome/test/base/in_process_browser_test.h"
#include "chrome/test/base/ui_test_utils.h"
#include "components/breadcrumbs/core/breadcrumb_manager.h"
#include "components/breadcrumbs/core/breadcrumb_manager_tab_helper.h"
#include "components/breadcrumbs/core/breadcrumbs_status.h"
#include "content/public/test/browser_test.h"
#include "net/test/embedded_test_server/embedded_test_server.h"
namespace {
// A WebContentsObserver to allow waiting on a change in visible security state.
class SecurityStyleTestObserver : public content::WebContentsObserver {
public:
explicit SecurityStyleTestObserver(content::WebContents* web_contents)
: content::WebContentsObserver(web_contents) {}
SecurityStyleTestObserver(const SecurityStyleTestObserver&) = delete;
SecurityStyleTestObserver& operator=(const SecurityStyleTestObserver&) =
delete;
~SecurityStyleTestObserver() override = default;
void DidChangeVisibleSecurityState() override { run_loop_.Quit(); }
void WaitForDidChangeVisibleSecurityState() { run_loop_.Run(); }
private:
base::RunLoop run_loop_;
};
const base::circular_deque<std::string>& GetEvents() {
return breadcrumbs::BreadcrumbManager::GetInstance().GetEvents();
}
// Returns the latest breadcrumb event containing substring `str.` If no logged
// breadcrumb event contains `str`, returns `std::nullopt`.
const std::optional<std::string> FindEventContaining(const std::string& str) {
const auto& events_new_to_old = base::Reversed(GetEvents());
for (const auto& event : events_new_to_old) {
if (event.find(str) != std::string::npos) {
return event;
}
}
return std::nullopt;
}
} // namespace
// Test fixture for BreadcrumbManagerTabHelper class.
class BreadcrumbManagerTabHelperBrowserTest : public InProcessBrowserTest {
protected:
void SetUpOnMainThread() override {
BreadcrumbManagerTabHelper::CreateForWebContents(
browser()->tab_strip_model()->GetActiveWebContents());
}
private:
breadcrumbs::ScopedEnableBreadcrumbsForTesting enable_breadcrumbs_;
};
// Tests download navigation.
IN_PROC_BROWSER_TEST_F(BreadcrumbManagerTabHelperBrowserTest, Download) {
const GURL url =
ui_test_utils::GetTestUrl(base::FilePath().AppendASCII("downloads"),
base::FilePath().AppendASCII("a_zip_file.zip"));
ui_test_utils::DownloadURL(browser(), url);
// Breadcrumbs should have been logged for starting and finishing the
// navigation, and the navigation should be labeled as a download.
EXPECT_NE(std::nullopt,
FindEventContaining(breadcrumbs::kBreadcrumbDidStartNavigation));
const auto finish_nav_event =
FindEventContaining(breadcrumbs::kBreadcrumbDidFinishNavigation);
EXPECT_NE(std::nullopt, finish_nav_event);
EXPECT_NE(std::string::npos,
finish_nav_event->find(breadcrumbs::kBreadcrumbDownload));
}
// Tests changes in security states.
class BreadcrumbManagerTabHelperSecurityStateBrowserTest
: public CertVerifierBrowserTest {
public:
BreadcrumbManagerTabHelperSecurityStateBrowserTest() {
https_server_.ServeFilesFromSourceDirectory(GetChromeTestDataDir());
}
void SetUpCommandLine(base::CommandLine* command_line) override {
command_line->AppendSwitch(switches::kAllowRunningInsecureContent);
CertVerifierBrowserTest::SetUpCommandLine(command_line);
}
void SetUpOnMainThread() override {
BreadcrumbManagerTabHelper::CreateForWebContents(
browser()->tab_strip_model()->GetActiveWebContents());
ASSERT_TRUE(https_server_.Start());
}
protected:
void SetUpMockCertVerifierForHttpsServer(net::CertStatus cert_status,
int net_result) {
scoped_refptr<net::X509Certificate> cert(https_server_.GetCertificate());
net::CertVerifyResult verify_result;
verify_result.is_issued_by_known_root = false;
verify_result.verified_cert = cert;
verify_result.cert_status = cert_status;
mock_cert_verifier()->AddResultForCert(cert, verify_result, net_result);
}
net::EmbeddedTestServer https_server_{net::EmbeddedTestServer::TYPE_HTTPS};
private:
breadcrumbs::ScopedEnableBreadcrumbsForTesting enable_breadcrumbs_;
};
// Broken authentication.
IN_PROC_BROWSER_TEST_F(BreadcrumbManagerTabHelperSecurityStateBrowserTest,
BrokenAuthentication) {
SetUpMockCertVerifierForHttpsServer(net::CERT_STATUS_DATE_INVALID,
net::ERR_CERT_DATE_INVALID);
ASSERT_TRUE(ui_test_utils::NavigateToURL(
browser(), https_server_.GetURL("/ssl/google.html")));
// The breadcrumb event for broken authentication should have been logged.
const auto security_change_event = FindEventContaining(
breadcrumbs::kBreadcrumbDidChangeVisibleSecurityState);
EXPECT_NE(std::nullopt, security_change_event);
EXPECT_NE(std::string::npos,
security_change_event->find(
breadcrumbs::kBreadcrumbAuthenticationBroken));
EXPECT_EQ(std::string::npos,
security_change_event->find(breadcrumbs::kBreadcrumbMixedContent));
EXPECT_NE(std::nullopt,
FindEventContaining(breadcrumbs::kBreadcrumbPageLoaded));
}
|