Home | History | Annotate | Download | only in tests
      1 /*
      2  * Copyright (C) 2009, 2010 Gustavo Noronha Silva
      3  * Copyright (C) 2009 Igalia S.L.
      4  *
      5  * This library is free software; you can redistribute it and/or
      6  * modify it under the terms of the GNU Library General Public
      7  * License as published by the Free Software Foundation; either
      8  * version 2 of the License, or (at your option) any later version.
      9  *
     10  * This library is distributed in the hope that it will be useful,
     11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
     12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
     13  * Library General Public License for more details.
     14  *
     15  * You should have received a copy of the GNU Library General Public License
     16  * along with this library; see the file COPYING.LIB.  If not, write to
     17  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
     18  * Boston, MA 02110-1301, USA.
     19  */
     20 
     21 #include <gtk/gtk.h>
     22 #include <libsoup/soup.h>
     23 #include <string.h>
     24 #include <webkit/webkit.h>
     25 
     26 #if GTK_CHECK_VERSION(2, 14, 0)
     27 
     28 /* This string has to be rather big because of the cancelled test - it
     29  * looks like soup refuses to send or receive a too small chunk */
     30 #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>"
     31 
     32 SoupURI* base_uri;
     33 
     34 /* For real request testing */
     35 static void
     36 server_callback(SoupServer* server, SoupMessage* msg,
     37                 const char* path, GHashTable* query,
     38                 SoupClientContext* context, gpointer data)
     39 {
     40     if (msg->method != SOUP_METHOD_GET) {
     41         soup_message_set_status(msg, SOUP_STATUS_NOT_IMPLEMENTED);
     42         return;
     43     }
     44 
     45     soup_message_set_status(msg, SOUP_STATUS_OK);
     46 
     47     if (g_str_equal(path, "/test_loading_status") || g_str_equal(path, "/test_loading_status2"))
     48         soup_message_body_append(msg->response_body, SOUP_MEMORY_STATIC, HTML_STRING, strlen(HTML_STRING));
     49     else if (g_str_equal(path, "/test_load_error")) {
     50         soup_message_set_status(msg, SOUP_STATUS_CANT_CONNECT);
     51     } else if (g_str_equal(path, "/test_loading_cancelled")) {
     52         soup_message_headers_set_encoding(msg->response_headers, SOUP_ENCODING_CHUNKED);
     53         soup_message_body_append(msg->response_body, SOUP_MEMORY_STATIC, HTML_STRING, strlen(HTML_STRING));
     54         soup_server_unpause_message(server, msg);
     55         return;
     56     }
     57 
     58     soup_message_body_complete(msg->response_body);
     59 }
     60 
     61 typedef struct {
     62     WebKitWebView* webView;
     63     GMainLoop *loop;
     64     gboolean has_been_provisional;
     65     gboolean has_been_committed;
     66     gboolean has_been_first_visually_non_empty_layout;
     67     gboolean has_been_finished;
     68     gboolean has_been_failed;
     69     gboolean has_been_load_error;
     70 } WebLoadingFixture;
     71 
     72 static void web_loading_fixture_setup(WebLoadingFixture* fixture, gconstpointer data)
     73 {
     74     fixture->webView = WEBKIT_WEB_VIEW(webkit_web_view_new());
     75     fixture->loop = g_main_loop_new(NULL, TRUE);
     76     g_object_ref_sink(fixture->webView);
     77     fixture->has_been_provisional = FALSE;
     78     fixture->has_been_committed = FALSE;
     79     fixture->has_been_first_visually_non_empty_layout = FALSE;
     80     fixture->has_been_finished = FALSE;
     81     fixture->has_been_failed = FALSE;
     82     fixture->has_been_load_error = FALSE;
     83 }
     84 
     85 static void web_loading_fixture_teardown(WebLoadingFixture* fixture, gconstpointer data)
     86 {
     87     g_object_unref(fixture->webView);
     88     g_main_loop_unref(fixture->loop);
     89 }
     90 
     91 static char* get_uri_for_path(const char* path)
     92 {
     93     SoupURI* uri;
     94     char* uri_string;
     95 
     96     uri = soup_uri_new_with_base(base_uri, path);
     97     uri_string = soup_uri_to_string(uri, FALSE);
     98     soup_uri_free (uri);
     99 
    100     return uri_string;
    101 }
    102 
    103 static void load_finished_cb(WebKitWebView* web_view, WebKitWebFrame* web_frame, WebLoadingFixture* fixture)
    104 {
    105     g_assert(fixture->has_been_provisional);
    106     g_assert(fixture->has_been_committed);
    107     g_assert(fixture->has_been_first_visually_non_empty_layout);
    108 
    109     g_main_loop_quit(fixture->loop);
    110 }
    111 
    112 
    113 static void status_changed_cb(GObject* object, GParamSpec* pspec, WebLoadingFixture* fixture)
    114 {
    115     WebKitLoadStatus status = webkit_web_view_get_load_status(WEBKIT_WEB_VIEW(object));
    116 
    117     switch (status) {
    118     case WEBKIT_LOAD_PROVISIONAL:
    119         g_assert(!fixture->has_been_provisional);
    120         g_assert(!fixture->has_been_committed);
    121         g_assert(!fixture->has_been_first_visually_non_empty_layout);
    122         fixture->has_been_provisional = TRUE;
    123         break;
    124     case WEBKIT_LOAD_COMMITTED:
    125         g_assert(fixture->has_been_provisional);
    126         g_assert(!fixture->has_been_committed);
    127         g_assert(!fixture->has_been_first_visually_non_empty_layout);
    128         fixture->has_been_committed = TRUE;
    129         break;
    130     case WEBKIT_LOAD_FIRST_VISUALLY_NON_EMPTY_LAYOUT:
    131         g_assert(fixture->has_been_provisional);
    132         g_assert(fixture->has_been_committed);
    133         g_assert(!fixture->has_been_first_visually_non_empty_layout);
    134         fixture->has_been_first_visually_non_empty_layout = TRUE;
    135         break;
    136     case WEBKIT_LOAD_FINISHED:
    137         g_assert(fixture->has_been_provisional);
    138         g_assert(fixture->has_been_committed);
    139         g_assert(fixture->has_been_first_visually_non_empty_layout);
    140         break;
    141     default:
    142         g_assert_not_reached();
    143     }
    144 }
    145 
    146 static void test_loading_status(WebLoadingFixture* fixture, gconstpointer data)
    147 {
    148     char* uri_string;
    149 
    150     g_assert_cmpint(webkit_web_view_get_load_status(fixture->webView), ==, WEBKIT_LOAD_PROVISIONAL);
    151 
    152     g_object_connect(G_OBJECT(fixture->webView),
    153                      "signal::notify::load-status", G_CALLBACK(status_changed_cb), fixture,
    154                      "signal::load-finished", G_CALLBACK(load_finished_cb), fixture,
    155                      NULL);
    156 
    157     uri_string = get_uri_for_path("/test_loading_status");
    158 
    159     /* load_uri will trigger the navigation-policy-decision-requested
    160      * signal emission;
    161      */
    162     webkit_web_view_load_uri(fixture->webView, uri_string);
    163     g_free(uri_string);
    164 
    165     g_main_loop_run(fixture->loop);
    166 }
    167 
    168 static void load_error_status_changed_cb(GObject* object, GParamSpec* pspec, WebLoadingFixture* fixture)
    169 {
    170     WebKitLoadStatus status = webkit_web_view_get_load_status(WEBKIT_WEB_VIEW(object));
    171 
    172     switch(status) {
    173     case WEBKIT_LOAD_PROVISIONAL:
    174         g_assert(!fixture->has_been_provisional);
    175         fixture->has_been_provisional = TRUE;
    176         break;
    177     case WEBKIT_LOAD_COMMITTED:
    178         g_assert(!fixture->has_been_committed);
    179         fixture->has_been_committed = TRUE;
    180         break;
    181     case WEBKIT_LOAD_FINISHED:
    182         g_assert(fixture->has_been_provisional);
    183         g_assert(fixture->has_been_load_error);
    184         g_assert(fixture->has_been_failed);
    185         g_assert(!fixture->has_been_finished);
    186         fixture->has_been_finished = TRUE;
    187         break;
    188     case WEBKIT_LOAD_FAILED:
    189         g_assert(!fixture->has_been_failed);
    190         fixture->has_been_failed = TRUE;
    191         g_main_loop_quit(fixture->loop);
    192         break;
    193     default:
    194         break;
    195     }
    196 }
    197 
    198 static gboolean load_error_cb(WebKitWebView* webView, WebKitWebFrame* frame, const char* uri, GError *error, WebLoadingFixture* fixture)
    199 {
    200     g_assert(fixture->has_been_provisional);
    201     g_assert(!fixture->has_been_load_error);
    202     fixture->has_been_load_error = TRUE;
    203 
    204     return FALSE;
    205 }
    206 
    207 static void test_loading_error(WebLoadingFixture* fixture, gconstpointer data)
    208 {
    209     char* uri_string;
    210 
    211     g_test_bug("28842");
    212 
    213     g_signal_connect(fixture->webView, "load-error", G_CALLBACK(load_error_cb), fixture);
    214     g_signal_connect(fixture->webView, "notify::load-status", G_CALLBACK(load_error_status_changed_cb), fixture);
    215 
    216     uri_string = get_uri_for_path("/test_load_error");
    217     webkit_web_view_load_uri(fixture->webView, uri_string);
    218     g_free(uri_string);
    219 
    220     g_main_loop_run(fixture->loop);
    221 
    222     g_assert(fixture->has_been_provisional);
    223     g_assert(!fixture->has_been_committed);
    224     g_assert(fixture->has_been_load_error);
    225     g_assert(fixture->has_been_failed);
    226     g_assert(!fixture->has_been_finished);
    227 }
    228 
    229 /* Cancelled load */
    230 
    231 static gboolean load_cancelled_cb(WebKitWebView* webView, WebKitWebFrame* frame, const char* uri, GError *error, WebLoadingFixture* fixture)
    232 {
    233     g_assert(fixture->has_been_provisional);
    234     g_assert(fixture->has_been_failed);
    235     g_assert(!fixture->has_been_load_error);
    236     g_assert(error->code == WEBKIT_NETWORK_ERROR_CANCELLED);
    237     fixture->has_been_load_error = TRUE;
    238 
    239     return TRUE;
    240 }
    241 
    242 static gboolean stop_load (gpointer data)
    243 {
    244     webkit_web_view_stop_loading(WEBKIT_WEB_VIEW(data));
    245     return FALSE;
    246 }
    247 
    248 static void load_cancelled_status_changed_cb(GObject* object, GParamSpec* pspec, WebLoadingFixture* fixture)
    249 {
    250     WebKitLoadStatus status = webkit_web_view_get_load_status(WEBKIT_WEB_VIEW(object));
    251 
    252     switch(status) {
    253     case WEBKIT_LOAD_PROVISIONAL:
    254         g_assert(!fixture->has_been_provisional);
    255         g_assert(!fixture->has_been_failed);
    256         fixture->has_been_provisional = TRUE;
    257         break;
    258     case WEBKIT_LOAD_COMMITTED:
    259         g_idle_add (stop_load, object);
    260         break;
    261     case WEBKIT_LOAD_FAILED:
    262         g_assert(fixture->has_been_provisional);
    263         g_assert(!fixture->has_been_failed);
    264         g_assert(!fixture->has_been_load_error);
    265         fixture->has_been_failed = TRUE;
    266         g_main_loop_quit(fixture->loop);
    267         break;
    268     case WEBKIT_LOAD_FINISHED:
    269         g_assert_not_reached();
    270         break;
    271     default:
    272         break;
    273     }
    274 }
    275 
    276 static void test_loading_cancelled(WebLoadingFixture* fixture, gconstpointer data)
    277 {
    278     char* uri_string;
    279 
    280     g_test_bug("29644");
    281 
    282     g_signal_connect(fixture->webView, "load-error", G_CALLBACK(load_cancelled_cb), fixture);
    283     g_signal_connect(fixture->webView, "notify::load-status", G_CALLBACK(load_cancelled_status_changed_cb), fixture);
    284 
    285     uri_string = get_uri_for_path("/test_loading_cancelled");
    286     webkit_web_view_load_uri(fixture->webView, uri_string);
    287     g_free(uri_string);
    288 
    289     g_main_loop_run(fixture->loop);
    290 }
    291 
    292 static void load_goback_status_changed_cb(GObject* object, GParamSpec* pspec, WebLoadingFixture* fixture)
    293 {
    294     WebKitLoadStatus status = webkit_web_view_get_load_status(WEBKIT_WEB_VIEW(object));
    295 
    296     switch(status) {
    297     case WEBKIT_LOAD_PROVISIONAL:
    298         g_assert(!fixture->has_been_provisional);
    299         fixture->has_been_provisional = TRUE;
    300         break;
    301     case WEBKIT_LOAD_COMMITTED:
    302         g_assert(fixture->has_been_provisional);
    303         fixture->has_been_committed = TRUE;
    304         break;
    305     case WEBKIT_LOAD_FAILED:
    306         g_assert_not_reached();
    307         break;
    308     case WEBKIT_LOAD_FINISHED:
    309         g_assert(fixture->has_been_provisional);
    310         g_assert(fixture->has_been_committed);
    311         fixture->has_been_finished = TRUE;
    312         g_main_loop_quit(fixture->loop);
    313         break;
    314     default:
    315         break;
    316     }
    317 }
    318 
    319 static void load_wentback_status_changed_cb(GObject* object, GParamSpec* pspec, WebLoadingFixture* fixture)
    320 {
    321     WebKitLoadStatus status = webkit_web_view_get_load_status(WEBKIT_WEB_VIEW(object));
    322     char* uri_string;
    323     char* uri_string2;
    324 
    325     uri_string = get_uri_for_path("/test_loading_status");
    326     uri_string2 = get_uri_for_path("/test_loading_status2");
    327 
    328     switch(status) {
    329     case WEBKIT_LOAD_PROVISIONAL:
    330         g_assert_cmpstr(webkit_web_view_get_uri(fixture->webView), ==, uri_string2);
    331         break;
    332     case WEBKIT_LOAD_COMMITTED:
    333         g_assert_cmpstr(webkit_web_view_get_uri(fixture->webView), ==, uri_string);
    334         break;
    335     case WEBKIT_LOAD_FAILED:
    336         g_assert_not_reached();
    337         break;
    338     case WEBKIT_LOAD_FINISHED:
    339         g_assert_cmpstr(webkit_web_view_get_uri(fixture->webView), ==, uri_string);
    340         g_main_loop_quit(fixture->loop);
    341         break;
    342     default:
    343         break;
    344     }
    345 
    346     g_free(uri_string);
    347     g_free(uri_string2);
    348 }
    349 
    350 static void load_error_test(WebKitWebView* webview, WebKitWebFrame* frame, const char* uri, GError* error)
    351 {
    352     g_debug("Error: %s", error->message);
    353 }
    354 
    355 static void test_loading_goback(WebLoadingFixture* fixture, gconstpointer data)
    356 {
    357     char* uri_string;
    358 
    359     g_signal_connect(fixture->webView, "notify::load-status", G_CALLBACK(load_goback_status_changed_cb), fixture);
    360 
    361     g_signal_connect(fixture->webView, "load-error", G_CALLBACK(load_error_test), fixture);
    362 
    363     uri_string = get_uri_for_path("/test_loading_status");
    364     webkit_web_view_load_uri(fixture->webView, uri_string);
    365     g_free(uri_string);
    366 
    367     g_main_loop_run(fixture->loop);
    368 
    369     fixture->has_been_provisional = FALSE;
    370     fixture->has_been_committed = FALSE;
    371     fixture->has_been_first_visually_non_empty_layout = FALSE;
    372     fixture->has_been_finished = FALSE;
    373     fixture->has_been_failed = FALSE;
    374     fixture->has_been_load_error = FALSE;
    375 
    376     uri_string = get_uri_for_path("/test_loading_status2");
    377     webkit_web_view_load_uri(fixture->webView, uri_string);
    378     g_free(uri_string);
    379 
    380     g_main_loop_run(fixture->loop);
    381 
    382     g_signal_handlers_disconnect_by_func(fixture->webView, load_goback_status_changed_cb, fixture);
    383 
    384     fixture->has_been_provisional = FALSE;
    385     fixture->has_been_committed = FALSE;
    386     fixture->has_been_first_visually_non_empty_layout = FALSE;
    387     fixture->has_been_finished = FALSE;
    388     fixture->has_been_failed = FALSE;
    389     fixture->has_been_load_error = FALSE;
    390 
    391     g_signal_connect(fixture->webView, "notify::load-status", G_CALLBACK(load_wentback_status_changed_cb), fixture);
    392     webkit_web_view_go_back(fixture->webView);
    393 
    394     g_main_loop_run(fixture->loop);
    395 
    396     g_signal_handlers_disconnect_by_func(fixture->webView, load_wentback_status_changed_cb, fixture);
    397 }
    398 
    399 int main(int argc, char** argv)
    400 {
    401     SoupServer* server;
    402 
    403     g_thread_init(NULL);
    404     gtk_test_init(&argc, &argv, NULL);
    405 
    406     server = soup_server_new(SOUP_SERVER_PORT, 0, NULL);
    407     soup_server_run_async(server);
    408 
    409     soup_server_add_handler(server, NULL, server_callback, NULL, NULL);
    410 
    411     base_uri = soup_uri_new("http://127.0.0.1/");
    412     soup_uri_set_port(base_uri, soup_server_get_port(server));
    413 
    414     g_test_bug_base("https://bugs.webkit.org/");
    415     g_test_add("/webkit/loading/status",
    416                WebLoadingFixture, NULL,
    417                web_loading_fixture_setup,
    418                test_loading_status,
    419                web_loading_fixture_teardown);
    420     g_test_add("/webkit/loading/error",
    421                WebLoadingFixture, NULL,
    422                web_loading_fixture_setup,
    423                test_loading_error,
    424                web_loading_fixture_teardown);
    425     g_test_add("/webkit/loading/cancelled",
    426                WebLoadingFixture, NULL,
    427                web_loading_fixture_setup,
    428                test_loading_cancelled,
    429                web_loading_fixture_teardown);
    430     g_test_add("/webkit/loading/goback",
    431                WebLoadingFixture, NULL,
    432                web_loading_fixture_setup,
    433                test_loading_goback,
    434                web_loading_fixture_teardown);
    435     return g_test_run();
    436 }
    437 
    438 #else
    439 int main(int argc, char** argv)
    440 {
    441     g_critical("You will need gtk-2.14.0 to run the unit tests. Doing nothing now.");
    442     return 0;
    443 }
    444 
    445 #endif
    446