command: added command "idle"

"idle" waits until something noteworthy happens on the server,
e.g. song change, playlist modified, database updated.  This allows
clients to keep up to date without polling.
This commit is contained in:
Max Kellermann 2008-10-14 22:38:14 +02:00
parent 30c86d8ae6
commit a3e3d2c950
14 changed files with 305 additions and 3 deletions

View File

@ -265,6 +265,27 @@ volume <int change>
change volume by amount _change_ change volume by amount _change_
NOTE: volume command is deprecated, use setvol instead NOTE: volume command is deprecated, use setvol instead
idle
Waits until there is a noteworthy change in one or more of
MPD's subsystems. As soon as there is one, it lists all
changed systems in a line in the format "changed: SUBSYSTEM",
where SUBSYSTEM is one of the following:
database: the song database has been updated
stored_playlist: a stored playlist has been modified, renamed,
created or deleted
playlist: the current playlist has been modified
player: the player has been started, stopped or seeked
mixer: the volume has been changed
output: an audio output has been enabled or disabled
options: options like "repeat", "random", "crossfade"
While a client waits for "idle" results, the server disables
timeouts, allowing a client to wait for events as long as mpd
runs. The "idle" command can be canceled by sending a new
command.
COMMAND LIST COMMAND LIST
------------ ------------

View File

@ -45,6 +45,7 @@ mpd_headers = \
buffer2array.h \ buffer2array.h \
charConv.h \ charConv.h \
command.h \ command.h \
idle.h \
condition.h \ condition.h \
conf.h \ conf.h \
crossfade.h \ crossfade.h \
@ -130,6 +131,7 @@ mpd_SOURCES = \
buffer2array.c \ buffer2array.c \
charConv.c \ charConv.c \
command.c \ command.c \
idle.c \
condition.c \ condition.c \
conf.c \ conf.c \
crossfade.c \ crossfade.c \

View File

@ -24,6 +24,7 @@
#include "log.h" #include "log.h"
#include "path.h" #include "path.h"
#include "client.h" #include "client.h"
#include "idle.h"
#include "utils.h" #include "utils.h"
#include "os_compat.h" #include "os_compat.h"
@ -379,6 +380,7 @@ int enableAudioDevice(unsigned int device)
return -1; return -1;
audioDeviceStates[device] = true; audioDeviceStates[device] = true;
idle_add(IDLE_OUTPUT);
return 0; return 0;
} }
@ -389,6 +391,7 @@ int disableAudioDevice(unsigned int device)
return -1; return -1;
audioDeviceStates[device] = false; audioDeviceStates[device] = false;
idle_add(IDLE_OUTPUT);
return 0; return 0;
} }

View File

