summaryrefslogtreecommitdiff
path: root/libgupnp-dlna/gupnp-dlna-discoverer.c
diff options
context:
space:
mode:
Diffstat (limited to 'libgupnp-dlna/gupnp-dlna-discoverer.c')
-rw-r--r--libgupnp-dlna/gupnp-dlna-discoverer.c443
1 files changed, 443 insertions, 0 deletions
diff --git a/libgupnp-dlna/gupnp-dlna-discoverer.c b/libgupnp-dlna/gupnp-dlna-discoverer.c
new file mode 100644
index 0000000..faf21ff
--- /dev/null
+++ b/libgupnp-dlna/gupnp-dlna-discoverer.c
@@ -0,0 +1,443 @@
+/*
+ * Copyright (C) 2010 Nokia Corporation.
+ *
+ * Authors: Arun Raghavan <arun.raghavan@collabora.co.uk>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser 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
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#include "gupnp-dlna-discoverer.h"
+#include "gupnp-dlna-marshal.h"
+#include "profile-loading.h"
+
+/**
+ * SECTION:gupnp-dlna-discoverer
+ * @short_description: Utility API for discovering DLNA profile/mime type and
+ * other metadata for given media.
+ *
+ * The GUPnPDLNADiscoverer object provides a light-weight wrapper over the
+ * #GstDiscoverer API. The latter provides a simple interface to discover
+ * media metadata given a URI. GUPnPDLNADiscoverer extends this API to also
+ * provide a DLNA profile name and mime type for the media.
+ *
+ * The API provided corresponds very closely to the API provided by
+ * #GstDiscoverer - both synchronous and asynchronous discovery of metadata
+ * are possible.
+ *
+ * The asynchronous mode requires a running #GMainLoop in the default
+ * #GMainContext, where one connects to the various signals, appends the
+ * URIs to be processed and then asks for the discovery to begin.
+ */
+enum {
+ DONE,
+ SIGNAL_LAST
+};
+
+static guint signals[SIGNAL_LAST];
+
+
+G_DEFINE_TYPE (GUPnPDLNADiscoverer, gupnp_dlna_discoverer, GST_TYPE_DISCOVERER)
+
+#define GET_PRIVATE(o) \
+ (G_TYPE_INSTANCE_GET_PRIVATE ((o), \
+ GUPNP_TYPE_DLNA_DISCOVERER, \
+ GUPnPDLNADiscovererPrivate))
+
+typedef struct _GUPnPDLNADiscovererPrivate GUPnPDLNADiscovererPrivate;
+
+struct _GUPnPDLNADiscovererPrivate {
+ gboolean relaxed_mode;
+ gboolean extended_mode;
+};
+
+enum {
+ PROP_0,
+ PROP_DLNA_RELAXED_MODE,
+ PROP_DLNA_EXTENDED_MODE,
+};
+
+static void
+gupnp_dlna_discoverer_set_property (GObject *object,
+ guint property_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ GUPnPDLNADiscoverer *self = GUPNP_DLNA_DISCOVERER (object);
+ GUPnPDLNADiscovererPrivate *priv = GET_PRIVATE (self);
+
+ switch (property_id) {
+ case PROP_DLNA_RELAXED_MODE:
+ priv->relaxed_mode = g_value_get_boolean (value);
+ break;
+
+ case PROP_DLNA_EXTENDED_MODE:
+ priv->extended_mode = g_value_get_boolean (value);
+ break;
+
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object,
+ property_id,
+ pspec);
+ break;
+ }
+}
+
+static void
+gupnp_dlna_discoverer_get_property (GObject *object,
+ guint property_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ GUPnPDLNADiscoverer *self = GUPNP_DLNA_DISCOVERER (object);
+ GUPnPDLNADiscovererPrivate *priv = GET_PRIVATE (self);
+
+ switch (property_id) {
+ case PROP_DLNA_RELAXED_MODE:
+ g_value_set_boolean (value, priv->relaxed_mode);
+ break;
+
+ case PROP_DLNA_EXTENDED_MODE:
+ g_value_set_boolean (value, priv->extended_mode);
+ break;
+
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object,
+ property_id,
+ pspec);
+ break;
+ }
+}
+
+static void
+gupnp_dlna_discoverer_dispose (GObject *object)
+{
+ G_OBJECT_CLASS (gupnp_dlna_discoverer_parent_class)->dispose (object);
+}
+
+static void
+gupnp_dlna_discoverer_finalize (GObject *object)
+{
+ G_OBJECT_CLASS (gupnp_dlna_discoverer_parent_class)->finalize (object);
+}
+
+static void
+gupnp_dlna_discovered_cb (GstDiscoverer *discoverer,
+ GstDiscovererInfo *info,
+ GError *err)
+{
+ GUPnPDLNAInformation *dlna = NULL;
+ GUPnPDLNADiscovererClass *klass =
+ GUPNP_DLNA_DISCOVERER_GET_CLASS (discoverer);
+ GUPnPDLNADiscovererPrivate *priv =
+ GET_PRIVATE (GUPNP_DLNA_DISCOVERER (discoverer));
+ gboolean relaxed = priv->relaxed_mode;
+ gboolean extended = priv->extended_mode;
+
+ if (info)
+ dlna = gupnp_dlna_information_new_from_discoverer_info
+ (info,
+ klass->profiles_list
+ [relaxed][extended]);
+
+ g_signal_emit (GUPNP_DLNA_DISCOVERER (discoverer),
+ signals[DONE], 0, dlna, err);
+
+ if (dlna)
+ g_object_unref (dlna);
+}
+
+static void
+gupnp_dlna_discoverer_class_init (GUPnPDLNADiscovererClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+ GParamSpec *pspec;
+
+ g_type_class_add_private (klass, sizeof (GUPnPDLNADiscovererPrivate));
+
+ object_class->get_property = gupnp_dlna_discoverer_get_property;
+ object_class->set_property = gupnp_dlna_discoverer_set_property;
+ object_class->dispose = gupnp_dlna_discoverer_dispose;
+ object_class->finalize = gupnp_dlna_discoverer_finalize;
+
+ /**
+ * GUPnPDLNADiscoverer::relaxed-mode:
+ * @relaxed_mode: setting to true will enable relaxed mode
+ *
+ * The current release does not support relaxed mode yet
+ */
+ pspec = g_param_spec_boolean ("relaxed-mode",
+ "Relaxed mode property",
+ "Indicates that profile matching should"
+ "be strictly compliant with the DLNA "
+ "specification",
+ FALSE,
+ G_PARAM_READWRITE |
+ G_PARAM_CONSTRUCT_ONLY);
+ g_object_class_install_property (object_class,
+ PROP_DLNA_RELAXED_MODE,
+ pspec);
+
+ /**
+ * GUPnPDLNADiscoverer::extended-mode:
+ * @extended: setting true will enable extended profile support
+ *
+ * The current release does not support extended mode yet
+ */
+ pspec = g_param_spec_boolean ("extended-mode",
+ "Extended mode property",
+ "Indicates support for profiles that are "
+ "not part of the DLNA specification",
+ FALSE,
+ G_PARAM_READWRITE |
+ G_PARAM_CONSTRUCT_ONLY);
+ g_object_class_install_property (object_class,
+ PROP_DLNA_EXTENDED_MODE,
+ pspec);
+
+ /**
+ * GUPnPDLNADiscoverer::done:
+ * @discoverer: the #GUPnPDLNADiscoverer
+ * @dlna: the results as #GUPnPDLNAInformation
+ * @err: contains details of the error if discovery fails, else is NULL
+ *
+ * Will be emitted when all information on a URI could be discovered.
+ *
+ * The reciever must unref @dlna with when done using it.
+ */
+ signals[DONE] =
+ g_signal_new ("done", G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_LAST,
+ G_STRUCT_OFFSET (GUPnPDLNADiscovererClass, done),
+ NULL, NULL,
+ gupnp_dlna_marshal_VOID__OBJECT_BOXED,
+ G_TYPE_NONE, 2, GUPNP_TYPE_DLNA_INFORMATION,
+ GST_TYPE_G_ERROR);
+
+ /* Load DLNA profiles from disk */
+ if (g_type_from_name ("GstElement")) {
+ klass->profiles_list [0][0]
+ = gupnp_dlna_load_profiles_from_disk (FALSE,
+ FALSE);
+ klass->profiles_list [0][1]
+ = gupnp_dlna_load_profiles_from_disk (FALSE,
+ TRUE);
+ klass->profiles_list [1][0]
+ = gupnp_dlna_load_profiles_from_disk (TRUE,
+ FALSE);
+ klass->profiles_list [1][1]
+ = gupnp_dlna_load_profiles_from_disk (TRUE,
+ TRUE);
+ } else {
+ klass->profiles_list [0][0] = NULL;
+ klass->profiles_list [0][1] = NULL;
+ klass->profiles_list [1][0] = NULL;
+ klass->profiles_list [1][1] = NULL;
+ g_warning ("GStreamer has not yet been initialised. You need "
+ "to call gst_init()/gst_init_check() for discovery "
+ "to work.");
+ }
+}
+
+static void
+gupnp_dlna_discoverer_init (GUPnPDLNADiscoverer *self)
+{
+ g_signal_connect (&self->parent,
+ "discovered",
+ G_CALLBACK (gupnp_dlna_discovered_cb),
+ NULL);
+}
+
+/**
+ * gupnp_dlna_discoverer_new:
+ * @timeout: default discovery timeout, in nanoseconds
+ * @relaxed_mode: set to TRUE, to enable relaxed mode support. FALSE otherwise
+ * @extended_mode: set to TRUE, to enable extended mode support. FALSE otherwise
+ *
+ * Creates a new #GUPnPDLNADiscoverer object with the given default timeout
+ * value.
+ *
+ * Returns: A new #GUPnPDLNADiscoverer object.
+ */
+GUPnPDLNADiscoverer*
+gupnp_dlna_discoverer_new (GstClockTime timeout,
+ gboolean relaxed_mode,
+ gboolean extended_mode)
+{
+ return g_object_new (GUPNP_TYPE_DLNA_DISCOVERER,
+ "timeout", timeout,
+ "relaxed-mode", relaxed_mode,
+ "extended-mode", extended_mode,
+ NULL);
+}
+
+/* Asynchronous API */
+
+/**
+ * gupnp_dlna_discoverer_start:
+ * @discoverer: #GUPnPDLNADiscoverer object to start discovery on
+ *
+ * Allows asynchronous discovery of URIs to begin.
+ */
+
+/**
+ * gupnp_dlna_discoverer_stop:
+ * @discoverer: #GUPnPDLNADiscoverer object to stop discovery on
+ *
+ * Stops asynchronous discovery of URIs.
+ */
+
+/**
+ * gupnp_dlna_discoverer_discover_uri:
+ * @discoverer: #GUPnPDLNADiscoverer object to use for discovery
+ * @uri: URI to gather metadata for
+ *
+ * Queues @uri for metadata discovery. When discovery is completed, the
+ * "discovered" signal is emitted on @discoverer.
+ *
+ * Returns: TRUE if @uri was successfully queued, FALSE otherwise.
+ */
+gboolean
+gupnp_dlna_discoverer_discover_uri (GUPnPDLNADiscoverer *discoverer,
+ const gchar *uri)
+{
+ return gst_discoverer_discover_uri_async (GST_DISCOVERER (discoverer),
+ uri);
+}
+
+/* Synchronous API */
+
+/**
+ * gupnp_dlna_discoverer_discover_uri_sync:
+ * @discoverer: #GUPnPDLNADiscoverer object to use for discovery
+ * @uri: URI to gather metadata for
+ * @err: contains details of the error if discovery fails, else is NULL
+ *
+ * Synchronously gathers metadata for @uri.
+ *
+ * Returns: (transfer full): a #GUPnPDLNAInformation with the metadata for @uri
+ * on success, NULL otherwise
+ */
+GUPnPDLNAInformation *
+gupnp_dlna_discoverer_discover_uri_sync (GUPnPDLNADiscoverer *discoverer,
+ const gchar *uri,
+ GError **err)
+{
+ GstDiscovererInfo *info;
+ GUPnPDLNADiscovererClass *klass =
+ GUPNP_DLNA_DISCOVERER_GET_CLASS (discoverer);
+ GUPnPDLNADiscovererPrivate *priv = GET_PRIVATE (discoverer);
+ gboolean relaxed = priv->relaxed_mode;
+ gboolean extended = priv->extended_mode;
+
+ info = gst_discoverer_discover_uri (GST_DISCOVERER (discoverer),
+ uri,
+ err);
+
+ if (info)
+ return gupnp_dlna_information_new_from_discoverer_info
+ (info, klass->profiles_list [relaxed][extended]);
+
+ return NULL;
+}
+
+/**
+ * gupnp_dlna_discoverer_get_profile:
+ * @self: The #GUPnPDLNADiscoverer object
+ * @name: The name of the DLNA profile to be retrieved
+ *
+ * Given @name, this finds the corresponding DLNA profile information (stored
+ * as a #GUPnPDLNAProfile).
+ *
+ * Returns: (transfer full): a #GUPnPDLNAProfile on success, NULL otherwise.
+ **/
+GUPnPDLNAProfile *
+gupnp_dlna_discoverer_get_profile (GUPnPDLNADiscoverer *self,
+ const gchar *name)
+{
+ GList *i;
+ GUPnPDLNADiscovererClass *klass;
+ GUPnPDLNADiscovererPrivate *priv = GET_PRIVATE (self);
+ gboolean relaxed = priv->relaxed_mode;
+ gboolean extended = priv->extended_mode;
+
+ g_return_val_if_fail (self != NULL, NULL);
+ klass = GUPNP_DLNA_DISCOVERER_GET_CLASS (self);
+
+ for (i = klass->profiles_list [relaxed][extended];
+ i != NULL;
+ i = i->next) {
+ GUPnPDLNAProfile *profile = (GUPnPDLNAProfile *) i->data;
+
+ if (g_str_equal (gupnp_dlna_profile_get_name (profile), name)) {
+ g_object_ref (profile);
+ return profile;
+ }
+ }
+
+ return NULL;
+}
+
+/**
+ * gupnp_dlna_discoverer_list_profiles:
+ * @self: The #GUPnPDLNADiscoverer whose profile list is required
+ *
+ * Retuns a list of the all the DLNA profiles supported by @self.
+ *
+ * Returns: (transfer none) (element-type GUPnPDLNAProfile*): a #GList of
+ * #GUPnPDLNAProfile on success, NULL otherwise.
+ **/
+const GList *
+gupnp_dlna_discoverer_list_profiles (GUPnPDLNADiscoverer *self)
+{
+ GUPnPDLNADiscovererClass *klass;
+ GUPnPDLNADiscovererPrivate *priv = GET_PRIVATE (self);
+ gboolean relaxed = priv->relaxed_mode;
+ gboolean extended = priv->extended_mode;
+
+ g_return_val_if_fail (self != NULL, NULL);
+
+ klass = GUPNP_DLNA_DISCOVERER_GET_CLASS (self);
+
+ return klass->profiles_list [relaxed][extended];
+}
+
+/**
+ * gupnp_dlna_discoverer_get_relaxed_mode:
+ * @self: The #GUPnPDLNADiscoverer object
+ *
+ * Returns: true if relaxed mode is set and false otherwise
+ */
+gboolean
+gupnp_dlna_discoverer_get_relaxed_mode (GUPnPDLNADiscoverer *self)
+{
+ GUPnPDLNADiscovererPrivate *priv = GET_PRIVATE (self);
+
+ return priv->relaxed_mode;
+}
+
+/**
+ * gupnp_dlna_discoverer_get_extended_mode:
+ * @self: The #GUPnPDLNADiscoverer object
+ *
+ * Returns: true if application is using extended mode and false otherwise
+ */
+gboolean
+gupnp_dlna_discoverer_get_extended_mode (GUPnPDLNADiscoverer *self)
+{
+ GUPnPDLNADiscovererPrivate *priv = GET_PRIVATE (self);
+
+ return priv->extended_mode;
+}