| /* |
| * Copyright (C) 2012 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. |
| */ |
| |
| #include "config.h" |
| #include "WebProcessTest.h" |
| #include <JavaScriptCore/JSContextRef.h> |
| #include <JavaScriptCore/JSRetainPtr.h> |
| #include <fcntl.h> |
| #include <gio/gio.h> |
| #include <gio/gunixfdlist.h> |
| #include <glib/gstdio.h> |
| #if USE(GSTREAMER) |
| #include <gst/gst.h> |
| #endif |
| #include <jsc/jsc.h> |
| #include <stdlib.h> |
| #include <string.h> |
| #include <wtf/Deque.h> |
| #include <wtf/ProcessID.h> |
| #include <wtf/glib/GRefPtr.h> |
| #include <wtf/glib/GUniquePtr.h> |
| #include <wtf/text/CString.h> |
| |
| #if PLATFORM(GTK) |
| #include <webkit2/webkit-web-extension.h> |
| #elif PLATFORM(WPE) |
| #include <wpe/webkit-web-extension.h> |
| #endif |
| |
| static const char introspectionXML[] = |
| "<node>" |
| " <interface name='org.webkit.gtk.WebExtensionTest'>" |
| " <method name='GetTitle'>" |
| " <arg type='t' name='pageID' direction='in'/>" |
| " <arg type='s' name='title' direction='out'/>" |
| " </method>" |
| " <method name='InputElementIsUserEdited'>" |
| " <arg type='t' name='pageID' direction='in'/>" |
| " <arg type='s' name='elementID' direction='in'/>" |
| " <arg type='b' name='isUserEdited' direction='out'/>" |
| " </method>" |
| " <method name='AbortProcess'>" |
| " </method>" |
| " <method name='RunJavaScriptInIsolatedWorld'>" |
| " <arg type='t' name='pageID' direction='in'/>" |
| " <arg type='s' name='script' direction='in'/>" |
| " </method>" |
| " <method name='GetProcessIdentifier'>" |
| " <arg type='u' name='identifier' direction='out'/>" |
| " </method>" |
| " <method name='RemoveAVPluginsFromGSTRegistry'>" |
| " </method>" |
| " <signal name='PageCreated'>" |
| " <arg type='t' name='pageID' direction='out'/>" |
| " </signal>" |
| " <signal name='DocumentLoaded'/>" |
| " <signal name='FormControlsAssociated'>" |
| " <arg type='s' name='formIds' direction='out'/>" |
| " </signal>" |
| " <signal name='FormSubmissionWillSendDOMEvent'>" |
| " <arg type='s' name='formID' direction='out'/>" |
| " <arg type='s' name='textFieldNames' direction='out'/>" |
| " <arg type='s' name='textFieldValues' direction='out'/>" |
| " <arg type='b' name='targetFrameIsMainFrame' direction='out'/>" |
| " <arg type='b' name='sourceFrameIsMainFrame' direction='out'/>" |
| " </signal>" |
| " <signal name='FormSubmissionWillComplete'>" |
| " <arg type='s' name='formID' direction='out'/>" |
| " <arg type='s' name='textFieldNames' direction='out'/>" |
| " <arg type='s' name='textFieldValues' direction='out'/>" |
| " <arg type='b' name='targetFrameIsMainFrame' direction='out'/>" |
| " <arg type='b' name='sourceFrameIsMainFrame' direction='out'/>" |
| " </signal>" |
| " <signal name='URIChanged'>" |
| " <arg type='s' name='uri' direction='out'/>" |
| " </signal>" |
| " </interface>" |
| "</node>"; |
| |
| |
| typedef enum { |
| PageCreatedSignal, |
| DocumentLoadedSignal, |
| URIChangedSignal, |
| FormControlsAssociatedSignal, |
| FormSubmissionWillSendDOMEventSignal, |
| FormSubmissionWillCompleteSignal, |
| } DelayedSignalType; |
| |
| struct DelayedSignal { |
| explicit DelayedSignal(DelayedSignalType type) |
| : type(type) |
| { |
| } |
| |
| DelayedSignal(DelayedSignalType type, guint64 n) |
| : type(type) |
| , n(n) |
| { |
| } |
| |
| DelayedSignal(DelayedSignalType type, const char* str) |
| : type(type) |
| , str(str) |
| { |
| } |
| |
| DelayedSignal(DelayedSignalType type, const char* str, const char* str2, const char* str3, gboolean b, gboolean b2) |
| : type(type) |
| , str(str) |
| , str2(str2) |
| , str3(str3) |
| , b(b) |
| , b2(b2) |
| { |
| } |
| |
| DelayedSignalType type; |
| CString str; |
| CString str2; |
| CString str3; |
| gboolean b; |
| gboolean b2; |
| guint64 n; |
| }; |
| |
| Deque<DelayedSignal> delayedSignalsQueue; |
| |
| static void emitDocumentLoaded(GDBusConnection* connection) |
| { |
| bool ok = g_dbus_connection_emit_signal( |
| connection, |
| 0, |
| "/org/webkit/gtk/WebExtensionTest", |
| "org.webkit.gtk.WebExtensionTest", |
| "DocumentLoaded", |
| 0, |
| 0); |
| g_assert_true(ok); |
| } |
| |
| static void documentLoadedCallback(WebKitWebPage* webPage, WebKitWebExtension* extension) |
| { |
| #if PLATFORM(GTK) |
| WebKitDOMDocument* document = webkit_web_page_get_dom_document(webPage); |
| G_GNUC_BEGIN_IGNORE_DEPRECATIONS; |
| GRefPtr<WebKitDOMDOMWindow> window = adoptGRef(webkit_dom_document_get_default_view(document)); |
| webkit_dom_dom_window_webkit_message_handlers_post_message(window.get(), "dom", "DocumentLoaded"); |
| G_GNUC_END_IGNORE_DEPRECATIONS; |
| #endif |
| |
| webkit_web_page_send_message_to_view(webPage, webkit_user_message_new("DocumentLoaded", nullptr), nullptr, nullptr, nullptr); |
| |
| gpointer data = g_object_get_data(G_OBJECT(extension), "dbus-connection"); |
| if (data) |
| emitDocumentLoaded(G_DBUS_CONNECTION(data)); |
| else |
| delayedSignalsQueue.append(DelayedSignal(DocumentLoadedSignal)); |
| } |
| |
| static void emitURIChanged(GDBusConnection* connection, const char* uri) |
| { |
| bool ok = g_dbus_connection_emit_signal( |
| connection, |
| 0, |
| "/org/webkit/gtk/WebExtensionTest", |
| "org.webkit.gtk.WebExtensionTest", |
| "URIChanged", |
| g_variant_new("(s)", uri), |
| 0); |
| g_assert_true(ok); |
| } |
| |
| static void uriChangedCallback(WebKitWebPage* webPage, GParamSpec* pspec, WebKitWebExtension* extension) |
| { |
| gpointer data = g_object_get_data(G_OBJECT(extension), "dbus-connection"); |
| if (data) |
| emitURIChanged(G_DBUS_CONNECTION(data), webkit_web_page_get_uri(webPage)); |
| else |
| delayedSignalsQueue.append(DelayedSignal(URIChangedSignal, webkit_web_page_get_uri(webPage))); |
| } |
| |
| static gboolean sendRequestCallback(WebKitWebPage*, WebKitURIRequest* request, WebKitURIResponse* redirectResponse, gpointer) |
| { |
| gboolean returnValue = FALSE; |
| const char* requestURI = webkit_uri_request_get_uri(request); |
| g_assert_nonnull(requestURI); |
| |
| if (const char* suffix = g_strrstr(requestURI, "/remove-this/javascript.js")) { |
| GUniquePtr<char> prefix(g_strndup(requestURI, strlen(requestURI) - strlen(suffix))); |
| GUniquePtr<char> newURI(g_strdup_printf("%s/javascript.js", prefix.get())); |
| webkit_uri_request_set_uri(request, newURI.get()); |
| } else if (const char* suffix = g_strrstr(requestURI, "/remove-this/javascript-after-redirection.js")) { |
| // Redirected from /redirected.js, redirectResponse should be nullptr. |
| g_assert_true(WEBKIT_IS_URI_RESPONSE(redirectResponse)); |
| g_assert_true(g_str_has_suffix(webkit_uri_response_get_uri(redirectResponse), "/redirected.js")); |
| |
| GUniquePtr<char> prefix(g_strndup(requestURI, strlen(requestURI) - strlen(suffix))); |
| GUniquePtr<char> newURI(g_strdup_printf("%s/javascript-after-redirection.js", prefix.get())); |
| webkit_uri_request_set_uri(request, newURI.get()); |
| } else if (g_str_has_suffix(requestURI, "/redirected.js")) { |
| // Original request, redirectResponse should be nullptr. |
| g_assert_null(redirectResponse); |
| } else if (g_str_has_suffix(requestURI, "/add-do-not-track-header")) { |
| SoupMessageHeaders* headers = webkit_uri_request_get_http_headers(request); |
| g_assert_nonnull(headers); |
| soup_message_headers_append(headers, "DNT", "1"); |
| } else if (g_str_has_suffix(requestURI, "/normal-change-request") && !g_strrstr(requestURI, "/redirect-js/")) { |
| GUniquePtr<char> prefix(g_strndup(requestURI, strlen(requestURI) - strlen("/normal-change-request"))); |
| GUniquePtr<char> newURI(g_strdup_printf("%s/request-changed%s", prefix.get(), redirectResponse ? "-on-redirect" : "")); |
| webkit_uri_request_set_uri(request, newURI.get()); |
| } else if (g_str_has_suffix(requestURI, "/http-get-method")) { |
| g_assert_cmpstr(webkit_uri_request_get_http_method(request), ==, "GET"); |
| g_assert_cmpstr(webkit_uri_request_get_http_method(request), ==, SOUP_METHOD_GET); |
| } else if (g_str_has_suffix(requestURI, "/http-post-method")) { |
| g_assert_cmpstr(webkit_uri_request_get_http_method(request), ==, "POST"); |
| g_assert_cmpstr(webkit_uri_request_get_http_method(request), ==, SOUP_METHOD_POST); |
| returnValue = TRUE; |
| } else if (g_str_has_suffix(requestURI, "/cancel-this.js")) |
| returnValue = TRUE; |
| |
| return returnValue; |
| } |
| |
| static GVariant* serializeContextMenu(WebKitContextMenu* menu) |
| { |
| GVariantBuilder builder; |
| g_variant_builder_init(&builder, G_VARIANT_TYPE_ARRAY); |
| GList* items = webkit_context_menu_get_items(menu); |
| for (GList* it = items; it; it = g_list_next(it)) |
| g_variant_builder_add(&builder, "u", webkit_context_menu_item_get_stock_action(WEBKIT_CONTEXT_MENU_ITEM(it->data))); |
| return g_variant_builder_end(&builder); |
| } |
| |
| static GVariant* serializeNode(JSCValue* node) |
| { |
| GVariantBuilder builder; |
| g_variant_builder_init(&builder, G_VARIANT_TYPE_ARRAY); |
| GRefPtr<JSCValue> value = adoptGRef(jsc_value_object_get_property(node, "nodeName")); |
| g_variant_builder_add(&builder, "{sv}", "Name", g_variant_new_take_string(jsc_value_to_string(value.get()))); |
| value = adoptGRef(jsc_value_object_get_property(node, "nodeType")); |
| g_variant_builder_add(&builder, "{sv}", "Type", g_variant_new_uint32(jsc_value_to_int32(value.get()))); |
| value = adoptGRef(jsc_value_object_get_property(node, "textContent")); |
| g_variant_builder_add(&builder, "{sv}", "Contents", g_variant_new_take_string(jsc_value_to_string(value.get()))); |
| value = adoptGRef(jsc_value_object_get_property(node, "parentNode")); |
| if (jsc_value_is_null(value.get())) |
| g_variant_builder_add(&builder, "{sv}", "Parent", g_variant_new_string("ROOT")); |
| else { |
| value = jsc_value_object_get_property(value.get(), "nodeName"); |
| g_variant_builder_add(&builder, "{sv}", "Parent", g_variant_new_take_string(jsc_value_to_string(value.get()))); |
| } |
| return g_variant_builder_end(&builder); |
| } |
| |
| static gboolean contextMenuCallback(WebKitWebPage* page, WebKitContextMenu* menu, WebKitWebHitTestResult* hitTestResult, gpointer) |
| { |
| const char* pageURI = webkit_web_page_get_uri(page); |
| if (!g_strcmp0(pageURI, "ContextMenuTestDefault")) { |
| webkit_context_menu_set_user_data(menu, serializeContextMenu(menu)); |
| return FALSE; |
| } |
| |
| if (!g_strcmp0(pageURI, "ContextMenuTestCustom")) { |
| // Remove Back and Forward, and add Inspector action. |
| webkit_context_menu_remove(menu, webkit_context_menu_first(menu)); |
| webkit_context_menu_remove(menu, webkit_context_menu_first(menu)); |
| webkit_context_menu_append(menu, webkit_context_menu_item_new_separator()); |
| webkit_context_menu_append(menu, webkit_context_menu_item_new_from_stock_action(WEBKIT_CONTEXT_MENU_ACTION_INSPECT_ELEMENT)); |
| webkit_context_menu_set_user_data(menu, serializeContextMenu(menu)); |
| return TRUE; |
| } |
| |
| if (!g_strcmp0(pageURI, "ContextMenuTestClear")) { |
| webkit_context_menu_remove_all(menu); |
| return TRUE; |
| } |
| |
| if (!g_strcmp0(pageURI, "ContextMenuTestNode")) { |
| WebKitDOMNode* node = webkit_web_hit_test_result_get_node(hitTestResult); |
| g_assert_true(WEBKIT_DOM_IS_NODE(node)); |
| auto* frame = webkit_web_page_get_main_frame(page); |
| GRefPtr<JSCValue> jsNode = adoptGRef(webkit_frame_get_js_value_for_dom_object(frame, WEBKIT_DOM_OBJECT(node))); |
| webkit_context_menu_set_user_data(menu, serializeNode(jsNode.get())); |
| return TRUE; |
| } |
| |
| return FALSE; |
| } |
| |
| static void consoleMessageSentCallback(WebKitWebPage* webPage, WebKitConsoleMessage* consoleMessage) |
| { |
| g_assert_nonnull(consoleMessage); |
| GRefPtr<GVariant> variant = g_variant_new("(uusus)", webkit_console_message_get_source(consoleMessage), |
| webkit_console_message_get_level(consoleMessage), webkit_console_message_get_text(consoleMessage), |
| webkit_console_message_get_line(consoleMessage), webkit_console_message_get_source_id(consoleMessage)); |
| GUniquePtr<char> messageString(g_variant_print(variant.get(), FALSE)); |
| GRefPtr<JSCContext> jsContext = adoptGRef(webkit_frame_get_js_context(webkit_web_page_get_main_frame(webPage))); |
| GRefPtr<JSCValue> console = adoptGRef(jsc_context_evaluate(jsContext.get(), "window.webkit.messageHandlers.console", -1)); |
| g_assert_true(JSC_IS_VALUE(console.get())); |
| if (jsc_value_is_object(console.get())) { |
| GRefPtr<JSCValue> result = adoptGRef(jsc_value_object_invoke_method(console.get(), "postMessage", G_TYPE_STRING, messageString.get(), G_TYPE_NONE)); |
| g_assert_true(JSC_IS_VALUE(result.get())); |
| } |
| } |
| |
| static void emitFormControlsAssociated(GDBusConnection* connection, const char* formIds) |
| { |
| bool ok = g_dbus_connection_emit_signal( |
| connection, |
| nullptr, |
| "/org/webkit/gtk/WebExtensionTest", |
| "org.webkit.gtk.WebExtensionTest", |
| "FormControlsAssociated", |
| g_variant_new("(s)", formIds), |
| nullptr); |
| g_assert_true(ok); |
| } |
| |
| static void formControlsAssociatedForFrameCallback(WebKitWebPage* webPage, GPtrArray* formElements, WebKitFrame* frame, WebKitWebExtension* extension) |
| { |
| GString* formIdsBuilder = g_string_new(nullptr); |
| for (guint i = 0; i < formElements->len; ++i) { |
| g_assert_true(WEBKIT_DOM_IS_ELEMENT(g_ptr_array_index(formElements, i))); |
| GRefPtr<JSCValue> value = adoptGRef(webkit_frame_get_js_value_for_dom_object(frame, WEBKIT_DOM_OBJECT(g_ptr_array_index(formElements, i)))); |
| g_assert_true(JSC_IS_VALUE(value.get())); |
| g_assert_true(jsc_value_is_object(value.get())); |
| g_assert_true(jsc_value_object_is_instance_of(value.get(), "Element")); |
| GRefPtr<JSCValue> idValue = adoptGRef(jsc_value_object_get_property(value.get(), "id")); |
| GUniquePtr<char> elementID(jsc_value_to_string(idValue.get())); |
| g_string_append(formIdsBuilder, elementID.get()); |
| } |
| if (!formIdsBuilder->len) { |
| g_string_free(formIdsBuilder, TRUE); |
| return; |
| } |
| GUniquePtr<char> formIds(g_string_free(formIdsBuilder, FALSE)); |
| gpointer data = g_object_get_data(G_OBJECT(extension), "dbus-connection"); |
| if (data) |
| emitFormControlsAssociated(G_DBUS_CONNECTION(data), formIds.get()); |
| else |
| delayedSignalsQueue.append(DelayedSignal(FormControlsAssociatedSignal, formIds.get())); |
| } |
| |
| static void emitFormSubmissionEvent(GDBusConnection* connection, const char* methodName, const char* formID, const char* names, const char* values, gboolean targetFrameIsMainFrame, gboolean sourceFrameIsMainFrame) |
| { |
| bool ok = g_dbus_connection_emit_signal( |
| connection, |
| nullptr, |
| "/org/webkit/gtk/WebExtensionTest", |
| "org.webkit.gtk.WebExtensionTest", |
| methodName, |
| g_variant_new("(sssbb)", formID ? formID : "", names, values, targetFrameIsMainFrame, sourceFrameIsMainFrame), |
| nullptr); |
| g_assert_true(ok); |
| } |
| |
| static void handleFormSubmissionCallback(WebKitWebPage* webPage, DelayedSignalType delayedSignalType, const char* methodName, const char* formID, WebKitFrame* sourceFrame, WebKitFrame* targetFrame, GPtrArray* textFieldNames, GPtrArray* textFieldValues, WebKitWebExtension* extension) |
| { |
| GString* namesBuilder = g_string_new(nullptr); |
| for (guint i = 0; i < textFieldNames->len; ++i) { |
| auto* name = static_cast<char*>(g_ptr_array_index(textFieldNames, i)); |
| g_string_append(namesBuilder, name); |
| g_string_append_c(namesBuilder, ','); |
| } |
| GUniquePtr<char> names(g_string_free(namesBuilder, FALSE)); |
| |
| GString* valuesBuilder = g_string_new(nullptr); |
| for (guint i = 0; i < textFieldValues->len; ++i) { |
| auto* value = static_cast<char*>(g_ptr_array_index(textFieldValues, i)); |
| g_string_append(valuesBuilder, value); |
| g_string_append_c(valuesBuilder, ','); |
| } |
| GUniquePtr<char> values(g_string_free(valuesBuilder, FALSE)); |
| |
| gpointer data = g_object_get_data(G_OBJECT(extension), "dbus-connection"); |
| if (data) |
| emitFormSubmissionEvent(G_DBUS_CONNECTION(data), methodName, formID, names.get(), values.get(), webkit_frame_is_main_frame(targetFrame), webkit_frame_is_main_frame(sourceFrame)); |
| else |
| delayedSignalsQueue.append(DelayedSignal(delayedSignalType, formID, names.get(), values.get(), webkit_frame_is_main_frame(targetFrame), webkit_frame_is_main_frame(sourceFrame))); |
| } |
| |
| static void willSubmitFormCallback(WebKitWebPage* webPage, WebKitDOMElement* formElement, WebKitFormSubmissionStep step, WebKitFrame* sourceFrame, WebKitFrame* targetFrame, GPtrArray* textFieldNames, GPtrArray* textFieldValues, WebKitWebExtension* extension) |
| { |
| #if PLATFORM(GTK) |
| G_GNUC_BEGIN_IGNORE_DEPRECATIONS; |
| g_assert_true(WEBKIT_DOM_IS_HTML_FORM_ELEMENT(formElement)); |
| G_GNUC_END_IGNORE_DEPRECATIONS; |
| #endif |
| GRefPtr<JSCValue> jsFormElement = adoptGRef(webkit_frame_get_js_value_for_dom_object(sourceFrame, WEBKIT_DOM_OBJECT(formElement))); |
| g_assert_true(JSC_IS_VALUE(jsFormElement.get())); |
| g_assert_true(jsc_value_is_object(jsFormElement.get())); |
| g_assert_true(jsc_value_object_is_instance_of(jsFormElement.get(), "HTMLFormElement")); |
| GRefPtr<JSCValue> idValue = adoptGRef(jsc_value_object_get_property(jsFormElement.get(), "id")); |
| GUniquePtr<char> formID(jsc_value_to_string(idValue.get())); |
| |
| switch (step) { |
| case WEBKIT_FORM_SUBMISSION_WILL_SEND_DOM_EVENT: |
| handleFormSubmissionCallback(webPage, FormSubmissionWillSendDOMEventSignal, "FormSubmissionWillSendDOMEvent", formID.get(), sourceFrame, targetFrame, textFieldNames, textFieldValues, extension); |
| break; |
| case WEBKIT_FORM_SUBMISSION_WILL_COMPLETE: |
| handleFormSubmissionCallback(webPage, FormSubmissionWillCompleteSignal, "FormSubmissionWillComplete", formID.get(), sourceFrame, targetFrame, textFieldNames, textFieldValues, extension); |
| break; |
| default: |
| g_assert_not_reached(); |
| } |
| } |
| |
| static gboolean pageMessageReceivedCallback(WebKitWebPage* webPage, WebKitUserMessage* message, WebKitWebExtension* extension) |
| { |
| const char* messageName = webkit_user_message_get_name(message); |
| if (!g_strcmp0(messageName, "Test.Hello")) { |
| auto* parameters = webkit_user_message_get_parameters(message); |
| g_assert_nonnull(parameters); |
| const char* parameter = nullptr; |
| g_variant_get(parameters, "&s", ¶meter); |
| g_assert_cmpstr(parameter, ==, "WebProcess"); |
| g_assert_null(webkit_user_message_get_fd_list(message)); |
| webkit_user_message_send_reply(message, webkit_user_message_new("Test.Hello", g_variant_new("s", "UIProcess"))); |
| return TRUE; |
| } |
| |
| if (!g_strcmp0(messageName, "Test.Optional")) { |
| auto* parameters = webkit_user_message_get_parameters(message); |
| g_assert_nonnull(parameters); |
| const char* parameter1 = nullptr; |
| const char* parameter2 = nullptr; |
| g_variant_get(parameters, "(&sm&s)", ¶meter1, ¶meter2); |
| g_assert_cmpstr(parameter1, ==, "Hello"); |
| webkit_user_message_send_reply(message, webkit_user_message_new("Test.Optional", g_variant_new("s", parameter2 ? parameter2 : "NULL"))); |
| return TRUE; |
| } |
| |
| if (!g_strcmp0(messageName, "Test.Ping")) { |
| g_assert_null(webkit_user_message_get_parameters(message)); |
| webkit_user_message_send_reply(message, webkit_user_message_new("Test.Pong", nullptr)); |
| return TRUE; |
| } |
| |
| if (!g_strcmp0(messageName, "Test.OpenFile")) { |
| auto* parameters = webkit_user_message_get_parameters(message); |
| g_assert_nonnull(parameters); |
| const char* filename = nullptr; |
| g_variant_get(parameters, "&s", &filename); |
| int fd = g_open(filename, O_RDONLY, 0); |
| g_assert_cmpint(fd, !=, -1); |
| GRefPtr<GUnixFDList> fdList = adoptGRef(g_unix_fd_list_new()); |
| GUniqueOutPtr<GError> error; |
| g_unix_fd_list_append(fdList.get(), fd, &error.outPtr()); |
| g_assert_no_error(error.get()); |
| close(fd); |
| |
| webkit_user_message_send_reply(message, webkit_user_message_new_with_fd_list("Test.OpenFile", g_variant_new("h", 0), fdList.get())); |
| return TRUE; |
| } |
| |
| if (!g_strcmp0(messageName, "Test.Infinite")) { |
| abort(); |
| return TRUE; |
| } |
| |
| if (!g_strcmp0(messageName, "Test.AsyncPing")) { |
| webkit_web_page_send_message_to_view(webPage, webkit_user_message_new("Test.Ping", nullptr), nullptr, |
| [](GObject* object, GAsyncResult* result, gpointer) { |
| auto* webPage = WEBKIT_WEB_PAGE(object); |
| GUniqueOutPtr<GError> error; |
| GRefPtr<WebKitUserMessage> reply = adoptGRef(webkit_web_page_send_message_to_view_finish(webPage, result, &error.outPtr())); |
| g_assert_no_error(error.get()); |
| g_assert_cmpstr(webkit_user_message_get_name(reply.get()), ==, "Test.Pong"); |
| webkit_web_page_send_message_to_view(webPage, webkit_user_message_new("Test.AsyncPong", nullptr), nullptr, nullptr, nullptr); |
| }, nullptr); |
| return TRUE; |
| } |
| |
| return FALSE; |
| } |
| |
| static void emitPageCreated(GDBusConnection* connection, guint64 pageID) |
| { |
| bool ok = g_dbus_connection_emit_signal( |
| connection, |
| nullptr, |
| "/org/webkit/gtk/WebExtensionTest", |
| "org.webkit.gtk.WebExtensionTest", |
| "PageCreated", |
| g_variant_new("(t)", pageID), |
| nullptr); |
| g_assert_true(ok); |
| } |
| |
| static void pageCreatedCallback(WebKitWebExtension* extension, WebKitWebPage* webPage, gpointer) |
| { |
| if (auto* data = g_object_get_data(G_OBJECT(extension), "dbus-connection")) |
| emitPageCreated(G_DBUS_CONNECTION(data), webkit_web_page_get_id(webPage)); |
| else |
| delayedSignalsQueue.append(DelayedSignal(PageCreatedSignal, webkit_web_page_get_id(webPage))); |
| |
| webkit_web_extension_send_message_to_context(extension, webkit_user_message_new("PageCreated", g_variant_new("(t)", webkit_web_page_get_id(webPage))), |
| nullptr, nullptr, nullptr); |
| |
| g_signal_connect(webPage, "document-loaded", G_CALLBACK(documentLoadedCallback), extension); |
| g_signal_connect(webPage, "notify::uri", G_CALLBACK(uriChangedCallback), extension); |
| g_signal_connect(webPage, "send-request", G_CALLBACK(sendRequestCallback), nullptr); |
| g_signal_connect(webPage, "console-message-sent", G_CALLBACK(consoleMessageSentCallback), nullptr); |
| g_signal_connect(webPage, "context-menu", G_CALLBACK(contextMenuCallback), nullptr); |
| g_signal_connect(webPage, "form-controls-associated-for-frame", G_CALLBACK(formControlsAssociatedForFrameCallback), extension); |
| g_signal_connect(webPage, "will-submit-form", G_CALLBACK(willSubmitFormCallback), extension); |
| g_signal_connect(webPage, "user-message-received", G_CALLBACK(pageMessageReceivedCallback), extension); |
| } |
| |
| static gboolean extensionMessageReceivedCallback(WebKitWebExtension* extension, WebKitUserMessage* message) |
| { |
| const char* messageName = webkit_user_message_get_name(message); |
| if (g_strcmp0(messageName, "RequestPing")) { |
| webkit_web_extension_send_message_to_context(extension, webkit_user_message_new("Ping", nullptr), nullptr, |
| [](GObject* object, GAsyncResult* result, gpointer) { |
| auto* extension = WEBKIT_WEB_EXTENSION(object); |
| GUniqueOutPtr<GError> error; |
| GRefPtr<WebKitUserMessage> reply = adoptGRef(webkit_web_extension_send_message_to_context_finish(extension, result, &error.outPtr())); |
| g_assert_no_error(error.get()); |
| g_assert_cmpstr(webkit_user_message_get_name(reply.get()), ==, "Pong"); |
| webkit_web_extension_send_message_to_context(extension, webkit_user_message_new("Test.FinishedPingRequest", nullptr), nullptr, nullptr, nullptr); |
| }, nullptr); |
| return TRUE; |
| } |
| |
| return FALSE; |
| } |
| |
| static char* echoCallback(const char* message) |
| { |
| return g_strdup(message); |
| } |
| |
| static void windowObjectCleared(WebKitScriptWorld* world, WebKitWebPage* page, WebKitFrame* frame, gpointer) |
| { |
| GRefPtr<JSCContext> jsContext = adoptGRef(webkit_frame_get_js_context_for_script_world(frame, world)); |
| g_assert_true(JSC_IS_CONTEXT(jsContext.get())); |
| GRefPtr<JSCValue> function = adoptGRef(jsc_value_new_function(jsContext.get(), "echo", G_CALLBACK(echoCallback), NULL, NULL, G_TYPE_STRING, 1, G_TYPE_STRING)); |
| jsc_context_set_value(jsContext.get(), "echo", function.get()); |
| |
| auto* fileClass = jsc_context_register_class(jsContext.get(), "GFile", nullptr, nullptr, static_cast<GDestroyNotify>(g_object_unref)); |
| GRefPtr<JSCValue> constructor = adoptGRef(jsc_class_add_constructor(fileClass, "GFile", G_CALLBACK(g_file_new_for_path), nullptr, nullptr, G_TYPE_OBJECT, 1, G_TYPE_STRING)); |
| jsc_class_add_method(fileClass, "path", G_CALLBACK(g_file_get_path), nullptr, nullptr, G_TYPE_STRING, 0, G_TYPE_NONE); |
| jsc_context_set_value(jsContext.get(), "GFile", constructor.get()); |
| } |
| |
| static WebKitWebPage* getWebPage(WebKitWebExtension* extension, uint64_t pageID, GDBusMethodInvocation* invocation) |
| { |
| WebKitWebPage* page = webkit_web_extension_get_page(extension, pageID); |
| if (!page) { |
| g_dbus_method_invocation_return_error( |
| invocation, G_DBUS_ERROR, G_DBUS_ERROR_INVALID_ARGS, |
| "Invalid page ID: %" G_GUINT64_FORMAT, pageID); |
| return 0; |
| } |
| |
| g_assert_cmpuint(webkit_web_page_get_id(page), ==, pageID); |
| return page; |
| } |
| |
| static void methodCallCallback(GDBusConnection* connection, const char* sender, const char* objectPath, const char* interfaceName, const char* methodName, GVariant* parameters, GDBusMethodInvocation* invocation, gpointer userData) |
| { |
| if (g_strcmp0(interfaceName, "org.webkit.gtk.WebExtensionTest")) |
| return; |
| |
| if (!g_strcmp0(methodName, "GetTitle")) { |
| uint64_t pageID; |
| g_variant_get(parameters, "(t)", &pageID); |
| WebKitWebPage* page = getWebPage(WEBKIT_WEB_EXTENSION(userData), pageID, invocation); |
| if (!page) |
| return; |
| |
| GRefPtr<JSCContext> jsContext = adoptGRef(webkit_frame_get_js_context(webkit_web_page_get_main_frame(page))); |
| GRefPtr<JSCValue> titleValue = adoptGRef(jsc_context_evaluate(jsContext.get(), "document.title", -1)); |
| GUniquePtr<char> title(jsc_value_to_string(titleValue.get())); |
| g_dbus_method_invocation_return_value(invocation, g_variant_new("(s)", title.get())); |
| } else if (!g_strcmp0(methodName, "InputElementIsUserEdited")) { |
| uint64_t pageID; |
| const char* elementID; |
| g_variant_get(parameters, "(t&s)", &pageID, &elementID); |
| WebKitWebPage* page = getWebPage(WEBKIT_WEB_EXTENSION(userData), pageID, invocation); |
| if (!page) |
| return; |
| |
| WebKitDOMDocument* document = webkit_web_page_get_dom_document(page); |
| WebKitFrame* frame = webkit_web_page_get_main_frame(page); |
| GRefPtr<JSCContext> jsContext = adoptGRef(webkit_frame_get_js_context(frame)); |
| GRefPtr<JSCValue> jsDocument = adoptGRef(webkit_frame_get_js_value_for_dom_object(frame, WEBKIT_DOM_OBJECT(document))); |
| GRefPtr<JSCValue> jsInputElement = adoptGRef(jsc_value_object_invoke_method(jsDocument.get(), "getElementById", G_TYPE_STRING, elementID, G_TYPE_NONE)); |
| WebKitDOMNode* node = webkit_dom_node_for_js_value(jsInputElement.get()); |
| gboolean isUserEdited = webkit_dom_element_html_input_element_is_user_edited(WEBKIT_DOM_ELEMENT(node)); |
| g_dbus_method_invocation_return_value(invocation, g_variant_new("(b)", isUserEdited)); |
| } else if (!g_strcmp0(methodName, "RunJavaScriptInIsolatedWorld")) { |
| uint64_t pageID; |
| const char* script; |
| g_variant_get(parameters, "(t&s)", &pageID, &script); |
| WebKitWebPage* page = getWebPage(WEBKIT_WEB_EXTENSION(userData), pageID, invocation); |
| if (!page) |
| return; |
| |
| GRefPtr<WebKitScriptWorld> world = adoptGRef(webkit_script_world_new()); |
| g_assert_true(webkit_script_world_get_default() != world.get()); |
| g_assert_true(g_str_has_prefix(webkit_script_world_get_name(world.get()), "UniqueWorld_")); |
| WebKitFrame* frame = webkit_web_page_get_main_frame(page); |
| GRefPtr<JSCContext> jsContext = adoptGRef(webkit_frame_get_js_context_for_script_world(frame, world.get())); |
| GRefPtr<JSCValue> result = adoptGRef(jsc_context_evaluate(jsContext.get(), script, -1)); |
| g_dbus_method_invocation_return_value(invocation, 0); |
| } else if (!g_strcmp0(methodName, "AbortProcess")) { |
| abort(); |
| } else if (!g_strcmp0(methodName, "GetProcessIdentifier")) { |
| g_dbus_method_invocation_return_value(invocation, |
| g_variant_new("(u)", static_cast<guint32>(getCurrentProcessID()))); |
| } else if (!g_strcmp0(methodName, "RemoveAVPluginsFromGSTRegistry")) { |
| #if USE(GSTREAMER) |
| gst_init(nullptr, nullptr); |
| static const char* avPlugins[] = { "libav", "omx", "vaapi", nullptr }; |
| GstRegistry* registry = gst_registry_get(); |
| for (unsigned i = 0; avPlugins[i]; ++i) { |
| if (GstPlugin* plugin = gst_registry_find_plugin(registry, avPlugins[i])) { |
| gst_registry_remove_plugin(registry, plugin); |
| gst_object_unref(plugin); |
| } |
| } |
| #endif |
| g_dbus_method_invocation_return_value(invocation, nullptr); |
| } |
| } |
| |
| static const GDBusInterfaceVTable interfaceVirtualTable = { |
| methodCallCallback, 0, 0, { 0, } |
| }; |
| |
| static void dbusConnectionCreated(GObject*, GAsyncResult* result, gpointer userData) |
| { |
| GUniqueOutPtr<GError> error; |
| GDBusConnection* connection = g_dbus_connection_new_for_address_finish(result, &error.outPtr()); |
| g_assert(G_IS_DBUS_CONNECTION(connection)); |
| g_assert_no_error(error.get()); |
| |
| static GDBusNodeInfo* introspectionData = nullptr; |
| if (!introspectionData) |
| introspectionData = g_dbus_node_info_new_for_xml(introspectionXML, nullptr); |
| |
| unsigned registrationID = g_dbus_connection_register_object( |
| connection, |
| "/org/webkit/gtk/WebExtensionTest", |
| introspectionData->interfaces[0], |
| &interfaceVirtualTable, |
| g_object_ref(userData), |
| static_cast<GDestroyNotify>(g_object_unref), |
| &error.outPtr()); |
| if (!registrationID) |
| g_error("Failed to register object: %s\n", error->message); |
| |
| g_object_set_data_full(G_OBJECT(userData), "dbus-connection", connection, g_object_unref); |
| while (delayedSignalsQueue.size()) { |
| DelayedSignal delayedSignal = delayedSignalsQueue.takeFirst(); |
| switch (delayedSignal.type) { |
| case PageCreatedSignal: |
| emitPageCreated(connection, delayedSignal.n); |
| break; |
| case DocumentLoadedSignal: |
| emitDocumentLoaded(connection); |
| break; |
| case URIChangedSignal: |
| emitURIChanged(connection, delayedSignal.str.data()); |
| break; |
| case FormControlsAssociatedSignal: |
| emitFormControlsAssociated(connection, delayedSignal.str.data()); |
| break; |
| case FormSubmissionWillCompleteSignal: |
| emitFormSubmissionEvent(connection, "FormSubmissionWillComplete", delayedSignal.str.data(), delayedSignal.str2.data(), delayedSignal.str3.data(), delayedSignal.b, delayedSignal.b2); |
| break; |
| case FormSubmissionWillSendDOMEventSignal: |
| emitFormSubmissionEvent(connection, "FormSubmissionWillSendDOMEvent", delayedSignal.str.data(), delayedSignal.str2.data(), delayedSignal.str3.data(), delayedSignal.b, delayedSignal.b2); |
| break; |
| } |
| } |
| } |
| |
| static void registerGResource(void) |
| { |
| GUniquePtr<char> resourcesPath(g_build_filename(WEBKIT_TEST_RESOURCES_DIR, "webkitglib-tests-resources.gresource", nullptr)); |
| GResource* resource = g_resource_load(resourcesPath.get(), nullptr); |
| g_assert_nonnull(resource); |
| |
| g_resources_register(resource); |
| g_resource_unref(resource); |
| } |
| |
| extern "C" void webkit_web_extension_initialize_with_user_data(WebKitWebExtension* extension, GVariant* userData) |
| { |
| WebKitScriptWorld* isolatedWorld = webkit_script_world_new_with_name("WebExtensionTestScriptWorld"); |
| g_assert_true(WEBKIT_IS_SCRIPT_WORLD(isolatedWorld)); |
| g_assert_cmpstr(webkit_script_world_get_name(isolatedWorld), ==, "WebExtensionTestScriptWorld"); |
| g_object_set_data_full(G_OBJECT(extension), "wk-script-world", isolatedWorld, g_object_unref); |
| |
| g_signal_connect(extension, "user-message-received", G_CALLBACK(extensionMessageReceivedCallback), nullptr); |
| g_signal_connect(extension, "page-created", G_CALLBACK(pageCreatedCallback), extension); |
| g_signal_connect(webkit_script_world_get_default(), "window-object-cleared", G_CALLBACK(windowObjectCleared), nullptr); |
| |
| registerGResource(); |
| |
| g_assert_nonnull(userData); |
| const char* guid; |
| const char* address; |
| g_variant_get(userData, "(&s&s)", &guid, &address); |
| g_assert_nonnull(guid); |
| g_assert_nonnull(address); |
| |
| g_dbus_connection_new_for_address(address, G_DBUS_CONNECTION_FLAGS_AUTHENTICATION_CLIENT, nullptr, nullptr, |
| dbusConnectionCreated, extension); |
| } |