2 * Copyright (C) 1999,2000 Erik Walthinsen <omega@cse.ogi.edu>
3 * 2000 Wim Taymans <wtay@chello.be>
4 * Copyright (C) 2011 Tim-Philipp Müller <tim centricular net>
6 * gsturi.c: register URI handlers
8 * This library is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Library General Public
10 * License as published by the Free Software Foundation; either
11 * version 2 of the License, or (at your option) any later version.
13 * This library is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Library General Public License for more details.
18 * You should have received a copy of the GNU Library General Public
19 * License along with this library; if not, write to the
20 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
21 * Boston, MA 02110-1301, USA.
25 * SECTION:gsturihandler
26 * @short_description: Interface to ease URI handling in plugins.
28 * The URIHandler is an interface that is implemented by Source and Sink
29 * #GstElement to simplify then handling of URI.
31 * An application can use the following functions to quickly get an element
32 * that handles the given URI for reading or writing
33 * (gst_element_make_from_uri()).
35 * Source and Sink plugins should implement this interface when possible.
37 * Last reviewed on 2005-11-09 (0.9.4)
44 #include "gst_private.h"
48 #include "gstregistry.h"
50 #include "gst-i18n-lib.h"
54 GST_DEBUG_CATEGORY_STATIC (gst_uri_handler_debug);
55 #define GST_CAT_DEFAULT gst_uri_handler_debug
58 gst_uri_handler_get_type (void)
60 static volatile gsize urihandler_type = 0;
62 if (g_once_init_enter (&urihandler_type)) {
64 static const GTypeInfo urihandler_info = {
65 sizeof (GstURIHandlerInterface),
77 _type = g_type_register_static (G_TYPE_INTERFACE,
78 "GstURIHandler", &urihandler_info, 0);
80 GST_DEBUG_CATEGORY_INIT (gst_uri_handler_debug, "GST_URI", GST_DEBUG_BOLD,
82 g_once_init_leave (&urihandler_type, _type);
84 return urihandler_type;
88 gst_uri_error_quark (void)
90 return g_quark_from_static_string ("gst-uri-error-quark");
93 static const guchar acceptable[96] = { /* X0 X1 X2 X3 X4 X5 X6 X7 X8 X9 XA XB XC XD XE XF */
94 0x00, 0x3F, 0x20, 0x20, 0x20, 0x00, 0x2C, 0x3F, 0x3F, 0x3F, 0x3F, 0x22, 0x20, 0x3F, 0x3F, 0x1C, /* 2X !"#$%&'()*+,-./ */
95 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x38, 0x20, 0x20, 0x2C, 0x20, 0x2C, /* 3X 0123456789:;<=>? */
96 0x30, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, /* 4X @ABCDEFGHIJKLMNO */
97 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x20, 0x20, 0x20, 0x20, 0x3F, /* 5X PQRSTUVWXYZ[\]^_ */
98 0x20, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, /* 6X `abcdefghijklmno */
99 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x3F, 0x20, 0x20, 0x20, 0x3F, 0x20 /* 7X pqrstuvwxyz{|}~DEL */
104 UNSAFE_ALL = 0x1, /* Escape all unsafe characters */
105 UNSAFE_ALLOW_PLUS = 0x2, /* Allows '+' */
106 UNSAFE_PATH = 0x4, /* Allows '/' and '?' and '&' and '=' */
107 UNSAFE_DOS_PATH = 0x8, /* Allows '/' and '?' and '&' and '=' and ':' */
108 UNSAFE_HOST = 0x10, /* Allows '/' and ':' and '@' */
109 UNSAFE_SLASHES = 0x20 /* Allows all characters except for '/' and '%' */
110 } UnsafeCharacterSet;
112 #define HEX_ESCAPE '%'
114 /* Escape undesirable characters using %
115 * -------------------------------------
117 * This function takes a pointer to a string in which
118 * some characters may be unacceptable unescaped.
119 * It returns a string which has these characters
120 * represented by a '%' character followed by two hex digits.
122 * This routine returns a g_malloced string.
125 static const gchar hex[16] = "0123456789ABCDEF";
128 escape_string_internal (const gchar * string, UnsafeCharacterSet mask)
130 #define ACCEPTABLE_CHAR(a) ((a)>=32 && (a)<128 && (acceptable[(a)-32] & use_mask))
137 UnsafeCharacterSet use_mask;
139 g_return_val_if_fail (mask == UNSAFE_ALL
140 || mask == UNSAFE_ALLOW_PLUS
141 || mask == UNSAFE_PATH
142 || mask == UNSAFE_DOS_PATH
143 || mask == UNSAFE_HOST || mask == UNSAFE_SLASHES, NULL);
145 if (string == NULL) {
151 for (p = string; *p != '\0'; p++) {
153 if (!ACCEPTABLE_CHAR (c)) {
156 if ((use_mask == UNSAFE_HOST) && (unacceptable || (c == '/'))) {
157 /* when escaping a host, if we hit something that needs to be escaped, or we finally
158 * hit a path separator, revert to path mode (the host segment of the url is over).
160 use_mask = UNSAFE_PATH;
164 result = g_malloc (p - string + unacceptable * 2 + 1);
167 for (q = result, p = string; *p != '\0'; p++) {
170 if (!ACCEPTABLE_CHAR (c)) {
171 *q++ = HEX_ESCAPE; /* means hex coming */
177 if ((use_mask == UNSAFE_HOST) && (!ACCEPTABLE_CHAR (c) || (c == '/'))) {
178 use_mask = UNSAFE_PATH;
188 * @string: string to be escaped
190 * Escapes @string, replacing any and all special characters
191 * with equivalent escape sequences.
193 * Return value: a newly allocated string equivalent to @string
194 * but with all special characters escaped
197 escape_string (const gchar * string)
199 return escape_string_internal (string, UNSAFE_ALL);
205 return c >= '0' && c <= '9' ? c - '0'
206 : c >= 'A' && c <= 'F' ? c - 'A' + 10
207 : c >= 'a' && c <= 'f' ? c - 'a' + 10 : -1;
211 unescape_character (const char *scanner)
216 first_digit = hex_to_int (*scanner++);
217 if (first_digit < 0) {
221 second_digit = hex_to_int (*scanner);
222 if (second_digit < 0) {
226 return (first_digit << 4) | second_digit;
230 * @escaped_string: an escaped URI, path, or other string
231 * @illegal_characters: a string containing a sequence of characters
232 * considered "illegal", '\0' is automatically in this list.
234 * Decodes escaped characters (i.e. PERCENTxx sequences) in @escaped_string.
235 * Characters are encoded in PERCENTxy form, where xy is the ASCII hex code
236 * for character 16x+y.
238 * Return value: a newly allocated string with the unescaped equivalents,
239 * or %NULL if @escaped_string contained one of the characters
240 * in @illegal_characters.
243 unescape_string (const gchar * escaped_string, const gchar * illegal_characters)
249 if (escaped_string == NULL) {
253 result = g_malloc (strlen (escaped_string) + 1);
256 for (in = escaped_string; *in != '\0'; in++) {
258 if (*in == HEX_ESCAPE) {
259 character = unescape_character (in + 1);
261 /* Check for an illegal character. We consider '\0' illegal here. */
263 || (illegal_characters != NULL
264 && strchr (illegal_characters, (char) character) != NULL)) {
270 *out++ = (char) character;
274 g_assert ((gsize) (out - result) <= strlen (escaped_string));
281 gst_uri_protocol_check_internal (const gchar * uri, gchar ** endptr)
283 gchar *check = (gchar *) uri;
285 g_assert (uri != NULL);
286 g_assert (endptr != NULL);
288 if (g_ascii_isalpha (*check)) {
290 while (g_ascii_isalnum (*check) || *check == '+'
291 || *check == '-' || *check == '.')
299 * gst_uri_protocol_is_valid:
300 * @protocol: A string
302 * Tests if the given string is a valid protocol identifier. Protocols
303 * must consist of alphanumeric characters, '+', '-' and '.' and must
304 * start with a alphabetic character. See RFC 3986 Section 3.1.
306 * Returns: TRUE if the string is a valid protocol identifier, FALSE otherwise.
309 gst_uri_protocol_is_valid (const gchar * protocol)
313 g_return_val_if_fail (protocol != NULL, FALSE);
315 gst_uri_protocol_check_internal (protocol, &endptr);
317 return *endptr == '\0' && ((gsize) (endptr - protocol)) >= 2;
324 * Tests if the given string is a valid URI identifier. URIs start with a valid
325 * scheme followed by ":" and maybe a string identifying the location.
327 * Returns: TRUE if the string is a valid URI
330 gst_uri_is_valid (const gchar * uri)
334 g_return_val_if_fail (uri != NULL, FALSE);
336 gst_uri_protocol_check_internal (uri, &endptr);
338 return *endptr == ':' && ((gsize) (endptr - uri)) >= 2;
342 * gst_uri_get_protocol:
345 * Extracts the protocol out of a given valid URI. The returned string must be
346 * freed using g_free().
348 * Returns: The protocol for this URI.
351 gst_uri_get_protocol (const gchar * uri)
355 g_return_val_if_fail (uri != NULL, NULL);
356 g_return_val_if_fail (gst_uri_is_valid (uri), NULL);
358 colon = strstr (uri, ":");
360 return g_ascii_strdown (uri, colon - uri);
364 * gst_uri_has_protocol:
366 * @protocol: a protocol string (e.g. "http")
368 * Checks if the protocol of a given valid URI matches @protocol.
370 * Returns: %TRUE if the protocol matches.
373 gst_uri_has_protocol (const gchar * uri, const gchar * protocol)
377 g_return_val_if_fail (uri != NULL, FALSE);
378 g_return_val_if_fail (protocol != NULL, FALSE);
379 g_return_val_if_fail (gst_uri_is_valid (uri), FALSE);
381 colon = strstr (uri, ":");
386 return (g_ascii_strncasecmp (uri, protocol, (gsize) (colon - uri)) == 0);
390 * gst_uri_get_location:
393 * Extracts the location out of a given valid URI, ie. the protocol and "://"
394 * are stripped from the URI, which means that the location returned includes
395 * the hostname if one is specified. The returned string must be freed using
398 * Free-function: g_free
400 * Returns: (transfer full): the location for this URI. Returns NULL if the
401 * URI isn't valid. If the URI does not contain a location, an empty
402 * string is returned.
405 gst_uri_get_location (const gchar * uri)
408 gchar *unescaped = NULL;
410 g_return_val_if_fail (uri != NULL, NULL);
411 g_return_val_if_fail (gst_uri_is_valid (uri), NULL);
413 colon = strstr (uri, "://");
417 unescaped = unescape_string (colon + 3, "/");
419 /* On Windows an URI might look like file:///c:/foo/bar.txt or
420 * file:///c|/foo/bar.txt (some Netscape versions) and we want to
421 * return c:/foo/bar.txt as location rather than /c:/foo/bar.txt.
422 * Can't use g_filename_from_uri() here because it will only handle the
423 * file:// protocol */
425 if (unescaped != NULL && unescaped[0] == '/' &&
426 g_ascii_isalpha (unescaped[1]) &&
427 (unescaped[2] == ':' || unescaped[2] == '|')) {
429 g_memmove (unescaped, unescaped + 1, strlen (unescaped + 1) + 1);
433 GST_LOG ("extracted location '%s' from URI '%s'", GST_STR_NULL (unescaped),
440 * @protocol: Protocol for URI
441 * @location: (transfer none): Location for URI
443 * Constructs a URI for a given valid protocol and location.
445 * Free-function: g_free
447 * Returns: (transfer full): a new string for this URI. Returns NULL if the
448 * given URI protocol is not valid, or the given location is NULL.
451 gst_uri_construct (const gchar * protocol, const gchar * location)
453 char *escaped, *proto_lowercase;
456 g_return_val_if_fail (gst_uri_protocol_is_valid (protocol), NULL);
457 g_return_val_if_fail (location != NULL, NULL);
459 proto_lowercase = g_ascii_strdown (protocol, -1);
460 escaped = escape_string (location);
461 retval = g_strdup_printf ("%s://%s", proto_lowercase, escaped);
463 g_free (proto_lowercase);
471 const gchar *protocol;
476 search_by_entry (GstPluginFeature * feature, gpointer search_entry)
478 const gchar *const *protocols;
479 GstElementFactory *factory;
480 SearchEntry *entry = (SearchEntry *) search_entry;
482 if (!GST_IS_ELEMENT_FACTORY (feature))
484 factory = GST_ELEMENT_FACTORY_CAST (feature);
486 if (factory->uri_type != entry->type)
489 protocols = gst_element_factory_get_uri_protocols (factory);
491 if (protocols == NULL) {
492 g_warning ("Factory '%s' implements GstUriHandler interface but returned "
493 "no supported protocols!", gst_plugin_feature_get_name (feature));
497 while (*protocols != NULL) {
498 if (g_ascii_strcasecmp (*protocols, entry->protocol) == 0)
506 sort_by_rank (GstPluginFeature * first, GstPluginFeature * second)
508 return gst_plugin_feature_get_rank (second) -
509 gst_plugin_feature_get_rank (first);
513 get_element_factories_from_uri_protocol (const GstURIType type,
514 const gchar * protocol)
516 GList *possibilities;
519 g_return_val_if_fail (protocol, NULL);
522 entry.protocol = protocol;
523 possibilities = gst_registry_feature_filter (gst_registry_get (),
524 search_by_entry, FALSE, &entry);
526 return possibilities;
530 * gst_uri_protocol_is_supported:
531 * @type: Whether to check for a source or a sink
532 * @protocol: Protocol that should be checked for (e.g. "http" or "smb")
534 * Checks if an element exists that supports the given URI protocol. Note
535 * that a positive return value does not imply that a subsequent call to
536 * gst_element_make_from_uri() is guaranteed to work.
541 gst_uri_protocol_is_supported (const GstURIType type, const gchar * protocol)
543 GList *possibilities;
545 g_return_val_if_fail (protocol, FALSE);
547 possibilities = get_element_factories_from_uri_protocol (type, protocol);
550 g_list_free (possibilities);
557 * gst_element_make_from_uri:
558 * @type: Whether to create a source or a sink
559 * @uri: URI to create an element for
560 * @elementname: (allow-none): Name of created element, can be NULL.
561 * @error: (allow-none): address where to store error information, or NULL.
563 * Creates an element for handling the given URI.
565 * Returns: (transfer floating): a new element or NULL if none could be created
568 gst_element_make_from_uri (const GstURIType type, const gchar * uri,
569 const gchar * elementname, GError ** error)
571 GList *possibilities, *walk;
573 GstElement *ret = NULL;
575 g_return_val_if_fail (gst_is_initialized (), NULL);
576 g_return_val_if_fail (GST_URI_TYPE_IS_VALID (type), NULL);
577 g_return_val_if_fail (gst_uri_is_valid (uri), NULL);
578 g_return_val_if_fail (error == NULL || *error == NULL, NULL);
580 GST_DEBUG ("type:%d, uri:%s, elementname:%s", type, uri, elementname);
582 protocol = gst_uri_get_protocol (uri);
583 possibilities = get_element_factories_from_uri_protocol (type, protocol);
585 if (!possibilities) {
586 GST_DEBUG ("No %s for URI '%s'", type == GST_URI_SINK ? "sink" : "source",
588 /* The error message isn't great, but we don't expect applications to
589 * show that error to users, but call the missing plugins functions */
590 g_set_error (error, GST_URI_ERROR, GST_URI_ERROR_UNSUPPORTED_PROTOCOL,
591 _("No URI handler for the %s protocol found"), protocol);
597 possibilities = g_list_sort (possibilities, (GCompareFunc) sort_by_rank);
598 walk = possibilities;
600 GstElementFactory *factory = walk->data;
601 GError *uri_err = NULL;
603 ret = gst_element_factory_create (factory, elementname);
605 GstURIHandler *handler = GST_URI_HANDLER (ret);
607 if (gst_uri_handler_set_uri (handler, uri, &uri_err))
610 GST_WARNING ("%s didn't accept URI '%s': %s", GST_OBJECT_NAME (ret), uri,
613 if (error != NULL && *error == NULL)
614 g_propagate_error (error, uri_err);
616 g_error_free (uri_err);
618 gst_object_unref (ret);
623 gst_plugin_feature_list_free (possibilities);
625 GST_LOG_OBJECT (ret, "created %s for URL '%s'",
626 type == GST_URI_SINK ? "sink" : "source", uri);
628 /* if the first handler didn't work, but we found another one that works */
630 g_clear_error (error);
636 * gst_uri_handler_get_uri_type:
637 * @handler: A #GstURIHandler.
639 * Gets the type of the given URI handler
641 * Returns: the #GstURIType of the URI handler.
642 * Returns #GST_URI_UNKNOWN if the @handler isn't implemented correctly.
645 gst_uri_handler_get_uri_type (GstURIHandler * handler)
647 GstURIHandlerInterface *iface;
650 g_return_val_if_fail (GST_IS_URI_HANDLER (handler), GST_URI_UNKNOWN);
652 iface = GST_URI_HANDLER_GET_INTERFACE (handler);
653 g_return_val_if_fail (iface != NULL, GST_URI_UNKNOWN);
654 g_return_val_if_fail (iface->get_type != NULL, GST_URI_UNKNOWN);
656 ret = iface->get_type (G_OBJECT_TYPE (handler));
657 g_return_val_if_fail (GST_URI_TYPE_IS_VALID (ret), GST_URI_UNKNOWN);
663 * gst_uri_handler_get_protocols:
664 * @handler: A #GstURIHandler.
666 * Gets the list of protocols supported by @handler. This list may not be
669 * Returns: (transfer none) (element-type utf8): the supported protocols.
670 * Returns NULL if the @handler isn't implemented properly, or the @handler
671 * doesn't support any protocols.
674 gst_uri_handler_get_protocols (GstURIHandler * handler)
676 GstURIHandlerInterface *iface;
677 const gchar *const *ret;
679 g_return_val_if_fail (GST_IS_URI_HANDLER (handler), NULL);
681 iface = GST_URI_HANDLER_GET_INTERFACE (handler);
682 g_return_val_if_fail (iface != NULL, NULL);
683 g_return_val_if_fail (iface->get_protocols != NULL, NULL);
685 ret = iface->get_protocols (G_OBJECT_TYPE (handler));
686 g_return_val_if_fail (ret != NULL, NULL);
692 * gst_uri_handler_get_uri:
693 * @handler: A #GstURIHandler
695 * Gets the currently handled URI.
697 * Returns: (transfer full): the URI currently handled by the @handler.
698 * Returns NULL if there are no URI currently handled. The
699 * returned string must be freed with g_free() when no longer needed.
702 gst_uri_handler_get_uri (GstURIHandler * handler)
704 GstURIHandlerInterface *iface;
707 g_return_val_if_fail (GST_IS_URI_HANDLER (handler), NULL);
709 iface = GST_URI_HANDLER_GET_INTERFACE (handler);
710 g_return_val_if_fail (iface != NULL, NULL);
711 g_return_val_if_fail (iface->get_uri != NULL, NULL);
712 ret = iface->get_uri (handler);
714 g_return_val_if_fail (gst_uri_is_valid (ret), NULL);
720 * gst_uri_handler_set_uri:
721 * @handler: A #GstURIHandler
723 * @error: (allow-none): address where to store a #GError in case of
726 * Tries to set the URI of the given handler.
728 * Returns: TRUE if the URI was set successfully, else FALSE.
731 gst_uri_handler_set_uri (GstURIHandler * handler, const gchar * uri,
734 GstURIHandlerInterface *iface;
736 gchar *new_uri, *protocol, *location, *colon;
738 g_return_val_if_fail (GST_IS_URI_HANDLER (handler), FALSE);
739 g_return_val_if_fail (gst_uri_is_valid (uri), FALSE);
740 g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
742 iface = GST_URI_HANDLER_GET_INTERFACE (handler);
743 g_return_val_if_fail (iface != NULL, FALSE);
744 g_return_val_if_fail (iface->set_uri != NULL, FALSE);
746 protocol = gst_uri_get_protocol (uri);
748 if (iface->get_protocols) {
749 const gchar *const *protocols;
750 const gchar *const *p;
751 gboolean found_protocol = FALSE;
753 protocols = iface->get_protocols (G_OBJECT_TYPE (handler));
754 if (protocols != NULL) {
755 for (p = protocols; *p != NULL; ++p) {
756 if (g_ascii_strcasecmp (protocol, *p) == 0) {
757 found_protocol = TRUE;
762 if (!found_protocol) {
763 g_set_error (error, GST_URI_ERROR, GST_URI_ERROR_UNSUPPORTED_PROTOCOL,
764 _("URI scheme '%s' not supported"), protocol);
771 colon = strstr (uri, ":");
772 location = g_strdup (colon);
774 new_uri = g_strdup_printf ("%s%s", protocol, location);
776 ret = iface->set_uri (handler, uri, error);
786 gst_file_utils_canonicalise_path (const gchar * path)
788 gchar **parts, **p, *clean_path;
792 GST_WARNING ("FIXME: canonicalise win32 path");
793 return g_strdup (path);
797 parts = g_strsplit (path, "/", -1);
801 if (strcmp (*p, ".") == 0) {
802 /* just move all following parts on top of this, incl. NUL terminator */
804 g_memmove (p, p + 1, (g_strv_length (p + 1) + 1) * sizeof (gchar *));
805 /* re-check the new current part again in the next iteration */
807 } else if (strcmp (*p, "..") == 0 && p > parts) {
808 /* just move all following parts on top of the previous part, incl.
812 g_memmove (p - 1, p + 1, (g_strv_length (p + 1) + 1) * sizeof (gchar *));
813 /* re-check the new current part again in the next iteration */
822 num_parts = g_strv_length (parts) + 1; /* incl. terminator */
823 parts = g_renew (gchar *, parts, num_parts + 1);
824 g_memmove (parts + 1, parts, num_parts * sizeof (gchar *));
825 parts[0] = g_strdup ("/");
828 clean_path = g_build_filenamev (parts);
834 file_path_contains_relatives (const gchar * path)
836 return (strstr (path, "/./") != NULL || strstr (path, "/../") != NULL ||
837 strstr (path, G_DIR_SEPARATOR_S "." G_DIR_SEPARATOR_S) != NULL ||
838 strstr (path, G_DIR_SEPARATOR_S ".." G_DIR_SEPARATOR_S) != NULL);
842 * gst_filename_to_uri:
843 * @filename: absolute or relative file name path
844 * @error: pointer to error, or NULL
846 * Similar to g_filename_to_uri(), but attempts to handle relative file paths
847 * as well. Before converting @filename into an URI, it will be prefixed by
848 * the current working directory if it is a relative path, and then the path
849 * will be canonicalised so that it doesn't contain any './' or '../' segments.
851 * On Windows #filename should be in UTF-8 encoding.
854 gst_filename_to_uri (const gchar * filename, GError ** error)
856 gchar *abs_location = NULL;
857 gchar *uri, *abs_clean;
859 g_return_val_if_fail (filename != NULL, NULL);
860 g_return_val_if_fail (error == NULL || *error == NULL, NULL);
862 if (g_path_is_absolute (filename)) {
863 if (!file_path_contains_relatives (filename)) {
864 uri = g_filename_to_uri (filename, NULL, error);
868 abs_location = g_strdup (filename);
872 cwd = g_get_current_dir ();
873 abs_location = g_build_filename (cwd, filename, NULL);
876 if (!file_path_contains_relatives (abs_location)) {
877 uri = g_filename_to_uri (abs_location, NULL, error);
882 /* path is now absolute, but contains '.' or '..' */
883 abs_clean = gst_file_utils_canonicalise_path (abs_location);
884 GST_LOG ("'%s' -> '%s' -> '%s'", filename, abs_location, abs_clean);
885 uri = g_filename_to_uri (abs_clean, NULL, error);
890 g_free (abs_location);
891 GST_DEBUG ("'%s' -> '%s'", filename, uri);