summaryrefslogtreecommitdiffstats
path: root/ext/soup
diff options
context:
space:
mode:
Diffstat (limited to 'ext/soup')
-rw-r--r--ext/soup/Makefile.am9
-rw-r--r--ext/soup/gstsouphttpsrc.c378
-rw-r--r--ext/soup/gstsouphttpsrc.h61
3 files changed, 448 insertions, 0 deletions
diff --git a/ext/soup/Makefile.am b/ext/soup/Makefile.am
new file mode 100644
index 00000000..39d89dda
--- /dev/null
+++ b/ext/soup/Makefile.am
@@ -0,0 +1,9 @@
+plugin_LTLIBRARIES = libgstsouphttpsrc.la
+
+libgstsouphttpsrc_la_SOURCES = gstsouphttpsrc.c
+
+libgstsouphttpsrc_la_CFLAGS = $(GST_CFLAGS) $(SOUP_CFLAGS)
+libgstsouphttpsrc_la_LIBADD = $(GST_BASE_LIBS) $(SOUP_LIBS)
+libgstsouphttpsrc_la_LDFLAGS = $(GST_PLUGIN_LDFLAGS)
+
+noinst_HEADERS = gstsouphttpsrc.h
diff --git a/ext/soup/gstsouphttpsrc.c b/ext/soup/gstsouphttpsrc.c
new file mode 100644
index 00000000..330f9791
--- /dev/null
+++ b/ext/soup/gstsouphttpsrc.c
@@ -0,0 +1,378 @@
+/* GStreamer
+ * Copyright (C) <2007> Wouter Cloetens <wouter@mind.be>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Library General Public License for more
+ */
+
+#ifdef HAVE_CONFIG_H
+#include "config.h"
+#endif
+
+#include <string.h>
+#include <libsoup/soup.h>
+#include "gstsouphttpsrc.h"
+
+GST_DEBUG_CATEGORY_STATIC (souphttpsrc_debug);
+#define GST_CAT_DEFAULT souphttpsrc_debug
+
+static const GstElementDetails gst_souphttp_src_details =
+GST_ELEMENT_DETAILS ("HTTP client source",
+ "Source/Network",
+ "Receive data as a client over the network via HTTP using SOUP",
+ "Wouter Cloetens <wouter@mind.be>");
+
+static GstStaticPadTemplate srctemplate = GST_STATIC_PAD_TEMPLATE ("src",
+ GST_PAD_SRC,
+ GST_PAD_ALWAYS,
+ GST_STATIC_CAPS_ANY);
+
+enum
+{
+ PROP_0,
+ PROP_LOCATION,
+};
+
+static void gst_souphttp_src_dispose (GObject * gobject);
+static void gst_souphttp_src_set_property (GObject * object, guint prop_id,
+ const GValue * value, GParamSpec * pspec);
+static void gst_souphttp_src_get_property (GObject * object, guint prop_id,
+ GValue * value, GParamSpec * pspec);
+
+static GstFlowReturn gst_souphttp_src_create (GstPushSrc * psrc,
+ GstBuffer ** outbuf);
+static gboolean gst_souphttp_src_start (GstBaseSrc * bsrc);
+static gboolean gst_souphttp_src_stop (GstBaseSrc * bsrc);
+static gboolean gst_souphttp_src_unlock (GstBaseSrc * bsrc);
+
+static gboolean gst_souphttp_src_set_location (GstSouphttpSrc * src,
+ const gchar * uri);
+
+static void soup_got_chunk (SoupMessage * msg, GstSouphttpSrc * src);
+static void soup_response (SoupMessage * msg, gpointer user_data);
+static void soup_session_close (GstSouphttpSrc * src);
+
+static void
+_do_init (GType type)
+{
+ GST_DEBUG_CATEGORY_INIT (souphttpsrc_debug, "souphttpsrc", 0,
+ "SOUP HTTP src");
+}
+
+GST_BOILERPLATE_FULL (GstSouphttpSrc, gst_souphttp_src, GstPushSrc,
+ GST_TYPE_PUSH_SRC, _do_init);
+
+static void
+gst_souphttp_src_base_init (gpointer g_class)
+{
+ GstElementClass *element_class = GST_ELEMENT_CLASS (g_class);
+
+ gst_element_class_add_pad_template (element_class,
+ gst_static_pad_template_get (&srctemplate));
+
+ gst_element_class_set_details (element_class, &gst_souphttp_src_details);
+}
+
+static void
+gst_souphttp_src_class_init (GstSouphttpSrcClass * klass)
+{
+ GObjectClass *gobject_class;
+ GstBaseSrcClass *gstbasesrc_class;
+ GstPushSrcClass *gstpushsrc_class;
+
+ gobject_class = (GObjectClass *) klass;
+ gstbasesrc_class = (GstBaseSrcClass *) klass;
+ gstpushsrc_class = (GstPushSrcClass *) klass;
+
+ gobject_class->set_property = gst_souphttp_src_set_property;
+ gobject_class->get_property = gst_souphttp_src_get_property;
+ gobject_class->dispose = gst_souphttp_src_dispose;
+
+ g_object_class_install_property
+ (gobject_class, PROP_LOCATION,
+ g_param_spec_string ("location", "Location",
+ "Location to read from", "", G_PARAM_READWRITE));
+
+ gstbasesrc_class->start = GST_DEBUG_FUNCPTR (gst_souphttp_src_start);
+ gstbasesrc_class->stop = GST_DEBUG_FUNCPTR (gst_souphttp_src_stop);
+ gstbasesrc_class->unlock = GST_DEBUG_FUNCPTR (gst_souphttp_src_unlock);
+
+ gstpushsrc_class->create = GST_DEBUG_FUNCPTR (gst_souphttp_src_create);
+
+ GST_DEBUG_CATEGORY_INIT (souphttpsrc_debug, "souphttpsrc", 0,
+ "SOUP HTTP Client Source");
+}
+
+static void
+gst_souphttp_src_init (GstSouphttpSrc * src, GstSouphttpSrcClass * g_class)
+{
+ src->location = NULL;
+ src->loop = NULL;
+ src->session = NULL;
+ src->msg = NULL;
+ src->interrupted = FALSE;
+}
+
+static void
+gst_souphttp_src_dispose (GObject * gobject)
+{
+ GstSouphttpSrc *src = GST_SOUPHTTP_SRC (gobject);
+
+ GST_DEBUG_OBJECT (src, "dispose");
+ soup_session_close (src);
+ if (src->loop) {
+ g_main_loop_unref (src->loop);
+ src->loop = NULL;
+ }
+ if (src->location) {
+ g_free (src->location);
+ }
+
+ G_OBJECT_CLASS (parent_class)->dispose (gobject);
+}
+
+static void
+gst_souphttp_src_set_property (GObject * object, guint prop_id,
+ const GValue * value, GParamSpec * pspec)
+{
+ GstSouphttpSrc *src = GST_SOUPHTTP_SRC (object);
+
+ switch (prop_id) {
+ case PROP_LOCATION:
+ {
+ const gchar *location;
+
+ location = g_value_get_string (value);
+
+ if (location == NULL) {
+ GST_WARNING ("location property cannot be NULL");
+ goto done;
+ }
+ if (!gst_souphttp_src_set_location (src, location)) {
+ GST_WARNING ("badly formatted location");
+ goto done;
+ }
+ break;
+ }
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+done:
+ return;
+}
+
+static void
+gst_souphttp_src_get_property (GObject * object, guint prop_id,
+ GValue * value, GParamSpec * pspec)
+{
+ GstSouphttpSrc *souphttpsrc = GST_SOUPHTTP_SRC (object);
+
+ switch (prop_id) {
+ case PROP_LOCATION:
+ g_value_set_string (value, souphttpsrc->location);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+}
+
+static GstFlowReturn
+gst_souphttp_src_create (GstPushSrc * psrc, GstBuffer ** outbuf)
+{
+ GstSouphttpSrc *src;
+
+ src = GST_SOUPHTTP_SRC (psrc);
+
+ src->ret = GST_FLOW_CUSTOM_ERROR;
+ src->outbuf = outbuf;
+ do {
+ if (src->interrupted) {
+ soup_session_close (src);
+ src->ret = GST_FLOW_UNEXPECTED;
+ break;
+ }
+ if (!src->session) {
+ GST_DEBUG_OBJECT (src, "EOS reached");
+ src->ret = GST_FLOW_UNEXPECTED;
+ break;
+ }
+
+ switch (src->msg->status) {
+ case SOUP_MESSAGE_STATUS_IDLE:
+ GST_DEBUG_OBJECT (src, "Queueing connection request");
+ soup_session_queue_message (src->session, src->msg, soup_response, src);
+ break;
+ case SOUP_MESSAGE_STATUS_FINISHED:
+ GST_DEBUG_OBJECT (src, "Connection closed");
+ soup_session_close (src);
+ src->ret = GST_FLOW_UNEXPECTED;
+ break;
+ case SOUP_MESSAGE_STATUS_QUEUED:
+ case SOUP_MESSAGE_STATUS_CONNECTING:
+ case SOUP_MESSAGE_STATUS_RUNNING:
+ default:
+ soup_message_io_unpause (src->msg);
+ break;
+ }
+
+ if (src->ret == GST_FLOW_CUSTOM_ERROR)
+ g_main_loop_run (src->loop);
+ } while (src->ret == GST_FLOW_CUSTOM_ERROR);
+
+ return src->ret;
+}
+
+/* create a socket for connecting to remote server */
+static gboolean
+gst_souphttp_src_start (GstBaseSrc * bsrc)
+{
+ GstSouphttpSrc *src = GST_SOUPHTTP_SRC (bsrc);
+ GMainContext *context;
+
+ if (!src->location) {
+ GST_ELEMENT_ERROR (src, LIBRARY, INIT,
+ (NULL), ("Missing location property"));
+ return FALSE;
+ }
+
+ context = g_main_context_new ();
+
+ src->loop = g_main_loop_new (context, TRUE);
+ if (!src->loop) {
+ GST_ELEMENT_ERROR (src, LIBRARY, INIT,
+ (NULL), ("Failed to start GMainLoop"));
+ return FALSE;
+ }
+
+ src->session =
+ soup_session_async_new_with_options (SOUP_SESSION_ASYNC_CONTEXT, context,
+ NULL);
+ if (!src->session) {
+ GST_ELEMENT_ERROR (src, LIBRARY, INIT,
+ (NULL), ("Failed to create async session"));
+ return FALSE;
+ }
+
+ src->msg = soup_message_new (SOUP_METHOD_GET, src->location);
+ if (!src->msg) {
+ GST_ELEMENT_ERROR (src, LIBRARY, INIT, (NULL), ("Error parsing URL"));
+ return FALSE;
+ }
+
+ g_signal_connect (src->msg, "got_chunk", G_CALLBACK (soup_got_chunk), src);
+ soup_message_set_flags (src->msg, SOUP_MESSAGE_OVERWRITE_CHUNKS);
+
+ return TRUE;
+}
+
+/* close the socket and associated resources
+ * used both to recover from errors and go to NULL state */
+static gboolean
+gst_souphttp_src_stop (GstBaseSrc * bsrc)
+{
+ GstSouphttpSrc *src;
+
+ src = GST_SOUPHTTP_SRC (bsrc);
+ GST_DEBUG_OBJECT (src, "stop()");
+ soup_session_close (src);
+
+ return TRUE;
+}
+
+/* Interrupt a blocking request. */
+static gboolean
+gst_souphttp_src_unlock (GstBaseSrc * bsrc)
+{
+ GstSouphttpSrc *src;
+
+ src = GST_SOUPHTTP_SRC (bsrc);
+ GST_DEBUG_OBJECT (src, "unlock()");
+
+ src->interrupted = TRUE;
+ g_main_loop_quit (src->loop);
+
+ return TRUE;
+}
+
+static gboolean
+gst_souphttp_src_set_location (GstSouphttpSrc * src, const gchar * uri)
+{
+ if (src->location) {
+ g_free (src->location);
+ src->location = NULL;
+ }
+ src->location = g_strdup (uri);
+
+ return TRUE;
+}
+
+static void
+soup_got_chunk (SoupMessage * msg, GstSouphttpSrc * src)
+{
+ GstBaseSrc *basesrc;
+
+ basesrc = GST_BASE_SRC_CAST (src);
+ GST_DEBUG_OBJECT (src, "got chunk of %d bytes", msg->response.length);
+
+ /* Create the buffer. */
+ src->ret = gst_pad_alloc_buffer (GST_BASE_SRC_PAD (basesrc),
+ basesrc->segment.last_stop, msg->response.length,
+ GST_PAD_CAPS (GST_BASE_SRC_PAD (basesrc)), src->outbuf);
+ if (G_LIKELY (src->ret == GST_FLOW_OK))
+ memcpy (GST_BUFFER_DATA (*src->outbuf), msg->response.body,
+ msg->response.length);
+
+ g_main_loop_quit (src->loop);
+ soup_message_io_pause (msg);
+}
+
+static void
+soup_response (SoupMessage * msg, gpointer user_data)
+{
+ GstSouphttpSrc *src = (GstSouphttpSrc *) user_data;
+
+ GST_DEBUG_OBJECT (src, "got response %d: %s", msg->status_code,
+ msg->reason_phrase);
+ g_main_loop_quit (src->loop);
+}
+
+static void
+soup_session_close (GstSouphttpSrc * src)
+{
+ GST_DEBUG_OBJECT (src, "Connection closed");
+ if (src->session) {
+ soup_session_abort (src->session); /* This unrefs the message. */
+ g_object_unref (src->session);
+ src->session = NULL;
+ src->msg = NULL;
+ }
+}
+
+/* entry point to initialize the plug-in
+ * initialize the plug-in itself
+ * register the element factories and pad templates
+ * register the features
+ */
+static gboolean
+plugin_init (GstPlugin * plugin)
+{
+ return gst_element_register (plugin, "souphttpsrc", GST_RANK_NONE,
+ GST_TYPE_SOUPHTTP_SRC);
+}
+
+/* this is the structure that gst-register looks for
+ * so keep the name plugin_desc, or you cannot get your plug-in registered */
+GST_PLUGIN_DEFINE (GST_VERSION_MAJOR,
+ GST_VERSION_MINOR,
+ "soup",
+ "libsoup http client src",
+ plugin_init, VERSION, "LGPL", "GStreamer", "http://gstreamer.net/")
diff --git a/ext/soup/gstsouphttpsrc.h b/ext/soup/gstsouphttpsrc.h
new file mode 100644
index 00000000..71d1a83d
--- /dev/null
+++ b/ext/soup/gstsouphttpsrc.h
@@ -0,0 +1,61 @@
+/* GStreamer
+ * Copyright (C) <2007> Wouter Cloetens <wouter@mind.be>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Library General Public License for more
+ */
+
+#ifndef __GST_SOUPHTTP_SRC_H__
+#define __GST_SOUPHTTP_SRC_H__
+
+#include <gst/gst.h>
+#include <gst/base/gstpushsrc.h>
+#include <glib.h>
+
+G_BEGIN_DECLS
+
+#include <libsoup/soup.h>
+
+#define GST_TYPE_SOUPHTTP_SRC \
+ (gst_souphttp_src_get_type())
+#define GST_SOUPHTTP_SRC(obj) \
+ (G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_SOUPHTTP_SRC,GstSouphttpSrc))
+#define GST_SOUPHTTP_SRC_CLASS(klass) \
+ (G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_SOUPHTTP_SRC,GstSouphttpSrcClass))
+#define GST_IS_SOUPHTTP_SRC(obj) \
+ (G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_SOUPHTTP_SRC))
+#define GST_IS_SOUPHTTP_SRC_CLASS(klass) \
+ (G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_SOUPHTTP_SRC))
+
+typedef struct _GstSouphttpSrc GstSouphttpSrc;
+typedef struct _GstSouphttpSrcClass GstSouphttpSrcClass;
+
+struct _GstSouphttpSrc {
+ GstPushSrc element;
+
+ gchar *location; /* Full URI. */
+ GMainLoop *loop; /* Event loop. */
+ SoupSession *session; /* Async context. */
+ SoupMessage *msg; /* Request message. */
+ GstFlowReturn ret; /* Return code from callback. */
+ GstBuffer **outbuf; /* Return buffer allocated by callback. */
+ gboolean interrupted; /* Signal unlock(). */
+};
+
+struct _GstSouphttpSrcClass {
+ GstPushSrcClass parent_class;
+};
+
+GType gst_souphttp_src_get_type (void);
+
+G_END_DECLS
+
+#endif /* __GST_SOUPHTTP_SRC_H__ */
+