cfgfile.c 42.7 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12
/* Icecast
 *
 * This program is distributed under the GNU General Public License, version 2.
 * A copy of this license is included with this source.
 *
 * Copyright 2000-2004, Jack Moffitt <jack@xiph.org, 
 *                      Michael Smith <msmith@xiph.org>,
 *                      oddsock <oddsock@xiph.org>,
 *                      Karl Heyes <karl@xiph.org>
 *                      and others (see AUTHORS for details).
 */

13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <libxml/xmlmemory.h>
#include <libxml/parser.h>

#include "thread/thread.h"
#include "cfgfile.h"
#include "refbuf.h"
#include "client.h"
#include "logging.h" 

#define CATMODULE "CONFIG"
#define CONFIG_DEFAULT_LOCATION "Earth"
#define CONFIG_DEFAULT_ADMIN "icemaster@localhost"
#define CONFIG_DEFAULT_CLIENT_LIMIT 256
#define CONFIG_DEFAULT_SOURCE_LIMIT 16
34
#define CONFIG_DEFAULT_QUEUE_SIZE_LIMIT (500*1024)
35
#define CONFIG_DEFAULT_BURST_SIZE (64*1024)
36 37 38 39 40 41
#define CONFIG_DEFAULT_THREADPOOL_SIZE 4
#define CONFIG_DEFAULT_CLIENT_TIMEOUT 30
#define CONFIG_DEFAULT_HEADER_TIMEOUT 15
#define CONFIG_DEFAULT_SOURCE_TIMEOUT 10
#define CONFIG_DEFAULT_SOURCE_PASSWORD "changeme"
#define CONFIG_DEFAULT_RELAY_PASSWORD "changeme"
42
#define CONFIG_DEFAULT_MASTER_USERNAME "relay"
43
#define CONFIG_DEFAULT_SHOUTCAST_MOUNT "/stream"
44 45 46 47
#define CONFIG_DEFAULT_ICE_LOGIN 0
#define CONFIG_DEFAULT_FILESERVE 1
#define CONFIG_DEFAULT_TOUCH_FREQ 5
#define CONFIG_DEFAULT_HOSTNAME "localhost"
48
#define CONFIG_DEFAULT_PLAYLIST_LOG NULL
49 50 51 52 53 54 55 56 57 58 59 60 61 62 63
#define CONFIG_DEFAULT_ACCESS_LOG "access.log"
#define CONFIG_DEFAULT_ERROR_LOG "error.log"
#define CONFIG_DEFAULT_LOG_LEVEL 4
#define CONFIG_DEFAULT_CHROOT 0
#define CONFIG_DEFAULT_CHUID 0
#define CONFIG_DEFAULT_USER NULL
#define CONFIG_DEFAULT_GROUP NULL
#define CONFIG_MASTER_UPDATE_INTERVAL 120
#define CONFIG_YP_URL_TIMEOUT 10

#ifndef _WIN32
#define CONFIG_DEFAULT_BASE_DIR "/usr/local/icecast"
#define CONFIG_DEFAULT_LOG_DIR "/usr/local/icecast/logs"
#define CONFIG_DEFAULT_WEBROOT_DIR "/usr/local/icecast/webroot"
#define CONFIG_DEFAULT_ADMINROOT_DIR "/usr/local/icecast/admin"
64
#define MIMETYPESFILE "/etc/mime.types"
65 66 67 68 69
#else
#define CONFIG_DEFAULT_BASE_DIR ".\\"
#define CONFIG_DEFAULT_LOG_DIR ".\\logs"
#define CONFIG_DEFAULT_WEBROOT_DIR ".\\webroot"
#define CONFIG_DEFAULT_ADMINROOT_DIR ".\\admin"
70
#define MIMETYPESFILE ".\\mime.types"
71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90
#endif

static ice_config_t _current_configuration;
static ice_config_locks _locks;

static void _set_defaults(ice_config_t *c);
static void _parse_root(xmlDocPtr doc, xmlNodePtr node, ice_config_t *c);
static void _parse_limits(xmlDocPtr doc, xmlNodePtr node, ice_config_t *c);
static void _parse_directory(xmlDocPtr doc, xmlNodePtr node, ice_config_t *c);
static void _parse_paths(xmlDocPtr doc, xmlNodePtr node, ice_config_t *c);
static void _parse_logging(xmlDocPtr doc, xmlNodePtr node, ice_config_t *c);
static void _parse_security(xmlDocPtr doc, xmlNodePtr node, ice_config_t *c);
static void _parse_authentication(xmlDocPtr doc, xmlNodePtr node, 
        ice_config_t *c);
static void _parse_relay(xmlDocPtr doc, xmlNodePtr node, ice_config_t *c);
static void _parse_mount(xmlDocPtr doc, xmlNodePtr node, ice_config_t *c);
static void _parse_listen_socket(xmlDocPtr doc, xmlNodePtr node, 
        ice_config_t *c);
static void _add_server(xmlDocPtr doc, xmlNodePtr node, ice_config_t *c);

91
static void create_locks(void) {
92
    thread_mutex_create(&_locks.relay_lock);
93
    thread_rwlock_create(&_locks.config_lock);
94 95
}

96
static void release_locks(void) {
97
    thread_mutex_destroy(&_locks.relay_lock);
98
    thread_rwlock_destroy(&_locks.config_lock);
99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117
}

void config_initialize(void) {
    create_locks();
}

void config_shutdown(void) {
    config_get_config();
    config_clear(&_current_configuration);
    config_release_config();
    release_locks();
}

void config_init_configuration(ice_config_t *configuration)
{
    memset(configuration, 0, sizeof(ice_config_t));
    _set_defaults(configuration);
}

