mirror of
https://bitbucket.org/chromiumembedded/cef
synced 2025-01-22 07:27:55 +01:00
53ef570f57
Add new CefBrowserHost::[Can]ExecuteChromeCommand methods for executing arbitrary Chrome commands. Add support for existing CefBrowserHost::ShowDevTools, CloseDevTools and HasDevTools methods. DevTools windows now support the same Views callbacks as normal popup windows with the new CefLifeSpanHandler::OnBeforeDevToolsPopup callback as the DevTools-specific equivalent of OnBeforePopup. Always create DevTools as an undocked window to support use of ShowDevTools with default Chrome browser windows. To test: Run `ceftests --enable-chrome-runtime [--use-views] --gtest_filter=V8Test.OnUncaughtExceptionDevTools` OR: 1. Run `cefclient --enable-chrome-runtime [--use-native]` 2. Select "Show DevTools", "Close DevTools" or "Inspect" from the right-click menu. 3. Notice that the DevTools window is Views-hosted (or native-hosted) and works as expected. Add --use-default-popup to get a default styled popup in step 3.
743 lines
21 KiB
C++
743 lines
21 KiB
C++
// Copyright (c) 2013 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 "tests/ceftests/test_handler.h"
|
|
|
|
#include <sstream>
|
|
|
|
#include "include/base/cef_callback.h"
|
|
#include "include/base/cef_logging.h"
|
|
#include "include/cef_command_line.h"
|
|
#include "include/cef_stream.h"
|
|
#include "include/views/cef_browser_view.h"
|
|
#include "include/views/cef_window.h"
|
|
#include "include/wrapper/cef_closure_task.h"
|
|
#include "include/wrapper/cef_stream_resource_handler.h"
|
|
#include "tests/ceftests/test_request.h"
|
|
#include "tests/ceftests/test_util.h"
|
|
#include "tests/shared/common/client_switches.h"
|
|
|
|
// Set to 1 to enable verbose debugging info logging.
|
|
#define VERBOSE_DEBUGGING 0
|
|
|
|
namespace {
|
|
|
|
bool UseViewsGlobal() {
|
|
static bool use_views = []() {
|
|
return CefCommandLine::GetGlobalCommandLine()->HasSwitch(
|
|
client::switches::kUseViews);
|
|
}();
|
|
return use_views;
|
|
}
|
|
|
|
// Delegate implementation for the CefWindow that will host the Views-based
|
|
// browser.
|
|
class TestWindowDelegate : public CefWindowDelegate {
|
|
public:
|
|
// Create a new top-level Window hosting |browser_view|.
|
|
static void CreateBrowserWindow(TestHandler* handler,
|
|
CefRefPtr<CefBrowserView> browser_view,
|
|
const std::string& title) {
|
|
CefWindow::CreateTopLevelWindow(new TestWindowDelegate(
|
|
handler, browser_view, "CefUnitTestViews " + title));
|
|
}
|
|
|
|
// CefWindowDelegate methods:
|
|
|
|
void OnWindowCreated(CefRefPtr<CefWindow> window) override {
|
|
// Add the browser view and show the window.
|
|
window->CenterWindow(CefSize(800, 600));
|
|
window->SetTitle(title_);
|
|
window->AddChildView(browser_view_);
|
|
window->Show();
|
|
|
|
// With Chrome runtime, the Browser is not created until after the
|
|
// BrowserView is assigned to the Window.
|
|
browser_id_ = browser_view_->GetBrowser()->GetIdentifier();
|
|
handler_->OnWindowCreated(browser_id_);
|
|
}
|
|
|
|
void OnWindowDestroyed(CefRefPtr<CefWindow> window) override {
|
|
auto browser = browser_view_->GetBrowser();
|
|
browser_view_ = nullptr;
|
|
handler_->OnWindowDestroyed(browser_id_);
|
|
}
|
|
|
|
bool CanClose(CefRefPtr<CefWindow> window) override {
|
|
// Allow the window to close if the browser says it's OK.
|
|
CefRefPtr<CefBrowser> browser = browser_view_->GetBrowser();
|
|
if (browser) {
|
|
return browser->GetHost()->TryCloseBrowser();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
private:
|
|
TestWindowDelegate(TestHandler* handler,
|
|
CefRefPtr<CefBrowserView> browser_view,
|
|
const CefString& title)
|
|
: handler_(handler), browser_view_(browser_view), title_(title) {}
|
|
|
|
TestHandler* const handler_;
|
|
CefRefPtr<CefBrowserView> browser_view_;
|
|
int browser_id_ = 0;
|
|
CefString title_;
|
|
|
|
IMPLEMENT_REFCOUNTING(TestWindowDelegate);
|
|
DISALLOW_COPY_AND_ASSIGN(TestWindowDelegate);
|
|
};
|
|
|
|
// Delegate implementation for the CefBrowserView.
|
|
class TestBrowserViewDelegate : public CefBrowserViewDelegate {
|
|
public:
|
|
explicit TestBrowserViewDelegate(TestHandler* handler) : handler_(handler) {}
|
|
|
|
// CefBrowserViewDelegate methods:
|
|
|
|
void OnBrowserDestroyed(CefRefPtr<CefBrowserView> browser_view,
|
|
CefRefPtr<CefBrowser> browser) override {
|
|
#if VERBOSE_DEBUGGING
|
|
LOG(INFO) << handler_->debug_string_prefix() << browser->GetIdentifier()
|
|
<< ": OnBrowserDestroyed";
|
|
#endif
|
|
// Always close the containing Window when the browser is destroyed.
|
|
if (auto window = browser_view->GetWindow()) {
|
|
#if VERBOSE_DEBUGGING
|
|
LOG(INFO) << handler_->debug_string_prefix() << browser->GetIdentifier()
|
|
<< ": OnBrowserDestroyed Close";
|
|
#endif
|
|
window->Close();
|
|
}
|
|
}
|
|
|
|
CefRefPtr<CefBrowserViewDelegate> GetDelegateForPopupBrowserView(
|
|
CefRefPtr<CefBrowserView> browser_view,
|
|
const CefBrowserSettings& settings,
|
|
CefRefPtr<CefClient> client,
|
|
bool is_devtools) override {
|
|
if (client.get() == handler_) {
|
|
// Use the same Delegate when using the same TestHandler instance.
|
|
return this;
|
|
}
|
|
|
|
// Return a new Delegate when using a different TestHandler instance.
|
|
auto* handler = static_cast<TestHandler*>(client.get());
|
|
return new TestBrowserViewDelegate(handler);
|
|
}
|
|
|
|
bool OnPopupBrowserViewCreated(CefRefPtr<CefBrowserView> browser_view,
|
|
CefRefPtr<CefBrowserView> popup_browser_view,
|
|
bool is_devtools) override {
|
|
// The popup may use a different TestHandler instance.
|
|
auto* handler = static_cast<TestHandler*>(
|
|
popup_browser_view->GetBrowser()->GetHost()->GetClient().get());
|
|
|
|
// Create our own Window for popups. It will show itself after creation.
|
|
TestWindowDelegate::CreateBrowserWindow(handler, popup_browser_view,
|
|
is_devtools ? "DevTools" : "Popup");
|
|
return true;
|
|
}
|
|
|
|
private:
|
|
TestHandler* const handler_;
|
|
|
|
IMPLEMENT_REFCOUNTING(TestBrowserViewDelegate);
|
|
DISALLOW_COPY_AND_ASSIGN(TestBrowserViewDelegate);
|
|
};
|
|
|
|
} // namespace
|
|
|
|
// TestHandler::CompletionState
|
|
|
|
TestHandler::CompletionState::CompletionState(int total)
|
|
: total_(total), count_(0) {
|
|
event_ = CefWaitableEvent::CreateWaitableEvent(true, false);
|
|
}
|
|
|
|
void TestHandler::CompletionState::TestComplete() {
|
|
if (++count_ == total_) {
|
|
count_ = 0;
|
|
|
|
// Signal that the test is now complete. Do not access any object members
|
|
// after this call because |this| might be deleted.
|
|
event_->Signal();
|
|
}
|
|
}
|
|
|
|
void TestHandler::CompletionState::WaitForTests() {
|
|
// Wait for the test to complete
|
|
event_->Wait();
|
|
|
|
// Reset the event so the same test can be executed again.
|
|
event_->Reset();
|
|
}
|
|
|
|
// TestHandler::Collection
|
|
|
|
TestHandler::Collection::Collection(CompletionState* completion_state)
|
|
: completion_state_(completion_state) {
|
|
EXPECT_TRUE(completion_state_);
|
|
}
|
|
|
|
void TestHandler::Collection::AddTestHandler(TestHandler* test_handler) {
|
|
EXPECT_EQ(test_handler->completion_state_, completion_state_);
|
|
handler_list_.push_back(test_handler);
|
|
}
|
|
|
|
void TestHandler::Collection::ExecuteTests() {
|
|
EXPECT_GT(handler_list_.size(), 0UL);
|
|
|
|
TestHandlerList::const_iterator it;
|
|
|
|
it = handler_list_.begin();
|
|
for (; it != handler_list_.end(); ++it) {
|
|
(*it)->SetupTest();
|
|
}
|
|
|
|
completion_state_->WaitForTests();
|
|
|
|
it = handler_list_.begin();
|
|
for (; it != handler_list_.end(); ++it) {
|
|
(*it)->RunTest();
|
|
}
|
|
|
|
completion_state_->WaitForTests();
|
|
}
|
|
|
|
// TestHandler::UIThreadHelper
|
|
|
|
TestHandler::UIThreadHelper::UIThreadHelper() : weak_ptr_factory_(this) {}
|
|
|
|
void TestHandler::UIThreadHelper::PostTask(base::OnceClosure task) {
|
|
EXPECT_UI_THREAD();
|
|
CefPostTask(TID_UI,
|
|
base::BindOnce(&UIThreadHelper::TaskHelper,
|
|
weak_ptr_factory_.GetWeakPtr(), std::move(task)));
|
|
}
|
|
|
|
void TestHandler::UIThreadHelper::PostDelayedTask(base::OnceClosure task,
|
|
int delay_ms) {
|
|
EXPECT_UI_THREAD();
|
|
CefPostDelayedTask(
|
|
TID_UI,
|
|
base::BindOnce(&UIThreadHelper::TaskHelper,
|
|
weak_ptr_factory_.GetWeakPtr(), std::move(task)),
|
|
delay_ms);
|
|
}
|
|
|
|
void TestHandler::UIThreadHelper::TaskHelper(base::OnceClosure task) {
|
|
EXPECT_UI_THREAD();
|
|
std::move(task).Run();
|
|
}
|
|
|
|
// TestHandler
|
|
|
|
// static
|
|
std::atomic<size_t> TestHandler::test_handler_count_{0U};
|
|
|
|
TestHandler::TestHandler(CompletionState* completion_state)
|
|
: debug_string_prefix_(MakeDebugStringPrefix()),
|
|
use_views_(UseViewsGlobal()) {
|
|
test_handler_count_++;
|
|
|
|
if (completion_state) {
|
|
completion_state_ = completion_state;
|
|
completion_state_owned_ = false;
|
|
} else {
|
|
completion_state_ = new CompletionState(1);
|
|
completion_state_owned_ = true;
|
|
}
|
|
}
|
|
|
|
TestHandler::~TestHandler() {
|
|
DCHECK(!ui_thread_helper_.get());
|
|
if (destroy_test_expected_) {
|
|
EXPECT_TRUE(destroy_test_called_);
|
|
} else {
|
|
EXPECT_FALSE(destroy_test_called_);
|
|
}
|
|
EXPECT_TRUE(browser_map_.empty());
|
|
EXPECT_TRUE(browser_status_map_.empty());
|
|
|
|
if (completion_state_owned_) {
|
|
delete completion_state_;
|
|
}
|
|
|
|
if (destroy_event_) {
|
|
destroy_event_->Signal();
|
|
}
|
|
|
|
test_handler_count_--;
|
|
}
|
|
|
|
void TestHandler::OnAfterCreated(CefRefPtr<CefBrowser> browser) {
|
|
EXPECT_UI_THREAD();
|
|
|
|
const int browser_id = browser->GetIdentifier();
|
|
EXPECT_EQ(browser_map_.find(browser_id), browser_map_.end());
|
|
if (browser_map_.empty()) {
|
|
first_browser_id_ = browser_id;
|
|
first_browser_ = browser;
|
|
}
|
|
browser_map_.insert(std::make_pair(browser_id, browser));
|
|
|
|
const bool views_hosted = !!CefBrowserView::GetForBrowser(browser);
|
|
OnCreated(browser_id, NT_BROWSER, views_hosted);
|
|
}
|
|
|
|
void TestHandler::OnBeforeClose(CefRefPtr<CefBrowser> browser) {
|
|
EXPECT_UI_THREAD();
|
|
|
|
// Free the browser pointer so that the browser can be destroyed.
|
|
const int browser_id = browser->GetIdentifier();
|
|
BrowserMap::iterator it = browser_map_.find(browser_id);
|
|
EXPECT_NE(it, browser_map_.end());
|
|
browser_map_.erase(it);
|
|
|
|
if (browser_id == first_browser_id_) {
|
|
first_browser_id_ = 0;
|
|
first_browser_ = nullptr;
|
|
}
|
|
|
|
OnClosed(browser_id, NT_BROWSER);
|
|
}
|
|
|
|
void TestHandler::OnWindowCreated(int browser_id) {
|
|
CHECK(use_views_);
|
|
EXPECT_UI_THREAD();
|
|
OnCreated(browser_id, NT_WINDOW, /*views_hosted=*/true);
|
|
}
|
|
|
|
void TestHandler::OnWindowDestroyed(int browser_id) {
|
|
CHECK(use_views_);
|
|
EXPECT_UI_THREAD();
|
|
OnClosed(browser_id, NT_WINDOW);
|
|
}
|
|
|
|
void TestHandler::OnCreated(int browser_id,
|
|
NotifyType type,
|
|
bool views_hosted) {
|
|
CHECK(use_views_ || !views_hosted);
|
|
|
|
const bool has_value = browser_status_map_.contains(browser_id);
|
|
|
|
auto& browser_status = browser_status_map_[browser_id];
|
|
if (has_value) {
|
|
CHECK_EQ(browser_status.views_hosted, views_hosted);
|
|
} else {
|
|
browser_status.views_hosted = views_hosted;
|
|
}
|
|
|
|
EXPECT_FALSE(browser_status.got_created[type])
|
|
<< "Duplicate call to OnCreated(" << browser_id << ", "
|
|
<< (type == NT_BROWSER ? "BROWSER" : "WINDOW") << ")";
|
|
browser_status.got_created[type].yes();
|
|
|
|
#if VERBOSE_DEBUGGING
|
|
bool creation_complete = false;
|
|
|
|
// When using Views, wait for both Browser and Window notifications.
|
|
if (browser_status.views_hosted) {
|
|
creation_complete = browser_status.got_created[NT_BROWSER] &&
|
|
browser_status.got_created[NT_WINDOW];
|
|
} else {
|
|
creation_complete = browser_status.got_created[NT_BROWSER];
|
|
}
|
|
|
|
LOG(INFO) << debug_string_prefix_ << browser_id << ": OnCreated type="
|
|
<< (type == NT_BROWSER ? "BROWSER" : "WINDOW")
|
|
<< " creation_complete=" << creation_complete;
|
|
#endif // VERBOSE_DEBUGGING
|
|
}
|
|
|
|
void TestHandler::OnClosed(int browser_id, NotifyType type) {
|
|
bool close_complete = false;
|
|
|
|
auto& browser_status = browser_status_map_[browser_id];
|
|
EXPECT_FALSE(browser_status.got_closed[type])
|
|
<< "Duplicate call to OnClosed(" << browser_id << ", "
|
|
<< (type == NT_BROWSER ? "BROWSER" : "WINDOW") << ")";
|
|
browser_status.got_closed[type].yes();
|
|
|
|
// When using Views, wait for both Browser and Window notifications.
|
|
if (browser_status.views_hosted) {
|
|
close_complete = browser_status.got_closed[NT_BROWSER] &&
|
|
browser_status.got_closed[NT_WINDOW];
|
|
} else {
|
|
close_complete = browser_status.got_closed[NT_BROWSER];
|
|
}
|
|
|
|
if (close_complete) {
|
|
browser_status_map_.erase(browser_id);
|
|
}
|
|
|
|
// Test may be complete if no Browsers/Windows are remaining.
|
|
const bool all_browsers_closed = AllBrowsersClosed();
|
|
|
|
#if VERBOSE_DEBUGGING
|
|
LOG(INFO) << debug_string_prefix_ << browser_id
|
|
<< ": OnClosed type=" << (type == NT_BROWSER ? "BROWSER" : "WINDOW")
|
|
<< " close_complete=" << close_complete
|
|
<< " all_browsers_closed=" << all_browsers_closed;
|
|
#endif
|
|
|
|
if (all_browsers_closed) {
|
|
// May result in |this| being deleted.
|
|
MaybeTestComplete();
|
|
}
|
|
}
|
|
|
|
std::string TestHandler::MakeDebugStringPrefix() const {
|
|
#if VERBOSE_DEBUGGING
|
|
std::stringstream ss;
|
|
ss << "TestHandler [0x" << std::hex << this << "]: ";
|
|
return ss.str();
|
|
#else
|
|
return std::string();
|
|
#endif
|
|
}
|
|
|
|
namespace {
|
|
|
|
CefResponse::HeaderMap ToCefHeaderMap(
|
|
const ResourceContent::HeaderMap& headerMap) {
|
|
CefResponse::HeaderMap result;
|
|
ResourceContent::HeaderMap::const_iterator it = headerMap.begin();
|
|
for (; it != headerMap.end(); ++it) {
|
|
result.insert(std::pair<CefString, CefString>(it->first, it->second));
|
|
}
|
|
return result;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
CefRefPtr<CefResourceHandler> TestHandler::GetResourceHandler(
|
|
CefRefPtr<CefBrowser> browser,
|
|
CefRefPtr<CefFrame> frame,
|
|
CefRefPtr<CefRequest> request) {
|
|
EXPECT_IO_THREAD();
|
|
|
|
if (resource_map_.size() > 0) {
|
|
const std::string& url = test_request::GetPathURL(request->GetURL());
|
|
ResourceMap::const_iterator it = resource_map_.find(url);
|
|
if (it != resource_map_.end()) {
|
|
// Return the previously mapped resource
|
|
CefRefPtr<CefStreamReader> stream = CefStreamReader::CreateForData(
|
|
static_cast<void*>(const_cast<char*>(it->second.content().c_str())),
|
|
it->second.content().length());
|
|
return new CefStreamResourceHandler(
|
|
200, "OK", it->second.mimeType(),
|
|
ToCefHeaderMap(it->second.headerMap()), stream);
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
void TestHandler::OnRenderProcessTerminated(CefRefPtr<CefBrowser> browser,
|
|
TerminationStatus status) {
|
|
LOG(WARNING) << "OnRenderProcessTerminated: status = " << status << ".";
|
|
}
|
|
|
|
CefRefPtr<CefBrowser> TestHandler::GetBrowser() const {
|
|
return first_browser_;
|
|
}
|
|
|
|
int TestHandler::GetBrowserId() const {
|
|
return first_browser_id_;
|
|
}
|
|
|
|
void TestHandler::GetAllBrowsers(BrowserMap* map) {
|
|
EXPECT_UI_THREAD();
|
|
EXPECT_TRUE(map);
|
|
*map = browser_map_;
|
|
}
|
|
|
|
void TestHandler::ExecuteTest() {
|
|
EXPECT_EQ(completion_state_->total(), 1);
|
|
|
|
// Reset any state from the previous run.
|
|
if (test_timeout_called_) {
|
|
test_timeout_called_ = false;
|
|
}
|
|
if (destroy_test_called_) {
|
|
destroy_test_called_ = false;
|
|
}
|
|
|
|
// Run the test.
|
|
RunTest();
|
|
|
|
// Wait for the test to complete.
|
|
completion_state_->WaitForTests();
|
|
}
|
|
|
|
void TestHandler::SetupComplete() {
|
|
// Signal that the test setup is complete.
|
|
completion_state_->TestComplete();
|
|
}
|
|
|
|
void TestHandler::DestroyTest() {
|
|
if (!CefCurrentlyOn(TID_UI)) {
|
|
CefPostTask(TID_UI, base::BindOnce(&TestHandler::DestroyTest, this));
|
|
return;
|
|
}
|
|
|
|
EXPECT_TRUE(destroy_test_expected_);
|
|
if (destroy_test_called_) {
|
|
return;
|
|
}
|
|
destroy_test_called_ = true;
|
|
|
|
if (!browser_map_.empty()) {
|
|
// Use a copy of the map since the original may be modified while we're
|
|
// iterating.
|
|
BrowserMap browser_map = browser_map_;
|
|
|
|
// Tell all browsers to close.
|
|
BrowserMap::const_iterator it = browser_map.begin();
|
|
for (; it != browser_map.end(); ++it) {
|
|
CloseBrowser(it->second, false);
|
|
}
|
|
}
|
|
}
|
|
|
|
void TestHandler::OnTestTimeout(int timeout_ms, bool treat_as_error) {
|
|
EXPECT_UI_THREAD();
|
|
|
|
EXPECT_FALSE(test_timeout_called_);
|
|
test_timeout_called_ = true;
|
|
|
|
if (treat_as_error) {
|
|
EXPECT_TRUE(false) << "Test timed out after " << timeout_ms << "ms";
|
|
}
|
|
|
|
EXPECT_FALSE(AllBrowsersClosed() && AllowTestCompletionWhenAllBrowsersClose())
|
|
<< "Test timed out unexpectedly; should be complete";
|
|
|
|
// Keep |this| alive until after the method completes.
|
|
CefRefPtr<TestHandler> self(this);
|
|
|
|
// Close any remaining browsers.
|
|
DestroyTest();
|
|
|
|
// Reset signal completion count.
|
|
if (signal_completion_count_ > 0) {
|
|
signal_completion_count_ = 0;
|
|
MaybeTestComplete();
|
|
}
|
|
}
|
|
|
|
void TestHandler::CreateBrowser(const CefString& url,
|
|
CefRefPtr<CefRequestContext> request_context,
|
|
CefRefPtr<CefDictionaryValue> extra_info) {
|
|
if (!CefCurrentlyOn(TID_UI)) {
|
|
CefPostTask(TID_UI, base::BindOnce(&TestHandler::CreateBrowser, this, url,
|
|
request_context, extra_info));
|
|
return;
|
|
}
|
|
|
|
CefWindowInfo windowInfo;
|
|
CefBrowserSettings settings;
|
|
|
|
if (use_views_) {
|
|
// Create the BrowserView.
|
|
CefRefPtr<CefBrowserView> browser_view = CefBrowserView::CreateBrowserView(
|
|
this, url, settings, extra_info, request_context,
|
|
new TestBrowserViewDelegate(this));
|
|
|
|
// Create the Window. It will show itself after creation.
|
|
TestWindowDelegate::CreateBrowserWindow(this, browser_view, std::string());
|
|
} else {
|
|
#if defined(OS_WIN)
|
|
windowInfo.SetAsPopup(nullptr, "CefUnitTest");
|
|
windowInfo.style |= WS_VISIBLE;
|
|
#endif
|
|
CefBrowserHost::CreateBrowser(windowInfo, this, url, settings, extra_info,
|
|
request_context);
|
|
}
|
|
}
|
|
|
|
// static
|
|
void TestHandler::CloseBrowser(CefRefPtr<CefBrowser> browser,
|
|
bool force_close) {
|
|
#if VERBOSE_DEBUGGING
|
|
LOG(INFO) << "TestHandler: " << browser->GetIdentifier()
|
|
<< ": CloseBrowser force_close=" << force_close;
|
|
#endif
|
|
browser->GetHost()->CloseBrowser(force_close);
|
|
}
|
|
|
|
void TestHandler::AddResource(const std::string& url,
|
|
const std::string& content,
|
|
const std::string& mime_type,
|
|
const ResourceContent::HeaderMap& header_map) {
|
|
ResourceContent rc = ResourceContent(content, mime_type, header_map);
|
|
AddResourceEx(url, rc);
|
|
}
|
|
|
|
void TestHandler::AddResourceEx(const std::string& url,
|
|
const ResourceContent& content) {
|
|
if (!CefCurrentlyOn(TID_IO)) {
|
|
CefPostTask(TID_IO, base::BindOnce(&TestHandler::AddResourceEx, this, url,
|
|
content));
|
|
return;
|
|
}
|
|
|
|
// Ignore the query component, if any.
|
|
std::string urlStr = url;
|
|
size_t idx = urlStr.find('?');
|
|
if (idx > 0) {
|
|
urlStr = urlStr.substr(0, idx);
|
|
}
|
|
|
|
resource_map_.insert(std::make_pair(urlStr, content));
|
|
}
|
|
|
|
void TestHandler::ClearResources() {
|
|
if (!CefCurrentlyOn(TID_IO)) {
|
|
CefPostTask(TID_IO, base::BindOnce(&TestHandler::ClearResources, this));
|
|
return;
|
|
}
|
|
|
|
resource_map_.clear();
|
|
}
|
|
|
|
void TestHandler::SetTestTimeout(int timeout_ms, bool treat_as_error) {
|
|
if (!CefCurrentlyOn(TID_UI)) {
|
|
CefPostTask(TID_UI, base::BindOnce(&TestHandler::SetTestTimeout, this,
|
|
timeout_ms, treat_as_error));
|
|
return;
|
|
}
|
|
|
|
if (destroy_test_called_) {
|
|
// No need to set the timeout if the test has already completed.
|
|
return;
|
|
}
|
|
|
|
const auto timeout = GetConfiguredTestTimeout(timeout_ms);
|
|
if (treat_as_error && !timeout) {
|
|
return;
|
|
}
|
|
|
|
// Use a weak reference to |this| via UIThreadHelper so that the TestHandler
|
|
// can be destroyed before the timeout expires.
|
|
GetUIThreadHelper()->PostDelayedTask(
|
|
base::BindOnce(&TestHandler::OnTestTimeout, base::Unretained(this),
|
|
*timeout, treat_as_error),
|
|
*timeout);
|
|
}
|
|
|
|
void TestHandler::SetUseViews(bool use_views) {
|
|
if (!CefCurrentlyOn(TID_UI)) {
|
|
CefPostTask(TID_UI,
|
|
base::BindOnce(&TestHandler::SetUseViews, this, use_views));
|
|
return;
|
|
}
|
|
|
|
use_views_ = use_views;
|
|
}
|
|
|
|
void TestHandler::SetSignalTestCompletionCount(size_t count) {
|
|
#if VERBOSE_DEBUGGING
|
|
LOG(INFO) << debug_string_prefix_
|
|
<< "SetSignalTestCompletionCount count=" << count;
|
|
#endif
|
|
signal_completion_count_ = count;
|
|
}
|
|
|
|
void TestHandler::SignalTestCompletion() {
|
|
if (!CefCurrentlyOn(TID_UI)) {
|
|
CefPostTask(TID_UI,
|
|
base::BindOnce(&TestHandler::SignalTestCompletion, this));
|
|
return;
|
|
}
|
|
|
|
if (test_timeout_called_) {
|
|
// Ignore any signals that arrive after test timeout.
|
|
return;
|
|
}
|
|
|
|
CHECK_GT(signal_completion_count_, 0U);
|
|
signal_completion_count_--;
|
|
|
|
#if VERBOSE_DEBUGGING
|
|
LOG(INFO) << debug_string_prefix_
|
|
<< "SignalTestComplete remaining=" << signal_completion_count_;
|
|
#endif
|
|
|
|
if (signal_completion_count_ == 0) {
|
|
// May result in |this| being deleted.
|
|
MaybeTestComplete();
|
|
}
|
|
}
|
|
|
|
bool TestHandler::AllowTestCompletionWhenAllBrowsersClose() const {
|
|
EXPECT_UI_THREAD();
|
|
return signal_completion_count_ == 0U;
|
|
}
|
|
|
|
bool TestHandler::AllBrowsersClosed() const {
|
|
EXPECT_UI_THREAD();
|
|
return browser_status_map_.empty();
|
|
}
|
|
|
|
void TestHandler::MaybeTestComplete() {
|
|
EXPECT_UI_THREAD();
|
|
|
|
const bool all_browsers_closed = AllBrowsersClosed();
|
|
const bool allow_test_completion = AllowTestCompletionWhenAllBrowsersClose();
|
|
|
|
#if VERBOSE_DEBUGGING
|
|
LOG(INFO) << debug_string_prefix_
|
|
<< "MaybeTestComplete all_browsers_closed=" << all_browsers_closed
|
|
<< " allow_test_completion=" << allow_test_completion;
|
|
#endif
|
|
|
|
if (all_browsers_closed && allow_test_completion) {
|
|
TestComplete();
|
|
}
|
|
}
|
|
|
|
void TestHandler::TestComplete() {
|
|
EXPECT_UI_THREAD();
|
|
EXPECT_TRUE(AllBrowsersClosed());
|
|
EXPECT_TRUE(AllowTestCompletionWhenAllBrowsersClose());
|
|
|
|
#if VERBOSE_DEBUGGING
|
|
LOG(INFO) << debug_string_prefix_ << "TestComplete";
|
|
#endif
|
|
|
|
// Cancel any pending tasks posted via UIThreadHelper.
|
|
ui_thread_helper_.reset();
|
|
|
|
completion_state_->TestComplete();
|
|
}
|
|
|
|
TestHandler::UIThreadHelper* TestHandler::GetUIThreadHelper() {
|
|
EXPECT_UI_THREAD();
|
|
CHECK(!destroy_test_called_);
|
|
|
|
if (!ui_thread_helper_.get()) {
|
|
ui_thread_helper_.reset(new UIThreadHelper());
|
|
}
|
|
return ui_thread_helper_.get();
|
|
}
|
|
|
|
// global functions
|
|
|
|
bool TestFailed() {
|
|
CefRefPtr<CefCommandLine> command_line =
|
|
CefCommandLine::GetGlobalCommandLine();
|
|
if (command_line->HasSwitch("single-process")) {
|
|
// Check for a failure on the current test only.
|
|
return ::testing::UnitTest::GetInstance()
|
|
->current_test_info()
|
|
->result()
|
|
->Failed();
|
|
} else {
|
|
// Check for any global failure.
|
|
return ::testing::UnitTest::GetInstance()->Failed();
|
|
}
|
|
}
|