cef/tests/ceftests/certificate_error_unittest.cc
2022-08-16 16:23:27 -04:00

522 lines
16 KiB
C++

// Copyright (c) 2022 The Chromium Embedded Framework 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 "include/base/cef_callback.h"
#include "include/test/cef_test_server.h"
#include "include/wrapper/cef_closure_task.h"
#include "include/wrapper/cef_stream_resource_handler.h"
#include "tests/ceftests/test_handler.h"
#include "tests/ceftests/test_server_observer.h"
#include "tests/ceftests/test_util.h"
#include "tests/gtest/include/gtest/gtest.h"
#include "tests/shared/common/string_util.h"
namespace {
// Used to observe HTTP and HTTPS server requests.
class OtherServerObserver : public test_server::ObserverHelper {
public:
using ReadyCallback = base::OnceCallback<void(const std::string& url)>;
using RequestCallback =
base::RepeatingCallback<bool(CefRefPtr<CefRequest> request,
const ResponseCallback& response_callback)>;
OtherServerObserver(bool https_server,
ReadyCallback ready_callback,
base::OnceClosure done_callback,
const RequestCallback& request_callback)
: ready_callback_(std::move(ready_callback)),
done_callback_(std::move(done_callback)),
request_callback_(request_callback) {
EXPECT_TRUE(ready_callback_);
EXPECT_TRUE(done_callback_);
EXPECT_TRUE(request_callback_);
Initialize(https_server);
}
void OnInitialized(const std::string& server_origin) override {
EXPECT_UI_THREAD();
std::move(ready_callback_).Run(server_origin);
}
void OnShutdown() override {
EXPECT_UI_THREAD();
std::move(done_callback_).Run();
delete this;
}
bool OnTestServerRequest(CefRefPtr<CefRequest> request,
const ResponseCallback& response_callback) override {
EXPECT_UI_THREAD();
return request_callback_.Run(request, response_callback);
}
private:
ReadyCallback ready_callback_;
base::OnceClosure done_callback_;
RequestCallback request_callback_;
DISALLOW_COPY_AND_ASSIGN(OtherServerObserver);
};
class CertificateErrorTest : public TestHandler, public CefTestServerHandler {
public:
using ResponseCallback = test_server::ObserverHelper::ResponseCallback;
enum class OtherServerType {
NONE,
HTTP,
HTTPS,
};
CertificateErrorTest(
cef_test_cert_type_t cert_type,
bool expect_load_success,
bool expect_certificate_error,
OtherServerType other_server_type = OtherServerType::NONE)
: cert_type_(cert_type),
expect_load_success_(expect_load_success),
expect_certificate_error_(expect_certificate_error),
other_server_type_(other_server_type) {}
void RunTest() override {
SetTestTimeout();
CefPostTask(TID_UI,
base::BindOnce(&CertificateErrorTest::StartHttpsServer, this));
}
bool OnTestServerRequest(
CefRefPtr<CefTestServer> server,
CefRefPtr<CefRequest> request,
CefRefPtr<CefTestServerConnection> connection) override {
CefPostTask(TID_UI,
base::BindOnce(&CertificateErrorTest::HandleHttpsRequest, this,
request, connection));
return true;
}
bool OnOtherServerRequest(CefRefPtr<CefRequest> request,
const ResponseCallback& response_callback) {
EXPECT_UI_THREAD();
HandleOtherServerRequest(request, response_callback);
return true;
}
bool OnCertificateError(CefRefPtr<CefBrowser> browser,
cef_errorcode_t cert_error,
const CefString& request_url,
CefRefPtr<CefSSLInfo> ssl_info,
CefRefPtr<CefCallback> callback) override {
EXPECT_UI_THREAD();
got_certificate_error_.yes();
EXPECT_STREQ(GetEndURL().c_str(), request_url.ToString().c_str());
if (expect_load_success_) {
// Continue the load.
callback->Continue();
return true;
}
// Cancel the load.
return false;
}
void OnLoadEnd(CefRefPtr<CefBrowser> browser,
CefRefPtr<CefFrame> frame,
int httpStatusCode) override {
EXPECT_UI_THREAD();
TestHandler::OnLoadEnd(browser, frame, httpStatusCode);
const std::string& url = frame->GetURL();
if (url == GetEndURL()) {
got_load_end_.yes();
MaybeEndTest();
}
}
void OnLoadError(CefRefPtr<CefBrowser> browser,
CefRefPtr<CefFrame> frame,
ErrorCode errorCode,
const CefString& errorText,
const CefString& failedUrl) override {
EXPECT_UI_THREAD();
const std::string& url = frame->GetURL();
if (url == GetEndURL()) {
got_load_error_.yes();
MaybeEndTest();
}
}
void DestroyTest() override {
EXPECT_FALSE(server_);
EXPECT_FALSE(other_server_);
EXPECT_TRUE(got_load_end_);
if (expect_load_success_) {
EXPECT_TRUE(got_request_);
EXPECT_FALSE(got_load_error_);
} else {
EXPECT_FALSE(got_request_);
EXPECT_TRUE(got_load_error_);
}
EXPECT_EQ(expect_certificate_error_, got_certificate_error_);
TestHandler::DestroyTest();
}
protected:
virtual std::string GetStartURL() const {
return server_origin() + "/index.html";
}
virtual std::string GetEndURL() const { return GetStartURL(); }
const std::string& server_origin() const { return server_origin_; }
const std::string& other_origin() const { return other_origin_; }
virtual void HandleOtherServerRequest(
CefRefPtr<CefRequest> request,
const ResponseCallback& response_callback) {}
private:
void HandleHttpsRequest(CefRefPtr<CefRequest> request,
CefRefPtr<CefTestServerConnection> connection) {
EXPECT_UI_THREAD();
got_request_.yes();
const std::string response = "<html><body>Certificate Test</body></html>";
connection->SendHttp200Response("text/html", response.c_str(),
response.size());
MaybeEndTest();
}
void MaybeEndTest() {
EXPECT_UI_THREAD();
bool end_test;
if (expect_load_success_) {
end_test = got_load_end_ && got_request_;
} else {
end_test = got_load_end_ && got_load_error_;
}
if (end_test) {
StopHttpsServer();
}
}
void StartHttpsServer() {
EXPECT_UI_THREAD();
server_ = CefTestServer::CreateAndStart(/*port=*/0, /*https_server=*/true,
cert_type_, this);
server_origin_ = server_->GetOrigin();
if (other_server_type_ != OtherServerType::NONE) {
StartOtherServer();
} else {
DoCreateBrowser();
}
}
void StartOtherServer() {
EXPECT_UI_THREAD();
EXPECT_NE(other_server_type_, OtherServerType::NONE);
// Will delete itself after the server stops.
other_server_ = new OtherServerObserver(
other_server_type_ == OtherServerType::HTTPS,
base::BindOnce(&CertificateErrorTest::StartedOtherServer, this),
base::BindOnce(&CertificateErrorTest::StoppedOtherServer, this),
base::BindRepeating(&CertificateErrorTest::OnOtherServerRequest, this));
}
void StartedOtherServer(const std::string& other_origin) {
EXPECT_UI_THREAD();
EXPECT_NE(other_server_type_, OtherServerType::NONE);
other_origin_ = other_origin;
DoCreateBrowser();
}
void DoCreateBrowser() {
EXPECT_UI_THREAD();
// Create a new in-memory context so certificate decisions aren't cached.
auto request_context = CreateTestRequestContext(
TEST_RC_MODE_CUSTOM, /*cache_path=*/std::string());
CreateBrowser(GetStartURL(), request_context);
}
void StopHttpsServer() {
EXPECT_UI_THREAD();
server_->Stop();
server_ = nullptr;
if (other_server_type_ != OtherServerType::NONE) {
// Stop the other server. Results in a call to StoppedOtherServer().
other_server_->Shutdown();
} else {
DestroyTest();
}
}
void StoppedOtherServer() {
EXPECT_UI_THREAD();
EXPECT_NE(other_server_type_, OtherServerType::NONE);
other_server_ = nullptr;
DestroyTest();
}
const cef_test_cert_type_t cert_type_;
const bool expect_load_success_;
const bool expect_certificate_error_;
const OtherServerType other_server_type_;
CefRefPtr<CefTestServer> server_;
std::string server_origin_;
OtherServerObserver* other_server_ = nullptr;
std::string other_origin_;
TrackCallback got_request_;
TrackCallback got_certificate_error_;
TrackCallback got_load_end_;
TrackCallback got_load_error_;
IMPLEMENT_REFCOUNTING(CertificateErrorTest);
};
} // namespace
TEST(CertificateErrorTest, DirectNoError) {
CefRefPtr<CertificateErrorTest> handler = new CertificateErrorTest(
/*cert_type=*/CEF_TEST_CERT_OK_DOMAIN, /*expect_load_success=*/true,
/*expect_certificate_error=*/false);
handler->ExecuteTest();
ReleaseAndWaitForDestructor(handler);
}
TEST(CertificateErrorTest, DirectExpired) {
CefRefPtr<CertificateErrorTest> handler = new CertificateErrorTest(
/*cert_type=*/CEF_TEST_CERT_EXPIRED, /*expect_load_success=*/false,
/*expect_certificate_error=*/true);
handler->ExecuteTest();
ReleaseAndWaitForDestructor(handler);
}
namespace {
class DirectMismatchedTest : public CertificateErrorTest {
public:
DirectMismatchedTest(bool continue_invalid_certificate)
: CertificateErrorTest(
/*cert_type=*/CEF_TEST_CERT_OK_DOMAIN,
/*expect_load_success=*/continue_invalid_certificate,
/*expect_certificate_error=*/true) {}
protected:
std::string GetStartURL() const {
// Load by IP address when the certificate expects a domain.
return client::AsciiStrReplace(server_origin(), "localhost", "127.0.0.1") +
"/index.html";
}
};
} // namespace
TEST(CertificateErrorTest, DirectMismatchedCancel) {
CefRefPtr<DirectMismatchedTest> handler =
new DirectMismatchedTest(/*continue_invalid_certificate=*/false);
handler->ExecuteTest();
ReleaseAndWaitForDestructor(handler);
}
TEST(CertificateErrorTest, DirectMismatchedContinue) {
CefRefPtr<DirectMismatchedTest> handler =
new DirectMismatchedTest(/*continue_invalid_certificate=*/true);
handler->ExecuteTest();
ReleaseAndWaitForDestructor(handler);
}
namespace {
class RedirectMismatchedFromHandlerTest : public CertificateErrorTest {
public:
RedirectMismatchedFromHandlerTest(bool continue_invalid_certificate,
bool redirect_from_https)
: CertificateErrorTest(
/*cert_type=*/CEF_TEST_CERT_OK_DOMAIN,
/*expect_load_success=*/continue_invalid_certificate,
/*expect_certificate_error=*/true),
redirect_from_https_(redirect_from_https) {}
CefRefPtr<CefResourceHandler> GetResourceHandler(
CefRefPtr<CefBrowser> browser,
CefRefPtr<CefFrame> frame,
CefRefPtr<CefRequest> request) override {
EXPECT_IO_THREAD();
const std::string& url = request->GetURL();
if (url == GetStartURL()) {
// Redirect to the end URL.
return new CefStreamResourceHandler(
301, "Permanent Redirect", "text/html", {{"Location", GetEndURL()}},
/*stream=*/nullptr);
}
return TestHandler::GetResourceHandler(browser, frame, request);
}
protected:
std::string GetStartURL() const override {
return redirect_from_https_ ? "https://certificate-test.com/index.html"
: "http://certificate-test.com/index.html";
}
std::string GetEndURL() const override {
// Load by IP address when the certificate expects a domain.
return client::AsciiStrReplace(server_origin(), "localhost", "127.0.0.1") +
"/index.html";
}
private:
const bool redirect_from_https_;
};
} // namespace
TEST(CertificateErrorTest, RedirectMismatchedFromHttpsResourceCancel) {
CefRefPtr<RedirectMismatchedFromHandlerTest> handler =
new RedirectMismatchedFromHandlerTest(
/*continue_invalid_certificate=*/false,
/*redirect_from_https=*/true);
handler->ExecuteTest();
ReleaseAndWaitForDestructor(handler);
}
TEST(CertificateErrorTest, RedirectMismatchedFromHttpsResourceContinue) {
CefRefPtr<RedirectMismatchedFromHandlerTest> handler =
new RedirectMismatchedFromHandlerTest(
/*continue_invalid_certificate=*/true,
/*redirect_from_https=*/true);
handler->ExecuteTest();
ReleaseAndWaitForDestructor(handler);
}
TEST(CertificateErrorTest, RedirectMismatchedFromHttpResourceCancel) {
CefRefPtr<RedirectMismatchedFromHandlerTest> handler =
new RedirectMismatchedFromHandlerTest(
/*continue_invalid_certificate=*/false,
/*redirect_from_https=*/false);
handler->ExecuteTest();
ReleaseAndWaitForDestructor(handler);
}
TEST(CertificateErrorTest, RedirectMismatchedFromHttpResourceContinue) {
CefRefPtr<RedirectMismatchedFromHandlerTest> handler =
new RedirectMismatchedFromHandlerTest(
/*continue_invalid_certificate=*/true,
/*redirect_from_https=*/false);
handler->ExecuteTest();
ReleaseAndWaitForDestructor(handler);
}
namespace {
class RedirectMismatchedFromServerTest : public CertificateErrorTest {
public:
RedirectMismatchedFromServerTest(bool continue_invalid_certificate,
bool redirect_from_https)
: CertificateErrorTest(
/*cert_type=*/CEF_TEST_CERT_OK_DOMAIN,
/*expect_load_success=*/continue_invalid_certificate,
/*expect_certificate_error=*/true,
redirect_from_https ? OtherServerType::HTTPS
: OtherServerType::HTTP) {}
void HandleOtherServerRequest(
CefRefPtr<CefRequest> request,
const ResponseCallback& response_callback) override {
EXPECT_UI_THREAD();
EXPECT_FALSE(got_other_request_);
got_other_request_.yes();
const std::string& url = request->GetURL();
EXPECT_STREQ(GetStartURL().c_str(), url.c_str());
// Redirect to the end URL.
auto response = CefResponse::Create();
response->SetMimeType("text/html");
response->SetStatus(301); // Permanent Redirect
response->SetHeaderByName("Location", GetEndURL(),
/*overwrite=*/true);
response_callback.Run(response, /*response_body=*/std::string());
}
void DestroyTest() override {
EXPECT_TRUE(got_other_request_);
CertificateErrorTest::DestroyTest();
}
protected:
std::string GetStartURL() const override {
return other_origin() + "/index.html";
}
std::string GetEndURL() const override {
// Load by IP address when the certificate expects a domain.
return client::AsciiStrReplace(server_origin(), "localhost", "127.0.0.1") +
"/index.html";
}
private:
TrackCallback got_other_request_;
};
} // namespace
TEST(CertificateErrorTest, RedirectMismatchedFromHttpsServerCancel) {
CefRefPtr<RedirectMismatchedFromServerTest> handler =
new RedirectMismatchedFromServerTest(
/*continue_invalid_certificate=*/false, /*redirect_from_https=*/true);
handler->ExecuteTest();
ReleaseAndWaitForDestructor(handler);
}
TEST(CertificateErrorTest, RedirectMismatchedFromHttpsServerContinue) {
CefRefPtr<RedirectMismatchedFromServerTest> handler =
new RedirectMismatchedFromServerTest(
/*continue_invalid_certificate=*/true, /*redirect_from_https=*/true);
handler->ExecuteTest();
ReleaseAndWaitForDestructor(handler);
}
TEST(CertificateErrorTest, RedirectMismatchedFromHttpServerCancel) {
CefRefPtr<RedirectMismatchedFromServerTest> handler =
new RedirectMismatchedFromServerTest(
/*continue_invalid_certificate=*/false,
/*redirect_from_https=*/false);
handler->ExecuteTest();
ReleaseAndWaitForDestructor(handler);
}
TEST(CertificateErrorTest, RedirectMismatchedFromHttpServerContinue) {
CefRefPtr<RedirectMismatchedFromServerTest> handler =
new RedirectMismatchedFromServerTest(
/*continue_invalid_certificate=*/true, /*redirect_from_https=*/false);
handler->ExecuteTest();
ReleaseAndWaitForDestructor(handler);
}