118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135
static void config_clear_mount (mount_proxy *mount)
{
    config_options_t *option;

    xmlFree (mount->mountname);
    xmlFree (mount->username);
    xmlFree (mount->password);
    xmlFree (mount->dumpfile);
    xmlFree (mount->intro_filename);
    xmlFree (mount->on_connect);
    xmlFree (mount->on_disconnect);
    xmlFree (mount->fallback_mount);
    xmlFree (mount->stream_name);
    xmlFree (mount->stream_description);
    xmlFree (mount->stream_url);
    xmlFree (mount->stream_genre);
    xmlFree (mount->bitrate);
    xmlFree (mount->type);
136
    xmlFree (mount->charset);
137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153
    xmlFree (mount->cluster_password);

    xmlFree (mount->auth_type);
    option = mount->auth_options;
    while (option)
    {
        config_options_t *nextopt = option->next;
        xmlFree (option->name);
        xmlFree (option->value);
        free (option);
        option = nextopt;
    }
    auth_release (mount->auth);
    free (mount);
}


154 155 156 157 158 159 160 161
void config_clear(ice_config_t *c)
{
    ice_config_dir_t *dirnode, *nextdirnode;
    relay_server *relay, *nextrelay;
    mount_proxy *mount, *nextmount;
    aliases *alias, *nextalias;
    int i;

162
    free(c->config_filename);
163

164 165 166
    if (c->location) xmlFree(c->location);
    if (c->admin) xmlFree(c->admin);
    if (c->source_password) xmlFree(c->source_password);
167 168 169 170
    if (c->admin_username)
        xmlFree(c->admin_username);
    if (c->admin_password)
        xmlFree(c->admin_password);
171 172 173 174
    if (c->relay_username)
        xmlFree(c->relay_username);
    if (c->relay_password)
        xmlFree(c->relay_password);
175 176 177 178 179
    if (c->hostname) xmlFree(c->hostname);
    if (c->base_dir) xmlFree(c->base_dir);
    if (c->log_dir) xmlFree(c->log_dir);
    if (c->webroot_dir) xmlFree(c->webroot_dir);
    if (c->adminroot_dir) xmlFree(c->adminroot_dir);
Karl Heyes's avatar
Karl Heyes committed
180
    if (c->pidfile)
181
        xmlFree(c->pidfile);
182 183 184 185
    if (c->playlist_log) xmlFree(c->playlist_log);
    if (c->access_log) xmlFree(c->access_log);
    if (c->error_log) xmlFree(c->error_log);
    if (c->shoutcast_mount) xmlFree(c->shoutcast_mount);
186 187 188 189
    for(i=0; i < MAX_LISTEN_SOCKETS; i++) {
        if (c->listeners[i].bind_address) xmlFree(c->listeners[i].bind_address);
    }
    if (c->master_server) xmlFree(c->master_server);
190
    if (c->master_username) xmlFree(c->master_username);
191 192 193
    if (c->master_password) xmlFree(c->master_password);
    if (c->user) xmlFree(c->user);
    if (c->group) xmlFree(c->group);
194
    if (c->mimetypes_fn) xmlFree (c->mimetypes_fn);
195 196 197 198 199 200 201

    thread_mutex_lock(&(_locks.relay_lock));
    relay = c->relay;
    while(relay) {
        nextrelay = relay->next;
        xmlFree(relay->server);
        xmlFree(relay->mount);
Karl Heyes's avatar
Karl Heyes committed
202
        xmlFree(relay->localmount);
203 204 205 206 207 208 209 210
        free(relay);
        relay = nextrelay;
    }
    thread_mutex_unlock(&(_locks.relay_lock));

    mount = c->mounts;
    while(mount) {
        nextmount = mount->next;
211
        config_clear_mount (mount);
212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231
        mount = nextmount;
    }

    alias = c->aliases;
    while(alias) {
        nextalias = alias->next;
        xmlFree(alias->source);
        xmlFree(alias->destination);
        xmlFree(alias->bind_address);
        free(alias);
        alias = nextalias;
    }

    dirnode = c->dir_list;
    while(dirnode) {
        nextdirnode = dirnode->next;
        xmlFree(dirnode->host);
        free(dirnode);
        dirnode = nextdirnode;
    }
232
#ifdef USE_YP
Karl Heyes's avatar
Karl Heyes committed
233
    i = 0;
234
    while (i < c->num_yp_directories)
Karl Heyes's avatar
Karl Heyes committed
235 236 237 238 239
    {
        xmlFree (c->yp_url[i]);
        i++;
    }
#endif
240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269

    memset(c, 0, sizeof(ice_config_t));
}

int config_initial_parse_file(const char *filename)
{
    /* Since we're already pointing at it, we don't need to copy it in place */
    return config_parse_file(filename, &_current_configuration);
}

int config_parse_file(const char *filename, ice_config_t *configuration)
{
    xmlDocPtr doc;
    xmlNodePtr node;

    if (filename == NULL || strcmp(filename, "") == 0) return CONFIG_EINSANE;
    
    xmlInitParser();
    doc = xmlParseFile(filename);
    if (doc == NULL) {
        return CONFIG_EPARSE;
    }

    node = xmlDocGetRootElement(doc);
    if (node == NULL) {
        xmlFreeDoc(doc);
        xmlCleanupParser();
        return CONFIG_ENOROOT;
    }

270
    if (xmlStrcmp (node->name, XMLSTR("icecast")) != 0) {
271 272 273 274 275 276 277
        xmlFreeDoc(doc);
        xmlCleanupParser();
        return CONFIG_EBADROOT;
    }

    config_init_configuration(configuration);

278
    configuration->config_filename = strdup (filename);
279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298

    _parse_root(doc, node->xmlChildrenNode, configuration);

    xmlFreeDoc(doc);

    return 0;
}

