blob: 8c7900049a3ba2444cc46732f45a1376b79f7b4b [file] [log] [blame]
/*
* Copyright (C) 2011 Igalia S.L.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public License
* along with this library; see the file COPYING.LIB. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#pragma once
#include <cairo.h>
#include <glib-object.h>
#include <wtf/HashMap.h>
#include <wtf/HashSet.h>
#include <wtf/Vector.h>
#include <wtf/glib/GRefPtr.h>
#include <wtf/glib/GUniquePtr.h>
#include <wtf/text/CString.h>
#if PLATFORM(GTK)
#include <webkit2/webkit2.h>
#elif PLATFORM(WPE)
#include "HeadlessViewBackend.h"
#include <wpe/webkit.h>
#endif
#define TEST_PATH_FORMAT "/webkit/%s/%s"
#define MAKE_GLIB_TEST_FIXTURE(ClassName) \
static void setUp(ClassName* fixture, gconstpointer data) \
{ \
new (fixture) ClassName; \
} \
static void tearDown(ClassName* fixture, gconstpointer data) \
{ \
fixture->~ClassName(); \
} \
static void add(const char* suiteName, const char* testName, void (*testFunc)(ClassName*, const void*)) \
{ \
GUniquePtr<gchar> testPath(g_strdup_printf(TEST_PATH_FORMAT, suiteName, testName)); \
g_test_add(testPath.get(), ClassName, 0, ClassName::setUp, testFunc, ClassName::tearDown); \
}
#define MAKE_GLIB_TEST_FIXTURE_WITH_SETUP_TEARDOWN(ClassName, setup, teardown) \
static void setUp(ClassName* fixture, gconstpointer data) \
{ \
setup(); \
new (fixture) ClassName; \
} \
static void tearDown(ClassName* fixture, gconstpointer data) \
{ \
fixture->~ClassName(); \
teardown(); \
} \
static void add(const char* suiteName, const char* testName, void (*testFunc)(ClassName*, const void*)) \
{ \
GUniquePtr<gchar> testPath(g_strdup_printf(TEST_PATH_FORMAT, suiteName, testName)); \
g_test_add(testPath.get(), ClassName, 0, ClassName::setUp, testFunc, ClassName::tearDown); \
}
#define ASSERT_CMP_CSTRING(s1, cmp, s2) \
do { \
CString __s1 = (s1); \
CString __s2 = (s2); \
if (g_strcmp0(__s1.data(), __s2.data()) cmp 0) ; \
else { \
g_assertion_message_cmpstr(G_LOG_DOMAIN, __FILE__, __LINE__, \
G_STRFUNC, #s1 " " #cmp " " #s2, __s1.data(), #cmp, __s2.data()); \
} \
} while (0)
#if !defined(g_assert_cmpfloat_with_epsilon)
#define g_assert_cmpfloat_with_epsilon(n1,n2,epsilon) \
do { \
double __n1 = (n1); \
double __n2 = (n2); \
double __epsilon = (epsilon); \
if ((((__n1) > (__n2) ? (__n1) - (__n2) : (__n2) - (__n1)) < (__epsilon))) ; \
else { \
g_assertion_message_cmpnum (G_LOG_DOMAIN, __FILE__, __LINE__, \
G_STRFUNC, #n1 " == " #n2 " (+/- " #epsilon ")", __n1, "==", __n2, 'f'); \
} \
} while(0)
#endif
class Test {
public:
MAKE_GLIB_TEST_FIXTURE(Test);
static GRefPtr<WebKitWebView> adoptView(gpointer view)
{
g_assert_true(WEBKIT_IS_WEB_VIEW(view));
#if PLATFORM(GTK)
g_assert_true(g_object_is_floating(view));
return GRefPtr<WebKitWebView>(WEBKIT_WEB_VIEW(view));
#elif PLATFORM(WPE)
return adoptGRef(WEBKIT_WEB_VIEW(view));
#endif
}
static const char* dataDirectory();
static void initializeWebExtensionsCallback(WebKitWebContext* context, Test* test)
{
test->initializeWebExtensions();
}
Test()
{
GUniquePtr<char> localStorageDirectory(g_build_filename(dataDirectory(), "local-storage", nullptr));
GUniquePtr<char> indexedDBDirectory(g_build_filename(dataDirectory(), "indexeddb", nullptr));
GUniquePtr<char> diskCacheDirectory(g_build_filename(dataDirectory(), "disk-cache", nullptr));
GUniquePtr<char> applicationCacheDirectory(g_build_filename(dataDirectory(), "appcache", nullptr));
GUniquePtr<char> webSQLDirectory(g_build_filename(dataDirectory(), "websql", nullptr));
GUniquePtr<char> hstsDirectory(g_build_filename(dataDirectory(), "hsts", nullptr));
GUniquePtr<char> itpDirectory(g_build_filename(dataDirectory(), "itp", nullptr));
GUniquePtr<char> swRegistrationsDirectory(g_build_filename(dataDirectory(), "serviceworkers", nullptr));
GUniquePtr<char> domCacheDirectory(g_build_filename(dataDirectory(), "dom-cache", nullptr));
GRefPtr<WebKitWebsiteDataManager> websiteDataManager = adoptGRef(webkit_website_data_manager_new(
"local-storage-directory", localStorageDirectory.get(), "indexeddb-directory", indexedDBDirectory.get(),
"disk-cache-directory", diskCacheDirectory.get(), "offline-application-cache-directory", applicationCacheDirectory.get(),
"websql-directory", webSQLDirectory.get(), "hsts-cache-directory", hstsDirectory.get(),
"itp-directory", itpDirectory.get(), "service-worker-registrations-directory", swRegistrationsDirectory.get(),
"dom-cache-directory", domCacheDirectory.get(), nullptr));
m_webContext = adoptGRef(WEBKIT_WEB_CONTEXT(g_object_new(WEBKIT_TYPE_WEB_CONTEXT,
"website-data-manager", websiteDataManager.get(),
#if PLATFORM(GTK)
"process-swap-on-cross-site-navigation-enabled", TRUE,
#if !USE(GTK4)
"use-system-appearance-for-scrollbars", FALSE,
#endif
#endif
"memory-pressure-settings", s_memoryPressureSettings,
nullptr)));
assertObjectIsDeletedWhenTestFinishes(G_OBJECT(m_webContext.get()));
g_signal_connect(m_webContext.get(), "initialize-web-extensions", G_CALLBACK(initializeWebExtensionsCallback), this);
}
virtual ~Test()
{
g_signal_handlers_disconnect_matched(m_webContext.get(), G_SIGNAL_MATCH_DATA, 0, 0, nullptr, nullptr, this);
m_webContext = nullptr;
if (m_watchedObjects.isEmpty())
return;
g_print("Leaked objects:");
HashSet<GObject*>::const_iterator end = m_watchedObjects.end();
for (HashSet<GObject*>::const_iterator it = m_watchedObjects.begin(); it != end; ++it)
g_print(" %s(%p - %u left)", g_type_name_from_instance(reinterpret_cast<GTypeInstance*>(*it)), *it, (*it)->ref_count);
g_print("\n");
g_assert_true(m_watchedObjects.isEmpty());
}
virtual void initializeWebExtensions()
{
webkit_web_context_set_web_extensions_directory(m_webContext.get(), WEBKIT_TEST_WEB_EXTENSIONS_DIR);
webkit_web_context_set_web_extensions_initialization_user_data(m_webContext.get(),
g_variant_new("(ss)", g_dbus_server_get_guid(s_dbusServer.get()), g_dbus_server_get_client_address(s_dbusServer.get())));
}
#if PLATFORM(WPE)
static WebKitWebViewBackend* createWebViewBackend()
{
// Don't make warnings fatal when creating the backend, since atk produces warnings when a11y bus is not running.
removeLogFatalFlag(G_LOG_LEVEL_WARNING);
auto* headlessBackend = new WPEToolingBackends::HeadlessViewBackend(800, 600);
addLogFatalFlag(G_LOG_LEVEL_WARNING);
// Make the view initially hidden for consistency with GTK+ tests.
wpe_view_backend_remove_activity_state(headlessBackend->backend(), wpe_view_activity_state_visible | wpe_view_activity_state_focused);
return webkit_web_view_backend_new(headlessBackend->backend(), [](gpointer userData) {
delete static_cast<WPEToolingBackends::HeadlessViewBackend*>(userData);
}, headlessBackend);
}
#endif
static WebKitWebView* createWebView()
{
#if PLATFORM(GTK)
return WEBKIT_WEB_VIEW(webkit_web_view_new());
#elif PLATFORM(WPE)
return webkit_web_view_new(createWebViewBackend());
#endif
}
static WebKitWebView* createWebView(WebKitWebContext* context)
{
#if PLATFORM(GTK)
return WEBKIT_WEB_VIEW(webkit_web_view_new_with_context(context));
#elif PLATFORM(WPE)
return webkit_web_view_new_with_context(createWebViewBackend(), context);
#endif
}
static WebKitWebView* createWebView(WebKitWebView* relatedView)
{
#if PLATFORM(GTK)
return WEBKIT_WEB_VIEW(webkit_web_view_new_with_related_view(relatedView));
#elif PLATFORM(WPE)
return webkit_web_view_new_with_related_view(createWebViewBackend(), relatedView);
#endif
}
static WebKitWebView* createWebView(WebKitUserContentManager* contentManager)
{
#if PLATFORM(GTK)
return WEBKIT_WEB_VIEW(webkit_web_view_new_with_user_content_manager(contentManager));
#elif PLATFORM(WPE)
return webkit_web_view_new_with_user_content_manager(createWebViewBackend(), contentManager);
#endif
}
static WebKitWebView* createWebView(WebKitSettings* settings)
{
#if PLATFORM(GTK)
return WEBKIT_WEB_VIEW(webkit_web_view_new_with_settings(settings));
#elif PLATFORM(WPE)
return webkit_web_view_new_with_settings(createWebViewBackend(), settings);
#endif
}
static void objectFinalized(Test* test, GObject* finalizedObject)
{
test->m_watchedObjects.remove(finalizedObject);
}
void assertObjectIsDeletedWhenTestFinishes(GObject* object)
{
m_watchedObjects.add(object);
g_object_weak_ref(object, reinterpret_cast<GWeakNotify>(objectFinalized), this);
}
enum ResourcesDir {
WebKitGLibResources,
WebKit2Resources,
};
static CString getResourcesDir(ResourcesDir resourcesDir = WebKitGLibResources)
{
switch (resourcesDir) {
case WebKitGLibResources: {
GUniquePtr<char> resourcesDir(g_build_filename(WEBKIT_SRC_DIR, "Tools", "TestWebKitAPI", "Tests", "WebKitGLib", "resources", nullptr));
return resourcesDir.get();
}
case WebKit2Resources: {
GUniquePtr<char> resourcesDir(g_build_filename(WEBKIT_SRC_DIR, "Tools", "TestWebKitAPI", "Tests", "WebKit", nullptr));
return resourcesDir.get();
}
}
RELEASE_ASSERT_NOT_REACHED();
}
static void addLogFatalFlag(unsigned flag)
{
unsigned fatalMask = g_log_set_always_fatal(static_cast<GLogLevelFlags>(G_LOG_FATAL_MASK));
fatalMask |= flag;
g_log_set_always_fatal(static_cast<GLogLevelFlags>(fatalMask));
}
static void removeLogFatalFlag(unsigned flag)
{
unsigned fatalMask = g_log_set_always_fatal(static_cast<GLogLevelFlags>(G_LOG_FATAL_MASK));
fatalMask &= ~flag;
g_log_set_always_fatal(static_cast<GLogLevelFlags>(fatalMask));
}
static bool cairoSurfacesEqual(cairo_surface_t* s1, cairo_surface_t* s2)
{
return (cairo_image_surface_get_format(s1) == cairo_image_surface_get_format(s2)
&& cairo_image_surface_get_width(s1) == cairo_image_surface_get_width(s2)
&& cairo_image_surface_get_height(s1) == cairo_image_surface_get_height(s2)
&& cairo_image_surface_get_stride(s1) == cairo_image_surface_get_stride(s2)
&& !memcmp(const_cast<const void*>(reinterpret_cast<void*>(cairo_image_surface_get_data(s1))),
const_cast<const void*>(reinterpret_cast<void*>(cairo_image_surface_get_data(s2))),
cairo_image_surface_get_height(s1)*cairo_image_surface_get_stride(s1)));
}
HashSet<GObject*> m_watchedObjects;
GRefPtr<WebKitWebContext> m_webContext;
static GRefPtr<GDBusServer> s_dbusServer;
static Vector<GRefPtr<GDBusConnection>> s_dbusConnections;
static HashMap<uint64_t, GDBusConnection*> s_dbusConnectionPageMap;
static WebKitMemoryPressureSettings* s_memoryPressureSettings;
};