298 lines
12 KiB
C++
298 lines
12 KiB
C++
/*
|
|
* 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;
|
|
};
|