int config_parse_cmdline(int arg, char **argv)
{
    return 0;
}

ice_config_locks *config_locks(void)
{
    return &_locks;
}

void config_release_config(void)
{
299
    thread_rwlock_unlock(&(_locks.config_lock));
300 301 302 303
}

ice_config_t *config_get_config(void)
{
304 305 306 307 308 309 310
    thread_rwlock_rlock(&(_locks.config_lock));
    return &_current_configuration;
}

ice_config_t *config_grab_config(void)
{
    thread_rwlock_wlock(&(_locks.config_lock));
311 312 313 314 315 316 317 318 319 320 321 322 323 324 325
    return &_current_configuration;
}

/* MUST be called with the lock held! */
void config_set_config(ice_config_t *config) {
    memcpy(&_current_configuration, config, sizeof(ice_config_t));
}

ice_config_t *config_get_config_unlocked(void)
{
    return &_current_configuration;
}

static void _set_defaults(ice_config_t *configuration)
{
326
    configuration->location = (char *)xmlCharStrdup (CONFIG_DEFAULT_LOCATION);
327
    configuration->server_id = (char *)xmlCharStrdup (ICECAST_VERSION_STRING);
328
    configuration->admin = (char *)xmlCharStrdup (CONFIG_DEFAULT_ADMIN);
329 330 331 332 333 334 335
    configuration->client_limit = CONFIG_DEFAULT_CLIENT_LIMIT;
    configuration->source_limit = CONFIG_DEFAULT_SOURCE_LIMIT;
    configuration->queue_size_limit = CONFIG_DEFAULT_QUEUE_SIZE_LIMIT;
    configuration->threadpool_size = CONFIG_DEFAULT_THREADPOOL_SIZE;
    configuration->client_timeout = CONFIG_DEFAULT_CLIENT_TIMEOUT;
    configuration->header_timeout = CONFIG_DEFAULT_HEADER_TIMEOUT;
    configuration->source_timeout = CONFIG_DEFAULT_SOURCE_TIMEOUT;
336 337
    configuration->source_password = (char *)xmlCharStrdup (CONFIG_DEFAULT_SOURCE_PASSWORD);
    configuration->shoutcast_mount = (char *)xmlCharStrdup (CONFIG_DEFAULT_SHOUTCAST_MOUNT);
338 339 340
    configuration->ice_login = CONFIG_DEFAULT_ICE_LOGIN;
    configuration->fileserve = CONFIG_DEFAULT_FILESERVE;
    configuration->touch_interval = CONFIG_DEFAULT_TOUCH_FREQ;
341
    configuration->on_demand = 0;
342
    configuration->dir_list = NULL;
343 344
    configuration->hostname = (char *)xmlCharStrdup (CONFIG_DEFAULT_HOSTNAME);
    configuration->mimetypes_fn = (char *)xmlCharStrdup (MIMETYPESFILE);
345 346 347
    configuration->port = 0;
    configuration->listeners[0].port = 0;
    configuration->listeners[0].bind_address = NULL;
348
    configuration->listeners[0].shoutcast_compat = 0;
349 350 351
    configuration->master_server = NULL;
    configuration->master_server_port = 0;
    configuration->master_update_interval = CONFIG_MASTER_UPDATE_INTERVAL;
352
    configuration->master_username = (char *)xmlCharStrdup (CONFIG_DEFAULT_MASTER_USERNAME);
353
    configuration->master_password = NULL;
354 355 356 357 358 359 360
    configuration->base_dir = (char *)xmlCharStrdup (CONFIG_DEFAULT_BASE_DIR);
    configuration->log_dir = (char *)xmlCharStrdup (CONFIG_DEFAULT_LOG_DIR);
    configuration->webroot_dir = (char *)xmlCharStrdup (CONFIG_DEFAULT_WEBROOT_DIR);
    configuration->adminroot_dir = (char *)xmlCharStrdup (CONFIG_DEFAULT_ADMINROOT_DIR);
    configuration->playlist_log = (char *)xmlCharStrdup (CONFIG_DEFAULT_PLAYLIST_LOG);
    configuration->access_log = (char *)xmlCharStrdup (CONFIG_DEFAULT_ACCESS_LOG);
    configuration->error_log = (char *)xmlCharStrdup (CONFIG_DEFAULT_ERROR_LOG);
361 362 363
    configuration->loglevel = CONFIG_DEFAULT_LOG_LEVEL;
    configuration->chroot = CONFIG_DEFAULT_CHROOT;
    configuration->chuid = CONFIG_DEFAULT_CHUID;
364 365
    configuration->user = NULL;
    configuration->group = NULL;
366
    configuration->num_yp_directories = 0;
367
    configuration->relay_username = (char *)xmlCharStrdup (CONFIG_DEFAULT_MASTER_USERNAME);
368
    configuration->relay_password = NULL;
Karl Heyes's avatar
Karl Heyes committed
369
    /* default to a typical prebuffer size used by clients */
370
    configuration->burst_size = CONFIG_DEFAULT_BURST_SIZE;
371 372 373 374 375 376 377 378 379 380 381
}

