[libsoup] Add missing file for last commit
- From: Gustavo Noronha Silva <gns src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [libsoup] Add missing file for last commit
- Date: Mon, 20 Aug 2012 18:22:24 +0000 (UTC)
commit 1fcf9757746de2d4d19b6c07bc59c8b5f1bf14b2
Author: Gustavo Noronha Silva <gns gnome org>
Date: Mon Aug 20 15:20:22 2012 -0300
Add missing file for last commit
tests/multipart-test.c | 617 ++++++++++++++++++++++++++++++++++++++++++++++++
1 files changed, 617 insertions(+), 0 deletions(-)
---
diff --git a/tests/multipart-test.c b/tests/multipart-test.c
new file mode 100644
index 0000000..136ee0d
--- /dev/null
+++ b/tests/multipart-test.c
@@ -0,0 +1,617 @@
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
+/*
+ * Copyright (C) 2011 Collabora Ltd.
+ */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <unistd.h>
+
+#define LIBSOUP_USE_UNSTABLE_REQUEST_API
+#include <libsoup/soup-multipart-input-stream.h>
+#include <libsoup/soup-request-http.h>
+#include <libsoup/soup-requester.h>
+#include <libsoup/soup.h>
+
+#include "test-utils.h"
+
+#define READ_BUFFER_SIZE 8192
+
+typedef enum {
+ NO_MULTIPART,
+ SYNC_MULTIPART,
+ ASYNC_MULTIPART,
+ ASYNC_MULTIPART_SMALL_READS
+} MultipartMode;
+
+char *buffer;
+SoupRequester *requester;
+char *base_uri_string;
+SoupURI *base_uri;
+SoupMultipartInputStream *multipart;
+unsigned passes;
+
+
+/* This payload contains 4 different responses.
+ *
+ * First, a text/html response with a Content-Length (31);
+ * Second, a response lacking Content-Type with Content-Length (11);
+ * Third, a text/css response with no Content-Length;
+ * Fourth, same as the third, but with different content;
+ */
+const char *payload = \
+ "--cut-here\r\n" \
+ "Content-Type: text/html\n"
+ "Content-Length: 30\r\n" \
+ "\r\n" \
+ "<html><body>Hey!</body></html>" \
+ "\r\n--cut-here\r\n" \
+ "Content-Length: 10\r\n" \
+ "\r\n" \
+ "soup rocks" \
+ "\r\n--cut-here\r\n" \
+ "Content-Type: text/css\r\n" \
+ "\r\n" \
+ ".soup { before: rocks; }" \
+ "\r\n--cut-here\n" /* Tests boundary ending in a single \n. */ \
+ "Content-Type: text/css\r\n" \
+ "\r\n" \
+ "#soup { background-color: black; }" \
+ "\r\n--cut-here--";
+
+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);
+
+ soup_message_headers_append (msg->response_headers,
+ "Content-Type", "multipart/x-mixed-replace; boundary=cut-here");
+
+ soup_message_body_append (msg->response_body,
+ SOUP_MEMORY_STATIC,
+ payload,
+ strlen (payload));
+
+ soup_message_body_complete (msg->response_body);
+}
+
+static void
+content_sniffed (SoupMessage *msg, char *content_type, GHashTable *params, int *sniffed_count)
+{
+ *sniffed_count = *sniffed_count + 1;
+ debug_printf (2, " content-sniffed -> %s\n", content_type);
+}
+
+static void
+check_is_next (gboolean is_next)
+{
+ if (!is_next) {
+ debug_printf (1, " expected a header, but there are no more headers\n");
+ errors++;
+ }
+}
+
+static void
+got_headers (SoupMessage *msg, int *headers_count)
+{
+ SoupMessageHeadersIter iter;
+ gboolean is_next;
+ const char* name, *value;
+
+ *headers_count = *headers_count + 1;
+
+ soup_message_headers_iter_init (&iter, msg->response_headers);
+
+ is_next = soup_message_headers_iter_next (&iter, &name, &value);
+ check_is_next (is_next);
+
+ if (g_str_equal (name, "Date")) {
+ is_next = soup_message_headers_iter_next (&iter, &name, &value);
+ check_is_next (is_next);
+ }
+
+ if (!g_str_equal (name, "Content-Type")) {
+ debug_printf (1, " expected `Content-Type' got %s\n", name);
+ errors++;
+ }
+
+ if (!g_str_equal (value, "multipart/x-mixed-replace; boundary=cut-here")) {
+ debug_printf (1, " expected `multipart/x-mixed-replace; boundary=cut-here' got %s\n", value);
+ errors++;
+ }
+}
+
+static void
+read_cb (GObject *source, GAsyncResult *asyncResult, gpointer data)
+{
+ GMainLoop *loop = (GMainLoop*)data;
+ GInputStream *stream = G_INPUT_STREAM (source);
+ GError *error = NULL;
+ gssize bytes_read = g_input_stream_read_finish (stream, asyncResult, &error);
+ if (error) {
+ debug_printf (1, " failed read: %s\n", error->message);
+ errors++;
+
+ g_main_loop_quit (loop);
+ return;
+ }
+
+ if (!bytes_read) {
+ g_input_stream_close (stream, NULL, &error);
+
+ if (error) {
+ debug_printf (1, " failed close: %s\n", error->message);
+ errors++;
+ }
+
+ g_main_loop_quit (loop);
+ return;
+ }
+
+ g_input_stream_read_async (stream, buffer, READ_BUFFER_SIZE,
+ G_PRIORITY_DEFAULT, NULL,
+ read_cb, data);
+}
+
+static void
+no_multipart_handling_cb (GObject *source, GAsyncResult *res, gpointer data)
+{
+ GMainLoop *loop = (GMainLoop*)data;
+ SoupRequest *request = SOUP_REQUEST (source);
+ GError *error = NULL;
+ GInputStream* in;
+
+ in = soup_request_send_finish (request, res, &error);
+
+ if (error) {
+ debug_printf (1, " failed send: %s\n", error->message);
+ errors++;
+
+ g_main_loop_quit (loop);
+ return;
+ }
+
+ g_input_stream_read_async (in, buffer, READ_BUFFER_SIZE,
+ G_PRIORITY_DEFAULT, NULL,
+ read_cb, data);
+}
+
+static void
+multipart_close_part_cb (GObject *source, GAsyncResult *res, gpointer data)
+{
+ GInputStream *in = G_INPUT_STREAM (source);
+ GError *error = NULL;
+
+ g_input_stream_close_finish (in, res, &error);
+ if (error) {
+ debug_printf (1, " error closing stream: %s\n", error->message);
+ errors++;
+ }
+}
+
+static void multipart_next_part_cb (GObject *source,
+ GAsyncResult *res,
+ gpointer data);
+
+static void
+check_read (gsize nread, unsigned passes)
+{
+ switch (passes) {
+ case 0:
+ if (nread != 30) {
+ debug_printf (1, " expected to read 30 bytes, got: %d\n", (int)nread);
+ errors++;
+ }
+ break;
+ case 1:
+ if (nread != 10) {
+ debug_printf (1, " expected to read 10 bytes, got: %d\n", (int)nread);
+ errors++;
+ }
+ break;
+ case 2:
+ if (nread != 24) {
+ debug_printf (1, " expected to read 24 bytes, got: %d\n", (int)nread);
+ errors++;
+ }
+ break;
+ case 3:
+ if (nread != 34) {
+ debug_printf (1, " expected to read 34 bytes, got: %d\n", (int)nread);
+ errors++;
+ }
+ break;
+ default:
+ debug_printf (1, " unexpected read of size: %d\n", (int)nread);
+ errors++;
+ }
+}
+
+static void
+multipart_read_cb (GObject *source, GAsyncResult *asyncResult, gpointer data)
+{
+ GMainLoop *loop = (GMainLoop*)data;
+ GInputStream *in = G_INPUT_STREAM (source);
+ GError *error = NULL;
+ static gssize bytes_read_for_part = 0;
+ gssize bytes_read;
+
+ bytes_read = g_input_stream_read_finish (in, asyncResult, &error);
+
+ if (error) {
+ debug_printf (1, " failed read: %s\n", error->message);
+ errors++;
+
+ g_input_stream_close_async (in, G_PRIORITY_DEFAULT, NULL,
+ multipart_close_part_cb, NULL);
+
+ g_main_loop_quit (loop);
+ return;
+ }
+
+ /* Read 0 bytes - try to start reading another part. */
+ if (!bytes_read) {
+ check_read (bytes_read_for_part, passes);
+ bytes_read_for_part = 0;
+ passes++;
+
+ g_input_stream_close_async (in, G_PRIORITY_DEFAULT, NULL,
+ multipart_close_part_cb, NULL);
+
+ soup_multipart_input_stream_next_part_async (multipart, G_PRIORITY_DEFAULT, NULL,
+ multipart_next_part_cb, data);
+ return;
+ }
+
+ bytes_read_for_part += bytes_read;
+ g_input_stream_read_async (in, buffer, READ_BUFFER_SIZE,
+ G_PRIORITY_DEFAULT, NULL,
+ multipart_read_cb, data);
+}
+
+static void
+check_headers (SoupMultipartInputStream* multipart, unsigned passes)
+{
+ SoupMessageHeaders *headers;
+ SoupMessageHeadersIter iter;
+ gboolean is_next;
+ const char *name, *value;
+
+ headers = soup_multipart_input_stream_get_headers (multipart);
+ soup_message_headers_iter_init (&iter, headers);
+
+ switch (passes) {
+ case 0:
+ is_next = soup_message_headers_iter_next (&iter, &name, &value);
+ check_is_next (is_next);
+
+ if (!g_str_equal (name, "Content-Type")) {
+ debug_printf (1, " [0] expected `Content-Type' got %s\n", name);
+ errors++;
+ }
+
+ if (!g_str_equal (value, "text/html")) {
+ debug_printf (1, " [0] expected `text/html' got %s\n", value);
+ errors++;
+ }
+
+ is_next = soup_message_headers_iter_next (&iter, &name, &value);
+ check_is_next (is_next);
+
+ if (!g_str_equal (name, "Content-Length")) {
+ debug_printf (1, " [0] expected `Content-Length' got %s\n", name);
+ errors++;
+ }
+
+ if (!g_str_equal (value, "30")) {
+ debug_printf (1, " [0] expected `30' got %s\n", value);
+ errors++;
+ }
+
+ break;
+ case 1:
+ is_next = soup_message_headers_iter_next (&iter, &name, &value);
+ check_is_next (is_next);
+
+ if (!g_str_equal (name, "Content-Length")) {
+ debug_printf (1, " [1] expected `Content-Length' got %s\n", name);
+ errors++;
+ }
+
+ if (!g_str_equal (value, "10")) {
+ debug_printf (1, " [1] expected `10' got %s\n", value);
+ errors++;
+ }
+
+ break;
+ case 2:
+ case 3:
+ is_next = soup_message_headers_iter_next (&iter, &name, &value);
+ check_is_next (is_next);
+
+ if (!g_str_equal (name, "Content-Type")) {
+ debug_printf (1, " [%d] expected `Content-Type' got %s\n", passes, name);
+ errors++;
+ }
+
+ if (!g_str_equal (value, "text/css")) {
+ debug_printf (1, " [%d] expected `text/html' got %s\n", passes, value);
+ errors++;
+ }
+
+ break;
+ default:
+ debug_printf (1, " unexpected part received\n");
+ break;
+ }
+}
+
+static void
+multipart_next_part_cb (GObject *source, GAsyncResult *res, gpointer data)
+{
+ GMainLoop *loop = (GMainLoop*)data;
+ GError *error = NULL;
+ GInputStream *in;
+ gsize read_size = READ_BUFFER_SIZE;
+
+ g_assert (SOUP_MULTIPART_INPUT_STREAM (source) == multipart);
+
+ in = soup_multipart_input_stream_next_part_finish (multipart, res, &error);
+
+ if (error) {
+ debug_printf (1, " failed next part: %s\n", error->message);
+ errors++;
+
+ g_main_loop_quit (loop);
+ return;
+ }
+
+ if (!in) {
+ if (passes != 4) {
+ debug_printf (1, " expected 4 parts, got %u\n", passes);
+ errors++;
+ }
+
+ g_main_loop_quit (loop);
+ return;
+ }
+
+ check_headers (multipart, passes);
+
+ if (g_object_get_data (G_OBJECT (multipart), "multipart-small-reads"))
+ read_size = 4;
+
+ g_input_stream_read_async (in, buffer, read_size,
+ G_PRIORITY_DEFAULT, NULL,
+ multipart_read_cb, data);
+}
+
+static void
+multipart_handling_cb (GObject *source, GAsyncResult *res, gpointer data)
+{
+ GMainLoop *loop = (GMainLoop*)data;
+ SoupRequest *request = SOUP_REQUEST (source);
+ GError *error = NULL;
+ GInputStream *in;
+ SoupMessage *message;
+
+ in = soup_request_send_finish (request, res, &error);
+ message = soup_request_http_get_message (SOUP_REQUEST_HTTP (request));
+ multipart = soup_multipart_input_stream_new (message, in);
+
+ if (error) {
+ debug_printf (1, " failed send: %s\n", error->message);
+ errors++;
+
+ g_main_loop_quit (loop);
+ return;
+ }
+
+ if (g_object_get_data (source, "multipart-small-reads"))
+ g_object_set_data (G_OBJECT (multipart), "multipart-small-reads", GINT_TO_POINTER(1));
+
+ soup_multipart_input_stream_next_part_async (multipart, G_PRIORITY_DEFAULT, NULL,
+ multipart_next_part_cb, data);
+}
+
+static void
+sync_multipart_handling_cb (GObject *source, GAsyncResult *res, gpointer data)
+{
+ GMainLoop *loop = (GMainLoop*)data;
+ SoupRequest *request = SOUP_REQUEST (source);
+ GError *error = NULL;
+ GInputStream *in;
+ SoupMessage *message;
+ char buffer[READ_BUFFER_SIZE];
+ gsize bytes_read;
+
+ in = soup_request_send_finish (request, res, &error);
+ message = soup_request_http_get_message (SOUP_REQUEST_HTTP (request));
+ multipart = soup_multipart_input_stream_new (message, in);
+
+ if (error) {
+ debug_printf (1, " failed send: %s\n", error->message);
+ errors++;
+
+ g_main_loop_quit (loop);
+ return;
+ }
+
+ while (TRUE) {
+ in = soup_multipart_input_stream_next_part (multipart, NULL, &error);
+
+ if (error) {
+ debug_printf (1, " failed sync next part: %s\n", error->message);
+ errors++;
+ g_clear_error (&error);
+ break;
+ }
+
+ if (!in)
+ break;
+
+ check_headers (multipart, passes);
+
+ g_input_stream_read_all (in, (void*)buffer, sizeof (buffer), &bytes_read, NULL, &error);
+
+ if (error) {
+ debug_printf (1, " failed sync read: %s\n", error->message);
+ errors++;
+ g_clear_error (&error);
+ break;
+ }
+
+ check_read (bytes_read, passes);
+
+ passes++;
+ }
+
+ if (passes != 4) {
+ debug_printf (1, " expected 4 parts, got %u\n", passes);
+ errors++;
+ }
+
+ g_main_loop_quit (loop);
+
+}
+
+static const char*
+multipart_mode_to_string (MultipartMode mode)
+{
+ if (mode == NO_MULTIPART)
+ return "NO_MULTIPART";
+ else if (mode == SYNC_MULTIPART)
+ return "SYNC_MULTIPART";
+ else if (mode == ASYNC_MULTIPART_SMALL_READS)
+ return "SYNC_MULTIPART_SMALL_READS";
+
+ return "ASYNC_MULTIPART";
+}
+
+static void
+test_multipart (int headers_expected, int sniffed_expected, MultipartMode multipart_mode)
+{
+ GError* error = NULL;
+ SoupRequest* request = soup_requester_request (requester, base_uri_string, &error);
+
+ SoupMessage *msg = soup_request_http_get_message (SOUP_REQUEST_HTTP (request));
+ GMainLoop *loop = g_main_loop_new (NULL, TRUE);
+ int headers_count = 0;
+ int sniffed_count = 0;
+ GHashTable *params;
+ const char *content_type;
+ gboolean message_is_multipart = FALSE;
+
+ debug_printf (1, "test_multipart(%s)\n", multipart_mode_to_string (multipart_mode));
+
+ /* This is used to track the number of parts. */
+ passes = 0;
+
+ /* Force the server to close the connection. */
+ soup_message_headers_append (msg->request_headers,
+ "Connection", "close");
+
+ g_signal_connect (msg, "got_headers",
+ G_CALLBACK (got_headers), &headers_count);
+
+ g_signal_connect (msg, "content-sniffed",
+ G_CALLBACK (content_sniffed), &sniffed_count);
+
+ g_object_ref (msg);
+
+ if (multipart_mode == ASYNC_MULTIPART)
+ soup_request_send_async (request, NULL, multipart_handling_cb, loop);
+ else if (multipart_mode == ASYNC_MULTIPART_SMALL_READS) {
+ g_object_set_data (G_OBJECT (request), "multipart-small-reads", GINT_TO_POINTER(1));
+ soup_request_send_async (request, NULL, multipart_handling_cb, loop);
+ } else if (multipart_mode == SYNC_MULTIPART)
+ soup_request_send_async (request, NULL, sync_multipart_handling_cb, loop);
+ else
+ soup_request_send_async (request, NULL, no_multipart_handling_cb, loop);
+
+ g_main_loop_run (loop);
+
+ content_type = soup_message_headers_get_content_type (msg->response_headers, ¶ms);
+
+ if (content_type &&
+ g_str_has_prefix (content_type, "multipart/") &&
+ g_hash_table_lookup (params, "boundary")) {
+ message_is_multipart = TRUE;
+ }
+ g_clear_pointer (¶ms, g_hash_table_unref);
+
+ if (!message_is_multipart) {
+ debug_printf (1,
+ " Header does not indicate a multipart message!\n");
+ errors++;
+ }
+
+ if (headers_count != headers_expected) {
+ debug_printf (1,
+ " expected got_header %d times, got %d!\n",
+ headers_expected, headers_count);
+ errors++;
+ }
+
+ if (sniffed_count != sniffed_expected) {
+ debug_printf (1,
+ " expected content_sniffed %d times, got %d!\n",
+ sniffed_expected, sniffed_count);
+ errors++;
+ }
+
+ g_object_unref (msg);
+ g_object_unref (request);
+ g_main_loop_unref (loop);
+}
+
+int
+main (int argc, char **argv)
+{
+ SoupServer *server;
+ SoupSession *session;
+
+ test_init (argc, argv, NULL);
+
+ buffer = g_malloc (READ_BUFFER_SIZE);
+
+ server = soup_test_server_new (FALSE);
+ 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));
+ base_uri_string = soup_uri_to_string (base_uri, FALSE);
+
+ /* FIXME: I had to raise the number of connections allowed here, otherwise I
+ * was hitting the limit, which indicates some connections are not dying.
+ */
+ session = soup_test_session_new (SOUP_TYPE_SESSION_ASYNC,
+ "use-thread-context", TRUE,
+ "max-conns", 20,
+ "max-conns-per-host", 20,
+ NULL);
+ soup_session_add_feature_by_type (session, SOUP_TYPE_REQUESTER);
+ soup_session_add_feature_by_type (session, SOUP_TYPE_CONTENT_SNIFFER);
+
+ requester = SOUP_REQUESTER (soup_session_get_feature (session, SOUP_TYPE_REQUESTER));
+
+ test_multipart (1, 1, NO_MULTIPART);
+ test_multipart (1, 1, SYNC_MULTIPART);
+ test_multipart (1, 1, ASYNC_MULTIPART);
+ test_multipart (1, 1, ASYNC_MULTIPART_SMALL_READS);
+
+ soup_uri_free (base_uri);
+ g_free (base_uri_string);
+
+ soup_test_session_abort_unref (session);
+ soup_test_server_quit_unref (server);
+ test_cleanup ();
+ return errors != 0;
+}
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]