@ -27,6 +27,7 @@
#include "ioops.h" #include "ioops.h"
#include "main_notify.h" #include "main_notify.h"
#include "dlist.h" #include "dlist.h"
#include "idle.h"
#include "../config.h" #include "../config.h"
@ -87,6 +88,13 @@ struct client {
size_t send_buf_used; /* bytes used this instance */ size_t send_buf_used; /* bytes used this instance */
size_t send_buf_size; /* bytes usable this instance */ size_t send_buf_size; /* bytes usable this instance */
size_t send_buf_alloc; /* bytes actually allocated */ size_t send_buf_alloc; /* bytes actually allocated */
/** is this client waiting for an "idle" response? */
bool idle_waiting;
/** idle flags pending on this client, to be sent as soon as
the client enters "idle" */
unsigned idle_flags;
}; };
static LIST_HEAD(clients); static LIST_HEAD(clients);
@ -409,6 +417,9 @@ static int client_input_received(struct client *client, int bytesRead)
int ret; int ret;
char *buf_tail = &(client->buffer[client->bufferLength - 1]); char *buf_tail = &(client->buffer[client->bufferLength - 1]);
/* any input from the client makes it leave "idle" mode */
client->idle_waiting = false;
while (bytesRead > 0) { while (bytesRead > 0) {
client->bufferLength++; client->bufferLength++;
bytesRead--; bytesRead--;
@ -635,7 +646,9 @@ void client_manager_expire(void)
if (client_is_expired(client)) { if (client_is_expired(client)) {
DEBUG("client %i: expired\n", client->num); DEBUG("client %i: expired\n", client->num);
client_close(client); client_close(client);
} else if (time(NULL) - client->lastTime > } else if (!client->idle_waiting && /* idle clients
never expire */
time(NULL) - client->lastTime >
client_timeout) { client_timeout) {
DEBUG("client %i: timeout\n", client->num); DEBUG("client %i: timeout\n", client->num);
client_close(client); client_close(client);
@ -807,3 +820,71 @@ mpd_fprintf void client_printf(struct client *client, const char *fmt, ...)
client_vprintf(client, fmt, args); client_vprintf(client, fmt, args);
va_end(args); va_end(args);
} }
static const char *const idle_names[] = {
"database",
"stored_playlist",
"playlist",
"player",
"mixer",
"output",
"options",
};
/**
* Send "idle" response to this client.
*/
static void
client_idle_notify(struct client *client)
{
unsigned flags, i;
assert(client->idle_waiting);
assert(client->idle_flags != 0);
flags = client->idle_flags;
client->idle_flags = 0;
client->idle_waiting = false;
for (i = 0; i < sizeof(idle_names) / sizeof(idle_names[0]); ++i) {
assert(idle_names[i] != NULL);
if (flags & (1 << i))
client_printf(client, "changed: %s\n",
idle_names[i]);
}
client_puts(client, "OK\n");
client->lastTime = time(NULL);
}
void client_manager_idle_add(unsigned flags)
{
struct client *client;
assert(flags != 0);
list_for_each_entry(client, &clients, siblings) {
if (client_is_expired(client))
continue;
client->idle_flags |= flags;
if (client->idle_waiting) {
client_idle_notify(client);
client_write_output(client);
}
}
}
bool client_idle_wait(struct client *client)
{
assert(!client->idle_waiting);
client->idle_waiting = true;
if (client->idle_flags != 0) {
client_idle_notify(client);
return true;
} else
return false;
}

View File

@ -21,6 +21,7 @@
#include "gcc.h" #include "gcc.h"
#include <stdbool.h>
#include <stddef.h> #include <stddef.h>
#include <stdarg.h> #include <stdarg.h>
#include <sys/socket.h> #include <sys/socket.h>
@ -60,4 +61,17 @@ void client_vprintf(struct client *client, const char *fmt, va_list args);
*/ */
mpd_fprintf void client_printf(struct client *client, const char *fmt, ...); mpd_fprintf void client_printf(struct client *client, const char *fmt, ...);
/**
* Adds the specified idle flags to all clients and immediately sends
* notifications to all waiting clients.
*/
void client_manager_idle_add(unsigned flags);
/**
* Checks whether the client has pending idle flags. If yes, they are
* sent immediately and "true" is returned". If no, it puts the
* client into waiting mode and returns false.
*/
bool client_idle_wait(struct client *client);
#endif #endif

View File

@ -1239,6 +1239,17 @@ static int handlePlaylistAdd(struct client *client,
return print_playlist_result(client, result); return print_playlist_result(client, result);
} }
static int
handle_idle(struct client *client,
mpd_unused int argc, mpd_unused char *argv[])
{
/* enable "idle" mode on this client */
client_idle_wait(client);
/* return value is "1" so the caller won't print "OK" */
return 1;
}
void initCommands(void) void initCommands(void)
{ {
commandList = makeList(free, 1); commandList = makeList(free, 1);
@ -1307,6 +1318,7 @@ void initCommands(void)
addCommand(COMMAND_TAGTYPES, PERMISSION_READ, 0, 0, handleTagTypes); addCommand(COMMAND_TAGTYPES, PERMISSION_READ, 0, 0, handleTagTypes);
addCommand(COMMAND_COUNT, PERMISSION_READ, 2, -1, handleCount); addCommand(COMMAND_COUNT, PERMISSION_READ, 2, -1, handleCount);
addCommand(COMMAND_RENAME, PERMISSION_CONTROL, 2, 2, handleRename); addCommand(COMMAND_RENAME, PERMISSION_CONTROL, 2, 2, handleRename);
addCommand("idle", PERMISSION_READ, 0, 0, handle_idle);
sortList(commandList); sortList(commandList);
} }

56
src/idle.c Normal file
View File

@ -0,0 +1,56 @@
/* the Music Player Daemon (MPD)
* Copyright (C) 2008 Max Kellermann <max@duempel.org>
* This project's homepage is: http://www.musicpd.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
*/
/*
* Support library for the "idle" command.
*
*/
#include "idle.h"
#include "main_notify.h"
#include <assert.h>
#include <pthread.h>
static unsigned idle_flags;
static pthread_mutex_t idle_mutex = PTHREAD_MUTEX_INITIALIZER;
void
idle_add(unsigned flags)
{
assert(flags != 0);
pthread_mutex_lock(&idle_mutex);
idle_flags |= flags;
pthread_mutex_unlock(&idle_mutex);
wakeup_main_task();
}
unsigned
idle_get(void)
{
unsigned flags;
pthread_mutex_lock(&idle_mutex);
flags = idle_flags;
idle_flags = 0;
pthread_mutex_unlock(&idle_mutex);
return flags;
}

64
src/idle.h Normal file
View File

@ -0,0 +1,64 @@
/* the Music Player Daemon (MPD)
* Copyright (C) 2008 Max Kellermann <max@duempel.org>
* This project's homepage is: http://www.musicpd.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
*/
/*
* Support library for the "idle" command.
*
*/
#ifndef MPD_IDLE_H
#define MPD_IDLE_H
enum {
/** song database has been updated*/
IDLE_DATABASE = 0x1,
/** a stored playlist has been modified, created, deleted or
renamed */
IDLE_STORED_PLAYLIST = 0x2,
/** the current playlist has been modified */
IDLE_PLAYLIST = 0x4,
/** the player state has changed: play, stop, pause, seek, ... */
IDLE_PLAYER = 0x8,
/** the volume has been modified */
IDLE_MIXER = 0x10,
/** an audio output device has been enabled or disabled */
IDLE_OUTPUT = 0x20,
/** options have changed: crossfade, random, repeat, ... */
IDLE_OPTIONS = 0x40,
};
/**
* Adds idle flag (with bitwise "or") and queues notifications to all
* clients.
*/
void
idle_add(unsigned flags);
/**
* Atomically reads and resets the global idle flags value.
*/
unsigned
idle_get(void);
#endif

View File

@ -17,6 +17,7 @@
*/ */
#include "client.h" #include "client.h"
#include "idle.h"
#include "command.h" #include "command.h"
#include "playlist.h" #include "playlist.h"
#include "database.h" #include "database.h"
@ -445,9 +446,17 @@ int main(int argc, char *argv[])
while (COMMAND_RETURN_KILL != client_manager_io() && while (COMMAND_RETURN_KILL != client_manager_io() &&
COMMAND_RETURN_KILL != handlePendingSignals()) { COMMAND_RETURN_KILL != handlePendingSignals()) {
unsigned flags;
syncPlayerAndPlaylist(); syncPlayerAndPlaylist();
client_manager_expire(); client_manager_expire();
reap_update_task(); reap_update_task();
/* send "idle" notificaions to all subscribed
clients */
flags = idle_get();
if (flags != 0)
client_manager_idle_add(flags);
} }
write_state_file(); write_state_file();

View File

@ -21,6 +21,7 @@
#include "log.h" #include "log.h"
#include "tag.h" #include "tag.h"
#include "song.h" #include "song.h"
#include "idle.h"
#include "os_compat.h" #include "os_compat.h"
#include "main_notify.h" #include "main_notify.h"
@ -61,6 +62,8 @@ playerPlay(struct song *song)
pc.next_song = song; pc.next_song = song;
player_command(PLAYER_COMMAND_PLAY); player_command(PLAYER_COMMAND_PLAY);
idle_add(IDLE_PLAYER);
} }
void pc_cancel(void) void pc_cancel(void)
@ -71,17 +74,23 @@ void pc_cancel(void)
void playerWait(void) void playerWait(void)
{ {
player_command(PLAYER_COMMAND_CLOSE_AUDIO); player_command(PLAYER_COMMAND_CLOSE_AUDIO);
idle_add(IDLE_PLAYER);
} }
void playerKill(void) void playerKill(void)
{ {
player_command(PLAYER_COMMAND_EXIT); player_command(PLAYER_COMMAND_EXIT);
idle_add(IDLE_PLAYER);
} }
void playerPause(void) void playerPause(void)
{ {
if (pc.state != PLAYER_STATE_STOP) if (pc.state != PLAYER_STATE_STOP) {
player_command(PLAYER_COMMAND_PAUSE); player_command(PLAYER_COMMAND_PAUSE);
idle_add(IDLE_PLAYER);
}
} }
void playerSetPause(int pause_flag) void playerSetPause(int pause_flag)
@ -185,6 +194,8 @@ playerSeek(struct song *song, float seek_time)
if (pc.error == PLAYER_ERROR_NOERROR) { if (pc.error == PLAYER_ERROR_NOERROR) {
pc.seekWhere = seek_time; pc.seekWhere = seek_time;
player_command(PLAYER_COMMAND_SEEK); player_command(PLAYER_COMMAND_SEEK);
idle_add(IDLE_PLAYER);
} }
return 0; return 0;
@ -200,6 +211,8 @@ void setPlayerCrossFade(float crossFadeInSeconds)
if (crossFadeInSeconds < 0) if (crossFadeInSeconds < 0)
crossFadeInSeconds = 0; crossFadeInSeconds = 0;
pc.crossFade = crossFadeInSeconds; pc.crossFade = crossFadeInSeconds;
idle_add(IDLE_OPTIONS);
} }
void setPlayerSoftwareVolume(int volume) void setPlayerSoftwareVolume(int volume)