static void _parse_root(xmlDocPtr doc, xmlNodePtr node, 
        ice_config_t *configuration)
{
    char *tmp;

    do {
        if (node == NULL) break;
        if (xmlIsBlankNode(node)) continue;

382
        if (xmlStrcmp (node->name, XMLSTR("location")) == 0) {
383
            if (configuration->location) xmlFree(configuration->location);
384
            configuration->location = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
385
        } else if (xmlStrcmp (node->name, XMLSTR("admin")) == 0) {
386
            if (configuration->admin) xmlFree(configuration->admin);
387
            configuration->admin = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
388
        } else if (xmlStrcmp (node->name, XMLSTR("server-id")) == 0) {
389 390
            xmlFree (configuration->server_id);
            configuration->server_id = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
391
        } else if(xmlStrcmp (node->name, XMLSTR("authentication")) == 0) {
392
            _parse_authentication(doc, node->xmlChildrenNode, configuration);
393
        } else if (xmlStrcmp (node->name, XMLSTR("source-password")) == 0) {
394 395
            /* TODO: This is the backwards-compatibility location */
            char *mount, *pass;
396
            if ((mount = (char *)xmlGetProp(node, XMLSTR("mount"))) != NULL) {
397 398 399 400
                pass = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
                /* FIXME: This is a placeholder for per-mount passwords */
            }
            else {
401
                if (configuration->source_password) xmlFree(configuration->source_password);
402 403
                configuration->source_password = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            }
404
        } else if (xmlStrcmp (node->name, XMLSTR("icelogin")) == 0) {
405 406 407
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            configuration->ice_login = atoi(tmp);
            if (tmp) xmlFree(tmp);
408
        } else if (xmlStrcmp (node->name, XMLSTR("fileserve")) == 0) {
409 410 411
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            configuration->fileserve = atoi(tmp);
            if (tmp) xmlFree(tmp);
412
        } else if (xmlStrcmp (node->name, XMLSTR("relays-on-demand")) == 0) {
413 414 415
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            configuration->on_demand = atoi(tmp);
            if (tmp) xmlFree(tmp);
416
        } else if (xmlStrcmp (node->name, XMLSTR("hostname")) == 0) {
417
            if (configuration->hostname) xmlFree(configuration->hostname);
418
            configuration->hostname = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
419
        } else if (xmlStrcmp (node->name, XMLSTR("mime-types")) == 0) {
420 421
            if (configuration->mimetypes_fn) xmlFree(configuration->mimetypes_fn);
            configuration->mimetypes_fn = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
422
        } else if (xmlStrcmp (node->name, XMLSTR("listen-socket")) == 0) {
423
            _parse_listen_socket(doc, node->xmlChildrenNode, configuration);
424
        } else if (xmlStrcmp (node->name, XMLSTR("port")) == 0) {
425 426 427 428
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            configuration->port = atoi(tmp);
            configuration->listeners[0].port = atoi(tmp);
            if (tmp) xmlFree(tmp);
429
        } else if (xmlStrcmp (node->name, XMLSTR("bind-address")) == 0) {
430 431 432
            if (configuration->listeners[0].bind_address) 
                xmlFree(configuration->listeners[0].bind_address);
            configuration->listeners[0].bind_address = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
433
        } else if (xmlStrcmp (node->name, XMLSTR("master-server")) == 0) {
434 435
            if (configuration->master_server) xmlFree(configuration->master_server);
            configuration->master_server = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
436
        } else if (xmlStrcmp (node->name, XMLSTR("master-username")) == 0) {
437 438
            if (configuration->master_username) xmlFree(configuration->master_username);
            configuration->master_username = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
439
        } else if (xmlStrcmp (node->name, XMLSTR("master-password")) == 0) {
440 441
            if (configuration->master_password) xmlFree(configuration->master_password);
            configuration->master_password = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
442
        } else if (xmlStrcmp (node->name, XMLSTR("master-server-port")) == 0) {
443 444
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            configuration->master_server_port = atoi(tmp);
445
            xmlFree (tmp);
446
        } else if (xmlStrcmp (node->name, XMLSTR("master-update-interval")) == 0) {
447 448
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            configuration->master_update_interval = atoi(tmp);
449
            xmlFree (tmp);
450
        } else if (xmlStrcmp (node->name, XMLSTR("shoutcast-mount")) == 0) {
451
            if (configuration->shoutcast_mount) xmlFree(configuration->shoutcast_mount);
452
            configuration->shoutcast_mount = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
453
        } else if (xmlStrcmp (node->name, XMLSTR("limits")) == 0) {
454
            _parse_limits(doc, node->xmlChildrenNode, configuration);
455
        } else if (xmlStrcmp (node->name, XMLSTR("relay")) == 0) {
456
            _parse_relay(doc, node->xmlChildrenNode, configuration);
457
        } else if (xmlStrcmp (node->name, XMLSTR("mount")) == 0) {
458
            _parse_mount(doc, node->xmlChildrenNode, configuration);
459
        } else if (xmlStrcmp (node->name, XMLSTR("directory")) == 0) {
460
            _parse_directory(doc, node->xmlChildrenNode, configuration);
461
        } else if (xmlStrcmp (node->name, XMLSTR("paths")) == 0) {
462
            _parse_paths(doc, node->xmlChildrenNode, configuration);
463
        } else if (xmlStrcmp (node->name, XMLSTR("logging")) == 0) {
464
            _parse_logging(doc, node->xmlChildrenNode, configuration);
465
        } else if (xmlStrcmp (node->name, XMLSTR("security")) == 0) {
466 467 468 469 470 471 472 473 474 475 476 477 478 479
            _parse_security(doc, node->xmlChildrenNode, configuration);
        }
    } while ((node = node->next));
}

