// Copyright (c) 2009 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.
#undef LOG
#include "webkit/tools/test_shell/test_shell.h"
#include "app/gfx/codec/png_codec.h"
#include "base/base_paths.h"
#include "base/command_line.h"
#include "base/debug_on_start.h"
#include "base/file_path.h"
#include "base/file_util.h"
#include "base/gfx/size.h"
#if defined(OS_MACOSX)
#include "base/mac_util.h"
#endif
#include "base/md5.h"
#include "base/message_loop.h"
#include "base/path_service.h"
#include "base/stats_table.h"
#include "base/string_util.h"
#include "build/build_config.h"
#include "googleurl/src/url_util.h"
#include "grit/webkit_strings.h"
#include "net/base/mime_util.h"
#include "net/base/net_util.h"
#include "net/url_request/url_request_file_job.h"
#include "net/url_request/url_request_filter.h"
#include "skia/ext/bitmap_platform_device.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/skia/include/core/SkBitmap.h"
#include "third_party/WebKit/WebKit/chromium/public/WebAccessibilityObject.h"
#include "third_party/WebKit/WebKit/chromium/public/WebFrame.h"
#include "third_party/WebKit/WebKit/chromium/public/WebKit.h"
#include "third_party/WebKit/WebKit/chromium/public/WebScriptController.h"
#include "third_party/WebKit/WebKit/chromium/public/WebRect.h"
#include "third_party/WebKit/WebKit/chromium/public/WebSize.h"
#include "third_party/WebKit/WebKit/chromium/public/WebString.h"
#include "third_party/WebKit/WebKit/chromium/public/WebURL.h"
#include "third_party/WebKit/WebKit/chromium/public/WebURLRequest.h"
#include "third_party/WebKit/WebKit/chromium/public/WebURLResponse.h"
#include "third_party/WebKit/WebKit/chromium/public/WebView.h"
#include "webkit/glue/glue_serialize.h"
#include "webkit/glue/webkit_glue.h"
#include "webkit/glue/webpreferences.h"
#include "webkit/tools/test_shell/accessibility_controller.h"
#include "webkit/tools/test_shell/simple_resource_loader_bridge.h"
#include "webkit/tools/test_shell/test_navigation_controller.h"
#include "webkit/tools/test_shell/test_shell_switches.h"
using WebKit::WebCanvas;
using WebKit::WebFrame;
using WebKit::WebNavigationPolicy;
using WebKit::WebRect;
using WebKit::WebScriptController;
using WebKit::WebSize;
using WebKit::WebURLRequest;
using WebKit::WebView;
namespace {
// Default timeout in ms for file page loads when in layout test mode.
const int kDefaultFileTestTimeoutMillisecs = 10 * 1000;
// Content area size for newly created windows.
const int kTestWindowWidth = 800;
const int kTestWindowHeight = 600;
// The W3C SVG layout tests use a different size than the other layout
// tests.
const int kSVGTestWindowWidth = 480;
const int kSVGTestWindowHeight = 360;
// URLRequestTestShellFileJob is used to serve the inspector
class URLRequestTestShellFileJob : public URLRequestFileJob {
public:
static URLRequestJob* InspectorFactory(URLRequest* request,
const std::string& scheme) {
FilePath path;
PathService::Get(base::DIR_EXE, &path);
path = path.AppendASCII("resources");
path = path.AppendASCII("inspector");
path = path.AppendASCII(request->url().path().substr(1));
return new URLRequestTestShellFileJob(request, path);
}
private:
URLRequestTestShellFileJob(URLRequest* request, const FilePath& path)
: URLRequestFileJob(request, path) {
}
virtual ~URLRequestTestShellFileJob() { }
DISALLOW_COPY_AND_ASSIGN(URLRequestTestShellFileJob);
};
} // namespace
// Initialize static member variable
WindowList* TestShell::window_list_;
WebPreferences* TestShell::web_prefs_ = NULL;
bool TestShell::layout_test_mode_ = false;
int TestShell::file_test_timeout_ms_ = kDefaultFileTestTimeoutMillisecs;
bool TestShell::test_is_preparing_ = false;
bool TestShell::test_is_pending_ = false;
TestShell::TestShell()
: m_mainWnd(NULL),
m_editWnd(NULL),
m_webViewHost(NULL),
m_popupHost(NULL),
m_focusedWidgetHost(NULL),
#if defined(OS_WIN)
default_edit_wnd_proc_(0),
#endif
test_params_(NULL),
is_modal_(false),
dump_stats_table_on_exit_(false) {
accessibility_controller_.reset(new AccessibilityController(this));
delegate_.reset(new TestWebViewDelegate(this));
popup_delegate_.reset(new TestWebViewDelegate(this));
layout_test_controller_.reset(new LayoutTestController(this));
event_sending_controller_.reset(new EventSendingController(this));
plain_text_controller_.reset(new PlainTextController(this));
text_input_controller_.reset(new TextInputController(this));
navigation_controller_.reset(new TestNavigationController(this));
URLRequestFilter* filter = URLRequestFilter::GetInstance();
filter->AddHostnameHandler("test-shell-resource", "inspector",
&URLRequestTestShellFileJob::InspectorFactory);
url_util::AddStandardScheme("test-shell-resource");
}
TestShell::~TestShell() {
delegate_->RevokeDragDrop();
// Navigate to an empty page to fire all the destruction logic for the
// current page.
LoadURL(GURL("about:blank"));
// Call GC twice to clean up garbage.
CallJSGC();
CallJSGC();
// Destroy the WebView before the TestWebViewDelegate.
m_webViewHost.reset();
PlatformCleanUp();
StatsTable *table = StatsTable::current();
if (dump_stats_table_on_exit_) {
// Dump the stats table.
printf("\n");
if (table != NULL) {
int counter_max = table->GetMaxCounters();
for (int index = 0; index < counter_max; index++) {
std::string name(table->GetRowName(index));
if (name.length() > 0) {
int value = table->GetRowValue(index);
printf("%s:\t%d\n", name.c_str(), value);
}
}
}
printf("\n");
}
}
bool TestShell::CreateNewWindow(const GURL& starting_url,
TestShell** result) {
TestShell* shell = new TestShell();
bool rv = shell->Initialize(starting_url);
if (rv) {
if (result)
*result = shell;
TestShell::windowList()->push_back(shell->m_mainWnd);
}
return rv;
}
void TestShell::ShutdownTestShell() {
PlatformShutdown();
SimpleResourceLoaderBridge::Shutdown();
delete window_list_;
delete TestShell::web_prefs_;
}
// All fatal log messages (e.g. DCHECK failures) imply unit test failures
static void UnitTestAssertHandler(const std::string& str) {
FAIL() << str;
}
// static
void TestShell::Dump(TestShell* shell) {
const TestParams* params = NULL;
if ((shell == NULL) || ((params = shell->test_params()) == NULL))
return;
WebScriptController::flushConsoleMessages();
// Dump the requested representation.
WebFrame* frame = shell->webView()->mainFrame();
if (frame) {
bool should_dump_as_text =
shell->layout_test_controller_->ShouldDumpAsText();
bool dumped_anything = false;
if (params->dump_tree) {
dumped_anything = true;
// Text output: the test page can request different types of output
// which we handle here.
if (!should_dump_as_text) {
// Plain text pages should be dumped as text
const string16& mime_type =
frame->dataSource()->response().mimeType();
should_dump_as_text = EqualsASCII(mime_type, "text/plain");
}
if (should_dump_as_text) {
bool recursive = shell->layout_test_controller_->
ShouldDumpChildFramesAsText();
std::string data_utf8 = WideToUTF8(
webkit_glue::DumpFramesAsText(frame, recursive));
if (fwrite(data_utf8.c_str(), 1, data_utf8.size(), stdout) !=
data_utf8.size()) {
LOG(FATAL) << "Short write to stdout, disk full?";
}
} else {
printf("%s", WideToUTF8(
webkit_glue::DumpRenderer(frame)).c_str());
bool recursive = shell->layout_test_controller_->
ShouldDumpChildFrameScrollPositions();
printf("%s", WideToUTF8(
webkit_glue::DumpFrameScrollPosition(frame, recursive)).c_str());
}
if (shell->layout_test_controller_->ShouldDumpBackForwardList()) {
std::wstring bfDump;
DumpAllBackForwardLists(&bfDump);
printf("%s", WideToUTF8(bfDump).c_str());
}
}
if (params->dump_pixels && !should_dump_as_text) {
// Image output: we write the image data to the file given on the
// command line (for the dump pixels argument), and the MD5 sum to
// stdout.
dumped_anything = true;
WebViewHost* view_host = shell->webViewHost();
view_host->webview()->layout();
view_host->Paint();
std::string md5sum = DumpImage(view_host->canvas(),
params->pixel_file_name, params->pixel_hash);
printf("#MD5:%s\n", md5sum.c_str());
}
if (dumped_anything)
printf("#EOF\n");
fflush(stdout);
}
}
// static
std::string TestShell::DumpImage(skia::PlatformCanvas* canvas,
const std::wstring& file_name, const std::string& pixel_hash) {
skia::BitmapPlatformDevice& device =
static_cast(canvas->getTopPlatformDevice());
const SkBitmap& src_bmp = device.accessBitmap(false);
// Encode image.
std::vector png;
SkAutoLockPixels src_bmp_lock(src_bmp);
gfx::PNGCodec::ColorFormat color_format = gfx::PNGCodec::FORMAT_BGRA;
// Fix the alpha. The expected PNGs on Mac have an alpha channel, so we want
// to keep it. On Windows, the alpha channel is wrong since text/form control
// drawing may have erased it in a few places. So on Windows we force it to
// opaque and also don't write the alpha channel for the reference. Linux
// doesn't have the wrong alpha like Windows, but we ignore it anyway.
#if defined(OS_WIN)
bool discard_transparency = true;
device.makeOpaque(0, 0, src_bmp.width(), src_bmp.height());
#elif defined(OS_LINUX)
bool discard_transparency = true;
#elif defined(OS_MACOSX)
bool discard_transparency = false;
#endif
// Compute MD5 sum. We should have done this before calling
// device.makeOpaque on Windows. Because we do it after the call, there are
// some images that are the pixel identical on windows and other platforms
// but have different MD5 sums. At this point, rebaselining all the windows
// tests is too much of a pain, so we just check in different baselines.
MD5Context ctx;
MD5Init(&ctx);
MD5Update(&ctx, src_bmp.getPixels(), src_bmp.getSize());
MD5Digest digest;
MD5Final(&digest, &ctx);
std::string md5hash = MD5DigestToBase16(digest);
// Only encode and dump the png if the hashes don't match. Encoding the image
// is really expensive.
if (md5hash.compare(pixel_hash) != 0) {
gfx::PNGCodec::Encode(
reinterpret_cast(src_bmp.getPixels()),
color_format, src_bmp.width(), src_bmp.height(),
static_cast(src_bmp.rowBytes()), discard_transparency, &png);
// Write to disk.
file_util::WriteFile(file_name, reinterpret_cast(&png[0]),
png.size());
}
return md5hash;
}
// static
void TestShell::InitLogging(bool suppress_error_dialogs,
bool layout_test_mode,
bool enable_gp_fault_error_box) {
if (suppress_error_dialogs)
logging::SetLogAssertHandler(UnitTestAssertHandler);
#if defined(OS_WIN)
if (!IsDebuggerPresent()) {
UINT new_flags = SEM_FAILCRITICALERRORS |
SEM_NOOPENFILEERRORBOX;
if (!enable_gp_fault_error_box)
new_flags |= SEM_NOGPFAULTERRORBOX;
// Preserve existing error mode, as discussed at
// http://blogs.msdn.com/oldnewthing/archive/2004/07/27/198410.aspx
UINT existing_flags = SetErrorMode(new_flags);
SetErrorMode(existing_flags | new_flags);
}
#endif
// Only log to a file if we're running layout tests. This prevents debugging
// output from disrupting whether or not we pass.
logging::LoggingDestination destination =
logging::LOG_TO_BOTH_FILE_AND_SYSTEM_DEBUG_LOG;
if (layout_test_mode)
destination = logging::LOG_ONLY_TO_FILE;
// We might have multiple test_shell processes going at once
FilePath log_filename;
PathService::Get(base::DIR_EXE, &log_filename);
log_filename = log_filename.AppendASCII("test_shell.log");
logging::InitLogging(log_filename.value().c_str(),
destination,
logging::LOCK_LOG_FILE,
logging::DELETE_OLD_LOG_FILE);
// we want process and thread IDs because we may have multiple processes
logging::SetLogItems(true, true, false, true);
// Turn on logging of notImplemented()s inside WebKit, but only if we're
// not running layout tests (because otherwise they'd corrupt the test
// output).
if (!layout_test_mode)
webkit_glue::EnableWebCoreNotImplementedLogging();
}
// static
void TestShell::CleanupLogging() {
logging::CloseLogFile();
}
// static
void TestShell::SetAllowScriptsToCloseWindows() {
if (web_prefs_)
web_prefs_->allow_scripts_to_close_windows = true;
}
// static
void TestShell::ResetWebPreferences() {
DCHECK(web_prefs_);
// Match the settings used by Mac DumpRenderTree, with the exception of
// fonts.
if (web_prefs_) {
*web_prefs_ = WebPreferences();
#if defined(OS_MACOSX)
web_prefs_->serif_font_family = L"Times";
web_prefs_->cursive_font_family = L"Apple Chancery";
web_prefs_->fantasy_font_family = L"Papyrus";
#else
// NOTE: case matters here, this must be 'times new roman', else
// some layout tests fail.
web_prefs_->serif_font_family = L"times new roman";
// These two fonts are picked from the intersection of
// Win XP font list and Vista font list :
// http://www.microsoft.com/typography/fonts/winxp.htm
// http://blogs.msdn.com/michkap/archive/2006/04/04/567881.aspx
// Some of them are installed only with CJK and complex script
// support enabled on Windows XP and are out of consideration here.
// (although we enabled both on our buildbots.)
// They (especially Impact for fantasy) are not typical cursive
// and fantasy fonts, but it should not matter for layout tests
// as long as they're available.
web_prefs_->cursive_font_family = L"Comic Sans MS";
web_prefs_->fantasy_font_family = L"Impact";
#endif
web_prefs_->standard_font_family = web_prefs_->serif_font_family;
web_prefs_->fixed_font_family = L"Courier";
web_prefs_->sans_serif_font_family = L"Helvetica";
web_prefs_->default_encoding = "ISO-8859-1";
web_prefs_->default_font_size = 16;
web_prefs_->default_fixed_font_size = 13;
web_prefs_->minimum_font_size = 1;
web_prefs_->minimum_logical_font_size = 9;
web_prefs_->javascript_can_open_windows_automatically = true;
web_prefs_->dom_paste_enabled = true;
web_prefs_->developer_extras_enabled = !layout_test_mode_;
web_prefs_->shrinks_standalone_images_to_fit = false;
web_prefs_->uses_universal_detector = false;
web_prefs_->text_areas_are_resizable = false;
web_prefs_->java_enabled = true;
web_prefs_->allow_scripts_to_close_windows = false;
web_prefs_->xss_auditor_enabled = false;
// It's off by default for Chrome, but we don't want to
// lose the coverage of dynamic font tests in webkit test.
web_prefs_->remote_fonts_enabled = true;
web_prefs_->local_storage_enabled = true;
web_prefs_->application_cache_enabled = true;
web_prefs_->databases_enabled = true;
// LayoutTests were written with Safari Mac in mind which does not allow
// tabbing to links by default.
web_prefs_->tabs_to_links = false;
// Allow those layout tests running as local files, i.e. under
// LayoutTests/http/tests/local, to access http server.
if (layout_test_mode_)
web_prefs_->allow_universal_access_from_file_urls = true;
}
}
// static
bool TestShell::RemoveWindowFromList(gfx::NativeWindow window) {
WindowList::iterator entry =
std::find(TestShell::windowList()->begin(),
TestShell::windowList()->end(),
window);
if (entry != TestShell::windowList()->end()) {
TestShell::windowList()->erase(entry);
return true;
}
return false;
}
void TestShell::TestTimedOut() {
puts("#TEST_TIMED_OUT\n");
TestFinished();
}
void TestShell::Show(WebNavigationPolicy policy) {
delegate_->show(policy);
}
void TestShell::BindJSObjectsToWindow(WebFrame* frame) {
// Only bind the test classes if we're running tests.
if (layout_test_mode_) {
accessibility_controller_->BindToJavascript(
frame, L"accessibilityController");
layout_test_controller_->BindToJavascript(frame, L"layoutTestController");
event_sending_controller_->BindToJavascript(frame, L"eventSender");
plain_text_controller_->BindToJavascript(frame, L"plainText");
text_input_controller_->BindToJavascript(frame, L"textInputController");
}
}
void TestShell::DumpBackForwardEntry(int index, std::wstring* result) {
int current_index = navigation_controller_->GetLastCommittedEntryIndex();
std::string content_state =
navigation_controller_->GetEntryAtIndex(index)->GetContentState();
if (content_state.empty()) {
content_state = webkit_glue::CreateHistoryStateForURL(
navigation_controller_->GetEntryAtIndex(index)->GetURL());
}
result->append(
webkit_glue::DumpHistoryState(content_state, 8, index == current_index));
}
void TestShell::DumpBackForwardList(std::wstring* result) {
result->append(L"\n============== Back Forward List ==============\n");
for (int i = 0; i < navigation_controller_->GetEntryCount(); ++i)
DumpBackForwardEntry(i, result);
result->append(L"===============================================\n");
}
void TestShell::CallJSGC() {
webView()->mainFrame()->collectGarbage();
}
WebView* TestShell::CreateWebView() {
// If we're running layout tests, only open a new window if the test has
// called layoutTestController.setCanOpenWindows()
if (layout_test_mode_ && !layout_test_controller_->CanOpenWindows())
return NULL;
TestShell* new_win;
if (!CreateNewWindow(GURL(), &new_win))
return NULL;
return new_win->webView();
}
bool TestShell::IsSVGTestURL(const GURL& url) {
return url.is_valid() && url.spec().find("W3C-SVG-1.1") != std::string::npos;
}
void TestShell::SizeToSVG() {
SizeTo(kSVGTestWindowWidth, kSVGTestWindowHeight);
}
void TestShell::SizeToDefault() {
SizeTo(kTestWindowWidth, kTestWindowHeight);
}
void TestShell::ResetTestController() {
accessibility_controller_->Reset();
layout_test_controller_->Reset();
event_sending_controller_->Reset();
delegate_->Reset();
}
void TestShell::LoadFile(const FilePath& file) {
LoadURLForFrame(net::FilePathToFileURL(file), std::wstring());
}
void TestShell::LoadURL(const GURL& url) {
// Used as a sentinal for run_webkit_tests.py to know when to start reading
// test output for this test and so we know we're not getting out of sync.
if (layout_test_mode_ && test_params())
printf("#URL:%s\n", test_params()->test_url.c_str());
LoadURLForFrame(url, std::wstring());
}
bool TestShell::Navigate(const TestNavigationEntry& entry, bool reload) {
// Get the right target frame for the entry.
WebFrame* frame = webView()->mainFrame();
if (!entry.GetTargetFrame().empty()) {
frame = webView()->findFrameByName(
WideToUTF16Hack(entry.GetTargetFrame()));
}
// TODO(mpcomplete): should we clear the target frame, or should
// back/forward navigations maintain the target frame?
// A navigation resulting from loading a javascript URL should not be
// treated as a browser initiated event. Instead, we want it to look as if
// the page initiated any load resulting from JS execution.
if (!entry.GetURL().SchemeIs("javascript")) {
delegate_->set_pending_extra_data(
new TestShellExtraData(entry.GetPageID()));
}
// If we are reloading, then WebKit will use the state of the current page.
// Otherwise, we give it the state to navigate to.
if (reload) {
frame->reload();
} else if (!entry.GetContentState().empty()) {
DCHECK_NE(entry.GetPageID(), -1);
frame->loadHistoryItem(
webkit_glue::HistoryItemFromString(entry.GetContentState()));
} else {
DCHECK_EQ(entry.GetPageID(), -1);
frame->loadRequest(WebURLRequest(entry.GetURL()));
}
// In case LoadRequest failed before DidCreateDataSource was called.
delegate_->set_pending_extra_data(NULL);
// Restore focus to the main frame prior to loading new request.
// This makes sure that we don't have a focused iframe. Otherwise, that
// iframe would keep focus when the SetFocus called immediately after
// LoadRequest, thus making some tests fail (see http://b/issue?id=845337
// for more details).
webView()->setFocusedFrame(frame);
SetFocus(webViewHost(), true);
return true;
}
void TestShell::GoBackOrForward(int offset) {
navigation_controller_->GoToOffset(offset);
}
void TestShell::DumpDocumentText() {
std::wstring file_path;
if (!PromptForSaveFile(L"Dump document text", &file_path))
return;
const std::string data =
WideToUTF8(webkit_glue::DumpDocumentText(webView()->mainFrame()));
file_util::WriteFile(file_path, data.c_str(), data.length());
}
void TestShell::DumpRenderTree() {
std::wstring file_path;
if (!PromptForSaveFile(L"Dump render tree", &file_path))
return;
const std::string data =
WideToUTF8(webkit_glue::DumpRenderer(webView()->mainFrame()));
file_util::WriteFile(file_path, data.c_str(), data.length());
}
std::wstring TestShell::GetDocumentText() {
return webkit_glue::DumpDocumentText(webView()->mainFrame());
}
void TestShell::Reload() {
navigation_controller_->Reload();
}
void TestShell::SetFocus(WebWidgetHost* host, bool enable) {
if (!layout_test_mode_) {
InteractiveSetFocus(host, enable);
} else {
// Simulate the effects of InteractiveSetFocus(), which includes calling
// both setFocus() and setIsActive().
if (enable) {
if (m_focusedWidgetHost != host) {
if (m_focusedWidgetHost)
m_focusedWidgetHost->webwidget()->setFocus(false);
webView()->setIsActive(enable);
host->webwidget()->setFocus(enable);
m_focusedWidgetHost = host;
}
} else {
if (m_focusedWidgetHost == host) {
host->webwidget()->setFocus(enable);
webView()->setIsActive(enable);
m_focusedWidgetHost = NULL;
}
}
}
}
//-----------------------------------------------------------------------------
namespace webkit_glue {
void PrecacheUrl(const char16* url, int url_length) {}
void AppendToLog(const char* file, int line, const char* msg) {
logging::LogMessage(file, line).stream() << msg;
}
bool GetApplicationDirectory(FilePath* path) {
return PathService::Get(base::DIR_EXE, path);
}
bool GetExeDirectory(FilePath* path) {
return GetApplicationDirectory(path);
}
bool IsPluginRunningInRendererProcess() {
return true;
}
bool GetPluginFinderURL(std::string* plugin_finder_url) {
return false;
}
bool IsDefaultPluginEnabled() {
#if defined(OS_WIN)
FilePath exe_path;
if (PathService::Get(base::FILE_EXE, &exe_path)) {
std::wstring exe_name = file_util::GetFilenameFromPath(
exe_path.ToWStringHack());
if (StartsWith(exe_name, L"test_shell_tests", false))
return true;
}
#endif // OS_WIN
return false;
}
bool IsProtocolSupportedForMedia(const GURL& url) {
if (url.SchemeIsFile() ||
url.SchemeIs("http") ||
url.SchemeIs("https") ||
url.SchemeIs("data"))
return true;
return false;
}
std::wstring GetWebKitLocale() {
return L"en-US";
}
void CloseIdleConnections() {
// Used in benchmarking, Ignored for test_shell.
}
void SetCacheMode(bool enabled) {
// Used in benchmarking, Ignored for test_shell.
}
} // namespace webkit_glue