View File

@ -34,6 +34,7 @@
#include "state_file.h" #include "state_file.h"
#include "storedPlaylist.h" #include "storedPlaylist.h"
#include "ack.h" #include "ack.h"
#include "idle.h"
#include "os_compat.h" #include "os_compat.h"
#define PLAYLIST_STATE_STOP 0 #define PLAYLIST_STATE_STOP 0
@ -87,6 +88,8 @@ static void incrPlaylistVersion(void)
playlist.version = 1; playlist.version = 1;
} }
idle_add(IDLE_PLAYLIST);
} }
void playlistVersionChange(void) void playlistVersionChange(void)
@ -504,6 +507,8 @@ static void syncPlaylistWithQueue(void)
if (pc.next_song == NULL && playlist.queued != -1) { if (pc.next_song == NULL && playlist.queued != -1) {
playlist.current = playlist.queued; playlist.current = playlist.queued;
playlist.queued = -1; playlist.queued = -1;
idle_add(IDLE_PLAYER);
} }
} }
@ -951,6 +956,8 @@ void setPlaylistRepeatStatus(bool status)
clearPlayerQueue(); clearPlayerQueue();
playlist.repeat = status; playlist.repeat = status;
idle_add(IDLE_OPTIONS);
} }
enum playlist_result moveSongInPlaylist(int from, int to) enum playlist_result moveSongInPlaylist(int from, int to)
@ -1123,6 +1130,8 @@ void setPlaylistRandomStatus(bool status)
} }
} else } else
orderPlaylist(); orderPlaylist();
idle_add(IDLE_OPTIONS);
} }
void previousSongInPlaylist(void) void previousSongInPlaylist(void)
@ -1219,6 +1228,7 @@ enum playlist_result savePlaylist(const char *utf8file)
while (fclose(fp) && errno == EINTR) ; while (fclose(fp) && errno == EINTR) ;
idle_add(IDLE_STORED_PLAYLIST);
return PLAYLIST_RESULT_SUCCESS; return PLAYLIST_RESULT_SUCCESS;
} }