static void _parse_limits(xmlDocPtr doc, xmlNodePtr node, 
        ice_config_t *configuration)
{
    char *tmp;

    do {
        if (node == NULL) break;
        if (xmlIsBlankNode(node)) continue;

480
        if (xmlStrcmp (node->name, XMLSTR("clients")) == 0) {
481 482 483
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            configuration->client_limit = atoi(tmp);
            if (tmp) xmlFree(tmp);
484
        } else if (xmlStrcmp (node->name, XMLSTR("sources")) == 0) {
485 486 487
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            configuration->source_limit = atoi(tmp);
            if (tmp) xmlFree(tmp);
488
        } else if (xmlStrcmp (node->name, XMLSTR("queue-size")) == 0) {
489 490 491
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            configuration->queue_size_limit = atoi(tmp);
            if (tmp) xmlFree(tmp);
492
        } else if (xmlStrcmp (node->name, XMLSTR("threadpool")) == 0) {
493 494 495
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            configuration->threadpool_size = atoi(tmp);
            if (tmp) xmlFree(tmp);
496
        } else if (xmlStrcmp (node->name, XMLSTR("client-timeout")) == 0) {
497 498 499
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            configuration->client_timeout = atoi(tmp);
            if (tmp) xmlFree(tmp);
500
        } else if (xmlStrcmp (node->name, XMLSTR("header-timeout")) == 0) {
501 502 503
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            configuration->header_timeout = atoi(tmp);
            if (tmp) xmlFree(tmp);
504
        } else if (xmlStrcmp (node->name, XMLSTR("source-timeout")) == 0) {
505 506 507
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            configuration->source_timeout = atoi(tmp);
            if (tmp) xmlFree(tmp);
508
        } else if (xmlStrcmp (node->name, XMLSTR("burst-on-connect")) == 0) {
509
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
Karl Heyes's avatar
Karl Heyes committed
510 511 512
            if (atoi(tmp) == 0)
                configuration->burst_size = 0;
            if (tmp) xmlFree(tmp);
513
        } else if (xmlStrcmp (node->name, XMLSTR("burst-size")) == 0) {
Karl Heyes's avatar
Karl Heyes committed
514 515
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            configuration->burst_size = atoi(tmp);
516
            if (tmp) xmlFree(tmp);
517 518 519 520 521 522 523 524 525 526 527 528
        }
    } while ((node = node->next));
}

