Commit c39b64d5 authored by Yoann Peronneau's avatar Yoann Peronneau

* demux/playlist/itml.c: iTunes Media Library importer (not tested yet)

parent 930bf06c
...@@ -14,4 +14,6 @@ SOURCES_playlist = \ ...@@ -14,4 +14,6 @@ SOURCES_playlist = \
qtl.c \ qtl.c \
gvp.c \ gvp.c \
ifo.c \ ifo.c \
itml.c \
itml.h \
$(NULL) $(NULL)
/*******************************************************************************
* itml.c : iTunes Music Library import functions
*******************************************************************************
* Copyright (C) 2007 the VideoLAN team
* $Id: $
*
* Authors: Yoann Peronneau <yoann@videolan.org>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
*******************************************************************************/
/**
* \file modules/demux/playlist/itml.c
* \brief iTunes Music Library import functions
*/
#include <vlc/vlc.h>
#include <vlc_demux.h>
#include "playlist.h"
#include "vlc_xml.h"
#include "vlc_strings.h"
#include "vlc_url.h"
#include "itml.h"
struct demux_sys_t
{
int i_ntracks;
};
static int Control( demux_t *, int, va_list );
static int Demux( demux_t * );
/**
* \brief iTML submodule initialization function
*/
int E_(Import_iTML)( vlc_object_t *p_this )
{
DEMUX_BY_EXTENSION_OR_FORCED_MSG( ".xml", "itml",
"using iTunes Media Library reader" );
return VLC_SUCCESS;
}
void E_(Close_iTML)( vlc_object_t *p_this )
{
demux_t *p_demux = (demux_t *)p_this;
free( p_demux->p_sys );
}
/**
* \brief demuxer function for iTML parsing
*/
int Demux( demux_t *p_demux )
{
int i_ret = VLC_SUCCESS;
xml_t *p_xml = NULL;
xml_reader_t *p_xml_reader = NULL;
char *psz_name = NULL;
INIT_PLAYLIST_STUFF;
p_demux->p_sys->i_ntracks = 0;
/* create new xml parser from stream */
p_xml = xml_Create( p_demux );
if( !p_xml )
i_ret = VLC_ENOMOD;
else
{
p_xml_reader = xml_ReaderCreate( p_xml, p_demux->s );
if( !p_xml_reader )
i_ret = VLC_EGENERIC;
}
/* locating the root node */
if( i_ret == VLC_SUCCESS )
{
do
{
if( xml_ReaderRead( p_xml_reader ) != 1 )
{
msg_Err( p_demux, "can't read xml stream" );
i_ret = VLC_EGENERIC;
}
} while( i_ret == VLC_SUCCESS &&
xml_ReaderNodeType( p_xml_reader ) != XML_READER_STARTELEM );
}
/* checking root node name */
if( i_ret == VLC_SUCCESS )
{
psz_name = xml_ReaderName( p_xml_reader );
if( !psz_name || strcmp( psz_name, "plist" ) )
{
msg_Err( p_demux, "invalid root node name: %s", psz_name );
i_ret = VLC_EGENERIC;
}
FREE_NAME();
}
if( i_ret == VLC_SUCCESS )
{
xml_elem_hnd_t pl_elements[] =
{ {"dict", COMPLEX_CONTENT, {.cmplx = parse_plist_dict} } };
i_ret = parse_plist_node( p_demux, p_playlist, p_current_input,
NULL, p_xml_reader, "plist",
pl_elements );
HANDLE_PLAY_AND_RELEASE;
}
if( p_xml_reader )
xml_ReaderDelete( p_xml, p_xml_reader );
if( p_xml )
xml_Delete( p_xml );
return -1; /* Needed for correct operation of go back */
}
/** \brief dummy function for demux callback interface */
static int Control( demux_t *p_demux, int i_query, va_list args )
{
return VLC_EGENERIC;
}
/**
* \brief parse the root node of the playlist
*/
static vlc_bool_t parse_plist_node COMPLEX_INTERFACE
{
char *psz_name = NULL;
char *psz_value = NULL;
vlc_bool_t b_version_found = VLC_FALSE;
/* read all playlist attributes */
while( xml_ReaderNextAttr( p_xml_reader ) == VLC_SUCCESS )
{
psz_name = xml_ReaderName( p_xml_reader );
psz_value = xml_ReaderValue( p_xml_reader );
if( !psz_name || !psz_value )
{
msg_Err( p_demux, "invalid xml stream @ <plist>" );
FREE_ATT();
return VLC_FALSE;
}
/* attribute: version */
if( !strcmp( psz_name, "version" ) )
{
b_version_found = VLC_TRUE;
if( strcmp( psz_value, "1.0" ) )
msg_Warn( p_demux, "unsupported iTunes Media Library version" );
}
/* unknown attribute */
else
msg_Warn( p_demux, "invalid <plist> attribute:\"%s\"", psz_name);
FREE_ATT();
}
/* attribute version is mandatory !!! */
if( !b_version_found )
msg_Warn( p_demux, "<plist> requires \"version\" attribute" );
return parse_dict( p_demux, p_playlist, p_input_item, NULL, p_xml_reader,
"plist", p_handlers );
}
/**
* \brief parse a <dict>
* \param COMPLEX_INTERFACE
*/
static vlc_bool_t parse_dict COMPLEX_INTERFACE
{
int i_node;
char *psz_name = NULL;
char *psz_value = NULL;
char *psz_key = NULL;
xml_elem_hnd_t *p_handler = NULL;
while( xml_ReaderRead( p_xml_reader ) == 1 )
{
i_node = xml_ReaderNodeType( p_xml_reader );
switch( i_node )
{
case XML_READER_NONE:
break;
case XML_READER_STARTELEM:
/* element start tag */
psz_name = xml_ReaderName( p_xml_reader );
if( !psz_name || !*psz_name )
{
msg_Err( p_demux, "invalid xml stream" );
FREE_ATT_KEY();
return VLC_FALSE;
}
/* choose handler */
for( p_handler = p_handlers;
p_handler->name && strcmp( psz_name, p_handler->name );
p_handler++ );
if( !p_handler->name )
{
msg_Err( p_demux, "unexpected element <%s>", psz_name );
FREE_ATT_KEY();
return VLC_FALSE;
}
FREE_NAME();
/* complex content is parsed in a separate function */
if( p_handler->type == COMPLEX_CONTENT )
{
if( p_handler->pf_handler.cmplx( p_demux,
p_playlist,
p_input_item,
NULL,
p_xml_reader,
p_handler->name,
NULL ) )
{
p_handler = NULL;
FREE_ATT_KEY();
}
else
{
FREE_ATT_KEY();
return VLC_FALSE;
}
}
break;
case XML_READER_TEXT:
/* simple element content */
FREE_ATT();
psz_value = xml_ReaderValue( p_xml_reader );
if( !psz_value )
{
msg_Err( p_demux, "invalid xml stream" );
FREE_ATT_KEY();
return VLC_FALSE;
}
break;
case XML_READER_ENDELEM:
/* element end tag */
psz_name = xml_ReaderName( p_xml_reader );
if( !psz_name )
{
msg_Err( p_demux, "invalid xml stream" );
FREE_ATT_KEY();
return VLC_FALSE;
}
/* leave if the current parent node <track> is terminated */
if( !strcmp( psz_name, psz_element ) )
{
FREE_ATT_KEY();
return VLC_TRUE;
}
/* there MUST have been a start tag for that element name */
if( !p_handler || !p_handler->name
|| strcmp( p_handler->name, psz_name ))
{
msg_Err( p_demux, "there's no open element left for <%s>",
psz_name );
FREE_ATT_KEY();
return VLC_FALSE;
}
/* special case: key */
if( !strcmp( p_handler->name, "key" ) )
{
psz_key = strdup( psz_value );
}
/* call the simple handler */
else if( p_handler->pf_handler.smpl )
{
p_handler->pf_handler.smpl( p_track, psz_key, psz_value );
}
FREE_ATT();
p_handler = NULL;
break;
default:
/* unknown/unexpected xml node */
msg_Err( p_demux, "unexpected xml node %i", i_node );
FREE_ATT_KEY();
return VLC_FALSE;
}
FREE_NAME();
}
msg_Err( p_demux, "unexpected end of xml data" );
FREE_ATT_KEY();
return VLC_FALSE;
}
static vlc_bool_t parse_plist_dict COMPLEX_INTERFACE
{
xml_elem_hnd_t pl_elements[] =
{ {"dict", COMPLEX_CONTENT, {.cmplx = parse_tracks_dict} },
{"array", SIMPLE_CONTENT, {NULL} },
{"key", SIMPLE_CONTENT, {NULL} },
{"integer", SIMPLE_CONTENT, {NULL} },
{"string", SIMPLE_CONTENT, {NULL} },
{"date", SIMPLE_CONTENT, {NULL} },
{"true", SIMPLE_CONTENT, {NULL} },
{"false", SIMPLE_CONTENT, {NULL} },
{NULL, UNKNOWN_CONTENT, {NULL} }
};
return parse_dict( p_demux, p_playlist, p_input_item, NULL, p_xml_reader,
"dict", pl_elements );
}
static vlc_bool_t parse_tracks_dict COMPLEX_INTERFACE
{
xml_elem_hnd_t tracks_elements[] =
{ {"dict", COMPLEX_CONTENT, {.cmplx = parse_track_dict} },
{"key", SIMPLE_CONTENT, {NULL} },
{NULL, UNKNOWN_CONTENT, {NULL} }
};
parse_dict( p_demux, p_playlist, p_input_item, NULL, p_xml_reader,
"dict", tracks_elements );
msg_Info( p_demux, "added %i tracks successfully",
p_demux->p_sys->i_ntracks );
return VLC_TRUE;
}
static vlc_bool_t parse_track_dict COMPLEX_INTERFACE
{
input_item_t *p_new_input = NULL;
int i_ret = -1;
char *psz_uri = NULL;
p_track = new_track();
xml_elem_hnd_t track_elements[] =
{ {"array", COMPLEX_CONTENT, {.cmplx = skip_element} },
{"key", SIMPLE_CONTENT, {.smpl = save_data} },
{"integer", SIMPLE_CONTENT, {.smpl = save_data} },
{"string", SIMPLE_CONTENT, {.smpl = save_data} },
{"date", SIMPLE_CONTENT, {.smpl = save_data} },
{"true", SIMPLE_CONTENT, {NULL} },
{"false", SIMPLE_CONTENT, {NULL} },
{NULL, UNKNOWN_CONTENT, {NULL} }
};
i_ret = parse_dict( p_demux, p_playlist, p_input_item, p_track,
p_xml_reader, "dict", track_elements );
msg_Dbg( p_demux, "name: %s, artist: %s, album: %s, genre: %s, trackNum: %s, location: %s",
p_track->name, p_track->artist, p_track->album, p_track->genre, p_track->trackNum, p_track->location );
if( !p_track->location )
{
msg_Err( p_demux, "Track needs Location" );
free_track( p_track );
return VLC_FALSE;
}
psz_uri = decode_URI_duplicate( p_track->location );
if( psz_uri )
{
if( strlen( psz_uri ) > 17 &&
!strncmp( psz_uri, "file://localhost/", 17 ) )
{
/* remove 'localhost/' */
strcpy( psz_uri + 7, psz_uri + 17 );
msg_Info( p_demux, "Adding '%s'", psz_uri );
p_new_input = input_ItemNewExt( p_playlist, psz_uri,
NULL, 0, NULL, -1 );
input_ItemAddSubItem( p_input_item, p_new_input );
/* add meta info */
add_meta( p_new_input, p_track );
p_demux->p_sys->i_ntracks++;
}
else
{
msg_Err( p_demux, "Don't know how to handle %s", psz_uri );
}
free( psz_uri );
}
free_track( p_track );
return i_ret;
}
static track_elem_t *new_track()
{
track_elem_t *p_track = NULL;
p_track = (track_elem_t *)malloc( sizeof( track_elem_t ) );
if( p_track )
{
p_track->name = NULL;
p_track->artist = NULL;
p_track->album = NULL;
p_track->genre = NULL;
p_track->trackNum = NULL;
p_track->location = NULL;
p_track->duration = 0;
}
return p_track;
}
static void free_track( track_elem_t *p_track )
{
fprintf( stderr, "free track\n" );
if ( !p_track )
return;
FREE( p_track->name )
FREE( p_track->artist )
FREE( p_track->album )
FREE( p_track->genre )
FREE( p_track->trackNum )
FREE( p_track->location )
p_track->duration = 0;
free( p_track );
}
static vlc_bool_t save_data SIMPLE_INTERFACE
{
/* exit if setting is impossible */
if( !psz_name || !psz_value || !p_track )
return VLC_FALSE;
/* re-convert xml special characters inside psz_value */
resolve_xml_special_chars( psz_value );
#define SAVE_INFO( name, value ) \
if( !strcmp( psz_name, name ) ) { p_track->value = strdup( psz_value ); }
SAVE_INFO( "Name", name )
else SAVE_INFO( "Artist", artist )
else SAVE_INFO( "Album", album )
else SAVE_INFO( "Genre", genre )
else SAVE_INFO( "Track Number", trackNum )
else SAVE_INFO( "Location", location )
else if( !strcmp( psz_name, "Total Time" ) )
{
long i_num = atol( psz_value );
p_track->duration = (mtime_t) i_num*1000;
}
return VLC_TRUE;
}
/**
* \brief handles the supported <track> sub-elements
*/
static vlc_bool_t add_meta( input_item_t *p_input_item,
track_elem_t *p_track )
{
/* exit if setting is impossible */
if( !p_input_item || !p_track )
return VLC_FALSE;
#define SET_INFO( func, prop ) \
if( p_track->prop ) { func( p_input_item, p_track->prop ); }
SET_INFO( input_item_SetTitle, name )
SET_INFO( input_item_SetArtist, artist )
SET_INFO( input_item_SetAlbum, album )
SET_INFO( input_item_SetGenre, genre )
SET_INFO( input_item_SetTrackNum, trackNum )
SET_INFO( input_item_SetDuration, duration )
return VLC_TRUE;
}
/**
* \brief skips complex element content that we can't manage
*/
static vlc_bool_t skip_element COMPLEX_INTERFACE
{
char *psz_endname;
while( xml_ReaderRead( p_xml_reader ) == 1 )
{
if( xml_ReaderNodeType( p_xml_reader ) == XML_READER_ENDELEM )
{
psz_endname = xml_ReaderName( p_xml_reader );
if( !psz_endname )
return VLC_FALSE;
if( !strcmp( psz_element, psz_endname ) )
{
free( psz_endname );
return VLC_TRUE;
}
else
free( psz_endname );
}
}
return VLC_FALSE;
}
/*******************************************************************************
* itml.c : iTunes Music Library import functions
*******************************************************************************
* Copyright (C) 2007 the VideoLAN team
* $Id: $
*
* Authors: Yoann Peronneau <yoann@videolan.org>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
*******************************************************************************/
/**
* \file modules/demux/playlist/itml.h
* \brief iTunes Music Library import: prototypes, datatypes, defines
*/
/* defines */
#define FREE(v) if (v) {free(v);v=NULL;}
#define FREE_NAME() if (psz_name) {free(psz_name);psz_name=NULL;}
#define FREE_VALUE() if (psz_value) {free(psz_value);psz_value=NULL;}
#define FREE_KEY() if (psz_key) {free(psz_key);psz_key=NULL;}
#define FREE_ATT() FREE_NAME();FREE_VALUE()
#define FREE_ATT_KEY() FREE_NAME();FREE_VALUE();FREE_KEY()
#define UNKNOWN_CONTENT 0
#define SIMPLE_CONTENT 1
#define COMPLEX_CONTENT 2
#define SIMPLE_INTERFACE (track_elem_t *p_track,\
const char *psz_name,\
char *psz_value)
#define COMPLEX_INTERFACE (demux_t *p_demux,\
playlist_t *p_playlist,\
input_item_t *p_input_item,\
track_elem_t *p_track,\
xml_reader_t *p_xml_reader,\
const char *psz_element,\
struct xml_elem_hnd *p_handlers)
/* datatypes */
typedef struct
{
char *name, *artist, *album, *genre, *trackNum, *location;
mtime_t duration;
} track_elem_t;
struct xml_elem_hnd
{
const char *name;
int type;
union
{
vlc_bool_t (*smpl) SIMPLE_INTERFACE;
vlc_bool_t (*cmplx) COMPLEX_INTERFACE;
} pf_handler;
};
typedef struct xml_elem_hnd xml_elem_hnd_t;
/* prototypes */
static vlc_bool_t parse_plist_node COMPLEX_INTERFACE;
static vlc_bool_t skip_element COMPLEX_INTERFACE;
static vlc_bool_t parse_dict COMPLEX_INTERFACE;
static vlc_bool_t parse_plist_dict COMPLEX_INTERFACE;
static vlc_bool_t parse_tracks_dict COMPLEX_INTERFACE;
static vlc_bool_t parse_track_dict COMPLEX_INTERFACE;
static vlc_bool_t save_data SIMPLE_INTERFACE;
static vlc_bool_t add_meta( input_item_t*, track_elem_t* );
static track_elem_t *new_track( void );
static void free_track( track_elem_t* );
...@@ -123,6 +123,11 @@ vlc_module_begin(); ...@@ -123,6 +123,11 @@ vlc_module_begin();
set_description( _("Dummy ifo demux") ); set_description( _("Dummy ifo demux") );
set_capability( "demux2", 12 ); set_capability( "demux2", 12 );
set_callbacks( E_(Import_IFO), E_(Close_IFO) ); set_callbacks( E_(Import_IFO), E_(Close_IFO) );
add_submodule();
set_description( _("iTunes Music Library importer") );
add_shortcut( "itml" );
set_capability( "demux2", 10 );
set_callbacks( E_(Import_iTML), E_(Close_iTML) );
vlc_module_end(); vlc_module_end();
......
...@@ -73,6 +73,9 @@ void E_(Close_IFO) ( vlc_object_t * ); ...@@ -73,6 +73,9 @@ void E_(Close_IFO) ( vlc_object_t * );
int E_(Import_VideoPortal) ( vlc_object_t * ); int E_(Import_VideoPortal) ( vlc_object_t * );
void E_(Close_VideoPortal) ( vlc_object_t * ); void E_(Close_VideoPortal) ( vlc_object_t * );
int E_(Import_iTML) ( vlc_object_t * );
void E_(Close_iTML) ( vlc_object_t * );
#define INIT_PLAYLIST_STUFF \ #define INIT_PLAYLIST_STUFF \
playlist_t *p_playlist = pl_Yield( p_demux ); \ playlist_t *p_playlist = pl_Yield( p_demux ); \
input_thread_t *p_input_thread = (input_thread_t *)vlc_object_find( p_demux, VLC_OBJECT_INPUT, FIND_PARENT ); \ input_thread_t *p_input_thread = (input_thread_t *)vlc_object_find( p_demux, VLC_OBJECT_INPUT, FIND_PARENT ); \
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment