// Copyright (c) 2006-2008 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 "v8_proxy.h"
#undef LOG
#include "webkit/tools/test_shell/test_shell.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/png_encoder.h"
#include "base/gfx/size.h"
#include "base/icu_util.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/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 "webkit/glue/webdatasource.h"
#include "webkit/glue/webframe.h"
#include "webkit/glue/webkit_glue.h"
#include "webkit/glue/webpreferences.h"
#include "webkit/glue/webresponse.h"
#include "webkit/glue/weburlrequest.h"
#include "webkit/glue/webview.h"
#include "webkit/glue/webwidget.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"
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:
virtual ~URLRequestTestShellFileJob() { }
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) {
}
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;
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),
test_is_preparing_(false),
test_is_pending_(false),
is_modal_(false),
dump_stats_table_on_exit_(false) {
delegate_ = new TestWebViewDelegate(this);
layout_test_controller_.reset(new LayoutTestController(this));
event_sending_controller_.reset(new EventSendingController(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() {
// Navigate to an empty page to fire all the destruction logic for the
// current page.
LoadURL(L"about:blank");
// Call GC twice to clean up garbage.
CallJSGC();
CallJSGC();
webView()->SetDelegate(NULL);
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 std::wstring& startingURL,
TestShell** result) {
TestShell* shell = new TestShell();
bool rv = shell->Initialize(startingURL);
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;
WebCore::V8Proxy::ProcessConsoleMessages();
// Echo the url in the output so we know we're not getting out of sync.
printf("#URL:%s\n", params->test_url.c_str());
// Dump the requested representation.
WebFrame* webFrame = shell->webView()->GetMainFrame();
if (webFrame) {
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
std::string mime_type =
webFrame->GetDataSource()->GetResponse().GetMimeType();
should_dump_as_text = (mime_type == "text/plain");
}
if (should_dump_as_text) {
bool recursive = shell->layout_test_controller_->
ShouldDumpChildFramesAsText();
std::string data_utf8 = WideToUTF8(
webkit_glue::DumpFramesAsText(webFrame, 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(webFrame)).c_str());
bool recursive = shell->layout_test_controller_->
ShouldDumpChildFrameScrollPositions();
printf("%s", WideToUTF8(
webkit_glue::DumpFrameScrollPosition(webFrame, 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;
std::string md5sum = DumpImage(webFrame, 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(WebFrame* web_frame,
const std::wstring& file_name, const std::string& pixel_hash) {
scoped_ptr device;
if (!web_frame->CaptureImage(&device, true))
return std::string();
const SkBitmap& src_bmp = device->accessBitmap(false);
// Encode image.
std::vector png;
SkAutoLockPixels src_bmp_lock(src_bmp);
PNGEncoder::ColorFormat color_format = PNGEncoder::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) {
PNGEncoder::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 = L"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;
// 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;
}
}
// 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::Show(WebView* webview, WindowOpenDisposition disposition) {
delegate_->Show(webview, disposition);
}
void TestShell::BindJSObjectsToWindow(WebFrame* frame) {
// Only bind the test classes if we're running tests.
if (layout_test_mode_) {
layout_test_controller_->BindToJavascript(frame, L"layoutTestController");
event_sending_controller_->BindToJavascript(frame, L"eventSender");
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()->GetMainFrame()->CallJSGC();
}
WebView* TestShell::CreateWebView(WebView* webview) {
// 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(std::wstring(), &new_win))
return NULL;
return new_win->webView();
}
void TestShell::SizeToSVG() {
SizeTo(kSVGTestWindowWidth, kSVGTestWindowHeight);
}
void TestShell::SizeToDefault() {
SizeTo(kTestWindowWidth, kTestWindowHeight);
}
void TestShell::ResetTestController() {
layout_test_controller_->Reset();
event_sending_controller_->Reset();
// Reset state in the test webview delegate.
delegate_ = new TestWebViewDelegate(this);
webView()->SetDelegate(delegate_);
}
void TestShell::LoadURL(const wchar_t* url) {
LoadURLForFrame(url, NULL);
}
bool TestShell::Navigate(const TestNavigationEntry& entry, bool reload) {
// Get the right target frame for the entry.
WebFrame* frame = webView()->GetMainFrame();
if (!entry.GetTargetFrame().empty())
frame = webView()->GetFrameWithName(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 && !entry.GetContentState().empty()) {
DCHECK(entry.GetPageID() != -1);
frame->LoadHistoryState(entry.GetContentState());
} else {
WebRequestCachePolicy cache_policy;
if (reload) {
cache_policy = WebRequestReloadIgnoringCacheData;
} else {
DCHECK(entry.GetPageID() == -1);
cache_policy = WebRequestUseProtocolCachePolicy;
}
scoped_ptr request(WebRequest::Create(entry.GetURL()));
request->SetCachePolicy(cache_policy);
frame->LoadRequest(request.get());
}
// 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()->GetMainFrame()));
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()->GetMainFrame()));
file_util::WriteFile(file_path, data.c_str(), data.length());
}
std::wstring TestShell::GetDocumentText() {
return webkit_glue::DumpDocumentText(webView()->GetMainFrame());
}
void TestShell::Reload() {
navigation_controller_->Reload();
}
void TestShell::SetFocus(WebWidgetHost* host, bool enable) {
if (!layout_test_mode_) {
InteractiveSetFocus(host, enable);
} else {
if (enable) {
if (m_focusedWidgetHost != host) {
if (m_focusedWidgetHost)
m_focusedWidgetHost->webwidget()->SetFocus(false);
host->webwidget()->SetFocus(enable);
m_focusedWidgetHost = host;
}
} else {
if (m_focusedWidgetHost == host) {
host->webwidget()->SetFocus(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(std::wstring *path) {
bool r;
FilePath fp;
r = PathService::Get(base::DIR_EXE, &fp);
*path = fp.ToWStringHack();
return r;
}
GURL GetInspectorURL() {
return GURL("test-shell-resource://inspector/inspector.html");
}
std::string GetUIResourceProtocol() {
return "test-shell-resource";
}
bool GetExeDirectory(std::wstring *path) {
return GetApplicationDirectory(path);
}
bool SpellCheckWord(const wchar_t* word, int word_len,
int* misspelling_start, int* misspelling_len) {
// Report all words being correctly spelled.
*misspelling_start = 0;
*misspelling_len = 0;
return true;
}
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;
}
std::wstring GetWebKitLocale() {
return L"en-US";
}
} // namespace webkit_glue