static void _parse_mount(xmlDocPtr doc, xmlNodePtr node, 
        ice_config_t *configuration)
{
    char *tmp;
    mount_proxy *mount = calloc(1, sizeof(mount_proxy));
    mount_proxy *current = configuration->mounts;
    mount_proxy *last=NULL;
    
Karl Heyes's avatar
Karl Heyes committed
529
    /* default <mount> settings */
530
    mount->max_listeners = -1;
Karl Heyes's avatar
Karl Heyes committed
531
    mount->burst_size = -1;
532
    mount->mp3_meta_interval = -1;
533
    mount->yp_public = -1;
534 535 536 537 538 539
    mount->next = NULL;

    do {
        if (node == NULL) break;
        if (xmlIsBlankNode(node)) continue;

540 541
        if (xmlStrcmp (node->name, XMLSTR("mount-name")) == 0) {
            mount->mountname = (char *)xmlNodeListGetString (doc, node->xmlChildrenNode, 1);
542
        }
543
        else if (xmlStrcmp (node->name, XMLSTR("username")) == 0) {
544 545 546
            mount->username = (char *)xmlNodeListGetString(
                    doc, node->xmlChildrenNode, 1);
        }
547
        else if (xmlStrcmp (node->name, XMLSTR("password")) == 0) {
548 549 550
            mount->password = (char *)xmlNodeListGetString(
                    doc, node->xmlChildrenNode, 1);
        }
551
        else if (xmlStrcmp (node->name, XMLSTR("dump-file")) == 0) {
552 553
            mount->dumpfile = (char *)xmlNodeListGetString(
                    doc, node->xmlChildrenNode, 1);
Karl Heyes's avatar
Karl Heyes committed
554
        }
555
        else if (xmlStrcmp (node->name, XMLSTR("intro")) == 0) {
Karl Heyes's avatar
Karl Heyes committed
556 557
            mount->intro_filename = (char *)xmlNodeListGetString(
                    doc, node->xmlChildrenNode, 1);
558
        }
559
        else if (xmlStrcmp (node->name, XMLSTR("fallback-mount")) == 0) {
560 561 562
            mount->fallback_mount = (char *)xmlNodeListGetString(
                    doc, node->xmlChildrenNode, 1);
        }
563
        else if (xmlStrcmp (node->name, XMLSTR("fallback-when-full")) == 0) {
564 565 566 567
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            mount->fallback_when_full = atoi(tmp);
            if(tmp) xmlFree(tmp);
        }
568
        else if (xmlStrcmp (node->name, XMLSTR("max-listeners")) == 0) {
569 570 571 572
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            mount->max_listeners = atoi(tmp);
            if(tmp) xmlFree(tmp);
        }
573
        else if (xmlStrcmp (node->name, XMLSTR("charset")) == 0) {
574 575 576
            mount->charset = (char *)xmlNodeListGetString(doc,
                    node->xmlChildrenNode, 1);
        }
577
        else if (xmlStrcmp (node->name, XMLSTR("mp3-metadata-interval")) == 0) {
578 579 580 581
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            mount->mp3_meta_interval = atoi(tmp);
            if(tmp) xmlFree(tmp);
        }
582
        else if (xmlStrcmp (node->name, XMLSTR("fallback-override")) == 0) {
Michael Smith's avatar
Michael Smith committed
583 584 585 586
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            mount->fallback_override = atoi(tmp);
            if(tmp) xmlFree(tmp);
        }
587
        else if (xmlStrcmp (node->name, XMLSTR("no-mount")) == 0) {
Michael Smith's avatar
Michael Smith committed
588 589 590 591
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            mount->no_mount = atoi(tmp);
            if(tmp) xmlFree(tmp);
        }
592
        else if (xmlStrcmp (node->name, XMLSTR("no-yp")) == 0) {
Karl Heyes's avatar
Karl Heyes committed
593
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
594
            mount->yp_public = atoi(tmp) == 0 ? -1 : 0;
Karl Heyes's avatar
Karl Heyes committed
595 596
            if(tmp) xmlFree(tmp);
        }
597
        else if (xmlStrcmp (node->name, XMLSTR("hidden")) == 0) {
598 599 600 601
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            mount->hidden = atoi(tmp);
            if(tmp) xmlFree(tmp);
        }
602
        else if (xmlStrcmp (node->name, XMLSTR("authentication")) == 0) {
603
            mount->auth = auth_get_authenticator (node);
Michael Smith's avatar
Michael Smith committed
604
        }
605
        else if (xmlStrcmp (node->name, XMLSTR("on-connect")) == 0) {
606 607 608
            mount->on_connect = (char *)xmlNodeListGetString(
                    doc, node->xmlChildrenNode, 1);
        }
609
        else if (xmlStrcmp (node->name, XMLSTR("on-disconnect")) == 0) {
610 611 612
            mount->on_disconnect = (char *)xmlNodeListGetString(
                    doc, node->xmlChildrenNode, 1);
        }
613
        else if (xmlStrcmp (node->name, XMLSTR("max-listener-duration")) == 0) {
614 615 616 617
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            mount->max_listener_duration = atoi(tmp);
            if(tmp) xmlFree(tmp);
        }
618
        else if (xmlStrcmp (node->name, XMLSTR("queue-size")) == 0) {
619 620 621 622
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            mount->queue_size_limit = atoi (tmp);
            if(tmp) xmlFree(tmp);
        }
623
        else if (xmlStrcmp (node->name, XMLSTR("source-timeout")) == 0) {
624 625 626 627 628 629
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            if (tmp)
            {
                mount->source_timeout = atoi (tmp);
                xmlFree(tmp);
            }
630
        } else if (xmlStrcmp (node->name, XMLSTR("burst-size")) == 0) {
Karl Heyes's avatar
Karl Heyes committed
631 632 633
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            mount->burst_size = atoi(tmp);
            if (tmp) xmlFree(tmp);
634
        } else if (xmlStrcmp (node->name, XMLSTR("cluster-password")) == 0) {
635 636
            mount->cluster_password = (char *)xmlNodeListGetString(
                    doc, node->xmlChildrenNode, 1);
637
        } else if (xmlStrcmp (node->name, XMLSTR("stream-name")) == 0) {
638 639
            mount->stream_name = (char *)xmlNodeListGetString(
                    doc, node->xmlChildrenNode, 1);
640
        } else if (xmlStrcmp (node->name, XMLSTR("stream-description")) == 0) {
641 642
            mount->stream_description = (char *)xmlNodeListGetString(
                    doc, node->xmlChildrenNode, 1);
643
        } else if (xmlStrcmp (node->name, XMLSTR("stream-url")) == 0) {
644 645
            mount->stream_url = (char *)xmlNodeListGetString(
                    doc, node->xmlChildrenNode, 1);
646
        } else if (xmlStrcmp (node->name, XMLSTR("genre")) == 0) {
647 648
            mount->stream_genre = (char *)xmlNodeListGetString(
                    doc, node->xmlChildrenNode, 1);
649
        } else if (xmlStrcmp (node->name, XMLSTR("bitrate")) == 0) {
650 651
            mount->bitrate = (char *)xmlNodeListGetString(
                    doc, node->xmlChildrenNode, 1);
652
        } else if (xmlStrcmp (node->name, XMLSTR("public")) == 0) {
653 654 655
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            mount->yp_public = atoi (tmp);
            if(tmp) xmlFree(tmp);
656
        } else if (xmlStrcmp (node->name, XMLSTR("type")) == 0) {
657 658
            mount->type = (char *)xmlNodeListGetString(
                    doc, node->xmlChildrenNode, 1);
659
        } else if (xmlStrcmp (node->name, XMLSTR("subtype")) == 0) {
660 661
            mount->subtype = (char *)xmlNodeListGetString(
                    doc, node->xmlChildrenNode, 1);
662
        }
663
    } while ((node = node->next));
664 665 666 667 668 669 670

    /* make sure we have at least the mountpoint name */
    if (mount->mountname == NULL)
    {
        config_clear_mount (mount);
        return;
    }
671
    if (mount->auth)
672
        mount->auth->mount = strdup ((char *)mount->mountname);
673 674 675 676 677 678 679 680 681
    while(current) {
        last = current;
        current = current->next;
    }

    if(last)
        last->next = mount;
    else
        configuration->mounts = mount;
682 683
}

684

