[libsoup/carlosgc/request-body-stream: 1/4] message: add support for stream based request body




commit 2cebb238316ceac9499be0e0941640df0bd1c51e
Author: Carlos Garcia Campos <cgarcia igalia com>
Date:   Tue Oct 6 14:52:06 2020 +0200

    message: add support for stream based request body
    
    Add soup_message_set_request_body() that takes a GInputStream and
    soup_message_set_request_body_from_bytes() for convenience that creates
    a GMemoryInputStream for the given bytes.

 docs/reference/libsoup-3.0-sections.txt |   2 +
 libsoup/soup-body-output-stream.c       |  32 ++-
 libsoup/soup-message-io.c               | 118 +++++++--
 libsoup/soup-message.c                  |  72 ++++++
 libsoup/soup-message.h                  |  10 +
 libsoup/soup-version.h.in               |   1 +
 tests/chunk-test.c                      | 407 --------------------------------
 tests/meson.build                       |   2 +-
 tests/request-body-test.c               | 211 +++++++++++++++++
 9 files changed, 424 insertions(+), 431 deletions(-)
---
diff --git a/docs/reference/libsoup-3.0-sections.txt b/docs/reference/libsoup-3.0-sections.txt
index ce2bcb75..b31a5fb9 100644
--- a/docs/reference/libsoup-3.0-sections.txt
+++ b/docs/reference/libsoup-3.0-sections.txt
@@ -7,6 +7,8 @@ SoupMessage
 soup_message_new
 soup_message_new_from_uri
 soup_message_set_request
+soup_message_set_request_body
+soup_message_set_request_body_from_bytes
 soup_message_set_response
 <SUBSECTION>
 SoupHTTPVersion
diff --git a/libsoup/soup-body-output-stream.c b/libsoup/soup-body-output-stream.c
index 7ba59dca..94fe2811 100644
--- a/libsoup/soup-body-output-stream.c
+++ b/libsoup/soup-body-output-stream.c
@@ -40,6 +40,14 @@ enum {
        PROP_CONTENT_LENGTH
 };
 
+enum {
+       WROTE_DATA,
+
+       LAST_SIGNAL
+};
+
+static guint signals[LAST_SIGNAL] = { 0 };
+
 static void soup_body_output_stream_pollable_init (GPollableOutputStreamInterface *pollable_interface, 
gpointer interface_data);
 
 G_DEFINE_TYPE_WITH_CODE (SoupBodyOutputStream, soup_body_output_stream, G_TYPE_FILTER_OUTPUT_STREAM,
@@ -99,6 +107,13 @@ soup_body_output_stream_get_property (GObject *object, guint prop_id,
        }
 }
 
