/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ /* * Copyright (C) 2007 Red Hat, Inc. * Copyright (C) 2011 Igalia, S.L. */ #include "test-utils.h" SoupServer *server; SoupURI *base_uri; static void server_callback (SoupServer *server, SoupMessage *msg, const char *path, GHashTable *query, SoupClientContext *context, gpointer data) { const char *accept_encoding, *options; GSList *codings; SoupBuffer *response = NULL; options = soup_message_headers_get_one (msg->request_headers, "X-Test-Options"); if (!options) options = ""; accept_encoding = soup_message_headers_get_list (msg->request_headers, "Accept-Encoding"); if (accept_encoding && !soup_header_contains (options, "force-encode")) codings = soup_header_parse_quality_list (accept_encoding, NULL); else codings = NULL; if (codings) { gboolean claim_deflate, claim_gzip; const char *extension = NULL, *encoding = NULL; claim_deflate = g_slist_find_custom (codings, "deflate", (GCompareFunc)g_ascii_strcasecmp) != NULL; claim_gzip = g_slist_find_custom (codings, "gzip", (GCompareFunc)g_ascii_strcasecmp) != NULL; if (claim_gzip && (!claim_deflate || (!soup_header_contains (options, "prefer-deflate-zlib") && !soup_header_contains (options, "prefer-deflate-raw")))) { extension = "gz"; encoding = "gzip"; } else if (claim_deflate) { if (soup_header_contains (options, "prefer-deflate-raw")) { extension = "raw"; encoding = "deflate"; } else { extension = "zlib"; encoding = "deflate"; } } if (extension && encoding) { char *resource; resource = g_strdup_printf ("%s.%s", path, extension); response = soup_test_load_resource (resource, NULL); if (response) { soup_message_headers_append (msg->response_headers, "Content-Encoding", encoding); } g_free (resource); } } soup_header_free_list (codings); if (!response) response = soup_test_load_resource (path, NULL); if (!response) { /* If path.gz exists but can't be read, we'll send back * the error with "Content-Encoding: gzip" but there's * no body, so, eh. */ soup_message_set_status (msg, SOUP_STATUS_NOT_FOUND); return; } if (soup_header_contains (options, "force-encode")) { const gchar *encoding = "gzip"; if (soup_header_contains (options, "prefer-deflate-zlib") || soup_header_contains (options, "prefer-deflate-raw")) encoding = "deflate"; soup_message_headers_replace (msg->response_headers, "Content-Encoding", encoding); } /* Content-Type matches the "real" format, not the sent format */ if (g_str_has_suffix (path, ".gz")) { soup_message_headers_append (msg->response_headers, "Content-Type", "application/gzip"); } else { soup_message_headers_append (msg->response_headers, "Content-Type", "text/plain"); } soup_message_set_status (msg, SOUP_STATUS_OK); soup_message_headers_set_encoding (msg->response_headers, SOUP_ENCODING_CHUNKED); if (!soup_header_contains (options, "empty")) soup_message_body_append_buffer (msg->response_body, response); soup_buffer_free (response); if (soup_header_contains (options, "trailing-junk")) { soup_message_body_append (msg->response_body, SOUP_MEMORY_COPY, options, strlen (options)); } soup_message_body_complete (msg->response_body); } typedef struct { SoupSession *session; SoupMessage *msg; SoupRequest *req; SoupBuffer *response; } CodingTestData; typedef enum { CODING_TEST_DEFAULT = 0, CODING_TEST_NO_DECODER = (1 << 0), CODING_TEST_REQUEST_API = (1 << 1), CODING_TEST_EMPTY = (1 << 2) } CodingTestType; typedef enum { NO_CHECK, EXPECT_DECODED, EXPECT_NOT_DECODED } MessageContentStatus; static void check_response (CodingTestData *data, const char *expected_encoding, const char *expected_content_type, MessageContentStatus status, GByteArray *body) { const char *coding, *type; soup_test_assert_message_status (data->msg, SOUP_STATUS_OK); coding = soup_message_headers_get_one (data->msg->response_headers, "Content-Encoding"); g_assert_cmpstr (coding, ==, expected_encoding); if (status != NO_CHECK) { if (status == EXPECT_DECODED) g_assert_true (soup_message_get_flags (data->msg) & SOUP_MESSAGE_CONTENT_DECODED); else g_assert_false (soup_message_get_flags (data->msg) & SOUP_MESSAGE_CONTENT_DECODED); } type = soup_message_headers_get_one (data->msg->response_headers, "Content-Type"); g_assert_cmpstr (type, ==, expected_content_type); if (body) { soup_assert_cmpmem (body->data, body->len, data->response->data, data->response->length); } else { soup_assert_cmpmem (data->msg->response_body->data, data->msg->response_body->length, data->response->data, data->response->length); } } static void setup_coding_test (CodingTestData *data, gconstpointer test_data) { CodingTestType test_type = GPOINTER_TO_INT (test_data); SoupMessage *msg; SoupURI *uri; data->session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC, SOUP_SESSION_USE_THREAD_CONTEXT, TRUE, NULL); uri = soup_uri_new_with_base (base_uri, "/mbox"); if (test_type & CODING_TEST_EMPTY) data->response = soup_buffer_new (SOUP_MEMORY_STATIC, "", 0); else { msg = soup_message_new_from_uri ("GET", uri); soup_session_send_message (data->session, msg); data->response = soup_message_body_flatten (msg->response_body); g_object_unref (msg); } if (test_type & CODING_TEST_REQUEST_API) { SoupRequestHTTP *reqh; reqh = soup_session_request_http_uri (data->session, "GET", uri, NULL); data->req = SOUP_REQUEST (reqh); data->msg = soup_request_http_get_message (reqh); } else data->msg = soup_message_new_from_uri ("GET", uri); soup_uri_free (uri); if (! (test_type & CODING_TEST_NO_DECODER)) soup_session_add_feature_by_type (data->session, SOUP_TYPE_CONTENT_DECODER); } static void teardown_coding_test (CodingTestData *data, gconstpointer test_data) { soup_buffer_free (data->response); g_clear_object (&data->req); g_object_unref (data->msg); soup_test_session_abort_unref (data->session); } static void do_coding_test_plain (CodingTestData *data, gconstpointer test_data) { soup_session_send_message (data->session, data->msg); check_response (data, NULL, "text/plain", EXPECT_NOT_DECODED, NULL); } static void do_coding_test_gzip (CodingTestData *data, gconstpointer test_data) { soup_session_send_message (data->session, data->msg); check_response (data, "gzip", "text/plain", EXPECT_DECODED, NULL); } static void do_coding_test_gzip_with_junk (CodingTestData *data, gconstpointer test_data) { g_test_bug ("606352"); g_test_bug ("676477"); soup_message_headers_append (data->msg->request_headers, "X-Test-Options", "trailing-junk"); soup_session_send_message (data->session, data->msg); check_response (data, "gzip", "text/plain", EXPECT_DECODED, NULL); } static void do_coding_test_gzip_bad_server (CodingTestData *data, gconstpointer test_data) { g_test_bug ("613361"); soup_message_headers_append (data->msg->request_headers, "X-Test-Options", "force-encode"); soup_session_send_message (data->session, data->msg); /* Failed content-decoding should have left the body untouched * from what the server sent... which happens to be the * uncompressed data. */ check_response (data, "gzip", "text/plain", EXPECT_NOT_DECODED, NULL); } static void do_coding_test_deflate (CodingTestData *data, gconstpointer test_data) { soup_message_headers_append (data->msg->request_headers, "X-Test-Options", "prefer-deflate-zlib"); soup_session_send_message (data->session, data->msg); check_response (data, "deflate", "text/plain", EXPECT_DECODED, NULL); } static void do_coding_test_deflate_with_junk (CodingTestData *data, gconstpointer test_data) { g_test_bug ("606352"); g_test_bug ("676477"); soup_message_headers_append (data->msg->request_headers, "X-Test-Options", "prefer-deflate-zlib, trailing-junk"); soup_session_send_message (data->session, data->msg); check_response (data, "deflate", "text/plain", EXPECT_DECODED, NULL); } static void do_coding_test_deflate_bad_server (CodingTestData *data, gconstpointer test_data) { g_test_bug ("613361"); soup_message_headers_append (data->msg->request_headers, "X-Test-Options", "force-encode, prefer-deflate-zlib"); soup_session_send_message (data->session, data->msg); check_response (data, "deflate", "text/plain", EXPECT_NOT_DECODED, NULL); } static void do_coding_test_deflate_raw (CodingTestData *data, gconstpointer test_data) { soup_message_headers_append (data->msg->request_headers, "X-Test-Options", "prefer-deflate-raw"); soup_session_send_message (data->session, data->msg); check_response (data, "deflate", "text/plain", EXPECT_DECODED, NULL); } static void do_coding_test_deflate_raw_bad_server (CodingTestData *data, gconstpointer test_data) { g_test_bug ("613361"); soup_message_headers_append (data->msg->request_headers, "X-Test-Options", "force-encode, prefer-deflate-raw"); soup_session_send_message (data->session, data->msg); check_response (data, "deflate", "text/plain", EXPECT_NOT_DECODED, NULL); } static void read_finished (GObject *stream, GAsyncResult *result, gpointer user_data) { gssize *nread = user_data; GError *error = NULL; *nread = g_input_stream_read_finish (G_INPUT_STREAM (stream), result, &error); g_assert_no_error (error); g_clear_error (&error); } static void do_single_coding_req_test (CodingTestData *data, const char *expected_encoding, const char *expected_content_type, MessageContentStatus status) { GInputStream *stream; GByteArray *body; guchar buf[1024]; gssize nread; GError *error = NULL; body = g_byte_array_new (); stream = soup_test_request_send (data->req, NULL, 0, &error); if (!stream) { g_assert_no_error (error); g_error_free (error); return; } do { nread = -2; g_input_stream_read_async (stream, buf, sizeof (buf), G_PRIORITY_DEFAULT, NULL, read_finished, &nread); while (nread == -2) g_main_context_iteration (NULL, TRUE); if (nread > 0) g_byte_array_append (body, buf, nread); } while (nread > 0); soup_test_request_close_stream (data->req, stream, NULL, &error); g_assert_no_error (error); g_clear_error (&error); g_object_unref (stream); check_response (data, expected_encoding, expected_content_type, status, body); g_byte_array_free (body, TRUE); } static void do_coding_req_test_plain (CodingTestData *data, gconstpointer test_data) { do_single_coding_req_test (data, NULL, "text/plain", EXPECT_NOT_DECODED); } static void do_coding_req_test_gzip (CodingTestData *data, gconstpointer test_data) { do_single_coding_req_test (data, "gzip", "text/plain", EXPECT_DECODED); } static void do_coding_req_test_gzip_with_junk (CodingTestData *data, gconstpointer test_data) { g_test_bug ("606352"); g_test_bug ("676477"); soup_message_headers_append (data->msg->request_headers, "X-Test-Options", "trailing-junk"); do_single_coding_req_test (data, "gzip", "text/plain", EXPECT_DECODED); } static void do_coding_req_test_gzip_bad_server (CodingTestData *data, gconstpointer test_data) { g_test_bug ("613361"); soup_message_headers_append (data->msg->request_headers, "X-Test-Options", "force-encode"); do_single_coding_req_test (data, "gzip", "text/plain", EXPECT_NOT_DECODED); } static void do_coding_req_test_deflate (CodingTestData *data, gconstpointer test_data) { soup_message_headers_append (data->msg->request_headers, "X-Test-Options", "prefer-deflate-zlib"); do_single_coding_req_test (data, "deflate", "text/plain", EXPECT_DECODED); } static void do_coding_req_test_deflate_with_junk (CodingTestData *data, gconstpointer test_data) { g_test_bug ("606352"); g_test_bug ("676477"); soup_message_headers_append (data->msg->request_headers, "X-Test-Options", "prefer-deflate-zlib, trailing-junk"); do_single_coding_req_test (data, "deflate", "text/plain", EXPECT_DECODED); } static void do_coding_req_test_deflate_bad_server (CodingTestData *data, gconstpointer test_data) { g_test_bug ("613361"); soup_message_headers_append (data->msg->request_headers, "X-Test-Options", "force-encode, prefer-deflate-zlib"); do_single_coding_req_test (data, "deflate", "text/plain", EXPECT_NOT_DECODED); } static void do_coding_req_test_deflate_raw (CodingTestData *data, gconstpointer test_data) { soup_message_headers_append (data->msg->request_headers, "X-Test-Options", "prefer-deflate-raw"); do_single_coding_req_test (data, "deflate", "text/plain", EXPECT_DECODED); } static void do_coding_req_test_deflate_raw_bad_server (CodingTestData *data, gconstpointer test_data) { g_test_bug ("613361"); soup_message_headers_append (data->msg->request_headers, "X-Test-Options", "force-encode, prefer-deflate-raw"); do_single_coding_req_test (data, "deflate", "text/plain", EXPECT_NOT_DECODED); } static void do_coding_msg_empty_test (CodingTestData *data, gconstpointer test_data) { g_test_bug ("697527"); soup_message_headers_append (data->msg->request_headers, "X-Test-Options", "empty"); soup_session_send_message (data->session, data->msg); check_response (data, "gzip", "text/plain", EXPECT_NOT_DECODED, NULL); } static void do_coding_req_empty_test (CodingTestData *data, gconstpointer test_data) { g_test_bug ("697527"); soup_message_headers_append (data->msg->request_headers, "X-Test-Options", "empty"); do_single_coding_req_test (data, "gzip", "text/plain", EXPECT_NOT_DECODED); } int main (int argc, char **argv) { int ret; test_init (argc, argv, NULL); server = soup_test_server_new (SOUP_TEST_SERVER_IN_THREAD); soup_server_add_handler (server, NULL, server_callback, NULL, NULL); base_uri = soup_test_server_get_uri (server, "http", NULL); g_test_add ("/coding/message/plain", CodingTestData, GINT_TO_POINTER (CODING_TEST_NO_DECODER), setup_coding_test, do_coding_test_plain, teardown_coding_test); g_test_add ("/coding/message/gzip", CodingTestData, GINT_TO_POINTER (CODING_TEST_DEFAULT), setup_coding_test, do_coding_test_gzip, teardown_coding_test); g_test_add ("/coding/message/gzip/with-junk", CodingTestData, GINT_TO_POINTER (CODING_TEST_DEFAULT), setup_coding_test, do_coding_test_gzip_with_junk, teardown_coding_test); g_test_add ("/coding/message/gzip/bad-server", CodingTestData, GINT_TO_POINTER (CODING_TEST_DEFAULT), setup_coding_test, do_coding_test_gzip_bad_server, teardown_coding_test); g_test_add ("/coding/message/deflate", CodingTestData, GINT_TO_POINTER (CODING_TEST_DEFAULT), setup_coding_test, do_coding_test_deflate, teardown_coding_test); g_test_add ("/coding/message/deflate/with-junk", CodingTestData, GINT_TO_POINTER (CODING_TEST_DEFAULT), setup_coding_test, do_coding_test_deflate_with_junk, teardown_coding_test); g_test_add ("/coding/message/deflate/bad-server", CodingTestData, GINT_TO_POINTER (CODING_TEST_DEFAULT), setup_coding_test, do_coding_test_deflate_bad_server, teardown_coding_test); g_test_add ("/coding/message/deflate-raw", CodingTestData, GINT_TO_POINTER (CODING_TEST_DEFAULT), setup_coding_test, do_coding_test_deflate_raw, teardown_coding_test); g_test_add ("/coding/message/deflate-raw/bad-server", CodingTestData, GINT_TO_POINTER (CODING_TEST_DEFAULT), setup_coding_test, do_coding_test_deflate_raw_bad_server, teardown_coding_test); g_test_add ("/coding/request/plain", CodingTestData, GINT_TO_POINTER (CODING_TEST_NO_DECODER | CODING_TEST_REQUEST_API), setup_coding_test, do_coding_req_test_plain, teardown_coding_test); g_test_add ("/coding/request/gzip", CodingTestData, GINT_TO_POINTER (CODING_TEST_REQUEST_API), setup_coding_test, do_coding_req_test_gzip, teardown_coding_test); g_test_add ("/coding/request/gzip/with-junk", CodingTestData, GINT_TO_POINTER (CODING_TEST_REQUEST_API), setup_coding_test, do_coding_req_test_gzip_with_junk, teardown_coding_test); g_test_add ("/coding/request/gzip/bad-server", CodingTestData, GINT_TO_POINTER (CODING_TEST_REQUEST_API), setup_coding_test, do_coding_req_test_gzip_bad_server, teardown_coding_test); g_test_add ("/coding/request/deflate", CodingTestData, GINT_TO_POINTER (CODING_TEST_REQUEST_API), setup_coding_test, do_coding_req_test_deflate, teardown_coding_test); g_test_add ("/coding/request/deflate/with-junk", CodingTestData, GINT_TO_POINTER (CODING_TEST_REQUEST_API), setup_coding_test, do_coding_req_test_deflate_with_junk, teardown_coding_test); g_test_add ("/coding/request/deflate/bad-server", CodingTestData, GINT_TO_POINTER (CODING_TEST_REQUEST_API), setup_coding_test, do_coding_req_test_deflate_bad_server, teardown_coding_test); g_test_add ("/coding/request/deflate-raw", CodingTestData, GINT_TO_POINTER (CODING_TEST_REQUEST_API), setup_coding_test, do_coding_req_test_deflate_raw, teardown_coding_test); g_test_add ("/coding/request/deflate-raw/bad-server", CodingTestData, GINT_TO_POINTER (CODING_TEST_REQUEST_API), setup_coding_test, do_coding_req_test_deflate_raw_bad_server, teardown_coding_test); g_test_add ("/coding/message/empty", CodingTestData, GINT_TO_POINTER (CODING_TEST_EMPTY), setup_coding_test, do_coding_msg_empty_test, teardown_coding_test); g_test_add ("/coding/request/empty", CodingTestData, GINT_TO_POINTER (CODING_TEST_REQUEST_API | CODING_TEST_EMPTY), setup_coding_test, do_coding_req_empty_test, teardown_coding_test); ret = g_test_run (); soup_uri_free (base_uri); soup_test_server_quit_unref (server); test_cleanup (); return ret; }