View File

@ -24,6 +24,7 @@
#include "utils.h" #include "utils.h"
#include "ls.h" #include "ls.h"
#include "database.h" #include "database.h"
#include "idle.h"
#include "os_compat.h" #include "os_compat.h"
static ListNode *nodeOfStoredPlaylist(List *list, int idx) static ListNode *nodeOfStoredPlaylist(List *list, int idx)
@ -192,6 +193,7 @@ static int moveSongInStoredPlaylist(List *list, int src, int dest)
} }
} }
idle_add(IDLE_STORED_PLAYLIST);
return 0; return 0;
} }
@ -212,6 +214,8 @@ moveSongInStoredPlaylistByPath(const char *utf8path, int src, int dest)
result = writeStoredPlaylistToPath(list, utf8path); result = writeStoredPlaylistToPath(list, utf8path);
freeList(list); freeList(list);
idle_add(IDLE_STORED_PLAYLIST);
return result; return result;
} }
@ -231,6 +235,8 @@ removeAllFromStoredPlaylistByPath(const char *utf8path)
return PLAYLIST_RESULT_ERRNO; return PLAYLIST_RESULT_ERRNO;
while (fclose(file) != 0 && errno == EINTR); while (fclose(file) != 0 && errno == EINTR);
idle_add(IDLE_STORED_PLAYLIST);
return PLAYLIST_RESULT_SUCCESS; return PLAYLIST_RESULT_SUCCESS;
} }
@ -262,6 +268,8 @@ removeOneSongFromStoredPlaylistByPath(const char *utf8path, int pos)
result = writeStoredPlaylistToPath(list, utf8path); result = writeStoredPlaylistToPath(list, utf8path);
freeList(list); freeList(list);
idle_add(IDLE_STORED_PLAYLIST);
return result; return result;
} }
@ -299,6 +307,8 @@ appendSongToStoredPlaylistByPath(const char *utf8path, struct song *song)
playlist_print_song(file, song); playlist_print_song(file, song);
while (fclose(file) != 0 && errno == EINTR); while (fclose(file) != 0 && errno == EINTR);
idle_add(IDLE_STORED_PLAYLIST);
return PLAYLIST_RESULT_SUCCESS; return PLAYLIST_RESULT_SUCCESS;
} }
@ -325,5 +335,6 @@ renameStoredPlaylist(const char *utf8from, const char *utf8to)
if (rename(from, to) < 0) if (rename(from, to) < 0)
return PLAYLIST_RESULT_ERRNO; return PLAYLIST_RESULT_ERRNO;
idle_add(IDLE_STORED_PLAYLIST);
return PLAYLIST_RESULT_SUCCESS; return PLAYLIST_RESULT_SUCCESS;
} }