685 686 687 688 689 690 691
static void _parse_relay(xmlDocPtr doc, xmlNodePtr node,
        ice_config_t *configuration)
{
    char *tmp;
    relay_server *relay = calloc(1, sizeof(relay_server));
    relay_server *current = configuration->relay;
    relay_server *last=NULL;
692

693 694 695 696 697 698 699 700 701 702 703
    while(current) {
        last = current;
        current = current->next;
    }

    if(last)
        last->next = relay;
    else
        configuration->relay = relay;

    relay->next = NULL;
704
    relay->mp3metadata = 1;
705
    relay->on_demand = configuration->on_demand;
706 707 708 709 710

    do {
        if (node == NULL) break;
        if (xmlIsBlankNode(node)) continue;

711
        if (xmlStrcmp (node->name, XMLSTR("server")) == 0) {
712 713 714
            relay->server = (char *)xmlNodeListGetString(
                    doc, node->xmlChildrenNode, 1);
        }
715
        else if (xmlStrcmp (node->name, XMLSTR("port")) == 0) {
716 717 718 719
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            relay->port = atoi(tmp);
            if(tmp) xmlFree(tmp);
        }
720
        else if (xmlStrcmp (node->name, XMLSTR("mount")) == 0) {
721 722 723
            relay->mount = (char *)xmlNodeListGetString(
                    doc, node->xmlChildrenNode, 1);
        }
724
        else if (xmlStrcmp (node->name, XMLSTR("local-mount")) == 0) {
725 726 727
            relay->localmount = (char *)xmlNodeListGetString(
                    doc, node->xmlChildrenNode, 1);
        }
728
        else if (xmlStrcmp (node->name, XMLSTR("relay-shoutcast-metadata")) == 0) {
729 730 731 732
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            relay->mp3metadata = atoi(tmp);
            if(tmp) xmlFree(tmp);
        }
733
        else if (xmlStrcmp (node->name, XMLSTR("username")) == 0) {
734 735 736
            relay->username = (char *)xmlNodeListGetString(doc,
                    node->xmlChildrenNode, 1);
        }
737
        else if (xmlStrcmp (node->name, XMLSTR("password")) == 0) {
738 739 740
            relay->password = (char *)xmlNodeListGetString(doc,
                    node->xmlChildrenNode, 1);
        }
741
        else if (xmlStrcmp (node->name, XMLSTR("on-demand")) == 0) {
742 743 744 745
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            relay->on_demand = atoi(tmp);
            if (tmp) xmlFree(tmp);
        }
746
    } while ((node = node->next));
Karl Heyes's avatar
Karl Heyes committed
747
    if (relay->localmount == NULL)
748
        relay->localmount = (char *)xmlStrdup (XMLSTR(relay->mount));
749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764
}

static void _parse_listen_socket(xmlDocPtr doc, xmlNodePtr node,
        ice_config_t *configuration)
{
    listener_t *listener = NULL;
    int i;
    char *tmp;

    for(i=0; i < MAX_LISTEN_SOCKETS; i++) {
        if(configuration->listeners[i].port <= 0) {
            listener = &(configuration->listeners[i]);
            break;
        }
    }

765 766
    if (listener == NULL)
        return;
767 768 769 770
    do {
        if (node == NULL) break;
        if (xmlIsBlankNode(node)) continue;

771
        if (xmlStrcmp (node->name, XMLSTR("port")) == 0) {
772 773 774 775 776 777
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            if(configuration->port == 0)
                configuration->port = atoi(tmp);
            listener->port = atoi(tmp);
            if(tmp) xmlFree(tmp);
        }
778
        else if (xmlStrcmp (node->name, XMLSTR("ssl")) == 0) {
779 780 781 782
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            listener->ssl = atoi(tmp);
            if(tmp) xmlFree(tmp);
        }
783
        else if (xmlStrcmp (node->name, XMLSTR("shoutcast-compat")) == 0) {
784 785 786 787
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            listener->shoutcast_compat = atoi(tmp);
            if(tmp) xmlFree(tmp);
        }
788
        else if (xmlStrcmp (node->name, XMLSTR("bind-address")) == 0) {
789 790 791 792 793 794 795 796 797 798 799 800 801
            listener->bind_address = (char *)xmlNodeListGetString(doc, 
                    node->xmlChildrenNode, 1);
        }
    } while ((node = node->next));
}

