blob: 88c049f2786238ed5f9ea8cee077130a7017086f [file] [log] [blame] [edit]
/*
* Copyright (C) 2011, 2017 Igalia S.L.
* Portions Copyright (c) 2011 Motorola Mobility, Inc. All rights reserved.
*
* 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.
*/
#include "config.h"
#include "WebViewTest.h"
#include <JavaScriptCore/JSRetainPtr.h>
#include <WebKitWebViewInternal.h>
bool WebViewTest::shouldInitializeWebViewInConstructor = true;
bool WebViewTest::shouldCreateEphemeralWebView = false;
WebViewTest::WebViewTest()
: m_userContentManager(adoptGRef(webkit_user_content_manager_new()))
, m_mainLoop(g_main_loop_new(nullptr, TRUE))
{
if (shouldInitializeWebViewInConstructor)
initializeWebView();
assertObjectIsDeletedWhenTestFinishes(G_OBJECT(m_userContentManager.get()));
}
WebViewTest::~WebViewTest()
{
platformDestroy();
s_dbusConnectionPageMap.remove(webkit_web_view_get_page_id(m_webView.get()));
g_main_loop_unref(m_mainLoop);
}
void WebViewTest::initializeWebView()
{
g_assert_null(m_webView.get());
#if ENABLE(2022_GLIB_API)
GRefPtr<WebKitNetworkSession> networkSession = shouldCreateEphemeralWebView ? adoptGRef(webkit_network_session_new_ephemeral()) : m_networkSession;
#endif
m_webView = createWebView(
"user-content-manager", m_userContentManager.get(),
#if ENABLE(2022_GLIB_API)
"network-session", networkSession.get(),
#else
"is-ephemeral", shouldCreateEphemeralWebView,
#endif
nullptr);
assertObjectIsDeletedWhenTestFinishes(G_OBJECT(m_webView.get()));
#if ENABLE(WPE_PLATFORM)
if (auto* view = webkit_web_view_get_wpe_view(m_webView.get()))
assertObjectIsDeletedWhenTestFinishes(G_OBJECT(view));
#endif
g_signal_connect(m_webView.get(), "web-process-terminated", G_CALLBACK(WebViewTest::webProcessTerminated), this);
}
gboolean WebViewTest::webProcessTerminated(WebKitWebView*, WebKitWebProcessTerminationReason, WebViewTest* test)
{
if (test->m_expectedWebProcessCrash) {
test->m_expectedWebProcessCrash = false;
return FALSE;
}
g_assert_not_reached();
return TRUE;
}
void WebViewTest::loadURI(const char* uri)
{
m_activeURI = uri;
webkit_web_view_load_uri(m_webView.get(), uri);
g_assert_true(webkit_web_view_is_loading(m_webView.get()));
g_assert_cmpstr(webkit_web_view_get_uri(m_webView.get()), ==, m_activeURI.data());
}
void WebViewTest::loadHtml(const char* html, const char* baseURI, WebKitWebView* webView)
{
if (!baseURI)
m_activeURI = "about:blank";
else
m_activeURI = baseURI;
if (!webView)
webView = m_webView.get();
webkit_web_view_load_html(webView, html, baseURI);
g_assert_true(webkit_web_view_is_loading(webView));
g_assert_cmpstr(webkit_web_view_get_uri(webView), ==, m_activeURI.data());
}
void WebViewTest::loadPlainText(const char* plainText)
{
m_activeURI = "about:blank";
webkit_web_view_load_plain_text(m_webView.get(), plainText);
g_assert_true(webkit_web_view_is_loading(m_webView.get()));
g_assert_cmpstr(webkit_web_view_get_uri(m_webView.get()), ==, m_activeURI.data());
}
void WebViewTest::loadBytes(GBytes* bytes, const char* mimeType, const char* encoding, const char* baseURI)
{
if (!baseURI)
m_activeURI = "about:blank";
else
m_activeURI = baseURI;
webkit_web_view_load_bytes(m_webView.get(), bytes, mimeType, encoding, baseURI);
g_assert_true(webkit_web_view_is_loading(m_webView.get()));
g_assert_cmpstr(webkit_web_view_get_uri(m_webView.get()), ==, m_activeURI.data());
}
void WebViewTest::loadRequest(WebKitURIRequest* request)
{
m_activeURI = webkit_uri_request_get_uri(request);
webkit_web_view_load_request(m_webView.get(), request);
g_assert_true(webkit_web_view_is_loading(m_webView.get()));
g_assert_cmpstr(webkit_web_view_get_uri(m_webView.get()), ==, m_activeURI.data());
}
void WebViewTest::loadAlternateHTML(const char* html, const char* contentURI, const char* baseURI)
{
m_activeURI = contentURI;
webkit_web_view_load_alternate_html(m_webView.get(), html, contentURI, baseURI);
g_assert_true(webkit_web_view_is_loading(m_webView.get()));
g_assert_cmpstr(webkit_web_view_get_uri(m_webView.get()), ==, m_activeURI.data());
}
void WebViewTest::goBack()
{
bool canGoBack = webkit_web_view_can_go_back(m_webView.get());
if (canGoBack) {
WebKitBackForwardList* list = webkit_web_view_get_back_forward_list(m_webView.get());
WebKitBackForwardListItem* item = webkit_back_forward_list_get_nth_item(list, -1);
m_activeURI = webkit_back_forward_list_item_get_original_uri(item);
}
// Call go_back even when can_go_back returns FALSE to check nothing happens.
webkit_web_view_go_back(m_webView.get());
if (canGoBack) {
g_assert_true(webkit_web_view_is_loading(m_webView.get()));
g_assert_cmpstr(webkit_web_view_get_uri(m_webView.get()), ==, m_activeURI.data());
}
}
void WebViewTest::goForward()
{
bool canGoForward = webkit_web_view_can_go_forward(m_webView.get());
if (canGoForward) {
WebKitBackForwardList* list = webkit_web_view_get_back_forward_list(m_webView.get());
WebKitBackForwardListItem* item = webkit_back_forward_list_get_nth_item(list, 1);
m_activeURI = webkit_back_forward_list_item_get_original_uri(item);
}
// Call go_forward even when can_go_forward returns FALSE to check nothing happens.
webkit_web_view_go_forward(m_webView.get());
if (canGoForward) {
g_assert_true(webkit_web_view_is_loading(m_webView.get()));
g_assert_cmpstr(webkit_web_view_get_uri(m_webView.get()), ==, m_activeURI.data());
}
}
void WebViewTest::goToBackForwardListItem(WebKitBackForwardListItem* item)
{
m_activeURI = webkit_back_forward_list_item_get_original_uri(item);
webkit_web_view_go_to_back_forward_list_item(m_webView.get(), item);
g_assert_true(webkit_web_view_is_loading(m_webView.get()));
g_assert_cmpstr(webkit_web_view_get_uri(m_webView.get()), ==, m_activeURI.data());
}
void WebViewTest::quitMainLoop()
{
g_main_loop_quit(m_mainLoop);
}
void WebViewTest::wait(double seconds)
{
g_timeout_add(seconds * 1000, [](gpointer userData) -> gboolean {
static_cast<WebViewTest*>(userData)->quitMainLoop();
return G_SOURCE_REMOVE;
}, this);
g_main_loop_run(m_mainLoop);
}
static void loadChanged(WebKitWebView* webView, WebKitLoadEvent loadEvent, WebViewTest* test)
{
if (loadEvent != WEBKIT_LOAD_FINISHED)
return;
g_signal_handlers_disconnect_by_func(webView, reinterpret_cast<void*>(loadChanged), test);
g_main_loop_quit(test->m_mainLoop);
}
void WebViewTest::waitUntilLoadFinished(WebKitWebView* webView)
{
if (!webView)
webView = m_webView.get();
g_signal_connect(webView, "load-changed", G_CALLBACK(loadChanged), this);
g_main_loop_run(m_mainLoop);
}
static void titleChanged(WebKitWebView* webView, GParamSpec*, WebViewTest* test)
{
if (!test->m_expectedTitle.isNull() && test->m_expectedTitle != webkit_web_view_get_title(webView))
return;
g_signal_handlers_disconnect_by_func(webView, reinterpret_cast<void*>(titleChanged), test);
g_main_loop_quit(test->m_mainLoop);
}
void WebViewTest::waitUntilTitleChangedTo(const char* expectedTitle)
{
if (expectedTitle && !g_strcmp0(expectedTitle, webkit_web_view_get_title(m_webView.get())))
return;
m_expectedTitle = expectedTitle;
g_signal_connect(m_webView.get(), "notify::title", G_CALLBACK(titleChanged), this);
g_main_loop_run(m_mainLoop);
m_expectedTitle = CString();
}
void WebViewTest::waitUntilTitleChanged()
{
waitUntilTitleChangedTo(nullptr);
}
static void isWebProcessResponsiveChanged(WebKitWebView* webView, GParamSpec*, WebViewTest* test)
{
g_signal_handlers_disconnect_by_func(webView, reinterpret_cast<void*>(isWebProcessResponsiveChanged), test);
g_main_loop_quit(test->m_mainLoop);
}
void WebViewTest::waitUntilIsWebProcessResponsiveChanged()
{
g_signal_connect(m_webView.get(), "notify::is-web-process-responsive", G_CALLBACK(isWebProcessResponsiveChanged), this);
g_main_loop_run(m_mainLoop);
}
void WebViewTest::selectAll()
{
webkit_web_view_execute_editing_command(m_webView.get(), "SelectAll");
}
bool WebViewTest::isEditable()
{
return webkit_web_view_is_editable(m_webView.get());
}
void WebViewTest::setEditable(bool editable)
{
webkit_web_view_set_editable(m_webView.get(), editable);
}
void WebViewTest::assertFileIsCreated(const char *filename)
{
constexpr double intervalInSeconds = 0.25;
unsigned tries = 4;
while (!g_file_test(filename, G_FILE_TEST_EXISTS) && --tries)
wait(intervalInSeconds);
g_assert_true(g_file_test(filename, G_FILE_TEST_EXISTS));
}
void WebViewTest::assertFileIsNotCreated(const char* filename)
{
constexpr double intervalInSeconds = 0.25;
unsigned tries = 4;
while (!g_file_test(filename, G_FILE_TEST_EXISTS) && --tries)
wait(intervalInSeconds);
g_assert_false(g_file_test(filename, G_FILE_TEST_EXISTS));
}
void WebViewTest::assertJavaScriptBecomesTrue(const char* javascript)
{
unsigned triesCount = 4;
bool becameTrue = false;
while (!becameTrue && triesCount--) {
auto jsValue = runJavaScriptAndWaitUntilFinished(javascript, nullptr);
if (jsc_value_is_boolean(jsValue) && jsc_value_to_boolean(jsValue)) {
becameTrue = true;
break;
}
wait(0.25);
}
g_assert_true(becameTrue);
}
static void resourceGetDataCallback(GObject* object, GAsyncResult* result, gpointer userData)
{
size_t dataSize;
GUniqueOutPtr<GError> error;
unsigned char* data = webkit_web_resource_get_data_finish(WEBKIT_WEB_RESOURCE(object), result, &dataSize, &error.outPtr());
g_assert_nonnull(data);
WebViewTest* test = static_cast<WebViewTest*>(userData);
test->m_resourceData.reset(reinterpret_cast<char*>(data));
test->m_resourceDataSize = dataSize;
g_main_loop_quit(test->m_mainLoop);
}
const char* WebViewTest::mainResourceData(size_t& mainResourceDataSize)
{
m_resourceDataSize = 0;
m_resourceData.reset();
WebKitWebResource* resource = webkit_web_view_get_main_resource(m_webView.get());
g_assert_nonnull(resource);
webkit_web_resource_get_data(resource, 0, resourceGetDataCallback, this);
g_main_loop_run(m_mainLoop);
mainResourceDataSize = m_resourceDataSize;
return m_resourceData.get();
}
static void runJavaScriptReadyCallback(GObject* object, GAsyncResult* result, WebViewTest* test)
{
test->m_javascriptResult = adoptGRef(webkit_web_view_evaluate_javascript_finish(WEBKIT_WEB_VIEW(object), result, test->m_javascriptError));
g_main_loop_quit(test->m_mainLoop);
}
static void runAsyncJavaScriptFunctionInWorldReadyCallback(GObject*, GAsyncResult* result, WebViewTest* test)
{
test->m_javascriptResult = adoptGRef(webkit_web_view_call_async_javascript_function_finish(test->webView(), result, test->m_javascriptError));
g_main_loop_quit(test->m_mainLoop);
}
JSCValue* WebViewTest::runJavaScriptAndWaitUntilFinished(const char* javascript, gsize length, GError** error)
{
m_javascriptResult = nullptr;
m_javascriptError = error;
webkit_web_view_evaluate_javascript(m_webView.get(), javascript, length, nullptr, nullptr, nullptr, reinterpret_cast<GAsyncReadyCallback>(runJavaScriptReadyCallback), this);
g_main_loop_run(m_mainLoop);
return m_javascriptResult.get();
}
JSCValue* WebViewTest::runJavaScriptAndWaitUntilFinished(const char* javascript, GError** error, WebKitWebView* webView)
{
m_javascriptResult = nullptr;
m_javascriptError = error;
if (!webView)
webView = m_webView.get();
webkit_web_view_evaluate_javascript(webView, javascript, -1, nullptr, nullptr, nullptr, reinterpret_cast<GAsyncReadyCallback>(runJavaScriptReadyCallback), this);
g_main_loop_run(m_mainLoop);
return m_javascriptResult.get();
}
JSCValue* WebViewTest::runJavaScriptFromGResourceAndWaitUntilFinished(const char* resource, GError** error)
{
m_javascriptResult = nullptr;
GRefPtr<GBytes> bytes = adoptGRef(g_resources_lookup_data(resource, G_RESOURCE_LOOKUP_FLAGS_NONE, error));
if (!bytes)
return nullptr;
gsize length;
const auto* script = g_bytes_get_data(bytes.get(), &length);
GUniquePtr<char> sourceURI(g_strdup_printf("resource://%s", resource));
m_javascriptError = error;
webkit_web_view_evaluate_javascript(m_webView.get(), reinterpret_cast<const char*>(script), length, nullptr, sourceURI.get(), nullptr, reinterpret_cast<GAsyncReadyCallback>(runJavaScriptReadyCallback), this);
g_main_loop_run(m_mainLoop);
return m_javascriptResult.get();
}
JSCValue* WebViewTest::runJavaScriptInWorldAndWaitUntilFinished(const char* javascript, const char* world, const char* sourceURI, GError** error)
{
m_javascriptResult = nullptr;
m_javascriptError = error;
webkit_web_view_evaluate_javascript(m_webView.get(), javascript, -1, world, sourceURI, nullptr, reinterpret_cast<GAsyncReadyCallback>(runJavaScriptReadyCallback), this);
g_main_loop_run(m_mainLoop);
return m_javascriptResult.get();
}
JSCValue* WebViewTest::runAsyncJavaScriptFunctionInWorldAndWaitUntilFinished(const char* body, GVariant* arguments, const char* world, GError** error)
{
m_javascriptResult = nullptr;
m_javascriptError = error;
webkit_web_view_call_async_javascript_function(m_webView.get(), body, -1, arguments, world, nullptr, nullptr, reinterpret_cast<GAsyncReadyCallback>(runAsyncJavaScriptFunctionInWorldReadyCallback), this);
g_main_loop_run(m_mainLoop);
return m_javascriptResult.get();
}
JSCValue* WebViewTest::runJavaScriptWithoutForcedUserGesturesAndWaitUntilFinished(const char* javascript, GError** error)
{
m_javascriptResult = nullptr;
m_javascriptError = error;
webkitWebViewRunJavascriptWithoutForcedUserGestures(m_webView.get(), javascript, nullptr, reinterpret_cast<GAsyncReadyCallback>(runJavaScriptReadyCallback), this);
g_main_loop_run(m_mainLoop);
return m_javascriptResult.get();
}
void WebViewTest::runJavaScriptAndWait(const char* javascript)
{
webkit_web_view_evaluate_javascript(m_webView.get(), javascript, -1, nullptr, nullptr, nullptr, nullptr, nullptr);
g_main_loop_run(m_mainLoop);
}
char* WebViewTest::javascriptResultToCString(JSCValue* value)
{
g_assert_true(JSC_IS_VALUE(value));
g_assert_true(jsc_value_is_string(value));
return jsc_value_to_string(value);
}
#if !ENABLE(2022_GLIB_API)
char* WebViewTest::javascriptResultToCString(WebKitJavascriptResult* javascriptResult)
{
return javascriptResultToCString(webkit_javascript_result_get_js_value(javascriptResult));
}
#endif
double WebViewTest::javascriptResultToNumber(JSCValue* value)
{
g_assert_true(JSC_IS_VALUE(value));
g_assert_true(jsc_value_is_number(value));
return jsc_value_to_double(value);
}
#if !ENABLE(2022_GLIB_API)
double WebViewTest::javascriptResultToNumber(WebKitJavascriptResult* javascriptResult)
{
return javascriptResultToNumber(webkit_javascript_result_get_js_value(javascriptResult));
}
#endif
bool WebViewTest::javascriptResultToBoolean(JSCValue* value)
{
g_assert_true(JSC_IS_VALUE(value));
g_assert_true(jsc_value_is_boolean(value));
return jsc_value_to_boolean(value);
}
#if !ENABLE(2022_GLIB_API)
bool WebViewTest::javascriptResultToBoolean(WebKitJavascriptResult* javascriptResult)
{
return javascriptResultToBoolean(webkit_javascript_result_get_js_value(javascriptResult));
}
#endif
bool WebViewTest::javascriptResultIsNull(JSCValue* value)
{
g_assert_true(JSC_IS_VALUE(value));
return jsc_value_is_null(value);
}
#if !ENABLE(2022_GLIB_API)
bool WebViewTest::javascriptResultIsNull(WebKitJavascriptResult* javascriptResult)
{
return javascriptResultIsNull(webkit_javascript_result_get_js_value(javascriptResult));
}
#endif
bool WebViewTest::javascriptResultIsUndefined(JSCValue* value)
{
g_assert_true(JSC_IS_VALUE(value));
return jsc_value_is_undefined(value);
}
#if !ENABLE(2022_GLIB_API)
bool WebViewTest::javascriptResultIsUndefined(WebKitJavascriptResult* javascriptResult)
{
return javascriptResultIsUndefined(webkit_javascript_result_get_js_value(javascriptResult));
}
#endif
bool WebViewTest::runWebProcessTest(const char* suiteName, const char* testName, const char* contents, const char* contentType)
{
if (!contentType) {
static const char* emptyHTML = "<html><body></body></html>";
loadHtml(contents ? contents : emptyHTML, "webprocess://test");
} else {
GRefPtr<GBytes> bytes = adoptGRef(g_bytes_new_static(contents, strlen(contents)));
loadBytes(bytes.get(), contentType, nullptr, "webprocess://test");
}
waitUntilLoadFinished();
GUniquePtr<char> script(g_strdup_printf("WebProcessTestRunner.runTest('%s/%s');", suiteName, testName));
GUniqueOutPtr<GError> error;
JSCValue* value = runJavaScriptAndWaitUntilFinished(script.get(), &error.outPtr());
g_assert_no_error(error.get());
loadURI("about:blank");
waitUntilLoadFinished();
return javascriptResultToBoolean(value);
}
GRefPtr<GDBusProxy> WebViewTest::extensionProxy()
{
GDBusConnection* connection = nullptr;
// If nothing has been loaded yet, load about:blank to force the web process to be spawned.
if (!webkit_web_view_get_uri(m_webView.get())) {
loadURI("about:blank");
waitUntilLoadFinished();
connection = s_dbusConnectionPageMap.get(webkit_web_view_get_page_id(m_webView.get()));
if (!connection) {
// Wait for page created signal.
g_idle_add([](gpointer userData) -> gboolean {
auto* test = static_cast<WebViewTest*>(userData);
if (!s_dbusConnectionPageMap.get(webkit_web_view_get_page_id(test->webView())))
return TRUE;
test->quitMainLoop();
return FALSE;
}, this);
g_main_loop_run(m_mainLoop);
// FIXME: we can cache this once we can monitor the page id on the web view.
connection = s_dbusConnectionPageMap.get(webkit_web_view_get_page_id(m_webView.get()));
}
} else
connection = s_dbusConnectionPageMap.get(webkit_web_view_get_page_id(m_webView.get()));
return adoptGRef(g_dbus_proxy_new_sync(connection, static_cast<GDBusProxyFlags>(G_DBUS_PROXY_FLAGS_DO_NOT_LOAD_PROPERTIES | G_DBUS_PROXY_FLAGS_DO_NOT_CONNECT_SIGNALS),
nullptr, nullptr, "/org/webkit/gtk/WebProcessExtensionTest", "org.webkit.gtk.WebProcessExtensionTest", nullptr, nullptr));
}
WebViewTest::NetworkPolicyGuard::NetworkPolicyGuard(WebViewTest* test, WebKitTLSErrorsPolicy policy)
{
#if ENABLE(2022_GLIB_API)
m_test = test;
m_originalPolicy = webkit_network_session_get_tls_errors_policy(test->m_networkSession.get());
webkit_network_session_set_tls_errors_policy(test->m_networkSession.get(), policy);
#else
m_manager = webkit_web_context_get_website_data_manager(test->m_webContext.get());
m_originalPolicy = webkit_website_data_manager_get_tls_errors_policy(m_manager);
webkit_website_data_manager_set_tls_errors_policy(m_manager, policy);
#endif
}
WebViewTest::NetworkPolicyGuard::~NetworkPolicyGuard()
{
#if ENABLE(2022_GLIB_API)
webkit_network_session_set_tls_errors_policy(m_test->m_networkSession.get(), m_originalPolicy);
#else
webkit_website_data_manager_set_tls_errors_policy(m_manager, m_originalPolicy);
#endif
}
void WebViewTest::NetworkPolicyGuard::setErrorPolicy(WebKitTLSErrorsPolicy policy)
{
#if ENABLE(2022_GLIB_API)
webkit_network_session_set_tls_errors_policy(m_test->m_networkSession.get(), policy);
#else
webkit_website_data_manager_set_tls_errors_policy(m_manager, policy);
#endif
}