+static void
+soup_body_output_stream_wrote_data (SoupBodyOutputStream *bostream,
+                                   gsize                 count)
+{
+       g_signal_emit (bostream, signals[WROTE_DATA], 0, count);
+}
+
 static gssize
 soup_body_output_stream_write_raw (SoupBodyOutputStream  *bostream,
                                   const void            *buffer,
@@ -126,8 +141,10 @@ soup_body_output_stream_write_raw (SoupBodyOutputStream  *bostream,
                                          buffer, my_count,
                                          blocking, cancellable, error);
 
-       if (nwrote > 0 && bostream->priv->write_length)
+       if (nwrote > 0 && bostream->priv->write_length) {
                bostream->priv->written += nwrote;
+               soup_body_output_stream_wrote_data (bostream, nwrote);
+       }
 
        if (nwrote == my_count && my_count != count)
                nwrote = count;
@@ -173,6 +190,9 @@ again:
                nwrote = g_pollable_stream_write (bostream->priv->base_stream,
                                                  buffer, count, blocking,
                                                  cancellable, error);
+               if (nwrote > 0)
+                       soup_body_output_stream_wrote_data (bostream, nwrote);
+
                if (nwrote < (gssize)count)
                        return nwrote;
 
@@ -300,6 +320,16 @@ soup_body_output_stream_class_init (SoupBodyOutputStreamClass *stream_class)
        output_stream_class->write_fn = soup_body_output_stream_write_fn;
        output_stream_class->close_fn = soup_body_output_stream_close_fn;
 
+        signals[WROTE_DATA] =
+                g_signal_new ("wrote-data",
+                              G_OBJECT_CLASS_TYPE (object_class),
+                              G_SIGNAL_RUN_LAST,
+                              0,
+                              NULL, NULL,
+                              NULL,
+                              G_TYPE_NONE, 1,
+                              G_TYPE_UINT);
+
        g_object_class_install_property (
                object_class, PROP_ENCODING,
                g_param_spec_enum ("encoding",
diff --git a/libsoup/soup-message-io.c b/libsoup/soup-message-io.c
index 1d85daa8..cfbba0f7 100644
--- a/libsoup/soup-message-io.c
+++ b/libsoup/soup-message-io.c
@@ -85,8 +85,8 @@ typedef struct {
        GSource *unpause_source;
        gboolean paused;
 
-       GCancellable *async_close_wait;
-       GError       *async_close_error;
+       GCancellable *async_wait;
+       GError       *async_error;
 
        SoupMessageGetHeadersFn   get_headers_cb;
        SoupMessageParseHeadersFn parse_headers_cb;
@@ -130,11 +130,11 @@ soup_message_io_cleanup (SoupMessage *msg)
        g_string_free (io->write_buf, TRUE);
         g_clear_pointer (&io->write_chunk, g_bytes_unref);
 
-       if (io->async_close_wait) {
-               g_cancellable_cancel (io->async_close_wait);
-               g_clear_object (&io->async_close_wait);
+       if (io->async_wait) {
+               g_cancellable_cancel (io->async_wait);
+               g_clear_object (&io->async_wait);
        }
-       g_clear_error (&io->async_close_error);
+       g_clear_error (&io->async_error);
 
        g_slice_free (SoupMessageIOData, io);
 }
@@ -321,6 +321,50 @@ soup_message_setup_body_istream (GInputStream *body_stream,
        return istream;
 }
 
+static void
+request_body_stream_wrote_data_cb (SoupMessage *msg,
+                                  guint        count)
+{
+       GBytes *chunk;
+
+       /* FIXME: Change SoupMessage::wrote-body-data to pass just the size */
+       chunk = g_bytes_new_static ("", count);
+       soup_message_wrote_body_data (msg, chunk);
+       g_bytes_unref (chunk);
+}
+
+static void
+request_body_stream_wrote_cb (GOutputStream *ostream,
+                             GAsyncResult  *result,
+                             SoupMessage   *msg)
+{
+       SoupMessageIOData *io;
+       gssize nwrote;
+       GCancellable *async_wait;
+       GError *error = NULL;
+
+       nwrote = g_output_stream_splice_finish (ostream, result, &error);
+
+       io = soup_message_get_io_data (msg);
+       if (!io || !io->async_wait || io->body_ostream != ostream) {
+               g_clear_error (&error);
+               g_object_unref (msg);
+               return;
+       }
+
+       if (nwrote != -1)
+               io->write_state = SOUP_MESSAGE_IO_STATE_BODY_FLUSH;
+
+       if (error)
+               g_propagate_error (&io->async_error, error);
+       async_wait = io->async_wait;
+       io->async_wait = NULL;
+       g_cancellable_cancel (async_wait);
+       g_object_unref (async_wait);
+
+       g_object_unref (msg);
+}
+
 static void
 closed_async (GObject      *source,
              GAsyncResult *result,
@@ -329,21 +373,21 @@ closed_async (GObject      *source,
        GOutputStream *body_ostream = G_OUTPUT_STREAM (source);
        SoupMessage *msg = user_data;
        SoupMessageIOData *io;
-       GCancellable *async_close_wait;
+       GCancellable *async_wait;
 
        io = soup_message_get_io_data (msg);
-       if (!io || !io->async_close_wait || io->body_ostream != body_ostream) {
+       if (!io || !io->async_wait || io->body_ostream != body_ostream) {
                g_object_unref (msg);
                return;
        }
 
-       g_output_stream_close_finish (body_ostream, result, &io->async_close_error);
+       g_output_stream_close_finish (body_ostream, result, &io->async_error);
        g_clear_object (&io->body_ostream);
 
-       async_close_wait = io->async_close_wait;
-       io->async_close_wait = NULL;
-       g_cancellable_cancel (async_close_wait);
-       g_object_unref (async_close_wait);
+       async_wait = io->async_wait;
+       io->async_wait = NULL;
+       g_cancellable_cancel (async_wait);
+       g_object_unref (async_wait);
 
        g_object_unref (msg);
 }
@@ -388,11 +432,11 @@ io_write (SoupMessage *msg, gboolean blocking,
        GBytes *chunk;
        gssize nwrote;
 
-       if (io->async_close_error) {
-               g_propagate_error (error, io->async_close_error);
-               io->async_close_error = NULL;
+       if (io->async_error) {
+               g_propagate_error (error, io->async_error);
+               io->async_error = NULL;
                return FALSE;
-       } else if (io->async_close_wait) {
+       } else if (io->async_wait) {
                g_set_error_literal (error, G_IO_ERROR,
                                     G_IO_ERROR_WOULD_BLOCK,
                                     _("Operation would block"));
@@ -503,6 +547,36 @@ io_write (SoupMessage *msg, gboolean blocking,
                        break;
                }
 
+               if (io->mode == SOUP_MESSAGE_IO_CLIENT && msg->request_body_stream) {
+                       g_signal_connect_object (io->body_ostream,
+                                                "wrote-data",
+                                                G_CALLBACK (request_body_stream_wrote_data_cb),
+                                                msg, G_CONNECT_SWAPPED);
+                       if (blocking) {
+                               nwrote = g_output_stream_splice (io->body_ostream,
+                                                                msg->request_body_stream,
+                                                                G_OUTPUT_STREAM_SPLICE_CLOSE_SOURCE,
+                                                                cancellable,
+                                                                error);
+                               if (nwrote == -1)
+                                       return FALSE;
+                               io->write_state = SOUP_MESSAGE_IO_STATE_BODY_FLUSH;
+                               break;
+                       } else {
+                               io->async_wait = g_cancellable_new ();
+                               g_main_context_push_thread_default (io->async_context);
+                               g_output_stream_splice_async (io->body_ostream,
+                                                             msg->request_body_stream,
+                                                             G_OUTPUT_STREAM_SPLICE_CLOSE_SOURCE,
+                                                             G_PRIORITY_DEFAULT,
+                                                             cancellable,
+                                                             
(GAsyncReadyCallback)request_body_stream_wrote_cb,
+                                                             g_object_ref (msg));
+                               g_main_context_pop_thread_default (io->async_context);
+                               return FALSE;
+                       }
+               }
+
                if (!io->write_chunk) {
                        io->write_chunk = soup_message_body_get_chunk (io->write_body, io->write_body_offset);
                        if (!io->write_chunk) {
@@ -562,7 +636,7 @@ io_write (SoupMessage *msg, gboolean blocking,
                                        return FALSE;
                                g_clear_object (&io->body_ostream);
                        } else {
-                               io->async_close_wait = g_cancellable_new ();
+                               io->async_wait = g_cancellable_new ();
                                g_main_context_push_thread_default (io->async_context);
                                g_output_stream_close_async (io->body_ostream,
                                                             G_PRIORITY_DEFAULT, cancellable,
@@ -883,8 +957,8 @@ soup_message_io_get_source (SoupMessage *msg, GCancellable *cancellable,
                base_source = g_timeout_source_new (0);
        } else if (io->paused) {
                base_source = NULL;
-       } else if (io->async_close_wait) {
-               base_source = g_cancellable_source_new (io->async_close_wait);
+       } else if (io->async_wait) {
+               base_source = g_cancellable_source_new (io->async_wait);
        } else if (SOUP_MESSAGE_IO_STATE_POLLABLE (io->read_state)) {
                GPollableInputStream *istream;
 
@@ -958,7 +1032,7 @@ io_run_until (SoupMessage *msg, gboolean blocking,
 
        g_object_ref (msg);
 
-       while (progress && soup_message_get_io_data (msg) == io && !io->paused && !io->async_close_wait &&
+       while (progress && soup_message_get_io_data (msg) == io && !io->paused && !io->async_wait &&
               (io->read_state < read_state || io->write_state < write_state)) {
 
                if (SOUP_MESSAGE_IO_STATE_ACTIVE (io->read_state))
@@ -988,7 +1062,7 @@ io_run_until (SoupMessage *msg, gboolean blocking,
                                     _("Operation was cancelled"));
                g_object_unref (msg);
                return FALSE;
-       } else if (!io->async_close_wait &&
+       } else if (!io->async_wait &&
                   g_cancellable_set_error_if_cancelled (cancellable, error)) {
                g_object_unref (msg);
                return FALSE;
diff --git a/libsoup/soup-message.c b/libsoup/soup-message.c
index 393dbd35..aacd8008 100644
--- a/libsoup/soup-message.c
+++ b/libsoup/soup-message.c
@@ -183,6 +183,7 @@ soup_message_finalize (GObject *object)
 
        soup_message_body_free (msg->request_body);
        soup_message_headers_free (msg->request_headers);
+       g_clear_object (&msg->request_body_stream);
        soup_message_body_free (msg->response_body);
        soup_message_headers_free (msg->response_headers);
 
@@ -1129,6 +1130,75 @@ soup_message_set_response (SoupMessage    *msg,
        }
 }
 
+/**
+ * soup_message_set_request_body:
+ * @msg: the message
+ * @content_type: (allow-none): MIME Content-Type of the body
+ * @stream: (allow-none): a #GInputStream to read the request body from
+ * @content_length: the byte length of @tream or -1 if unknown
+ *
+ * Set the request body of a #SoupMessage. If
+ * @content_type is %NULL, the request body must be empty (or @stream %NULL) as well.
+ * The request body needs to be set again in case @msg is restarted
+ * (in case of redirection or authentication).
+ */
+void
+soup_message_set_request_body (SoupMessage  *msg,
+                               const char   *content_type,
+                               GInputStream *stream,
+                               gssize        content_length)
+{
+        g_return_if_fail (SOUP_IS_MESSAGE (msg));
+        g_return_if_fail (content_type != NULL || content_length == 0);
+        g_return_if_fail (content_type == NULL || G_IS_INPUT_STREAM (stream));
+
+        g_clear_object (&msg->request_body_stream);
+
+        if (content_type) {
+                g_warn_if_fail (strchr (content_type, '/') != NULL);
+
+                soup_message_headers_replace (msg->request_headers, "Content-Type", content_type);
+                if (content_length == -1)
+                        soup_message_headers_set_encoding (msg->request_headers, SOUP_ENCODING_CHUNKED);
+                else
+                        soup_message_headers_set_content_length (msg->request_headers, content_length);
+                msg->request_body_stream = g_object_ref (stream);
+        } else {
+                soup_message_headers_remove (msg->request_headers, "Content-Type");
+                soup_message_headers_remove (msg->request_headers, "Content-Length");
+        }
+}
+
+/**
+ * soup_message_set_request_body_from_bytes:
+ * @msg: the message
+ * @content_type: (allow-none): MIME Content-Type of the body
+ * @bytes: (allow-none): a #GBytes with the request body data
+ *
+ * Set the request body of a #SoupMessage from #GBytes. If
+ * @content_type is %NULL, the request body must be empty (or @bytes %NULL) as well.
+ * The request body needs to be set again in case @msg is restarted
+ * (in case of redirection or authentication).
+ */
+void
+soup_message_set_request_body_from_bytes (SoupMessage  *msg,
+                                          const char   *content_type,
+                                          GBytes       *bytes)
+{
+        g_return_if_fail (SOUP_IS_MESSAGE (msg));
+        g_return_if_fail (content_type == NULL || bytes != NULL);
+        g_return_if_fail (content_type != NULL || g_bytes_get_size (bytes) == 0);
+
+       if (bytes) {
+               GInputStream *stream;
+
+               stream = g_memory_input_stream_new_from_bytes (bytes);
+               soup_message_set_request_body (msg, content_type, stream, g_bytes_get_size (bytes));
+               g_object_unref (stream);
+       } else
+               soup_message_set_request_body (msg, NULL, NULL, 0);
+}
+
 void
 soup_message_wrote_informational (SoupMessage *msg)
 {
@@ -1203,6 +1273,8 @@ soup_message_restarted (SoupMessage *msg)
        if (priv->msg_flags & SOUP_MESSAGE_CAN_REBUILD)
                soup_message_body_truncate (msg->request_body);
 
+       g_clear_object (&msg->request_body_stream);
+
        g_signal_emit (msg, signals[RESTARTED], 0);
 }
 
diff --git a/libsoup/soup-message.h b/libsoup/soup-message.h
index 70a0becb..91cd5f7d 100644
--- a/libsoup/soup-message.h
+++ b/libsoup/soup-message.h
@@ -30,6 +30,7 @@ struct _SoupMessage {
        char               *reason_phrase;
 
        SoupMessageBody    *request_body;
+       GInputStream       *request_body_stream;
        SoupMessageHeaders *request_headers;
 
        SoupMessageBody    *response_body;
@@ -100,6 +101,15 @@ void           soup_message_set_response        (SoupMessage       *msg,
                                                 SoupMemoryUse      resp_use,
                                                 const char        *resp_body,
                                                 gsize              resp_length);
+SOUP_AVAILABLE_IN_ALL
+void           soup_message_set_request_body    (SoupMessage       *msg,
+                                                const char        *content_type,
+                                                GInputStream      *stream,
+                                                gssize             content_length);
+SOUP_AVAILABLE_IN_ALL
+void           soup_message_set_request_body_from_bytes (SoupMessage  *msg,
+                                                        const char   *content_type,
+                                                        GBytes       *bytes);
 
 typedef enum {
        SOUP_HTTP_1_0 = 0, /*< nick=http-1-0 >*/
diff --git a/libsoup/soup-version.h.in b/libsoup/soup-version.h.in
index 53605dfa..308c2f27 100644
--- a/libsoup/soup-version.h.in
+++ b/libsoup/soup-version.h.in
@@ -109,6 +109,7 @@ G_BEGIN_DECLS
 #error "SOUP_VERSION_MIN_REQUIRED must be >= SOUP_VERSION_2_24"
 #endif
 
+#define SOUP_AVAILABLE_IN_ALL                   _SOUP_EXTERN
 #define SOUP_AVAILABLE_IN_2_4                   _SOUP_EXTERN
 
 #if SOUP_VERSION_MIN_REQUIRED >= SOUP_VERSION_2_24
diff --git a/tests/meson.build b/tests/meson.build
index 84e679ea..b01397cd 100644
--- a/tests/meson.build
+++ b/tests/meson.build
@@ -35,7 +35,6 @@ endif
 # ['name', is_parallel, extra_deps]
 tests = [
   ['cache', true, []],
-  ['chunk', true, []],
   ['chunk-io', true, []],
   ['coding', true, []],
   ['context', true, []],
@@ -51,6 +50,7 @@ tests = [
   ['no-ssl', true, []],
   ['ntlm', true, []],
   ['redirect', true, []],
+  ['request-body', true, []],
   ['resource', true, []],
   ['samesite', true, []],
   ['session', true, []],
diff --git a/tests/request-body-test.c b/tests/request-body-test.c
new file mode 100644
index 00000000..fb5cd31d
--- /dev/null
+++ b/tests/request-body-test.c
@@ -0,0 +1,211 @@
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
+/*
+ * Copyright (C) 2008 Red Hat, Inc.
+ * Copyright (C) 2010 Igalia S.L.
+ */
+
+#include "test-utils.h"
+
+static SoupSession *session;
+static SoupURI *base_uri;
+
+typedef struct {
+        SoupSession *session;
+        GInputStream *stream;
+        GBytes *bytes;
+        int nwrote;
+} PutTestData;
+
+typedef enum {
+        BYTES  = 1 << 0,
+        RESTART = 1 << 1,
+        ASYNC = 1 << 2,
+        LARGE = 1 << 3,
+        EMPTY = 1 << 4
+} RequestTestFlags;
+
+static void
+wrote_body_data (SoupMessage *msg,
+                 GBytes      *chunk,
+                 PutTestData *ptd)
+{
+        debug_printf (2, "  wrote_body_data, %d bytes\n",
+                      (int)g_bytes_get_size (chunk));
+        ptd->nwrote += g_bytes_get_size (chunk);
+}
+
+static GChecksum *
+setup_request_body (PutTestData     *ptd,
+                    RequestTestFlags flags)
+{
+        GChecksum *check;
+
+        ptd->nwrote = 0;
+        check = g_checksum_new (G_CHECKSUM_MD5);
+        if (flags & LARGE) {
+                static const unsigned int large_size = 1000000;
+                char *large_data;
+                unsigned int i;
+
+                large_data = g_malloc (large_size);
+                for (i = 0; i < large_size; i++)
+                        large_data[i] = i & 0xFF;
+                ptd->bytes = g_bytes_new_take (large_data, large_size);
+                g_checksum_update (check, (guchar *)large_data, large_size);
+        } else if (flags & EMPTY) {
+                ptd->bytes = g_bytes_new_static (NULL, 0);
+        } else {
+                static const char *data = "one two three";
+
+                ptd->bytes = g_bytes_new_static (data, strlen (data));
+                g_checksum_update (check, (guchar *)data, strlen (data));
+        }
+        ptd->stream = flags & BYTES ? NULL : g_memory_input_stream_new_from_bytes (ptd->bytes);
+
+        return check;
+}
+
+static void
+restarted (SoupMessage *msg,
+           PutTestData *ptd)
+{
+        debug_printf (2, "  --restarting--\n");
+
+        ptd->nwrote = 0;
+
+        /* FIXME: The 302 redirect will turn it into a GET request */
+        msg->method = SOUP_METHOD_PUT;
+
+        if (ptd->stream) {
+                g_object_unref (ptd->stream);
+                ptd->stream = g_memory_input_stream_new_from_bytes (ptd->bytes);
+                soup_message_set_request_body (msg, "text/plain", ptd->stream, -1);
+        } else {
+                soup_message_set_request_body_from_bytes (msg, "text/plain", ptd->bytes);
+        }
+}
+
+static void
+do_request_test (gconstpointer data)
+{
+        RequestTestFlags flags = GPOINTER_TO_UINT (data);
+        SoupURI *uri;
+        PutTestData ptd;
+        SoupMessage *msg;
+        const char *client_md5, *server_md5;
+        GChecksum *check;
+
+        if (flags & RESTART)
+                uri = soup_uri_new_with_base (base_uri, "/redirect");
+        else
+                uri = soup_uri_copy (base_uri);
+
+        ptd.session = session;
+        check = setup_request_body (&ptd, flags);
+        client_md5 = g_checksum_get_string (check);
+
+        msg = soup_message_new_from_uri ("PUT", uri);
+        if (flags & BYTES)
+                soup_message_set_request_body_from_bytes (msg, flags & EMPTY ? NULL : "text/plain", 
ptd.bytes);
+        else
+                soup_message_set_request_body (msg, "text/plain", ptd.stream, -1);
+
+        if (flags & RESTART) {
+                g_signal_connect (msg, "restarted",
+                                  G_CALLBACK (restarted), &ptd);
+        }
+
+        g_signal_connect (msg, "wrote-body-data",
+                          G_CALLBACK (wrote_body_data), &ptd);
+
+        if (flags & ASYNC)
+                soup_test_session_async_send (session, msg);
+        else
+                soup_test_session_send_message (session, msg);
+        soup_test_assert_message_status (msg, SOUP_STATUS_CREATED);
+        g_assert_cmpint (g_bytes_get_size (ptd.bytes), ==, ptd.nwrote);
+
+        server_md5 = soup_message_headers_get_one (msg->response_headers,
+                                                   "Content-MD5");
+        g_assert_cmpstr (client_md5, ==, server_md5);
+
+        g_bytes_unref (ptd.bytes);
+        g_clear_object (&ptd.stream);
+        g_object_unref (msg);
+        g_checksum_free (check);
+
+        soup_uri_free (uri);
+}
+
+static void
+server_callback (SoupServer *server, SoupMessage *msg,
+                 const char *path, GHashTable *query,
+                 SoupClientContext *context, gpointer data)
+{
+        SoupMessageBody *md5_body;
+        char *md5;
+
+        if (g_str_has_prefix (path, "/redirect")) {
+                soup_message_set_redirect (msg, SOUP_STATUS_FOUND, "/");
+                return;
+        }
+
+        if (msg->method == SOUP_METHOD_PUT) {
+                soup_message_set_status (msg, SOUP_STATUS_CREATED);
+                md5_body = msg->request_body;
+        } else {
+                soup_message_set_status (msg, SOUP_STATUS_METHOD_NOT_ALLOWED);
+                return;
+        }
+
+        md5 = g_compute_checksum_for_data (G_CHECKSUM_MD5,
+                                           (guchar *)md5_body->data,
+                                           md5_body->length);
+        soup_message_headers_append (msg->response_headers,
+                                     "Content-MD5", md5);
+        g_free (md5);
+}
+
+int
+main (int argc, char **argv)
+{
+        GMainLoop *loop;
+        SoupServer *server;
+        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);
+
+        loop = g_main_loop_new (NULL, TRUE);
+
+        base_uri = soup_test_server_get_uri (server, "http", NULL);
+        session = soup_test_session_new (SOUP_TYPE_SESSION, NULL);
+
+        g_test_add_data_func ("/request-body/sync/stream", GINT_TO_POINTER (0), do_request_test);
+        g_test_add_data_func ("/request-body/sync/bytes", GINT_TO_POINTER (BYTES), do_request_test);
+        g_test_add_data_func ("/request-body/sync/restart-stream", GINT_TO_POINTER (RESTART), 
do_request_test);
+        g_test_add_data_func ("/request-body/sync/restart-bytes", GINT_TO_POINTER (RESTART | BYTES), 
do_request_test);
+        g_test_add_data_func ("/request-body/sync/large", GINT_TO_POINTER (BYTES | LARGE), do_request_test);
+        g_test_add_data_func ("/request-body/sync/empty", GINT_TO_POINTER (BYTES | EMPTY), do_request_test);
+        g_test_add_data_func ("/request-body/async/stream", GINT_TO_POINTER (ASYNC), do_request_test);
+        g_test_add_data_func ("/request-body/async/bytes", GINT_TO_POINTER (BYTES | ASYNC), do_request_test);
+        g_test_add_data_func ("/request-body/async/restart-stream", GINT_TO_POINTER (RESTART | ASYNC), 
do_request_test);
+        g_test_add_data_func ("/request-body/async/restart-bytes", GINT_TO_POINTER (RESTART | ASYNC | 
BYTES), do_request_test);
+        g_test_add_data_func ("/request-body/async/large", GINT_TO_POINTER (BYTES | LARGE | ASYNC), 
do_request_test);
+        g_test_add_data_func ("/request-body/async/empty", GINT_TO_POINTER (BYTES | EMPTY | ASYNC), 
do_request_test);
+
+        ret = g_test_run ();
+
+        soup_test_session_abort_unref (session);
+
+        soup_uri_free (base_uri);
+
+        g_main_loop_unref (loop);
+        soup_test_server_quit_unref (server);
+
+        test_cleanup ();
+        return ret;
+}


[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]