/* * Copyright (C) 2009, 2010 Gustavo Noronha Silva * Copyright (C) 2009 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 <gtk/gtk.h> #include <libsoup/soup.h> #include <string.h> #include <webkit/webkit.h> #if GTK_CHECK_VERSION(2, 14, 0) /* This string has to be rather big because of the cancelled test - it * looks like soup refuses to send or receive a too small chunk */ #define HTML_STRING "<html><body>Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!Testing!</body></html>" SoupURI* base_uri; /* For real request testing */ static void server_callback(SoupServer* server, SoupMessage* msg, const char* path, GHashTable* query, SoupClientContext* context, gpointer data) { if (msg->method != SOUP_METHOD_GET) { soup_message_set_status(msg, SOUP_STATUS_NOT_IMPLEMENTED); return; } soup_message_set_status(msg, SOUP_STATUS_OK); if (g_str_equal(path, "/test_loading_status") || g_str_equal(path, "/test_loading_status2")) soup_message_body_append(msg->response_body, SOUP_MEMORY_STATIC, HTML_STRING, strlen(HTML_STRING)); else if (g_str_equal(path, "/test_load_error")) { soup_message_set_status(msg, SOUP_STATUS_CANT_CONNECT); } else if (g_str_equal(path, "/test_loading_cancelled")) { soup_message_headers_set_encoding(msg->response_headers, SOUP_ENCODING_CHUNKED); soup_message_body_append(msg->response_body, SOUP_MEMORY_STATIC, HTML_STRING, strlen(HTML_STRING)); soup_server_unpause_message(server, msg); return; } soup_message_body_complete(msg->response_body); } typedef struct { WebKitWebView* webView; GMainLoop *loop; gboolean has_been_provisional; gboolean has_been_committed; gboolean has_been_first_visually_non_empty_layout; gboolean has_been_finished; gboolean has_been_failed; gboolean has_been_load_error; } WebLoadingFixture; static void web_loading_fixture_setup(WebLoadingFixture* fixture, gconstpointer data) { fixture->webView = WEBKIT_WEB_VIEW(webkit_web_view_new()); fixture->loop = g_main_loop_new(NULL, TRUE); g_object_ref_sink(fixture->webView); fixture->has_been_provisional = FALSE; fixture->has_been_committed = FALSE; fixture->has_been_first_visually_non_empty_layout = FALSE; fixture->has_been_finished = FALSE; fixture->has_been_failed = FALSE; fixture->has_been_load_error = FALSE; } static void web_loading_fixture_teardown(WebLoadingFixture* fixture, gconstpointer data) { g_object_unref(fixture->webView); g_main_loop_unref(fixture->loop); } static char* get_uri_for_path(const char* path) { SoupURI* uri; char* uri_string; uri = soup_uri_new_with_base(base_uri, path); uri_string = soup_uri_to_string(uri, FALSE); soup_uri_free (uri); return uri_string; } static void load_finished_cb(WebKitWebView* web_view, WebKitWebFrame* web_frame, WebLoadingFixture* fixture) { g_assert(fixture->has_been_provisional); g_assert(fixture->has_been_committed); g_assert(fixture->has_been_first_visually_non_empty_layout); g_main_loop_quit(fixture->loop); } static void status_changed_cb(GObject* object, GParamSpec* pspec, WebLoadingFixture* fixture) { WebKitLoadStatus status = webkit_web_view_get_load_status(WEBKIT_WEB_VIEW(object)); switch (status) { case WEBKIT_LOAD_PROVISIONAL: g_assert(!fixture->has_been_provisional); g_assert(!fixture->has_been_committed); g_assert(!fixture->has_been_first_visually_non_empty_layout); fixture->has_been_provisional = TRUE; break; case WEBKIT_LOAD_COMMITTED: g_assert(fixture->has_been_provisional); g_assert(!fixture->has_been_committed); g_assert(!fixture->has_been_first_visually_non_empty_layout); fixture->has_been_committed = TRUE; break; case WEBKIT_LOAD_FIRST_VISUALLY_NON_EMPTY_LAYOUT: g_assert(fixture->has_been_provisional); g_assert(fixture->has_been_committed); g_assert(!fixture->has_been_first_visually_non_empty_layout); fixture->has_been_first_visually_non_empty_layout = TRUE; break; case WEBKIT_LOAD_FINISHED: g_assert(fixture->has_been_provisional); g_assert(fixture->has_been_committed); g_assert(fixture->has_been_first_visually_non_empty_layout); break; default: g_assert_not_reached(); } } static void test_loading_status(WebLoadingFixture* fixture, gconstpointer data) { char* uri_string; g_assert_cmpint(webkit_web_view_get_load_status(fixture->webView), ==, WEBKIT_LOAD_PROVISIONAL); g_object_connect(G_OBJECT(fixture->webView), "signal::notify::load-status", G_CALLBACK(status_changed_cb), fixture, "signal::load-finished", G_CALLBACK(load_finished_cb), fixture, NULL); uri_string = get_uri_for_path("/test_loading_status"); /* load_uri will trigger the navigation-policy-decision-requested * signal emission; */ webkit_web_view_load_uri(fixture->webView, uri_string); g_free(uri_string); g_main_loop_run(fixture->loop); } static void load_error_status_changed_cb(GObject* object, GParamSpec* pspec, WebLoadingFixture* fixture) { WebKitLoadStatus status = webkit_web_view_get_load_status(WEBKIT_WEB_VIEW(object)); switch(status) { case WEBKIT_LOAD_PROVISIONAL: g_assert(!fixture->has_been_provisional); fixture->has_been_provisional = TRUE; break; case WEBKIT_LOAD_COMMITTED: g_assert(!fixture->has_been_committed); fixture->has_been_committed = TRUE; break; case WEBKIT_LOAD_FINISHED: g_assert(fixture->has_been_provisional); g_assert(fixture->has_been_load_error); g_assert(fixture->has_been_failed); g_assert(!fixture->has_been_finished); fixture->has_been_finished = TRUE; break; case WEBKIT_LOAD_FAILED: g_assert(!fixture->has_been_failed); fixture->has_been_failed = TRUE; g_main_loop_quit(fixture->loop); break; default: break; } } static gboolean load_error_cb(WebKitWebView* webView, WebKitWebFrame* frame, const char* uri, GError *error, WebLoadingFixture* fixture) { g_assert(fixture->has_been_provisional); g_assert(!fixture->has_been_load_error); fixture->has_been_load_error = TRUE; return FALSE; } static void test_loading_error(WebLoadingFixture* fixture, gconstpointer data) { char* uri_string; g_test_bug("28842"); g_signal_connect(fixture->webView, "load-error", G_CALLBACK(load_error_cb), fixture); g_signal_connect(fixture->webView, "notify::load-status", G_CALLBACK(load_error_status_changed_cb), fixture); uri_string = get_uri_for_path("/test_load_error"); webkit_web_view_load_uri(fixture->webView, uri_string); g_free(uri_string); g_main_loop_run(fixture->loop); g_assert(fixture->has_been_provisional); g_assert(!fixture->has_been_committed); g_assert(fixture->has_been_load_error); g_assert(fixture->has_been_failed); g_assert(!fixture->has_been_finished); } /* Cancelled load */ static gboolean load_cancelled_cb(WebKitWebView* webView, WebKitWebFrame* frame, const char* uri, GError *error, WebLoadingFixture* fixture) { g_assert(fixture->has_been_provisional); g_assert(fixture->has_been_failed); g_assert(!fixture->has_been_load_error); g_assert(error->code == WEBKIT_NETWORK_ERROR_CANCELLED); fixture->has_been_load_error = TRUE; return TRUE; } static gboolean stop_load (gpointer data) { webkit_web_view_stop_loading(WEBKIT_WEB_VIEW(data)); return FALSE; } static void load_cancelled_status_changed_cb(GObject* object, GParamSpec* pspec, WebLoadingFixture* fixture) { WebKitLoadStatus status = webkit_web_view_get_load_status(WEBKIT_WEB_VIEW(object)); switch(status) { case WEBKIT_LOAD_PROVISIONAL: g_assert(!fixture->has_been_provisional); g_assert(!fixture->has_been_failed); fixture->has_been_provisional = TRUE; break; case WEBKIT_LOAD_COMMITTED: g_idle_add (stop_load, object); break; case WEBKIT_LOAD_FAILED: g_assert(fixture->has_been_provisional); g_assert(!fixture->has_been_failed); g_assert(!fixture->has_been_load_error); fixture->has_been_failed = TRUE; g_main_loop_quit(fixture->loop); break; case WEBKIT_LOAD_FINISHED: g_assert_not_reached(); break; default: break; } } static void test_loading_cancelled(WebLoadingFixture* fixture, gconstpointer data) { char* uri_string; g_test_bug("29644"); g_signal_connect(fixture->webView, "load-error", G_CALLBACK(load_cancelled_cb), fixture); g_signal_connect(fixture->webView, "notify::load-status", G_CALLBACK(load_cancelled_status_changed_cb), fixture); uri_string = get_uri_for_path("/test_loading_cancelled"); webkit_web_view_load_uri(fixture->webView, uri_string); g_free(uri_string); g_main_loop_run(fixture->loop); } static void load_goback_status_changed_cb(GObject* object, GParamSpec* pspec, WebLoadingFixture* fixture) { WebKitLoadStatus status = webkit_web_view_get_load_status(WEBKIT_WEB_VIEW(object)); switch(status) { case WEBKIT_LOAD_PROVISIONAL: g_assert(!fixture->has_been_provisional); fixture->has_been_provisional = TRUE; break; case WEBKIT_LOAD_COMMITTED: g_assert(fixture->has_been_provisional); fixture->has_been_committed = TRUE; break; case WEBKIT_LOAD_FAILED: g_assert_not_reached(); break; case WEBKIT_LOAD_FINISHED: g_assert(fixture->has_been_provisional); g_assert(fixture->has_been_committed); fixture->has_been_finished = TRUE; g_main_loop_quit(fixture->loop); break; default: break; } } static void load_wentback_status_changed_cb(GObject* object, GParamSpec* pspec, WebLoadingFixture* fixture) { WebKitLoadStatus status = webkit_web_view_get_load_status(WEBKIT_WEB_VIEW(object)); char* uri_string; char* uri_string2; uri_string = get_uri_for_path("/test_loading_status"); uri_string2 = get_uri_for_path("/test_loading_status2"); switch(status) { case WEBKIT_LOAD_PROVISIONAL: g_assert_cmpstr(webkit_web_view_get_uri(fixture->webView), ==, uri_string2); break; case WEBKIT_LOAD_COMMITTED: g_assert_cmpstr(webkit_web_view_get_uri(fixture->webView), ==, uri_string); break; case WEBKIT_LOAD_FAILED: g_assert_not_reached(); break; case WEBKIT_LOAD_FINISHED: g_assert_cmpstr(webkit_web_view_get_uri(fixture->webView), ==, uri_string); g_main_loop_quit(fixture->loop); break; default: break; } g_free(uri_string); g_free(uri_string2); } static void load_error_test(WebKitWebView* webview, WebKitWebFrame* frame, const char* uri, GError* error) { g_debug("Error: %s", error->message); } static void test_loading_goback(WebLoadingFixture* fixture, gconstpointer data) { char* uri_string; g_signal_connect(fixture->webView, "notify::load-status", G_CALLBACK(load_goback_status_changed_cb), fixture); g_signal_connect(fixture->webView, "load-error", G_CALLBACK(load_error_test), fixture); uri_string = get_uri_for_path("/test_loading_status"); webkit_web_view_load_uri(fixture->webView, uri_string); g_free(uri_string); g_main_loop_run(fixture->loop); fixture->has_been_provisional = FALSE; fixture->has_been_committed = FALSE; fixture->has_been_first_visually_non_empty_layout = FALSE; fixture->has_been_finished = FALSE; fixture->has_been_failed = FALSE; fixture->has_been_load_error = FALSE; uri_string = get_uri_for_path("/test_loading_status2"); webkit_web_view_load_uri(fixture->webView, uri_string); g_free(uri_string); g_main_loop_run(fixture->loop); g_signal_handlers_disconnect_by_func(fixture->webView, load_goback_status_changed_cb, fixture); fixture->has_been_provisional = FALSE; fixture->has_been_committed = FALSE; fixture->has_been_first_visually_non_empty_layout = FALSE; fixture->has_been_finished = FALSE; fixture->has_been_failed = FALSE; fixture->has_been_load_error = FALSE; g_signal_connect(fixture->webView, "notify::load-status", G_CALLBACK(load_wentback_status_changed_cb), fixture); webkit_web_view_go_back(fixture->webView); g_main_loop_run(fixture->loop); g_signal_handlers_disconnect_by_func(fixture->webView, load_wentback_status_changed_cb, fixture); } int main(int argc, char** argv) { SoupServer* server; g_thread_init(NULL); gtk_test_init(&argc, &argv, NULL); server = soup_server_new(SOUP_SERVER_PORT, 0, NULL); soup_server_run_async(server); soup_server_add_handler(server, NULL, server_callback, NULL, NULL); base_uri = soup_uri_new("http://127.0.0.1/"); soup_uri_set_port(base_uri, soup_server_get_port(server)); g_test_bug_base("https://bugs.webkit.org/"); g_test_add("/webkit/loading/status", WebLoadingFixture, NULL, web_loading_fixture_setup, test_loading_status, web_loading_fixture_teardown); g_test_add("/webkit/loading/error", WebLoadingFixture, NULL, web_loading_fixture_setup, test_loading_error, web_loading_fixture_teardown); g_test_add("/webkit/loading/cancelled", WebLoadingFixture, NULL, web_loading_fixture_setup, test_loading_cancelled, web_loading_fixture_teardown); g_test_add("/webkit/loading/goback", WebLoadingFixture, NULL, web_loading_fixture_setup, test_loading_goback, web_loading_fixture_teardown); return g_test_run(); } #else int main(int argc, char** argv) { g_critical("You will need gtk-2.14.0 to run the unit tests. Doing nothing now."); return 0; } #endif