1 /*******************************************************************************
2 * itml.c : iTunes Music Library import functions
3 *******************************************************************************
4 * Copyright (C) 2007 the VideoLAN team
7 * Authors: Yoann Peronneau <yoann@videolan.org>
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (at your option) any later version.
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22 *******************************************************************************/
24 * \file modules/demux/playlist/itml.c
25 * \brief iTunes Music Library import functions
29 #include <vlc_demux.h>
33 #include "vlc_strings.h"
42 static int Control( demux_t *, int, va_list );
43 static int Demux( demux_t * );
46 * \brief iTML submodule initialization function
48 int E_(Import_iTML)( vlc_object_t *p_this )
50 DEMUX_BY_EXTENSION_OR_FORCED_MSG( ".xml", "itml",
51 "using iTunes Media Library reader" );
55 void E_(Close_iTML)( vlc_object_t *p_this )
57 demux_t *p_demux = (demux_t *)p_this;
58 free( p_demux->p_sys );
62 * \brief demuxer function for iTML parsing
64 int Demux( demux_t *p_demux )
66 int i_ret = VLC_SUCCESS;
68 xml_reader_t *p_xml_reader = NULL;
69 char *psz_name = NULL;
71 p_demux->p_sys->i_ntracks = 0;
73 /* create new xml parser from stream */
74 p_xml = xml_Create( p_demux );
79 p_xml_reader = xml_ReaderCreate( p_xml, p_demux->s );
84 /* locating the root node */
85 if( i_ret == VLC_SUCCESS )
89 if( xml_ReaderRead( p_xml_reader ) != 1 )
91 msg_Err( p_demux, "can't read xml stream" );
94 } while( i_ret == VLC_SUCCESS &&
95 xml_ReaderNodeType( p_xml_reader ) != XML_READER_STARTELEM );
97 /* checking root node name */
98 if( i_ret == VLC_SUCCESS )
100 psz_name = xml_ReaderName( p_xml_reader );
101 if( !psz_name || strcmp( psz_name, "plist" ) )
103 msg_Err( p_demux, "invalid root node name: %s", psz_name );
104 i_ret = VLC_EGENERIC;
109 if( i_ret == VLC_SUCCESS )
111 xml_elem_hnd_t pl_elements[] =
112 { {"dict", COMPLEX_CONTENT, {.cmplx = parse_plist_dict} } };
113 i_ret = parse_plist_node( p_demux, p_playlist, p_current_input,
114 NULL, p_xml_reader, "plist",
116 HANDLE_PLAY_AND_RELEASE;
120 xml_ReaderDelete( p_xml, p_xml_reader );
123 return 0; /* Needed for correct operation of go back */
126 /** \brief dummy function for demux callback interface */
127 static int Control( demux_t *p_demux, int i_query, va_list args )
133 * \brief parse the root node of the playlist
135 static vlc_bool_t parse_plist_node COMPLEX_INTERFACE
137 char *psz_name = NULL;
138 char *psz_value = NULL;
139 vlc_bool_t b_version_found = VLC_FALSE;
141 /* read all playlist attributes */
142 while( xml_ReaderNextAttr( p_xml_reader ) == VLC_SUCCESS )
144 psz_name = xml_ReaderName( p_xml_reader );
145 psz_value = xml_ReaderValue( p_xml_reader );
146 if( !psz_name || !psz_value )
148 msg_Err( p_demux, "invalid xml stream @ <plist>" );
152 /* attribute: version */
153 if( !strcmp( psz_name, "version" ) )
155 b_version_found = VLC_TRUE;
156 if( strcmp( psz_value, "1.0" ) )
157 msg_Warn( p_demux, "unsupported iTunes Media Library version" );
159 /* unknown attribute */
161 msg_Warn( p_demux, "invalid <plist> attribute:\"%s\"", psz_name);
165 /* attribute version is mandatory !!! */
166 if( !b_version_found )
167 msg_Warn( p_demux, "<plist> requires \"version\" attribute" );
169 return parse_dict( p_demux, p_playlist, p_input_item, NULL, p_xml_reader,
170 "plist", p_handlers );
174 * \brief parse a <dict>
175 * \param COMPLEX_INTERFACE
177 static vlc_bool_t parse_dict COMPLEX_INTERFACE
180 char *psz_name = NULL;
181 char *psz_value = NULL;
182 char *psz_key = NULL;
183 xml_elem_hnd_t *p_handler = NULL;
185 while( xml_ReaderRead( p_xml_reader ) == 1 )
187 i_node = xml_ReaderNodeType( p_xml_reader );
190 case XML_READER_NONE:
193 case XML_READER_STARTELEM:
194 /* element start tag */
195 psz_name = xml_ReaderName( p_xml_reader );
196 if( !psz_name || !*psz_name )
198 msg_Err( p_demux, "invalid xml stream" );
203 for( p_handler = p_handlers;
204 p_handler->name && strcmp( psz_name, p_handler->name );
206 if( !p_handler->name )
208 msg_Err( p_demux, "unexpected element <%s>", psz_name );
213 /* complex content is parsed in a separate function */
214 if( p_handler->type == COMPLEX_CONTENT )
216 if( p_handler->pf_handler.cmplx( p_demux,
235 case XML_READER_TEXT:
236 /* simple element content */
238 psz_value = xml_ReaderValue( p_xml_reader );
241 msg_Err( p_demux, "invalid xml stream" );
247 case XML_READER_ENDELEM:
248 /* element end tag */
249 psz_name = xml_ReaderName( p_xml_reader );
252 msg_Err( p_demux, "invalid xml stream" );
256 /* leave if the current parent node <track> is terminated */
257 if( !strcmp( psz_name, psz_element ) )
262 /* there MUST have been a start tag for that element name */
263 if( !p_handler || !p_handler->name
264 || strcmp( p_handler->name, psz_name ))
266 msg_Err( p_demux, "there's no open element left for <%s>",
271 /* special case: key */
272 if( !strcmp( p_handler->name, "key" ) )
274 psz_key = strdup( psz_value );
276 /* call the simple handler */
277 else if( p_handler->pf_handler.smpl )
279 p_handler->pf_handler.smpl( p_track, psz_key, psz_value );
286 /* unknown/unexpected xml node */
287 msg_Err( p_demux, "unexpected xml node %i", i_node );
293 msg_Err( p_demux, "unexpected end of xml data" );
298 static vlc_bool_t parse_plist_dict COMPLEX_INTERFACE
300 xml_elem_hnd_t pl_elements[] =
301 { {"dict", COMPLEX_CONTENT, {.cmplx = parse_tracks_dict} },
302 {"array", SIMPLE_CONTENT, {NULL} },
303 {"key", SIMPLE_CONTENT, {NULL} },
304 {"integer", SIMPLE_CONTENT, {NULL} },
305 {"string", SIMPLE_CONTENT, {NULL} },
306 {"date", SIMPLE_CONTENT, {NULL} },
307 {"true", SIMPLE_CONTENT, {NULL} },
308 {"false", SIMPLE_CONTENT, {NULL} },
309 {NULL, UNKNOWN_CONTENT, {NULL} }
312 return parse_dict( p_demux, p_playlist, p_input_item, NULL, p_xml_reader,
313 "dict", pl_elements );
316 static vlc_bool_t parse_tracks_dict COMPLEX_INTERFACE
318 xml_elem_hnd_t tracks_elements[] =
319 { {"dict", COMPLEX_CONTENT, {.cmplx = parse_track_dict} },
320 {"key", SIMPLE_CONTENT, {NULL} },
321 {NULL, UNKNOWN_CONTENT, {NULL} }
324 parse_dict( p_demux, p_playlist, p_input_item, NULL, p_xml_reader,
325 "dict", tracks_elements );
327 msg_Info( p_demux, "added %i tracks successfully",
328 p_demux->p_sys->i_ntracks );
333 static vlc_bool_t parse_track_dict COMPLEX_INTERFACE
335 input_item_t *p_new_input = NULL;
337 char *psz_uri = NULL;
338 p_track = new_track();
340 xml_elem_hnd_t track_elements[] =
341 { {"array", COMPLEX_CONTENT, {.cmplx = skip_element} },
342 {"key", SIMPLE_CONTENT, {.smpl = save_data} },
343 {"integer", SIMPLE_CONTENT, {.smpl = save_data} },
344 {"string", SIMPLE_CONTENT, {.smpl = save_data} },
345 {"date", SIMPLE_CONTENT, {.smpl = save_data} },
346 {"true", SIMPLE_CONTENT, {NULL} },
347 {"false", SIMPLE_CONTENT, {NULL} },
348 {NULL, UNKNOWN_CONTENT, {NULL} }
351 i_ret = parse_dict( p_demux, p_playlist, p_input_item, p_track,
352 p_xml_reader, "dict", track_elements );
354 msg_Dbg( p_demux, "name: %s, artist: %s, album: %s, genre: %s, trackNum: %s, location: %s",
355 p_track->name, p_track->artist, p_track->album, p_track->genre, p_track->trackNum, p_track->location );
357 if( !p_track->location )
359 msg_Err( p_demux, "Track needs Location" );
360 free_track( p_track );
364 psz_uri = decode_URI_duplicate( p_track->location );
368 if( strlen( psz_uri ) > 17 &&
369 !strncmp( psz_uri, "file://localhost/", 17 ) )
371 /* remove 'localhost/' */
372 strcpy( psz_uri + 7, psz_uri + 17 );
373 msg_Info( p_demux, "Adding '%s'", psz_uri );
375 p_new_input = input_ItemNewExt( p_playlist, psz_uri,
377 input_ItemAddSubItem( p_input_item, p_new_input );
380 add_meta( p_new_input, p_track );
381 vlc_gc_decref( p_new_input );
383 p_demux->p_sys->i_ntracks++;
387 msg_Err( p_demux, "Don't know how to handle %s", psz_uri );
392 free_track( p_track );
396 static track_elem_t *new_track()
398 track_elem_t *p_track = NULL;
399 p_track = (track_elem_t *)malloc( sizeof( track_elem_t ) );
402 p_track->name = NULL;
403 p_track->artist = NULL;
404 p_track->album = NULL;
405 p_track->genre = NULL;
406 p_track->trackNum = NULL;
407 p_track->location = NULL;
408 p_track->duration = 0;
413 static void free_track( track_elem_t *p_track )
415 fprintf( stderr, "free track\n" );
419 FREE( p_track->name )
420 FREE( p_track->artist )
421 FREE( p_track->album )
422 FREE( p_track->genre )
423 FREE( p_track->trackNum )
424 FREE( p_track->location )
425 p_track->duration = 0;
429 static vlc_bool_t save_data SIMPLE_INTERFACE
431 /* exit if setting is impossible */
432 if( !psz_name || !psz_value || !p_track )
435 /* re-convert xml special characters inside psz_value */
436 resolve_xml_special_chars( psz_value );
438 #define SAVE_INFO( name, value ) \
439 if( !strcmp( psz_name, name ) ) { p_track->value = strdup( psz_value ); }
441 SAVE_INFO( "Name", name )
442 else SAVE_INFO( "Artist", artist )
443 else SAVE_INFO( "Album", album )
444 else SAVE_INFO( "Genre", genre )
445 else SAVE_INFO( "Track Number", trackNum )
446 else SAVE_INFO( "Location", location )
447 else if( !strcmp( psz_name, "Total Time" ) )
449 long i_num = atol( psz_value );
450 p_track->duration = (mtime_t) i_num*1000;
456 * \brief handles the supported <track> sub-elements
458 static vlc_bool_t add_meta( input_item_t *p_input_item,
459 track_elem_t *p_track )
461 /* exit if setting is impossible */
462 if( !p_input_item || !p_track )
465 #define SET_INFO( func, prop ) \
466 if( p_track->prop ) { func( p_input_item, p_track->prop ); }
468 SET_INFO( input_item_SetTitle, name )
469 SET_INFO( input_item_SetArtist, artist )
470 SET_INFO( input_item_SetAlbum, album )
471 SET_INFO( input_item_SetGenre, genre )
472 SET_INFO( input_item_SetTrackNum, trackNum )
473 SET_INFO( input_item_SetDuration, duration )
478 * \brief skips complex element content that we can't manage
480 static vlc_bool_t skip_element COMPLEX_INTERFACE
484 while( xml_ReaderRead( p_xml_reader ) == 1 )
486 if( xml_ReaderNodeType( p_xml_reader ) == XML_READER_ENDELEM )
488 psz_endname = xml_ReaderName( p_xml_reader );
491 if( !strcmp( psz_element, psz_endname ) )