mirror of
https://gitlab.xiph.org/xiph/ezstream.git
synced 2024-12-04 14:46:31 -05:00
304908bff4
program/script to get metadata from. Also include SIGUSR2 handling that triggers metadata updates from the external program mid-stream. git-svn-id: https://svn.xiph.org/trunk/ezstream@12693 0101bb08-14d6-0310-b084-bc0e0c8e3800
1103 lines
26 KiB
C
1103 lines
26 KiB
C
/*
|
|
* ezstream - source client for Icecast with external en-/decoder support
|
|
* Copyright (C) 2003, 2004, 2005, 2006 Ed Zaleski <oddsock@oddsock.org>
|
|
* Copyright (C) 2007 Moritz Grimm <gtgbr@gmx.net>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 2 as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#ifdef HAVE_SYS_TYPES_H
|
|
# include <sys/types.h>
|
|
#endif
|
|
#ifdef HAVE_SYS_STAT_H
|
|
# include <sys/stat.h>
|
|
#endif
|
|
#ifdef HAVE_SYS_TIME_H
|
|
# include <sys/time.h>
|
|
#endif
|
|
#include <ctype.h>
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#ifdef HAVE_LIBGEN_H
|
|
# include <libgen.h>
|
|
#endif
|
|
#include <limits.h>
|
|
#ifdef HAVE_PATHS_H
|
|
# include <paths.h>
|
|
#endif
|
|
#ifdef HAVE_SIGNAL_H
|
|
# include <signal.h>
|
|
#endif
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#ifdef WIN32
|
|
# include <io.h>
|
|
# include <windows.h>
|
|
#endif /* WIN32 */
|
|
#ifdef HAVE_UNISTD_H
|
|
# include <unistd.h>
|
|
#endif
|
|
#include <shout/shout.h>
|
|
|
|
#include "compat.h"
|
|
#include "configfile.h"
|
|
#ifndef HAVE_GETOPT
|
|
# include "getopt.h"
|
|
#endif
|
|
#include "metadata.h"
|
|
#include "playlist.h"
|
|
#include "strfctns.h"
|
|
#include "util.h"
|
|
|
|
#define STREAM_DONE 0
|
|
#define STREAM_CONT 1
|
|
#define STREAM_SKIP 2
|
|
#define STREAM_SERVERR 3
|
|
#define STREAM_UPDMDATA 4
|
|
|
|
#ifdef HAVE___PROGNAME
|
|
extern char *__progname;
|
|
#else
|
|
char *__progname;
|
|
#endif /* HAVE___PROGNAME */
|
|
|
|
int qFlag;
|
|
int vFlag;
|
|
int metadataFromProgram;
|
|
|
|
EZCONFIG *pezConfig = NULL;
|
|
static const char *blankString = "";
|
|
playlist_t *playlist = NULL;
|
|
int playlistMode = 0;
|
|
|
|
#ifdef HAVE_SIGNALS
|
|
const int ezstream_signals[] = { SIGHUP, SIGUSR1, SIGUSR2 };
|
|
|
|
volatile sig_atomic_t rereadPlaylist = 0;
|
|
volatile sig_atomic_t rereadPlaylist_notify = 0;
|
|
volatile sig_atomic_t skipTrack = 0;
|
|
volatile sig_atomic_t queryMetadata = 0;
|
|
#else
|
|
int rereadPlaylist = 0;
|
|
int rereadPlaylist_notify = 0;
|
|
int skipTrack = 0;
|
|
int queryMetadata = 0;
|
|
#endif /* HAVE_SIGNALS */
|
|
|
|
typedef struct tag_ID3Tag {
|
|
char tag[3];
|
|
char trackName[30];
|
|
char artistName[30];
|
|
char albumName[30];
|
|
char year[3];
|
|
char comment[30];
|
|
char genre;
|
|
} ID3Tag;
|
|
|
|
int urlParse(const char *, char **, int *, char **);
|
|
void replaceString(const char *, char *, size_t, const char *, const char *);
|
|
char * buildCommandString(const char *, const char *, const char *);
|
|
char * processMetadata(shout_t *, const char *);
|
|
FILE * openResource(shout_t *, const char *, int *, char **, int *);
|
|
int reconnectServer(shout_t *, int);
|
|
int sendStream(shout_t *, FILE *, const char *, int, void *);
|
|
int streamFile(shout_t *, const char *);
|
|
int streamPlaylist(shout_t *, const char *);
|
|
char * getProgname(const char *);
|
|
void usage(void);
|
|
void usageHelp(void);
|
|
|
|
#ifdef HAVE_SIGNALS
|
|
void sig_handler(int);
|
|
|
|
void
|
|
sig_handler(int sig)
|
|
{
|
|
switch (sig) {
|
|
case SIGHUP:
|
|
rereadPlaylist = 1;
|
|
rereadPlaylist_notify = 1;
|
|
break;
|
|
case SIGUSR1:
|
|
skipTrack = 1;
|
|
break;
|
|
case SIGUSR2:
|
|
queryMetadata = 1;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
#endif /* HAVE_SIGNALS */
|
|
|
|
int
|
|
urlParse(const char *url, char **hostname, int *port, char **mountname)
|
|
{
|
|
char *p1, *p2, *p3;
|
|
char tmpPort[6] = "";
|
|
size_t hostsiz, mountsiz;
|
|
const char *errstr;
|
|
|
|
if (hostname == NULL || port == NULL || mountname == NULL) {
|
|
printf("%s: urlParse(): Internal error: Bad arguments\n",
|
|
__progname);
|
|
exit(1);
|
|
}
|
|
|
|
if (strncmp(url, "http://", strlen("http://")) != 0) {
|
|
printf("%s: Error: Invalid <url>: Not an HTTP address\n",
|
|
__progname);
|
|
return (0);
|
|
}
|
|
|
|
p1 = (char *)(url) + strlen("http://");
|
|
p2 = strchr(p1, ':');
|
|
if (p2 == NULL) {
|
|
printf("%s: Error: Invalid <url>: Missing port\n",
|
|
__progname);
|
|
return (0);
|
|
}
|
|
hostsiz = (p2 - p1) + 1;
|
|
*hostname = xmalloc(hostsiz);
|
|
strlcpy(*hostname, p1, hostsiz);
|
|
|
|
p2++;
|
|
p3 = strchr(p2, '/');
|
|
if (p3 == NULL || p3 - p2 >= (int)sizeof(tmpPort)) {
|
|
printf("%s: Error: Invalid <url>: Missing mountpoint or too long port number\n",
|
|
__progname);
|
|
xfree(*hostname);
|
|
return (0);
|
|
}
|
|
|
|
strlcpy(tmpPort, p2, (p3 - p2) + 1);
|
|
*port = (int)strtonum(tmpPort, 1, 65535, &errstr);
|
|
if (errstr) {
|
|
printf("%s: Error: Invalid <url>: Port '%s' is %s\n",
|
|
__progname, tmpPort, errstr);
|
|
xfree(*hostname);
|
|
return (0);
|
|
}
|
|
|
|
mountsiz = strlen(p3) + 1;
|
|
*mountname = xmalloc(mountsiz);
|
|
strlcpy(*mountname, p3, mountsiz);
|
|
|
|
return (1);
|
|
}
|
|
|
|
void
|
|
replaceString(const char *source, char *dest, size_t size,
|
|
const char *from, const char *to)
|
|
{
|
|
char *p1 = (char *)source;
|
|
char *p2;
|
|
|
|
p2 = strstr(p1, from);
|
|
if (p2 != NULL) {
|
|
if ((unsigned int)(p2 - p1) >= size) {
|
|
printf("%s: replaceString(): Internal error: p2 - p1 >= size\n",
|
|
__progname);
|
|
abort();
|
|
}
|
|
strncat(dest, p1, p2 - p1);
|
|
strlcat(dest, to, size);
|
|
p1 = p2 + strlen(from);
|
|
}
|
|
strlcat(dest, p1, size);
|
|
}
|
|
|
|
char *
|
|
buildCommandString(const char *extension, const char *fileName,
|
|
const char *metadata)
|
|
{
|
|
char *commandString = NULL;
|
|
size_t commandStringLen = 0;
|
|
char *encoder = NULL;
|
|
char *decoder = NULL;
|
|
char *newDecoder = NULL;
|
|
size_t newDecoderLen = 0;
|
|
char *newEncoder = NULL;
|
|
size_t newEncoderLen = 0;
|
|
|
|
decoder = xstrdup(getFormatDecoder(extension));
|
|
if (strlen(decoder) == 0) {
|
|
printf("%s: Unknown extension '%s', cannot decode '%s'\n",
|
|
__progname, extension, fileName);
|
|
xfree(decoder);
|
|
return (NULL);
|
|
}
|
|
newDecoderLen = strlen(decoder) + strlen(fileName) + 1;
|
|
newDecoder = xcalloc(1, newDecoderLen);
|
|
replaceString(decoder, newDecoder, newDecoderLen, TRACK_PLACEHOLDER,
|
|
fileName);
|
|
if (strstr(decoder, METADATA_PLACEHOLDER) != NULL) {
|
|
size_t tmpLen = strlen(newDecoder) + strlen(metadata) + 1;
|
|
char *tmpStr = xcalloc(1, tmpLen);
|
|
replaceString(newDecoder, tmpStr, tmpLen, METADATA_PLACEHOLDER,
|
|
metadata);
|
|
xfree(newDecoder);
|
|
newDecoder = tmpStr;
|
|
}
|
|
|
|
encoder = xstrdup(getFormatEncoder(pezConfig->format));
|
|
if (strlen(encoder) == 0) {
|
|
if (vFlag)
|
|
printf("%s: Passing through%s%s data from the decoder\n",
|
|
__progname,
|
|
(strcmp(pezConfig->format, THEORA_FORMAT) != 0) ? " (unsupported) " : " ",
|
|
pezConfig->format);
|
|
commandStringLen = strlen(newDecoder) + 1;
|
|
commandString = xcalloc(1, commandStringLen);
|
|
strlcpy(commandString, newDecoder, commandStringLen);
|
|
xfree(decoder);
|
|
xfree(encoder);
|
|
xfree(newDecoder);
|
|
return (commandString);
|
|
}
|
|
|
|
newEncoderLen = strlen(encoder) + strlen(metadata) + 1;
|
|
newEncoder = xcalloc(1, newEncoderLen);
|
|
replaceString(encoder, newEncoder, newEncoderLen, METADATA_PLACEHOLDER,
|
|
metadata);
|
|
|
|
commandStringLen = strlen(newDecoder) + strlen(" | ") +
|
|
strlen(newEncoder) + 1;
|
|
commandString = xcalloc(1, commandStringLen);
|
|
snprintf(commandString, commandStringLen, "%s | %s", newDecoder,
|
|
newEncoder);
|
|
|
|
xfree(decoder);
|
|
xfree(encoder);
|
|
xfree(newDecoder);
|
|
xfree(newEncoder);
|
|
|
|
return (commandString);
|
|
}
|
|
|
|
char *
|
|
processMetadata(shout_t *shout, const char *fileName)
|
|
{
|
|
char *songInfo = NULL;
|
|
shout_metadata_t *shout_mdata = NULL;
|
|
metadata_t *mdata = NULL;
|
|
|
|
if (metadataFromProgram) {
|
|
if ((mdata = metadata_program(fileName)) == NULL)
|
|
return (NULL);
|
|
|
|
if (!metadata_program_update(mdata, METADATA_STRING)) {
|
|
metadata_free(&mdata);
|
|
songInfo = xstrdup(blankString);
|
|
return (songInfo);
|
|
}
|
|
} else {
|
|
if ((mdata = metadata_file(fileName)) == NULL)
|
|
return (NULL);
|
|
|
|
if (!metadata_file_update(mdata)) {
|
|
metadata_free(&mdata);
|
|
songInfo = xstrdup(blankString);
|
|
return (songInfo);
|
|
}
|
|
}
|
|
|
|
songInfo = xstrdup(metadata_get_string(mdata));
|
|
metadata_free(&mdata);
|
|
|
|
if ((shout_mdata = shout_metadata_new()) == NULL) {
|
|
printf("%s: shout_metadata_new(): %s\n", __progname,
|
|
strerror(ENOMEM));
|
|
exit(1);
|
|
}
|
|
shout_metadata_add(shout_mdata, "song", songInfo);
|
|
shout_set_metadata(shout, shout_mdata);
|
|
shout_metadata_free(shout_mdata);
|
|
|
|
return (songInfo);
|
|
}
|
|
|
|
FILE *
|
|
openResource(shout_t *shout, const char *fileName, int *popenFlag,
|
|
char **metaCopy, int *isStdin)
|
|
{
|
|
FILE *filep = NULL;
|
|
char extension[25];
|
|
char *p = NULL;
|
|
char *pMetadata = NULL;
|
|
char *pCommandString = NULL;
|
|
|
|
if (strcmp(fileName, "stdin") == 0) {
|
|
if (metadataFromProgram &&
|
|
processMetadata(shout, pezConfig->metadataProgram) == NULL)
|
|
return (filep);
|
|
|
|
if (vFlag)
|
|
printf("%s: Reading from standard input\n",
|
|
__progname);
|
|
if (isStdin != NULL)
|
|
*isStdin = 1;
|
|
#ifdef WIN32
|
|
_setmode(_fileno(stdin), _O_BINARY);
|
|
#endif
|
|
filep = stdin;
|
|
return (filep);
|
|
}
|
|
if (isStdin != NULL)
|
|
*isStdin = 0;
|
|
|
|
extension[0] = '\0';
|
|
p = strrchr(fileName, '.');
|
|
if (p != NULL)
|
|
strlcpy(extension, p, sizeof(extension));
|
|
for (p = extension; *p != '\0'; p++)
|
|
*p = tolower((int)*p);
|
|
|
|
if (strlen(extension) == 0) {
|
|
printf("%s: Error: Cannot determine file type of '%s'\n",
|
|
__progname, fileName);
|
|
return (filep);
|
|
}
|
|
|
|
if (metadataFromProgram)
|
|
pMetadata = processMetadata(shout, pezConfig->metadataProgram);
|
|
else
|
|
pMetadata = processMetadata(shout, fileName);
|
|
if (pMetadata == NULL)
|
|
return (filep);
|
|
if (metaCopy != NULL)
|
|
*metaCopy = xstrdup(pMetadata);
|
|
|
|
*popenFlag = 0;
|
|
if (pezConfig->reencode) {
|
|
int stderr_fd = dup(fileno(stderr));
|
|
|
|
pCommandString = buildCommandString(extension, fileName, pMetadata);
|
|
if (vFlag > 1)
|
|
printf("%s: Running command `%s`\n", __progname,
|
|
pCommandString);
|
|
|
|
if (qFlag) {
|
|
int fd;
|
|
|
|
if ((fd = open(_PATH_DEVNULL, O_RDWR, 0)) == -1) {
|
|
printf("%s: Cannot open %s for redirecting STDERR output: %s\n",
|
|
__progname, _PATH_DEVNULL, strerror(errno));
|
|
exit(1);
|
|
}
|
|
|
|
dup2(fd, fileno(stderr));
|
|
if (fd > 2)
|
|
close(fd);
|
|
}
|
|
|
|
fflush(NULL);
|
|
errno = 0;
|
|
if ((filep = popen(pCommandString, "r")) == NULL) {
|
|
printf("%s: popen(): Error while executing '%s'",
|
|
__progname, pCommandString);
|
|
/* popen() does not set errno reliably ... */
|
|
if (errno)
|
|
printf(": %s\n", strerror(errno));
|
|
else
|
|
printf("\n");
|
|
} else {
|
|
*popenFlag = 1;
|
|
#ifdef WIN32
|
|
_setmode(_fileno(filep), _O_BINARY );
|
|
#endif
|
|
}
|
|
xfree(pCommandString);
|
|
|
|
if (qFlag)
|
|
dup2(stderr_fd, fileno(stderr));
|
|
|
|
xfree(pMetadata);
|
|
return (filep);
|
|
}
|
|
|
|
xfree(pMetadata);
|
|
|
|
if ((filep = fopen(fileName, "rb")) == NULL)
|
|
printf("%s: %s: %s\n", __progname, fileName,
|
|
strerror(errno));
|
|
|
|
return (filep);
|
|
}
|
|
|
|
int
|
|
reconnectServer(shout_t *shout, int closeConn)
|
|
{
|
|
unsigned int i;
|
|
int close_conn = closeConn;
|
|
|
|
printf("%s: Connection to %s lost\n", __progname, pezConfig->URL);
|
|
|
|
i = 0;
|
|
while (++i) {
|
|
printf("%s: Attempting reconnection #", __progname);
|
|
if (pezConfig->reconnectAttempts > 0)
|
|
printf("%u/%u: ", i,
|
|
pezConfig->reconnectAttempts);
|
|
else
|
|
printf("%u: ", i);
|
|
|
|
if (close_conn == 0)
|
|
close_conn = 1;
|
|
else
|
|
shout_close(shout);
|
|
if (shout_open(shout) == SHOUTERR_SUCCESS) {
|
|
printf("OK\n%s: Reconnect to %s successful\n",
|
|
__progname, pezConfig->URL);
|
|
return (1);
|
|
}
|
|
|
|
printf("FAILED: %s\n", shout_get_error(shout));
|
|
|
|
if (pezConfig->reconnectAttempts > 0 &&
|
|
i >= pezConfig->reconnectAttempts)
|
|
break;
|
|
|
|
printf("%s: Waiting 5s for %s to come back ...\n",
|
|
__progname, pezConfig->URL);
|
|
sleep(5);
|
|
};
|
|
|
|
printf("%s: Giving up\n", __progname);
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
sendStream(shout_t *shout, FILE *filepstream, const char *fileName,
|
|
int isStdin, void *tv)
|
|
{
|
|
unsigned char buff[4096];
|
|
size_t read, total, oldTotal;
|
|
int ret;
|
|
#ifdef HAVE_GETTIMEOFDAY
|
|
double kbps = -1.0;
|
|
struct timeval timeStamp, *startTime = (struct timeval *)tv;
|
|
|
|
if (startTime == NULL) {
|
|
printf("%s: sendStream(): Internal error: startTime is NULL\n",
|
|
__progname);
|
|
abort();
|
|
}
|
|
|
|
timeStamp.tv_sec = startTime->tv_sec;
|
|
timeStamp.tv_usec = startTime->tv_usec;
|
|
#endif /* HAVE_GETTIMEOFDAY */
|
|
|
|
total = oldTotal = 0;
|
|
ret = STREAM_DONE;
|
|
while ((read = fread(buff, 1, sizeof(buff), filepstream)) > 0) {
|
|
if (shout_get_connected(shout) != SHOUTERR_CONNECTED &&
|
|
reconnectServer(shout, 0) == 0) {
|
|
ret = STREAM_SERVERR;
|
|
break;
|
|
}
|
|
|
|
shout_sync(shout);
|
|
|
|
if (shout_send(shout, buff, read) != SHOUTERR_SUCCESS) {
|
|
printf("%s: shout_send(): %s\n", __progname,
|
|
shout_get_error(shout));
|
|
if (reconnectServer(shout, 1))
|
|
break;
|
|
else {
|
|
ret = STREAM_SERVERR;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (rereadPlaylist_notify) {
|
|
rereadPlaylist_notify = 0;
|
|
if (!pezConfig->fileNameIsProgram)
|
|
printf("%s: SIGHUP signal received, will reread playlist after this file\n",
|
|
__progname);
|
|
}
|
|
if (skipTrack) {
|
|
skipTrack = 0;
|
|
ret = STREAM_SKIP;
|
|
break;
|
|
}
|
|
if (queryMetadata) {
|
|
queryMetadata = 0;
|
|
if (metadataFromProgram) {
|
|
ret = STREAM_UPDMDATA;
|
|
break;
|
|
}
|
|
}
|
|
|
|
total += read;
|
|
if (qFlag && vFlag) {
|
|
#ifdef HAVE_GETTIMEOFDAY
|
|
struct timeval tv;
|
|
double oldTime, newTime;
|
|
unsigned int hrs, mins, secs;
|
|
#endif /* HAVE_GETTIMEOFDAY */
|
|
|
|
if (!isStdin && playlistMode) {
|
|
if (pezConfig->fileNameIsProgram) {
|
|
char *tmp = xstrdup(pezConfig->fileName);
|
|
printf(" [%s]",
|
|
basename(tmp));
|
|
xfree(tmp);
|
|
} else
|
|
printf(" [%4lu/%-4lu]",
|
|
playlist_get_position(playlist),
|
|
playlist_get_num_items(playlist));
|
|
}
|
|
|
|
#ifdef HAVE_GETTIMEOFDAY
|
|
oldTime = (double)timeStamp.tv_sec
|
|
+ (double)timeStamp.tv_usec / 1000000.0;
|
|
gettimeofday(&tv, NULL);
|
|
newTime = (double)tv.tv_sec
|
|
+ (double)tv.tv_usec / 1000000.0;
|
|
secs = tv.tv_sec - startTime->tv_sec;
|
|
hrs = secs / 3600;
|
|
secs %= 3600;
|
|
mins = secs / 60;
|
|
secs %= 60;
|
|
if (newTime - oldTime >= 1.0) {
|
|
kbps = (((double)(total - oldTotal) / (newTime - oldTime)) * 8.0) / 1000.0;
|
|
timeStamp.tv_sec = tv.tv_sec;
|
|
timeStamp.tv_usec = tv.tv_usec;
|
|
oldTotal = total;
|
|
}
|
|
printf(" [ %uh%02um%02us]", hrs, mins, secs);
|
|
if (kbps < 0)
|
|
printf(" ");
|
|
else
|
|
printf(" [%8.2f kbps]", kbps);
|
|
#endif /* HAVE_GETTIMEOFDAY */
|
|
|
|
printf(" \r");
|
|
fflush(stdout);
|
|
}
|
|
}
|
|
if (ferror(filepstream)) {
|
|
if (errno == EINTR) {
|
|
clearerr(filepstream);
|
|
ret = STREAM_CONT;
|
|
} else
|
|
printf("%s: streamFile(): Error while reading '%s': %s\n",
|
|
__progname, fileName, strerror(errno));
|
|
}
|
|
|
|
return (ret);
|
|
}
|
|
|
|
int
|
|
streamFile(shout_t *shout, const char *fileName)
|
|
{
|
|
FILE *filepstream = NULL;
|
|
int popenFlag = 0;
|
|
char *metaData = NULL;
|
|
int isStdin = 0;
|
|
int ret, retval = 0;
|
|
#ifdef HAVE_GETTIMEOFDAY
|
|
struct timeval startTime;
|
|
#endif
|
|
|
|
if ((filepstream = openResource(shout, fileName, &popenFlag,
|
|
&metaData, &isStdin))
|
|
== NULL) {
|
|
return (retval);
|
|
}
|
|
|
|
if (metaData != NULL) {
|
|
printf("%s: Streaming ``%s''", __progname, metaData);
|
|
if (vFlag)
|
|
printf(" (file: %s)\n", fileName);
|
|
else
|
|
printf("\n");
|
|
xfree(metaData);
|
|
}
|
|
|
|
#ifdef HAVE_GETTIMEOFDAY
|
|
gettimeofday(&startTime, NULL);
|
|
do {
|
|
ret = sendStream(shout, filepstream, fileName, isStdin,
|
|
(void *)&startTime);
|
|
#else
|
|
do {
|
|
ret = sendStream(shout, filepstream, fileName, isStdin, NULL);
|
|
#endif
|
|
if (ret != STREAM_DONE) {
|
|
if ((skipTrack && rereadPlaylist) ||
|
|
(skipTrack && queryMetadata)) {
|
|
skipTrack = 0;
|
|
ret = STREAM_CONT;
|
|
}
|
|
if (queryMetadata && rereadPlaylist) {
|
|
queryMetadata = 0;
|
|
ret = STREAM_CONT;
|
|
}
|
|
if (ret == STREAM_SKIP || skipTrack) {
|
|
skipTrack = 0;
|
|
if (!isStdin && vFlag)
|
|
printf("%s: SIGUSR1 signal received, skipping current track\n",
|
|
__progname);
|
|
retval = 1;
|
|
ret = STREAM_DONE;
|
|
}
|
|
if (ret == STREAM_UPDMDATA || queryMetadata) {
|
|
queryMetadata = 0;
|
|
if (metadataFromProgram) {
|
|
char *mdataStr;
|
|
|
|
if (vFlag > 1)
|
|
printf("%s: Querying '%s' for fresh metadata\n",
|
|
__progname, pezConfig->metadataProgram);
|
|
if ((mdataStr = processMetadata(shout, pezConfig->metadataProgram)) == NULL) {
|
|
retval = 0;
|
|
ret = STREAM_DONE;
|
|
}
|
|
printf("%s: New metadata: ``%s''\n",
|
|
__progname, mdataStr);
|
|
xfree(mdataStr);
|
|
}
|
|
}
|
|
if (ret == STREAM_SERVERR) {
|
|
retval = 0;
|
|
ret = STREAM_DONE;
|
|
}
|
|
} else
|
|
retval = 1;
|
|
} while (ret != STREAM_DONE);
|
|
|
|
if (popenFlag)
|
|
pclose(filepstream);
|
|
else
|
|
fclose(filepstream);
|
|
|
|
return (retval);
|
|
}
|
|
|
|
int
|
|
streamPlaylist(shout_t *shout, const char *fileName)
|
|
{
|
|
const char *song;
|
|
char lastSong[PATH_MAX];
|
|
|
|
if (playlist == NULL) {
|
|
if (pezConfig->fileNameIsProgram) {
|
|
if ((playlist = playlist_program(fileName)) == NULL)
|
|
return (0);
|
|
} else {
|
|
if ((playlist = playlist_read(fileName)) == NULL)
|
|
return (0);
|
|
}
|
|
} else
|
|
/*
|
|
* XXX: This preserves traditional behavior, however,
|
|
* rereading the playlist after each walkthrough seems a
|
|
* bit more logical.
|
|
*/
|
|
playlist_rewind(playlist);
|
|
|
|
if (!pezConfig->fileNameIsProgram && pezConfig->shuffle)
|
|
playlist_shuffle(playlist);
|
|
|
|
while ((song = playlist_get_next(playlist)) != NULL) {
|
|
strlcpy(lastSong, song, sizeof(lastSong));
|
|
if (!streamFile(shout, song))
|
|
return (0);
|
|
if (rereadPlaylist) {
|
|
rereadPlaylist = rereadPlaylist_notify = 0;
|
|
if (pezConfig->fileNameIsProgram)
|
|
continue;
|
|
printf("%s: Rereading playlist\n", __progname);
|
|
if (!playlist_reread(&playlist))
|
|
return (0);
|
|
if (pezConfig->shuffle)
|
|
playlist_shuffle(playlist);
|
|
else {
|
|
playlist_goto_entry(playlist, lastSong);
|
|
playlist_skip_next(playlist);
|
|
}
|
|
continue;
|
|
}
|
|
}
|
|
|
|
return (1);
|
|
}
|
|
|
|
/* Borrowed from OpenNTPd-portable's compat-openbsd/bsd-misc.c */
|
|
char *
|
|
getProgname(const char *argv0)
|
|
{
|
|
#ifdef HAVE___PROGNAME
|
|
return (xstrdup(__progname));
|
|
#else
|
|
char *p;
|
|
|
|
if (argv0 == NULL)
|
|
return ((char *)"ezstream");
|
|
p = strrchr(argv0, PATH_SEPARATOR);
|
|
if (p == NULL)
|
|
p = (char *)argv0;
|
|
else
|
|
p++;
|
|
|
|
return (xstrdup(p));
|
|
#endif /* HAVE___PROGNAME */
|
|
}
|
|
|
|
void
|
|
usage(void)
|
|
{
|
|
printf("usage: %s [-hqVv] [-c configfile]\n", __progname);
|
|
}
|
|
|
|
void
|
|
usageHelp(void)
|
|
{
|
|
printf("\n");
|
|
printf(" -c configfile use XML configuration in configfile\n");
|
|
printf(" -h display this additional help and exit\n");
|
|
printf(" -q suppress STDERR output from external en-/decoders\n");
|
|
printf(" -V print the version number and exit\n");
|
|
printf(" -v verbose output (use twice for more effect)\n");
|
|
printf("\n");
|
|
printf("See the ezstream(1) manual for detailed information.\n");
|
|
}
|
|
|
|
int
|
|
main(int argc, char *argv[])
|
|
{
|
|
int c;
|
|
char *configFile = NULL;
|
|
char *host = NULL;
|
|
int port = 0;
|
|
char *mount = NULL;
|
|
shout_t *shout;
|
|
extern char *optarg;
|
|
extern int optind;
|
|
#ifdef HAVE_SIGNALS
|
|
struct sigaction act;
|
|
unsigned int i;
|
|
#endif
|
|
|
|
__progname = getProgname(argv[0]);
|
|
pezConfig = getEZConfig();
|
|
|
|
qFlag = 0;
|
|
vFlag = 0;
|
|
|
|
while ((c = getopt(argc, argv, "c:hqVv")) != -1) {
|
|
switch (c) {
|
|
case 'c':
|
|
if (configFile != NULL) {
|
|
printf("Error: multiple -c arguments given\n");
|
|
usage();
|
|
return (2);
|
|
}
|
|
configFile = xstrdup(optarg);
|
|
break;
|
|
case 'h':
|
|
usage();
|
|
usageHelp();
|
|
return (0);
|
|
case 'q':
|
|
qFlag = 1;
|
|
break;
|
|
case 'V':
|
|
printf("%s\n", PACKAGE_STRING);
|
|
return (0);
|
|
case 'v':
|
|
vFlag++;
|
|
break;
|
|
case '?':
|
|
usage();
|
|
return (2);
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
argc -= optind;
|
|
argv += optind;
|
|
|
|
if (configFile == NULL) {
|
|
printf("You must supply a config file with the -c argument.\n");
|
|
usage();
|
|
return (2);
|
|
} else {
|
|
/*
|
|
* Attempt to open configFile here for a more meaningful error
|
|
* message. Where possible, do it with stat() and check for
|
|
* safe config file permissions.
|
|
*/
|
|
#ifdef HAVE_STAT
|
|
struct stat st;
|
|
|
|
if (stat(configFile, &st) == -1) {
|
|
printf("%s: %s\n", configFile, strerror(errno));
|
|
usage();
|
|
return (2);
|
|
}
|
|
if (vFlag && (st.st_mode & (S_IRGRP | S_IROTH)))
|
|
printf("%s: Warning: %s is group and/or world readable\n",
|
|
__progname, configFile);
|
|
if (st.st_mode & (S_IWGRP | S_IWOTH)) {
|
|
printf("%s: Error: %s is group and/or world writeable\n",
|
|
__progname, configFile);
|
|
return (2);
|
|
}
|
|
#else
|
|
FILE *tmp;
|
|
|
|
if ((tmp = fopen(configFile, "r")) == NULL) {
|
|
printf("%s: %s\n", configFile, strerror(errno));
|
|
usage();
|
|
return (2);
|
|
}
|
|
fclose(tmp);
|
|
#endif /* HAVE_STAT */
|
|
}
|
|
|
|
if (!parseConfig(configFile))
|
|
return (2);
|
|
|
|
shout_init();
|
|
playlist_init();
|
|
|
|
if (pezConfig->URL == NULL) {
|
|
printf("%s: Error: Missing <url>\n", configFile);
|
|
return (2);
|
|
}
|
|
if (!urlParse(pezConfig->URL, &host, &port, &mount)) {
|
|
printf("Must be of the form ``http://server:port/mountpoint''\n");
|
|
return (2);
|
|
}
|
|
if (strlen(host) == 0) {
|
|
printf("%s: Error: Invalid <url>: Missing server:\n", configFile);
|
|
printf("Must be of the form ``http://server:port/mountpoint''\n");
|
|
return (2);
|
|
}
|
|
if (strlen(mount) == 0) {
|
|
printf("%s: Error: Invalid <url>: Missing mountpoint:\n", configFile);
|
|
printf("Must be of the form ``http://server:port/mountpoint''\n");
|
|
return (2);
|
|
}
|
|
if (pezConfig->password == NULL) {
|
|
printf("%s: Error: Missing <sourcepassword>\n", configFile);
|
|
return (2);
|
|
}
|
|
if (pezConfig->fileName == NULL) {
|
|
printf("%s: Error: Missing <filename>\n", configFile);
|
|
return (2);
|
|
}
|
|
if (pezConfig->format == NULL) {
|
|
printf("%s: Warning: Missing <format>:\n", configFile);
|
|
printf("Specify a stream format of either MP3, VORBIS or THEORA\n");
|
|
}
|
|
|
|
xfree(configFile);
|
|
|
|
if ((shout = shout_new()) == NULL) {
|
|
printf("%s: shout_new(): %s", __progname, strerror(ENOMEM));
|
|
return (1);
|
|
}
|
|
|
|
if (shout_set_host(shout, host) != SHOUTERR_SUCCESS) {
|
|
printf("%s: shout_set_host(): %s\n", __progname,
|
|
shout_get_error(shout));
|
|
return (1);
|
|
}
|
|
if (shout_set_protocol(shout, SHOUT_PROTOCOL_HTTP) != SHOUTERR_SUCCESS) {
|
|
printf("%s: shout_set_protocol(): %s\n", __progname,
|
|
shout_get_error(shout));
|
|
return (1);
|
|
}
|
|
if (shout_set_port(shout, port) != SHOUTERR_SUCCESS) {
|
|
printf("%s: shout_set_port: %s\n", __progname,
|
|
shout_get_error(shout));
|
|
return (1);
|
|
}
|
|
if (shout_set_password(shout, pezConfig->password) != SHOUTERR_SUCCESS) {
|
|
printf("%s: shout_set_password(): %s\n", __progname,
|
|
shout_get_error(shout));
|
|
return (1);
|
|
}
|
|
if (shout_set_mount(shout, mount) != SHOUTERR_SUCCESS) {
|
|
printf("%s: shout_set_mount(): %s\n", __progname,
|
|
shout_get_error(shout));
|
|
return (1);
|
|
}
|
|
if (shout_set_user(shout, "source") != SHOUTERR_SUCCESS) {
|
|
printf("%s: shout_set_user(): %s\n", __progname,
|
|
shout_get_error(shout));
|
|
return (1);
|
|
}
|
|
|
|
if (!strcmp(pezConfig->format, MP3_FORMAT)) {
|
|
if (shout_set_format(shout, SHOUT_FORMAT_MP3) != SHOUTERR_SUCCESS) {
|
|
printf("%s: shout_set_format(MP3): %s\n",
|
|
__progname, shout_get_error(shout));
|
|
return (1);
|
|
}
|
|
}
|
|
if (!strcmp(pezConfig->format, VORBIS_FORMAT) ||
|
|
!strcmp(pezConfig->format, THEORA_FORMAT)) {
|
|
if (shout_set_format(shout, SHOUT_FORMAT_OGG) != SHOUTERR_SUCCESS) {
|
|
printf("%s: shout_set_format(OGG): %s\n",
|
|
__progname, shout_get_error(shout));
|
|
return (1);
|
|
}
|
|
}
|
|
|
|
if (pezConfig->serverName) {
|
|
if (shout_set_name(shout, pezConfig->serverName) != SHOUTERR_SUCCESS) {
|
|
printf("%s: shout_set_name(): %s\n",
|
|
__progname, shout_get_error(shout));
|
|
return (1);
|
|
}
|
|
}
|
|
if (pezConfig->serverURL) {
|
|
if (shout_set_url(shout, pezConfig->serverURL) != SHOUTERR_SUCCESS) {
|
|
printf("%s: shout_set_url(): %s\n",
|
|
__progname, shout_get_error(shout));
|
|
return (1);
|
|
}
|
|
}
|
|
if (pezConfig->serverGenre) {
|
|
if (shout_set_genre(shout, pezConfig->serverGenre) != SHOUTERR_SUCCESS) {
|
|
printf("%s: shout_set_genre(): %s\n",
|
|
__progname, shout_get_error(shout));
|
|
return (1);
|
|
}
|
|
}
|
|
if (pezConfig->serverDescription) {
|
|
if (shout_set_description(shout, pezConfig->serverDescription) != SHOUTERR_SUCCESS) {
|
|
printf("%s: shout_set_description(): %s\n",
|
|
__progname, shout_get_error(shout));
|
|
return (1);
|
|
}
|
|
}
|
|
if (pezConfig->serverBitrate) {
|
|
if (shout_set_audio_info(shout, SHOUT_AI_BITRATE, pezConfig->serverBitrate) != SHOUTERR_SUCCESS) {
|
|
printf("%s: shout_set_audio_info(AI_BITRATE): %s\n",
|
|
__progname, shout_get_error(shout));
|
|
return (1);
|
|
}
|
|
}
|
|
if (pezConfig->serverChannels) {
|
|
if (shout_set_audio_info(shout, SHOUT_AI_CHANNELS, pezConfig->serverChannels) != SHOUTERR_SUCCESS) {
|
|
printf("%s: shout_set_audio_info(AI_CHANNELS): %s\n",
|
|
__progname, shout_get_error(shout));
|
|
return (1);
|
|
}
|
|
}
|
|
if (pezConfig->serverSamplerate) {
|
|
if (shout_set_audio_info(shout, SHOUT_AI_SAMPLERATE, pezConfig->serverSamplerate) != SHOUTERR_SUCCESS) {
|
|
printf("%s: shout_set_audio_info(AI_SAMPLERATE): %s\n",
|
|
__progname, shout_get_error(shout));
|
|
return (1);
|
|
}
|
|
}
|
|
if (pezConfig->serverQuality) {
|
|
if (shout_set_audio_info(shout, SHOUT_AI_QUALITY, pezConfig->serverQuality) != SHOUTERR_SUCCESS) {
|
|
printf("%s: shout_set_audio_info(AI_QUALITY): %s\n",
|
|
__progname, shout_get_error(shout));
|
|
return (1);
|
|
}
|
|
}
|
|
|
|
if (shout_set_public(shout, pezConfig->serverPublic) != SHOUTERR_SUCCESS) {
|
|
printf("%s: shout_set_public(): %s\n",
|
|
__progname, shout_get_error(shout));
|
|
return (1);
|
|
}
|
|
|
|
if (pezConfig->metadataProgram != NULL)
|
|
metadataFromProgram = 1;
|
|
else
|
|
metadataFromProgram = 0;
|
|
|
|
#ifdef HAVE_SIGNALS
|
|
memset(&act, 0, sizeof(act));
|
|
act.sa_handler = sig_handler;
|
|
# ifdef SA_RESTART
|
|
act.sa_flags = SA_RESTART;
|
|
# endif
|
|
for (i = 0; i < sizeof(ezstream_signals) / sizeof(int); i++) {
|
|
if (sigaction(ezstream_signals[i], &act, NULL) == -1) {
|
|
printf("%s: sigaction(): %s\n",
|
|
__progname, strerror(errno));
|
|
return (1);
|
|
}
|
|
}
|
|
#endif /* HAVE_SIGNALS */
|
|
|
|
if (shout_open(shout) == SHOUTERR_SUCCESS) {
|
|
int ret;
|
|
char *tmpFileName, *p;
|
|
|
|
printf("%s: Connected to http://%s:%d%s\n", __progname,
|
|
host, port, mount);
|
|
|
|
tmpFileName = xstrdup(pezConfig->fileName);
|
|
for (p = tmpFileName; *p != '\0'; p++)
|
|
*p = tolower((int)*p);
|
|
if (pezConfig->fileNameIsProgram ||
|
|
strrcmp(tmpFileName, ".m3u") == 0 ||
|
|
strrcmp(tmpFileName, ".txt") == 0)
|
|
playlistMode = 1;
|
|
else
|
|
playlistMode = 0;
|
|
xfree(tmpFileName);
|
|
|
|
if (vFlag && pezConfig->fileNameIsProgram)
|
|
printf("%s: Using program '%s' to get filenames for streaming\n",
|
|
__progname, pezConfig->fileName);
|
|
|
|
ret = 1;
|
|
do {
|
|
if (playlistMode)
|
|
ret = streamPlaylist(shout,
|
|
pezConfig->fileName);
|
|
else {
|
|
ret = streamFile(shout, pezConfig->fileName);
|
|
if (pezConfig->streamOnce)
|
|
break;
|
|
}
|
|
} while (ret);
|
|
} else
|
|
printf("%s: Connection to http://%s:%d%s failed: %s\n", __progname,
|
|
host, port, mount, shout_get_error(shout));
|
|
|
|
if (vFlag)
|
|
printf("%s: Exiting ...\n", __progname);
|
|
|
|
shout_close(shout);
|
|
|
|
playlist_free(&playlist);
|
|
playlist_shutdown();
|
|
|
|
shout_shutdown();
|
|
|
|
xfree(host);
|
|
xfree(mount);
|
|
|
|
return 0;
|
|
}
|