mirror of
https://github.com/owntone/owntone-server.git
synced 2025-01-03 19:13:24 -05:00
2642 lines
61 KiB
C
2642 lines
61 KiB
C
/*
|
|
* Copyright (C) 2009-2010 Julien BLACHE <jb@jblache.org>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include <config.h>
|
|
#endif
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
#include <fcntl.h>
|
|
#include <limits.h>
|
|
#include <errno.h>
|
|
#include <pthread.h>
|
|
#include <sys/param.h>
|
|
#include <sys/queue.h>
|
|
#include <sys/types.h>
|
|
#include <stdint.h>
|
|
#include <inttypes.h>
|
|
|
|
#if defined(HAVE_SYS_EVENTFD_H) && defined(HAVE_EVENTFD)
|
|
# define USE_EVENTFD
|
|
# include <sys/eventfd.h>
|
|
#endif
|
|
|
|
#include "logger.h"
|
|
#include "db.h"
|
|
#include "conffile.h"
|
|
#include "misc.h"
|
|
#include "mpd.h"
|
|
|
|
#include "player.h"
|
|
#include "filescanner.h"
|
|
|
|
|
|
static pthread_t tid_mpd;
|
|
|
|
struct event_base *evbase_mpd;
|
|
static int g_exit_pipe[2];
|
|
static struct event *g_exitev;
|
|
|
|
#define COMMAND_ARGV_MAX 37
|
|
|
|
/* MPD error codes (taken from ack.h) */
|
|
enum ack
|
|
{
|
|
ACK_ERROR_NOT_LIST = 1,
|
|
ACK_ERROR_ARG = 2,
|
|
ACK_ERROR_PASSWORD = 3,
|
|
ACK_ERROR_PERMISSION = 4,
|
|
ACK_ERROR_UNKNOWN = 5,
|
|
|
|
ACK_ERROR_NO_EXIST = 50,
|
|
ACK_ERROR_PLAYLIST_MAX = 51,
|
|
ACK_ERROR_SYSTEM = 52,
|
|
ACK_ERROR_PLAYLIST_LOAD = 53,
|
|
ACK_ERROR_UPDATE_ALREADY = 54,
|
|
ACK_ERROR_PLAYER_SYNC = 55,
|
|
ACK_ERROR_EXIST = 56,
|
|
};
|
|
|
|
enum command_list_type
|
|
{
|
|
COMMAND_LIST = 1,
|
|
COMMAND_LIST_OK = 2,
|
|
COMMAND_LIST_NONE = 3
|
|
};
|
|
|
|
static void
|
|
thread_exit(void)
|
|
{
|
|
int dummy = 42;
|
|
|
|
DPRINTF(E_DBG, L_MPD, "Killing mpd thread\n");
|
|
|
|
if (write(g_exit_pipe[1], &dummy, sizeof(dummy)) != sizeof(dummy))
|
|
DPRINTF(E_LOG, L_MPD, "Could not write to exit fd: %s\n", strerror(errno));
|
|
}
|
|
|
|
|
|
/* Thread: mpd */
|
|
static void *
|
|
mpd(void *arg)
|
|
{
|
|
int ret;
|
|
|
|
ret = db_perthread_init();
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Error: DB init failed\n");
|
|
|
|
pthread_exit(NULL);
|
|
}
|
|
|
|
event_base_dispatch(evbase_mpd);
|
|
|
|
db_perthread_deinit();
|
|
|
|
pthread_exit(NULL);
|
|
}
|
|
|
|
static void
|
|
exit_cb(int fd, short what, void *arg)
|
|
{
|
|
int dummy;
|
|
int ret;
|
|
|
|
ret = read(g_exit_pipe[0], &dummy, sizeof(dummy));
|
|
if (ret != sizeof(dummy))
|
|
DPRINTF(E_LOG, L_MPD, "Error reading from exit pipe\n");
|
|
|
|
event_base_loopbreak(evbase_mpd);
|
|
|
|
event_add(g_exitev, NULL);
|
|
}
|
|
|
|
/*
|
|
* Parses a rage argument of the form START:END (the END item is not included in the range)
|
|
* into its start and end position.
|
|
*
|
|
* @param range the range argument
|
|
* @param start_pos set by this method to the start position
|
|
* @param end_pos set by this method to the end postion
|
|
* @return 0 on success, -1 on failure
|
|
*/
|
|
static int
|
|
mpd_pars_range_arg(char *range, int *start_pos, int *end_pos)
|
|
{
|
|
int ret;
|
|
|
|
if (strchr(range, ':'))
|
|
{
|
|
ret = sscanf(range, "%d:%d", start_pos, end_pos);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Error parsing range argument '%s' (return code = %d)\n", range, ret);
|
|
return -1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ret = safe_atoi32(range, start_pos);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Error parsing integer argument '%s' (return code = %d)\n", range, ret);
|
|
return -1;
|
|
}
|
|
|
|
*end_pos = (*start_pos) + 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Returns the next unquoted string argument from the input string
|
|
*/
|
|
static char*
|
|
mpd_pars_unquoted(char **input)
|
|
{
|
|
char *arg;
|
|
|
|
arg = *input;
|
|
|
|
while (**input != 0)
|
|
{
|
|
if (**input == ' ')
|
|
{
|
|
**input = '\0';
|
|
(*input)++;
|
|
return arg;
|
|
}
|
|
|
|
(*input)++;
|
|
}
|
|
|
|
return arg;
|
|
}
|
|
|
|
/*
|
|
* Returns the next quoted string argument from the input string
|
|
* with the quotes removed
|
|
*/
|
|
static char*
|
|
mpd_pars_quoted(char **input)
|
|
{
|
|
char *arg;
|
|
|
|
// skip double quote character
|
|
(*input)++;
|
|
|
|
arg = *input;
|
|
|
|
while (**input != '"')
|
|
{
|
|
// A backslash character escapes the following character
|
|
if (**input == '\\')
|
|
{
|
|
(*input)++;
|
|
}
|
|
|
|
if (**input == 0)
|
|
{
|
|
// Error handling for missing double quote at end of parameter
|
|
DPRINTF(E_LOG, L_MPD, "Error missing closing double quote in argument\n");
|
|
return NULL;
|
|
}
|
|
|
|
(*input)++;
|
|
}
|
|
|
|
**input = '\0';
|
|
(*input)++;
|
|
|
|
return arg;
|
|
}
|
|
|
|
/*
|
|
* Parses the argument string into an array of strings.
|
|
* Arguments are seperated by a whitespace character and may be wrapped in double quotes.
|
|
*
|
|
* @param args the arguments
|
|
* @param argc the number of arguments in the argument string
|
|
* @param argv the array containing the found arguments
|
|
*/
|
|
static int
|
|
mpd_parse_args(char *args, int *argc, char **argv)
|
|
{
|
|
char *input;
|
|
|
|
input = args;
|
|
*argc = 0;
|
|
|
|
while (*input != 0)
|
|
{
|
|
// Ignore whitespace characters
|
|
if (*input == ' ')
|
|
{
|
|
input++;
|
|
continue;
|
|
}
|
|
|
|
// Check if the parameter is wrapped in double quotes
|
|
if (*input == '"')
|
|
{
|
|
argv[*argc] = mpd_pars_quoted(&input);
|
|
if (argv[*argc] == NULL)
|
|
{
|
|
return -1;
|
|
}
|
|
*argc = *argc + 1;
|
|
}
|
|
else
|
|
{
|
|
argv[*argc] = mpd_pars_unquoted(&input);
|
|
*argc = *argc + 1;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Adds the informations (path, id, tags, etc.) for the given song to the given buffer
|
|
* with additional information for the position of this song in the playqueue.
|
|
*
|
|
* Example output:
|
|
* file: foo/bar/song.mp3
|
|
* Last-Modified: 2013-07-14T06:57:59Z
|
|
* Time: 172
|
|
* Artist: foo
|
|
* AlbumArtist: foo
|
|
* ArtistSort: foo
|
|
* AlbumArtistSort: foo
|
|
* Title: song
|
|
* Album: bar
|
|
* Track: 1/11
|
|
* Date: 2012-09-11
|
|
* Genre: Alternative
|
|
* Disc: 1/1
|
|
* MUSICBRAINZ_ALBUMARTISTID: c5c2ea1c-4bde-4f4d-bd0b-47b200bf99d6
|
|
* MUSICBRAINZ_ARTISTID: c5c2ea1c-4bde-4f4d-bd0b-47b200bf99d6
|
|
* MUSICBRAINZ_ALBUMID: 812f4b87-8ad9-41bd-be79-38151f17a2b4
|
|
* MUSICBRAINZ_TRACKID: fde95c39-ee51-48f6-a7f9-b5631c2ed156
|
|
* Pos: 0
|
|
* Id: 1
|
|
*
|
|
* @param evbuf the response event buffer
|
|
* @param mfi media information
|
|
* @param pos_pl position in the playqueue, if -1 the position is ignored
|
|
* @return the number of bytes added if successful, or -1 if an error occurred.
|
|
*/
|
|
static int
|
|
mpd_add_mediainfo(struct evbuffer *evbuf, struct media_file_info *mfi, int pos_pl)
|
|
{
|
|
int ret;
|
|
|
|
if (pos_pl < 0)
|
|
{
|
|
ret = evbuffer_add_printf(evbuf,
|
|
"file: %s\n"
|
|
"Time: %d\n"
|
|
"Artist: %s\n"
|
|
"Album: %s\n"
|
|
"Title: %s\n"
|
|
"Id: %d\n",
|
|
(mfi->virtual_path + 1),
|
|
(mfi->song_length / 1000),
|
|
mfi->artist,
|
|
mfi->album,
|
|
mfi->title,
|
|
mfi->id);
|
|
}
|
|
else
|
|
{
|
|
ret = evbuffer_add_printf(evbuf,
|
|
"file: %s\n"
|
|
"Time: %d\n"
|
|
"Artist: %s\n"
|
|
"Album: %s\n"
|
|
"Title: %s\n"
|
|
"Pos: %d\n"
|
|
"Id: %d\n",
|
|
(mfi->virtual_path + 1),
|
|
(mfi->song_length / 1000),
|
|
mfi->artist,
|
|
mfi->album,
|
|
mfi->title,
|
|
pos_pl,
|
|
mfi->id);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Adds the informations (path, id, tags, etc.) for the given song to the given buffer.
|
|
*
|
|
* Example output:
|
|
* file: foo/bar/song.mp3
|
|
* Last-Modified: 2013-07-14T06:57:59Z
|
|
* Time: 172
|
|
* Artist: foo
|
|
* AlbumArtist: foo
|
|
* ArtistSort: foo
|
|
* AlbumArtistSort: foo
|
|
* Title: song
|
|
* Album: bar
|
|
* Track: 1/11
|
|
* Date: 2012-09-11
|
|
* Genre: Alternative
|
|
* Disc: 1/1
|
|
* MUSICBRAINZ_ALBUMARTISTID: c5c2ea1c-4bde-4f4d-bd0b-47b200bf99d6
|
|
* MUSICBRAINZ_ARTISTID: c5c2ea1c-4bde-4f4d-bd0b-47b200bf99d6
|
|
* MUSICBRAINZ_ALBUMID: 812f4b87-8ad9-41bd-be79-38151f17a2b4
|
|
* MUSICBRAINZ_TRACKID: fde95c39-ee51-48f6-a7f9-b5631c2ed156
|
|
* Id: 1
|
|
*
|
|
* @param evbuf the response event buffer
|
|
* @param mfi media information
|
|
* @return the number of bytes added if successful, or -1 if an error occurred.
|
|
*/
|
|
/*static int
|
|
mpd_add_db_media_file_info(struct evbuffer *evbuf, struct db_media_file_info *dbmfi)
|
|
{
|
|
uint32_t songlength;
|
|
int ret;
|
|
|
|
if (safe_atou32(dbmfi->song_length, &songlength) != 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Error converting song length to uint32_t: %s\n", dbmfi->song_length);
|
|
return -1;
|
|
}
|
|
|
|
ret = evbuffer_add_printf(evbuf,
|
|
"file: %s\n"
|
|
"Time: %d\n"
|
|
"Artist: %s\n"
|
|
"Album: %s\n"
|
|
"Title: %s\n"
|
|
"Id: %s\n",
|
|
(dbmfi->path + 1),
|
|
(songlength / 1000),
|
|
dbmfi->artist,
|
|
dbmfi->album,
|
|
dbmfi->title,
|
|
dbmfi->id);
|
|
|
|
return ret;
|
|
}*/
|
|
|
|
/*
|
|
* Command handler function for 'currentsong'
|
|
*/
|
|
static int
|
|
mpd_command_currentsong(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
|
|
struct player_status status;
|
|
struct media_file_info *mfi;
|
|
int ret;
|
|
|
|
player_get_status(&status);
|
|
|
|
if (status.status == PLAY_STOPPED)
|
|
{
|
|
// Return empty evbuffer if there is no current playing song
|
|
return 0;
|
|
}
|
|
|
|
mfi = db_file_fetch_byid(status.id);
|
|
if (!mfi)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Error fetching file by id: %d\n", status.id);
|
|
ret = asprintf(errmsg, "Error fetching file by id: %d", status.id);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
ret = mpd_add_mediainfo(evbuf, mfi, status.pos_pl);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Error adding media info for file with id: %d\n", status.id);
|
|
ret = asprintf(errmsg, "Error adding media info for file with id: %d", status.id);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
|
|
free_mfi(mfi, 0);
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
free_mfi(mfi, 0);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'status'
|
|
*
|
|
* Example output:
|
|
* volume: -1
|
|
* repeat: 0
|
|
* random: 0
|
|
* single: 0
|
|
* consume: 0
|
|
* playlist: 2
|
|
* playlistlength: 34
|
|
* mixrampdb: 0.000000
|
|
* state: stop
|
|
* song: 0
|
|
* songid: 1
|
|
* time: 28:306
|
|
* elapsed: 28.178
|
|
* bitrate: 278
|
|
* audio: 44100:f:2
|
|
* nextsong: 1
|
|
* nextsongid: 2
|
|
*/
|
|
static int
|
|
mpd_command_status(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
struct player_status status;
|
|
char *state;
|
|
|
|
player_get_status(&status);
|
|
|
|
switch (status.status)
|
|
{
|
|
case PLAY_PAUSED:
|
|
state = "pause";
|
|
break;
|
|
|
|
case PLAY_PLAYING:
|
|
state = "play";
|
|
break;
|
|
|
|
default:
|
|
state = "stop";
|
|
break;
|
|
}
|
|
|
|
evbuffer_add_printf(evbuf,
|
|
"volume: %d\n"
|
|
"repeat: %d\n"
|
|
"random: %d\n"
|
|
"single: %d\n"
|
|
"consume: %d\n"
|
|
"playlist: %d\n"
|
|
"playlistlength: %d\n"
|
|
"mixrampdb: 0.000000\n"
|
|
"state: %s\n",
|
|
status.volume,
|
|
(status.repeat == REPEAT_OFF ? 0 : 1),
|
|
status.shuffle,
|
|
(status.repeat == REPEAT_SONG ? 1 : 0),
|
|
0 /* consume: not supported by forked-daapd, always return 'off' */,
|
|
status.plid,
|
|
status.playlistlength,
|
|
state);
|
|
|
|
if (status.status != PLAY_STOPPED)
|
|
{
|
|
evbuffer_add_printf(evbuf,
|
|
"song: %d\n"
|
|
"songid: %d\n"
|
|
"time: %d:%d\n"
|
|
"elapsed: %#.3f\n"
|
|
"bitrate: 128\n"
|
|
"audio: 44100:16:2\n"
|
|
"nextsong: %d\n"
|
|
"nextsongid: %d\n",
|
|
status.pos_pl,
|
|
status.id,
|
|
(status.pos_ms / 1000), (status.songlength_ms / 1000),
|
|
(status.pos_ms / 1000.0),
|
|
status.nextsong_pos_pl,
|
|
status.nextsong_id);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'stats'
|
|
*/
|
|
static int
|
|
mpd_command_stats(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
//TODO implement command stats
|
|
evbuffer_add_printf(evbuf,
|
|
"artists: %d\n"
|
|
"albums: %d\n"
|
|
"songs: %d\n"
|
|
"uptime: %d\n" //in seceonds
|
|
"db_playtime: %d\n"
|
|
"db_update: %d\n"
|
|
"playtime: %d\n",
|
|
1,
|
|
2,
|
|
3,
|
|
4,
|
|
5,
|
|
6,
|
|
7);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'random'
|
|
* Sets the shuffle mode, expects argument argv[1] to be an integer with
|
|
* 0 = disable shuffle
|
|
* 1 = enable shuffle
|
|
*/
|
|
static int
|
|
mpd_command_random(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
int enable;
|
|
int ret;
|
|
|
|
if (argc < 2)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Missing argument for command 'random'\n");
|
|
ret = asprintf(errmsg, "Missing argument for command 'random'");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
ret = safe_atoi32(argv[1], &enable);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Argument doesn't convert to integer: '%s'\n", argv[1]);
|
|
ret = asprintf(errmsg, "Argument doesn't convert to integer: '%s'", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
player_shuffle_set(enable);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'repeat'
|
|
* Sets the repeat mode, expects argument argv[1] to be an integer with
|
|
* 0 = repeat off
|
|
* 1 = repeat all
|
|
*/
|
|
static int
|
|
mpd_command_repeat(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
int enable;
|
|
int ret;
|
|
|
|
if (argc < 2)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Missing argument for command 'repeat'\n");
|
|
ret = asprintf(errmsg, "Missing argument for command 'repeat'");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
ret = safe_atoi32(argv[1], &enable);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Argument doesn't convert to integer: '%s'\n", argv[1]);
|
|
ret = asprintf(errmsg, "Argument doesn't convert to integer: '%s'", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
if (enable == 0)
|
|
player_repeat_set(REPEAT_OFF);
|
|
else
|
|
player_repeat_set(REPEAT_ALL);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'setvol'
|
|
* Sets the volume, expects argument argv[1] to be an integer 0-100
|
|
*/
|
|
static int
|
|
mpd_command_setvol(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
int volume;
|
|
int ret;
|
|
|
|
if (argc < 2)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Missing argument for command 'setvol'\n");
|
|
ret = asprintf(errmsg, "Missing argument for command 'setvol'");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
ret = safe_atoi32(argv[1], &volume);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Argument doesn't convert to integer: '%s'\n", argv[1]);
|
|
ret = asprintf(errmsg, "Argument doesn't convert to integer: '%s'", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
player_volume_set(volume);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'single'
|
|
* Sets the repeat mode, expects argument argv[1] to be an integer.
|
|
* forked-daapd only allows single-mode in combination with repeat, therefor the command
|
|
* single translates (depending on the current repeat mode) into:
|
|
* a) if repeat off:
|
|
* 0 = repeat off
|
|
* 1 = repeat song
|
|
* b) if repeat all:
|
|
* 0 = repeat all
|
|
* 1 = repeat song
|
|
* c) if repeat song:
|
|
* 0 = repeat all
|
|
* 1 = repeat song
|
|
*/
|
|
static int
|
|
mpd_command_single(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
int enable;
|
|
struct player_status status;
|
|
int ret;
|
|
|
|
if (argc < 2)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Missing argument for command 'single'\n");
|
|
ret = asprintf(errmsg, "Missing argument for command 'single'");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
ret = safe_atoi32(argv[1], &enable);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Argument doesn't convert to integer: '%s'\n", argv[1]);
|
|
ret = asprintf(errmsg, "Argument doesn't convert to integer: '%s'", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
player_get_status(&status);
|
|
|
|
if (enable == 0 && status.repeat != REPEAT_OFF)
|
|
player_repeat_set(REPEAT_ALL);
|
|
else if (enable == 0)
|
|
player_repeat_set(REPEAT_OFF);
|
|
else
|
|
player_repeat_set(REPEAT_SONG);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'replay_gain_status'
|
|
* forked-daapd does not support replay gain, therefor this function returns always
|
|
* "replay_gain_mode: off".
|
|
*/
|
|
static int
|
|
mpd_command_replay_gain_status(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
evbuffer_add(evbuf, "replay_gain_mode: off\n", 22);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'volume'
|
|
* Changes the volume by the given amount, expects argument argv[1] to be an integer
|
|
*
|
|
* According to the mpd protocoll specification this function is deprecated.
|
|
*/
|
|
static int
|
|
mpd_command_volume(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
struct player_status status;
|
|
int volume;
|
|
int ret;
|
|
|
|
if (argc < 2)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Missing argument for command 'volume'\n");
|
|
ret = asprintf(errmsg, "Missing argument for command 'volume'");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
ret = safe_atoi32(argv[1], &volume);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Argument doesn't convert to integer: '%s'\n", argv[1]);
|
|
ret = asprintf(errmsg, "Argument doesn't convert to integer: '%s'", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
player_get_status(&status);
|
|
|
|
volume += status.volume;
|
|
|
|
player_volume_set(volume);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'next'
|
|
* Skips to the next song in the playqueue
|
|
*/
|
|
static int
|
|
mpd_command_next(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
int ret;
|
|
|
|
ret = player_playback_next();
|
|
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_DBG, L_MPD, "Failed to skip to next song\n");
|
|
ret = asprintf(errmsg, "Failed to skip to next song");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
ret = player_playback_start(NULL);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Player returned an error for start after nextitem\n");
|
|
ret = asprintf(errmsg, "Player returned an error for start after nextitem\n");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'pause'
|
|
* Toggles pause/play, if the optional argument argv[1] is present, it must be an integer with
|
|
* 0 = play
|
|
* 1 = pause
|
|
*/
|
|
static int
|
|
mpd_command_pause(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
int pause;
|
|
struct player_status status;
|
|
int ret;
|
|
|
|
pause = 1;
|
|
if (argc > 1)
|
|
{
|
|
ret = safe_atoi32(argv[1], &pause);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Argument doesn't convert to integer: '%s'\n", argv[1]);
|
|
ret = asprintf(errmsg, "Argument doesn't convert to integer: '%s'", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
player_get_status(&status);
|
|
|
|
if (status.status != PLAY_PLAYING)
|
|
pause = 0;
|
|
}
|
|
|
|
if (pause == 1)
|
|
ret = player_playback_pause();
|
|
else
|
|
ret = player_playback_start(NULL);
|
|
|
|
if (ret != 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Failed to pause playback\n");
|
|
ret = asprintf(errmsg, "Failed to pause playback");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'play'
|
|
* Starts playback, the optional argument argv[1] represents the position in the playqueue
|
|
* where to start playback.
|
|
*/
|
|
static int
|
|
mpd_command_play(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
int songpos;
|
|
struct player_status status;
|
|
int ret;
|
|
|
|
player_get_status(&status);
|
|
|
|
//TODO verfiy handling of play with parameter if already playing
|
|
if (status.status == PLAY_PLAYING)
|
|
{
|
|
ret = player_playback_pause();
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Error pausing playback\n");
|
|
}
|
|
}
|
|
|
|
songpos = 0;
|
|
if (argc > 1)
|
|
{
|
|
ret = safe_atoi32(argv[1], &songpos);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Argument doesn't convert to integer: '%s'\n", argv[1]);
|
|
ret = asprintf(errmsg, "Argument doesn't convert to integer: '%s'", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
}
|
|
|
|
if (songpos > 0)
|
|
ret = player_playback_startpos(songpos, NULL);
|
|
else
|
|
ret = player_playback_start(NULL);
|
|
|
|
if (ret != 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Failed to start playback\n");
|
|
ret = asprintf(errmsg, "Failed to start playback");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'playid'
|
|
* Starts playback, the optional argument argv[1] represents the songid of the song
|
|
* where to start playback.
|
|
*/
|
|
static int
|
|
mpd_command_playid(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
uint32_t id;
|
|
struct player_status status;
|
|
int ret;
|
|
|
|
player_get_status(&status);
|
|
|
|
//TODO verfiy handling of play with parameter if already playing
|
|
if (status.status == PLAY_PLAYING)
|
|
{
|
|
ret = player_playback_pause();
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Error pausing playback\n");
|
|
}
|
|
}
|
|
|
|
id = 0;
|
|
if (argc > 1)
|
|
{
|
|
ret = safe_atou32(argv[1], &id);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Argument doesn't convert to integer: '%s'\n", argv[1]);
|
|
ret = asprintf(errmsg, "Argument doesn't convert to integer: '%s'", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
}
|
|
|
|
if (id > 0)
|
|
ret = player_playback_startid(id, NULL);
|
|
else
|
|
ret = player_playback_start(NULL);
|
|
|
|
if (ret != 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Failed to start playback\n");
|
|
ret = asprintf(errmsg, "Failed to start playback");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'previous'
|
|
* Skips to the previous song in the playqueue
|
|
*/
|
|
static int
|
|
mpd_command_previous(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
int ret;
|
|
|
|
ret = player_playback_prev();
|
|
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_DBG, L_MPD, "Failed to skip to previous song\n");
|
|
ret = asprintf(errmsg, "Failed to skip to previous song");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
ret = player_playback_start(NULL);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Player returned an error for start after previtem\n");
|
|
ret = asprintf(errmsg, "Player returned an error for start after previtem\n");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'seekid'
|
|
* Seeks to song at the given position in argv[1] to the position in seconds given in argument argv[2]
|
|
* (fractions allowed).
|
|
*/
|
|
static int
|
|
mpd_command_seek(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
uint32_t songpos;
|
|
float seek_target_sec;
|
|
int seek_target_msec;
|
|
int ret;
|
|
|
|
if (argc < 3)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Missing argument for command 'seekcur'\n");
|
|
ret = asprintf(errmsg, "Missing argument for command 'seekcur'");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
ret = safe_atou32(argv[1], &songpos);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Argument doesn't convert to integer: '%s'\n", argv[1]);
|
|
ret = asprintf(errmsg, "Argument doesn't convert to integer: '%s'", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
//TODO Allow seeking in songs not currently playing
|
|
if (songpos != 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Given song is not the current playing one, seeking is not supported\n");
|
|
ret = asprintf(errmsg, "Given song is not the current playing one, seeking is not supported\n");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
seek_target_sec = strtof(argv[2], NULL);
|
|
seek_target_msec = seek_target_sec * 1000;
|
|
|
|
ret = player_playback_seek(seek_target_msec);
|
|
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_DBG, L_MPD, "Failed to seek current song to time %d msec\n", seek_target_msec);
|
|
ret = asprintf(errmsg, "Failed to seek current song to time %d msec\n", seek_target_msec);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
ret = player_playback_start(NULL);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Player returned an error for start after seekcur\n");
|
|
ret = asprintf(errmsg, "Player returned an error for start after seekcur\n");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'seekid'
|
|
* Seeks to song with id given in argv[1] to the position in seconds given in argument argv[2]
|
|
* (fractions allowed).
|
|
*/
|
|
static int
|
|
mpd_command_seekid(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
struct player_status status;
|
|
uint32_t id;
|
|
float seek_target_sec;
|
|
int seek_target_msec;
|
|
int ret;
|
|
|
|
if (argc < 3)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Missing argument for command 'seekcur'\n");
|
|
ret = asprintf(errmsg, "Missing argument for command 'seekcur'");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
ret = safe_atou32(argv[1], &id);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Argument doesn't convert to integer: '%s'\n", argv[1]);
|
|
ret = asprintf(errmsg, "Argument doesn't convert to integer: '%s'", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
//TODO Allow seeking in songs not currently playing
|
|
player_get_status(&status);
|
|
if (status.id != id)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Given song is not the current playing one, seeking is not supported\n");
|
|
ret = asprintf(errmsg, "Given song is not the current playing one, seeking is not supported\n");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
seek_target_sec = strtof(argv[2], NULL);
|
|
seek_target_msec = seek_target_sec * 1000;
|
|
|
|
ret = player_playback_seek(seek_target_msec);
|
|
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_DBG, L_MPD, "Failed to seek current song to time %d msec\n", seek_target_msec);
|
|
ret = asprintf(errmsg, "Failed to seek current song to time %d msec\n", seek_target_msec);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
ret = player_playback_start(NULL);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Player returned an error for start after seekcur\n");
|
|
ret = asprintf(errmsg, "Player returned an error for start after seekcur\n");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'seekcur'
|
|
* Seeks the current song to the position in seconds given in argument argv[1] (fractions allowed).
|
|
*/
|
|
static int
|
|
mpd_command_seekcur(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
float seek_target_sec;
|
|
int seek_target_msec;
|
|
int ret;
|
|
|
|
if (argc < 2)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Missing argument for command 'seekcur'\n");
|
|
ret = asprintf(errmsg, "Missing argument for command 'seekcur'");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
seek_target_sec = strtof(argv[1], NULL);
|
|
seek_target_msec = seek_target_sec * 1000;
|
|
|
|
// TODO If prefixed by '+' or '-', then the time is relative to the current playing position.
|
|
ret = player_playback_seek(seek_target_msec);
|
|
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_DBG, L_MPD, "Failed to seek current song to time %d msec\n", seek_target_msec);
|
|
ret = asprintf(errmsg, "Failed to seek current song to time %d msec\n", seek_target_msec);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
ret = player_playback_start(NULL);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Player returned an error for start after seekcur\n");
|
|
ret = asprintf(errmsg, "Player returned an error for start after seekcur\n");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'stop'
|
|
* Stop playback.
|
|
*/
|
|
static int
|
|
mpd_command_stop(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
int ret;
|
|
|
|
ret = player_playback_stop();
|
|
|
|
if (ret != 0)
|
|
{
|
|
DPRINTF(E_DBG, L_MPD, "Failed to stop playback\n");
|
|
ret = asprintf(errmsg, "Failed to stop playback");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'add'
|
|
* Adds the all songs under the given path to the end of the playqueue (directories add recursively).
|
|
* Expects argument argv[1] to be a path to a single file or directory.
|
|
*/
|
|
static int
|
|
mpd_command_add(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
struct player_source *ps;
|
|
int ret;
|
|
|
|
if (argc < 2)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Missing argument for command 'add'\n");
|
|
ret = asprintf(errmsg, "Missing argument for command 'add'");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
ps = player_queue_make_mpd(argv[1], 1);
|
|
|
|
if (!ps)
|
|
{
|
|
DPRINTF(E_DBG, L_MPD, "Failed to add song '%s' to playlist\n", argv[1]);
|
|
ret = asprintf(errmsg, "Failed to add song '%s' to playlist\n", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
player_queue_add(ps);
|
|
|
|
ret = player_playback_start(NULL);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Could not start playback\n");
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'addid'
|
|
* Adds the song under the given path to the end or to the given position of the playqueue.
|
|
* Expects argument argv[1] to be a path to a single file. argv[2] is optional, if present
|
|
* it must be an integer representing the position in the playqueue.
|
|
*/
|
|
static int
|
|
mpd_command_addid(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
struct player_source *ps;
|
|
int ret;
|
|
|
|
if (argc < 2)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Missing argument for command 'addid'\n");
|
|
ret = asprintf(errmsg, "Missing argument for command 'addid'");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
//TODO if argc > 2 add song at position argv[2]
|
|
if (argc > 2)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Adding at a specified position not supported for 'addid', adding songs at end of queue.\n");
|
|
}
|
|
|
|
ps = player_queue_make_mpd(argv[1], 0);
|
|
|
|
if (!ps)
|
|
{
|
|
DPRINTF(E_DBG, L_MPD, "Failed to add song '%s' to playlist\n", argv[1]);
|
|
ret = asprintf(errmsg, "Failed to add song '%s' to playlist\n", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
|
|
player_queue_add(ps);
|
|
|
|
evbuffer_add_printf(evbuf,
|
|
"addid: %s\n"
|
|
"Id: %d\n",
|
|
argv[1],
|
|
ps->id);
|
|
|
|
ret = player_playback_start(NULL);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Could not start playback\n");
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'clear'
|
|
* Stops playback and removes all songs from the playqueue
|
|
*/
|
|
static int
|
|
mpd_command_clear(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
int ret;
|
|
|
|
ret = player_playback_stop();
|
|
if (ret != 0)
|
|
{
|
|
DPRINTF(E_DBG, L_MPD, "Failed to stop playback\n");
|
|
}
|
|
|
|
player_queue_clear();
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'delete'
|
|
* Removes songs from the playqueue. Expects argument argv[1] (optional) to be an integer or
|
|
* an integer range {START:END} representing the position of the songs in the playlist, that
|
|
* should be removed.
|
|
*/
|
|
static int
|
|
mpd_command_delete(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
struct player_status status;
|
|
uint32_t start_pos;
|
|
int pos;
|
|
int ret;
|
|
|
|
// If argv[1] is ommited clear the whole queue except the current playing one
|
|
if (argc < 2)
|
|
{
|
|
player_queue_empty(0);
|
|
return 0;
|
|
}
|
|
|
|
// If argument argv[1] is present remove only the specified songs
|
|
//TODO support ranges for argv[1]
|
|
ret = safe_atou32(argv[1], &start_pos);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Argument doesn't convert to integer: '%s'\n", argv[1]);
|
|
ret = asprintf(errmsg, "Argument doesn't convert to integer: '%s'", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
player_get_status(&status);
|
|
|
|
pos = start_pos - status.pos_pl;
|
|
|
|
if (pos < 1)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Removing playing or previously played song not supported (song position %d)\n", pos);
|
|
ret = asprintf(errmsg, "Removing playing or previously played song not supported (song position %d)", pos);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
ret = player_queue_remove(pos);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Failed to remove song at position '%d'\n", pos);
|
|
ret = asprintf(errmsg, "Failed to remove song at position '%d'", pos);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'deleteid'
|
|
* Removes the song with given id from the playqueue. Expects argument argv[1] to be an integer (song id).
|
|
*/
|
|
static int
|
|
mpd_command_deleteid(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
uint32_t songid;
|
|
int ret;
|
|
|
|
if (argc < 2)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Missing argument for command 'deleteid'\n");
|
|
ret = asprintf(errmsg, "Missing argument for command 'deleteid'");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
ret = safe_atou32(argv[1], &songid);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Argument doesn't convert to integer: '%s'\n", argv[1]);
|
|
ret = asprintf(errmsg, "Argument doesn't convert to integer: '%s'", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
ret = player_queue_removeid(songid);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Failed to remove song with id '%s'\n", argv[1]);
|
|
ret = asprintf(errmsg, "Failed to remove song with id '%s'", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'playlistinfo'
|
|
* Displays a list of all songs in the queue, or if the optional argument is given, displays information
|
|
* only for the song SONGPOS or the range of songs START:END given in argv[1].
|
|
*
|
|
* The order of the songs is always the not shuffled order.
|
|
*/
|
|
static int
|
|
mpd_command_playlistinfo(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
struct player_queue *queue;
|
|
struct media_file_info *mfi;
|
|
int start_pos;
|
|
int end_pos;
|
|
int pos_pl;
|
|
int i;
|
|
int ret;
|
|
|
|
start_pos = 0;
|
|
end_pos = -1;
|
|
|
|
if (argc > 1)
|
|
{
|
|
ret = mpd_pars_range_arg(argv[1], &start_pos, &end_pos);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Argument doesn't convert to integer or range: '%s'\n", argv[1]);
|
|
ret = asprintf(errmsg, "Argument doesn't convert to integer or range: '%s'", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
}
|
|
|
|
queue = player_queue_get(start_pos, end_pos, 0);
|
|
|
|
if (!queue)
|
|
{
|
|
// Queue is emtpy
|
|
return 0;
|
|
}
|
|
|
|
pos_pl = queue->start_pos;
|
|
for (i = 0; i < queue->count; i++)
|
|
{
|
|
mfi = db_file_fetch_byid(queue->queue[i]);
|
|
if (!mfi)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Error fetching file by id: %d\n", queue->queue[i]);
|
|
ret = asprintf(errmsg, "Error fetching file by id: %d", queue->queue[i]);
|
|
|
|
queue_free(queue);
|
|
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
ret = mpd_add_mediainfo(evbuf, mfi, pos_pl);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Error adding media info for file with id: %d\n", queue->queue[i]);
|
|
ret = asprintf(errmsg, "Error adding media info for file with id: %d\n", queue->queue[i]);
|
|
|
|
queue_free(queue);
|
|
free_mfi(mfi, 0);
|
|
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
free_mfi(mfi, 0);
|
|
pos_pl++;
|
|
}
|
|
|
|
queue_free(queue);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'load'
|
|
* Adds the playlist given by virtual-path in argv[1] to the queue.
|
|
*/
|
|
static int
|
|
mpd_command_load(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
char path[PATH_MAX];
|
|
struct playlist_info *pli;
|
|
struct player_source *ps;
|
|
uint32_t pos;
|
|
int ret;
|
|
|
|
if (argc < 2)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Missing argument for command 'load'\n");
|
|
ret = asprintf(errmsg, "Missing argument for command 'load'");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
if (strncmp(argv[1], "/", 1) == 0)
|
|
{
|
|
ret = snprintf(path, sizeof(path), "%s", argv[1]);
|
|
}
|
|
else
|
|
{
|
|
ret = snprintf(path, sizeof(path), "/%s", argv[1]);
|
|
}
|
|
|
|
pli = db_pl_fetch_byvirtualpath(path);
|
|
if (!pli)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Playlist not found for path '%s'\n", argv[1]);
|
|
ret = asprintf(errmsg, "Playlist not found for path '%s'", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
//TODO If a second parameter is given only add the specified range of songs to the playqueue
|
|
|
|
ps = player_queue_make_pl(pli->id, &pos);
|
|
|
|
if (!ps)
|
|
{
|
|
free_pli(pli, 0);
|
|
|
|
DPRINTF(E_DBG, L_MPD, "Failed to add song '%s' to playlist\n", argv[1]);
|
|
ret = asprintf(errmsg, "Failed to add song '%s' to playlist\n", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
player_queue_add(ps);
|
|
|
|
ret = player_playback_start(NULL);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Could not start playback\n");
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'lsinfo'
|
|
* Lists the contents of the directory given in argv[1].
|
|
*/
|
|
static int
|
|
mpd_command_lsinfo(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
struct query_params qp;
|
|
char parent[PATH_MAX];
|
|
struct filelist_info *fi;
|
|
struct media_file_info *mfi;
|
|
int ret;
|
|
|
|
if (strncmp(argv[1], "/", 1) == 0)
|
|
{
|
|
ret = snprintf(parent, sizeof(parent), "%s", argv[1]);
|
|
}
|
|
else
|
|
{
|
|
ret = snprintf(parent, sizeof(parent), "/%s", argv[1]);
|
|
}
|
|
|
|
if ((ret < 0) || (ret >= sizeof(parent)))
|
|
{
|
|
DPRINTF(E_INFO, L_MPD, "Parent path exceeds PATH_MAX\n");
|
|
return -1;
|
|
}
|
|
|
|
fi = (struct filelist_info*)malloc(sizeof(struct filelist_info));
|
|
if (!fi)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory for fi\n");
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
memset(&qp, 0, sizeof(struct query_params));
|
|
|
|
ret = db_build_query_filelist(&qp, parent);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Could not start query for path '%s'\n", argv[1]);
|
|
ret = asprintf(errmsg, "Could not start query for path '%s'\n", argv[1]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
|
|
free_fi(fi, 0);
|
|
return ACK_ERROR_UNKNOWN;
|
|
}
|
|
|
|
while (((ret = db_query_fetch_filelist(&qp, fi)) == 0) && (fi->path))
|
|
{
|
|
if (fi->type == F_DIR)
|
|
{
|
|
evbuffer_add_printf(evbuf,
|
|
"directory: %s\n"
|
|
"Last-Modified: 2014-07-11T14:13:56Z\n", //TODO Send correct last modified timestamp
|
|
(fi->path + 1));
|
|
}
|
|
else if (fi->type == F_PLAYLIST)
|
|
{
|
|
evbuffer_add_printf(evbuf,
|
|
"playlist: %s\n"
|
|
"Last-Modified: 2014-07-11T14:13:56Z\n", //TODO Send correct last modified timestamp
|
|
(fi->path + 1));
|
|
}
|
|
else if (fi->type == F_FILE)
|
|
{
|
|
mfi = db_file_fetch_byvirtualpath(fi->path);
|
|
if (mfi)
|
|
{
|
|
ret = mpd_add_mediainfo(evbuf, mfi, -1);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Could not add mediainfo for path '%s'\n", fi->path);
|
|
}
|
|
|
|
free_mfi(mfi, 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
db_query_end(&qp);
|
|
|
|
if (fi)
|
|
free_fi(fi, 0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Command handler function for 'update'
|
|
* Initiates an init-rescan (scans for new files)
|
|
*/
|
|
static int
|
|
mpd_command_update(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
int ret;
|
|
|
|
if (argc > 1 && strlen(argv[1]) > 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Update for specific uri not supported for command 'update'\n");
|
|
ret = asprintf(errmsg, "Update for specific uri not supported for command 'update'");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
filescanner_trigger_initscan();
|
|
|
|
evbuffer_add(evbuf, "updating_db: 1\n", 15);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
static int
|
|
mpd_command_rescan(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
int ret;
|
|
|
|
if (argc > 1)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Rescan for specific uri not supported for command 'rescan'\n");
|
|
ret = asprintf(errmsg, "Rescan for specific uri not supported for command 'rescan'");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
return ACK_ERROR_ARG;
|
|
}
|
|
|
|
filescanner_trigger_fullrescan();
|
|
|
|
evbuffer_add(evbuf, "updating_db: 1\n", 15);
|
|
|
|
return 0;
|
|
}
|
|
*/
|
|
|
|
/*
|
|
static void
|
|
speaker_enum_cb(uint64_t id, const char *name, int relvol, struct spk_flags flags, void *arg)
|
|
{
|
|
/ *
|
|
* outputid: 0
|
|
* outputname: My ALSA Device
|
|
* outputenabled: 0
|
|
* OK
|
|
* /
|
|
struct evbuffer *evbuf;
|
|
|
|
evbuf = (struct evbuffer *)arg;
|
|
|
|
evbuffer_add_printf(evbuf,
|
|
"outputid: %" PRIi64 "\n"
|
|
"outputname: %s\n"
|
|
"outputenabled: %d\n",
|
|
id,
|
|
name,
|
|
flags.selected);
|
|
}
|
|
|
|
static int
|
|
mpd_command_outputs(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
int ret;
|
|
|
|
player_speaker_enumerate(speaker_enum_cb, evbuf);
|
|
|
|
return 0;
|
|
}
|
|
*/
|
|
|
|
/*
|
|
* Dummy function to handle commands that are not supported by forked-daapd and should
|
|
* not raise an error.
|
|
*/
|
|
static int
|
|
mpd_command_ignore(struct evbuffer *evbuf, int argc, char **argv, char **errmsg)
|
|
{
|
|
//do nothing
|
|
DPRINTF(E_DBG, L_MPD, "Ignore command %s\n", argv[0]);
|
|
return 0;
|
|
}
|
|
|
|
struct command
|
|
{
|
|
/* The command name */
|
|
const char *mpdcommand;
|
|
|
|
/*
|
|
* The function to execute the command
|
|
*
|
|
* @param evbuf the response event buffer
|
|
* @param argc number of arguments in argv
|
|
* @param argv argument array, first entry is the commandname
|
|
* @param errmsg error message set by this function if an error occured
|
|
* @return 0 if successful, one of ack values if an error occured
|
|
*/
|
|
int (*handler)(struct evbuffer *evbuf, int argc, char **argv, char **errmsg);
|
|
};
|
|
|
|
static struct command mpd_handlers[] =
|
|
{
|
|
/*
|
|
* Commands for querying status
|
|
*/
|
|
{
|
|
.mpdcommand = "clearerror",
|
|
.handler = mpd_command_ignore
|
|
},
|
|
{
|
|
.mpdcommand = "currentsong",
|
|
.handler = mpd_command_currentsong
|
|
},
|
|
/*
|
|
{
|
|
.mpdcommand = "idle",
|
|
.handler = mpd_command_idle
|
|
},
|
|
*/
|
|
{
|
|
.mpdcommand = "status",
|
|
.handler = mpd_command_status
|
|
},
|
|
{
|
|
.mpdcommand = "stats",
|
|
.handler = mpd_command_stats
|
|
},
|
|
|
|
/*
|
|
* Playback options
|
|
*/
|
|
{
|
|
.mpdcommand = "consume",
|
|
.handler = mpd_command_ignore
|
|
},
|
|
{
|
|
.mpdcommand = "crossfade",
|
|
.handler = mpd_command_ignore
|
|
},
|
|
{
|
|
.mpdcommand = "mixrampdb",
|
|
.handler = mpd_command_ignore
|
|
},
|
|
{
|
|
.mpdcommand = "mixrampdelay",
|
|
.handler = mpd_command_ignore
|
|
},
|
|
{
|
|
.mpdcommand = "random",
|
|
.handler = mpd_command_random
|
|
},
|
|
{
|
|
.mpdcommand = "repeat",
|
|
.handler = mpd_command_repeat
|
|
},
|
|
{
|
|
.mpdcommand = "setvol",
|
|
.handler = mpd_command_setvol
|
|
},
|
|
{
|
|
.mpdcommand = "single",
|
|
.handler = mpd_command_single
|
|
},
|
|
{
|
|
.mpdcommand = "replay_gain_mode",
|
|
.handler = mpd_command_ignore
|
|
},
|
|
{
|
|
.mpdcommand = "replay_gain_status",
|
|
.handler = mpd_command_replay_gain_status
|
|
},
|
|
{
|
|
.mpdcommand = "volume",
|
|
.handler = mpd_command_volume
|
|
},
|
|
|
|
/*
|
|
* Controlling playback
|
|
*/
|
|
{
|
|
.mpdcommand = "next",
|
|
.handler = mpd_command_next
|
|
},
|
|
{
|
|
.mpdcommand = "pause",
|
|
.handler = mpd_command_pause
|
|
},
|
|
{
|
|
.mpdcommand = "play",
|
|
.handler = mpd_command_play
|
|
},
|
|
{
|
|
.mpdcommand = "playid",
|
|
.handler = mpd_command_playid
|
|
},
|
|
{
|
|
.mpdcommand = "previous",
|
|
.handler = mpd_command_previous
|
|
},
|
|
{
|
|
.mpdcommand = "seek",
|
|
.handler = mpd_command_seek
|
|
},
|
|
{
|
|
.mpdcommand = "seekid",
|
|
.handler = mpd_command_seekid
|
|
},
|
|
{
|
|
.mpdcommand = "seekcur",
|
|
.handler = mpd_command_seekcur
|
|
},
|
|
{
|
|
.mpdcommand = "stop",
|
|
.handler = mpd_command_stop
|
|
},
|
|
|
|
/*
|
|
* The current playlist
|
|
*/
|
|
{
|
|
.mpdcommand = "add",
|
|
.handler = mpd_command_add
|
|
},
|
|
{
|
|
.mpdcommand = "addid",
|
|
.handler = mpd_command_addid
|
|
},
|
|
{
|
|
.mpdcommand = "clear",
|
|
.handler = mpd_command_clear
|
|
},
|
|
{
|
|
.mpdcommand = "delete",
|
|
.handler = mpd_command_delete
|
|
},
|
|
{
|
|
.mpdcommand = "deleteid",
|
|
.handler = mpd_command_deleteid
|
|
},
|
|
/*
|
|
{
|
|
.mpdcommand = "move",
|
|
.handler = mpd_command_move
|
|
},
|
|
{
|
|
.mpdcommand = "moveid",
|
|
.handler = mpd_command_moveid
|
|
},
|
|
*/
|
|
// According to the mpd protocol the use of "playlist" is deprecated
|
|
{
|
|
.mpdcommand = "playlist",
|
|
.handler = mpd_command_playlistinfo
|
|
},
|
|
/*
|
|
{
|
|
.mpdcommand = "playlistfind",
|
|
.handler = mpd_command_playlistfind
|
|
},
|
|
{
|
|
.mpdcommand = "playlistid",
|
|
.handler = mpd_command_playlistid
|
|
},
|
|
*/
|
|
{
|
|
.mpdcommand = "playlistinfo",
|
|
.handler = mpd_command_playlistinfo
|
|
},
|
|
/*
|
|
{
|
|
.mpdcommand = "playlistsearch",
|
|
.handler = mpd_command_playlistsearch
|
|
},
|
|
{
|
|
.mpdcommand = "plchanges",
|
|
.handler = mpd_command_plchanges
|
|
},
|
|
{
|
|
.mpdcommand = "plchangesposid",
|
|
.handler = mpd_command_plchangesposid
|
|
},
|
|
{
|
|
.mpdcommand = "prio",
|
|
.handler = mpd_command_prio
|
|
},
|
|
{
|
|
.mpdcommand = "prioid",
|
|
.handler = mpd_command_prioid
|
|
},
|
|
{
|
|
.mpdcommand = "rangeid",
|
|
.handler = mpd_command_rangeid
|
|
},
|
|
{
|
|
.mpdcommand = "shuffle",
|
|
.handler = mpd_command_shuffle
|
|
},
|
|
{
|
|
.mpdcommand = "swap",
|
|
.handler = mpd_command_swap
|
|
},
|
|
{
|
|
.mpdcommand = "swapid",
|
|
.handler = mpd_command_swapid
|
|
},
|
|
{
|
|
.mpdcommand = "addtagid",
|
|
.handler = mpd_command_addtagid
|
|
},
|
|
{
|
|
.mpdcommand = "cleartagid",
|
|
.handler = mpd_command_cleartagid
|
|
},
|
|
*/
|
|
|
|
/*
|
|
* Stored playlists
|
|
*/
|
|
/*
|
|
{
|
|
.mpdcommand = "listplaylist",
|
|
.handler = mpd_command_listplaylist
|
|
},
|
|
{
|
|
.mpdcommand = "listplaylistinfo",
|
|
.handler = mpd_command_listplaylistinfo
|
|
},
|
|
{
|
|
.mpdcommand = "listplaylists",
|
|
.handler = mpd_command_listplaylists
|
|
},load
|
|
*/
|
|
{
|
|
.mpdcommand = "load",
|
|
.handler = mpd_command_load
|
|
},
|
|
/*
|
|
{
|
|
.mpdcommand = "playlistadd",
|
|
.handler = mpd_command_playlistadd
|
|
},
|
|
{
|
|
.mpdcommand = "playlistclear",
|
|
.handler = mpd_command_playlistclear
|
|
},
|
|
{
|
|
.mpdcommand = "playlistdelete",
|
|
.handler = mpd_command_playlistdelete
|
|
},
|
|
{
|
|
.mpdcommand = "playlistmove",
|
|
.handler = mpd_command_playlistmove
|
|
},
|
|
{
|
|
.mpdcommand = "rename",
|
|
.handler = mpd_command_rename
|
|
},
|
|
{
|
|
.mpdcommand = "rm",
|
|
.handler = mpd_command_rm
|
|
},
|
|
{
|
|
.mpdcommand = "save",
|
|
.handler = mpd_command_save
|
|
},
|
|
*/
|
|
|
|
/*
|
|
* The music database
|
|
*/
|
|
/*
|
|
{
|
|
.mpdcommand = "count",
|
|
.handler = mpd_command_count
|
|
},
|
|
{
|
|
.mpdcommand = "find",
|
|
.handler = mpd_command_find
|
|
},
|
|
{
|
|
.mpdcommand = "findadd",
|
|
.handler = mpd_command_findadd
|
|
},
|
|
{
|
|
.mpdcommand = "list",
|
|
.handler = mpd_command_list
|
|
},
|
|
{
|
|
.mpdcommand = "listall",
|
|
.handler = mpd_command_listall
|
|
},
|
|
{
|
|
.mpdcommand = "listallinfo",
|
|
.handler = mpd_command_listallinfo
|
|
},
|
|
{
|
|
.mpdcommand = "listfiles",
|
|
.handler = mpd_command_listfiles
|
|
},
|
|
*/
|
|
{
|
|
.mpdcommand = "lsinfo",
|
|
.handler = mpd_command_lsinfo
|
|
},
|
|
/*
|
|
{
|
|
.mpdcommand = "readcomments",
|
|
.handler = mpd_command_readcomments
|
|
},
|
|
{
|
|
.mpdcommand = "search",
|
|
.handler = mpd_command_search
|
|
},
|
|
{
|
|
.mpdcommand = "searchadd",
|
|
.handler = mpd_command_searchadd
|
|
},
|
|
{
|
|
.mpdcommand = "searchaddpl",
|
|
.handler = mpd_command_searchaddpl
|
|
},
|
|
*/
|
|
{
|
|
.mpdcommand = "update",
|
|
.handler = mpd_command_update
|
|
},
|
|
/*
|
|
{
|
|
.mpdcommand = "rescan",
|
|
.handler = mpd_command_rescan
|
|
},
|
|
*/
|
|
|
|
/*
|
|
* Mounts and neighbors
|
|
*/
|
|
/*
|
|
{
|
|
.mpdcommand = "mount",
|
|
.handler = mpd_command_mount
|
|
},
|
|
{
|
|
.mpdcommand = "unmount",
|
|
.handler = mpd_command_unmount
|
|
},
|
|
{
|
|
.mpdcommand = "listmounts",
|
|
.handler = mpd_command_listmounts
|
|
},
|
|
{
|
|
.mpdcommand = "listneighbors",
|
|
.handler = mpd_command_listneighbors
|
|
},
|
|
*/
|
|
|
|
/*
|
|
* Stickers
|
|
*/
|
|
/*
|
|
{
|
|
.mpdcommand = "sticker",
|
|
.handler = mpd_command_sticker
|
|
},
|
|
*/
|
|
|
|
/*
|
|
* Connection settings
|
|
*/
|
|
/*
|
|
{
|
|
.mpdcommand = "close",
|
|
.handler = mpd_command_close
|
|
},
|
|
{
|
|
.mpdcommand = "kill",
|
|
.handler = mpd_command_kill
|
|
},
|
|
{
|
|
.mpdcommand = "password",
|
|
.handler = mpd_command_password
|
|
},
|
|
{
|
|
.mpdcommand = "ping",
|
|
.handler = mpd_command_ping
|
|
},
|
|
*/
|
|
|
|
/*
|
|
* Audio output devices
|
|
*/
|
|
/*
|
|
{
|
|
.mpdcommand = "disableoutput",
|
|
.handler = mpd_command_disableoutput
|
|
},
|
|
{
|
|
.mpdcommand = "enableoutput",
|
|
.handler = mpd_command_enableoutput
|
|
},
|
|
{
|
|
.mpdcommand = "toggleoutput",
|
|
.handler = mpd_command_toggleoutput
|
|
},
|
|
{
|
|
.mpdcommand = "outputs",
|
|
.handler = mpd_command_outputs
|
|
},
|
|
*/
|
|
|
|
/*
|
|
* Reflection
|
|
*/
|
|
/*
|
|
{
|
|
.mpdcommand = "config",
|
|
.handler = mpd_command_config
|
|
},
|
|
{
|
|
.mpdcommand = "commands",
|
|
.handler = mpd_command_commands
|
|
},
|
|
{
|
|
.mpdcommand = "notcommands",
|
|
.handler = mpd_command_notcommands
|
|
},
|
|
{
|
|
.mpdcommand = "tagtypes",
|
|
.handler = mpd_command_tagtypes
|
|
},
|
|
{
|
|
.mpdcommand = "urlhandlers",
|
|
.handler = mpd_command_urlhandlers
|
|
},
|
|
{
|
|
.mpdcommand = "decoders",
|
|
.handler = mpd_command_decoders
|
|
},
|
|
*/
|
|
|
|
/*
|
|
* Client to client
|
|
*/
|
|
/*
|
|
{
|
|
.mpdcommand = "subscribe",
|
|
.handler = mpd_command_subscribe
|
|
},
|
|
{
|
|
.mpdcommand = "unsubscribe",
|
|
.handler = mpd_command_unsubscribe
|
|
},
|
|
{
|
|
.mpdcommand = "channels",
|
|
.handler = mpd_command_channels
|
|
},
|
|
{
|
|
.mpdcommand = "readmessages",
|
|
.handler = mpd_command_readmessages
|
|
},
|
|
{
|
|
.mpdcommand = "sendmessage",
|
|
.handler = mpd_command_sendmessage
|
|
},
|
|
*/
|
|
|
|
/*
|
|
* NULL command to terminate loop
|
|
*/
|
|
{
|
|
.mpdcommand = NULL,
|
|
.handler = NULL
|
|
}
|
|
};
|
|
|
|
/*
|
|
* Finds the command handler for the given command name
|
|
*
|
|
* @param name the name of the command
|
|
* @return the command or NULL if it is an unknown/unsupported command
|
|
*/
|
|
static struct command*
|
|
mpd_find_command(const char *name)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; mpd_handlers[i].handler; i++)
|
|
{
|
|
if (0 == strcmp(name, mpd_handlers[i].mpdcommand))
|
|
{
|
|
return &mpd_handlers[i];
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* The read callback function is invoked if a complete command sequence was received from the client
|
|
* (see mpd_input_filter function).
|
|
*
|
|
* @param bev the buffer event
|
|
* @param ctx (not used)
|
|
*/
|
|
static void
|
|
mpd_read_cb(struct bufferevent *bev, void *ctx)
|
|
{
|
|
struct evbuffer *input;
|
|
struct evbuffer *output;
|
|
int ret;
|
|
int ncmd;
|
|
char *line;
|
|
char *errmsg;
|
|
struct command *command;
|
|
enum command_list_type listtype;
|
|
char *argv[COMMAND_ARGV_MAX];
|
|
int argc;
|
|
|
|
/* Get the input evbuffer, contains the command sequence received from the client */
|
|
input = bufferevent_get_input(bev);
|
|
/* Get the output evbuffer, used to send the server response to the client */
|
|
output = bufferevent_get_output(bev);
|
|
|
|
DPRINTF(E_SPAM, L_MPD, "Received MPD command sequence\n");
|
|
|
|
listtype = COMMAND_LIST_NONE;
|
|
ncmd = 0;
|
|
|
|
while ((line = evbuffer_readln(input, NULL, EVBUFFER_EOL_ANY)))
|
|
{
|
|
DPRINTF(E_SPAM, L_MPD, "MPD message: %s\n", line);
|
|
|
|
// Split the read line into command name and arguments
|
|
ret = mpd_parse_args(line, &argc, argv);
|
|
if (ret != 0)
|
|
{
|
|
// Error handling for argument parsing error
|
|
DPRINTF(E_LOG, L_MPD, "Error parsing arguments for MPD message: %s\n", line);
|
|
ret = asprintf(&errmsg, "Error parsing arguments");
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
ret = ACK_ERROR_ARG;
|
|
evbuffer_add_printf(output, "ACK [%d@%d] {%s} %s\n", ret, ncmd, "unkown", errmsg);
|
|
free(errmsg);
|
|
free(line);
|
|
break;
|
|
}
|
|
|
|
/*
|
|
* Check if it is a list command
|
|
*/
|
|
if (0 == strcmp(argv[0], "command_list_ok_begin"))
|
|
{
|
|
listtype = COMMAND_LIST_OK;
|
|
free(line);
|
|
continue;
|
|
}
|
|
else if (0 == strcmp(argv[0], "command_list_begin"))
|
|
{
|
|
listtype = COMMAND_LIST;
|
|
free(line);
|
|
continue;
|
|
}
|
|
else if (0 == strcmp(argv[0], "command_list_end"))
|
|
{
|
|
free(line);
|
|
break;
|
|
}
|
|
|
|
/*
|
|
* Find the command handler and execute the command function
|
|
*/
|
|
command = mpd_find_command(argv[0]);
|
|
|
|
if (command == NULL)
|
|
{
|
|
ret = asprintf(&errmsg, "Unsupported command '%s'", argv[0]);
|
|
if (ret < 0)
|
|
DPRINTF(E_LOG, L_MPD, "Out of memory\n");
|
|
ret = ACK_ERROR_UNKNOWN;
|
|
}
|
|
else
|
|
ret = command->handler(output, argc, argv, &errmsg);
|
|
|
|
/*
|
|
* If an error occurred, add the ACK line to the response buffer and exit the loop
|
|
*/
|
|
if (ret != 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Error executing command '%s': %s\n", argv[0], errmsg);
|
|
evbuffer_add_printf(output, "ACK [%d@%d] {%s} %s\n", ret, ncmd, argv[0], errmsg);
|
|
free(errmsg);
|
|
free(line);
|
|
break;
|
|
}
|
|
|
|
/*
|
|
* If the command sequence started with command_list_ok_begin, add a list_ok line to the
|
|
* response buffer after each command output.
|
|
*/
|
|
if (listtype == COMMAND_LIST_OK)
|
|
{
|
|
evbuffer_add(output, "list_OK\n", 8);
|
|
}
|
|
|
|
free(line);
|
|
ncmd++;
|
|
}
|
|
|
|
DPRINTF(E_SPAM, L_MPD, "Finished MPD command sequence: %d\n", ret);
|
|
|
|
/*
|
|
* If everything was successful add OK line to signal clients end of message.
|
|
* If an error occured the necessary ACK line should already be added to the response buffer.
|
|
*/
|
|
if (ret == 0)
|
|
{
|
|
evbuffer_add(output, "OK\n", 3);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Callback when an event occurs on the bufferevent
|
|
*/
|
|
static void
|
|
mpd_event_cb(struct bufferevent *bev, short events, void *ctx)
|
|
{
|
|
if (events & BEV_EVENT_ERROR)
|
|
DPRINTF(E_LOG, L_MPD, "Error from buffer event\n");
|
|
|
|
if (events & (BEV_EVENT_EOF | BEV_EVENT_ERROR))
|
|
bufferevent_free(bev);
|
|
}
|
|
|
|
/*
|
|
* The input filter buffer callback checks if the data received from the client is a complete command sequence.
|
|
* A command sequence has end with '\n' and if it starts with "command_list_begin\n" or "command_list_ok_begin\n"
|
|
* the last line has to be "command_list_end\n".
|
|
*
|
|
* @param src evbuffer to read data from (contains the data received from the client)
|
|
* @param dst evbuffer to write data to (this is the evbuffer for the read callback)
|
|
* @param lim the upper bound of bytes to add to destination
|
|
* @param state write mode
|
|
* @param ctx (not used)
|
|
* @return BEV_OK if a complete command sequence was received otherwise BEV_NEED_MORE
|
|
*/
|
|
static enum bufferevent_filter_result
|
|
mpd_input_filter(struct evbuffer *src, struct evbuffer *dst, ev_ssize_t lim, enum bufferevent_flush_mode state, void *ctx)
|
|
{
|
|
struct evbuffer_ptr p;
|
|
char *line;
|
|
int ret;
|
|
|
|
while ((line = evbuffer_readln(src, NULL, EVBUFFER_EOL_ANY)))
|
|
{
|
|
ret = evbuffer_add_printf(dst, "%s\n", line);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Error adding line to buffer: '%s'\n", line);
|
|
free(line);
|
|
return BEV_ERROR;
|
|
}
|
|
free(line);
|
|
}
|
|
|
|
if (evbuffer_get_length(src) > 0)
|
|
{
|
|
DPRINTF(E_DBG, L_MPD, "Message incomplete, waiting for more data\n");
|
|
return BEV_NEED_MORE;
|
|
}
|
|
|
|
p = evbuffer_search(dst, "command_list_begin", 18, NULL);
|
|
if (p.pos < 0)
|
|
{
|
|
p = evbuffer_search(dst, "command_list_ok_begin", 21, NULL);
|
|
}
|
|
|
|
if (p.pos >= 0)
|
|
{
|
|
p = evbuffer_search(dst, "command_list_end", 16, NULL);
|
|
if (p.pos < 0)
|
|
{
|
|
DPRINTF(E_DBG, L_MPD, "Message incomplete (missing command_list_end), waiting for more data\n");
|
|
return BEV_NEED_MORE;
|
|
}
|
|
}
|
|
|
|
return BEV_OK;
|
|
}
|
|
|
|
/*
|
|
* The connection listener callback function is invoked when a new connection was received.
|
|
*
|
|
* @param listener the connection listener that received the connection
|
|
* @param sock the new socket
|
|
* @param address the address from which the connection was received
|
|
* @param socklen the length of that address
|
|
* @param ctx (not used)
|
|
*/
|
|
static void
|
|
mpd_accept_conn_cb(struct evconnlistener *listener,
|
|
evutil_socket_t sock, struct sockaddr *address, int socklen,
|
|
void *ctx)
|
|
{
|
|
/*
|
|
* For each new connection setup a new buffer event and wrap it around a filter event.
|
|
* The filter event ensures, that the read callback on the buffer event is only invoked if a complete
|
|
* command sequence from the client was received.
|
|
*/
|
|
struct event_base *base = evconnlistener_get_base(listener);
|
|
struct bufferevent *bev = bufferevent_socket_new(base, sock, BEV_OPT_CLOSE_ON_FREE);
|
|
|
|
bev = bufferevent_filter_new(bev, mpd_input_filter, NULL, BEV_OPT_CLOSE_ON_FREE, NULL, NULL);
|
|
bufferevent_setcb(bev, mpd_read_cb, NULL, mpd_event_cb, bev);
|
|
bufferevent_enable(bev, EV_READ | EV_WRITE);
|
|
|
|
/*
|
|
* According to the mpd protocol send "OK MPD <version>\n" to the client, where version is the version
|
|
* of the supported mpd protocol and not the server version.
|
|
*/
|
|
evbuffer_add(bufferevent_get_output(bev), "OK MPD 0.18.0\n", 14);
|
|
}
|
|
|
|
/*
|
|
* Error callback that gets called whenever an accept() call fails on the listener
|
|
* @param listener the connection listener that received the connection
|
|
* @param ctx (not used)
|
|
*/
|
|
static void
|
|
mpd_accept_error_cb(struct evconnlistener *listener, void *ctx)
|
|
{
|
|
int err;
|
|
|
|
err = EVUTIL_SOCKET_ERROR();
|
|
DPRINTF(E_LOG, L_MPD, "Error occured %d (%s) on the listener.\n", err, evutil_socket_error_to_string(err));
|
|
}
|
|
|
|
|
|
/* Thread: main */
|
|
int mpd_init(void)
|
|
{
|
|
struct evconnlistener *listener;
|
|
struct sockaddr_in sin;
|
|
unsigned short port;
|
|
int ret;
|
|
|
|
port = cfg_getint(cfg_getsec(cfg, "mpd"), "port");
|
|
if (port <= 0)
|
|
{
|
|
DPRINTF(E_INFO, L_MPD, "MPD not enabled\n");
|
|
return 0;
|
|
}
|
|
|
|
# if defined(__linux__)
|
|
ret = pipe2(g_exit_pipe, O_CLOEXEC);
|
|
# else
|
|
ret = pipe(g_exit_pipe);
|
|
# endif
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Could not create pipe: %s\n", strerror(errno));
|
|
goto exit_fail;
|
|
}
|
|
|
|
evbase_mpd = event_base_new();
|
|
if (!evbase_mpd)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Could not create an event base\n");
|
|
goto evbase_fail;
|
|
}
|
|
|
|
g_exitev = event_new(evbase_mpd, g_exit_pipe[0], EV_READ, exit_cb, NULL);
|
|
if (!g_exitev)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Could not create exit event\n");
|
|
goto evnew_fail;
|
|
}
|
|
|
|
event_add(g_exitev, NULL);
|
|
|
|
//TODO ipv6
|
|
memset(&sin, 0, sizeof(sin));
|
|
sin.sin_family = AF_INET;
|
|
sin.sin_addr.s_addr = htonl(0);
|
|
sin.sin_port = htons(port);
|
|
|
|
listener = evconnlistener_new_bind(
|
|
evbase_mpd,
|
|
mpd_accept_conn_cb,
|
|
NULL,
|
|
LEV_OPT_CLOSE_ON_FREE | LEV_OPT_REUSEABLE,
|
|
-1,
|
|
(struct sockaddr*) &sin,
|
|
sizeof(sin));
|
|
|
|
if (!listener)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Could not create connection listener for mpd clients on port %d\n", port);
|
|
|
|
goto connew_fail;
|
|
}
|
|
evconnlistener_set_error_cb(listener, mpd_accept_error_cb);
|
|
|
|
DPRINTF(E_INFO, L_MPD, "cache thread init\n");
|
|
|
|
ret = pthread_create(&tid_mpd, NULL, mpd, NULL);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MPD, "Could not spawn cache thread: %s\n", strerror(errno));
|
|
|
|
goto thread_fail;
|
|
}
|
|
|
|
return 0;
|
|
|
|
|
|
thread_fail:
|
|
connew_fail:
|
|
evnew_fail:
|
|
event_base_free(evbase_mpd);
|
|
evbase_mpd = NULL;
|
|
|
|
evbase_fail:
|
|
close(g_exit_pipe[0]);
|
|
close(g_exit_pipe[1]);
|
|
|
|
exit_fail:
|
|
return -1;
|
|
}
|
|
|
|
/* Thread: main */
|
|
void mpd_deinit(void)
|
|
{
|
|
unsigned short port;
|
|
int ret;
|
|
|
|
port = cfg_getint(cfg_getsec(cfg, "mpd"), "port");
|
|
if (port <= 0)
|
|
{
|
|
DPRINTF(E_INFO, L_MPD, "MPD not enabled\n");
|
|
return;
|
|
}
|
|
|
|
thread_exit();
|
|
|
|
ret = pthread_join(tid_mpd, NULL);
|
|
if (ret != 0)
|
|
{
|
|
DPRINTF(E_FATAL, L_MPD, "Could not join cache thread: %s\n", strerror(errno));
|
|
return;
|
|
}
|
|
|
|
// Free event base (should free events too)
|
|
event_base_free(evbase_mpd);
|
|
|
|
// Close pipes
|
|
close(g_exit_pipe[0]);
|
|
close(g_exit_pipe[1]);
|
|
}
|