/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ /* * Copyright 2007-2012 Red Hat, Inc. */ #include "test-utils.h" SoupServer *server, *ssl_server; SoupURI *base_uri, *ssl_base_uri; static gboolean auth_callback (SoupAuthDomain *auth_domain, SoupMessage *msg, const char *username, const char *password, gpointer data) { return !strcmp (username, "user") && !strcmp (password, "password"); } static gboolean timeout_finish_message (gpointer msg) { SoupServer *server = g_object_get_data (G_OBJECT (msg), "server"); soup_server_unpause_message (server, msg); return FALSE; } static void server_callback (SoupServer *server, SoupMessage *msg, const char *path, GHashTable *query, SoupClientContext *context, gpointer data) { SoupURI *uri = soup_message_get_uri (msg); const char *server_protocol = data; soup_message_headers_append (msg->response_headers, "X-Handled-By", "server_callback"); if (!strcmp (path, "*")) { debug_printf (1, " default server_callback got request for '*'!\n"); errors++; soup_message_set_status (msg, SOUP_STATUS_INTERNAL_SERVER_ERROR); return; } if (msg->method != SOUP_METHOD_GET && msg->method != SOUP_METHOD_POST) { soup_message_set_status (msg, SOUP_STATUS_NOT_IMPLEMENTED); return; } if (!strcmp (path, "/redirect")) { soup_message_set_redirect (msg, SOUP_STATUS_FOUND, "/"); return; } if (!strcmp (path, "/alias-redirect")) { SoupURI *redirect_uri; char *redirect_string; const char *redirect_protocol; redirect_protocol = soup_message_headers_get_one (msg->request_headers, "X-Redirect-Protocol"); redirect_uri = soup_uri_copy (uri); soup_uri_set_scheme (redirect_uri, "foo"); if (!g_strcmp0 (redirect_protocol, "https")) soup_uri_set_port (redirect_uri, ssl_base_uri->port); else soup_uri_set_port (redirect_uri, base_uri->port); soup_uri_set_path (redirect_uri, "/alias-redirected"); redirect_string = soup_uri_to_string (redirect_uri, FALSE); soup_message_set_redirect (msg, SOUP_STATUS_FOUND, redirect_string); g_free (redirect_string); soup_uri_free (redirect_uri); return; } else if (!strcmp (path, "/alias-redirected")) { soup_message_set_status (msg, SOUP_STATUS_OK); soup_message_headers_append (msg->response_headers, "X-Redirected-Protocol", server_protocol); return; } if (!strcmp (path, "/slow")) { soup_server_pause_message (server, msg); g_object_set_data (G_OBJECT (msg), "server", server); soup_add_timeout (soup_server_get_async_context (server), 1000, timeout_finish_message, msg); } soup_message_set_status (msg, SOUP_STATUS_OK); if (!strcmp (uri->host, "foo")) { soup_message_set_response (msg, "text/plain", SOUP_MEMORY_STATIC, "foo-index", 9); return; } else { soup_message_set_response (msg, "text/plain", SOUP_MEMORY_STATIC, "index", 5); return; } } static void server_star_callback (SoupServer *server, SoupMessage *msg, const char *path, GHashTable *query, SoupClientContext *context, gpointer data) { soup_message_headers_append (msg->response_headers, "X-Handled-By", "star_callback"); if (strcmp (path, "*") != 0) { debug_printf (1, " server_star_callback got request for '%s'!\n", path); errors++; soup_message_set_status (msg, SOUP_STATUS_INTERNAL_SERVER_ERROR); return; } if (msg->method != SOUP_METHOD_OPTIONS) { soup_message_set_status (msg, SOUP_STATUS_METHOD_NOT_ALLOWED); return; } soup_message_set_status (msg, SOUP_STATUS_OK); } /* Host header handling: client must be able to override the default * value, server must be able to recognize different Host values. * #539803. */ static void do_host_test (void) { SoupSession *session; SoupMessage *one, *two; debug_printf (1, "Host handling\n"); session = soup_test_session_new (SOUP_TYPE_SESSION_SYNC, NULL); one = soup_message_new_from_uri ("GET", base_uri); two = soup_message_new_from_uri ("GET", base_uri); soup_message_headers_replace (two->request_headers, "Host", "foo"); soup_session_send_message (session, one); soup_session_send_message (session, two); soup_test_session_abort_unref (session); if (!SOUP_STATUS_IS_SUCCESSFUL (one->status_code)) { debug_printf (1, " Message 1 failed: %d %s\n", one->status_code, one->reason_phrase); errors++; } else if (strcmp (one->response_body->data, "index") != 0) { debug_printf (1, " Unexpected response to message 1: '%s'\n", one->response_body->data); errors++; } g_object_unref (one); if (!SOUP_STATUS_IS_SUCCESSFUL (two->status_code)) { debug_printf (1, " Message 2 failed: %d %s\n", two->status_code, two->reason_phrase); errors++; } else if (strcmp (two->response_body->data, "foo-index") != 0) { debug_printf (1, " Unexpected response to message 2: '%s'\n", two->response_body->data); errors++; } g_object_unref (two); } /* Dropping the application's ref on the session from a callback * should not cause the session to be freed at an incorrect time. * (This test will crash if it fails.) #533473 */ static void cu_one_completed (SoupSession *session, SoupMessage *msg, gpointer loop) { debug_printf (2, " Message 1 completed\n"); if (msg->status_code != SOUP_STATUS_CANT_CONNECT) { debug_printf (1, " Unexpected status on Message 1: %d %s\n", msg->status_code, msg->reason_phrase); errors++; } g_object_unref (session); } static gboolean cu_idle_quit (gpointer loop) { g_main_loop_quit (loop); return FALSE; } static void cu_two_completed (SoupSession *session, SoupMessage *msg, gpointer loop) { debug_printf (2, " Message 2 completed\n"); if (msg->status_code != SOUP_STATUS_CANT_CONNECT) { debug_printf (1, " Unexpected status on Message 2: %d %s\n", msg->status_code, msg->reason_phrase); errors++; } g_idle_add (cu_idle_quit, loop); } static void do_callback_unref_test (void) { SoupServer *bad_server; SoupAddress *addr; SoupSession *session; SoupMessage *one, *two; GMainLoop *loop; char *bad_uri; debug_printf (1, "\nCallback unref handling (msg api)\n"); /* Get a guaranteed-bad URI */ addr = soup_address_new ("127.0.0.1", SOUP_ADDRESS_ANY_PORT); soup_address_resolve_sync (addr, NULL); bad_server = soup_server_new (SOUP_SERVER_INTERFACE, addr, NULL); g_object_unref (addr); bad_uri = g_strdup_printf ("http://127.0.0.1:%u/", soup_server_get_port (bad_server)); g_object_unref (bad_server); session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, NULL); g_object_add_weak_pointer (G_OBJECT (session), (gpointer *)&session); loop = g_main_loop_new (NULL, TRUE); one = soup_message_new ("GET", bad_uri); g_object_add_weak_pointer (G_OBJECT (one), (gpointer *)&one); two = soup_message_new ("GET", bad_uri); g_object_add_weak_pointer (G_OBJECT (two), (gpointer *)&two); g_free (bad_uri); soup_session_queue_message (session, one, cu_one_completed, loop); soup_session_queue_message (session, two, cu_two_completed, loop); g_main_loop_run (loop); g_main_loop_unref (loop); if (session) { g_object_remove_weak_pointer (G_OBJECT (session), (gpointer *)&session); debug_printf (1, " Session not destroyed?\n"); errors++; g_object_unref (session); } if (one) { g_object_remove_weak_pointer (G_OBJECT (one), (gpointer *)&one); debug_printf (1, " Message 1 not destroyed?\n"); errors++; g_object_unref (one); } if (two) { g_object_remove_weak_pointer (G_OBJECT (two), (gpointer *)&two); debug_printf (1, " Message 2 not destroyed?\n"); errors++; g_object_unref (two); } /* Otherwise, if we haven't crashed, we're ok. */ } static void cur_one_completed (GObject *source, GAsyncResult *result, gpointer session) { SoupRequest *one = SOUP_REQUEST (source); GError *error = NULL; debug_printf (2, " Request 1 completed\n"); if (soup_request_send_finish (one, result, &error)) { debug_printf (1, " Request 1 succeeded?\n"); errors++; } else if (!g_error_matches (error, SOUP_HTTP_ERROR, SOUP_STATUS_CANT_CONNECT)) { debug_printf (1, " Unexpected error on Request 1: %s\n", error->message); errors++; } g_clear_error (&error); g_object_unref (session); } static gboolean cur_idle_quit (gpointer loop) { g_main_loop_quit (loop); return FALSE; } static void cur_two_completed (GObject *source, GAsyncResult *result, gpointer loop) { SoupRequest *two = SOUP_REQUEST (source); GError *error = NULL; debug_printf (2, " Request 2 completed\n"); if (soup_request_send_finish (two, result, &error)) { debug_printf (1, " Request 2 succeeded?\n"); errors++; } else if (!g_error_matches (error, SOUP_HTTP_ERROR, SOUP_STATUS_CANT_CONNECT)) { debug_printf (1, " Unexpected error on Request 2: %s\n", error->message); errors++; } g_clear_error (&error); g_idle_add (cur_idle_quit, loop); } static void do_callback_unref_req_test (void) { SoupServer *bad_server; SoupAddress *addr; SoupSession *session; SoupRequest *one, *two; GMainLoop *loop; char *bad_uri; debug_printf (1, "\nCallback unref handling (request api)\n"); /* Get a guaranteed-bad URI */ addr = soup_address_new ("127.0.0.1", SOUP_ADDRESS_ANY_PORT); soup_address_resolve_sync (addr, NULL); bad_server = soup_server_new (SOUP_SERVER_INTERFACE, addr, NULL); g_object_unref (addr); bad_uri = g_strdup_printf ("http://127.0.0.1:%u/", soup_server_get_port (bad_server)); g_object_unref (bad_server); session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, SOUP_SESSION_USE_THREAD_CONTEXT, TRUE, NULL); g_object_add_weak_pointer (G_OBJECT (session), (gpointer *)&session); loop = g_main_loop_new (NULL, TRUE); one = soup_session_request (session, bad_uri, NULL); g_object_add_weak_pointer (G_OBJECT (one), (gpointer *)&one); two = soup_session_request (session, bad_uri, NULL); g_object_add_weak_pointer (G_OBJECT (two), (gpointer *)&two); g_free (bad_uri); soup_request_send_async (one, NULL, cur_one_completed, session); g_object_unref (one); soup_request_send_async (two, NULL, cur_two_completed, loop); g_object_unref (two); g_main_loop_run (loop); g_main_loop_unref (loop); if (session) { g_object_remove_weak_pointer (G_OBJECT (session), (gpointer *)&session); debug_printf (1, " Session not destroyed?\n"); errors++; g_object_unref (session); } if (one) { g_object_remove_weak_pointer (G_OBJECT (one), (gpointer *)&one); debug_printf (1, " Request 1 not destroyed?\n"); errors++; g_object_unref (one); } if (two) { g_object_remove_weak_pointer (G_OBJECT (two), (gpointer *)&two); debug_printf (1, " Request 2 not destroyed?\n"); errors++; g_object_unref (two); } /* Otherwise, if we haven't crashed, we're ok. */ } /* SoupSession should clean up all signal handlers on a message after * it is finished, allowing the message to be reused if desired. * #559054 */ static void ensure_no_signal_handlers (SoupMessage *msg, guint *signal_ids, guint n_signal_ids) { int i; for (i = 0; i < n_signal_ids; i++) { if (g_signal_handler_find (msg, G_SIGNAL_MATCH_ID, signal_ids[i], 0, NULL, NULL, NULL)) { debug_printf (1, " Message has handler for '%s'\n", g_signal_name (signal_ids[i])); errors++; } } } static void reuse_test_authenticate (SoupSession *session, SoupMessage *msg, SoupAuth *auth, gboolean retrying) { /* Get it wrong the first time, then succeed */ if (!retrying) soup_auth_authenticate (auth, "user", "wrong password"); else soup_auth_authenticate (auth, "user", "password"); } static void do_msg_reuse_test (void) { SoupSession *session; SoupMessage *msg; SoupURI *uri; guint *signal_ids, n_signal_ids; debug_printf (1, "\nSoupMessage reuse\n"); signal_ids = g_signal_list_ids (SOUP_TYPE_MESSAGE, &n_signal_ids); session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, NULL); g_signal_connect (session, "authenticate", G_CALLBACK (reuse_test_authenticate), NULL); debug_printf (1, " First message\n"); msg = soup_message_new_from_uri ("GET", base_uri); soup_session_send_message (session, msg); ensure_no_signal_handlers (msg, signal_ids, n_signal_ids); debug_printf (1, " Redirect message\n"); uri = soup_uri_new_with_base (base_uri, "/redirect"); soup_message_set_uri (msg, uri); soup_uri_free (uri); soup_session_send_message (session, msg); if (!soup_uri_equal (soup_message_get_uri (msg), base_uri)) { debug_printf (1, " Message did not get redirected!\n"); errors++; } ensure_no_signal_handlers (msg, signal_ids, n_signal_ids); debug_printf (1, " Auth message\n"); uri = soup_uri_new_with_base (base_uri, "/auth"); soup_message_set_uri (msg, uri); soup_uri_free (uri); soup_session_send_message (session, msg); if (!SOUP_STATUS_IS_SUCCESSFUL (msg->status_code)) { debug_printf (1, " Message did not get authenticated!\n"); errors++; } ensure_no_signal_handlers (msg, signal_ids, n_signal_ids); /* One last try to make sure the auth stuff got cleaned up */ debug_printf (1, " Last message\n"); soup_message_set_uri (msg, base_uri); soup_session_send_message (session, msg); ensure_no_signal_handlers (msg, signal_ids, n_signal_ids); soup_test_session_abort_unref (session); g_object_unref (msg); g_free (signal_ids); } /* Server handlers for "*" work but are separate from handlers for * all other URIs. #590751 */ static void do_star_test (void) { SoupSession *session; SoupMessage *msg; SoupURI *star_uri; const char *handled_by; debug_printf (1, "\nOPTIONS *\n"); session = soup_test_session_new (SOUP_TYPE_SESSION_SYNC, NULL); star_uri = soup_uri_copy (base_uri); soup_uri_set_path (star_uri, "*"); debug_printf (1, " Testing with no handler\n"); msg = soup_message_new_from_uri ("OPTIONS", star_uri); soup_session_send_message (session, msg); if (msg->status_code != SOUP_STATUS_NOT_FOUND) { debug_printf (1, " Unexpected response: %d %s\n", msg->status_code, msg->reason_phrase); errors++; } handled_by = soup_message_headers_get_one (msg->response_headers, "X-Handled-By"); if (handled_by) { /* Should have been rejected by SoupServer directly */ debug_printf (1, " Message reached handler '%s'\n", handled_by); errors++; } g_object_unref (msg); soup_server_add_handler (server, "*", server_star_callback, NULL, NULL); debug_printf (1, " Testing with handler\n"); msg = soup_message_new_from_uri ("OPTIONS", star_uri); soup_session_send_message (session, msg); if (msg->status_code != SOUP_STATUS_OK) { debug_printf (1, " Unexpected response: %d %s\n", msg->status_code, msg->reason_phrase); errors++; } handled_by = soup_message_headers_get_one (msg->response_headers, "X-Handled-By"); if (!handled_by) { debug_printf (1, " Message did not reach handler!\n"); errors++; } else if (strcmp (handled_by, "star_callback") != 0) { debug_printf (1, " Message reached incorrect handler '%s'\n", handled_by); errors++; } g_object_unref (msg); soup_test_session_abort_unref (session); soup_uri_free (star_uri); } /* Handle unexpectedly-early aborts. #596074, #618641 */ static void ea_msg_completed_one (SoupSession *session, SoupMessage *msg, gpointer loop) { debug_printf (2, " Message 1 completed\n"); if (msg->status_code != SOUP_STATUS_CANCELLED) { debug_printf (1, " Unexpected status on Message 1: %d %s\n", msg->status_code, msg->reason_phrase); errors++; } g_main_loop_quit (loop); } static gboolean ea_abort_session (gpointer session) { soup_session_abort (session); return FALSE; } static void ea_connection_state_changed (GObject *conn, GParamSpec *pspec, gpointer session) { SoupConnectionState state; g_object_get (conn, "state", &state, NULL); if (state == SOUP_CONNECTION_CONNECTING) { g_idle_add_full (G_PRIORITY_HIGH, ea_abort_session, session, NULL); g_signal_handlers_disconnect_by_func (conn, ea_connection_state_changed, session); } } static void ea_connection_created (SoupSession *session, GObject *conn, gpointer user_data) { g_signal_connect (conn, "notify::state", G_CALLBACK (ea_connection_state_changed), session); g_signal_handlers_disconnect_by_func (session, ea_connection_created, user_data); } static void ea_request_started (SoupSession *session, SoupMessage *msg, SoupSocket *socket, gpointer user_data) { soup_session_cancel_message (session, msg, SOUP_STATUS_CANCELLED); } static void do_early_abort_test (void) { SoupSession *session; SoupMessage *msg; GMainContext *context; GMainLoop *loop; debug_printf (1, "\nAbort with pending connection (msg api)\n"); session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, NULL); msg = soup_message_new_from_uri ("GET", base_uri); context = g_main_context_default (); loop = g_main_loop_new (context, TRUE); soup_session_queue_message (session, msg, ea_msg_completed_one, loop); g_main_context_iteration (context, FALSE); soup_session_abort (session); while (g_main_context_pending (context)) g_main_context_iteration (context, FALSE); g_main_loop_unref (loop); soup_test_session_abort_unref (session); session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, NULL); msg = soup_message_new_from_uri ("GET", base_uri); g_signal_connect (session, "connection-created", G_CALLBACK (ea_connection_created), NULL); soup_session_send_message (session, msg); debug_printf (2, " Message 2 completed\n"); if (msg->status_code != SOUP_STATUS_CANCELLED) { debug_printf (1, " Unexpected response: %d %s\n", msg->status_code, msg->reason_phrase); errors++; } g_object_unref (msg); while (g_main_context_pending (context)) g_main_context_iteration (context, FALSE); soup_test_session_abort_unref (session); session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, NULL); msg = soup_message_new_from_uri ("GET", base_uri); g_signal_connect (session, "request-started", G_CALLBACK (ea_request_started), NULL); soup_session_send_message (session, msg); debug_printf (2, " Message 3 completed\n"); if (msg->status_code != SOUP_STATUS_CANCELLED) { debug_printf (1, " Unexpected response: %d %s\n", msg->status_code, msg->reason_phrase); errors++; } g_object_unref (msg); while (g_main_context_pending (context)) g_main_context_iteration (context, FALSE); soup_test_session_abort_unref (session); } static void ear_one_completed (GObject *source, GAsyncResult *result, gpointer user_data) { GError *error = NULL; debug_printf (2, " Request 1 completed\n"); if (soup_request_send_finish (SOUP_REQUEST (source), result, &error)) { debug_printf (1, " Request 1 succeeded?\n"); errors++; } else if (!g_error_matches (error, SOUP_HTTP_ERROR, SOUP_STATUS_CANCELLED)) { debug_printf (1, " Unexpected error on Request 1: %s\n", error->message); errors++; } g_clear_error (&error); } static void ear_two_completed (GObject *source, GAsyncResult *result, gpointer loop) { GError *error = NULL; debug_printf (2, " Request 2 completed\n"); if (soup_request_send_finish (SOUP_REQUEST (source), result, &error)) { debug_printf (1, " Request 2 succeeded?\n"); errors++; } else if (!g_error_matches (error, SOUP_HTTP_ERROR, SOUP_STATUS_CANCELLED)) { debug_printf (1, " Unexpected error on Request 2: %s\n", error->message); errors++; } g_clear_error (&error); g_main_loop_quit (loop); } static void ear_three_completed (GObject *source, GAsyncResult *result, gpointer loop) { GError *error = NULL; debug_printf (2, " Request 3 completed\n"); if (soup_request_send_finish (SOUP_REQUEST (source), result, &error)) { debug_printf (1, " Request 3 succeeded?\n"); errors++; } else if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED)) { debug_printf (1, " Unexpected error on Request 3: %s\n", error->message); errors++; } g_clear_error (&error); g_main_loop_quit (loop); } static void ear_request_started (SoupSession *session, SoupMessage *msg, SoupSocket *socket, gpointer cancellable) { g_cancellable_cancel (cancellable); } static void do_early_abort_req_test (void) { SoupSession *session; SoupRequest *req; GMainContext *context; GMainLoop *loop; GCancellable *cancellable; debug_printf (1, "\nAbort with pending connection (request api)\n"); session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, SOUP_SESSION_USE_THREAD_CONTEXT, TRUE, NULL); req = soup_session_request_uri (session, base_uri, NULL); context = g_main_context_default (); loop = g_main_loop_new (context, TRUE); soup_request_send_async (req, NULL, ear_one_completed, NULL); g_object_unref (req); g_main_context_iteration (context, FALSE); soup_session_abort (session); while (g_main_context_pending (context)) g_main_context_iteration (context, FALSE); soup_test_session_abort_unref (session); session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, SOUP_SESSION_USE_THREAD_CONTEXT, TRUE, NULL); req = soup_session_request_uri (session, base_uri, NULL); g_signal_connect (session, "connection-created", G_CALLBACK (ea_connection_created), NULL); soup_request_send_async (req, NULL, ear_two_completed, loop); g_main_loop_run (loop); g_object_unref (req); while (g_main_context_pending (context)) g_main_context_iteration (context, FALSE); soup_test_session_abort_unref (session); session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, SOUP_SESSION_USE_THREAD_CONTEXT, TRUE, NULL); req = soup_session_request_uri (session, base_uri, NULL); cancellable = g_cancellable_new (); g_signal_connect (session, "request-started", G_CALLBACK (ear_request_started), cancellable); soup_request_send_async (req, cancellable, ear_three_completed, loop); g_main_loop_run (loop); g_object_unref (req); g_object_unref (cancellable); while (g_main_context_pending (context)) g_main_context_iteration (context, FALSE); soup_test_session_abort_unref (session); g_main_loop_unref (loop); } static void do_one_accept_language_test (const char *language, const char *expected_header) { SoupSession *session; SoupMessage *msg; const char *val; debug_printf (1, " LANGUAGE=%s\n", language); g_setenv ("LANGUAGE", language, TRUE); session = soup_test_session_new (SOUP_TYPE_SESSION_SYNC, SOUP_SESSION_ACCEPT_LANGUAGE_AUTO, TRUE, NULL); msg = soup_message_new_from_uri ("GET", base_uri); soup_session_send_message (session, msg); soup_test_session_abort_unref (session); if (!SOUP_STATUS_IS_SUCCESSFUL (msg->status_code)) { debug_printf (1, " Message failed? %d %s\n", msg->status_code, msg->reason_phrase); errors++; } val = soup_message_headers_get_list (msg->request_headers, "Accept-Language"); if (!val) { debug_printf (1, " No Accept-Language set!\n"); errors++; } else if (strcmp (val, expected_header) != 0) { debug_printf (1, " Wrong Accept-Language: expected '%s', got '%s'\n", expected_header, val); errors++; } g_object_unref (msg); } static void do_accept_language_test (void) { const char *orig_language; debug_printf (1, "\nAutomatic Accept-Language processing\n"); orig_language = g_getenv ("LANGUAGE"); do_one_accept_language_test ("C", "en"); do_one_accept_language_test ("fr_FR", "fr-fr, fr;q=0.9"); do_one_accept_language_test ("fr_FR:de:en_US", "fr-fr, fr;q=0.9, de;q=0.8, en-us;q=0.7, en;q=0.6"); if (orig_language) g_setenv ("LANGUAGE", orig_language, TRUE); else g_unsetenv ("LANGUAGE"); } static gboolean cancel_message_timeout (gpointer msg) { SoupSession *session = g_object_get_data (G_OBJECT (msg), "session"); soup_session_cancel_message (session, msg, SOUP_STATUS_CANCELLED); g_object_unref (msg); g_object_unref (session); return FALSE; } static gpointer cancel_message_thread (gpointer msg) { SoupSession *session = g_object_get_data (G_OBJECT (msg), "session"); g_usleep (100000); /* .1s */ soup_session_cancel_message (session, msg, SOUP_STATUS_CANCELLED); g_object_unref (msg); g_object_unref (session); return NULL; } static void set_done (SoupSession *session, SoupMessage *msg, gpointer user_data) { gboolean *done = user_data; *done = TRUE; } static void do_cancel_while_reading_test_for_session (SoupSession *session) { SoupMessage *msg; GThread *thread = NULL; SoupURI *uri; gboolean done = FALSE; uri = soup_uri_new_with_base (base_uri, "/slow"); msg = soup_message_new_from_uri ("GET", uri); soup_uri_free (uri); g_object_set_data (G_OBJECT (msg), "session", session); g_object_ref (msg); g_object_ref (session); if (SOUP_IS_SESSION_ASYNC (session)) g_timeout_add (100, cancel_message_timeout, msg); else thread = g_thread_new ("cancel_message_thread", cancel_message_thread, msg); /* We intentionally don't use soup_session_send_message() here, * because it holds an extra ref on the SoupMessageQueueItem * relative to soup_session_queue_message(). */ g_object_ref (msg); soup_session_queue_message (session, msg, set_done, &done); while (!done) g_main_context_iteration (NULL, TRUE); if (msg->status_code != SOUP_STATUS_CANCELLED) { debug_printf (1, " FAILED: %d %s (expected Cancelled)\n", msg->status_code, msg->reason_phrase); errors++; } g_object_unref (msg); if (thread) g_thread_join (thread); } static void do_cancel_while_reading_test (void) { SoupSession *session; debug_printf (1, "\nCancelling message while reading response (msg api)\n"); debug_printf (1, " Async session\n"); session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, NULL); do_cancel_while_reading_test_for_session (session); soup_test_session_abort_unref (session); debug_printf (1, " Sync session\n"); session = soup_test_session_new (SOUP_TYPE_SESSION_SYNC, NULL); do_cancel_while_reading_test_for_session (session); soup_test_session_abort_unref (session); } static void do_cancel_while_reading_req_test_for_session (SoupSession *session, guint flags) { SoupRequest *req; SoupURI *uri; GCancellable *cancellable; GError *error = NULL; uri = soup_uri_new_with_base (base_uri, "/slow"); req = soup_session_request_uri (session, uri, NULL); soup_uri_free (uri); cancellable = g_cancellable_new (); soup_test_request_send (req, cancellable, flags, &error); if (!error) { debug_printf (1, " Request succeeded?\n"); errors++; } else if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED)) { debug_printf (1, " Unexpected error: %s\n", error->message); errors++; } g_clear_error (&error); g_object_unref (req); g_object_unref (cancellable); } static void do_cancel_while_reading_req_test (void) { SoupSession *session; guint flags; debug_printf (1, "\nCancelling (immediately) message while reading response (request api)\n"); flags = SOUP_TEST_REQUEST_CANCEL_CANCELLABLE | SOUP_TEST_REQUEST_CANCEL_IMMEDIATE; debug_printf (1, " Async session\n"); session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, SOUP_SESSION_USE_THREAD_CONTEXT, TRUE, NULL); do_cancel_while_reading_req_test_for_session (session, flags); soup_test_session_abort_unref (session); debug_printf (1, " Sync session\n"); session = soup_test_session_new (SOUP_TYPE_SESSION_SYNC, NULL); do_cancel_while_reading_req_test_for_session (session, flags); soup_test_session_abort_unref (session); debug_printf (1, "\nCancelling (after 100ms) message while reading response (request api)\n"); flags = SOUP_TEST_REQUEST_CANCEL_CANCELLABLE | SOUP_TEST_REQUEST_CANCEL_SOON; debug_printf (1, " Async session\n"); session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, SOUP_SESSION_USE_THREAD_CONTEXT, TRUE, NULL); do_cancel_while_reading_req_test_for_session (session, flags); soup_test_session_abort_unref (session); debug_printf (1, " Sync session\n"); session = soup_test_session_new (SOUP_TYPE_SESSION_SYNC, NULL); do_cancel_while_reading_req_test_for_session (session, flags); soup_test_session_abort_unref (session); } static void do_aliases_test_for_session (SoupSession *session, const char *redirect_protocol) { SoupMessage *msg; SoupURI *uri; const char *redirected_protocol; uri = soup_uri_new_with_base (base_uri, "/alias-redirect"); msg = soup_message_new_from_uri ("GET", uri); if (redirect_protocol) soup_message_headers_append (msg->request_headers, "X-Redirect-Protocol", redirect_protocol); soup_uri_free (uri); soup_session_send_message (session, msg); redirected_protocol = soup_message_headers_get_one (msg->response_headers, "X-Redirected-Protocol"); if (g_strcmp0 (redirect_protocol, redirected_protocol)) { debug_printf (1, " redirect went to %s, should have gone to %s!\n", redirected_protocol ? redirected_protocol : "(none)", redirect_protocol ? redirect_protocol : "(none)"); errors++; } else if (redirect_protocol && !SOUP_STATUS_IS_SUCCESSFUL (msg->status_code)) { debug_printf (1, " msg failed? (%d %s)\n", msg->status_code, msg->reason_phrase); errors++; } else if (!redirect_protocol && SOUP_STATUS_IS_SUCCESSFUL (msg->status_code)) { debug_printf (1, " msg succeeded? (%d %s)\n", msg->status_code, msg->reason_phrase); errors++; } g_object_unref (msg); } static void do_aliases_test (void) { SoupSession *session; char *aliases[] = { "foo", NULL }; debug_printf (1, "\nhttp-aliases / https-aliases\n"); debug_printf (1, " Default behavior\n"); session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, NULL); do_aliases_test_for_session (session, "http"); soup_test_session_abort_unref (session); debug_printf (1, " foo-means-https\n"); session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, SOUP_SESSION_HTTPS_ALIASES, aliases, NULL); do_aliases_test_for_session (session, "https"); soup_test_session_abort_unref (session); debug_printf (1, " foo-means-nothing\n"); session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, SOUP_SESSION_HTTP_ALIASES, NULL, NULL); do_aliases_test_for_session (session, NULL); soup_test_session_abort_unref (session); } static void do_dot_dot_test (void) { SoupSession *session; SoupMessage *msg; SoupURI *uri; debug_printf (1, "\n'..' smuggling test\n"); session = soup_test_session_new (SOUP_TYPE_SESSION_SYNC, NULL); uri = soup_uri_new_with_base (base_uri, "/..%2ftest"); msg = soup_message_new_from_uri ("GET", uri); soup_uri_free (uri); soup_session_send_message (session, msg); if (msg->status_code != SOUP_STATUS_BAD_REQUEST) { debug_printf (1, " FAILED: %d %s (expected Bad Request)\n", msg->status_code, msg->reason_phrase); errors++; } g_object_unref (msg); soup_test_session_abort_unref (session); } static void ipv6_server_callback (SoupServer *server, SoupMessage *msg, const char *path, GHashTable *query, SoupClientContext *context, gpointer data) { const char *host; char expected_host[128]; host = soup_message_headers_get_one (msg->request_headers, "Host"); if (!host) { debug_printf (1, " request has no Host header!\n"); errors++; soup_message_set_status (msg, SOUP_STATUS_BAD_REQUEST); return; } g_snprintf (expected_host, sizeof (expected_host), "[::1]:%d", soup_server_get_port (server)); if (strcmp (host, expected_host) == 0) soup_message_set_status (msg, SOUP_STATUS_OK); else { debug_printf (1, " request has incorrect Host header '%s'\n", host); errors++; soup_message_set_status (msg, SOUP_STATUS_BAD_REQUEST); } } static void do_ipv6_test (void) { SoupServer *ipv6_server; SoupURI *ipv6_uri; SoupAddress *ipv6_addr; SoupSession *session; SoupMessage *msg; debug_printf (1, "\nIPv6 server test\n"); ipv6_addr = soup_address_new ("::1", SOUP_ADDRESS_ANY_PORT); soup_address_resolve_sync (ipv6_addr, NULL); ipv6_server = soup_server_new (SOUP_SERVER_INTERFACE, ipv6_addr, NULL); g_object_unref (ipv6_addr); soup_server_add_handler (ipv6_server, NULL, ipv6_server_callback, NULL, NULL); soup_server_run_async (ipv6_server); ipv6_uri = soup_uri_new ("http://[::1]/"); soup_uri_set_port (ipv6_uri, soup_server_get_port (ipv6_server)); session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, NULL); debug_printf (1, " HTTP/1.1\n"); msg = soup_message_new_from_uri ("GET", ipv6_uri); soup_session_send_message (session, msg); if (!SOUP_STATUS_IS_SUCCESSFUL (msg->status_code)) { debug_printf (1, " request failed: %d %s\n", msg->status_code, msg->reason_phrase); errors++; } g_object_unref (msg); debug_printf (1, " HTTP/1.0\n"); msg = soup_message_new_from_uri ("GET", ipv6_uri); soup_message_set_http_version (msg, SOUP_HTTP_1_0); soup_session_send_message (session, msg); if (!SOUP_STATUS_IS_SUCCESSFUL (msg->status_code)) { debug_printf (1, " request failed: %d %s\n", msg->status_code, msg->reason_phrase); errors++; } g_object_unref (msg); soup_uri_free (ipv6_uri); soup_test_session_abort_unref (session); soup_test_server_quit_unref (ipv6_server); } static void do_idle_on_dispose_test (void) { SoupSession *session; SoupMessage *msg; GMainContext *async_context; debug_printf (1, "\nTesting SoupSessionAsync dispose behavior\n"); async_context = g_main_context_new (); session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, SOUP_SESSION_ASYNC_CONTEXT, async_context, NULL); msg = soup_message_new_from_uri ("GET", base_uri); soup_session_send_message (session, msg); g_object_unref (msg); while (g_main_context_iteration (async_context, FALSE)) ; g_object_run_dispose (G_OBJECT (session)); if (g_main_context_iteration (async_context, FALSE)) { debug_printf (1, " idle was queued!\n"); errors++; } g_object_unref (session); g_main_context_unref (async_context); } static void do_pause_abort_test (void) { SoupSession *session; SoupMessage *msg; gpointer ptr; debug_printf (1, "\nTesting paused messages don't get leaked on abort\n"); session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, NULL); msg = soup_message_new_from_uri ("GET", base_uri); soup_session_queue_message (session, msg, NULL, NULL); soup_session_pause_message (session, msg); g_object_add_weak_pointer (G_OBJECT (msg), &ptr); soup_test_session_abort_unref (session); if (ptr) { debug_printf (1, " msg was leaked\n"); errors++; } } int main (int argc, char **argv) { SoupAuthDomain *auth_domain; test_init (argc, argv, NULL); server = soup_test_server_new (TRUE); soup_server_add_handler (server, NULL, server_callback, "http", NULL); base_uri = soup_uri_new ("http://127.0.0.1/"); soup_uri_set_port (base_uri, soup_server_get_port (server)); auth_domain = soup_auth_domain_basic_new ( SOUP_AUTH_DOMAIN_REALM, "misc-test", SOUP_AUTH_DOMAIN_ADD_PATH, "/auth", SOUP_AUTH_DOMAIN_BASIC_AUTH_CALLBACK, auth_callback, NULL); soup_server_add_auth_domain (server, auth_domain); g_object_unref (auth_domain); ssl_server = soup_test_server_new_ssl (TRUE); soup_server_add_handler (ssl_server, NULL, server_callback, "https", NULL); ssl_base_uri = soup_uri_new ("https://127.0.0.1/"); soup_uri_set_port (ssl_base_uri, soup_server_get_port (ssl_server)); do_host_test (); do_callback_unref_test (); do_callback_unref_req_test (); do_msg_reuse_test (); do_star_test (); do_early_abort_test (); do_early_abort_req_test (); do_accept_language_test (); do_cancel_while_reading_test (); do_cancel_while_reading_req_test (); do_aliases_test (); do_dot_dot_test (); do_ipv6_test (); do_idle_on_dispose_test (); do_pause_abort_test (); soup_uri_free (base_uri); soup_uri_free (ssl_base_uri); soup_test_server_quit_unref (server); soup_test_server_quit_unref (ssl_server); test_cleanup (); return errors != 0; }