]> git.sesse.net Git - vlc/blob - modules/control/http/mvar.c
HTTP old: use vlc_sd_GetNames(), fix #3490
[vlc] / modules / control / http / mvar.c
1 /*****************************************************************************
2  * mvar.c : Variables handling for the HTTP Interface
3  *****************************************************************************
4  * Copyright (C) 2001-2007 the VideoLAN team
5  * $Id$
6  *
7  * Authors: Gildas Bazin <gbazin@netcourrier.com>
8  *          Laurent Aimar <fenrir@via.ecp.fr>
9  *          Christophe Massiot <massiot@via.ecp.fr>
10  *
11  * This program is free software; you can redistribute it and/or modify
12  * it under the terms of the GNU General Public License as published by
13  * the Free Software Foundation; either version 2 of the License, or
14  * (at your option) any later version.
15  *
16  * This program is distributed in the hope that it will be useful,
17  * but WITHOUT ANY WARRANTY; without even the implied warranty of
18  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
19  * GNU General Public License for more details.
20  *
21  * You should have received a copy of the GNU General Public License
22  * along with this program; if not, write to the Free Software
23  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
24  *****************************************************************************/
25 #ifdef HAVE_CONFIG_H
26 # include "config.h"
27 #endif
28
29 #include "http.h"
30 #include <limits.h>
31 #include <errno.h>
32 #include <ctype.h>
33 #include <fcntl.h>
34 #ifdef HAVE_SYS_STAT_H
35 #include <sys/stat.h>
36 #endif
37 #include <vlc_fs.h>
38 #include <vlc_services_discovery.h>
39
40 /* Utility function for scandir */
41 static int Filter( const char *foo )
42 {
43     return strcmp( foo, "." );
44 };
45
46 static int InsensitiveAlphasort( const char **foo1,
47                                  const char **foo2 )
48 {
49     return strcasecmp( *foo1, *foo2 );
50 };
51
52
53
54 mvar_t *mvar_New( const char *name, const char *value )
55 {
56     mvar_t *v = malloc( sizeof( mvar_t ) );
57
58     if( !v ) return NULL;
59     v->name = strdup( name );
60     v->value = strdup( value ? value : "" );
61
62     v->i_field = 0;
63     v->field = xmalloc( sizeof( mvar_t * ) );
64     v->field[0] = NULL;
65
66     return v;
67 }
68
69 void mvar_Delete( mvar_t *v )
70 {
71     int i;
72
73     free( v->name );
74     free( v->value );
75
76     for( i = 0; i < v->i_field; i++ )
77     {
78         mvar_Delete( v->field[i] );
79     }
80     free( v->field );
81     free( v );
82 }
83
84 void mvar_AppendVar( mvar_t *v, mvar_t *f )
85 {
86     v->field = xrealloc( v->field, sizeof( mvar_t * ) * ( v->i_field + 2 ) );
87     v->field[v->i_field] = f;
88     v->i_field++;
89 }
90
91 mvar_t *mvar_Duplicate( const mvar_t *v )
92 {
93     int i;
94     mvar_t *n;
95
96     n = mvar_New( v->name, v->value );
97     for( i = 0; i < v->i_field; i++ )
98     {
99         mvar_AppendVar( n, mvar_Duplicate( v->field[i] ) );
100     }
101
102     return n;
103 }
104
105 void mvar_PushVar( mvar_t *v, mvar_t *f )
106 {
107     v->field = xrealloc( v->field, sizeof( mvar_t * ) * ( v->i_field + 2 ) );
108     if( v->i_field > 0 )
109     {
110         memmove( &v->field[1], &v->field[0], sizeof( mvar_t * ) * v->i_field );
111     }
112     v->field[0] = f;
113     v->i_field++;
114 }
115
116 void mvar_RemoveVar( mvar_t *v, mvar_t *f )
117 {
118     int i;
119     for( i = 0; i < v->i_field; i++ )
120     {
121         if( v->field[i] == f )
122         {
123             break;
124         }
125     }
126     if( i >= v->i_field )
127     {
128         return;
129     }
130
131     if( i + 1 < v->i_field )
132     {
133         memmove( &v->field[i], &v->field[i+1], sizeof( mvar_t * ) * ( v->i_field - i - 1 ) );
134     }
135     v->i_field--;
136     /* FIXME should do a realloc */
137 }
138
139 mvar_t *mvar_GetVar( mvar_t *s, const char *name )
140 {
141     /* format: name[index].field */
142     const char *field = strchr( name, '.' );
143     char base[1 + (field ? (size_t)(field - name) : strlen( name ))];
144     char *p;
145     int i_index, i;
146
147     strlcpy( base, name, sizeof (base) );
148     if( field != NULL )
149         field++;
150
151     if( ( p = strchr( base, '[' ) ) != NULL )
152     {
153         char *end;
154         unsigned long l = strtoul( p, &end, 0 );
155
156         if( ( l > INT_MAX ) || strcmp( "]", end ) )
157             return NULL;
158
159         *p++ = '\0';
160         i_index = (int)l;
161     }
162     else
163     {
164         i_index = 0;
165     }
166
167     for( i = 0; i < s->i_field; i++ )
168     {
169         if( !strcmp( s->field[i]->name, base ) )
170         {
171             if( i_index > 0 )
172             {
173                 i_index--;
174             }
175             else
176             {
177                 if( field )
178                 {
179                     return mvar_GetVar( s->field[i], field );
180                 }
181                 else
182                 {
183                     return s->field[i];
184                 }
185             }
186         }
187     }
188     return NULL;
189 }
190
191 const char *mvar_GetValue( mvar_t *v, const char *field )
192 {
193     if( *field == '\0' )
194     {
195         return v->value;
196     }
197     else
198     {
199         mvar_t *f = mvar_GetVar( v, field );
200         if( f )
201         {
202             return f->value;
203         }
204         else
205         {
206             return field;
207         }
208     }
209 }
210
211 void mvar_PushNewVar( mvar_t *vars, const char *name,
212                           const char *value )
213 {
214     mvar_t *f = mvar_New( name, value );
215     mvar_PushVar( vars, f );
216 }
217
218 void mvar_AppendNewVar( mvar_t *vars, const char *name,
219                             const char *value )
220 {
221     mvar_t *f = mvar_New( name, value );
222     mvar_AppendVar( vars, f );
223 }
224
225
226 /* arg= start[:stop[:step]],.. */
227 mvar_t *mvar_IntegerSetNew( const char *name, const char *arg )
228 {
229     char *dup = strdup( arg );
230     char *str = dup;
231     mvar_t *s = mvar_New( name, "set" );
232
233     while( str )
234     {
235         char *p;
236         int  i_start,i_stop,i_step;
237         int  i_match;
238
239         p = strchr( str, ',' );
240         if( p )
241         {
242             *p++ = '\0';
243         }
244
245         i_step = 0;
246         i_match = sscanf( str, "%d:%d:%d", &i_start, &i_stop, &i_step );
247
248         if( i_match == 1 )
249         {
250             i_stop = i_start;
251             i_step = 1;
252         }
253         else if( i_match == 2 )
254         {
255             i_step = i_start < i_stop ? 1 : -1;
256         }
257
258         if( i_match >= 1 )
259         {
260             int i;
261
262             if( ( i_start <= i_stop && i_step > 0 ) ||
263                 ( i_start >= i_stop && i_step < 0 ) )
264             {
265                 for( i = i_start; ; i += i_step )
266                 {
267                     char   value[79];
268
269                     if( ( i_step > 0 && i > i_stop ) ||
270                         ( i_step < 0 && i < i_stop ) )
271                     {
272                         break;
273                     }
274
275                     sprintf( value, "%d", i );
276
277                     mvar_PushNewVar( s, name, value );
278                 }
279             }
280         }
281         str = p;
282     }
283
284     free( dup );
285     return s;
286 }
287
288 /********************************************************************
289  * Special sets handling
290  ********************************************************************/
291
292 mvar_t *mvar_PlaylistSetNew( intf_thread_t *p_intf, char *name,
293                                  playlist_t *p_pl )
294 {
295     mvar_t *s = mvar_New( name, "set" );
296     playlist_Lock( p_pl );
297     PlaylistListNode( p_intf, p_pl, p_pl->p_root_category , name, s, 0 );
298     playlist_Unlock( p_pl );
299     return s;
300 }
301
302 mvar_t *mvar_InfoSetNew( char *name, input_thread_t *p_input )
303 {
304     mvar_t *s = mvar_New( name, "set" );
305     int i, j;
306
307     if( p_input == NULL || p_input->p == NULL /* workarround assert in input_GetItem */ )
308     {
309         return s;
310     }
311
312     vlc_mutex_lock( &input_GetItem(p_input)->lock );
313     for ( i = 0; i < input_GetItem(p_input)->i_categories; i++ )
314     {
315         info_category_t *p_category = input_GetItem(p_input)->pp_categories[i];
316
317         mvar_t *cat  = mvar_New( name, "set" );
318         mvar_t *iset = mvar_New( "info", "set" );
319
320         mvar_AppendNewVar( cat, "name", p_category->psz_name );
321         mvar_AppendVar( cat, iset );
322
323         for ( j = 0; j < p_category->i_infos; j++ )
324         {
325             info_t *p_info = p_category->pp_infos[j];
326             mvar_t *info = mvar_New( "info", "" );
327
328             /* msg_Dbg( p_input, "adding info name=%s value=%s",
329                      psz_name, psz_value ); */
330             mvar_AppendNewVar( info, "name",  p_info->psz_name );
331             mvar_AppendNewVar( info, "value", p_info->psz_value );
332             mvar_AppendVar( iset, info );
333         }
334         mvar_AppendVar( s, cat );
335     }
336     vlc_mutex_unlock( &input_GetItem(p_input)->lock );
337
338     return s;
339 }
340
341 mvar_t *mvar_ServicesSetNew( intf_thread_t *p_intf, char *psz_name )
342 {
343     mvar_t *s = mvar_New( psz_name, "set" );
344     char **longnames;
345     char **names = vlc_sd_GetNames( p_intf, &longnames, NULL );
346     if( names == NULL )
347         goto out;
348
349     for( size_t i = 0; names[i]; i++ )
350     {
351         mvar_t *sd = mvar_New( "sd", names[i] );
352         mvar_AppendNewVar( sd, "name", longnames[i] );
353         mvar_AppendVar( s, sd );
354         free( names[i] );
355         free( longnames[i] );
356     }
357
358     free( longnames );
359     free( names );
360 out:
361     return s;
362 }
363
364 mvar_t *mvar_InputVarSetNew( intf_thread_t *p_intf, char *name,
365                                  input_thread_t *p_input,
366                                  const char *psz_variable )
367 {
368     intf_sys_t     *p_sys = p_intf->p_sys;
369     mvar_t *s = mvar_New( name, "set" );
370     vlc_value_t val, val_list, text_list;
371     int i_type, i;
372
373     if( p_input == NULL )
374     {
375         return s;
376     }
377
378     /* Check the type of the object variable */
379     i_type = var_Type( p_sys->p_input, psz_variable );
380
381     /* Make sure we want to display the variable */
382     if( i_type & VLC_VAR_HASCHOICE )
383     {
384         var_Change( p_sys->p_input, psz_variable, VLC_VAR_CHOICESCOUNT, &val, NULL );
385         if( val.i_int == 0 ) return s;
386         if( (i_type & VLC_VAR_TYPE) != VLC_VAR_VARIABLE && val.i_int == 1 )
387             return s;
388     }
389     else
390     {
391         return s;
392     }
393
394     switch( i_type & VLC_VAR_TYPE )
395     {
396     case VLC_VAR_VOID:
397     case VLC_VAR_BOOL:
398     case VLC_VAR_VARIABLE:
399     case VLC_VAR_STRING:
400     case VLC_VAR_INTEGER:
401         break;
402     default:
403         /* Variable doesn't exist or isn't handled */
404         return s;
405     }
406
407     if( var_Get( p_sys->p_input, psz_variable, &val ) < 0 )
408     {
409         return s;
410     }
411
412     if( var_Change( p_sys->p_input, psz_variable, VLC_VAR_GETLIST,
413                     &val_list, &text_list ) < 0 )
414     {
415         if( (i_type & VLC_VAR_TYPE) == VLC_VAR_STRING ) free( val.psz_string );
416         return s;
417     }
418
419     for( i = 0; i < val_list.p_list->i_count; i++ )
420     {
421         char psz_int[16];
422         mvar_t *itm;
423
424         switch( i_type & VLC_VAR_TYPE )
425         {
426         case VLC_VAR_STRING:
427             itm = mvar_New( name, "set" );
428             mvar_AppendNewVar( itm, "name", text_list.p_list->p_values[i].psz_string );
429             mvar_AppendNewVar( itm, "id", val_list.p_list->p_values[i].psz_string );
430             snprintf( psz_int, sizeof(psz_int), "%d",
431                       ( !strcmp( val.psz_string,
432                                    val_list.p_list->p_values[i].psz_string )
433                            && !( i_type & VLC_VAR_ISCOMMAND ) ) );
434             mvar_AppendNewVar( itm, "selected", psz_int );
435             mvar_AppendVar( s, itm );
436             break;
437
438         case VLC_VAR_INTEGER:
439             itm = mvar_New( name, "set" );
440             mvar_AppendNewVar( itm, "name", text_list.p_list->p_values[i].psz_string );
441             snprintf( psz_int, sizeof(psz_int), "%d",
442                       val_list.p_list->p_values[i].i_int );
443             mvar_AppendNewVar( itm, "id", psz_int );
444             snprintf( psz_int, sizeof(psz_int), "%d",
445                       ( val.i_int == val_list.p_list->p_values[i].i_int )
446                          && !( i_type & VLC_VAR_ISCOMMAND ) );
447             mvar_AppendNewVar( itm, "selected", psz_int );
448             mvar_AppendVar( s, itm );
449             break;
450
451         default:
452             break;
453         }
454     }
455     /* clean up everything */
456     if( (i_type & VLC_VAR_TYPE) == VLC_VAR_STRING ) free( val.psz_string );
457     var_FreeList( &val_list, &text_list );
458     return s;
459 }
460
461 #if 0
462 mvar_t *mvar_HttpdInfoSetNew( char *name, httpd_t *p_httpd, int i_type )
463 {
464     mvar_t       *s = mvar_New( name, "set" );
465     httpd_info_t info;
466     int          i;
467
468     if( !p_httpd->pf_control( p_httpd, i_type, &info, NULL ) )
469     {
470         for( i= 0; i < info.i_count; )
471         {
472             mvar_t *inf;
473
474             inf = mvar_New( name, "set" );
475             do
476             {
477                 /* fprintf( stderr," mvar_HttpdInfoSetNew: append name=`%s' value=`%s'\n",
478                             info.info[i].psz_name, info.info[i].psz_value ); */
479                 mvar_AppendNewVar( inf,
480                                    info.info[i].psz_name,
481                                    info.info[i].psz_value );
482                 i++;
483             } while( i < info.i_count && strcmp( info.info[i].psz_name, "id" ) );
484             mvar_AppendVar( s, inf );
485         }
486     }
487
488     /* free mem */
489     for( i = 0; i < info.i_count; i++ )
490     {
491         free( info.info[i].psz_name );
492         free( info.info[i].psz_value );
493     }
494     if( info.i_count > 0 )
495     {
496         free( info.info );
497     }
498
499     return s;
500 }
501 #endif
502
503 mvar_t *mvar_FileSetNew( intf_thread_t *p_intf, char *name,
504                              char *psz_dir )
505 {
506     mvar_t *s = mvar_New( name, "set" );
507     char        **ppsz_dir_content;
508     int           i_dir_content, i;
509     psz_dir = RealPath( psz_dir );
510
511     /* parse psz_src dir */
512     if( ( i_dir_content = vlc_scandir( psz_dir, &ppsz_dir_content, Filter,
513                                         InsensitiveAlphasort ) ) == -1 )
514     {
515         if( errno != ENOENT && errno != ENOTDIR )
516             msg_Warn( p_intf, "error while scanning dir %s (%m)", psz_dir );
517         free( psz_dir );
518         return s;
519     }
520
521     for( i = 0; i < i_dir_content; i++ )
522     {
523 #ifdef HAVE_SYS_STAT_H
524         struct stat stat_info;
525 #endif
526         char *psz_name = ppsz_dir_content[i], *psz_ext, *psz_dummy;
527         char psz_tmp[strlen( psz_dir ) + 1 + strlen( psz_name ) + 1];
528         mvar_t *f;
529
530 #if defined( WIN32 )
531         if( psz_dir[0] == '\0' || (psz_dir[0] == '\\' && psz_dir[1] == '\0') )
532         {
533             strcpy( psz_tmp, psz_name );
534         }
535         else
536 #endif
537         {
538             sprintf( psz_tmp, "%s"DIR_SEP"%s", psz_dir, psz_name );
539
540 #ifdef HAVE_SYS_STAT_H
541             if( vlc_stat( psz_tmp, &stat_info ) == -1 )
542             {
543                 free( psz_name );
544                 continue;
545             }
546 #endif
547         }
548         f = mvar_New( name, "set" );
549
550         /* put lower-case file extension in 'ext' */
551         psz_ext = strrchr( psz_name, '.' );
552         psz_ext = strdup( psz_ext != NULL ? psz_ext + 1 : "" );
553         for( psz_dummy = psz_ext; *psz_dummy != '\0'; psz_dummy++ )
554             *psz_dummy = tolower( *psz_dummy );
555
556         mvar_AppendNewVar( f, "ext", psz_ext );
557         free( psz_ext );
558
559 #if defined( WIN32 )
560         if( psz_dir[0] == '\0' || (psz_dir[0] == '\\' && psz_dir[1] == '\0') )
561         {
562             char psz_tmp[3];
563             sprintf( psz_tmp, "%c:", psz_name[0] );
564             mvar_AppendNewVar( f, "name", psz_name );
565             mvar_AppendNewVar( f, "basename", psz_tmp );
566             mvar_AppendNewVar( f, "type", "directory" );
567             mvar_AppendNewVar( f, "size", "unknown" );
568             mvar_AppendNewVar( f, "date", "unknown" );
569         }
570         else
571 #endif
572         {
573             char psz_buf[26];
574             char psz_tmp[strlen( psz_dir ) + 1 + strlen( psz_name ) + 1];
575
576             sprintf( psz_tmp, "%s"DIR_SEP"%s", psz_dir, psz_name );
577             mvar_AppendNewVar( f, "name", psz_tmp );
578             mvar_AppendNewVar( f, "basename", psz_name );
579
580 #ifdef HAVE_SYS_STAT_H
581             if( S_ISDIR( stat_info.st_mode ) )
582             {
583                 mvar_AppendNewVar( f, "type", "directory" );
584             }
585             else if( S_ISREG( stat_info.st_mode ) )
586             {
587                 mvar_AppendNewVar( f, "type", "file" );
588             }
589             else
590             {
591                 mvar_AppendNewVar( f, "type", "unknown" );
592             }
593
594             snprintf( psz_buf, sizeof( psz_buf ), "%"PRId64,
595                       (int64_t)stat_info.st_size );
596             mvar_AppendNewVar( f, "size", psz_buf );
597
598             /* FIXME memory leak FIXME */
599 #   ifdef HAVE_CTIME_R
600             ctime_r( &stat_info.st_mtime, psz_buf );
601             mvar_AppendNewVar( f, "date", psz_buf );
602 #   else
603             mvar_AppendNewVar( f, "date", ctime( &stat_info.st_mtime ) );
604 #   endif
605
606 #else
607             mvar_AppendNewVar( f, "type", "unknown" );
608             mvar_AppendNewVar( f, "size", "unknown" );
609             mvar_AppendNewVar( f, "date", "unknown" );
610 #endif
611         }
612
613         mvar_AppendVar( s, f );
614
615         free( psz_name );
616     }
617
618     free( psz_dir );
619     free( ppsz_dir_content );
620     return s;
621 }
622
623 static void mvar_VlmSetNewLoop( char *name, vlm_t *vlm, mvar_t *s,
624                                 vlm_message_t *el, bool b_name )
625 {
626     /* Over name */
627     mvar_t        *set;
628     int k;
629
630     /* Add a node with name and info */
631     set = mvar_New( name, "set" );
632     if( b_name == true )
633     {
634         mvar_AppendNewVar( set, "name", el->psz_name );
635     }
636
637     for( k = 0; k < el->i_child; k++ )
638     {
639         vlm_message_t *ch = el->child[k];
640         if( ch->i_child > 0 )
641         {
642             mvar_VlmSetNewLoop( ch->psz_name, vlm, set, ch, false );
643         }
644         else
645         {
646             if( ch->psz_value )
647             {
648                 mvar_AppendNewVar( set, ch->psz_name, ch->psz_value );
649             }
650             else
651             {
652                 mvar_AppendNewVar( set, el->psz_name, ch->psz_name );
653             }
654         }
655     }
656
657     mvar_AppendVar( s, set );
658 }
659
660 mvar_t *mvar_VlmSetNew( char *name, vlm_t *vlm )
661 {
662     mvar_t        *s = mvar_New( name, "set" );
663 #ifdef ENABLE_VLM
664     vlm_message_t *msg;
665     int    i;
666
667     if( vlm == NULL ) return s;
668
669     if( vlm_ExecuteCommand( vlm, "show", &msg ) )
670         return s;
671
672     for( i = 0; i < msg->i_child; i++ )
673     {
674         /* Over media, schedule */
675         vlm_message_t *ch = msg->child[i];
676         int j;
677
678         for( j = 0; j < ch->i_child; j++ )
679         {
680             /* Over name */
681             vlm_message_t *el = ch->child[j];
682             vlm_message_t *inf, *desc;
683             char          psz[6 + strlen(el->psz_name)];
684
685             sprintf( psz, "show %s", el->psz_name );
686             if( vlm_ExecuteCommand( vlm, psz, &inf ) )
687                 continue;
688             desc = inf->child[0];
689
690             mvar_VlmSetNewLoop( el->psz_name, vlm, s, desc, true );
691
692             vlm_MessageDelete( inf );
693         }
694     }
695     vlm_MessageDelete( msg );
696 #endif /* ENABLE_VLM */
697     return s;
698 }