/* * Copyright (C) 2007 Apple Inc. All rights reserved. * Copyright (C) 2007 Eric Seidel <eric@webkit.org> * Copyright (C) 2008 Nuanti Ltd. * Copyright (C) 2009 Jan Michael Alonzo <jmalonzo@gmail.com> * Copyright (C) 2009 Collabora Ltd. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of * its contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "config.h" #include "LayoutTestController.h" #include "DumpRenderTree.h" #include "WorkQueue.h" #include "WorkQueueItem.h" #include <JavaScriptCore/JSRetainPtr.h> #include <JavaScriptCore/JSStringRef.h> #include <iostream> #include <sstream> #include <stdio.h> #include <glib.h> #include <libsoup/soup.h> #include <webkit/webkit.h> extern "C" { bool webkit_web_frame_pause_animation(WebKitWebFrame* frame, const gchar* name, double time, const gchar* element); bool webkit_web_frame_pause_transition(WebKitWebFrame* frame, const gchar* name, double time, const gchar* element); bool webkit_web_frame_pause_svg_animation(WebKitWebFrame* frame, const gchar* name, double time, const gchar* element); unsigned int webkit_web_frame_number_of_active_animations(WebKitWebFrame* frame); void webkit_application_cache_set_maximum_size(unsigned long long size); unsigned int webkit_worker_thread_count(void); void webkit_white_list_access_from_origin(const gchar* sourceOrigin, const gchar* destinationProtocol, const gchar* destinationHost, bool allowDestinationSubdomains); gchar* webkit_web_frame_counter_value_for_element_by_id(WebKitWebFrame* frame, const gchar* id); int webkit_web_frame_page_number_for_element_by_id(WebKitWebFrame* frame, const gchar* id, float pageWidth, float pageHeight); void webkit_web_inspector_execute_script(WebKitWebInspector* inspector, long callId, const gchar* script); } static gchar* copyWebSettingKey(gchar* preferenceKey) { static GHashTable* keyTable; if (!keyTable) { // If you add a pref here, make sure you reset the value in // DumpRenderTree::resetWebViewToConsistentStateBeforeTesting. keyTable = g_hash_table_new(g_str_hash, g_str_equal); g_hash_table_insert(keyTable, g_strdup("WebKitJavaScriptEnabled"), g_strdup("enable-scripts")); g_hash_table_insert(keyTable, g_strdup("WebKitDefaultFontSize"), g_strdup("default-font-size")); g_hash_table_insert(keyTable, g_strdup("WebKitEnableCaretBrowsing"), g_strdup("enable-caret-browsing")); g_hash_table_insert(keyTable, g_strdup("WebKitUsesPageCachePreferenceKey"), g_strdup("enable-page-cache")); } return g_strdup(static_cast<gchar*>(g_hash_table_lookup(keyTable, preferenceKey))); } LayoutTestController::~LayoutTestController() { // FIXME: implement } void LayoutTestController::addDisallowedURL(JSStringRef url) { // FIXME: implement } void LayoutTestController::clearBackForwardList() { WebKitWebView* webView = webkit_web_frame_get_web_view(mainFrame); WebKitWebBackForwardList* list = webkit_web_view_get_back_forward_list(webView); WebKitWebHistoryItem* item = webkit_web_back_forward_list_get_current_item(list); g_object_ref(item); // We clear the history by setting the back/forward list's capacity to 0 // then restoring it back and adding back the current item. gint limit = webkit_web_back_forward_list_get_limit(list); webkit_web_back_forward_list_set_limit(list, 0); webkit_web_back_forward_list_set_limit(list, limit); webkit_web_back_forward_list_add_item(list, item); webkit_web_back_forward_list_go_to_item(list, item); g_object_unref(item); } JSStringRef LayoutTestController::copyDecodedHostName(JSStringRef name) { // FIXME: implement return 0; } JSStringRef LayoutTestController::copyEncodedHostName(JSStringRef name) { // FIXME: implement return 0; } void LayoutTestController::dispatchPendingLoadRequests() { // FIXME: Implement for testing fix for 6727495 } void LayoutTestController::display() { displayWebView(); } JSRetainPtr<JSStringRef> LayoutTestController::counterValueForElementById(JSStringRef id) { gchar* idGChar = JSStringCopyUTF8CString(id); gchar* counterValueGChar = webkit_web_frame_counter_value_for_element_by_id(mainFrame, idGChar); g_free(idGChar); if (!counterValueGChar) return 0; JSRetainPtr<JSStringRef> counterValue(Adopt, JSStringCreateWithUTF8CString(counterValueGChar)); return counterValue; } void LayoutTestController::keepWebHistory() { // FIXME: implement } int LayoutTestController::pageNumberForElementById(JSStringRef id, float pageWidth, float pageHeight) { gchar* idGChar = JSStringCopyUTF8CString(id); int pageNumber = webkit_web_frame_page_number_for_element_by_id(mainFrame, idGChar, pageWidth, pageHeight); g_free(idGChar); return pageNumber; } int LayoutTestController::numberOfPages(float, float) { // FIXME: implement return -1; } size_t LayoutTestController::webHistoryItemCount() { // FIXME: implement return 0; } unsigned LayoutTestController::workerThreadCount() const { return webkit_worker_thread_count(); } void LayoutTestController::notifyDone() { if (m_waitToDump && !topLoadingFrame && !WorkQueue::shared()->count()) dump(); m_waitToDump = false; waitForPolicy = false; } JSStringRef LayoutTestController::pathToLocalResource(JSContextRef context, JSStringRef url) { // Function introduced in r28690. This may need special-casing on Windows. return JSStringRetain(url); // Do nothing on Unix. } void LayoutTestController::queueLoad(JSStringRef url, JSStringRef target) { gchar* relativeURL = JSStringCopyUTF8CString(url); SoupURI* baseURI = soup_uri_new(webkit_web_frame_get_uri(mainFrame)); SoupURI* absoluteURI = soup_uri_new_with_base(baseURI, relativeURL); soup_uri_free(baseURI); g_free(relativeURL); gchar* absoluteCString; if (absoluteURI) { absoluteCString = soup_uri_to_string(absoluteURI, FALSE); soup_uri_free(absoluteURI); } else absoluteCString = JSStringCopyUTF8CString(url); JSRetainPtr<JSStringRef> absoluteURL(Adopt, JSStringCreateWithUTF8CString(absoluteCString)); g_free(absoluteCString); WorkQueue::shared()->queue(new LoadItem(absoluteURL.get(), target)); } void LayoutTestController::setAcceptsEditing(bool acceptsEditing) { WebKitWebView* webView = webkit_web_frame_get_web_view(mainFrame); webkit_web_view_set_editable(webView, acceptsEditing); } void LayoutTestController::setAlwaysAcceptCookies(bool alwaysAcceptCookies) { // FIXME: Implement this (and restore the default value before running each test in DumpRenderTree.cpp). } void LayoutTestController::setCustomPolicyDelegate(bool setDelegate, bool permissive) { // FIXME: implement } void LayoutTestController::waitForPolicyDelegate() { waitForPolicy = true; setWaitToDump(true); } void LayoutTestController::whiteListAccessFromOrigin(JSStringRef sourceOrigin, JSStringRef protocol, JSStringRef host, bool includeSubdomains) { gchar* sourceOriginGChar = JSStringCopyUTF8CString(sourceOrigin); gchar* protocolGChar = JSStringCopyUTF8CString(protocol); gchar* hostGChar = JSStringCopyUTF8CString(host); webkit_white_list_access_from_origin(sourceOriginGChar, protocolGChar, hostGChar, includeSubdomains); g_free(sourceOriginGChar); g_free(protocolGChar); g_free(hostGChar); } void LayoutTestController::setMainFrameIsFirstResponder(bool flag) { // FIXME: implement } void LayoutTestController::setTabKeyCyclesThroughElements(bool cycles) { WebKitWebView* webView = webkit_web_frame_get_web_view(mainFrame); WebKitWebSettings* settings = webkit_web_view_get_settings(webView); g_object_set(G_OBJECT(settings), "tab-key-cycles-through-elements", cycles, NULL); } void LayoutTestController::setTimelineProfilingEnabled(bool flag) { WebKitWebView* view = webkit_web_frame_get_web_view(mainFrame); ASSERT(view); WebKitWebInspector* inspector = webkit_web_view_get_inspector(view); g_object_set(G_OBJECT(inspector), "timeline-profiling-enabled", flag, NULL); } void LayoutTestController::setUseDashboardCompatibilityMode(bool flag) { // FIXME: implement } static gchar* userStyleSheet = NULL; static gboolean userStyleSheetEnabled = TRUE; void LayoutTestController::setUserStyleSheetEnabled(bool flag) { userStyleSheetEnabled = flag; WebKitWebView* webView = webkit_web_frame_get_web_view(mainFrame); WebKitWebSettings* settings = webkit_web_view_get_settings(webView); if (flag && userStyleSheet) g_object_set(G_OBJECT(settings), "user-stylesheet-uri", userStyleSheet, NULL); else g_object_set(G_OBJECT(settings), "user-stylesheet-uri", "", NULL); } void LayoutTestController::setUserStyleSheetLocation(JSStringRef path) { g_free(userStyleSheet); userStyleSheet = JSStringCopyUTF8CString(path); if (userStyleSheetEnabled) setUserStyleSheetEnabled(true); } void LayoutTestController::setWindowIsKey(bool windowIsKey) { // FIXME: implement } void LayoutTestController::setSmartInsertDeleteEnabled(bool flag) { // FIXME: implement } static gboolean waitToDumpWatchdogFired(void*) { waitToDumpWatchdog = 0; gLayoutTestController->waitToDumpWatchdogTimerFired(); return FALSE; } void LayoutTestController::setWaitToDump(bool waitUntilDone) { static const int timeoutSeconds = 15; m_waitToDump = waitUntilDone; if (m_waitToDump && !waitToDumpWatchdog) waitToDumpWatchdog = g_timeout_add_seconds(timeoutSeconds, waitToDumpWatchdogFired, 0); } int LayoutTestController::windowCount() { // +1 -> including the main view return g_slist_length(webViewList) + 1; } void LayoutTestController::setPrivateBrowsingEnabled(bool flag) { WebKitWebView* view = webkit_web_frame_get_web_view(mainFrame); ASSERT(view); WebKitWebSettings* settings = webkit_web_view_get_settings(view); g_object_set(G_OBJECT(settings), "enable-private-browsing", flag, NULL); } void LayoutTestController::setXSSAuditorEnabled(bool flag) { WebKitWebView* view = webkit_web_frame_get_web_view(mainFrame); ASSERT(view); WebKitWebSettings* settings = webkit_web_view_get_settings(view); g_object_set(G_OBJECT(settings), "enable-xss-auditor", flag, NULL); } void LayoutTestController::setFrameSetFlatteningEnabled(bool flag) { // FIXME: implement } void LayoutTestController::setAllowUniversalAccessFromFileURLs(bool flag) { WebKitWebView* view = webkit_web_frame_get_web_view(mainFrame); ASSERT(view); WebKitWebSettings* settings = webkit_web_view_get_settings(view); g_object_set(G_OBJECT(settings), "enable-universal-access-from-file-uris", flag, NULL); } void LayoutTestController::setAuthorAndUserStylesEnabled(bool flag) { // FIXME: implement } void LayoutTestController::disableImageLoading() { // FIXME: Implement for testing fix for https://bugs.webkit.org/show_bug.cgi?id=27896 // Also need to make sure image loading is re-enabled for each new test. } void LayoutTestController::setMockGeolocationPosition(double latitude, double longitude, double accuracy) { // FIXME: Implement for Geolocation layout tests. // See https://bugs.webkit.org/show_bug.cgi?id=28264. } void LayoutTestController::setMockGeolocationError(int code, JSStringRef message) { // FIXME: Implement for Geolocation layout tests. // See https://bugs.webkit.org/show_bug.cgi?id=28264. } void LayoutTestController::setIconDatabaseEnabled(bool flag) { // FIXME: implement } void LayoutTestController::setJavaScriptProfilingEnabled(bool flag) { WebKitWebView* view = webkit_web_frame_get_web_view(mainFrame); ASSERT(view); WebKitWebSettings* settings = webkit_web_view_get_settings(view); g_object_set(G_OBJECT(settings), "enable-developer-extras", flag, NULL); WebKitWebInspector* inspector = webkit_web_view_get_inspector(view); g_object_set(G_OBJECT(inspector), "javascript-profiling-enabled", flag, NULL); } void LayoutTestController::setSelectTrailingWhitespaceEnabled(bool flag) { // FIXME: implement } void LayoutTestController::setPopupBlockingEnabled(bool flag) { WebKitWebView* view = webkit_web_frame_get_web_view(mainFrame); ASSERT(view); WebKitWebSettings* settings = webkit_web_view_get_settings(view); g_object_set(G_OBJECT(settings), "javascript-can-open-windows-automatically", !flag, NULL); } bool LayoutTestController::elementDoesAutoCompleteForElementWithId(JSStringRef id) { // FIXME: implement return false; } void LayoutTestController::execCommand(JSStringRef name, JSStringRef value) { // FIXME: implement } void LayoutTestController::setCacheModel(int) { // FIXME: implement } bool LayoutTestController::isCommandEnabled(JSStringRef /*name*/) { // FIXME: implement return false; } void LayoutTestController::setPersistentUserStyleSheetLocation(JSStringRef jsURL) { // FIXME: implement } void LayoutTestController::clearPersistentUserStyleSheet() { // FIXME: implement } void LayoutTestController::clearAllDatabases() { webkit_remove_all_web_databases(); } void LayoutTestController::setDatabaseQuota(unsigned long long quota) { WebKitSecurityOrigin* origin = webkit_web_frame_get_security_origin(mainFrame); webkit_security_origin_set_web_database_quota(origin, quota); } void LayoutTestController::setDomainRelaxationForbiddenForURLScheme(bool, JSStringRef) { // FIXME: implement } void LayoutTestController::setAppCacheMaximumSize(unsigned long long size) { webkit_application_cache_set_maximum_size(size); } bool LayoutTestController::pauseAnimationAtTimeOnElementWithId(JSStringRef animationName, double time, JSStringRef elementId) { gchar* name = JSStringCopyUTF8CString(animationName); gchar* element = JSStringCopyUTF8CString(elementId); bool returnValue = webkit_web_frame_pause_animation(mainFrame, name, time, element); g_free(name); g_free(element); return returnValue; } bool LayoutTestController::pauseTransitionAtTimeOnElementWithId(JSStringRef propertyName, double time, JSStringRef elementId) { gchar* name = JSStringCopyUTF8CString(propertyName); gchar* element = JSStringCopyUTF8CString(elementId); bool returnValue = webkit_web_frame_pause_transition(mainFrame, name, time, element); g_free(name); g_free(element); return returnValue; } bool LayoutTestController::sampleSVGAnimationForElementAtTime(JSStringRef animationId, double time, JSStringRef elementId) { gchar* name = JSStringCopyUTF8CString(animationId); gchar* element = JSStringCopyUTF8CString(elementId); bool returnValue = webkit_web_frame_pause_svg_animation(mainFrame, name, time, element); g_free(name); g_free(element); return returnValue; } unsigned LayoutTestController::numberOfActiveAnimations() const { return webkit_web_frame_number_of_active_animations(mainFrame); } void LayoutTestController::overridePreference(JSStringRef key, JSStringRef value) { gchar* name = JSStringCopyUTF8CString(key); gchar* strValue = JSStringCopyUTF8CString(value); WebKitWebView* view = webkit_web_frame_get_web_view(mainFrame); ASSERT(view); WebKitWebSettings* settings = webkit_web_view_get_settings(view); gchar* webSettingKey = copyWebSettingKey(name); if (webSettingKey) { GValue stringValue = { 0, { { 0 } } }; g_value_init(&stringValue, G_TYPE_STRING); g_value_set_string(&stringValue, const_cast<gchar*>(strValue)); WebKitWebSettingsClass* klass = WEBKIT_WEB_SETTINGS_GET_CLASS(settings); GParamSpec* pspec = g_object_class_find_property(G_OBJECT_CLASS(klass), webSettingKey); GValue propValue = { 0, { { 0 } } }; g_value_init(&propValue, pspec->value_type); if (g_value_type_transformable(G_TYPE_STRING, pspec->value_type)) { g_value_transform(const_cast<GValue*>(&stringValue), &propValue); g_object_set_property(G_OBJECT(settings), webSettingKey, const_cast<GValue*>(&propValue)); } else if (G_VALUE_HOLDS_BOOLEAN(&propValue)) { char* lowered = g_utf8_strdown(strValue, -1); g_object_set(G_OBJECT(settings), webSettingKey, g_str_equal(lowered, "true") || g_str_equal(strValue, "1"), NULL); g_free(lowered); } else if (G_VALUE_HOLDS_INT(&propValue)) { std::string str(strValue); std::stringstream ss(str); int val = 0; if (!(ss >> val).fail()) g_object_set(G_OBJECT(settings), webSettingKey, val, NULL); } else printf("LayoutTestController::overridePreference failed to override preference '%s'.\n", name); } g_free(webSettingKey); g_free(name); g_free(strValue); } void LayoutTestController::addUserScript(JSStringRef source, bool runAtStart) { printf("LayoutTestController::addUserScript not implemented.\n"); } void LayoutTestController::addUserStyleSheet(JSStringRef source) { printf("LayoutTestController::addUserStyleSheet not implemented.\n"); } void LayoutTestController::showWebInspector() { WebKitWebView* webView = webkit_web_frame_get_web_view(mainFrame); WebKitWebSettings* webSettings = webkit_web_view_get_settings(webView); WebKitWebInspector* inspector = webkit_web_view_get_inspector(webView); g_object_set(webSettings, "enable-developer-extras", TRUE, NULL); webkit_web_inspector_show(inspector); } void LayoutTestController::closeWebInspector() { WebKitWebView* webView = webkit_web_frame_get_web_view(mainFrame); WebKitWebSettings* webSettings = webkit_web_view_get_settings(webView); WebKitWebInspector* inspector = webkit_web_view_get_inspector(webView); webkit_web_inspector_close(inspector); g_object_set(webSettings, "enable-developer-extras", FALSE, NULL); } void LayoutTestController::evaluateInWebInspector(long callId, JSStringRef script) { WebKitWebView* webView = webkit_web_frame_get_web_view(mainFrame); WebKitWebInspector* inspector = webkit_web_view_get_inspector(webView); char* scriptString = JSStringCopyUTF8CString(script); webkit_web_inspector_execute_script(inspector, callId, scriptString); g_free(scriptString); } void LayoutTestController::evaluateScriptInIsolatedWorld(unsigned worldID, JSObjectRef globalObject, JSStringRef script) { // FIXME: Implement this. } void LayoutTestController::removeAllVisitedLinks() { // FIXME: Implement this. }