static void _parse_authentication(xmlDocPtr doc, xmlNodePtr node,
        ice_config_t *configuration)
{
    do {
        if (node == NULL) break;
        if (xmlIsBlankNode(node)) continue;

802
        if (xmlStrcmp (node->name, XMLSTR("source-password")) == 0) {
803
            char *mount, *pass;
804
            if ((mount = (char *)xmlGetProp(node, XMLSTR("mount"))) != NULL) {
805 806 807 808
                pass = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
                /* FIXME: This is a placeholder for per-mount passwords */
            }
            else {
809
                if (configuration->source_password)
810 811 812 813
                    xmlFree(configuration->source_password);
                configuration->source_password = 
                    (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            }
814
        } else if (xmlStrcmp (node->name, XMLSTR("admin-password")) == 0) {
815 816 817 818
            if(configuration->admin_password)
                xmlFree(configuration->admin_password);
            configuration->admin_password =
                (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
819
        } else if (xmlStrcmp (node->name, XMLSTR("admin-user")) == 0) {
820 821 822 823
            if(configuration->admin_username)
                xmlFree(configuration->admin_username);
            configuration->admin_username =
                (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
824
        } else if (xmlStrcmp (node->name, XMLSTR("relay-password")) == 0) {
825 826 827 828
            if(configuration->relay_password)
                xmlFree(configuration->relay_password);
            configuration->relay_password =
                (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
829
        } else if (xmlStrcmp (node->name, XMLSTR("relay-user")) == 0) {
830 831 832 833
            if(configuration->relay_username)
                xmlFree(configuration->relay_username);
            configuration->relay_username =
                (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850
        }
    } while ((node = node->next));
}

static void _parse_directory(xmlDocPtr doc, xmlNodePtr node,
        ice_config_t *configuration)
{
    char *tmp;

    if (configuration->num_yp_directories >= MAX_YP_DIRECTORIES) {
        ERROR0("Maximum number of yp directories exceeded!");
        return;
    }
    do {
        if (node == NULL) break;
        if (xmlIsBlankNode(node)) continue;

851
        if (xmlStrcmp (node->name, XMLSTR("yp-url")) == 0) {
852 853 854 855
            if (configuration->yp_url[configuration->num_yp_directories]) 
                xmlFree(configuration->yp_url[configuration->num_yp_directories]);
            configuration->yp_url[configuration->num_yp_directories] = 
                (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
856
        } else if (xmlStrcmp (node->name, XMLSTR("yp-url-timeout")) == 0) {
857 858 859
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            configuration->yp_url_timeout[configuration->num_yp_directories] = 
                atoi(tmp);
Karl Heyes's avatar
Karl Heyes committed
860
            if (tmp) xmlFree(tmp);
861
        } else if (xmlStrcmp (node->name, XMLSTR("server")) == 0) {
862
            _add_server(doc, node->xmlChildrenNode, configuration);
863
        } else if (xmlStrcmp (node->name, XMLSTR("touch-interval")) == 0) {
864
            tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
865 866
            configuration->yp_touch_interval[configuration->num_yp_directories] =
                atoi(tmp);
867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882
            if (tmp) xmlFree(tmp);
        }
    } while ((node = node->next));
    configuration->num_yp_directories++;
}

static void _parse_paths(xmlDocPtr doc, xmlNodePtr node,
        ice_config_t *configuration)
{
    char *temp;
    aliases *alias, *current, *last;

    do {
        if (node == NULL) break;
        if (xmlIsBlankNode(node)) continue;

883
        if (xmlStrcmp (node->name, XMLSTR("basedir")) == 0) {
884
            if (configuration->base_dir) xmlFree(configuration->base_dir);
885
            configuration->base_dir = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
886
        } else if (xmlStrcmp (node->name, XMLSTR("logdir")) == 0) {
887
            if (configuration->log_dir) xmlFree(configuration->log_dir);
888
            configuration->log_dir = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
889
        } else if (xmlStrcmp (node->name, XMLSTR("pidfile")) == 0) {
890 891
            if (configuration->pidfile) xmlFree(configuration->pidfile);
            configuration->pidfile = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
892
        } else if (xmlStrcmp (node->name, XMLSTR("ssl-certificate")) == 0) {
893 894
            if (configuration->cert_file) xmlFree(configuration->cert_file);
            configuration->cert_file = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
895
        } else if (xmlStrcmp (node->name, XMLSTR("webroot")) == 0) {
896
            if (configuration->webroot_dir) xmlFree(configuration->webroot_dir);
897 898 899
            configuration->webroot_dir = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            if(configuration->webroot_dir[strlen(configuration->webroot_dir)-1] == '/')
                configuration->webroot_dir[strlen(configuration->webroot_dir)-1] = 0;
900
        } else if (xmlStrcmp (node->name, XMLSTR("adminroot")) == 0) {
901
            if (configuration->adminroot_dir) 
902 903 904 905
                xmlFree(configuration->adminroot_dir);
            configuration->adminroot_dir = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            if(configuration->adminroot_dir[strlen(configuration->adminroot_dir)-1] == '/')
                configuration->adminroot_dir[strlen(configuration->adminroot_dir)-1] = 0;
906
        } else if (xmlStrcmp (node->name, XMLSTR("alias")) == 0) {
907 908
            alias = malloc(sizeof(aliases));
            alias->next = NULL;
909
            alias->source = (char *)xmlGetProp(node, XMLSTR("source"));
910 911 912 913
            if(alias->source == NULL) {
                free(alias);
                continue;
            }
914
            alias->destination = (char *)xmlGetProp(node, XMLSTR("dest"));
915 916 917 918 919 920
            if(alias->destination == NULL) {
                xmlFree(alias->source);
                free(alias);
                continue;
            }
            temp = NULL;
921
            temp = (char *)xmlGetProp(node, XMLSTR("port"));
922 923 924 925 926 927
            if(temp != NULL) {
                alias->port = atoi(temp);
                xmlFree(temp);
            }
            else
                alias->port = -1;
928
            alias->bind_address = (char *)xmlGetProp(node, XMLSTR("bind-address"));
929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949
            current = configuration->aliases;
            last = NULL;
            while(current) {
                last = current;
                current = current->next;
            }
            if(last)
                last->next = alias;
            else
                configuration->aliases = alias;
        }
    } while ((node = node->next));
}

static void _parse_logging(xmlDocPtr doc, xmlNodePtr node,
        ice_config_t *configuration)
{
    do {
        if (node == NULL) break;
        if (xmlIsBlankNode(node)) continue;

950
        if (xmlStrcmp (node->name, XMLSTR("accesslog")) == 0) {
951
            if (configuration->access_log) xmlFree(configuration->access_log);
952
            configuration->access_log = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
953
        } else if (xmlStrcmp (node->name, XMLSTR("errorlog")) == 0) {
954
            if (configuration->error_log) xmlFree(configuration->error_log);
955
            configuration->error_log = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
956
        } else if (xmlStrcmp (node->name, XMLSTR("playlistlog")) == 0) {
957
            if (configuration->playlist_log) xmlFree(configuration->playlist_log);
958
            configuration->playlist_log = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
959
        } else if (xmlStrcmp (node->name, XMLSTR("logsize")) == 0) {
960 961 962
            char *tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
            configuration->logsize = atoi(tmp);
            if (tmp) xmlFree(tmp);
963
        } else if (xmlStrcmp (node->name, XMLSTR("loglevel")) == 0) {
964 965 966
           char *tmp = (char *)xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
           configuration->loglevel = atoi(tmp);
           if (tmp) xmlFree(tmp);
967
        } else if (xmlStrcmp (node->name, XMLSTR("logarchive")) == 0