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
server_callback(SoupServer * server,SoupMessage * msg,const char * path,GHashTable * query,SoupClientContext * context,gpointer data)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
web_loading_fixture_setup(WebLoadingFixture * fixture,gconstpointer data)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
web_loading_fixture_teardown(WebLoadingFixture * fixture,gconstpointer data)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
get_uri_for_path(const char * path)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
load_finished_cb(WebKitWebView * web_view,WebKitWebFrame * web_frame,WebLoadingFixture * fixture)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
status_changed_cb(GObject * object,GParamSpec * pspec,WebLoadingFixture * fixture)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
test_loading_status(WebLoadingFixture * fixture,gconstpointer data)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
load_error_status_changed_cb(GObject * object,GParamSpec * pspec,WebLoadingFixture * fixture)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
load_error_cb(WebKitWebView * webView,WebKitWebFrame * frame,const char * uri,GError * error,WebLoadingFixture * fixture)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
test_loading_error(WebLoadingFixture * fixture,gconstpointer data)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
load_cancelled_cb(WebKitWebView * webView,WebKitWebFrame * frame,const char * uri,GError * error,WebLoadingFixture * fixture)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
stop_load(gpointer data)242 static gboolean stop_load (gpointer data)
243 {
244 webkit_web_view_stop_loading(WEBKIT_WEB_VIEW(data));
245 return FALSE;
246 }
247
load_cancelled_status_changed_cb(GObject * object,GParamSpec * pspec,WebLoadingFixture * fixture)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
test_loading_cancelled(WebLoadingFixture * fixture,gconstpointer data)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
load_goback_status_changed_cb(GObject * object,GParamSpec * pspec,WebLoadingFixture * fixture)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
load_wentback_status_changed_cb(GObject * object,GParamSpec * pspec,WebLoadingFixture * fixture)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
load_error_test(WebKitWebView * webview,WebKitWebFrame * frame,const char * uri,GError * error)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
test_loading_goback(WebLoadingFixture * fixture,gconstpointer data)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
main(int argc,char ** argv)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
main(int argc,char ** argv)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