[grilo-plugins] youtube: add quvi support
- From: Juan A. Suarez Romero <jasuarez src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [grilo-plugins] youtube: add quvi support
- Date: Tue, 21 Jun 2011 16:40:16 +0000 (UTC)
commit a7689ef4657e7681a7b76f57a0f03c8ba6a55a6a
Author: Juan A. Suarez Romero <jasuarez igalia com>
Date: Tue Jun 21 15:09:49 2011 +0000
youtube: add quvi support
Signed-off-by: Juan A. Suarez Romero <jasuarez igalia com>
configure.ac | 7 ++-
src/media/youtube/Makefile.am | 8 +-
src/media/youtube/grl-youtube.c | 172 ++++++++++-----------------------------
3 files changed, 56 insertions(+), 131 deletions(-)
---
diff --git a/configure.ac b/configure.ac
index b7f856e..9e1a7f7 100644
--- a/configure.ac
+++ b/configure.ac
@@ -119,6 +119,8 @@ PKG_CHECK_MODULES(SQLITE, sqlite3, HAVE_SQLITE=yes, HAVE_SQLITE=no)
PKG_CHECK_MODULES(GDATA, libgdata >= 0.4.0, HAVE_GDATA=yes, HAVE_GDATA=no)
+PKG_CHECK_MODULES(QUVI, libquvi >= 0.2.15, HAVE_QUVI=yes, HAVE_QUVI=no)
+
PKG_CHECK_MODULES(LIBSOUP, libsoup-2.4, HAVE_LIBSOUP=yes, HAVE_LIBSOUP=no)
AC_CHECK_HEADER([gcrypt.h], HAVE_GCRYPT=yes, HAVE_GCRYPT=no)
@@ -334,11 +336,14 @@ AC_ARG_ENABLE(youtube,
if test "x$HAVE_GDATA" = "xno"; then
AC_MSG_ERROR([libgdata not found, install it or use --disable-youtube])
fi
+ if test "x$HAVE_QUVI" = "xno"; then
+ AC_MSG_ERROR([libquvi not found, install it or use --disable-youtube])
+ fi
;;
esac
],
[
- if test "x$HAVE_GRLNET" = "xyes" -a "x$HAVE_XML" = "xyes" -a "x$HAVE_GDATA" = "xyes"; then
+ if test "x$HAVE_GRLNET" = "xyes" -a "x$HAVE_XML" = "xyes" -a "x$HAVE_GDATA" = "xyes" -a "x$HAVE_QUVI" = "xyes"; then
enable_youtube=yes
else
enable_youtube=no
diff --git a/src/media/youtube/Makefile.am b/src/media/youtube/Makefile.am
index 2384bb8..b173530 100644
--- a/src/media/youtube/Makefile.am
+++ b/src/media/youtube/Makefile.am
@@ -12,14 +12,16 @@ libgrlyoutube_la_CFLAGS = \
$(GRLNET_CFLAGS) \
$(XML_CFLAGS) \
$(GTHREAD_CFLAGS) \
- $(GDATA_CFLAGS)
+ $(GDATA_CFLAGS) \
+ $(QUVI_CFLAGS)
libgrlyoutube_la_LIBADD = \
$(DEPS_LIBS) \
$(GRLNET_LIBS) \
$(XML_LIBS) \
- $(GTHREAD_LIBS) \
- $(GDATA_LIBS)
+ $(GTHREAD_LIBS) \
+ $(GDATA_LIBS) \
+ $(QUVI_LIBS)
libgrlyoutube_la_LDFLAGS = \
-module \
diff --git a/src/media/youtube/grl-youtube.c b/src/media/youtube/grl-youtube.c
index 06ecd3d..ea7c7aa 100644
--- a/src/media/youtube/grl-youtube.c
+++ b/src/media/youtube/grl-youtube.c
@@ -28,6 +28,7 @@
#include <grilo.h>
#include <net/grl-net.h>
#include <gdata/gdata.h>
+#include <quvi/quvi.h>
#include <string.h>
#include "grl-youtube.h"
@@ -168,6 +169,7 @@ typedef enum {
struct _GrlYoutubeSourcePriv {
GDataService *service;
+ quvi_t quvi_handle;
GrlNetWc *wc;
};
@@ -321,6 +323,14 @@ grl_youtube_source_new (const gchar *api_key, const gchar *client_id)
"yt-service", service,
NULL));
+ /* Set up quvi */
+ if (quvi_init (&(source->priv->quvi_handle)) != QUVI_OK) {
+ source->priv->quvi_handle = NULL;
+ } else {
+ quvi_setopt (source->priv->quvi_handle, QUVIOPT_FORMAT, "mp4_360p");
+ quvi_setopt (source->priv->quvi_handle, QUVIOPT_NOVERIFY);
+ }
+
ytsrc = source;
return source;
@@ -394,6 +404,9 @@ grl_youtube_source_finalize (GObject *object)
if (self->priv->service)
g_object_unref (self->priv->service);
+ if (self->priv->quvi_handle)
+ quvi_close (&(self->priv->quvi_handle));
+
G_OBJECT_CLASS (grl_youtube_source_parent_class)->finalize (object);
}
@@ -498,126 +511,21 @@ read_url_async (const gchar *url,
}
static void
-set_media_url_async_read_cb (gchar *data, gpointer user_data)
-{
- SetMediaUrlAsyncReadCb *cb_data = (SetMediaUrlAsyncReadCb *) user_data;
- gchar *url = NULL;
- GMatchInfo *match_info = NULL;
- static GRegex *regex = NULL;
-
- if (!data) {
- goto done;
- }
-
- if (regex == NULL) {
- regex = g_regex_new (".*&fmt_url_map=([^&]+)&", G_REGEX_OPTIMIZE, 0, NULL);
- }
-
- /* Check if we find the url mapping */
- g_regex_match (regex, data, 0, &match_info);
- if (g_match_info_matches (match_info) == TRUE) {
- gchar *url_map_escaped, *url_map;
- gchar **mappings;
-
- url_map_escaped = g_match_info_fetch (match_info, 1);
- url_map = g_uri_unescape_string (url_map_escaped, NULL);
- g_free (url_map_escaped);
-
- mappings = g_strsplit (url_map, ",", 0);
- g_free (url_map);
-
- if (mappings != NULL) {
- /* TODO: We get the URL from the first format available.
- * We should provide the list of available urls or let the user
- * configure preferred formats
- */
- gchar **mapping = g_strsplit (mappings[0], "|", 2);
- url = g_strdup (mapping[1]);
- g_strfreev (mapping);
- }
- } else {
- GRL_DEBUG ("Format array not found, using token workaround");
- gchar *token_start;
- gchar *token_end;
- gchar *token;
- const gchar *video_id;
-
- token_start = g_strrstr (data, "&token=");
- if (!token_start) {
- goto done;
- }
- token_start += 7;
- token_end = strstr (token_start, "&");
- token = g_strndup (token_start, token_end - token_start);
-
- video_id = grl_media_get_id (cb_data->media);
- url = g_strdup_printf (YOUTUBE_VIDEO_URL, video_id, token);
- g_free (token);
- }
-
- done:
- if (url) {
- grl_media_set_url (cb_data->media, url);
- g_free (url);
- }
-
- cb_data->callback (cb_data->media, cb_data->user_data);
-
- g_free (cb_data);
-}
-
-static void
-set_media_url (GrlMedia *media,
- GCancellable *cancellable,
- BuildMediaFromEntryCbFunc callback,
- gpointer user_data)
-{
- const gchar *video_id;
- gchar *video_info_url;
- SetMediaUrlAsyncReadCb *set_media_url_async_read_data;
-
- /* The procedure to get the video url is:
- * 1) Read the video info URL using the video id (async operation)
- * 2) In the video info page, there should be an array of supported formats
- * and their corresponding URLs, right now we just use the first one we get.
- * (see set_media_url_async_read_cb).
- * TODO: we should be able to provide various urls or at least
- * select preferred formats via configuration
- * TODO: we should set mime-type accordingly to the format selected
- * 3) As a workaround in case no format array is found we get the video token
- * and figure out the url of the video using the video id and the token.
- */
-
- video_id = grl_media_get_id (media);
- video_info_url = g_strdup_printf (YOUTUBE_VIDEO_INFO_URL, video_id);
-
- set_media_url_async_read_data = g_new0 (SetMediaUrlAsyncReadCb, 1);
- set_media_url_async_read_data->media = media;
- set_media_url_async_read_data->cancellable = cancellable;
- set_media_url_async_read_data->callback = callback;
- set_media_url_async_read_data->user_data = user_data;
-
- read_url_async (video_info_url,
- cancellable,
- set_media_url_async_read_cb,
- set_media_url_async_read_data);
-
- g_free (video_info_url);
-}
-
-static void
-build_media_from_entry (GrlMedia *content,
- GDataEntry *entry,
+build_media_from_entry (GrlYoutubeSource *source,
+ GrlMedia *content,
+ GDataEntry *entry,
GCancellable *cancellable,
- const GList *keys,
- BuildMediaFromEntryCbFunc callback,
- gpointer user_data)
+ const GList *keys,
+ BuildMediaFromEntryCbFunc callback,
+ gpointer user_data)
{
GDataYouTubeVideo *video;
GDataMediaThumbnail *thumbnail;
GrlMedia *media;
GList *iter;
- gboolean need_url = FALSE;
+ quvi_media_t v;
+ QUVIcode rc;
+ gchar *url;
if (!content) {
media = grl_media_video_new ();
@@ -677,9 +585,17 @@ build_media_from_entry (GrlMedia *content,
gdouble average;
gdata_youtube_video_get_rating (video, NULL, NULL, NULL, &average);
grl_media_set_rating (media, average, 5.00);
- } else if (key == GRL_METADATA_KEY_URL) {
- /* This needs another query and will be resolved asynchronously p Q*/
- need_url = TRUE;
+ } else if (key == GRL_METADATA_KEY_URL && source->priv->quvi_handle) {
+ rc = quvi_parse (source->priv->quvi_handle,
+ (char *) gdata_youtube_video_get_player_uri (video),
+ &v);
+ if (rc == QUVI_OK) {
+ rc = quvi_getprop (v, QUVIPROP_MEDIAURL, &url);
+ if (rc == QUVI_OK) {
+ grl_media_set_url (media, url);
+ }
+ quvi_parse_close (&v);
+ }
} else if (key == GRL_METADATA_KEY_EXTERNAL_PLAYER) {
GDataYouTubeContent *youtube_content;
youtube_content =
@@ -694,12 +610,7 @@ build_media_from_entry (GrlMedia *content,
iter = g_list_next (iter);
}
- if (need_url) {
- /* URL resolution is async */
- set_media_url (media, cancellable, callback, user_data);
- } else {
- callback (media, user_data);
- }
+ callback (media, user_data);
}
static void
@@ -915,7 +826,8 @@ metadata_cb (GObject *object,
ms->callback (ms->source, ms->metadata_id, ms->media, ms->user_data, error);
g_error_free (error);
} else {
- build_media_from_entry (ms->media,
+ build_media_from_entry (GRL_YOUTUBE_SOURCE (ms->source),
+ ms->media,
video,
grl_metadata_source_get_operation_data (GRL_METADATA_SOURCE (ms->source),
ms->metadata_id),
@@ -950,8 +862,13 @@ search_progress_cb (GDataEntry *entry,
* we have to check if we got as many results as we requested or
* not, and handle that situation properly */
os->matches++;
- build_media_from_entry (NULL, entry, os->cancellable, os->keys,
- build_media_from_entry_search_cb, os);
+ build_media_from_entry (GRL_YOUTUBE_SOURCE (os->source),
+ NULL,
+ entry,
+ os->cancellable,
+ os->keys,
+ build_media_from_entry_search_cb,
+ os);
} else {
GRL_WARNING ("Invalid index/count received grom libgdata, ignoring result");
}
@@ -1337,7 +1254,8 @@ media_from_uri_cb (GObject *object, GAsyncResult *result, gpointer user_data)
mfus->callback (mfus->source, mfus->media_from_uri_id, NULL, mfus->user_data, error);
g_error_free (error);
} else {
- build_media_from_entry (NULL,
+ build_media_from_entry (GRL_YOUTUBE_SOURCE (mfus->source),
+ NULL,
video,
grl_metadata_source_get_operation_data (GRL_METADATA_SOURCE (mfus->source),
mfus->media_from_uri_id),
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]