View File

@ -30,6 +30,7 @@
#include "main_notify.h" #include "main_notify.h"
#include "condition.h" #include "condition.h"
#include "update.h" #include "update.h"
#include "idle.h"
static enum update_progress { static enum update_progress {
UPDATE_PROGRESS_IDLE = 0, UPDATE_PROGRESS_IDLE = 0,
@ -549,8 +550,10 @@ void reap_update_task(void)
if (pthread_join(update_thr, NULL)) if (pthread_join(update_thr, NULL))
FATAL("error joining update thread: %s\n", strerror(errno)); FATAL("error joining update thread: %s\n", strerror(errno));
if (modified) if (modified) {
playlistVersionChange(); playlistVersionChange();
idle_add(IDLE_DATABASE);
}
if (update_paths_nr) { if (update_paths_nr) {
char *path = update_paths[0]; char *path = update_paths[0];

View File

@ -21,6 +21,7 @@
#include "log.h" #include "log.h"
#include "player_control.h" #include "player_control.h"
#include "utils.h" #include "utils.h"
#include "idle.h"
#include "os_compat.h" #include "os_compat.h"
#include "../config.h" #include "../config.h"
@ -490,6 +491,8 @@ static int changeSoftwareVolume(int change, int rel)
int changeVolumeLevel(int change, int rel) int changeVolumeLevel(int change, int rel)
{ {
idle_add(IDLE_MIXER);
switch (volume_mixerType) { switch (volume_mixerType) {
#ifdef HAVE_ALSA #ifdef HAVE_ALSA
case VOLUME_MIXER_TYPE_ALSA: case VOLUME_MIXER_TYPE_ALSA: